123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447 |
- <?php
- if ( ! defined( 'ABSPATH' ) ) {
- exit; // Exit if accessed directly
- }
- if ( ! class_exists( 'acf_revisions' ) ) :
- class acf_revisions {
- // vars
- var $cache = array();
- /*
- * __construct
- *
- * A good place to add actions / filters
- *
- * @type function
- * @date 11/08/13
- *
- * @param N/A
- * @return N/A
- */
- function __construct() {
- // actions
- add_action( 'wp_restore_post_revision', array( $this, 'wp_restore_post_revision' ), 10, 2 );
- // filters
- add_filter( 'wp_save_post_revision_check_for_changes', array( $this, 'wp_save_post_revision_check_for_changes' ), 10, 3 );
- add_filter( '_wp_post_revision_fields', array( $this, 'wp_preview_post_fields' ), 10, 2 );
- add_filter( '_wp_post_revision_fields', array( $this, 'wp_post_revision_fields' ), 10, 2 );
- add_filter( 'acf/validate_post_id', array( $this, 'acf_validate_post_id' ), 10, 2 );
- }
- /*
- * wp_preview_post_fields
- *
- * This function is used to trick WP into thinking that one of the $post's fields has changed and
- * will allow an autosave to be updated.
- * Fixes an odd bug causing the preview page to render the non autosave post data on every odd attempt
- *
- * @type function
- * @date 21/10/2014
- * @since 5.1.0
- *
- * @param $fields (array)
- * @return $fields
- */
- function wp_preview_post_fields( $fields ) {
- // bail early if not previewing a post
- if ( acf_maybe_get_POST( 'wp-preview' ) !== 'dopreview' ) {
- return $fields;
- }
- // add to fields if ACF has changed
- if ( acf_maybe_get_POST( '_acf_changed' ) ) {
- $fields['_acf_changed'] = 'different than 1';
- }
- // return
- return $fields;
- }
- /*
- * wp_save_post_revision_check_for_changes
- *
- * This filter will return false and force WP to save a revision. This is required due to
- * WP checking only post_title, post_excerpt and post_content values, not custom fields.
- *
- * @type filter
- * @date 19/09/13
- *
- * @param $return (boolean) defaults to true
- * @param $last_revision (object) the last revision that WP will compare against
- * @param $post (object) the $post that WP will compare against
- * @return $return (boolean)
- */
- function wp_save_post_revision_check_for_changes( $return, $last_revision, $post ) {
- // if acf has changed, return false and prevent WP from performing 'compare' logic
- if ( acf_maybe_get_POST( '_acf_changed' ) ) {
- return false;
- }
- // return
- return $return;
- }
- /*
- * wp_post_revision_fields
- *
- * This filter will add the ACF fields to the returned array
- * Versions 3.5 and 3.6 of WP feature different uses of the revisions filters, so there are
- * some hacks to allow both versions to work correctly
- *
- * @type filter
- * @date 11/08/13
- *
- * @param $post_id (int)
- * @return $post_id (int)
- */
- function wp_post_revision_fields( $fields, $post = null ) {
- // validate page
- if ( acf_is_screen( 'revision' ) || acf_is_ajax( 'get-revision-diffs' ) ) {
- // bail early if is restoring
- if ( acf_maybe_get_GET( 'action' ) === 'restore' ) {
- return $fields;
- }
- // allow
- } else {
- // bail early (most likely saving a post)
- return $fields;
- }
- // vars
- $append = array();
- $order = array();
- $post_id = acf_maybe_get( $post, 'ID' );
- // compatibility with WP < 4.5 (test)
- if ( ! $post_id ) {
- global $post;
- $post_id = $post->ID;
- }
- // get all postmeta
- $meta = get_post_meta( $post_id );
- // bail early if no meta
- if ( ! $meta ) {
- return $fields;
- }
- // loop
- foreach ( $meta as $name => $value ) {
- // attempt to find key value
- $key = acf_maybe_get( $meta, '_' . $name );
- // bail early if no key
- if ( ! $key ) {
- continue;
- }
- // update vars
- $value = $value[0];
- $key = $key[0];
- // Load field.
- $field = acf_get_field( $key );
- if ( ! $field ) {
- continue;
- }
- // get field
- $field_title = $field['label'] . ' (' . $name . ')';
- $field_order = $field['menu_order'];
- $ancestors = acf_get_field_ancestors( $field );
- // ancestors
- if ( ! empty( $ancestors ) ) {
- // vars
- $count = count( $ancestors );
- $oldest = acf_get_field( $ancestors[ $count - 1 ] );
- // update vars
- $field_title = str_repeat( '- ', $count ) . $field_title;
- $field_order = $oldest['menu_order'] . '.1';
- }
- // append
- $append[ $name ] = $field_title;
- $order[ $name ] = $field_order;
- // hook into specific revision field filter and return local value
- add_filter( "_wp_post_revision_field_{$name}", array( $this, 'wp_post_revision_field' ), 10, 4 );
- }
- // append
- if ( ! empty( $append ) ) {
- // vars
- $prefix = '_';
- // add prefix
- $append = acf_add_array_key_prefix( $append, $prefix );
- $order = acf_add_array_key_prefix( $order, $prefix );
- // sort by name (orders sub field values correctly)
- array_multisort( $order, $append );
- // remove prefix
- $append = acf_remove_array_key_prefix( $append, $prefix );
- // append
- $fields = $fields + $append;
- }
- // return
- return $fields;
- }
- /*
- * wp_post_revision_field
- *
- * This filter will load the value for the given field and return it for rendering
- *
- * @type filter
- * @date 11/08/13
- *
- * @param $value (mixed) should be false as it has not yet been loaded
- * @param $field_name (string) The name of the field
- * @param $post (mixed) Holds the $post object to load from - in WP 3.5, this is not passed!
- * @param $direction (string) to / from - not used
- * @return $value (string)
- */
- function wp_post_revision_field( $value, $field_name, $post = null, $direction = false ) {
- // bail early if is empty.
- if ( empty( $value ) ) {
- return $value;
- }
- $value = maybe_unserialize( $value );
- $post_id = $post->ID;
- // load field.
- $field = acf_maybe_get_field( $field_name, $post_id );
- // default formatting.
- if ( is_array( $value ) ) {
- $value = implode( ', ', $value );
- } elseif ( is_object( $value ) ) {
- $value = serialize( $value );
- }
- // image.
- if ( is_array( $field ) && isset( $field['type'] ) && ( $field['type'] === 'image' || $field['type'] === 'file' ) ) {
- $url = wp_get_attachment_url( $value );
- $value = $value . ' (' . $url . ')';
- }
- return $value;
- }
- /*
- * wp_restore_post_revision
- *
- * This action will copy and paste the metadata from a revision to the post
- *
- * @type action
- * @date 11/08/13
- *
- * @param $parent_id (int) the destination post
- * @return $revision_id (int) the source post
- */
- function wp_restore_post_revision( $post_id, $revision_id ) {
- // copy postmeta from revision to post (restore from revision)
- acf_copy_postmeta( $revision_id, $post_id );
- // Make sure the latest revision is also updated to match the new $post data
- // get latest revision
- $revision = acf_get_post_latest_revision( $post_id );
- // save
- if ( $revision ) {
- // copy postmeta from revision to latest revision (potentialy may be the same, but most likely are different)
- acf_copy_postmeta( $revision_id, $revision->ID );
- }
- }
- /*
- * acf_validate_post_id
- *
- * This function will modify the $post_id and allow loading values from a revision
- *
- * @type function
- * @date 6/3/17
- * @since 5.5.10
- *
- * @param $post_id (int)
- * @param $_post_id (int)
- * @return $post_id (int)
- */
- function acf_validate_post_id( $post_id, $_post_id ) {
- // phpcs:disable WordPress.Security.NonceVerification.Recommended
- // bail early if no preview in URL
- if ( ! isset( $_GET['preview'] ) ) {
- return $post_id;
- }
- // bail early if $post_id is not numeric
- if ( ! is_numeric( $post_id ) ) {
- return $post_id;
- }
- // vars
- $k = $post_id;
- $preview_id = 0;
- // check cache
- if ( isset( $this->cache[ $k ] ) ) {
- return $this->cache[ $k ];
- }
- // validate
- if ( isset( $_GET['preview_id'] ) ) {
- $preview_id = (int) $_GET['preview_id'];
- } elseif ( isset( $_GET['p'] ) ) {
- $preview_id = (int) $_GET['p'];
- } elseif ( isset( $_GET['page_id'] ) ) {
- $preview_id = (int) $_GET['page_id'];
- }
- // phpcs:enable WordPress.Security.NonceVerification.Recommended
- // bail early id $preview_id does not match $post_id
- if ( $preview_id != $post_id ) {
- return $post_id;
- }
- // attempt find revision
- $revision = acf_get_post_latest_revision( $post_id );
- // save
- if ( $revision && $revision->post_parent == $post_id ) {
- $post_id = (int) $revision->ID;
- }
- // set cache
- $this->cache[ $k ] = $post_id;
- // return
- return $post_id;
- }
- }
- // initialize
- acf()->revisions = new acf_revisions();
- endif; // class_exists check
- /*
- * acf_save_post_revision
- *
- * This function will copy meta from a post to it's latest revision
- *
- * @type function
- * @date 26/09/2016
- * @since 5.4.0
- *
- * @param $post_id (int)
- * @return n/a
- */
- function acf_save_post_revision( $post_id = 0 ) {
- // get latest revision
- $revision = acf_get_post_latest_revision( $post_id );
- // save
- if ( $revision ) {
- acf_copy_postmeta( $post_id, $revision->ID );
- }
- }
- /*
- * acf_get_post_latest_revision
- *
- * This function will return the latest revision for a given post
- *
- * @type function
- * @date 25/06/2016
- * @since 5.3.8
- *
- * @param $post_id (int)
- * @return $post_id (int)
- */
- function acf_get_post_latest_revision( $post_id ) {
- // vars
- $revisions = wp_get_post_revisions( $post_id );
- // shift off and return first revision (will return null if no revisions)
- $revision = array_shift( $revisions );
- // return
- return $revision;
- }
|