defaults = apply_filters( 'email_post_changes_default_options', array( 'enable' => 1, 'users' => array(), 'emails' => array( get_option( 'admin_email' ) ), 'post_types' => array( 'post', 'page' ), 'drafts' => 0, ) ); $options = $this->get_options(); if ( $options['enable'] ) add_action( 'post_updated', array( $this, 'post_updated' ), 10, 3 ); // Ov3rfly: if ( $options['enable'] ) add_action( 'pre_post_update', array( $this, 'pre_post_update' ), 10, 2 ); if ( current_user_can( 'manage_options' ) ) add_action( 'admin_menu', array( $this, 'admin_menu' ), 115 ); } function get_post_types() { $post_types = get_post_types( array( 'public' => true ) ); $_post_types = array(); foreach ( $post_types as $post_type ) { if ( post_type_supports( $post_type, 'revisions' ) ) $_post_types[] = $post_type; } return $_post_types; } function get_options( $just_defaults = false ) { if ( $just_defaults ) return $this->defaults; $options = (array) get_option( 'email_post_changes' ); return wp_parse_args( $options, $this->defaults ); } // Ov3rfly: function pre_post_update( $post_id, $data ) { $left_taxonomies = array(); if ( isset( $data['post_type'] ) ) { $taxonomies = get_object_taxonomies( $data['post_type'], 'objects' ); foreach( $taxonomies as $taxonomy_slug => $taxonomy ) { $left_taxonomies[ $taxonomy_slug ] = strip_tags( get_the_term_list( $post_id, $taxonomy_slug, '', ', ', '' ) ); } } $this->left_taxonomies = $left_taxonomies; } // The meat of the plugin function post_updated( $post_id, $post_after, $post_before ) { $options = $this->get_options(); // If we're purely saving a draft, and don't have the draft option enabled, skip. If we're transitioning one way or the other, send a notification. if ( 0 == $options['drafts'] && 'draft' == $post_before->post_status && 'draft' == $post_after->post_status ) return; if ( isset( $_POST['autosave'] ) ) return; if ( !in_array( $post_before->post_type, $options['post_types'] ) ) return; $this->left_post = $post_before; $this->right_post = $post_after; // If this is a new post, set an empty title for $this->left_post so that it appears in the diff. $child_posts = wp_get_post_revisions( $post_id, array( 'numberposts' => 1 ) ); if ( count( $child_posts ) == 0 ) { $this->left_post->post_title = ''; } if ( !$this->left_post || !$this->right_post ) return; $html_diffs = array(); $text_diffs = array(); $identical = true; foreach ( _wp_post_revision_fields() as $field => $field_title ) { $left = apply_filters( "_wp_post_revision_field_$field", $this->left_post->$field, $field ); $right = apply_filters( "_wp_post_revision_field_$field", $this->right_post->$field, $field ); if ( !$diff = $this->wp_text_diff( $left, $right ) ) continue; $html_diffs[$field_title] = $diff; $left = normalize_whitespace( $left ); $right = normalize_whitespace( $right ); $left_lines = explode( "\n", $left ); $right_lines = explode( "\n", $right ); require_once( dirname( __FILE__ ) . '/unified.php' ); $text_diff = new Text_Diff( $left_lines, $right_lines ); $renderer = new Text_Diff_Renderer_unified(); $text_diffs[$field_title] = $renderer->render($text_diff); $identical = false; } // Ov3rfly: Detect draft to publish or similar if ( $this->left_post->post_status != $this->right_post->post_status ) { $left = $this->nice_post_status( $this->left_post->post_status ); $right = $this->nice_post_status( $this->right_post->post_status ); $field_title = __( 'Status' ); if ( $diff = $this->wp_text_diff( $left, $right ) ) { $html_diffs[$field_title] = $diff; require_once( dirname( __FILE__ ) . '/unified.php' ); $text_diff = new Text_Diff( array( $left ), array( $right ) ); $renderer = new Text_Diff_Renderer_unified(); $text_diffs[$field_title] = $renderer->render($text_diff); $identical = false; } } // Ov3rfly: Detect taxonomy changes (tags, categories.. ), see also function pre_post_update() if ( !empty( $this->left_taxonomies ) ) { $taxonomies = get_object_taxonomies( $this->right_post->post_type, 'objects' ); foreach( $taxonomies as $taxonomy_slug => $taxonomy ) { $left = $this->left_taxonomies[ $taxonomy_slug ]; $right = strip_tags( get_the_term_list( $this->right_post->ID, $taxonomy_slug, '', ', ', '' ) ); $field_title = __( $taxonomy->label ); if ( $diff = $this->wp_text_diff( $left, $right ) ) { $html_diffs[$field_title] = $diff; require_once( dirname( __FILE__ ) . '/unified.php' ); $text_diff = new Text_Diff( array( $left ), array( $right ) ); $renderer = new Text_Diff_Renderer_unified(); $text_diffs[$field_title] = $renderer->render($text_diff); $identical = false; } } } if ( $identical ) { $this->left_post = null; $this->right_post = null; return; } // Grab the meta data $the_author = get_the_author_meta( 'display_name', get_current_user_id() ); // The revision $the_title = get_the_title( $this->right_post->ID ); // New title (may be same as old title) $the_date = gmdate( 'j F, Y \a\t G:i \U\T\C', strtotime( $this->right_post->post_modified_gmt . '+0000' ) ); // Modified time $the_permalink = clean_url( get_permalink( $this->right_post->ID ) ); $the_edit_link = clean_url( get_edit_post_link( $this->right_post->ID ) ); $left_title = __( 'Revision' ); $right_title = sprintf( __( 'Current %s' ), $post_type = ucfirst( $this->right_post->post_type ) ); $head_sprintf = __( '%s made the following changes to the %s %s on %s' ); // HTML $html_diff_head = '

' . sprintf( __( '%s changed' ), $post_type ) . "

\n"; $html_diff_head .= '

' . sprintf( $head_sprintf, esc_html( $the_author ), sprintf( _x( '“%s” [%s]', '1 = link, 2 = "edit"' ), "" . esc_html( $the_title ) . '', "" . __( 'edit' ) . '' ), $this->right_post->post_type, $the_date ) . "

\n\n"; $html_diff_head .= "\n"; $html_diff_head .= "\n"; $html_diff_head .= "\n"; $html_diff_head .= "
" . esc_html( $left_title ) . ' @ ' . esc_html( $this->left_post->post_date_gmt ) . "" . esc_html( $right_title ) . ' @ ' . esc_html( $this->right_post->post_modified_gmt ) . "
\n\n"; $html_diff = ''; foreach ( $html_diffs as $field_title => $diff ) { $html_diff .= '

' . esc_html( $field_title ) . "

\n"; $html_diff .= "$diff\n\n"; } $html_diff = rtrim( $html_diff ); // Replace classes with inline style $html_diff = str_replace( "class='diff'", 'style="width: 100%; border-collapse: collapse; border: none; white-space: pre-wrap; word-wrap: break-word; font-family: Consolas,Monaco,Courier,monospace;"', $html_diff ); $html_diff = preg_replace( '#]+/?>#i', '', $html_diff ); $html_diff = str_replace( "class='diff-deletedline'", 'style="padding: 5px; width: 50%; background-color: #fdd;"', $html_diff ); $html_diff = str_replace( "class='diff-addedline'", 'style="padding: 5px; width: 50%; background-color: #dfd;"', $html_diff ); $html_diff = str_replace( "class='diff-context'", 'style="padding: 5px; width: 50%;"', $html_diff ); $html_diff = str_replace( '', '', $html_diff ); $html_diff = str_replace( '', '', $html_diff ); $html_diff = str_replace( '', '', $html_diff ); $html_diff = str_replace( array( '', '', '' ), array( "\n", "\n", "\n" ), $html_diff ); $html_diff = $html_diff_head . $html_diff; // Refactor some of the meta data for TEXT $length = max( strlen( $left_title ), strlen( $right_title ) ); $left_title = str_pad( $left_title, $length + 2 ); $right_title = str_pad( $right_title, $length + 2 ); // TEXT $text_diff = sprintf( $head_sprintf, $the_author, '"' . $the_title . '"', $this->right_post->post_type, $the_date ) . "\n"; $text_diff .= "URL: $the_permalink\n"; $text_diff .= "Edit: $the_edit_link\n\n"; foreach ( $text_diffs as $field_title => $diff ) { $text_diff .= "$field_title\n"; $text_diff .= "===================================================================\n"; $text_diff .= "--- $left_title ({$this->left_post->post_date_gmt})\n"; $text_diff .= "+++ $right_title ({$this->right_post->post_modified_gmt})\n"; $text_diff .= "$diff\n\n"; } $this->text_diff = $text_diff = rtrim( $text_diff ); // Send email $charset = apply_filters( 'wp_mail_charset', get_option( 'blog_charset' ) ); $blogname = html_entity_decode( get_option( 'blogname' ), ENT_QUOTES, $charset ); $title = html_entity_decode( $the_title, ENT_QUOTES, $charset ); add_action( 'phpmailer_init', array( $this, 'phpmailer_init' ) ); $user_emails = array(); foreach( $options['users'] as $user_id ) { if ( function_exists( 'is_multisite' ) && is_multisite() ) { if ( is_user_member_of_blog( $user_id, get_current_blog_id() ) ) $user_emails[] = get_user_option( 'user_email', $user_id ); } else { if ( $user_email = get_user_option( 'user_email', $user_id ) ) $user_emails[] = $user_email; } } $emails = array_unique( array_merge( $options['emails'], $user_emails ) ); if ( ! count( $emails ) && apply_filters( 'email_post_changes_admin_email_fallback', true ) ) $emails[] = get_option( 'admin_email' ); $emails = apply_filters( 'email_post_changes_emails', $emails, $this->left_post->ID, $this->right_post->ID ); foreach ( $emails as $email ) { wp_mail( $email, sprintf( __( '[%s] %s changed: %s' ), $blogname, $post_type, $title ), $html_diff ); } remove_action( 'phpmailer_init', array( &$this, 'phpmailer_init' ) ); do_action( 'email_post_changes_email_sent' ); } // Ov3rfly: function nice_post_status( $post_status ) { $nice = 'Unknown'; switch( $post_status ) { case 'publish': $nice = _x( 'Published', 'post' ); break; case 'future': $nice = _x( 'Scheduled', 'post' ); break; case 'draft': $nice = _x( 'Draft', 'post' ); break; case 'pending': $nice = _x( 'Pending', 'post' ); break; case 'private': $nice = _x( 'Private', 'post' ); break; case 'trash': $nice = _x( 'Trash', 'post' ); break; case 'auto-draft': case 'inherit': $nice = $post_status; break; } return $nice; } function phpmailer_init( &$phpmailer ) { $phpmailer->AltBody = $this->text_diff; $phpmailer->AddReplyTo( get_the_author_meta( 'email', $this->right_post->post_author ), get_the_author_meta( 'display_name', $this->right_post->post_author ) ); } function get_post_type_label( $post_type ) { // 2.9 if ( !function_exists( 'get_post_type_object' ) ) return ucwords( str_replace( '_', ' ', $post_type ) ); // 3.0 $post_type_object = get_post_type_object( $post_type ); if ( empty( $post_type_object->label ) ) return ucwords( str_replace( '_', ' ', $post_type ) ); return $post_type_object->label; } /* Admin */ function admin_menu() { register_setting( self::OPTION_GROUP, self::OPTION, array( $this, 'validate_options' ) ); add_settings_section( self::ADMIN_PAGE, __( 'Email Post Changes' ), array( $this, 'settings_section' ), self::ADMIN_PAGE ); add_settings_field( self::ADMIN_PAGE . '_enable', __( 'Enable' ), array( $this, 'enable_setting' ), self::ADMIN_PAGE, self::ADMIN_PAGE ); add_settings_field( self::ADMIN_PAGE . '_users', __( 'Users to Email' ), array( $this, 'users_setting' ), self::ADMIN_PAGE, self::ADMIN_PAGE ); add_settings_field( self::ADMIN_PAGE . '_emails', __( 'Additional Email Addresses' ), array( $this, 'emails_setting' ), self::ADMIN_PAGE, self::ADMIN_PAGE ); add_settings_field( self::ADMIN_PAGE . '_post_types', __( 'Post Types' ), array( $this, 'post_types_setting' ), self::ADMIN_PAGE, self::ADMIN_PAGE ); add_settings_field( self::ADMIN_PAGE . '_drafts', __( 'Drafts' ), array( $this, 'drafts_setting' ), self::ADMIN_PAGE, self::ADMIN_PAGE ); add_options_page( __( 'Email Post Changes' ), __( 'Email Post Changes' ), 'manage_options', self::ADMIN_PAGE, array( $this, 'admin_page' ) ); } function validate_options( $options ) { if ( !$options || !is_array( $options ) ) return $this->defaults; $return = array(); $return['enable'] = ( empty( $options['enable'] ) ) ? 0 : 1; if ( empty( $options['users'] ) || !is_array( $options ) ) { $return['users'] = $this->defaults['users']; } else { $return['users'] = $options['users']; } if ( empty( $options['emails'] ) ) { if ( count( $return['users'] ) ) $return['emails'] = array(); else $return['emails'] = $this->defaults['emails']; } else { $_emails = is_string( $options['emails'] ) ? preg_split( '(\n|\r)', $options['emails'], -1, PREG_SPLIT_NO_EMPTY ) : array(); $_emails = array_unique( $_emails ); $emails = array_filter( $_emails, 'is_email' ); $invalid_emails = array_diff( $_emails, $emails ); if ( $invalid_emails ) $return['invalid_emails'] = $invalid_emails; if ( $emails ) $return['emails'] = $emails; elseif ( count( $return['users'] ) ) $return['emails'] = array(); else $return['emails'] = $this->defaults['emails']; // Don't store a huge list of invalid emails addresses in the option if ( isset ( $return['invalid_emails'] ) && count( $return['invalid_emails'] ) > 200 ) { $return['invalid_emails'] = array_slice( $return['invalid_emails'], 0, 200 ); $return['invalid_emails'][] = __( 'and many more not listed here' ); } // Cap to at max 200 email addresses if ( count( $return['emails'] ) > 200 ) { $return['emails'] = array_slice( $return['emails'], 0, 200 ); } } if ( empty( $options['post_types'] ) || !is_array( $options ) ) { $return['post_types'] = $this->defaults['post_types']; } else { $post_types = array_intersect( $options['post_types'], $this->get_post_types() ); $return['post_types'] = $post_types ? $post_types : $this->defaults['post_types']; } $return['drafts'] = ( empty( $options['drafts'] ) ) ? 0 : 1; do_action( 'email_post_changes_validate_options', $this->get_options(), $return ); return $return; } function admin_page() { $options = $this->get_options(); ?>

' . join( ', ', array_map( 'esc_html', $options['invalid_emails'] ) ) ); ?>

get_options(); ?>

get_options(); ?>
display_name ), strtolower( $b->display_name ) ); } function emails_setting() { $options = $this->get_options(); ?>

get_options(); ?>
    get_post_types() as $post_type ) : $label = $this->get_post_type_label( $post_type ); ?>
get_options(); ?>

'', 'title_left' => '', 'title_right' => '' ); $args = wp_parse_args( $args, $defaults ); $left_string = normalize_whitespace( $left_string ); $right_string = normalize_whitespace( $right_string ); $left_lines = explode( "\n", $left_string ); $right_lines = explode( "\n", $right_string ); $text_diff = new Text_Diff( $left_lines, $right_lines ); $renderer = new Email_Post_Changes_Diff(); $diff = $renderer->render( $text_diff ); if ( !$diff ) return ''; $r = "\n"; $r .= ""; if ( $args['title'] || $args['title_left'] || $args['title_right'] ) $r .= ""; if ( $args['title'] ) $r .= "\n"; if ( $args['title_left'] || $args['title_right'] ) { $r .= "\n"; $r .= "\t\n"; $r .= "\t\n"; $r .= "\n"; } if ( $args['title'] || $args['title_left'] || $args['title_right'] ) $r .= "\n"; $r .= "\n$diff\n\n"; $r .= "
$args[title]
$args[title_left]$args[title_right]
"; return $r; } } if ( !class_exists( 'WP_Text_Diff_Renderer_Table' ) ) require( ABSPATH . WPINC . '/wp-diff.php' ); class Email_Post_Changes_Diff extends WP_Text_Diff_Renderer_Table { var $_leading_context_lines = 2; var $_trailing_context_lines = 2; }