Category Archives: PHP

Move WordPress to a New Domain Without Losing SEO

The process of switching to a new domain will temporarily affect your search engine rankings as Google and other search engines adjusts to the changes. This will also temporarily affect your search traffic as well. Please keep in mind that this is normal, and it happens to all sites that switch to a new domain.

Setting up Permanent 301 Redirects

Setting up a permanent 301 redirect is very important for both SEO and user experience. This allows you to redirect users and search engines to your new site. In other words, whenever someone lands on one of your old posts or pages, then they will be automatically redirected to your new site.

To setup a permanent 301 redirect, you need to connect to your old site using FTP and edit the .htaccess file. This will be located in the same directory as your wp-includes or wp-admin folder. Open the .htaccess file and paste the following code at the very top:

#Options +FollowSymLinks
RewriteEngine on
RewriteRule ^(.*)$ http://www.newsite.COM/$1 [R=301,L]

Note: Replace newsite.com with your domain in the above code.
Once you have applied these changes, then visit your old site. It should automatically redirect you to the new site. If it doesn’t, then it means the redirection is not setup properly.

Notifying Google About the Change

Login to your Google Webmaster Tools account to submit a change of address. Basically click on your site and look at the configuration menu -> Change od Address. This allows you to notify Google about your new site and the transfer. Yes, you have to verify your new site, so go ahead and do it.

Tour of the WordPress database

Source

In this tour I will focus on the tables for a standard single site install of WordPress and all of the tables will be referred to using the default prefix of ‘wp_’. However, for security reasons it isrecommended to use a different prefix when installing your WordPress sites.

The Tables

wp_posts

The posts table is arguably the most important table in the database. Its name sometimes throws people who believe it purely contains their blog posts. However, albeit badly named, it is an extremely powerful table that stores various types of content including posts, pages, menu items, media attachments and any custom post types that a site uses.

The table’s flexible content nature is provided by the ‘post_type’ column which denotes if the row is a post, page, attachment, nav_menu_item or another type. But this flexibility also makes it hard to discuss and describe. Essentially the table contains rows of content objects with different types, but for ease of reading, I will refer to the rows as “posts” throughout this article.

  • ID – unique number assigned to each post.
  • post_author – the user ID who created it. (Reference to the wp_users table.)
  • post_date – time and date of creation.
  • post_date_gmt – GMT time and date of creation. The GMT time and date is stored so there is no dependency on a site’s timezone in the future.
  • post_content – holds all the content for the post, including HTML, shortcodes and other content.
  • post_title – title of the post.
  • post_excerpt – custom intro or short version of the content.
  • post_status – status of the post, e.g. ‘draft’, ‘pending’, ‘private’, ‘publish’. Also a great WordPress news site.
  • comment_status – if comments are allowed.
  • ping_status – if the post allows ping and trackbacks.
  • post_password – optional password used to view the post.
  • post_name – URL friendly slug of the post title.
  • to_ping – a list of URLs WordPress should send pingbacks to when updated.
  • pinged – a list of URLs WordPress has sent pingbacks to when updated.
  • post_modified – time and date of last modification.
  • post_modified_gmt – GMT time and date of last modification.
  • post_content_filtered – used by plugins to cache a version of post_content typically passed through the ‘the_content’ filter. Not used by WordPress core itself.
  • post_parent – used to create a relationship between this post and another when this post is a revision, attachment or another type.
  • guid – Global Unique Identifier, the permanent URL to the post, not the permalink version.
  • menu_order – holds the display number for pages and other non-post types.
  • post_type – the content type identifier.
  • post_mime_type – only used for attachments, the MIME type of the uploaded file.
  • comment_count – total number of comments, pingbacks and trackbacks.

wp_postmeta

This table holds any extra information about individual posts. It is a vertical table using key/value pairs to store its data, a technique WordPress employs on a number of tables throughout the database allowing WordPress core, plugins and themes to store unlimited data.

  • meta_id – unique number assigned to each row of the table.
  • post_id – the ID of the post the data relates to. (Reference to the wp_posts table.)
  • meta_key – an identifying key for the piece of data.
  • meta_value – the actual piece of data.

wp_comments

Any post that allows discussion can have comments posted to it. This table stores those comments and some specific data about them. Further information can be stored inwp_commentmeta.

  • comment_ID – unique number assigned to each comment.
  • comment_post_ID – ID of the post this comment relates to. (Reference to the wp_poststable.)
  • comment_author – Name of the comment author.
  • comment_author_email – Email of the comment author.
  • comment_author_url – URL for the comment author.
  • comment_author_IP – IP Address of the comment author.
  • comment_date – Time and data the comment was posted.
  • comment_date_gmt – GMT time and data the comment was posted.
  • comment_content – the actual comment text.
  • comment_karma – unused by WordPress core, can be used by plugins to help manage comments.
  • comment_approved – if the comment has been approved.
  • comment_agent – where the comment was posted from, eg. browser, operating system etc.
  • comment_type – type of comment: comment, pingback or trackback.
  • comment_parent – refers to another comment when this comment is a reply.
  • user_id – ID of the comment author if they are a registered user on the site. (Reference to the wp_users table.)

wp_commentmeta

This table stores any further information related to a comment.

  • meta_id – unique number assigned to each row of the table.
  • comment_id – the ID of the post the data relates to. (Reference to the wp_commentstable.)
  • meta_key – an identifying key for the piece of data.
  • meta_value – the actual piece of data.

wp_terms

Terms are items of a taxonomy used to classify objects. Taxonomy what? WordPress allows items like posts and custom post types to be classified in various ways. For example, when creating a post in WordPress, by default you can add a category and some tags to it. Both ‘Category’ and ‘Tag’ are examples of a taxonomy, basically a way to group things together.

To classify this post (how meta of me) I would give it a category of ‘Guide’ and tags of ‘database’ and ‘mysql’. The category and tags are terms that would be contained in this table.

  • term_id – unique number assigned to each term.
  • name – the name of the term.
  • slug – the URL friendly slug of the name.
  • term_group – ability for themes or plugins to group terms together to use aliases. Not populated by WordPress core itself.

wp_term_taxonomy

Following the wp_terms example above, the terms ‘Guide’, ‘database’ and ‘mysql’ that are stored in wp_terms don’t exist yet as a ‘Category’ and as ‘Tags’ unless they are given context. Each term is assigned a taxonomy using this table.

The structure of this table allows you to use the same term across different taxonomies. For example ‘Database’ could be used as a category for posts and as a term of a custom taxonomy for a custom post type (think portfolio_category for portfolio items). The term of Database would exist once in wp_terms, but there would be two rows in wp_term_taxonomy for each taxonomy.

  • term_taxonomy_id – unique number assigned to each row of the table.
  • term_id – the ID of the related term. (Reference to the wp_terms table.)
  • taxonomy – the slug of the taxonomy. This can be the built in taxonomies or any taxonomy registered using register_taxonomy().
  • description – description of the term in this taxonomy.
  • parent – ID of a parent term. Used for hierarchical taxonomies like Categories.
  • count – number of post objects assigned the term for this taxonomy.

wp_term_relationships

So far we have seen how terms and their taxonomies are stored in the database, but have yet to see how WordPress stores the critical data when it comes to using taxonomies. This post exists in wp_posts and when we actually assign the category and tags through the WordPress dashboard this is the junction table that records that information. Each row defines a relationship between a post (object) in wp_posts and a term of a certain taxonomy in wp_term_taxonomy.

  • object_id – the ID of the post object. (Reference to the wp_posts table.)
  • term_taxonomy_id – the ID of the term / taxonomy pair. (Reference to thewp_term_taxonomy table.)
  • term_order – allow ordering of terms for an object, not fully used.

wp_users

WordPress’ user management is one of its strongest features and one that makes it great as an application framework. This table is the driving force behind it.

  • ID – unique number assigned to each user.
  • user_login – unique username for the user.
  • user_pass – hash of the user’s password.
  • user_nicename – display name for the user.
  • user_email – email address of the user.
  • user_url – URL of the user, e.g. website address.
  • user_registered – time and date the user registered.
  • user_activation_key – used for resetting passwords.
  • user_status – was used in Multisite pre WordPress 3.0 to indicate a spam user.
  • display_name – desired name to be used publicly in the site, can be user_login, user_nicename, first name or last name defined in wp_usermeta.

wp_usermeta

This table stores any further information related to the users. You will see other user profile fields for a user in the dashboard that are stored here.

  • umeta_id – unique number assigned to each row of the table.
  • user_id – ID of the related user. (Reference to the wp_users table.)
  • meta_key – an identifying key for the piece of data.
  • meta_value – the actual piece of data.

wp_options

The options table is the place where all of the site’s configuration is stored, including data about the theme, active plugins, widgets, and temporary cached data. It is typically where other plugins and themes store their settings.

The table is another example of a vertical key/value pair table to allow it to store all sorts of data for a variety of purposes.

  • option_id – unique number assigned to each row of the table.
  • option_name – an identifying key for the piece of data.
  • option_value – the actual piece of data. The data is often serialized so must be handled carefully.
  • autoload – controls if the option is automatically loaded by the functionwp_load_alloptions() (puts options into object cache on each page load).

Did you know that when performing migrations of databases using WP Migrate DB Pro you can tell the plugin to preserve specific options in the target database using the‘wpmdb_preserved_options’ filter?

During the rise of popularity of blogging having a blogroll (links to other sites) on your site was very much in fashion. This table holds all those links for you.

Nowadays blogrolls are used less and less and as of WordPress 3.5 the administration of links was removed from the admin UI. The table remains in the database for backwards compatibility and you can use the old link manager UI using this plugin.

  • link_id – unique number assigned to each row of the table.
  • link_url – URL of the link.
  • link_name – name of the link.
  • link_image – URL of an image related to the link.
  • link_target – the target frame for the link. e.g. _blank, _top, _none.
  • link_description – description of the link.
  • link_visible – control if the link is public or private.
  • link_owner – ID of user who created the link. (Reference to the wp_users table.)
  • link_rating – add a rating between 0-10 for the link.
  • link_updated – time and date of link update.
  • link_rel – relationship of link.
  • link_notes – notes about the link.
  • link_rss – RSS address for the link.

Someone has produced a helpful entity relationship diagram to explain the relationships between all the tables and posted it on the WordPress codex. This was created at version 3.8 but the structure is still current:

WordPress

WordPress is great at doing all the heavy lifting for you when it comes to reading and writing to the database, so even though we might know where the data is stored and how to get it, we always recommend using the WordPress APIs wherever possible.

WordPress, if no featured image get first image attachment from post

function bm_my_post_thumbnail_html( $html, $post_id, $thumbnail_id, $size = '' ) {

    if ( empty( $html ) ) {

        $values = get_children(
            array(
                'post_parent' => $post_id,
                'post_type' => 'attachment',
                'post_mime_type' => 'image',
                'order' => 'ASC',
                'orderby' => 'menu_order',
                'numberposts' => 1,
            )
        );

        if ( $values ) {
            foreach ( $values as $child_id => $attachment ) {
                $html = wp_get_attachment_image( $child_id, $size );
                break;
            }
        }

    }

    return $html;

}
add_filter( 'post_thumbnail_html', 'bm_my_post_thumbnail_html', 10, 4 );

 

Add sortable columns to admin

There are 3 different column filters: ‘manage_pages_columns’ for pages,
‘manage_posts_columns’ which covers ALL post types (including custom post types),
and ‘manage_{$post_type_name}_posts_columns’ which only covers, you guessed it,
the columns for the defined $post_type_name.

The ‘manage_pages_columns’ and ‘manage_{$post_type_name}_posts_columns’ filters only
pass $columns (an array), which is the column info, as an argument, but ‘manage_posts_columns’
passes $columns and $post_type (a string).

Note: Don’t forget that it’s a WordPress filter so you HAVE to return the first argument that’s
passed to the function, in this case $columns. And for filters that pass more than 1 argument,
you have to specify the number of accepted arguments in your add_filter() declaration,
following the priority argument.

add_filter( 'manage_posts_columns', 'manage_wp_posts_be_qe_manage_posts_columns', 10, 2 );
function manage_wp_posts_be_qe_manage_posts_columns( $columns, $post_type ) {
    
    if ( $post_type == 'movies' ) {
        $columns[ 'release_date' ] = 'Release Date';
        $columns[ 'coming_soon' ] = 'Coming Soon';
        $columns[ 'film_rating' ] = 'Film Rating';
    }
        
    return $columns;
}

 

The following filter allows you to make your column(s) sortable.

The ‘edit-movies’ section of the filter name is the custom part
of the filter name, which tells WordPress you want this to run
on the main ‘movies’ custom post type edit screen. So, e.g., if
your custom post type’s name was ‘books’, then the filter name
would be ‘manage_edit-books_sortable_columns’.

Don’t forget that filters must ALWAYS return a value.

add_filter( 'manage_edit-movies_sortable_columns', 'manage_wp_posts_be_qe_manage_sortable_columns' );
function manage_wp_posts_be_qe_manage_sortable_columns( $sortable_columns ) {
    $sortable_columns[ 'release_date_column' ] = 'release_date';    
    $sortable_columns[ 'film_rating_column' ] = 'film_rating';
    return $sortable_columns;    
}

 

Now that we have a column, we need to fill our column with data.
The filters to populate your custom column are pretty similar to the ones
that added your column: ‘manage_pages_custom_column’, ‘manage_posts_custom_column’,
and ‘manage_{$post_type_name}_posts_custom_column’. All three pass the same
2 arguments: $column_name (a string) and the $post_id (an integer).

Our custom column data is post meta so it will be a pretty simple case of retrieving
the post meta with the meta key ‘release_date’.

Note that we are wrapping our post meta in a div with an id of Òrelease_date-Ó plus the post id.
This will come in handy when we are populating our ÒQuick EditÓ row.

 

add_action( 'manage_posts_custom_column', 'manage_wp_posts_be_qe_manage_posts_custom_column', 10, 2 );
function manage_wp_posts_be_qe_manage_posts_custom_column( $column_name, $post_id ) {
    switch( $column_name ) {
    
        case 'release_date_column':
        
            echo get_post_meta( $post_id, 'release_date', true );
            break;
            
        case 'coming_soon_column':
        
            echo get_post_meta( $post_id, 'coming_soon', true );
            break;
            
        case 'film_rating_column':
        
            echo get_post_meta( $post_id, 'film_rating', true );
            break;            
    }
    
}

 

 

Just because we’ve made the column sortable doesn’t
mean the posts will sort by our column data. That’s where
this next 2 filters come into play.

If your sort data is simple, i.e. alphabetically or numerically,
then ‘pre_get_posts’ is the filter to use. This filter lets you
change up the query before it’s run.

If your orderby data is more complicated, like our release date
which is a date string stored in a custom field, then check out
the ‘posts_clauses’ filter example used below.

In the example below, when the main query is trying to order by
the ‘film_rating’, it’s a simple alphabetical sorting by a custom
field so we’re telling the query to set our ‘meta_key’ which is
‘film_rating’ and that we want to order by the query by the
custom field’s meta_value, e.g. PG, PG-13, R, etc.

Check out http://codex.wordpress.org/Class_Reference/WP_Query
for more info on WP Query parameters.

 

add_action( 'pre_get_posts', 'manage_wp_posts_be_qe_pre_get_posts', 1 );
function manage_wp_posts_be_qe_pre_get_posts( $query ) {
    /**
     * We only want our code to run in the main WP query
     * AND if an orderby query variable is designated.
     */
    if ( $query->is_main_query() && ( $orderby = $query->get( 'orderby' ) ) ) {
    
        switch( $orderby ) {
        
            // If we're ordering by 'film_rating'
            case 'film_rating':
            
                // set our query's meta_key, which is used for custom fields
                $query->set( 'meta_key', 'film_rating' );
                
                /**
                 * Tell the query to order by our custom field/meta_key's
                 * value, in this case: PG, PG-13, R, etc.
                 *
                 * If your meta value are numbers, change
                 * 'meta_value' to 'meta_value_num'.
                 */
                $query->set( 'orderby', 'meta_value' );
                
                break;
                
        }
    
    }    
}

 

Just because we’ve made the column sortable doesn’t
mean the posts will sort by our column data. That’s where
the filter above, ‘pre_get_posts’, and the filter below,
‘posts_clauses’, come into play.

If your sort data is simple, i.e. alphabetically or numerically,
then check out the ‘pre_get_posts’ filter used above.

If your orderby data is more complicated, like combining
several values or a date string stored in a custom field,
then the ‘posts_clauses’ filter used below is for you.
The ‘posts_clauses’ filter allows you to manually tweak
the query clauses in order to sort the posts by your
custom column data.

The reason more complicated sorts will not with the
“out of the box” WP Query is because the WP Query orderby
parameter will only order alphabetically and numerically.

Usually I would recommend simply using the ‘pre_get_posts’
and altering the WP Query itself but because our custom
field is a date, we have to manually set the query to
order our posts by a date.

 

add_filter( 'posts_clauses', 'manage_wp_posts_be_qe_posts_clauses', 1, 2 );
function manage_wp_posts_be_qe_posts_clauses( $pieces, $query ) {
    global $wpdb;
    
    /**
     * We only want our code to run in the main WP query
     * AND if an orderby query variable is designated.
     */
    if ( $query->is_main_query() && ( $orderby = $query->get( 'orderby' ) ) ) {
    
        // Get the order query variable - ASC or DESC
        $order = strtoupper( $query->get( 'order' ) );
        
        // Make sure the order setting qualifies. If not, set default as ASC
        if ( ! in_array( $order, array( 'ASC', 'DESC' ) ) )
            $order = 'ASC';
    
        switch( $orderby ) {
        
            // If we're ordering by release_date
            case 'release_date':
            
                /**
                 * We have to join the postmeta table to include
                 * our release date in the query.
                 */
                $pieces[ 'join' ] .= " LEFT JOIN $wpdb->postmeta wp_rd ON wp_rd.post_id = {$wpdb->posts}.ID AND wp_rd.meta_key = 'release_date'";
                
                // Then tell the query to order by our date
                $pieces[ 'orderby' ] = "STR_TO_DATE( wp_rd.meta_value,'%m/%d/%Y' ) $order, " . $pieces[ 'orderby' ];
                
                break;
        
        }
    
    }
    return $pieces;
}

 

PHP date to JS date

Kako pouzdano iz PHP-a setapirati JS date

$tz = new DateTimeZone($this_city_tz);
$date = new DateTime();
$date->setTimeZone($tz);
$date_time = $date->format('Y-m-d H:i:s');
$date_time_js = $date->format('Y/m/d H:i:s');

var currentDate = new Date('<?php echo gmdate("r", strtotime($date_time_js)); ?>');

console.log(currentDate);