// Remove WP Version From Styles add_filter( 'style_loader_src', 'sdt_remove_ver_css_js', 9999 ); // Remove WP Version From Scripts add_filter( 'script_loader_src', 'sdt_remove_ver_css_js', 9999 ); // Function to remove version numbers function sdt_remove_ver_css_js( $src ) { if ( strpos( $src, 'ver=' ) ) $src = remove_query_arg( 'ver', $src ); return $src; }
add_action('restrict_manage_posts','my_restrict_manage_posts'); function my_restrict_manage_posts() { global $typenow; if ($typenow=='job_listing'){ $args = array( 'show_option_all' => "Show All Cities", 'taxonomy' => 'listing_cities', 'name' => 'listing_cities' ); wp_dropdown_categories($args); } } add_action( 'request', 'my_request' ); function my_request($request) { if (is_admin() && $GLOBALS['PHP_SELF'] == '/wp-admin/edit.php' && isset($request['post_type']) && $request['post_type']=='job_listing') { $request['listing_cities'] = get_term($request['listing_cities'],'listing_cities')->name; } return $request; }
Send email
$data = @$_POST; $err = array(); if(!empty($_POST)){ if(empty($_POST["your_name"])){ $err["your_name"] = true; } if(empty($_POST["your_email"])){ $err["your_email"] = true; } if(empty($_POST["your_subject"])){ $err["your_subject"] = true; } if(empty($_POST["your_message"])){ $err["your_message"] = true; } if(empty($err)){ $your_name = strip_tags($_POST["your_name"]); $your_company = strip_tags($_POST["your_company"]); $your_phone = strip_tags($_POST["your_phone"]); $your_email = strip_tags($_POST["your_email"]); $your_subject = strip_tags($_POST["your_subject"]); $your_message = strip_tags($_POST["your_message"]); $caption = "Contactform"; $admin_email = get_option("admin_email"); ob_start(); include("template-contact-mail.php"); $mail_content = ob_get_contents(); ob_end_clean() ; $headers = 'Content-Type: text/html; charset=UTF-8'.'\r\n'; $headers .= 'From: '.$your_name.' <'.$your_email.'>' . "\r\n"; $success = wp_mail($admin_email, $caption, $mail_content, $headers); header("Location: ".get_permalink(PAGE_ID_SUCCESS_CONTACT)); die; } }
<form action="" method="post" class="wpcf7-form"> <p<?php if(!empty($err["your_name"])): ?> class="err"<?php endif; ?>> <label><?php _e("Your name") ?> (*):</label> <span class="wpcf7-form-control-wrap"><input type="text" name="your_name" value="<?php echo $data["your_name"] ?>" size="40" class="wpcf7-form-control wpcf7-text" /></span> </p> <p<?php if(!empty($err["your_company"])): ?> class="err"<?php endif; ?>> <label><?php _e("Company") ?>:</label> <span class="wpcf7-form-control-wrap"><input type="text" name="your_company" value="<?php echo $data["your_company"] ?>" size="40" class="wpcf7-form-control wpcf7-text" /></span> </p> <p<?php if(!empty($err["your_phone"])): ?> class="err"<?php endif; ?>> <label><?php _e("Phone") ?>:</label> <span class="wpcf7-form-control-wrap"><input type="text" name="your_phone" value="<?php echo $data["your_phone"] ?>" size="40" class="wpcf7-form-control wpcf7-text" /></span> </p> <p<?php if(!empty($err["your_email"])): ?> class="err"<?php endif; ?>> <label><?php _e("Email") ?> (*):</label> <span class="wpcf7-form-control-wrap"><input type="email" name="your_email" value="<?php echo $data["your_email"] ?>" size="40" class="wpcf7-form-control wpcf7-text" /></span> </p> <p<?php if(!empty($err["your_subject"])): ?> class="err"<?php endif; ?>> <label><?php _e("Subject") ?> (*):</label> <span class="wpcf7-form-control-wrap"><input type="text" name="your_subject" value="<?php echo $data["your_subject"] ?>" size="40" class="wpcf7-form-control wpcf7-text" /></span> </p> <p<?php if(!empty($err["your_message"])): ?> class="err"<?php endif; ?>> <label><?php _e("Message") ?> (*):</label> <span class="wpcf7-form-control-wrap"><textarea name="your_message" cols="40" rows="10" class="wpcf7-form-control wpcf7-textarea"><?php echo $data["your_message"] ?></textarea></span> </p> <p> <input type="submit" value="submit" class="wpcf7-form-control wpcf7-submit" /> </p> </form>
template-contact-mail.php
<br />Onderwerp: <?php echo $caption ?> <br /> <br />NAME: <br /><?php echo $your_name ?> <br /> <br />COMPANY: <br /><?php echo $your_company ?> <br /> <br />PHONE: <br /><?php echo $your_phone ?> <br /> <br />E-MAIL ADDRESS: <br /><?php echo $your_email ?> <br /> <br />SUBJECT: <br /><?php echo $your_subject ?> <br /> <br />MESSAGE: <br /><?php echo $your_message ?> <br /> <br />-- <br />atanasfilipov.com
Индекс на базата ( http://example.com/faq/
) [Индекс на цялата база]
- Основна категория (
http://example.com/faq/category/
) [Индекс на Основната категория]- Публикация в основната категория (
http://example.com/faq/category/post-1-slug/
) [Преглед на Публикация 1] - Под-категория (
http://example.com/faq/category/sub-category/
) [Индекс на под-категорията]- Публикация в под-категорията (
http://example.com/faq/category/sub-category/post-2-slug/
) [Преглед на Публикация 2]
- Публикация в под-категорията (
- Публикация в основната категория (
- Друга основна категория (
http://example.com/faq/another-category/
) [Индекс на другата основна категория]
Всичко изглежда нормално(като структура на линковете) за нормалния потребител, обаче WordPress е на друго мнение по въпроса Вместо линк като http://example.com/faq/category/
ние ще можем да видим индекса на основната категория на линк от рода на http://example.com/faq-category/category/
. Което както виждате не съвпада съвсем с първоначалната идея. За да оправим този проблем трябва да прибегнем до добавянето на няколко специално написани Permalink правила и функции.
Първата стъпка е да регистрираме новия пост тип(Custom Post Type). Това се случва със следната функция:
<?php add_action( 'init', 'register_faq_post_type' ); function register_faq_post_type() { register_post_type( 'faq', array( 'labels' => array( 'name' => 'Frequently Asked Questions', 'menu_name' => 'FAQ Manager', 'singular_name' => 'Question', 'all_items' => 'All Questions' ), 'public' => true, 'publicly_queryable' => true, 'show_ui' => true, 'show_in_menu' => true, 'show_in_nav_menus' => true, 'menu_position ' => 20, 'supports' => array( 'title', 'editor', 'author', 'thumbnail', 'comments', 'post-formats', 'revisions' ), 'hierarchical' => false, 'has_archive' => 'faq', 'rewrite' => array( 'slug' => 'faq-item', 'hierarchical' => true, 'with_front' => false ) ) ); register_taxonomy( 'faq-category', array( 'faq' ), array( 'labels' => array( 'name' => 'FAQ Categories', 'menu_name' => 'FAQ Categories', 'singular_name' => 'FAQ Category', 'all_items' => 'All Categories' ), 'public' => true, 'hierarchical' => true, 'show_ui' => true, 'rewrite' => array( 'slug' => 'faq-category', 'hierarchical' => true, 'with_front' => false ), ) ); } ?>
Този код ще регистрира новия пост тип, заедно с новата таксономия „FAQ Categories“, която ще използваме за да категоризираме въпросите.
След като регистрираме новия пост тип остава да добавим две функции които да направят така че Permalink-овете да работят както искаме ние
<?php add_action( 'generate_rewrite_rules', 'register_faq_rewrite_rules' ); function register_faq_rewrite_rules( $wp_rewrite ) { $new_rules = array( 'faq/([^/]+)/?$' => 'index.php?faq-category=' . $wp_rewrite->preg_index( 1 ), 'faq/([^/]+)/([^/]+)/?$' => 'index.php?post_type=faq&faq-category=' . $wp_rewrite->preg_index( 1 ) . '&faq=' . $wp_rewrite->preg_index( 2 ), 'faq/([^/]+)/([^/]+)/page/(\d{1,})/?$' => 'index.php?post_type=faq&faq-category=' . $wp_rewrite->preg_index( 1 ) . '&paged=' . $wp_rewrite->preg_index( 3 ), 'faq/([^/]+)/([^/]+)/([^/]+)/?$' => 'index.php?post_type=faq&faq-category=' . $wp_rewrite->preg_index( 2 ) . '&faq=' . $wp_rewrite->preg_index( 3 ), ); $wp_rewrite->rules = $new_rules + $wp_rewrite->rules; } // A hacky way of adding support for flexible custom permalinks // There is one case in which the rewrite rules from register_kb_rewrite_rules() fail: // When you visit the archive page for a child section(for example: http://example.com/faq/category/child-category) // The deal is that in this situation, the URL is parsed as a Knowledgebase post with slug "child-category" from the "category" section function fix_faq_subcategory_query($query) { if ( isset( $query['post_type'] ) && 'faq' == $query['post_type'] ) { if ( isset( $query['faq'] ) && $query['faq'] && isset( $query['faq-category'] ) && $query['faq-category'] ) { $query_old = $query; // Check if this is a paginated result(like search results) if ( 'page' == $query['faq-category'] ) { $query['paged'] = $query['name']; unset( $query['faq-category'], $query['name'], $query['faq'] ); } // Make it easier on the DB $query['fields'] = 'ids'; $query['posts_per_page'] = 1; // See if we have results or not $_query = new WP_Query( $query ); if ( ! $_query->posts ) { $query = array( 'faq-category' => $query['faq'] ); if ( isset( $query_old['faq-category'] ) && 'page' == $query_old['faq-category'] ) { $query['paged'] = $query_old['name']; } } } } return $query; } add_filter( 'request', 'fix_faq_subcategory_query', 10 ); ?>
В горния код регистрираме две функции – register_faq_rewrite_rules()
и fix_faq_subcategory_query()
.
Първата функция добавя следните няколко правила:
faq/([^/]+)/?$
– това правило ще бъде приложено когато адреса на страницата която се зарежда прилича на „faq/any-character/“ – тоест когато имаме начална част „faq/“ последвана от всякаква комбинация от символи, без „/“( „([^/]+)“ ), евентуално последвана от „/“( „/?“ ). Също така тази част трябва да е последната част от адреса на страницата, за да има съвпадение(т.е. няма да има съвпадение, ако адреса е „faq/any-character/something“).Когато адреса съвпадне, на текущата страница ще бъде показан архив за категорията(според примера който дадох по-горе) „any-character“.faq/([^/]+)/([^/]+)/?$
– това правило е почти същото като предишното, с изключение на това че вече адреса който би съвпаднал би изглеждал като „faq/any-character/post-slug/“.Това ще зареди публикация с кратко име(slug) „post-slug“ от категорията за ЧЗВ с кратко име „any-character“.faq/([^/]+)/([^/]+)/([^/]+)/?$
– това правило е почти същото като предишното, с изключение на това че вече адреса който би съвпаднал би изглеждал като „faq/any-character/sub-category/post-slug/“.Това ще зареди публикация с кратко име(slug) „post-slug“ от категорията за ЧЗВ с кратко име „sub-category“.
Втората функция от друга страна оправя проблема със под-категориите. А самият проблем се състои в това, че когато се опитате да заредите страница с адрес faq/category/child-category/
, WordPress ще се опита да зареди публикация с кратко име „child-category“ вместо под-категорията с кратко име „child-category“. Самата функция е по всяка вероятност не най-красивото решение на проблема, най-малкото защото трябва да направим една допълнителна заявка към базата данни, но това е единствения начин по който успях да реша проблема Понеже функцията проверява дали има публикации с краткото име „child-category“, ако случайно има такава публикация, може да получите неочаквани резултати
Това е достатъчно за да подкарате секцията за ЧЗВ, обаче в момента ако искате да използвате пълноценно новите адреси на категориите и публикациите, ще трябва да си ги пишете сами. Това е защото WordPress не знае как трябва да генерира линковете така че да съвпадат с нашата идея. За това добавяме още няколко функции:
<?php function faq_article_permalink( $article_id, $section_id = false, $leavename = false, $only_permalink = false ) { $taxonomy = 'faq-category'; $article = get_post( $article_id ); $return = '<a href="'; $permalink = ( $section_id ) ? trailingslashit( get_term_link( intval( $section_id ), 'faq-category' ) ) : home_url( '/faq/' ); $permalink .= trailingslashit( ( $leavename ? "%$article->post_type%" : $article->post_name ) ); $return .= $permalink . '/" >' . get_the_title( $article->ID ) . '</a>'; return ( $only_permalink ) ? $permalink : $return; } function filter_faq_post_link( $permalink, $post, $leavename ) { if ( get_post_type( $post->ID ) == 'faq' ) { $terms = wp_get_post_terms( $post->ID, 'faq-category' ); $term = ( $terms ) ? $terms[0]->term_id : false; $permalink = faq_article_permalink( $post->ID, $term, $leavename, true ); } return $permalink; } add_filter( 'post_type_link', 'filter_faq_post_link', 100, 3 ); function filter_faq_section_terms_link( $termlink, $term, $taxonomy = false ) { if ( $taxonomy == 'faq-category' ) { $section_ancestors = get_ancestors( $term->term_id, $taxonomy ); krsort( $section_ancestors ); $termlink = home_url( '/faq/' ); foreach ( $section_ancestors as $ancestor ) { $section_ancestor = get_term( $ancestor, $taxonomy ); $termlink .= $section_ancestor->slug . '/'; } $termlink .= trailingslashit( $term->slug ); } return $termlink; } add_filter( 'term_link', 'filter_faq_section_terms_link', 100, 3 ); ?>
Горните функции „филтрират“ резултата от две вградени в WordPress функции, които се грижат да създадат правилните линкове за всички страници, публикации и архиви на таксономии.
Остава един проблем, който може да не е никак малък за някои потребители – проблема с дубликиращото се съдържание. Има доста голяма вероятност Google или някоя друга търсачка която обхожда сайта ви да не остане очарована от факта, че намира идентично съдържание на два различни адреса(например faq-item/post-2-slug/
и faq/category/sub-category/post-2-slug/
). Това може да се избегне с една функция която да препраща потребителите към правилния адрес, но оставям това на вас.
Източник: http://themoonwatch.com/bg/%D0%BA%D0%B0%D1%82%D0%B5%D0%B3%D0%BE%D1%80%D0%B8%D1%8F/php-i-wordpress/#sthash.GBKCWoNy.dpuf
Първо спираме ETag header-а тъй като ще използваме EXPIRES CACHING:
# ---------------------------------------------------------------------- # Expire Header # ---------------------------------------------------------------------- # Use UTF-8 encoding for anything served text/plain or text/html AddDefaultCharset UTF-8 # Force UTF-8 for a number of file formats AddCharset UTF-8 .atom .css .js .json .rss .vtt .xml # FileETag None is not enough for every server. Header unset ETag # Since we’re sending far-future expires, we don’t need ETags for static content. # developer.yahoo.com/performance/rules.html#etags FileETag None # Send CORS headers if browsers request them; enabled by default for images. # mod_headers, y u no match by Content-Type?! SetEnvIf Origin ":" IS_CORS Header set Access-Control-Allow-Origin "*" env=IS_CORS # Allow access to web fonts from all domains. Header set Access-Control-Allow-Origin "*" Header set X-Powered-By "WP Rocket/2.8.14" Header unset Pragma Header append Cache-Control "public" Header unset Last-Modified Header unset Pragma Header append Cache-Control "public" # Expires headers (for better cache control) ExpiresActive on # Perhaps better to whitelist expires rules? Perhaps. ExpiresDefault "access plus 1 month" # cache.appcache needs re-requests in FF 3.6 (thanks Remy ~Introducing HTML5) ExpiresByType text/cache-manifest "access plus 0 seconds" # Your document html ExpiresByType text/html "access plus 0 seconds" # Data ExpiresByType text/xml "access plus 0 seconds" ExpiresByType application/xml "access plus 0 seconds" ExpiresByType application/json "access plus 0 seconds" # Feed ExpiresByType application/rss+xml "access plus 1 hour" ExpiresByType application/atom+xml "access plus 1 hour" # Favicon (cannot be renamed) ExpiresByType image/x-icon "access plus 1 week" # Media: images, video, audio ExpiresByType image/gif "access plus 1 month" ExpiresByType image/png "access plus 1 month" ExpiresByType image/jpg "access 1 year" ExpiresByType image/jpeg "access 1 year" ExpiresByType video/ogg "access plus 1 month" ExpiresByType audio/ogg "access plus 1 month" ExpiresByType video/mp4 "access plus 1 month" ExpiresByType video/webm "access plus 1 month" # HTC files (css3pie) ExpiresByType text/x-component "access plus 1 month" # Webfonts ExpiresByType application/x-font-ttf "access plus 1 month" ExpiresByType font/opentype "access plus 1 month" ExpiresByType application/x-font-woff "access plus 1 month" ExpiresByType application/x-font-woff2 "access plus 1 month" ExpiresByType image/svg+xml "access plus 1 month" ExpiresByType application/vnd.ms-fontobject "access plus 1 month" # CSS and JavaScript ExpiresByType text/css "access plus 1 year" ExpiresByType application/javascript "access plus 1 year" # Gzip compression # Active compression SetOutputFilter DEFLATE # Force deflate for mangled headers SetEnvIfNoCase ^(Accept-EncodXng|X-cept-Encoding|X{15}|~{15}|-{15})$ ^((gzip|deflate)\s*,?\s*)+|[X~-]{4,13}$ HAVE_Accept-Encoding RequestHeader append Accept-Encoding "gzip,deflate" env=HAVE_Accept-Encoding # Don’t compress images and other uncompressible content SetEnvIfNoCase Request_URI \ \.(?:gif|jpe?g|png|rar|zip|exe|flv|mov|wma|mp3|avi|swf|mp?g|mp4|webm|webp)$ no-gzip dont-vary # Compress all output labeled with one of the following MIME-types AddOutputFilterByType DEFLATE application/atom+xml \ application/javascript \ application/json \ application/rss+xml \ application/vnd.ms-fontobject \ application/x-font-ttf \ application/xhtml+xml \ application/xml \ font/opentype \ image/svg+xml \ image/x-icon \ text/css \ text/html \ text/plain \ text/x-component \ text/xml Header append Vary: Accept-Encoding # ---------------------------------------------------------------------- # Fonts # ---------------------------------------------------------------------- # Add correct content-type for fonts AddType application/vnd.ms-fontobject .eot AddType font/ttf .ttf AddType font/otf .otf AddType font/x-woff .woff AddType image/svg+xml .svg # Compress compressible fonts AddOutputFilterByType DEFLATE font/ttf font/otf image/svg+xml # Add a far future Expires header for fonts ExpiresByType application/vnd.ms-fontobject "access plus 1 year" ExpiresByType font/ttf "access plus 1 year" ExpiresByType font/otf "access plus 1 year" ExpiresByType font/x-woff "access plus 1 year" ExpiresByType image/svg+xml "access plus 1 year"
Remove Query Strings
Добавете следния код в function.php (WordPress)
function _remove_script_version( $src ){ $parts = explode( '?ver', $src ); return $parts[0]; } add_filter( 'script_loader_src', '_remove_script_version', 15, 1 ); add_filter( 'style_loader_src', '_remove_script_version', 15, 1 );
function get_cat_count_downloads_by_cat($cat_id){ global $wpdb; $total_downloads = $wpdb->get_col($wpdb->prepare ("SELECT meta_value FROM wp_postmeta as pm INNER JOIN wp_term_relationships as tr ON (pm.post_id = tr.object_id) INNER JOIN wp_term_taxonomy as tt ON (tr.term_taxonomy_id = tt.term_taxonomy_id) WHERE 1 AND tt.term_taxonomy_id = '".$cat_id."' AND pm.meta_key = 'mp3_downloads'")); return array_sum( $total_downloads ); }
Когато регистрирате нов тип пост “custom post type” във вашия function.php има опции rewrite, query_var, publicly_queryable, public. Те трябва да изглеждат така:
'rewrite' => false, 'query_var' => false, 'publicly_queryable' => false, 'public' => false
От WordPress документацията:
rewrite Triggers the handling of rewrites for this post type. To prevent rewrites, set to false. query_var 'false' - Disables query_var key use. A post type cannot be loaded at /?{query_var}={single_post_slug} publicly_queryable Whether queries can be performed on the front end as part of parse_request(). public 'false' - Post type is not intended to be used publicly and should generally be unavailable in wp-admin and on the front end unless explicitly planned for elsewhere.
Сега отивате в “Permalinks Settings” или ако сайта Ви е на български “Постоянни връзки” и без да променяте нищо натиснете запазване на промените, което ще презапише настройките.
С това действие ще забрани достъпа до вашия “custom post type” от предната част на сайта.
- 1
- 2