' . $message, $linkStart, $linkEnd);
+?>
+
\ No newline at end of file
diff --git a/wp-content/plugins/revisionary/mail-buffer_rvy.php b/wp-content/plugins/revisionary/mail-buffer_rvy.php
new file mode 100644
index 000000000..cbebbcb59
--- /dev/null
+++ b/wp-content/plugins/revisionary/mail-buffer_rvy.php
@@ -0,0 +1,169 @@
+ $default_minute_limit,
+ 'hour' => $default_hour_limit,
+ 'day' => $default_day_limit,
+ ]
+ );
+
+ return $send_limits;
+}
+
+function _rvy_mail_check_buffer($new_msg = [], $args = []) {
+ global $wpdb;
+
+ $log_only = !empty($args['log_only']);
+
+ if (!$log_only) {
+ wp_cache_delete('revisionary_mail_buffer', 'options');
+
+ // @todo: re-enable buffer after troubleshooting for working copy redirect error
+
+ if (true) {
+ $buffer = [];
+ $first_buffer = true;
+ }
+ }
+
+ $new_msg_buffered = false;
+
+ wp_cache_delete('revisionary_sent_mail', 'options');
+
+ if (!$sent_mail = get_option('revisionary_sent_mail')) {
+ $sent_mail = [];
+ $first_mail_log = true;
+ }
+
+ $current_time = time();
+
+ // check sending limits
+ $durations = ['minute' => 60, 'hour' => 3600, 'day' => 86400];
+ $sent_counts = ['minute' => 0, 'hour' => 0, 'day' => 0];
+
+ // by default, purge mail log entries older than 30 days
+ // @todo: purge mail log even when buffer is disabled
+ $purge_time = apply_filters('revisionary_mail_log_duration', 86400 * 30);
+
+ if ($purge_time < $durations['day'] * 2) {
+ $purge_time = $durations['day'] * 2;
+ }
+
+ $send_limits = _rvy_mail_send_limits();
+
+ foreach($sent_mail as $k => $mail) {
+ if (!isset($mail['time_gmt'])) {
+ continue;
+ }
+
+ $elapsed = $current_time - $mail['time_gmt'];
+
+ foreach($durations as $limit_key => $duration) {
+ if ($elapsed < $duration) {
+ $sent_counts[$limit_key]++;
+ }
+
+ if ($new_msg && ($sent_counts[$limit_key] >= $send_limits[$limit_key])) {
+ $new_msg_buffered = true;
+ }
+ }
+
+ if ($elapsed > $purge_time) {
+ unset($sent_mail[$k]);
+ $purged = true;
+ }
+ }
+
+ if (!$log_only && $new_msg_buffered) {
+ $buffer = array_merge([$new_msg], $buffer);
+ update_option('revisionary_mail_buffer', $buffer);
+ } else {
+ $buffer = [];
+ }
+
+ if (!empty($purged)) {
+ update_option('revisionary_sent_mail', $sent_mail);
+ }
+
+ if (!empty($first_mail_log) && $sent_mail) {
+ $wpdb->query("UPDATE $wpdb->options SET autoload = 'no' WHERE option_name = 'revisionary_sent_mail'");
+ }
+
+ if (!empty($first_buffer) && $buffer) {
+ $wpdb->query("UPDATE $wpdb->options SET autoload = 'no' WHERE option_name = 'revisionary_mail_buffer'");
+ }
+
+ return (object) compact('buffer', 'sent_mail', 'send_limits', 'sent_counts', 'new_msg_buffered');
+}
+
+// called by WP-cron hook
+function _rvy_send_buffered_mail() {
+ $buffer_status = rvy_mail_check_buffer();
+
+ if (empty($buffer_status->buffer)) {
+ return false;
+ }
+
+ $q = $buffer_status->buffer;
+
+ while ($q) {
+ foreach($buffer_status->sent_counts as $limit_key => $count) {
+ $buffer_status->sent_counts[$limit_key]++;
+
+ if ($count > $buffer_status->send_limits[$limit_key]) {
+ // A send limit has been reached
+ break 2;
+ }
+ }
+
+ $next_mail = array_pop($q);
+
+ // update truncated buffer immediately to prevent duplicate sending by another process
+ update_option('revisionary_mail_buffer', $q);
+
+ // If buffered notification is missing vital data, discard it
+ if (empty($next_mail['address']) || empty($next_mail['title']) || empty($next_mail['message']) || empty($next_mail['time_gmt'])) {
+ continue;
+ }
+
+ // If notification was buffered more than a week ago, discard it
+ if (time() - $next_mail['time_gmt'] > 3600 * 24 * 7 ) {
+ continue;
+ }
+
+ if (defined('RS_DEBUG')) {
+ $success = wp_mail($next_mail['address'], $next_mail['title'], $next_mail['message']);
+ } else {
+ $success = @wp_mail($next_mail['address'], $next_mail['title'], $next_mail['message']);
+ }
+
+ if (!$success && defined('REVISIONARY_MAIL_RETRY')) {
+ // message was not sent successfully, so put it back in the buffer
+ if ($q) {
+ $q = array_merge([$next_mail], $q);
+ } else {
+ $q = [$next_mail];
+ }
+ update_option('revisionary_mail_buffer', $q);
+ } else {
+ // log the sent mail
+ $next_mail['time'] = strtotime(current_time( 'mysql' ));
+ $next_mail['time_gmt'] = time();
+ $next_mail['success'] = intval(boolval($success));
+
+ if (!defined('RS_DEBUG') && !defined('REVISIONARY_LOG_EMAIL_MESSAGE')) {
+ unset($next_mail['message']);
+ }
+
+ $buffer_status->sent_mail[]= $next_mail;
+ update_option('revisionary_sent_mail', $buffer_status->sent_mail);
+ }
+ }
+}
diff --git a/wp-content/plugins/revisionary/multiple-authors_rvy.php b/wp-content/plugins/revisionary/multiple-authors_rvy.php
new file mode 100644
index 000000000..ecae4e647
--- /dev/null
+++ b/wp-content/plugins/revisionary/multiple-authors_rvy.php
@@ -0,0 +1,61 @@
+term_id)) {
+ return;
+ }
+
+ $author = $author->term_id;
+ }
+
+ $taxonomy = (!empty($multiple_authors_addon) && !empty($multiple_authors_addon->coauthor_taxonomy))
+ ? $multiple_authors_addon->coauthor_taxonomy
+ : 'author';
+
+ //$author = Author::get_by_term_id($author); // this returns an object with term_id property and no name
+ //$author = get_term($author, 'author'); // 'author' is actually an invalid taxonomy name per WP API
+ $author = $wpdb->get_row(
+ $wpdb->prepare(
+ "SELECT * FROM $wpdb->terms AS t INNER JOIN $wpdb->term_taxonomy AS tt ON t.term_id = tt.term_id"
+ . " WHERE tt.taxonomy = %s AND t.term_id = %d"
+ , $taxonomy, $author
+ )
+ );
+
+ if (!empty($author->name)) {
+ $names[] = $author->name;
+ }
+ }
+
+ if (!empty($names)) {
+ $names = implode(', ', $names);
+ rvy_update_post_meta($post_id, $metadata, $names);
+ } else {
+ delete_post_meta($post_id, $metadata);
+ }
+ }
+}
diff --git a/wp-content/plugins/revisionary/readme.txt b/wp-content/plugins/revisionary/readme.txt
new file mode 100644
index 000000000..ec29a94dd
--- /dev/null
+++ b/wp-content/plugins/revisionary/readme.txt
@@ -0,0 +1,1281 @@
+=== PublishPress Revisions: Duplicate Posts, Submit, Approve and Schedule Content Changes ===
+
+Contributors: publishpress, kevinB, stevejburge, andergmartins
+Author: PublishPress
+Author URI: https://publishpress.com
+Tags: revision, submit changes, duplicate post, duplicate page, revisions, approve changes, scheduled changes
+Requires at least: 5.5
+Requires PHP: 7.2.5
+Tested up to: 6.5
+Stable tag: 3.5.8.2
+License: GPLv2 or later
+License URI: http://www.gnu.org/licenses/gpl-2.0.html
+
+PublishPress Revisions gives you control over updating published content. Users can duplicate posts and submit changes. You can approve, reject or schedule them.
+
+== Description ==
+
+PublishPress Revisions is the ultimate tool for making content changes. Users get a safe space to work on content updates. When the changes are ready, they can ask for approval to publish the update. Administrators can approve or reject the changes. They can also schedule them to go live in the future.
+
+Your users can update posts using the normal WordPress editor, but their changes will not be published automatically. Instead, the changes are stored in a "Revision Queue". The changes can be approved, rejected or scheduled. PublishPress Revisions works with the familiar WordPress interface, including Gutenberg and the Classic Editor.
+
+## PublishPress Revisions Pro ##
+
+>
Upgrade to Revisions Pro
+> This plugin is the free version of the Revisions Pro plugin. The Free version of PublishPress Revisions supports the WordPress core. The Pro version of PublishPress Revisions supports ACF, Yoast SEO, PODS, WooCommerce, WPML, Elementor, Beaver Builder, and many more plugins and themes.
Click here to purchase the best premium WordPress content update plugin now!
+
+## Submit Change Requests for Posts ####
+PublishPress Revisions allows your users to submit change requests for published posts. Your users can update posts using the normal WordPress editor, but their changes will not be published automatically. Instead, the changes are stored as a "pending revision" that can be approved or rejected.
+
+[Click here to see how to submit revisions](https://publishpress.com/knowledge-base/revisions-start/).
+
+## Schedule Future Updates to Posts ##
+
+PublishPress Revisions allows you to schedule WordPress revisions to be published in the future. When you're editing a published post, all you need to do is select a future date and click "Schedule Revision". Your changes will be published at the specified time.
+
+[Click here to see how to schedule revisions](https://publishpress.com/knowledge-base/schedule-revisions-future/).
+
+## Manage and Moderate Changes to Content ##
+
+After you create a revision with PublishPress Revisions, you can find that revision on the Revision Queue screen. This screen shows you all the revisions that have been submitted for approval. Underneath each duplicate post you can choose from several moderation tools: Edit, Delete, Preview and Compare.
+
+[Click here to see how to manage and moderate revisions](https://publishpress.com/knowledge-base/schedule-or-publish-revisions/).
+
+## Compare Current and Former Versions ##
+Pending and Scheduled Revisions can include changes to post content, categories, tags, featured image, page parent and other options. Each of these changes can be reviewed in the familiar Compare Revisions interface.
+
+[Click here to see how to compare revisions](https://publishpress.com/knowledge-base/compare-revisions/).
+
+## Frontend Moderation of Content Changes ##
+
+It is possible to preview and moderate revisions via the frontend of your WordPress site. If you click Preview for a pending revision, you'll see a toolbar across the frontend of the site. This toolbar will change color so you can easily know the status of the revision. For example, if you're looking at a pending revision, the toolbar will be green. For scheduled revisions, the toolbar will be grey.
+
+[Click here to see how to manage from the frontend of your site](https://publishpress.com/knowledge-base/publishing-revisions-frontend/).
+
+## Email Notifications for Revisions ##
+
+PublishPress Revisions will notify Administrators and Editors when a new revision is submitted. They can log in to preview, compare and approve the changes. PublishPress Revisions can also send emails for revision approval and publication. The Settings screen lets you disable unwanted notifications.
+
+[Click here for more on revision notifications](https://publishpress.com/knowledge-base/emails-revisionary/).
+
+## Additional Plugins Supported in the Pro Version ##
+
+Page builder / front end editing integrations:
+
+* BeaverBuilder integration
+* Divi Theme, Divi Builder integration
+* Elementor integration
+
+Major plugins we maintain compatibility with:
+
+* Advanced Custom Fields
+* Pods
+* WPML
+* The Events Calendar
+* WooCommerce
+* Yoast SEO
+
+[Click here for the full list of supported plugins in the Pro version](https://publishpress.com/knowledge-base/plugins-revisions-support/).
+
+## You Can Set Very Detailed Revision Permissions ##
+
+PublishPress Revisions works with the default WordPress user roles, and also introduces a Revisor role:
+
+* Contributors can submit revisions to their own published posts.
+* Revisors can submit revisions to posts and pages published by others.
+* Authors, Editors and Administrators can approve revisions or schedule their own revisions.
+
+To schedule changes to a published post, just set the desired future date before hitting Update.
+
+By upgrading to Revisions Pro, you also gain advanced permissions control through the PublishPress Permissions Pro plugin. You can customize permissions by role or per-user, granting full editing or revision submission rights to specific posts, categories, or taxonomy terms.
+
+[Click here for more on revision permissions](https://publishpress.com/knowledge-base/permissions-revisions).
+
+## Join PublishPress and get the Pro plugins ##
+
+The Pro versions of the PublishPress plugins are well worth your investment. The Pro versions have extra features and faster support. [Click here to join PublishPress](https://publishpress.com/pricing/).
+
+Join PublishPress and you'll get access to these nine Pro plugins:
+
+* [PublishPress Authors Pro](https://publishpress.com/authors) allows you to add multiple authors and guest authors to WordPress posts.
+* [PublishPress Blocks Pro](https://publishpress.com/blocks) has everything you need to build professional websites with the WordPress block editor.
+* [PublishPress Capabilities Pro](https://publishpress.com/capabilities) is the plugin to manage your WordPress user roles, permissions, and capabilities.
+* [PublishPress Checklists Pro](https://publishpress.com/checklists) enables you to define tasks that must be completed before content is published.
+* [PublishPress Future Pro](https://publishpress.com/future) is the plugin for scheduling changes to your posts.
+* [PublishPress Permissions Pro](https://publishpress.com/permissions) is the plugin for advanced WordPress permissions.
+* [PublishPress Planner Pro](https://publishpress.com/publishpress) is the plugin for managing and scheduling WordPress content.
+* [PublishPress Revisions Pro](https://publishpress.com/revisions) allows you to update your published pages with teamwork and precision.
+* [PublishPress Series Pro](https://publishpress.com/series) enables you to group content together into a series
+
+Together, these plugins are a suite of powerful publishing tools for WordPress. If you need to create a professional workflow in WordPress, with moderation, revisions, permissions and more... then you should try PublishPress.
+
+## Bug Reports ##
+Bug reports for PublishPress Revisions are welcomed in our [repository on GitHub](https://github.com/publishpress/publishpress-revisions). Please note that GitHub is not a support forum, and that issues that aren't properly qualified as bugs will be closed.
+
+## Follow the PublishPress team ##
+
+Follow PublishPress on [Facebook](https://www.facebook.com/publishpress), [Twitter](https://www.twitter.com/publishpresscom) and [YouTube](https://www.youtube.com/publishpress)
+
+== Screenshots ==
+
+1. Schedule Revisions: PublishPress Revisions allows you to schedule WordPress revisions to be published in the future.
+2. Control Revision Permissions: PublishPress Revisions allows you to control which users are able to submit revisions to published posts.
+3. Submit Revisions: PublishPress Revisions enables you to submit change requests for published posts.
+4. Preview and Compare Revisions: Every requested change can be reviewed in the normal "Compare Revisions" interface in WordPress.
+5. Manage and Moderate Revisions: PublishPress Revisions has a Revision Queue screen where you can search, approve, or deny changes to a published post's content.
+6. Email Notifications for Revisions: Administrators and Editors are notified by email when a new revision is submitted. PublishPress Revisions can also send emails for revision approval and publication.
+7. Support for Plugins: Revisions Pro adds support for the plugins such as Yoast SEO, WooCommrce and Advanced Custom Fields.
+8. Divi, Beaver Builder Integration: With Revisions Pro, your Contributors and Revisors will be able to submit revisions to published content using the front end editor of the Divi theme or Beaver Builder plugin.
+9. WPML Integration: With Revisions Pro, pending and scheduled revisions work in multi-language sites controlled by the WPML plugin.
+10. Advanced Permissions: By upgrading to Revisions Pro, you also gain advanced permissions control through Permissions Pro. Customize permissions by role or per-user.
+
+== Frequently Asked Questions ==
+
+= How do I duplicate pages or other post types? =
+
+PublishPress Revisions can duplicate pages and any other post types. For example, to duplicate pages, go to the "Pages" screen in WordPress and click "New Revision" next to any page. You will be taken to a duplicate page that you can edit and save to the Revision Queue.
+
+= How do I duplicate WooCommerce products? =
+
+WooCommerce is the most popular way to build a WordPress eCommerce site. With PublishPress Revisions Pro you can submit, moderate, approve and schedule revisions to your WooCommerce products.
+
+When you are on the "Products" screen, click the "New Revision" link. Inside the product revision, the PublishPress buttons will appear in the top-right corner. When you save the product, it will appear on the main "Revision Queue" screen.
+
+It is possible to schedule updates to WooCommerce products.
+
+- While updating an post or page created created with Elementor, click "Immediately" in the right sidebar.
+- Choose the new date for these changes to go live.
+- Click the "Update Revision" button.
+
+All key WooCommerce fields are supported by PublishPress Revisions Pro. If you are using 3rd party WooCommerce plugins, we do recommend testing their compatibility with WordPress revisions and PublishPress Revisions Pro.
+
+[Click here for more on duplicate WooCommerce products](https://publishpress.com/knowledge-base/permissions-revisions).
+
+= How do I duplicate Elementor pages and posts? =
+
+PublishPress Revisions Pro has support for the Elementor plugin.
+
+Here's how to use PublishPress Revisions and Elementor together:
+
+- Find and a published post or page that was created with Elementor.
+- You will see a "New revision" link.
+- On the next screen, you can click "Edit with Elementor" to make changes.
+- You can use "Submit Revision" in the right sidebar to submit your changes.
+- In the WordPress admin area, you can go to visit the "Revision Queue" screen and you will see the submitted revision of your Elementor page. It will be waiting for your approval.
+
+It is possible to schedule content updates to Elementor posts.
+
+- While updating an post or page created created with Elementor, click "Immediately" in the right sidebar.
+- Choose the new date for these changes to go live.
+- Click the "Update Revision" button.
+
+[Click here for more on duplicate Elementor posts (https://publishpress.com/knowledge-base/publishpress-revisions-elementor/).
+
+= How do I duplicate The Events Calendar events? =
+
+The Events Calendar plugin is one the most popular ways to create and list events in WordPress. PublishPress Revisions Pro and The Events Calendar work together successfully.
+
+- Make sure you are using the Gutenberg option for creating events.
+- After installing PublishPress Revisions Pro, you will be able to submit revisions to events using the New Revision option on the main Events screen.
+- You will see the normal Event screen, but with additional PublishPress Revisions button in the top-right corner.
+- Your event revisions will appear in the "Revision Queue".
+- You will not be able to use the "Compare" link because The Events Calendar does not support that WordPress feature yet. However, all the other key features of PublishPress Revisions Pro will work. For example, PublishPress Revisions Pro will record changes to all your event fields and settings.
+
+[Click here for more on duplicate The Events Calendar events (https://publishpress.com/knowledge-base/revisions-events-calendar/).
+
+= How do I duplicate posts with Advanced Custom Fields? =
+
+PublishPress Revisions Pro has support for the Advanced Custom Fields (ACF) plugin. You can schedule, moderate and approve revisions for content that uses ACF.
+
+- To create a revision for post that use ACF, click on the "New Revision" link.
+- Go to the "Revision Queue".
+- Click "Edit" and you can update the revision, including the ACF fields.
+- Click "Compare" and you can see what's changed in this revision. The "Compare" screen will show you any changes that have happened with ACF Fields.
+
+[Click here for more on duplicate posts with Advanced Custom Fields (https://publishpress.com/knowledge-base/revisionary-acf/).
+
+= How do I duplicate posts with WPML? =
+
+PublishPress Revisions Pro has support for the WPML plugin. For example, you can create revisions for a post in English and create separate revisions for the Spanish translation of that post.
+
+- Go to your "Posts" screen.
+- Click "New Revisions" and you can create a translated version of your content in any language.
+
+[Click here for more on duplicate posts with WPML (https://publishpress.com/knowledge-base/revisionary-wpml/).
+
+= How do I duplicate posts with Yoast SEO? =
+
+Yoast SEO is the most popular SEO plugin in WordPress. With PublishPress Revisions Pro you can submit, moderate, approve and schedule revisions to your WordPress posts, and they will keep all your SEO data.
+
+In the image below, you can Yoast SEO data. With PublishPress Revisions Pro you enter this data and it will be saved throughout the revision approval process.
+
+[Click here for more on duplicate posts with Yoast SEO (https://publishpress.com/knowledge-base/revisions-yoast-seo/).
+
+= How do I duplicate posts with Beaver Builder? =
+
+PublishPress Revisions Pro has support for the Beaver Builder plugin. Here's how to use PublishPress Revisions Pro and Beaver Builder together:
+
+- In your WordPress admin area, click "New Revision" for a post or page that uses Beaver Builder.
+- On the next screen, you will be able to click "Launch Beaver Builder" and create a new version of this post.
+- Click "Submit Revision" to submit your updates.
+- In the WordPress admin area, you can visit the "Revision Queue" screen and you will see the submitted revision of your Beaver Builder page. It will be waiting for your approval.
+
+It is also possible to schedule revisions to Beaver Builder content.
+
+- When you're editing a revision, click "Immediately".
+- Choose the date you want your post changes to go live.
+- Click the "Submit Revision" button.
+
+[Click here for more on duplicate posts with Beaver Builder (https://publishpress.com/knowledge-base/revisionary-beaver-builder/).
+
+= How do I duplicate posts with Divi Theme and Builder? =
+
+Divi is one of the most popular page-builders in WordPress, and it does integrate with PublishPress Revisions Pro. It is possible for lower-level users to edit a page with Divi and submit an update for approval. This workflow will apply to anyone with the correct permissions to submit revisions.
+
+- In your WordPress admin area, click "New Revision" for a post or page that uses Divi.
+- You will see the normal editing layout in Divi, with the PublishPress Revisions buttons in the top toolbar and right sidebar. Click "Edit With The Divi Builder" to make content changes.
+
+[Click here for more on duplicate posts with Divi (https://publishpress.com/knowledge-base/divi-theme/).
+
+== Changelog ==
+
+= 3.5.8.2 =
+* Fixed : Nonce failure on revision decline attempts
+* Fixed : Saving with a future date in Classic Editor saves post as Published instead of Scheduled under some conditions
+* Fixed : Revision Queue - If multiple bulk actions are applied, confirmation notice from previous actions is repeated
+* Compat : Nested Pages - Fatal error on bulk deletion of revisions
+
+= 3.5.8.1 - 16 Apr 2024 =
+* Fixed : If limiting to one active revision per post, deleting a revision before saving it did not restore creation ability on some sites
+
+= 3.5.8 - 12 Apr 2024 =
+* Fixed : Revision Editor hid category selection checkboxes
+* Fixed : New Revision button was missing from Revision Editor sidebar
+* Fixed : Revision Editor did not display Preview Revision button
+* Fixed : Revision Preview top bar styling broken by external CSS on some sites
+* Fixed : Revision Preview top bar styling stuck in cache on some sites
+* Fixed : If limiting to one active revision per post, deleting that revision did not restore ability to create a new one
+* Fixed : Revision Queue - Bulk Decline failed with "link expired" message
+* Fixed : New revisions could become orphaned from main post on WordPress VIP
+
+= 3.5.7 - 4 Apr 2024 =
+* Compat : WordPress 6.5 - Revision editor sidebar did not display date, template UI
+* Feature : Decline link in Revision Queue row, Revision Preview top bar
+* Fixed : Preview of front page revision was not displayed correctly
+* Fixed : Previews of archived revisions did not display custom fields, multiple authors
+* Fixed : Revision preview styling broke third party javascript on some installations
+* Fixed : Classic Editor - Custom Fields metabox was unavailable while editing a revision
+* Fixed : WP-defined revision limit (for past revisions) was not imposed when saving current content as a past revision at pending / scheduled revision publication
+* Fixed : If "Limit to one active revision per post" setting is enabled, deleting a revision on a cached site did not restore ability to create a new revision to replace it
+* Fixed : Possible memory leaks in javascript on Revision edit with Gutenberg editor
+* API : New filter "revisionary_preview_link_type"
+
+= 3.5.6 - 4 Jan 2024 =
+* Feature : New setting Revisions > Settings > Preview > "Modify preview link for better theme compatibility". Some themes require one setting, some require the other.
+* Fixed : ACF - Revision update, preview fails if ACF plugin is active
+* Fixed : Edit Revision - Approve, Submit buttons missing in Gutenberg 17
+* Fixed : PHP 8.2: Warning for dynamic property declaration
+
+= 3.5.5 - 12 Dec 2023 =
+* Compat : Advanced Custom Fields - Prevent invalid filtering of revision ID
+* Fixed : Front page template was not loaded for revision previews of front page
+* Fixed : Revision previews were not loaded correctly on some sites
+* Fixed : Pagination setting on Queue, Archive screens was not stored
+* Fixed : Pagination on Archive screen not applied if Revision Deletion option disabled
+* Fixed : Classic Editor - Revision Status no longer displayed after changing publish time
+* Fixed : Classic Editor - Modified publish time discarded if Submit / Approve clicked before Update Revision
+* Fixed : Classic Editor - Revision Submit button was still (mal)functional when disabled
+* Change : Classic Editor - Separate buttons to Preview changes in progress, View / Approve Saved Revision
+
+= 3.5.4 - 17 Oct 2023 =
+* Fixed : Past Revision previews did not display (since 3.5.3)
+* Fixed : Scheduled Revisions were not published under some configurations
+* Fixed : Custom fields with multiple values stored to the same key were not revisioned correctly
+* Fixed : Improve html validation of Revisions Settings screens, possibly resolving rendering issue on some browsers
+* Change : Screen Options on Revision Queue and Revision Archive to set Revisions per Page (previously used setting from Pages screen)
+* Compat : PublishPress Authors - Default author setting overrides actual revision author
+
+= 3.5.3 - 30 Aug 2023 =
+* Feature : Option to enable deletion of revisions in Revision Archive
+* Fixed : Category / taxonomy changes were cleared by revision preview, not applied at revision approval
+* Fixed : Revision Archive included autosaves
+* Fixed : PHP 8.1 - Warning for dynamic property creation in revision preview
+
+= 3.5.2 - 17 Aug 2023 =
+* Fixed : Pro - Fatal error on activation if plugin settings were not previously saved (packaging error reverted fix in 3.5.1)
+* Fixed : Revision Edit - No progress caption was shown after Submit button click
+* Fixed : Options screen - html markup error (missing closing div tag)
+* Lang : Updated Spanish, French, Italian translations
+
+= 3.5.1 - 11 Aug 2023 =
+* Fixed : Bedrock - Unable to activate plugin due to three internal libraries missing from GitHub commit
+
+= 3.5 - 10 Aug 2023 =
+* Fixed : Unscheduling a revision using bulk edit in Revision Queue caused it to become inaccessible
+* Compat : Permalink Manager plugin - Suppress permalink modification in Revision edit
+* Change : Change revision deletion captions to clarify that only the revision is deleted
+* Change: Modified internal vendor library structure for shorter paths (lib/vendor)
+* Change: Free plugin is loaded through internal vendor library
+
+= 3.4.1 - 29 Jun 2023 =
+* Fixed : Incorrect admin menu item display if revision submission is disabled for all post types
+
+= 3.4 - 29 Jun 2023 =
+* Feature : Revision Archive screen
+
+= 3.3 - 19 Jun 2023 =
+* Feature : Revisions > Settings > Revision Creation > "Limit to one active revision per post"
+* Fixed : Plugin compat - dashboard_glance_items was improperly filtered
+* Fixed : Revisions were incorrectly attributed to original page author under some conditions
+* Compat : Revision Queue - Work around unidentified plugin conflict inserting inappropriate clauses into our posts query
+* Compat : PublishPress Authors - Author changes in revision were not applied at revision publication
+* Compat : PublishPress Authors - Revision Queue "Revised By" column always showed original post author
+* Compat : Permissions + WordFence - Add Media query was not filtered based on Permissions settings
+* Fixed : Multisite - Some settings UI were not displayed when plugin on network installations when plugin is not activated network-wide
+* Fixed : Multisite - Settings previously configured for network-wide activation were not displayed on sub-site settings screen even after network deactivation
+* Change : Bump WordPress version requirement to 5.5
+* Change : Bump PHP version requirement to 7.2.5
+* Change : Update vendor libraries based on revised PHP version support, using new internal-vendor structure
+
+= 3.1.13 - 30 Mar 2023 =
+* Fixed : Posts screen - Has Revision caption was not displayed unless "Revision Submission for Unpublished Posts" setting is enabled
+* Fixed : Settings > Notifications > "Show Notification Log / Buffer" link did not redirect back to Notifications tab
+* Change : Minimum PHP version 7.2.5
+
+= 3.1.12 - 23 Mar 2023 =
+* Fixed : Posts were left with invalid "Has Revision" links after publication or deletion of all their revisions
+* Fixed : If Revision Submission for Unpublished Posts is disabled but revisions of draft posts were previously created, those caused a "Has Revisions" label without displaying the revisions in Queue
+* Fixed : Links to Revision Queue for a specific published post (including "Has Revision" links from Posts listing) failed to list unsubmitted revisions
+* Fixed : Revision Compare > Manage : Non-rendered html tags, incorrect table formatting
+* Fixed : Compare Past Revisions - invalid link on Preview / Restore button
+* Fixed : Request Deletion link was displayed even if revisions are not enabled for post type
+
+= 3.1.11 - 9 Feb 2023 =
+* Fixed : Revision preview failure under some conditions
+* Fixed : Avoid PHP Warning on Revision Preview if requested revision cannot be loaded
+* Fixed : On Network-wide multisite activations, Settings screen displayed an unrendered html tag
+* Lang : Update Spanish, French, Italian translations
+
+= 3.1.10 - 22 Dec 2022 =
+* Perf : Improve performance of post revision count, dropdown pages queries
+* Fixed : Revisions were not correctly declined on some cached sites
+
+= 3.1.9 - 15 Dec 2022 =
+* Fixed : Scheduled Revisions missed by WP-Cron were not rescheduled. To disable auto-reschedule, define constant REVISIONARY_DISABLE_WP_CRON_RESTORATION
+* Fixed : Revisions Queue table was missing a "check all" box
+* Fixed : Fatal error on revision approval on some sites
+* Compat : Relevanssi - If filtering is enabled for admin searches, Submitted / Scheduled Revisions are included in search results
+* Compat : PublishPress Future - Attempted revision of post expiration settings caused unexpected behavior; disable for now
+* API : Classic Editor revision editor supports filter revisionary_submit_revision_metabox_classic
+
+= 3.1.8 - 3 Nov 2022 =
+* Fixed : PHP 8 - Unresponsive elements in post editor due to Javascript loading failure
+* Fixed : Avoid fatal error on plugin activation if loading sequence is non-standard
+* Compat : WP Buddy Rich Snippets - PHP error "The script tried to modify a property on an incomplete object" on revision creation
+
+= 3.1.7 - 12 Oct 2022 =
+* Feature : Revision submission and approval handles author selection
+* Fixed : Plugin review request dismissal was not applied
+* Fixed : Gutenberg Editor - for non-English locales, future date selection did not enable Scheduled Revision creation
+* Fixed : Gutenberg Editor - when editing an existing Scheduled Revision, publish date caption shows "Immediately"
+* Fixed : PHP Notice "Trying to get property 'post_type' of non-object" when another plugin hooks into "use_block_editor_for_post" filter
+* Lang : Update Spanish, French, Italian translations
+
+= 3.1.6 - 31 Aug 2022 =
+* Change : Support detection / indication of custom WP-Cron implementations when DISABLE_WP_CRON is set true
+
+= 3.1.5 - 30 Aug 2022 =
+* Fixed : Revision Submission from Gutenberg editor failed on some sites
+* Fixed : Bulk-deleted revisions were not removed from revision count
+* Fixed : PHP Warning for invalid regex if post type's REST registration or request has an invalid route string
+
+= 3.1.4 - 17 Aug 2022 =
+* Fixed : Approve Revision button in Gutenberg editor did not work
+* Fixed : Submit Revision button in Gutenberg editor failed under some conditions
+* Fixed : Detection of DISABLE_WP_CRON constant
+* Fixed : Administrators did not receive revisions submission notifications under some configurations
+* Fixed : Revision Queue column ordering / filtering links didn't work as expected in some use cases
+* Fixed : When previewing a scheduled revision that has been published, "Edit" link in top bar had invalid URL
+* Fixed : If Pro plugin is activated, deactivation of free plugin hides all revisions
+* Compat : Yoast SEO: PHP Notices on revision creation / submission
+* Lang : Update Spanish, Italian translations
+
+= 3.1.2 - 7 Jul 2022 =
+* Fixed : Filter 'use_block_editor_for_post' was not recognized in detecting block editor disable
+* Fixed : Revision Edit in Gutenberg: Enable non-Editors to set requested publish date
+* Fixed : Scheduled revisions were not published on some installations
+* Change : Allow WP-Cron publication method to be disabled for scheduled revisions
+* Feature : Option to Delete Editorial Comments on revision approval
+* Feature : Option to disable revisions per post type
+* Feature : Instance protection library warns about non-standard plugin folder, unnecessary free plugin activation
+* Feature : Plugin review request
+* Feature : Pro promo sidebar on settings screen
+
+= 3.1.1 - 8 Jun 2022 =
+* Fixed : Revision scheduling left a future date selection in editor. If the post is updated again without reloading the editor, it is unpublished until the selected date
+
+= 3.1 - 1 Jun 2022 =
+* Feature : Revision Edit - Administrators and editors have approval button for unsubmitted revisions
+* Feature : Revision Queue - Bulk Edit includes "Decline" action to return "Submitted" revisions back to "Not Submitted"
+* Feature : Revision Queue - New "Copy" row action
+* Compat : PublishPress Permissions - fatal error loading Customizer
+* Compat : PublishPress Authors - Revision submission failed
+* Compat : PublishPress Permissions - Fatal error loading Customizer (also requires Permissions 3.7.6)
+* Fixed : Scheduled Revisions fail on some sites
+* Fixed : Editors could not manage unsubmitted revisions without manage_unsubmitted_revisions capability, even if "Additional role capability required" setting disabled
+* Fixed : Non-public Post types had non-functional revision preview links, revision publication redirects
+* Fixed : Edit Revision - alignment and spacing in revision status sidebar
+* Fixed : Edit Revision - submit / approve buttons were re-enabled after toggle from Block to Post tab
+* Fixed : Revision Queue - checkbox column did not have a "select all" box
+
+= 3.0.16 - 20 Apr 2022 =
+* Fixed : Post Editor - New Revision button was not displayed
+* Lang : Some translations did not load
+* Fixed : Edit Revision with Classic Editor - submit button not displayed for some custom post types
+
+= 3.0.15 - 31 Mar 2022 =
+* Fixed : Scheduled Revisions failed on WP 5.9 if WP Cron scheduling was disabled in Revisions settings
+
+= 3.0.14 - 30 Mar 2022 =
+* Fixed : Output variable escaping, other coding standard improvements
+
+= 3.0.13 - 21 Mar 2022 =
+* Fixed : Revision submission in Classic Editor caused "Undefined" link next to "Preview" link
+* Fixed : New revisions created with "Auto-submit" option had wrong date (and listing order) in Revision Queue
+* Fixed : PHP error on revision approval under some configurations
+* Fixed : Dashboard Activity widget: Scheduled revisions were not included in Publishing Soon list
+* Fixed : API: wp_after_insert_post action was not applied at revision publication
+* Compat : Divi - Revision preview bar not visible on front end
+* Compat : Divi - Classic Editor setting did not trigger correct Revisions UI
+* Compat : Custom post types - revision submission button not displayed in some configurations
+* API: New filters pp_revisions_option_pending_revision_update_post_date, pp_revisions_option_pending_revision_update_modified_date, pp_revisions_option_scheduled_revision_update_post_date, pp_revisions_option_scheduled_revision_update_modified_date
+
+= 3.0.12 - 3 Mar 2022 =
+* Feature : Edit link in post editor after revision creation / scheduling
+* Change : Option to auto-submit revisions created by users who can publish the main post
+* Change : Option to publish scheduled revisions using WP-Cron
+* Fixed : Scheduled publication of revisions fails on WP Engine if caching enabled (fix by enabling WP Cron scheduling)
+* Fixed : WP 5.9 - After editing a revision, approval button remained disabled after revision update
+* Fixed : New revisions were not listed in Revision Queue or linked to Preview button under some conditions
+* Fixed : Compare Revisions: error if ACF is active on a PHP 8 installation
+* Fixed : Revision Queue - column headers (to set sort field) had a bad link
+
+= 3.0.10 - 10 Feb 2022 =
+* Fixed : WordPress 5.9 - When editing a submitted revision, Save button was hidden
+* Fixed : Revision Compare did not load on some sites
+* Fixed : Option "Editing others' revisions requires role capability" was not applied
+
+= 3.0.9 - 27 Jan 2022
+* Compat : WordPress 5.9 - Revisions could not be updated using Gutenberg editor
+
+= 3.0.8 - 26 Jan 2022
+* Compat : WordPress 5.9 - Fatal error adding / editing posts (work around WP hooking late-defined function _disable_block_editor_for_navigation_post_type)
+
+= 3.0.7 - 5 Jan 2022 =
+* Fixed : Revision status changes not updated on sites running an object cache
+* Fixed : Revision deletion from within editor left an inaccessible trashed revision and redirected to Edit Posts. Now deletes revision and redirects to Revision Queue.
+* Fixed : manage_unsubmitted_revisions capability was required without any hint in plugin settings; Now not required unless configured in Revisions > Settings
+* Fixed : Classic Editor - Javascript error breaks plugin compatibility
+* Fixed : Gutenberg Editor - Top margin for Preview button following revision submission
+* Fixed : Compare screen - could not approve revisions
+* Fixed : Compare screen - ampersands in user display name were displayed with html encoding
+* Fixed : Revision Queue - Revisions from deactivated post types were listed with invalid Edit, Delete links and a blank Post Type
+* Fixed : Compare Revisions - Preview / Restore button for past revisions linked back to Compare screen
+* Fixed : Input sanitization consistency
+* Change : Revision preview top bar styling
+* Compat : LifterLMS - LLMS query filters broke Revision Queue
+* Compat : WPML - Database error in Revision Queue
+* Fixed : Revision Queue: PHP error on bulk action under some site configurations
+
+= 3.0.6 - 29 Nov 2021 =
+* Fixed : Edit Revision - revisions could not be updated unless SCRIPT_DEBUG is enabled on the site
+* Fixed : Revisions could not be scheduled if Settings > Revisions > Revision Submission is disabled
+
+= 3.0.5 - 23 Nov 2021 =
+* Fixed : Published custom post types not listed to Revisors under some configurations (Permissions integration requires Permissions 3.6.4)
+* Fixed : Multisite - network-wide settings screens were non-functional
+
+= 3.0.4 - 19 Nov 2021 =
+* Fixed : New Revision button not displayed in Admin Bar
+* Fixed : Revision Queue: revisions not listed in My Activity view
+
+= 3.0.3 - 18 Nov 2021 =
+* Fixed : Revisions Queue empty on sites with a large number of published posts
+* Compat : WooCommerce - Hide Product Variations, Linked Products tabs when editing a Product Revision
+
+= 3.0.2 - 15 Nov 2021 =
+* Compat : PublishPress Permissions - Revisions Submission permissions did not correctly adjust Edit Posts / Pages listing in some configurations (also requires Permissions 3.6.3)
+* Compat : The Events Calendar - Revisions could not be submitted in Classic Editor
+* Fixed : Classic Editor - Schedule button was incorrectly displayed (leading to failed scheduling attempt) if a single time element is changed followed by OK button
+* Fixed : Scheduled Revisions could not be updated in Gutenberg editor
+* Fixed : Admin Bar button "New Revision" was displayed even if Submitted Revisions are disabled
+* Fixed : Revision Queue: bulk deletion did not reduce revision counts
+* Fixed : Duplicate page selection dropdown in Quick Edit and other instances of wp_dropdown_pages()
+* Fixed : Invalid page hierarchy and broken Pages admin if a past revision that was originally submitted through the Revisions plugin is restored
+* Fixed : Revision Preview was missing "Approve" button caption if revision is of an unpublished post
+* Change : Revisions Settings - clarify some captions
+
+= 3.0.1 - 10 Nov 2021 =
+* Fixed : Revisions created using Revisions 2.x were not listed in Revision Queue until plugin de/re-activation
+* Fixed : Front Page setting was cleared at revision submission on some sites
+* Fixed : Front Page setting dropdown in Settings > Reading included revisions
+* Fixed : Classic Editor : Schedule Revisions could not be submitted
+* Fixed : Classic Editor : Schedule button was displayed incorrectly while selecting a future date for scheduled revision
+* Compat : Gutenberg plugin - Edit Revision screen crashed on update attempt
+* API: Allow redirect to be disabled on revision creation
+
+= 3.0 - 9 Nov 2021 =
+* Feature : New revision submission mechanism: create an unsubmitted revision first instead of editing existing post. Edit revision directly, then submit for scheduling or publication.
+* Change : Revisions UI in Post / Revision editor
+* Feature : Admin Bar includes "New Revision" button
+* Change : Revision Queue filter captions
+* Compat : Permissions - revise_others_posts, revise_others_pages, etc. capabilties are equivalent to list_others capabilities in allowing uneditable items to be listed
+* Change : Include upload_files capability in the Revisor role
+
+= 2.6.3 - 11 Oct 2021 =
+* Fixed : Classic Editor - Error submitting a pending or scheduled revision
+
+= 2.6.2 - 7 Oct 2021 =
+* Change : Maintenance queries run at plugin activation to convert version 3.0 revisions back to 2.6 encoding
+* API : Support for WPML Translation Management support in Pro version
+
+= 2.6.1 - 15 Jul 2021 =
+* Fixed : Pending / Scheduled Revisions could become disassociated from main post due to third party plugin interactions
+* Compat : Permissions - Specific Permissions were not applied for editing / approval of custom post types under some conditions
+* Fixed : Revision Update triggered redirect back to Revision Queue even if "Confirmation redirect on Revision Update" setting disabled
+* Fixed : Users without full editing capabilities could not submit a pending revision with a future date selection
+* Fixed : Revision Previews were cached on some browsers
+* Fixed : Preview of front page revisions did not trigger front page template display
+* Fixed : Bulk Approval of revisions failed / caused PHP Notice on some sites due to post type not being registered early enough
+* Fixed : Pending Revision checkbox default selection from 'revisionary_default_pending_revision' filter application did not trigger revision submission
+* Lang : Revision Queue - Remove sample English translation "My Revisionz"
+
+= 2.6 - 23 Jun 2021 =
+* Fixed : Background fatal error on some sites on revision scheduling
+* Fixed : Scheduled Page Revisions - If published page has a non-default template which was not changed in the revision, the Compare and Edit Revision screens indicated a change to default template. This change was not actually applied unless the stored revision was updated prior to publication.
+* Feature : Option to send notifications on Revision Update
+* Feature : Option to redirect to confirmation screen after Revision Update
+* Compat : Custom Permalinks plugin
+* API : New filters allow submission confirmation message to be customized: revisionary_submit_message_links, revisionary_submit_message, revisionary_schedule_message_links, revisionary_schedule_message
+
+= 2.5.5 - 26 May 2021 =
+* Compat : Google Web Stories - Loss of story data due to clearance of post_content_filtered column
+* Fixed : Fatal error in post editor on sites that apply filter "rest_{$post_type}_collection_params" incorrectly (Uncaught ArgumentCountError: Too few arguments to function)
+* Fixed : PHP Warning on rvy_is_full_editor() call under some configurations
+* Change : Revision Queue - Standard link coloring (like Edit Posts)
+
+= 2.5.4 - 6 May 2021 =
+* Fixed : Database error on Revision approval - Unknown column 'filter' in 'field list' for query UPDATE `wp_posts` SET `post_author` =
+* Compat : TablePress - Tables could not be updated by non-Administrators
+
+= 2.5.3 - 6 Apr 2021 =
+* Compat : WP Rest Cache - Revision submission from Gutenberg failed
+* Compat : PublishPress Permissions - Edit Category / Term: Permissions metaboxes were not displayed
+* Compat : PublishPress Permissions - Pending Revision Monitors group ineffective; notifications were sent to all Editors and Administrators (also requires PublishPress Permissions 3.5.1)
+* Compat : Enable revisioning of non-public post types if they have type-specific capabilities defined
+
+= 2.5.2 - 30 Mar 2021 =
+* Fixed : Fatal error due to incorrect vendor library load request
+
+= 2.5.1 - 30 Mar 2021 =
+* Fixed : Revisors could not preview other users' pending revisions, even if they are listed in Revision Queue (Permissions integration requires 3.5.1)
+* Fixed : Post Title changes were not applied at revision publication
+* Fixed : Database error on revision publication, on some installations
+* Fixed : Compare Revisions - uneven column widths with WordPress 5.7
+* Feature : Option to update modified date on revision publication
+* Compat : PublishPress - Custom Status dropdown was hidden in post editor
+* Compat : Kinsta object cache - Revision submission using "Save as Revision" checkbox failed
+* Compat : Kinsta object cache - Revision submission clears Featured Image from published post
+
+= 2.5 - 4 Mar 2021 =
+* Fixed : After a Pending Revision is published, Compare Revisions screen for past revisions did not order it correctly
+* Fixed : Revisors could not access Compare Revisions screen unless PublishPress Permissions was active
+* Fixed : Some attachment fields implemented by third party plugins were not saved to Pending Revisions
+* Fixed : Revision submission - "submit another revision" link (following future date selection) led to submission failure on some sites
+* Compat : Advanced Custom Fields - attachment image fields were cleared out of new revision
+* Compat : REST API Cache - "Save as Revision" checkbox ineffective on some installations
+* Compat : PublishPress - Editing a revision caused it to be converted to a non-revision draft
+* Compat : PublishPress - Custom Status module interfered with Revisions scripts on Edit Revision screen
+* Compat : PublishPress - Adding an Editorial Comment on the Edit Revision screen, then updating, caused revision to become inaccessible
+* Compat : Some third party plugin interactions could cause Revision publication to fail, leaving published page unreadable
+* Change : Display a Revisions link (for past revisions) in Classic Editor for Revisor
+* Lang : Add Swedish translation
+
+= 2.4.9 - 14 Jan 2021 =
+* Compat : REST API Cache - REST caching failed for attachments
+* Fixed : Non-public post types displayed ineffective preview link. Now suppress link or link to Compare Revisions screen instead.
+* Fixed : Error loading Revisions > Settings page if nullstring license key is stored, on some installations
+* Fixed : Deprecated jQuery event handlers
+* Feature : Display notice to Revisor if they have already submitted a revision (in Gutenberg only if constant REVISIONARY_GUTEN_NOTICE defined)
+* Change : Enable Compare Revisions link (for past revisions) in page editor for Revisor
+
+= 2.4.8 - 17 Dec 2020 =
+* Compat : WPML - Revision Queue was not filtered by WPML language selector
+* Compat : WPML - Hide non-revisionable Languages settings in Revision editor
+* Compat : TablePress - Tables could not be updated by non-Administrators
+* Fixed : Revision Queue - PHP Notice if no revisions listed
+
+= 2.4.7 - 15 Dec 2020 =
+* Fixed : WP 5.6 - Pending Revision submission failed (since 2.4.6)
+* Fixed : Compare Pending Revisions always credited assigned post author for Current Revision, now shows user who created the last published update. To revert to previous behavior, define constant RVY_LEGACY_COMPARE_REVISIONS_AUTHOR_DISPLAY.
+* Change : Clarify captions for permissions-related checkboxes in Revisions > Settings > Revision Queue
+* Compat : PublishPress Capabilities - organize Revisions capabilities into own area
+
+= 2.4.6 - 8 Dec 2020 =
+* Lang: Include a .pot file (translation template)
+* Fixed : If no revisions are accessible to user, all revisions were listed in Revision Queue
+* Fixed : Post deletion triggered revision submission redirect under some conditions
+* Compat : REST API Cache plugin - "Save as Revision" checkbox was ineffective
+* Compat : PublishPress Authors - revision submission failed under some conditions
+
+= 2.4.5 - 30 Nov 2020 =
+* Compat : WP Engine, Kinsta object cache - "Pending Revision" checkbox was ineffective
+* Compat : PublishPress Permissions - Revision editing or deletion was improperly blocked under some configurations
+* Fixed : Revision submission by Administrator or Editor caused corruption of published block content (by slash removal) on some installations
+* Fixed : Edit Revision screen did not display Approve Revision button to users who cannot delete the revision
+* Fixed : Revision Queue - other plugin / theme filters could cause all revisions to be listed when a requested published post has no accessible revisions
+* Fixed : Revision Queue - filter links worked, but had original URL reconstructed incorrectly
+* Fixed : Revision Queue - pagination links had badly formatted arguments if a filtering argument is active
+* Fixed : Revisors had non-applicable elements hidden in post editor, but orphaned labels for those elements remained visible
+* API : New filter 'revisionary_notify_publishers_eligible' to modify eligible "Publishers to Notify" when Email Notification is optional
+* API : New filter 'revisionary_notify_publisher_default_ids' to modify default "Publishers to Notify" when Email Notification is optional
+
+= 2.4.4 - 18 Nov 2020 =
+* Fixed : Revision submission caused Post Thumbnail to be cleared from the published post
+* Fixed : Revision submission by a Revisor caused corruption of published block content (by slash removal) on some installations
+* Fixed : Compare Past Revisions - Editors did have "Preview / Restore" or "Manage" buttons
+
+= 2.4.3 - 5 Nov 2020 =
+* Compat : Polylang - language settings were not stored to revision (Fix also applies to other plugins using hidden taxonomies)
+* Compat : Project Nami (Microsoft SQL Server / ODBC) - No confirmation redirect on revision submission
+* Lang : Added .pot file
+
+= 2.4.2 - 26 Oct 2020 =
+* Compat : PublishPress Permissions - Users assigned Revise permissions for specific pages or categories could not compare Pending Revisions
+* Compat : PublishPress Permissions Pro - With Status Control module active, Edit Revision screen had invalid "Workflow" button (also requires Permissions Pro 3.3.8)
+* Compat : Public Post Preview - Author selector was hidden in post editor
+* Compat : Public Post Preview - Make public post previews of published posts redirect to the published post
+* Change : Revision Settings - Revision Queue section, includes "Compatibility Mode" setting to prevent revisions from being hidden from the queue in the case of plugin integration issues
+* Fixed : "Prevent Revisors from editing others' revisions" setting was not applied
+* Fixed : Error when Revision submission includes a template setting
+* Fixed : Custom plugins path caused PHP error
+* Fixed : Revison Queue - PHP Notice "Undefined variable: post_id" when URL includes published_post argument
+* Feature : If Revisors are blocked from editing other users' drafts, those can now be included (unclickable) in Edit Pages if the list_others_pages capability is granted
+* Feature : Support constant definitions REVISIONARY_DISABLE_SUBMISSION_REDIRECT, REVISIONARY_DISABLE_SCHEDULE_REDIRECT
+* Feature : New filters "revisionary_do_submission_redirect", "revisionary_do_schedule_redirect"
+
+= 2.4.1 - 9 Oct 2020 =
+* Compat : PublishPress Permissions - On new post creation, Revisors get a Publish button instead of a Submit button. Publishing fails; the only way to submit successfully is Save Draft, then Submit for Review.
+* Compat : PublishPress Permissions - Revisions were not listed in Queue under some configurations with PublishPress Permissions active
+* Fixed : With pre-Publish checks enabled in Gutenberg, after first save "Pending Revision" checkbox was moved off of pre-Publish Panel
+
+= 2.4 - 1 Oct 2020 =
+* Fixed : Revisors and other limited editors had editor elements hidden when adding a new post, under some site configurations
+* Fixed : Revision publication always set post publish date to current time. Now does only with enabled setting Revisions > Settings > Pending Revisions > Update Publish Date
+* Fixed : Classic Editor - after Preview Changes is clicked, limited editors see "Submit Revision" button recaptioned to "Update"
+* Compat : PublishPress Authors - Authors could not be changed on Edit Revision screen with Gutenberg editor if PublishPress is also active
+* Compat : Gutenberg Ramp - Revision submission UI did not load for post types that have Gutenberg enabled
+
+= 2.3.12 - 27 Aug 2020 =
+* Fixed : WP 5.5 - Post previews did not display correctly for Revisors
+* Fixed : Pending, Scheduled revisions not listed in Revision Queue following mirroring of posts database table from another installation (or possibly under other conditions)
+* Fixed : Editorial Comments added on "Edit Revision" screen did not trigger email notification to post author or revision author
+* Feature : Option to copy revision's editorial comments over to published post (at revision publication)
+
+= 2.3.11 - 13 Aug 2020 =
+* Compat : WP 5.5 - "Pending Revision" checkbox sometimes ineffective
+* Compat : WP 5.5 - Posts with pending or scheduled revisions stored had misplaced links in Gutenberg editor sidebar
+* Compat : WP 5.5 - With Classic Editor, javascript errors in post editor
+* Compat : WP 5.5 - PHP warning on post edit (deprecated function escape_attribute)
+* Compat : WP 5.5 - Edit Revision screen - Duplicate Preview link, misaligned
+* Compat : WP 5.5 - Edit Revision screen - View / Approve link misaligned
+* Fixed : "Has Revision" post state displayed for posts that have comments but no revisions (since 2.3.10)
+* Fixed : Scheduled revisions were not published under some conditions (since 2.3.9)
+* Fixed : In some conditions, fatal error on Plugins screen
+
+= 2.3.10 - 10 Aug 2020 =
+* Fixed : Revisions submitted without modifying tags had tags removed
+* Feature : Edit Posts screen - display "Has Revision" as a post state after post title
+
+= 2.3.9 - 6 Aug 2020 =
+* Fixed : Featured Image was removed from pending revision at creation
+* Fixed : Scheduled revision publication failed under some conditions, caused post to be unpublished
+* Fixed : Scheduled revisions could not be published ahead of schedule using "Publish Now" link on preview (since 2.3.4)
+* Lang : Add German translation
+* API : New filter 'revisionary_apply_revision_data' to adjust standard revision fields prior to publication
+
+= 2.3.8 - 30 Jul 2020 =
+* Feature : Revision Queue - new bulk action to Unschedule selected revisions
+* Lang : Add Spanish translation
+* Fixed : Revisors could not preview changes prior to submitting a pending revision
+* Fixed : Classic Editor plugin - when "Edit (Classic)" link is used, Revisors did not have Update button recaptioned to "Submit Revision"
+* Fixed : API - revisionary_enabled_post_types filter was not fully effective
+* Compat : Public Post Preview - support preview link generation on Edit Revision screen
+
+= 2.3.6 - 10 Jun 2020 =
+* Fixed : After revision submission, preview link was not always to latest revision
+* Fixed : Preview button on editor screen loaded preview with invalid thumbnail under some conditions
+* Fixed : When network-activated, Network Settings menu item loaded site-specific settings screen
+
+= 2.3.5 - 29 May 2020 =
+* Fixed : Compare link on Editor screen linked to Edit Posts screen instead of Compare Revisions
+
+= 2.3.4 - 29 May 2020 =
+* Fixed : Duplicate email notifications to users who have more than one WordPress role
+* Change : Suppress email notification when an Administrator or Editor creates a pending revision, if constant REVISIONARY_LIMIT_ADMIN_NOTIFICATIONS is defined
+* Compat : Relevanssi - Scheduled revisions were not published with Relevanssi active
+* Fixed : Scheduled revision publication caused other scheduled revisions (for the same post) to be hidden from Revision Queue
+* Fixed : has_revisions postmeta flag was not cleared when a post's last revision was deleted
+* Fixed : Revision Queue - Search function did not work
+* Fixed : Revision Queue link in Edit Posts / Edit Pages row was not removed after post's last pending or scheduled revision published or deleted
+
+= 2.3.3 - 14 May 2020 =
+* Compat : PublishPress Permissions - Fatal error on post creation
+
+= 2.3.2 - 12 May 2020 =
+* Fixed : Post meta flag "_rvy_has_revisions" was not cleared after last remaining pending / scheduled revision was published or deleted, affecting Revision Queue performance
+* Fixed : Revision Queue listed uneditable revisions under some conditions
+* Fixed : My Published Posts count was wrong under some conditions
+* Fixed : Dashboard At a Glance link for Pending Post Revisions linked to Revision Queue without filtering display to Posts only
+* Compat : PublishPress Permissions - suppress Permissions metaboxes on Edit Revision screen
+* Fixed : Published post content cleared on pending revisions submission, on a minority of installations
+
+= 2.2.4 - 6 Apr 2020 =
+* Fixed : Possible fatal error loading Revisions screen on a small percentage of installations
+
+= 2.2.3 - 3 Apr 2020 =
+* Fixed : Classic Editor - Category and Post Tag revisions were not applied
+
+= 2.2.2 - 2 Apr 2020 =
+* Feature : Option to disable revision preview links for non-Administrators (to work around themes that force a 404 Not Found response)
+* Fixed : Inline styles were stripped or modifield on scheduled revision publication
+* Fixed : Possible fatal error loading Revisions screen on a small percentage of installations
+* Fixed : PHP Notice for deprecated function contextual_help_list()
+* Change : Standardize sanitization of database queries
+* API: revisionary_enabled_post_types filter was not applied consistently
+* Compat : CMS Tree Page View - Suppress Pending Revisions and Scheduled Revisions from Page Tree View
+
+= 2.2.1 - 16 Mar 2020 =
+* Fixed : Page Template was cleared on revision submission in some installations
+* Fixed : Revision Queue - "Filter" link was ineffective in showing only revisions of the selected published post. This also applies to "View Revision Queue" link after revision creation.
+* Fixed : Edit Revision - Move to Trash button did not work (and created new pending revision)
+* Fixed : Duplicate email notifications for scheduled revision publication on some installations
+* Fixed : Safeguard to prevent duplicate email notifications
+* Feature : Plugin API - new filter 'revisionary_mail' allows adjustment to notification email address, title or message (or blockage of a particular email)
+* Change : Pro top banner on Revisions screens
+* Compat : New setting "Revision publication triggers API actions to mimic post update" causes save_post and transition_post_status actions to fire on revision publication
+* Compat : Yoast SEO - Revision submission stripped accented characters and emojis out of FAQ block
+* Compat : On revision publication, trigger 'transition_post_status' action, for plugins that use it
+
+= 2.2 - 12 Feb 2020 =
+* Feature : Email Notification - option to notify Editors and Administrators when a Pending Revision is approved
+* Fixed : Block Editor - Custom Taxonomies, if unchanged, were not saved to revision. Publication of revision cleared custom taxonomies for published post.
+* Fixed : Block Editor - Error setting Featured Image
+* Fixed : Revisions submitted by Administrators or Editors using "Pending Revision" checkbox caused published post title and content to be cleared if a future publish date was also selected
+* Compat : PublishPress Permissions Status Control - "Prevent Revisors from editing other users' drafts" setting also prevented other non-Editors from editing posts of a custom workflow status that uses custom capabilities (also requires PP Permissions Pro 2.9.1)
+* Compat : Block data from some plugins had html formatting tags displayed as unicode character codes
+* Fixed : Edit Revision screen - Date selector was displayed even if scheduled revisions feature disabled
+* Fixed : Compare Pending Revisions - Non-administrators could not edit Scheduled Revisions
+* Fixed : Compare Pending Revisions - for page slug change, original published slug was not displayed
+* Fixed : 'revisionary_skip_taxonomies' filter triggered a database error
+* Fixed : PHP Notice if third party code registers a post type without defining the edit_published capability
+* Fixed : PHP Notices on revision submission notification
+* Change : By default, enable "Prevent Revisors from viewing others'" setting
+* Change : Apply possible workaround for Revision Queue capability issues on some sites
+
+= 2.1.8 - 15 Jan 2020 =
+* Fixed : Custom Post Types did not have Pending Revisions or Scheduled Revisions available (since 2.1.7)
+* Lang : Correct textdomain on numerous translation calls
+* Lang : Improve translation string construction
+* Lang : Support translation of Revisor role name
+* Lang : Updated language files
+
+= 2.1.7 - 13 Jan 2020 =
+* Fixed : Excessive resource usage with some caching solutions
+* Fixed : Multisite - Super Administrators without a site role could not access Revision Queue
+* Fixed : Classic Editor - After updating a revision, "View Post" message linked to published post instead of revision preview
+* Feature : New filter 'revisionary_enabled_post_types', unset post types by key to disable PP Revisions involvement
+
+= 2.1.6 - 23 Dec 2019 =
+* Fixed : Edit Revision - Classic Editor "Approve" button ineffective
+* Fixed : Edit Revision - Classic Editor "View / Approve" button loaded live preview (of unsaved changes) instead
+* Compat : By default, prevent third party post query filtering on Revision Queue (to avoid non-display of Revisions)
+* Compat : PressPermit Pro - Updating a saved revision caused it to be changed to a regular pending post
+
+= 2.1.5 - 11 Dec 2019 =
+* Compat : PressPermit Pro - Pending revision previews could be viewed by any user (including anonymous) if "Prevent Revisors from viewing others' revisions" disabled (since 2.1.4)
+* Fixed : Contributors had other users' uneditable, unreadable revisions listed in Revision Queue
+* Fixed : Revision Preview - Under some configurations, users with read-only access to revisions had no top bar in revision preview display
+* Fixed : Revision Preview - Under some role configurations, users saw an ineffective "Publish" button in preview top bar
+* Fixed : PHP warning for undefined index 'preview'
+
+= 2.1.4 - 10 Dec 2019 =
+* Fixed : Revision previews were not displayed to Editors under some configurations
+* Feature : Separate settings for "Prevent Revisors from editing others'" and "Prevent Revisors from viewing others'"
+
+= 2.1.3 - 6 Dec 2019 =
+* Compat : Classic Editor plugin - View / Approve buttons missing on Edit Revision screen if Classic Editor active but settings default to Block Editor
+* Compat : Classic Editor plugin - Javascript errors on Edit Post / Edit Revision screen if Classic Editor active but currently using Block Editor
+* Compat : Thin Out Revisions plugin broke Preview / Approval buttons on Compare Pending Revisions screen
+* Compat : Multiple Authors - Revision Queue "Post Author" links did not work for secondary authors
+* Compat : Multiple Authors - Revision Queue "Post Author" links did not filter Revision Queue
+* Compat : JReviews - Live preview from Edit Revision screen failed if JReviews plugin active
+* Fixed : Preview Top Bar blocks admin bar dropdown menu if another fixed-position element on the page (other than #wpadminbar) has a z-index of 99999 or higher
+
+= 2.1.2 - 4 Dec 2019 =
+* Fixed : Scheduled Revisions were not published (since 2.1)
+* Fixed : Edit Revision - Preview of unsaved revision did not work from Gutenberg
+* Change : Edit Revision - Display "View / Approve" button if editor is unchanged from saved revision, otherwise "Preview" button for unsaved changes
+* Fixed : Classic Editor - Preview caused "Update Revision" button to be recaptioned to "Save Draft"
+* Feature : Support Post Slug revision
+* Fixed : Other users' revisions were not listed in Revision Queue even if "Prevent Revisors from editing others' revisions" disabled
+* Fixed : With "Prevent Revisors from editing others' revisions" setting enabled, Revisors and Authors could edit others' revisions by direct URL access
+* Feature : Support list_others_revisions capability to grant read access to other users' revisions (applies if "Prevent Revisors from editing others' revisions" is enabled)
+* Compat : PressPermit Pro - Revisors could not submit Beaver Builder revisions
+* Compat : PressPermit Pro - Revision Exceptions ("Also these" category / taxonomy assignments) assigned to Authors were not applied correctly
+* Compat : JReviews plugin
+
+= 2.1.1 - 26 Nov 2019 =
+* Compat : Multiple Authors - Fatal error on revision creation (since 2.1)
+
+= 2.1 - 26 Nov 2019 =
+* Feature : Bulk Approval / Publishing in Revision Queue
+* Feature : Revision Edit: Approve Button on Editor screen
+* Feature : Option for Approve, Edit buttons on Compare Revisions screen (instead of Preview button)
+* Feature : Email Notification Buffer to avoid failures due to exceeding server send limits
+* Fixed : Email Notification - For pending revision submission, submitter was misidentified on some sites
+* Fixed : Revisors could restore previous revisions through manual URL access
+* Fixed : Fatal error when WP_Privacy_Policy_Content::text_change_check() is triggered
+* Fixed : "Pending Revision" checkbox was displayed in Gutenberg editor, even for unpublished posts
+* Fixed : After clicking "Pending Revision" checkbox, unchecking did not prevent revision save
+* Fixed : Revision Preview - unsaved changes to saved revision could not be previewed with WP 5.3
+* Fixed : Revision Preview - top bar for edit / approval was not displayed on some sites
+* Change : Revision Preview URL - Default to using published post slug with revision page_id argument, for better theme compatibility. Option to use Revision slug or ID only.
+* Fixed : Edit Revision screen links to published post discarded customized slug
+* Fixed : Classic Editor - "View / Approve" link from Edit Revision screen loaded wrong preview URL and no top bar display for approval
+* Fixed : Classic Editor - No preview button was available to Revisors
+* Fixed : Classic Editor - Invalid Revisions > Browse link displayed to Revisors
+* Compat : Classic Editor plugin - with "Allow users to switch editors" enabled, non-default editor did not have correct javascript loaded for Revisions
+* Compat : On themes that use a fixed position header, display preview top bar above header
+* Compat : PressPermit Pro - revision preview could not be viewed by Contributors under some configurations
+* Fixed : On standard Compare Revisions screen (for past revisions), Preview and Manage button links did not update with slider selection change
+* Fixed : Pending, Schedule Revision notification - invalid preview link in some emails
+* Fixed : Trashed revisions were not identified as revisions in Edit Posts listing
+* Fixed : Trashed revisions were not deleted on parent post deletion
+* Fixed : Trashed revisions showed an invalid comment count value in Edit Posts listing
+* Fixed : PHP Warning in Gutenberg editor when editing is not being limited to revision submission
+* Compat : Multiple Authors - Compare Pending Revisions screen showed revisor as original post author under some conditions
+* Compat : Multiple Authors - Revision submission / approval caused published post author to be changed to revisor, under some conditions
+* Compat : Plugin interaction caused published post permalink custom slug to be replaced with default permalink structure at revision publication, on some sites
+* Change : Revision Queue - recaption "My Posts" to "My Published Posts"
+
+= 2.0.12 - 29 Oct 2019 =
+* Fixed : Fatal error on Post Preview
+
+= 2.0.11 - 28 Oct 2019 =
+* Fixed : Classic Editor - Post Preview showed last stored copy, not unsaved changes
+* Fixed : Revision Preview top bar covered admin menu dropdown
+* Fixed : Revision Edit - live preview showed revision author instead of published author (if Multiple Authors plugin not active)
+
+= 2.0.10 - 25 Oct 2019 =
+* Fixed : Post Preview showed last stored copy, not unsaved changes
+* Fixed : Post Preview (to view unsaved changes) was not available when editing a revision
+* Fixed : Revision Preview - Buttons were not clickable with some themes
+* Fixed : Filter revisionary_default_pending_revision was not effective in Gutenberg (check Save as Revision checkbox by default)
+* Compat : Multiple Authors - Incorrect author display in revision previews on some sites
+* Compat : PressPermit - Database error on Revision Queue screen under some configurations
+
+= 2.0.9 - 18 Oct 2019 =
+* Fixed : Compare Pending Revisions screen - link redirected to Edit Posts screen for some post types
+
+= 2.0.8 - 18 Oct 2019 =
+* Change : PostMeta Failsafe: to avoid the possibility of accidental clearance, Featured Image removal is not revisioned, until further testing. API filter available for experimental usage with specified meta keys.
+* Fixed : Featured Image, Page Template revisioning failed under some conditions
+* Fixed : Scheduled Revisions created with Gutenberg stored selected terms to published post, previous terms to revision
+* Fixed : Scheduled Revisions - If "Update Publish Date" enabled, 404 Not Found redirect after manually publishing a scheduled revision if the post type uses post date in permalink structure
+* Fixed : Revision Preview - Buttons were not clickable with some themes
+* Fixed : Settings - Disabling Pending or Scheduled Revisions did not remove UI from post editor
+* Fixed : Settings - If Pending Revisions disabled, Revisor could still edit published posts
+
+= 2.0.7 - 17 Oct 2019 =
+* Fixed : Scheduled Revisions - published post tags and categories were stripped out on scheduled revision publication
+* Fixed : Scheduled Revisions - manually publishing prior to scheduled time caused published post status to be set to Future (unpublished)
+
+= 2.0.6 - 17 Oct 2019 =
+* Fixed : Featured Image and Page Template revisions were not applied (but did work in PublishPress Revisions Pro)
+* Fixed : Publishing a revision imported from Revisionary 1.x caused tags and categories to be stripped out
+
+= 2.0.5 - 16 Oct 2019 =
+* Fixed : Import script for Revisionary 1.x revisions did not run on plugin activation
+* Fixed : Administrators, Editors and Authors were blocked from Quick Edit
+* Compat : Multiple Authors plugin
+* Fixed : Pending Revisions - Published post date was not updated even if "Update Publish Date" setting enabled
+* Change : Pending Revision Notification - Include link to Revision Queue
+* Fixed : Pending Revision Notification - If enabled for author only, email was sent with a blank title and message
+* Fixed : Empty Revision Queue was displayed to Subscribers with no Revision capabilities
+* Fixed : PHP notices on Revision Queue screen
+
+= 2.0.4 - 9 Oct 2019 =
+* Change : On installation over Revisionary 1.x, display a "heads up" notice about plugin name change, admin menu and Revision Queue
+* Fixed : Classic Editor - Revision Preview did not always include top bar (for Edit / Compare / Publish) if PressPermit Pro active
+* Fixed : Revision Preview - Edit url did not work on installations with non-conventional admin paths, due to hardcoded /wp-admin
+* Fixed : Schedule Revision notifications sent redundantly under some conditions
+* Change : On Revision Edit, recaption Preview button to "View" to clarify that it's a preview of the saved revision, not unsaved changes. (Future release will make it a true preview).
+
+= 2.0.3 - 3 Oct 2019 =
+* Fixed : Revisionary settings could not be changed
+* Fixed : Pending / Scheduled Revisions were listed in Revision Queue even if feature disabled in Revisions > Settings
+* Fixed : On post edit for revision, Revisors could not see the current or newly selected Featured Image
+* Fixed : On revision edit, Administrators and Editors did not have Trash button available
+* Fixed : Revisors could edit or delete their scheduled revisions
+* Fixed : Scheduled revision publication did not work with "Asynchronous publishing" setting enabled
+* Fixed : After revision publication reloading, the old revision preview returned "Not Found". Now redirects to published post and marks as "Current Revision"
+* Fixed : PHP Notices throughout wp-admin when WP_DEBUG enabled
+* Change : Revision Queue headline indicates when results are being filtered by post type, revision status, revision author or post author
+
+= 2.0.2 - 2 Oct 2019 =
+* Fixed : On post date change in Gutenberg editor, Publish button was recaptioned to "Schedule Revision" even on a past date selection (unless SCRIPT_DEBUG enabled)
+
+= 2.0.1 - 2 Oct 2019 =
+* Fixed : Fatal error if another copy of Revisionary already active
+
+= 2.0.0 - 1 Oct 2019 =
+* Feature : Submit revisions to Categories, Tags, Custom Terms, Page Parent, Featured Image, Page Template
+* Feature : Revisions editable in Gutenberg, Classic Editor
+* Feature : Voluntary pending revision submission by unrestricted editors in Gutenberg
+* Feature : Revision Queue screen is a sortable, filterable list of pending and scheduled revisions for all post types
+* Feature : Revision Queue screen includes "My Revisions" and "My Posts" filtering links
+* Feature : Revision Queue - Published Posts have "History" link to compare past revisions
+* Feature : Compare Revisions - for past revisions, add button links for "Preview / Restore" and "Manage"
+* Feature : Compare Pending Revisions using standard WordPress UI (link from Editor or Revision Queue)
+* Feature : Compare Scheduled Revisions using standard WordPress UI (link from Editor or Revision Queue)
+* Feature : Compare Pending / Scheduled Revisions shows changes to Categories, Tags, Terms, Page Parent, Featured Image, Page Template
+* Change : Improved styling for revision preview / approval top bar
+* Feature : "Update Publish Date" setting for Pending Revisions (defaults to disabled)
+
+= 1.3.8 - 30 Aug 2019 =
+* Fixed : Revisors could Quick Edit published posts (changing post title, slug, author, date, parent or template) since version 1.3. This could be used to unpublish (but not publish) posts. Sites also running PressPermit Pro were not affected.
+* Compat : PressPermit Pro - Under some configurations, Revisors were not allowed appropriate access (due to publish capability check)
+
+= 1.3.7 - 24 May 2019 =
+* Feature : Filter 'revisionary_default_pending_revision', return true to select "Send to Approval Queue" in Classic Editor by default
+
+= 1.3.6 - 30 Apr 2019 =
+* Fixed : Scheduled Revision publication updated post date even if "Update Publish Date" option disabled
+* Fixed : Gutenberg: Pending, Scheduled Revisions did not work for post types with show_in_rest property set false
+* Fixed : PHP Notice if REST Posts query executed without a corresponding rest_base property set for post type
+* Fixed : Better hiding of non-applicable sidebar metaboxes when post is being edited for Pending Revision
+
+= 1.3.5 - 3 Apr 2019 =
+* Fixed : With Classic Editor, Revision submission reset Page Template
+
+= 1.3.4 - 2 Apr 2019 =
+* Fixed : Pending Revision Notifications were not sent from Gutenberg editor if configured to send "by default" (selectable recipients)
+
+= 1.3.3 - 2 Apr 2019 =
+* Fixed : Scheduled Revision preview: "Publish Now" link failed with a fatal error
+* Change : Settings link in Plugins Row
+
+= 1.3.2 - 29 Mar 2019 =
+* Fixed : Email notifications were missing "Post" / "Page" caption
+* Fixed : PHP notices with Classic Editor
+* Fixed : With Classic Editor, revision approval from preview did not redirect back to Edit Posts / Pages screen
+* Fixed : In Classic Editor, setting a future date did not recaption Publish button to "Schedule Revision" if post has private visibility
+
+= 1.3.1 - 29 Mar 2019 =
+* Fixed : Scheduled Revision publication stripped out categories and tags, if "Update Publish Date" setting enabled
+* Fixed : Publish button was not recaptioned to Submit Revision under some conditions
+* Change : With Gutenberg active, revision approval defaults to front end preview
+* Feature : Better redirect logic following revision approval, scheduling or restoration (returns to screen that preview was linked from)
+* Feature : Preview link in Notification Emails
+* Feature : Previews of Scheduled Revisions and Pending Revisions with a future publish date include link to Revisions Manager to edit date
+* Change : Dismissable welcome message: To allow a user to submit Revisions to your published posts, set their role to "Revisor"
+
+= 1.3.0 - 28 Mar 2019 =
+* Feature : Gutenberg editor compatibility for Pending Revision, Scheduled Revision creation
+* Feature : By default, Scheduled Revisions also update publish date. New checkbox on Revisions > Settings to restore previous behavior of leaving publish date unchanged.
+* Feature : List Scheduled Revisions of any post type on Publishing Soon list in Activity dashboard widget
+* Fixed : If Scheduled Revision was first site access after scheduled publication time, changes were not displayed until page reload
+* Fixed : Scheduled post Revisions on Publishing Soon list in Activity dashboard widget had incorrect link
+* Fixed : Past Revisions list on Revision Manager screen had invalid preview links
+* Fixed : Better formatting for Publish Now / Schedule Now link
+* Fixed : Editing revision publication date updated revision author, even if post content not changed
+* Change : Use 12 hour format for revision dates
+* Change : Pending Revision lists show submission date
+* Change : Pending Revision lists show requested publication date if applicable
+
+= 1.2.7 - 13 Mar 2019 =
+* Fixed : Pending Revision Notification on Multisite installations. Due to failure to apply settings, e-mail notifications defaulted to "By default" option, which failed for Pending Revisions prior to version 1.2.6.
+* Fixed : Multisite - If network-activated, Revisionary settings screens unavailable. Last stored network-wide settings (or hardcoded defaults) applied instead.
+* Fixed : Multisite - If not network-activated, Revisionary settings screen was ineffective. Site-specific settings were stored, but network-wide settings or defaults applied instead.
+* Fixed : "Display Hints" setting had no checkbox on Settings screen
+* Change : Improved settings captions
+
+= 1.2.6 - 13 Mar 2019 =
+* Fixed : "Publishers to Notify" checkboxes were not displayed, and notifications not sent, if Email notification for Pending Revisions set to "By default"
+* Fixed : Revision previews - PHP Warning and failure to output "Publish Now" header
+* Change : Improved styling in "Publishers to Notify" metabox
+
+= 1.2.5 - 25 Feb 2019 =
+* Compat : TinyMCE Advanced - Failed to display editor on revision management screen
+* Compat : Multisite - Incorrect site switching, prevents Yoast SEO from saving post meta
+
+= 1.2.4 - 20 Feb 2019 =
+* Compat : PublishPress - publish button was hidden
+* Change : Capitalize "Save as Pending Revision" checkbox caption
+* Lang : Update .po file
+
+= 1.2.3 - 19 Feb 2019 =
+* FIXED : Scheduled revision publication failure, massive redundant email notifications (since 1.2)
+
+= 1.2.2 - 19 Feb 2019 =
+* Fixed : Temporarily disable scheduled revision publication emails, due to recently reported issue
+* Compat : PHP / coding standards - removed needless byref variable assignments
+* Fixed : PHP notices when viewing revision differences
+
+= 1.2.1 - 14 Feb 2019 =
+* Compat : Fatal error when another plugin hooks into 'user_has_cap' filter
+
+= 1.2 - 13 Feb 2019 =
+* Compat : PHP 7.2
+* Compat : WordPress 5.0.3
+* Fixed : Revision approval reset page template setting to default
+* Team : Revisionary is now owned and developed by PublishPress. The original author (Kevin Behrens) is excited to join forces in building and supporting effective tools for publishing teams.
+
+= 1.1.13 - 13 May 2015 =
+* Fixed : Previewing a Page revision from Revisions Manager screen caused fatal error / white screen
+* Fixed : When Previewing a revision, Publish Now link was not formatted properly on TwentyFifteen theme
+* Fixed : Pending Revision counts, links were not displayed in Dashboard At a Glance if PP Collaborative Editing plugin is not active
+* Compat : Jetpack Markdown - publishing a revision caused post content to be stripped
+* Compat : various caching plugins - post cache was not cleared after publishing a revision
+
+= 1.1.12 - 23 Dec 2013 =
+* WP 3.8 - Fixed Revisionary > Settings styling
+* Fixed : Email notifications were not sent on Pending Revision submission under some configurations
+* Fixed : Email notifications were not sent upon Scheduled Revision publishing unless Press Permit / Role Scoper active and Scheduled Revision Monitors group populated
+* Change : On network installations, email notifications to administrators will include super admins if constant RVY_NOTIFY_SUPER_ADMIN is defined
+* Fixed : Network-wide Revisionary Options could not be modified
+* Fixed : Revisions on Edit Posts screen were displayed with stored post title, ignoring modifications by previous filters (such as translations)
+* Fixed : Administrator did not have "save as pending revision" option when post is currently scheduled for publishing
+* Fixed : Revision Diff formatting (column alignment)
+* Fixed : Revision preview from Revisions Manager screen not displayed correctly under some configurations
+* Change : Revisions Manager screen marks a revision as "Current" only if it is published
+* Change : Better consistency with standard Revisions Manager behavior: post-assigned Revisor role is sufficient to edit others' revisions, but post-assigned Contributor role is not
+* Change : Better consistency with standard Revisions Manager behavior: prevent diff display of unreadable revisions
+* Change : When comparing revisions, if only one of the revisions is past, force it to left
+* Change : On Revisions Manager screen, add margins to Update Revision button
+* Fixed : PHP Notices for non-static function calls
+* Compat : Role Scoper - when Pending Revision Monitors group is used and notification is "by default", recipient checkboxes missing on Edit Post form and TinyMCE broken
+* Compat : Duplicate Right Now links on dashboard if Role Scoper or Press Permit active
+
+**1.1.11 - 18 Aug 2013**
+
+= WP 3.6 Compatibility =
+* WP 3.6 - Revisors could not submit revisions
+* WP 3.6 - Don't modify native WP revision links
+* WP 3.6 - In Publish metabox, re-caption Revisions as "Publication History" to distinguish from Pending Revisions (prevent this by defining constant RVY_PREVENT_PUBHIST_CAPTION)
+* WP 3.6 - Post Title metabox was unformatted on Revisions Manager screen
+
+= Email Notification =
+* Fixed : Publishers to Notify metabox was displayed even if no selections available (when notification for both Publishers and Author is set to Always or Never)
+* Fixed : PHP warning in Publishers to Notify metabox when a user has a very long name
+* Change : If Press Permit or Role Scoper are active but Monitors group does not contain any users who can publish the post, notifications go to all WP Administrators and Editors who have sufficient site-wide capabilities (prevent this by defining constant RVY_FORCE_MONITOR_GROUPS)
+* Change : On Revisionary Settings screen, expand caption to clarify email notification behavior
+
+= General =
+* Fixed : Revisors could not select desired publish date on Edit Post screen, even if Scheduled Revisions enabled
+* Fixed : "save as pending" checkbox caused poor layout of adjacent UI in Publish metabox
+* Perf : Eliminate some redundant queries on back-end for non-Administrators (especially with Press Permit or Role Scoper active)
+* Compat : Edit Flow - don't offer to revise EF Metadata
+
+= 1.1.10 - 29 May 2013 =
+* SECURITY FIX : Revisions could be viewed by any registered user
+* Feature : Option to prevent Revisors from viewing other user's drafts and regular pending posts (imposes edit_others_drafts cap requirement)
+* Fixed : Other users' revisions were viewable in Revisions Manager even if option to prevent is enabled
+* Fixed : "Publishers to Notify" metabox not displayed under some configurations
+* Fixed : "Publishers to Notify" metabox was displayed with checkboxes even if Revisionary settings are for both editors and author to always receive notification
+* Fixed : Email Notification for Pending Revision was not sent under some configurations
+* Fixed : Monitor Groups (with Press Permit or Role Scoper activated) did not regulate email notifications
+* Fixed : Users who cannot approve a revision received email notification under some configurations
+* Fixed : PHP warnings for deprecated WP function calls
+* Fixed : PHP warnings when "previewing" current revision
+* Fixed : Invalid notifications were sent on revision submission error
+* Fixed : JS warning on Edit Post form
+* Compat : Press Permit Core
+* Compat : Press Permit - revision previews could not be viewed by revisor (also requires PP Collaborative Editing 2.0.14-beta)
+* Compat : CForms (and possibly other plugins) - tinyMCE buttons were suppressed
+
+= 1.1.9 - 18 Jan 2012 =
+* Compat : Press Permit - PP roles were not applied under some configurations
+* Compat : Role Scoper - RS roles were not applied under some configurations (related fixes in RS 1.3.52)
+* Fixed: PHP Warning for mysql_get_server_info()
+
+= 1.1.8 - 20 Dec 2011 =
+* Compat : Role Scoper - duplicate Pending counts in Dashboard Right Now
+* API : new filter - rvy_hidden_meta_boxes
+* API : new action: - rvy-revisions_sidebar
+* API : new action - rvy-revisions_meta_boxes
+* API : new action - revision_approved
+* API : new action - post_revision_update
+
+= 1.1.7 - 11 Nov 2011 =
+* Compat : WP 3.3 - Revision Editor displayed redundantly, didn't work
+* Compat : Press Permit integration
+* Feature : By default, Revisor role does not enable editing other users' revisions (option to re-enable)
+* Fixed : If Visual Editor is disabled, html entities not displayed or updated correctly in Revisions Manager
+* Fixed : About Revisionary screen (linked from help menu) failed to display
+* Fixed : Revision previews used wrong template under some configurations
+* Fixed : Various PHP Notices
+
+= 1.1.6 - 7 Sep 2011 =
+* Fixed : Quick Edit was not disabled for Page Revisions, usage resulted in invalid revision data
+* Fixed : Revisionary Options were not available when plugin activatated per-site on a Multisite installation
+* Fixed : For Multisite installation, Revisionary Options on Sites menu caused a fatal error
+* Change : For Multisite installation, Revisionary Options Blog/Site captions changed to Site/Network
+* Fixed : Revised Post Title was not displayed in Revisions Manager
+* Fixed : Various PHP Notices
+
+= 1.1.5 - 29 June 2011 =
+* Fixed : Markup error in Revisions Manager for Administrators / Editors, especially noticeable in WP 3.2
+* Fixed : "save as pending revision" checkbox in Publish metabox caused formatting error with IE9
+* Fixed : Previews did not display post thumbnail or other meta data
+* Fixed : Previews could not be displayed for past revisions
+* Compat : WP 3.2 - revision previews did not work
+* Compat : WP 3.2 - preview link not displayed for Pending Revisions in edit.php listing
+* Compat : Builder theme - previews of page revisions could not be displayed
+* Compat : Events Calendar Pro - filtering fails when WP database prefix is non-default
+* Change : Better styling for revision approval link displayed above preview
+* Change : Remove Asynchronous Email option
+* Change : Change all require and include statements to absolute path to work around oddball servers that can't handle relative paths
+* Change : jQuery syntax change for forward compatibility
+
+= 1.1.4 - 5 Apr 2011 =
+* Fixed : Role Options, Role Defaults menu items were not available on 3.1 multisite
+* Fixed : Pending / Scheduled Revisions could not be previewed by Revisors
+* Fixed : "Submit Revision" button caption changed to "Update" or "Schedule" following publish date selection
+* Fixed : PHP Warning on post creation / update
+* Change : Hide Preview button from Revisors when editing for pending revision submission
+
+= 1.1.3 - 3 Dec 2010 =
+* Fixed : Autosave error message displayed while a revisor edits a published post prior to submitting a pending revision
+* Fixed : Email notifications failed on some servers if Asynchronous option enabled
+* Compat : Role Scoper - With RS 1.3 to 1.3.12, if another plugin (Events Manager) triggers a secondary edit_posts cap check when a Revisor attempts to edit another user's unpublished post, a pending revision is generated instead of just updating the unpublished post
+
+= 1.1.2 - 29 Nov 2010 =
+* Compat : Role Scoper - Post-assigned Revisor role was not honored to update another users' revision with RS 1.3+
+* Fixed : While in Revisions Manager, invalid "Revisions" submenu link was displayed in Settings menu
+
+= 1.1.1 - 5 Nov 2010 =
+* Fixed : Fatal Error if theme displays post edit link on front end
+* Fixed : Did not observe capability definitions for custom post types (assumed capability_type = post_type)
+* Compat : Event Calendar Pro - revisions of sp_events were not included in Edit Posts listing due to postmeta clause applied by ECP
+
+= 1.1 - 2 Nov 2010 =
+* Fixed : Revision Approval notices were not sent if "always send" option enabled
+* Feature : "save as pending revision" option when logged user has full editing capabilities in Edit Post/Page form
+
+= 1.1.RC3 - 29 Oct 2010 =
+* Fixed : Revision preview link returned 404 (since 1.1.RC)
+* Fixed : Revision Approval emails were not sent reliably with "Asynchronous Email" option enabled (since 1.0)
+* Fixed : Custom taxonomy selection UI was not hidden when submitting a revision
+* Fixed : In Quick Edit form, Published option sometimes displayed inappropriately
+
+= 1.1.RC.2 - 11 Oct 2010 =
+* Fixed : Listed revisions in Revision Editor were not linked for viewing / editing (since 1.1.RC)
+
+= 1.1.RC - 8 Oct 2010 =
+* Feature : Support Custom Post Types
+* Change : Better internal support for custom statuses
+* Fixed : On Options page, links to "Pending Revision Monitors" and "Scheduled Revision Monitors" were reversed
+* Fixed : Revision Edit link from Edit Posts/Pages listing led to uneditable revision display
+* Change : Raise minimum WP version to 3.0
+
+= 1.0.7 - 21 June 2010 =
+* Fixed : Revisionary prevented the normal scheduling of drafts for first-time publishing
+
+= 1.0.6 - 18 June 2010 =
+* Compat : CForms conflict broke TinyMCE edit form in Revisions Manager
+
+= 1.0.5 - 7 May 2010 =
+* Compat : WP 3.0 Multisite menu items had invalid link
+
+= 1.0.4 - 6 May 2010 =
+* Fixed : Pending Revision Approval email used invalid permalink if permalink structure changed since original post storage
+* Fixed : Schedule Revision Publication email used invalid permalink if permalink structure changed since original post storage
+
+= 1.0.3 - 6 May 2010 =
+* Compat : WP 3.0 elimination of page.php, edit-pages.php, page-new.php broke many aspects of page filtering
+* Fixed : Trash link did not work for revisions in Edit Posts/Pages listing
+* Change : Administrators and Editors now retain Quick Edit link for non-revisions in Edit Pages, Edit Posts listing
+* Fixed : "Publishers to Notify" metabox was included even if no eligible recipients are designated
+
+= 1.0.2 - 11 Mar 2010 =
+* Fixed : Email notification caused error if Role Scoper was not activated
+* Fixed : Database error message (nuisance) in non-MU installations (SELECT meta_key, meta_value FROM WHERE site_id...)
+* Fixed : Publish Now link on Scheduled Revision preview did not work
+* Fixed : With WP > 2.9, newly published revisions also remained listed as a Pending or Scheduled revision
+* Fixed : With WP > 2.9, revision date selection UI showed "undefined" caption next to new date selection
+* Fixed : Link for viewing Scheduled Revisions was captioned as "Pending Revisions" (since 1.0.1)
+* Compat : WMPL plugin
+
+= 1.0.1 - 6 Feb 2010 =
+* Fixed : Submitting a Pending Revision to a published Post failed with Fatal Error
+* Fixed : PHP short tag caused Parse Error on servers which were not configured to support it
+* Compat : Support TinyMCE Advanced and WP Super Edit for custom editor buttons on Revision Management form
+* Feature : Revision preview bar can be styled via CSS file
+* Lang : Fixed several string formatting issues for better translation support
+* Change : Use https link for Revisionary css and js files if ssl is being used / forced for the current uri
+
+= 1.0 - 30 Dec 2009 =
+* Feature : Use Blog Title and Admin Email as from address in revision notices, instead of "WordPress
"
+* Fixed : Revision Approval / Publication Notices used p=ID link instead of normal post permalink
+* Compat : Display workaround instructions for FolioPress conflict with visual revision display
+
+**1.0.RC1 - 12 Dec 2009**
+Initial release. Feature Changes and Bug Fixes are vs. Pending Revisions function in Role Scoper 1.0.8
+
+= General: =
+* Feature : Scheduled Revisions - submitter can specify a desired publication date for a revision
+* Feature : Any user with the delete_published_ and edit_published capabilities for a post/page can administer its revisions (must include those caps in RS Editor definitions and assign that role)
+* Feature : Scheduled Publishing and Email notification is processed asynchronously
+
+= Revisions Manager: =
+* Feature : Dedicated Revisions Manager provides more meaningful captions, classified by Past / Pending / Scheduled
+* Feature : RS Revision Manager form displays visually via TinyMCE, supports editing of content, title and date
+* Feature : Revisions Manager supports individual or bulk deletion
+* Feature : Users can view their own Pending and Scheduled Revisions
+* Feature : Users can delete their own Pending Revisions until approval
+
+= Preview: =
+* Feature : Preview a Pending Revision, with top link to publish / schedule it
+* Feature : Preview a Scheduled Revision, with top link fo publish it now
+* Feature : Preview a Past Revision, with top link for restore it
+
+= WP Admin: =
+* Feature : Pending and Scheduled revisions are included in Edit Posts / Pages list for all qualified users
+* Feature : Delete, View links on revisions in Edit Posts / Pages list redirect to RS Revisions Manager
+* Feature : Add pending posts and pages total to Dashboard Right Now list (includes both new post submissions and Pending Revisions)
+* Feature : Metaboxes in Edit Post/Page form for Pending / Scheduled Revisions
+* Fixed : Multiple Pending Revions created by autosave
+* Fixed : Users cannot preview their changes before submitting a Pending Revision on a published post/page
+* Fixed : Pending Post Revisions were not visible to Administrator in Edit Posts list
+* Fixed : Both Pending Page Revisions and Pending Post Revisions were visible to Administator in Edit Pages list
+* Fixed : Pending Revisions were not included in list for restoration
+* Fixed : Bulk Deletion attempt failed when pending / scheduled revisions were included in selection
+
+= Notification: =
+* Feature : Optional email (to editors or post author) on Pending Revision submission
+* Feature : Optional email (to editors, post author, or revisor) on Pending Revision approval
+* Feature : Optional email (to editors, post author, or revisor) on Scheduled Revision publication
+* Feature : If Role Scoper is active, Editors notification group can be customized via User Group
+
+== Upgrade Notice ==
+
+= 1.2.3 =
+Important Fix: Scheduled Revision publication failure with runaway email notifications (since 1.2)
+
+= 1.1.10 =
+SECURITY FIX: Revisions could be viewed by any registered user
+
+= 1.1.5 =
+Fixes: Markeup Err in Revisions Manager; Revision Previews (WP 3.2, Display of Post Thumbnail & other metadata, Past Revisions, Page Revisions in Builder theme, Approval link styling); IE9 formatting err in publish metabox; Events Calendar Pro conflict
diff --git a/wp-content/plugins/revisionary/rest_rvy.php b/wp-content/plugins/revisionary/rest_rvy.php
new file mode 100644
index 000000000..b5e37a440
--- /dev/null
+++ b/wp-content/plugins/revisionary/rest_rvy.php
@@ -0,0 +1,116 @@
+operation ) {
+ $post_type = get_post_field( 'post_type', $item_id );
+
+ if ( $type_obj = get_post_type_object( $post_type ) ) {
+ if ( $orig_cap == $type_obj->cap->read_post ) {
+ $orig_cap = 'edit_post';
+ }
+ }
+ }
+
+ return $orig_cap;
+ }
+
+ function pre_dispatch( $rest_response, $rest_server, $request ) {
+ $this->method = $request->get_method();
+ $path = $request->get_route();
+
+ foreach ( $rest_server->get_routes() as $route => $handlers ) {
+ if ( ! $match = @preg_match( '@^' . $route . '$@i', $path, $args ) )
+ continue;
+
+ foreach ( $handlers as $handler ) {
+ if ( ! empty( $handler['methods'][ $this->method ] ) && is_array($handler['callback']) && is_object($handler['callback'][0]) ) {
+
+ $this->endpoint_class = get_class( $handler['callback'][0] );
+
+ $compatible_endpoints = apply_filters(
+ 'revisionary_rest_post_endpoints',
+ ['WP_REST_Posts_Controller', 'LD_REST_Posts_Gutenberg_Controller']
+ );
+
+ if (!in_array($this->endpoint_class, $compatible_endpoints)) {
+ continue;
+ }
+
+ $this->request = $request;
+
+ $this->is_view_method = in_array( $this->method, array( WP_REST_Server::READABLE, 'GET' ) );
+
+ $post_id = self::get_id_element( $path );
+
+ if (is_numeric($post_id)) {
+ // back post type out of path because controller object does not expose it
+ $type_base = $this->get_path_element( $path );
+
+ $this->post_type = $this->get_type_from_rest_base( $type_base );
+ $this->post_id = $post_id;
+ $this->is_posts_request = true;
+ }
+ }
+ }
+ }
+
+ return $rest_response;
+ } // end function pre_dispatch
+
+ public static function get_id_element( $path, $position_from_right = 0 ) {
+ $arr_path = explode( '/', $path );
+
+ $count = -1;
+ for( $i=count($arr_path) - 1; $i>=0; $i-- ) {
+ $count++;
+
+ if ( $count == $position_from_right )
+ return $arr_path[$i];
+ }
+
+ return '';
+ }
+
+ function get_path_element( $path, $string_num_from_right = 1 ) {
+ $arr_path = explode( '/', $path );
+
+ $count = 0;
+ for( $i=count($arr_path) - 1; $i>=0; $i-- ) {
+ if ( is_numeric( $arr_path[$i] ) )
+ continue;
+
+ $count++;
+
+ if ( $count == $string_num_from_right )
+ return $arr_path[$i];
+ }
+
+ return '';
+ }
+
+ private function get_type_from_rest_base( $rest_base ) {
+ if ( $types = get_post_types( array( 'rest_base' => $rest_base ) ) ) {
+ $post_type = reset( $types );
+ return $post_type;
+ } elseif( post_type_exists( $rest_base ) ) {
+ return $rest_base;
+ } else {
+ return false;
+ }
+ }
+}
diff --git a/wp-content/plugins/revisionary/revision-creation_rvy.php b/wp-content/plugins/revisionary/revision-creation_rvy.php
new file mode 100644
index 000000000..f18e2abf9
--- /dev/null
+++ b/wp-content/plugins/revisionary/revision-creation_rvy.php
@@ -0,0 +1,264 @@
+revisionary = $args['revisionary'];
+ }
+ }
+
+ // @todo: status change handler for draft => future-revision
+ function flt_future_revision_status_change($revision_status, $old_status, $revision_id) {
+ if ('future-revision' == $revision_status) {
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ rvy_update_next_publish_date(['revision_id' => $revision_id]);
+ }
+
+ if ('pending-revision' == $revision_status) {
+ $revision = get_post($revision_id);
+
+ do_action('revisionary_submit_revision', $revision);
+
+ /**
+ * Trigger after a revision creation.
+ *
+ * @param int $revision The revision object.
+ */
+ do_action('revisionary_created_revision', $revision);
+ }
+ }
+
+ function flt_pending_revision_data( $data, $postarr ) {
+
+ if (rvy_is_revision_status($postarr['post_mime_type'])) {
+ if ($data['post_name'] != $postarr['post_name']) {
+ add_post_meta($revision_id, '_requested_slug', $data['post_name']);
+ $data['post_name'] = $postarr['post_name'];
+ }
+ }
+
+ return $data;
+ }
+
+ static function fltInterruptPostMetaOperation($interrupt) {
+ return true;
+ }
+
+ // Create a new revision, usually 'draft-revision' (Working Copy) or 'future-revision' (Scheduled Revision)
+
+ // If an autosave was stored for the current user prior to this creation, it will be retrieved in place of the main revision.
+ function createRevision($post_id, $revision_status, $args = []) {
+ global $wpdb, $current_user;
+
+ $is_revision = rvy_in_revision_workflow($post_id);
+
+ $main_post_id = $is_revision ? rvy_post_id($post_id) : $post_id;
+
+ $published_post = get_post($main_post_id);
+ $source_post = get_post($post_id);
+
+ $set_post_properties = [
+ 'post_content',
+ 'post_content_filtered',
+ 'post_title',
+ 'post_excerpt',
+ 'comment_status',
+ 'ping_status',
+ 'post_password',
+ 'menu_order',
+ ];
+
+ $data = [];
+
+ if (!$is_revision) {
+ if ($autosave_post = Utils::get_post_autosave($post_id, $current_user->ID)) {
+ if (strtotime($autosave_post->post_modified_gmt) > strtotime($source_post->post_modified_gmt)) {
+ $use_autosave = true;
+ $args['meta_post_id'] = $autosave_post->ID;
+ }
+ }
+ }
+
+ foreach($set_post_properties as $prop) {
+ $data[$prop] = (!empty($use_autosave) && !empty($autosave_post->$prop)) ? $autosave_post->$prop : $source_post->$prop;
+ }
+
+ $data['post_type'] = $source_post->post_type;
+ $data['post_parent'] = ($is_revision) ? $published_post->post_parent : $source_post->post_parent;
+
+ if (!defined('REVISIONARY_LEGACY_REVISION_AUTHOR') && !empty($current_user) && !empty($current_user->ID)) {
+ $data['post_author'] = $current_user->ID;
+ }
+
+ if (!empty($args['time_gmt'])) {
+ $timestamp = $args['time_gmt'];
+ $data['post_date_gmt'] = gmdate( 'Y-m-d H:i:s', $timestamp);
+ $data['post_date'] = gmdate( 'Y-m-d H:i:s', $timestamp + (int) ( get_option( 'gmt_offset' ) * HOUR_IN_SECONDS ));
+ }
+
+ $args['main_post_id'] = $main_post_id;
+
+ $revision_id = $this->insert_revision($data, $source_post->ID, $revision_status, $args);
+
+ if (!empty($use_autosave)) {
+ $wpdb->delete($wpdb->posts, ['ID' => $autosave_post->ID]);
+ }
+
+ if ('future-revision' == $revision_status) {
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ rvy_update_next_publish_date(['revision_id' => $revision_id]);
+ }
+
+ if (!$revision_id || !is_scalar($revision_id)) { // update_post_data() returns array or object on update abandon / failure
+ return;
+ }
+
+ $post = get_post($revision_id);
+
+ $url = apply_filters('revisionary_create_revision_redirect', rvy_admin_url("post.php?post=$revision_id&action=edit"), $revision_id);
+
+ if (!empty($args['suppress_redirect'])) {
+ return $revision_id;
+ }
+
+ wp_redirect($url);
+ exit;
+ }
+
+ private function insert_revision($data, $base_post_id, $revision_status, $args = []) {
+ global $wpdb, $current_user;
+
+ $data['post_mime_type'] = $revision_status;
+
+ switch($revision_status) {
+ case 'draft-revision':
+ $data['post_status'] = 'draft';
+
+ $data['post_date'] = current_time( 'mysql' );
+ $data['post_date_gmt'] = current_time( 'mysql', 1 );
+ break;
+
+ case 'future-revision':
+ $data['post_status'] = 'pending';
+ break;
+
+ default:
+ $data['post_status'] = 'pending';
+ }
+
+ $main_post_id = (!empty($args['main_post_id'])) ? $args['main_post_id'] : $base_post_id;
+
+ $base_post = get_post($main_post_id);
+
+ if (!empty($base_post) && !empty($base_post->post_status) && ('revision' == $base_post->post_type)) {
+ $main_post_id = $base_post->post_parent;
+
+ } elseif (!empty($base_post) && !empty($base_post->post_mime_type) && in_array($base_post->post_mime_type, ['draft-revision', 'pending-revision', 'future-revision'])) {
+ $main_post_id = $base_post->comment_count;
+ }
+
+ $data['comment_count'] = $main_post_id; // buffer this value in posts table for query efficiency (actual comment count stored for published post will not be overwritten)
+
+ $data['post_author'] = $current_user->ID; // store current user as revision author (but will retain current post_author on restoration)
+
+ $data['post_modified'] = current_time( 'mysql' );
+ $data['post_modified_gmt'] = current_time( 'mysql', 1 );
+
+ if ( $future_date = ! empty($data['post_date']) && ( strtotime($data['post_date_gmt'] ) > agp_time_gmt() ) ) { // in past versions, $future_date was also passed to get_revision_msg()
+ // round down to zero seconds
+ $data['post_date_gmt'] = date( 'Y-m-d H:i:00', strtotime( $data['post_date_gmt'] ) );
+ $data['post_date'] = date( 'Y-m-d H:i:00', strtotime( $data['post_date'] ) );
+ }
+
+ // @todo: confirm this is still needed
+ $data['guid'] = '';
+ $data['post_name'] = '';
+
+ $revision_id = wp_insert_post(\wp_slash($data), true);
+
+ if (is_wp_error($revision_id)) {
+ return new \WP_Error(esc_html__( 'Could not insert revision into the database', 'revisionary'));
+ }
+
+ $update_data = ('pending-revision' == $data['post_mime_type']) //
+ ? ['comment_count' => $main_post_id, 'post_modified_gmt' => $data['post_modified_gmt'], 'post_modified' => $data['post_modified']]
+ : ['comment_count' => $main_post_id];
+
+ $wpdb->update($wpdb->posts, $update_data, ['ID' => $revision_id]);
+
+ /**
+ * Fired when a new revision is being inserted into the database.
+ *
+ * @param int $revision_id The ID of the inserted revision.
+ * @param int $main_post_id The ID of the published post for this revision.
+ * @param array $data The post data used to create this revision.
+ */
+ do_action( 'revisionary_new_revision_inserting', $revision_id, $main_post_id, $data );
+
+ if (!defined('REVISONARY_CREATE_REVISION_NO_COMMENT_COUNT_UPDATE')) {
+ // Hack WP into updating the comment count to store the main post ID in the comment_count field.
+ add_filter(
+ 'pre_wp_update_comment_count_now',
+ function( $new, $old, $post_id ) use ( $revision_id, $main_post_id ) {
+ if ( (int) $revision_id === (int) $post_id ) {
+ return $main_post_id;
+ }
+
+ return $new;
+ },
+ 10,
+ 3
+ );
+
+ // Update the comment count.
+ wp_update_comment_count_now( $revision_id );
+ }
+
+ // Use the newly generated $post_ID.
+ $where = array( 'ID' => $revision_id );
+
+ // make sure autosave still exists
+ if (!empty($args['meta_post_id'])) {
+ $post = get_post($args['meta_post_id']);
+
+ if (empty($post) || empty($post->post_type)) {
+ unset($args['meta_post_id']);
+ }
+ }
+
+ if (!empty($args['meta_post_id']) && apply_filters('revisionary_use_autodraft_meta', true, $data)) {
+ revisionary_copy_terms($args['meta_post_id'], $revision_id);
+ revisionary_copy_postmeta($args['meta_post_id'], $revision_id);
+
+ // For taxonomies and meta keys not stored for the autosave, use published copies
+ revisionary_copy_terms($base_post_id, $revision_id, ['empty_target_only' => true]);
+ revisionary_copy_postmeta($base_post_id, $revision_id, ['empty_target_only' => true]);
+ } else {
+ // If term selections are not posted for revision, store current published terms
+ revisionary_copy_terms($base_post_id, $revision_id);
+ revisionary_copy_postmeta($base_post_id, $revision_id);
+ }
+
+ rvy_update_post_meta($revision_id, '_rvy_base_post_id', $base_post_id);
+
+ if (!defined('REVISIONARY_LIMIT_IGNORE_UNSUBMITTED')) {
+ rvy_update_post_meta($base_post_id, '_rvy_has_revisions', true);
+ }
+
+ // Set GUID. @todo: still needed?
+ if ( '' == get_post_field( 'guid', $revision_id ) ) {
+ // need to give revision a guid for 3rd party editor compat (post_ID is ID of revision)
+ $wpdb->update( $wpdb->posts, array( 'guid' => get_permalink( $revision_id ) ), $where );
+ }
+
+ $data['ID'] = $revision_id;
+ do_action('revisionary_new_revision', $revision_id, $revision_status);
+
+ return (int) $revision_id; // only return array in calling function should return
+ }
+}
diff --git a/wp-content/plugins/revisionary/revision-workflow_rvy.php b/wp-content/plugins/revisionary/revision-workflow_rvy.php
new file mode 100644
index 000000000..67f5a5777
--- /dev/null
+++ b/wp-content/plugins/revisionary/revision-workflow_rvy.php
@@ -0,0 +1,330 @@
+content_roles->ensure_init();
+
+ if ( $publisher_ids = $revisionary->content_roles->get_metagroup_members( 'Pending Revision Monitors' ) ) {
+ if ( $type_obj ) {
+ $cols = ( defined('COLS_ALL_RS') ) ? COLS_ALL_RS : 'all';
+ $post_publishers = $revisionary->content_roles->users_who_can( 'edit_post', $object_id, array( 'cols' => $cols, 'force_refresh' => true, 'user_ids' => $publisher_ids ) );
+
+ $can_publish_post = array();
+ foreach ( $post_publishers as $key => $user ) {
+ $can_publish_post []= $user->ID;
+
+ if ( ! in_array( $user->ID, $publisher_ids ) )
+ unset( $post_publishers[$key] );
+ }
+
+ $publisher_ids = array_intersect( $publisher_ids, $can_publish_post );
+ $publisher_ids = array_fill_keys( $publisher_ids, true );
+ }
+ }
+ }
+
+ if ( ! $publisher_ids && ( empty($monitor_groups_enabled) || ! defined('RVY_FORCE_MONITOR_GROUPS') ) ) {
+ // If RS is not active, default to sending to all Administrators and Editors who can publish the post
+ require_once(ABSPATH . 'wp-admin/includes/user.php');
+
+ if ( defined( 'SCOPER_MONITOR_ROLES' ) )
+ $use_wp_roles = SCOPER_MONITOR_ROLES;
+ else
+ $use_wp_roles = ( defined( 'RVY_MONITOR_ROLES' ) ) ? RVY_MONITOR_ROLES : 'administrator,editor';
+
+ $use_wp_roles = str_replace( ' ', '', $use_wp_roles );
+ $use_wp_roles = explode( ',', $use_wp_roles );
+
+ $recipients = array();
+
+ foreach ( $use_wp_roles as $role_name ) {
+ $search = new WP_User_Query( "search=&role=$role_name" );
+ $recipients = array_merge( $recipients, $search->results );
+ }
+
+ foreach ( $recipients as $_user ) {
+ $reqd_caps = map_meta_cap( 'edit_post', $_user->ID, $object_id );
+
+ if ( ! array_diff( $reqd_caps, array_keys( array_intersect( $_user->allcaps, array( true, 1, '1' ) ) ) ) ) {
+ $post_publishers []= $_user;
+ $publisher_ids [$_user->ID] = true;
+ }
+ }
+ }
+
+ // boolean array with user IDs as array keys
+ $default_ids = apply_filters('revisionary_notify_publisher_default_ids', $publisher_ids, $object_id);
+ }
+
+ if ( '1' === $notify_author ) {
+ global $post;
+
+ if (function_exists('get_multiple_authors')) {
+ $author_ids = [];
+ foreach(get_multiple_authors($post) as $_author) {
+ $author_ids []= $_author->ID;
+ }
+ } else {
+ $author_ids = [$post->post_author];
+ }
+
+ foreach($author_ids as $author_id) {
+ if ( empty( $default_ids[$author_id] ) ) {
+ if ( defined('RVY_CONTENT_ROLES') ) {
+ $cols = ( defined('COLS_ALL_RS') ) ? COLS_ALL_RS : 'all';
+ $author_notify = (bool) $revisionary->content_roles->users_who_can( 'edit_post', $object_id, array( 'cols' => $cols, 'force_refresh' => true, 'user_ids' => (array) $author_id ) );
+ } else {
+ $_user = new WP_User($author_id);
+ $reqd_caps = map_meta_cap( 'edit_post', $_user->ID, $object_id );
+ $author_notify = ! array_diff( $reqd_caps, array_keys( array_intersect( $_user->allcaps, array( true, 1, '1' ) ) ) );
+ }
+
+ if ( $author_notify ) {
+ $default_ids[$author_id] = true;
+
+ $user = new WP_User( $author_id );
+ $post_publishers[] = $user;
+ }
+ }
+ }
+ }
+
+ if ($default_ids) {
+ // array of WP_User objects
+ $post_publishers = apply_filters('revisionary_notify_publishers_eligible', $post_publishers, $object_id);
+ }
+
+ return compact('default_ids', 'post_publishers', 'publisher_ids');
+ }
+
+ function do_notifications( $notification_type, $status, $post_arr, $args ) {
+ global $revisionary, $current_user;
+
+ if ( 'pending-revision' != $notification_type ) {
+ return;
+ }
+
+ $defaults = array( 'revision_id' => 0, 'published_post' => false, 'object_type' => '', 'selected_recipients' => array() );
+ $args = array_merge( $defaults, $args );
+ foreach( array_keys($defaults) as $var ) { $$var = $args[$var]; }
+
+ $revision_id = (int) $revision_id;
+ $object_type = sanitize_key($object_type);
+
+ if ( ! $published_post ) {
+ return;
+ }
+
+ // Support workaround to prevent notification when an Administrator or Editor voluntarily creates a pending revision
+ if (defined('REVISIONARY_LIMIT_ADMIN_NOTIFICATIONS') && current_user_can('edit_post', $published_post->ID)) {
+ return;
+ }
+
+ if ( $revisionary->doing_rest && $revisionary->rest->is_posts_request && ! empty( $revisionary->rest->request ) ) {
+ $post_arr = array_merge( $revisionary->rest->request->get_params(), $post_arr );
+ }
+
+ $recipient_ids = [];
+
+ $admin_notify = rvy_get_option( 'pending_rev_notify_admin' );
+ $author_notify = rvy_get_option( 'pending_rev_notify_author' );
+
+ if ( ( $admin_notify || $author_notify ) && $revision_id ) {
+ $type_obj = get_post_type_object( $object_type );
+ $type_caption = strtolower($type_obj->labels->singular_name);
+ $post_arr['post_type'] = $published_post->post_type;
+
+ $blogname = wp_specialchars_decode( get_option('blogname'), ENT_QUOTES );
+
+ if (!empty($args['update'])) {
+ $title = sprintf( esc_html__('[%s] %s Updated', 'revisionary'), $blogname, pp_revisions_status_label('pending-revision', 'name') );
+
+ $message = sprintf( esc_html__('%1$s updated a %2$s of the %3$s "%4$s".', 'revisionary'), $current_user->display_name, strtolower(pp_revisions_status_label('pending-revision', 'name')), $type_caption, $post_arr['post_title'] ) . "\r\n\r\n";
+ } else {
+ $title = sprintf( esc_html__('[%s] %s', 'revisionary'), $blogname, pp_revisions_status_label('pending-revision', 'name') );
+
+ $message = sprintf( esc_html__('%1$s submitted changes to the %2$s "%3$s". You can review the changes for possible publication:', 'revisionary'), $current_user->display_name, $type_caption, $post_arr['post_title'] ) . "\r\n\r\n";
+ }
+
+ if ( $revision_id ) {
+ $revision = get_post($revision_id);
+
+ if (rvy_get_option('revision_preview_links') || current_user_can('administrator') || is_super_admin()) {
+ $preview_link = rvy_preview_url($revision);
+ $message .= esc_html__( 'Preview and Approval: ', 'revisionary' ) . $preview_link . "\r\n\r\n";
+ }
+
+ $message .= esc_html__( 'Revision Queue: ', 'revisionary' ) . rvy_admin_url("admin.php?page=revisionary-q&published_post={$published_post->ID}&all=1") . "\r\n\r\n";
+
+ $message .= sprintf(esc_html__( 'Edit %s: ', 'revisionary' ), pp_revisions_status_label('pending-revision', 'name')) . rvy_admin_url("post.php?action=edit&post={$revision_id}") . "\r\n";
+ }
+
+ if ( $admin_notify ) {
+ // establish the publisher recipients
+ $recipient_ids = apply_filters('revisionary_submission_notify_admin', self::getRecipients('rev_submission_notify_admin', compact('type_obj', 'published_post')), ['post_type' => $object_type, 'post_id' => $published_post->ID, 'revision_id' => $revision_id]);
+
+ if ( ( 'always' != $admin_notify ) && $selected_recipients ) {
+ // intersect default recipients with selected recipients
+ $recipient_ids = array_intersect( $selected_recipients, $recipient_ids );
+ }
+
+ if ( defined( 'RVY_NOTIFY_SUPER_ADMIN' ) && is_multisite() ) {
+ $super_admin_logins = get_super_admins();
+ foreach( $super_admin_logins as $user_login ) {
+ if ( $super = new WP_User($user_login) ) {
+ $recipient_ids []= $super->ID;
+ }
+ }
+ }
+ }
+
+ if ( $author_notify ) {
+ if (function_exists('get_multiple_authors')) {
+ $author_ids = [];
+ foreach(get_multiple_authors($published_post) as $_author) {
+ $author_ids []= $_author->ID;
+ }
+ } else {
+ $author_ids = [$published_post->post_author];
+ }
+
+ if ('always' != $author_notify) {
+ $author_ids = $selected_recipients ? array_intersect($author_ids, $selected_recipients) : [];
+ }
+
+ $recipient_ids = array_merge($recipient_ids, $author_ids);
+ }
+
+ if ( $recipient_ids ) {
+ $to_addresses = [];
+
+ foreach($recipient_ids as $user_id) {
+ $user = new WP_User($user_id);
+
+ if ($user->exists() && !empty($user->user_email)) {
+ $to_addresses[$user_id] = $user->user_email;
+ }
+ }
+
+ $to_addresses = array_unique($to_addresses);
+ } else {
+ $to_addresses = array();
+ }
+
+ $message = str_replace('"', '"', $message);
+
+ foreach ( $to_addresses as $user_id => $address ) {
+ if (!empty($author_ids) && in_array($user_id, $author_ids)) {
+ $notification_class = 'rev_submission_notify_author';
+ } elseif (!empty($monitor_ids) && in_array($user_id, $monitor_ids)) {
+ $notification_class = 'rev_submission_notify_monitor';
+ } else {
+ $notification_class = 'rev_submission_notify_admin';
+ }
+
+ rvy_mail(
+ $address,
+ $title,
+ $message,
+ [
+ 'revision_id' => $revision_id,
+ 'post_id' => $published_post->ID,
+ 'notification_type' => $notification_type,
+ 'notification_class' => $notification_class,
+ ]
+ );
+ }
+ }
+ }
+
+ static function getRecipients($notification_class, $args) {
+ $defaults = ['type_obj' => false, 'published_post' => false];
+ foreach(array_keys($defaults) as $key) {
+ if (!empty($args[$key])) {
+ $$key = $args[$key];
+ } else {
+ return [];
+ }
+ }
+
+ $recipient_ids = [];
+
+ switch ($notification_class) {
+ case 'rev_submission_notify_admin' :
+ case 'rev_approval_notify_admin' :
+
+ do_action('presspermit_init_rvy_interface');
+
+ if ( defined('RVY_CONTENT_ROLES') && ! defined('SCOPER_DEFAULT_MONITOR_GROUPS') && ! defined('REVISIONARY_LIMIT_ADMIN_NOTIFICATIONS') ) {
+ global $revisionary;
+
+ $monitor_groups_enabled = true;
+ $revisionary->content_roles->ensure_init();
+
+ $recipient_ids = $revisionary->content_roles->get_metagroup_members( 'Pending Revision Monitors' );
+
+ if ( $type_obj ) {
+ $post_publisher_ids = $revisionary->content_roles->users_who_can( 'edit_post', $published_post->ID, array( 'cols' => 'id', 'user_ids' => $recipient_ids ) );
+ $recipient_ids = array_intersect( $recipient_ids, $post_publisher_ids );
+ }
+
+ $monitor_ids = $recipient_ids;
+ }
+
+ if (!$recipient_ids && (empty($monitor_groups_enabled) || ! defined('RVY_FORCE_MONITOR_GROUPS'))) {
+ require_once(ABSPATH . 'wp-admin/includes/user.php');
+
+ if ( defined( 'SCOPER_MONITOR_ROLES' ) ) {
+ $use_wp_roles = SCOPER_MONITOR_ROLES;
+ } else {
+ $use_wp_roles = ( defined( 'RVY_MONITOR_ROLES' ) ) ? RVY_MONITOR_ROLES : 'administrator,editor';
+ }
+
+ $use_wp_roles = str_replace( ' ', '', $use_wp_roles );
+ $use_wp_roles = explode( ',', $use_wp_roles );
+
+ foreach ( $use_wp_roles as $role_name ) {
+ $search = new WP_User_Query( "search=&fields=id&role=$role_name" );
+ $recipient_ids = array_merge( $recipient_ids, $search->results );
+ $recipient_ids = array_unique($recipient_ids);
+ }
+
+ if ( $recipient_ids && $type_obj ) {
+ foreach( $recipient_ids as $key => $user_id ) {
+ $_user = new WP_User($user_id);
+ $reqd_caps = map_meta_cap( 'edit_post', $user_id, $published_post->ID );
+
+ if (
+ array_diff( $reqd_caps, array_keys( array_intersect( $_user->allcaps, array( true, 1, '1' ) ) ) )
+ && !in_array('administrator', $_user->allcaps)
+ ) {
+ unset( $recipient_ids[$key] );
+ }
+ }
+ }
+ }
+
+ break;
+ default:
+ } // end switch notification_class
+
+ return $recipient_ids;
+ }
+}
diff --git a/wp-content/plugins/revisionary/revisionary-front.css b/wp-content/plugins/revisionary/revisionary-front.css
new file mode 100644
index 000000000..b45eeb88c
--- /dev/null
+++ b/wp-content/plugins/revisionary/revisionary-front.css
@@ -0,0 +1,198 @@
+/* PublishPress Revisions. Copyright (C) 2019, PublishPress
+ * This file is licensed under the GNU GPL.
+ */
+
+.rvy_view_revision,
+.preview_approval_rvy {
+ height: 50px;
+ width: 100%;
+ text-align: center;
+ padding-right: 20px;
+ color: white;
+ font-family: sans-serif;
+ font-weight: normal;
+ font-size: 0.9em;
+ z-index: 99990;
+}
+
+.rvy_preview_msgspan {
+ font-size: 0.9em;
+ line-height: 50px;
+}
+
+.rvy_preview_msgspan > a, .rvy_preview_msgspan > a:visited{
+ color:white;
+ margin-left:10px;
+}
+
+.rvy_preview_linkspan {
+ margin-left: 15px;
+ line-height: 50px;
+ padding: 6px 10px;
+ font-family: sans-serif;
+ font-size: 0.9em;
+ font-style: normal;
+ font-weight: normal;
+ text-decoration: none !important;
+ -webkit-border-radius: 5px;
+ -moz-border-radius: 5px;
+ border-radius: 5px;
+ border: 1px solid;
+ cursor: pointer;
+ color: #333;
+ z-index: 99999;
+}
+
+.rvy_view_published {
+ background-color: #635b93;
+}
+.rvy_view_published .rvy_preview_linkspan {
+ background-color: #E4E3D4;
+ border-color: #b2b1a7;
+}
+.rvy_view_published .rvy_preview_linkspan:hover {
+ background-color: #f7f6e7;
+}
+
+.rvy_view_past {
+ background-color: #ab5454;
+}
+.rvy_view_past .rvy_preview_linkspan {
+ background-color: #dfcdd0;
+ border-color: #c0b0b3;
+}
+.rvy_view_past .rvy_preview_linkspan:hover {
+ background-color: #FFECEF;
+}
+
+.rvy_view_draft {
+ background-color: #999;
+}
+.rvy_view_draft .rvy_preview_linkspan {
+ background-color: #cde0d2;
+ border-color: #a6b5a9;
+}
+.rvy_view_draft .rvy_preview_linkspan:hover {
+ background-color: #E4F5E8;
+}
+
+.rvy_view_pending {
+ background-color: #35b194;
+}
+.rvy_view_pending .rvy_preview_linkspan {
+ background-color: #cde0d2;
+ border-color: #a6b5a9;
+}
+.rvy_view_pending .rvy_preview_linkspan:hover {
+ background-color: #E4F5E8;
+}
+
+.rvy_view_future,
+.rvy_view_scheduled {
+ background-color: #888;
+}
+.rvy_view_future .rvy_preview_linkspan,
+.rvy_view_scheduled .rvy_preview_linkspan {
+ background-color: #cde6e7;
+ border-color: #a5bdbe;
+}
+.rvy_view_future .rvy_preview_linkspan:hover,
+.rvy_view_scheduled .rvy_preview_linkspan:hover {
+ background-color: #def8f9;
+}
+
+.rvy_view_pending_future {
+ color: white;
+ background-color: #35b194;
+}
+.rvy_view_pending_future .rvy_preview_linkspan {
+ background-color: #E1F6E8;
+ border-color: #cad6cd;
+}
+.rvy_view_pending_future .rvy_preview_linkspan:hover {
+ background-color: #f0fff4;
+}
+
+.preview_approval_rvy {
+ background-color: #635b93;
+}
+.preview_approval_rvy .rvy_preview_linkspan {
+ border-color: #a2babc;
+ background-color: #c4dcde;
+}
+.preview_approval_rvy .rvy_preview_linkspan:hover {
+ background-color: #D9F3F5;
+}
+
+span.rvy_preview_linkspan a, span.rvy_preview_linkspan a:visited, span.rvy_preview_linkspan a:hover {
+ text-decoration:none;
+ color: #333;
+}
+
+.rvy_preview_linkspan:hover {
+ color: #4e4e4e;
+}
+
+/* Button styles */
+
+#pp_revisions_top_bar .button,
+#pp_revisions_top_bar .button-primary,
+#pp_revisions_top_bar .button-secondary {
+ display: inline-block;
+ max-width: 220px !important;
+ text-decoration: none;
+ font-size: 13px;
+ line-height: 2.15384615;
+ min-height: 30px;
+ margin: 0;
+ padding: 0 10px;
+ cursor: pointer;
+ border-width: 1px;
+ border-style: solid;
+ -webkit-appearance: none;
+ border-radius: 3px;
+ white-space: nowrap;
+ box-sizing: border-box;
+}
+
+#pp_revisions_top_bar .button-secondary {
+ color: #2271b1;
+ border-color: #2271b1;
+ background: #f6f7f7;
+ text-decoration: none;
+}
+
+#pp_revisions_top_bar .button-secondary:hover,
+#pp_revisions_top_bar .button-secondary:focus,
+#pp_revisions_top_bar .button-secondary:active {
+ background: #f0f0f1;
+ border-color: #0a4b78;
+ color: #0a4b78;
+ text-decoration: none !important;
+}
+
+#pp_revisions_top_bar .button-primary {
+ background: #FFB300;
+ border-color: #C58C07;
+ color: #754D26;
+ text-decoration: none;
+ text-shadow: none;
+}
+
+#pp_revisions_top_bar .button-primary:hover,
+#pp_revisions_top_bar .button-primary:focus,
+#pp_revisions_top_bar .button-primary:active {
+ background-color: #F3AC04;
+ border-color: #C58C07;
+ color: #333;
+ text-decoration: none !important;
+ text-shadow: none;
+}
+
+#pp_revisions_top_bar .button {
+ margin-right: 7px;
+}
+
+#pp_revisions_top_bar .button.rvy_has_empty_spacing {
+ margin-right: 3px;
+}
diff --git a/wp-content/plugins/revisionary/revisionary.php b/wp-content/plugins/revisionary/revisionary.php
new file mode 100644
index 000000000..be9452017
--- /dev/null
+++ b/wp-content/plugins/revisionary/revisionary.php
@@ -0,0 +1,287 @@
+
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see .
+ *
+ * @package PublishPress\Revisions
+ * @author PublishPress
+ * @copyright Copyright (C) 2024 PublishPress. All rights reserved.
+ *
+ **/
+
+ if (!defined('ABSPATH')) exit; // Exit if accessed directly
+
+// Temporary usage within this module only; avoids multiple instances of version string
+global $pp_revisions_version;
+
+$pp_revisions_version = '3.5.8.2';
+
+global $wp_version;
+
+$min_php_version = '7.2.5';
+$min_wp_version = '5.5';
+
+$invalid_php_version = version_compare(phpversion(), $min_php_version, '<');
+$invalid_wp_version = version_compare($wp_version, $min_wp_version, '<');
+
+// If the PHP version is not compatible, terminate the plugin execution and show an admin notice.
+if (is_admin() && $invalid_php_version) {
+ add_action(
+ 'admin_notices',
+ function () use ($min_php_version) {
+ if (current_user_can('activate_plugins')) {
+ echo '';
+ printf(
+ 'PublishPress Revisions requires PHP version %s or higher.',
+ $min_php_version
+ );
+ echo '
';
+ }
+ }
+ );
+}
+
+// If the WP version is not compatible, terminate the plugin execution and show an admin notice.
+if (is_admin() && $invalid_wp_version) {
+ add_action(
+ 'admin_notices',
+ function () use ($min_wp_version) {
+ if (current_user_can('activate_plugins')) {
+ echo '';
+ printf(
+ 'PublishPress Revisions requires WordPress version %s or higher.',
+ $min_wp_version
+ );
+ echo '
';
+ }
+ }
+ );
+}
+
+if ($invalid_php_version || $invalid_wp_version) {
+ return;
+}
+
+$revisionary_pro_active = false;
+
+global $revisionary_loaded_by_pro;
+
+$revisionary_loaded_by_pro = strpos(str_replace('\\', '/', __FILE__), 'vendor/publishpress/');
+
+// Detect separate Pro plugin activation, but not self-activation (this file loaded in vendor library by Pro)
+if (false === $revisionary_loaded_by_pro) {
+ foreach ((array)get_option('active_plugins') as $plugin_file) {
+ if (false !== strpos($plugin_file, 'revisionary-pro.php')) {
+ $revisionary_pro_active = true;
+ break;
+ }
+ }
+
+ if (!$revisionary_pro_active && is_multisite()) {
+ foreach (array_keys((array)get_site_option('active_sitewide_plugins')) as $plugin_file) {
+ if (false !== strpos($plugin_file, 'revisionary-pro.php')) {
+ $revisionary_pro_active = true;
+ break;
+ }
+ }
+ }
+
+ if ($revisionary_pro_active) {
+ add_filter(
+ 'plugin_row_meta',
+ function($links, $file)
+ {
+ if ($file == plugin_basename(__FILE__)) {
+ $links[]= __('This plugin can be deleted. ', 'revisionary');
+ }
+
+ return $links;
+ },
+ 10, 2
+ );
+
+ add_action(
+ 'admin_notices',
+ function () {
+ if (current_user_can('activate_plugins')) {
+ echo ''
+ . 'Revisions Pro requires the free plugin (PublishPress Revisions) to be deactivated.'
+ . '
';
+ }
+ }
+ );
+
+ return;
+ }
+}
+
+if ( isset($_SERVER['SCRIPT_NAME']) && strpos( esc_url_raw($_SERVER['SCRIPT_NAME']), 'p-admin/index-extra.php' ) || strpos( esc_url_raw($_SERVER['SCRIPT_NAME']), 'p-admin/update.php' ) ) {
+ return;
+}
+
+if (! defined('REVISIONS_INTERNAL_VENDORPATH')) {
+ define('REVISIONS_INTERNAL_VENDORPATH', __DIR__ . '/lib/vendor');
+}
+
+if (!defined('REVISIONARY_FILE') && !$revisionary_loaded_by_pro) {
+ $includeFileRelativePath = REVISIONS_INTERNAL_VENDORPATH . '/publishpress/publishpress-instance-protection/include.php';
+ if (file_exists($includeFileRelativePath)) {
+ require_once $includeFileRelativePath;
+ }
+
+ if (class_exists('PublishPressInstanceProtection\\Config')) {
+ $pluginCheckerConfig = new PublishPressInstanceProtection\Config();
+ $pluginCheckerConfig->pluginSlug = 'revisionary';
+ $pluginCheckerConfig->pluginFolder = 'revisionary';
+ $pluginCheckerConfig->pluginName = 'PublishPress Revisions';
+
+ $pluginChecker = new PublishPressInstanceProtection\InstanceChecker($pluginCheckerConfig);
+ }
+
+ if (! class_exists('ComposerAutoloaderInitRevisionary')
+ && file_exists(REVISIONS_INTERNAL_VENDORPATH . '/autoload.php')
+ ) {
+ require_once REVISIONS_INTERNAL_VENDORPATH . '/autoload.php';
+ }
+}
+
+if (!defined('REVISIONARY_FILE') && (!$revisionary_pro_active || $revisionary_loaded_by_pro)) {
+ define('REVISIONARY_FILE', __FILE__);
+
+ add_action(
+ 'init',
+ function() {
+ global $pp_revisions_version;
+
+ if (!function_exists('revisionary')) {
+ require_once(dirname(__FILE__).'/functions.php');
+ pp_revisions_plugin_updated($pp_revisions_version);
+ }
+ },
+ 2
+ );
+
+ // register these functions before any early exits so normal activation/deactivation can still run with RS_DEBUG
+ register_activation_hook(__FILE__, function()
+ {
+ global $pp_revisions_version;
+
+ if (!function_exists('revisionary')) {
+ require_once(dirname(__FILE__).'/functions.php');
+ }
+
+ pp_revisions_plugin_updated($pp_revisions_version);
+ pp_revisions_plugin_activation();
+ }
+ );
+
+ register_deactivation_hook(__FILE__, function()
+ {
+ if (!function_exists('rvy_init')) {
+ require_once( dirname(__FILE__).'/rvy_init.php');
+ }
+
+ if (!rvy_is_plugin_active('revisionary-pro/revisionary-pro.php')) {
+ pp_revisions_plugin_deactivation();
+ }
+ }
+ );
+
+ // negative priority to precede any default WP action handlers
+ function revisionary_load() {
+ global $pp_revisions_version;
+
+ define('PUBLISHPRESS_REVISIONS_VERSION', $pp_revisions_version);
+
+ if ( ! defined( 'RVY_VERSION' ) ) {
+ define( 'RVY_VERSION', PUBLISHPRESS_REVISIONS_VERSION ); // back compat
+ }
+
+ define ('COLS_ALL_RVY', 0);
+ define ('COL_ID_RVY', 1);
+
+ if ( defined('RS_DEBUG') ) {
+ include_once( dirname(__FILE__).'/lib/debug.php');
+ add_action( 'admin_footer', 'rvy_echo_usage_message' );
+ } else
+ include_once( dirname(__FILE__).'/lib/debug_shell.php');
+
+ require_once( dirname(__FILE__).'/defaults_rvy.php');
+
+ // === awp_is_mu() function definition and usage: must be executed in this order, and before any checks of IS_MU_RVY constant ===
+ require_once( dirname(__FILE__).'/lib/agapetry_wp_core_lib.php');
+ define( 'IS_MU_RVY', awp_is_mu() );
+ // -------------------------------------------
+
+ require_once( dirname(__FILE__).'/content-roles_rvy.php');
+
+ if ( is_admin() || defined('XMLRPC_REQUEST') ) {
+ require_once( dirname(__FILE__).'/lib/agapetry_wp_admin_lib.php');
+
+ // skip WP version check and init operations when a WP plugin auto-update is in progress
+ if (isset($_SERVER['SCRIPT_NAME']) && false !== strpos(esc_url_raw($_SERVER['SCRIPT_NAME']), 'update.php') )
+ return;
+ }
+
+ require_once( dirname(__FILE__).'/classes/PublishPress/Revisionary.php');
+ require_once( dirname(__FILE__).'/rvy_init.php'); // Contains activate, deactivate, init functions. Adds mod_rewrite_rules.
+ require_once( dirname(__FILE__).'/functions.php');
+
+ // avoid lockout in case of editing plugin via wp-admin
+ if ( defined('RS_DEBUG') && is_admin() && isset($_SERVER['REQUEST_URI']) && ( strpos( urldecode(esc_url_raw($_SERVER['REQUEST_URI'])), 'p-admin/plugin-editor.php' ) || strpos( urldecode(esc_url_raw($_SERVER['REQUEST_URI'])), 'p-admin/plugins.php' ) ) && false === strpos( esc_url_raw($_SERVER['REQUEST_URI']), 'activate' ) )
+ return;
+
+ define('RVY_ABSPATH', __DIR__);
+
+ if (is_admin() && !defined('PUBLISHPRESS_REVISIONS_PRO_VERSION')) {
+ require_once(__DIR__ . '/includes/CoreAdmin.php');
+ new \PublishPress\Revisions\CoreAdmin();
+ }
+
+ rvy_refresh_options_sitewide();
+
+ // since sequence of set_current_user and init actions seems unreliable, make sure our current_user is loaded first
+ add_action('init', 'rvy_init', 1);
+
+ if (!defined('IFRAME_REQUEST')) {
+ add_action('init', 'rvy_add_revisor_custom_caps', 99);
+ add_action('wp_loaded', 'rvy_add_revisor_custom_caps', 99);
+ }
+
+ add_action('init', 'rvy_configuration_late_init', PHP_INT_MAX - 1);
+
+ revisionary();
+ }
+
+ // negative priority to precede any default WP action handlers
+ if ($revisionary_loaded_by_pro) {
+ revisionary_load(); // Pro support
+ } else {
+ add_action('plugins_loaded', 'revisionary_load', -10);
+ }
+}
diff --git a/wp-content/plugins/revisionary/revisionary_main.php b/wp-content/plugins/revisionary/revisionary_main.php
new file mode 100644
index 000000000..8f8b8621a
--- /dev/null
+++ b/wp-content/plugins/revisionary/revisionary_main.php
@@ -0,0 +1,1212 @@
+
+ * @copyright Copyright (c) 2024 PublishPress. All rights reserved.
+ * @license GPLv2 or later
+ * @since 1.0.0
+ */
+class Revisionary
+{
+ var $content_roles; // object ref - instance of RevisionaryContentRoles subclass, set by external plugin
+ var $doing_rest = false;
+ var $rest = ''; // object ref - Revisionary_REST
+ var $internal_meta_update = false;
+ var $skip_filtering = false;
+ var $is_revisions_query = false;
+ var $front = false;
+
+ var $config_loaded = false; // configuration related to post types and statuses must be loaded late on the init action
+ var $enabled_post_types = []; // enabled_post_types property is set (keyed by post type slug) late on the init action.
+ var $enabled_post_types_archive = []; // enabled_post_types_archive property is set (keyed by post type slug) late on the init action.
+ var $post_edit_ui;
+
+ // minimal config retrieval to support pre-init usage by WP_Scoped_User before text domain is loaded
+ function __construct() {
+ }
+
+ function init() {
+ if (isset($_SERVER['REQUEST_URI']) && is_admin() && (false !== strpos(esc_url_raw($_SERVER['REQUEST_URI']), 'revision.php')) && (!empty($_REQUEST['revision']))) {
+ add_action('init', [$this, 'addFilters'], PHP_INT_MAX);
+ } else {
+ $this->addFilters();
+ }
+ }
+
+ function addFilters() {
+ global $script_name;
+
+ add_filter('pre_wp_update_comment_count_now', [$this, 'fltUpdateCommentCountBypass'], 10, 3);
+
+ // Ensure editing access to past revisions is not accidentally filtered.
+ // @todo: Correct selective application of filtering downstream so Revisors can use a read-only Compare [Past] Revisions screen
+ //
+ // Note: some filtering is needed to allow users with full editing permissions on the published post to access a Compare Revisions screen with Preview and Manage buttons
+ if (is_admin() && isset($_SERVER['REQUEST_URI']) && (false !== strpos(esc_url_raw($_SERVER['REQUEST_URI']), 'revision.php')) && (!empty($_REQUEST['revision'])) && !is_content_administrator_rvy()) {
+ if (!empty($_REQUEST['revision'])) {
+ $revision_id = (int) $_REQUEST['revision'];
+ } elseif (isset($_REQUEST['to'])) {
+ $revision_id = (int) $_REQUEST['to'];
+ } else {
+ $revision_id = 0;
+ }
+
+ if ($revision_id) {
+ if ($_post = get_post($revision_id)) {
+ if (!rvy_in_revision_workflow($_post)) {
+ if ($parent_post = get_post($_post->post_parent)) {
+ if (!empty($_POST) || (!empty($_REQUEST['action']) && ('restore' == $_REQUEST['action']))) {
+ if (!$this->canEditPost($parent_post, ['simple_cap_check' => true])) {
+ return;
+ }
+ }
+ }
+ }
+ }
+ }
+ }
+
+ $this->setPostTypes();
+ $this->setPostTypesArchive();
+
+ rvy_refresh_options_sitewide();
+
+ if (defined('DOING_CRON') && DOING_CRON) {
+ if (!rvy_get_option('wp_cron_usage_detected')) {
+ update_option('rvy_wp_cron_usage_detected', true);
+ }
+ }
+
+ require_once( dirname(__FILE__).'/classes/PublishPress/Revisions/PluginCompat.php' );
+ new PublishPress\Revisions\PluginCompat();
+
+ // NOTE: $_GET['preview'] and $_GET['post_type'] arguments are set by rvy_init() at response to ?p= request when the requested post is a revision.
+ if (!is_admin() && (!defined('REST_REQUEST') || ! REST_REQUEST)) { // preview_id indicates a regular preview via WP core, based on autosave revision
+ $preview_arg = (defined('RVY_PREVIEW_ARG')) ? sanitize_key(constant('RVY_PREVIEW_ARG')) : 'rv_preview';
+
+ if ((defined('FL_BUILDER_VERSION') && rvy_in_revision_workflow(rvy_detect_post_id())) || ((!empty($_GET[$preview_arg]) || !empty($_GET['_ppp'])) && empty($_REQUEST['preview_id'])) || !empty($_GET['mark_current_revision'])) {
+ require_once( dirname(__FILE__).'/front_rvy.php' );
+ $this->front = new RevisionaryFront();
+ }
+ }
+
+ if (!is_admin() && (!defined('REST_REQUEST') || ! REST_REQUEST) && (!empty($_GET['preview']) && !empty($_REQUEST['preview_id']))) {
+ if (!defined('PUBLISHPRESS_MULTIPLE_AUTHORS_VERSION')) {
+ require_once(dirname(__FILE__).'/classes/PublishPress/Revisions/PostPreview.php');
+ new PublishPress\Revisions\PostPreview();
+ }
+ }
+
+ add_filter('presspermit_is_preview', [$this, 'fltIsPreview']);
+ add_filter('presspermit_query_post_statuses', [$this, 'fltQueryPostStatuses'], 10, 2);
+
+ add_filter('map_meta_cap', [$this, 'fltStatusChangeCap'], 5, 4);
+
+ if ( ! is_content_administrator_rvy() ) {
+ add_filter( 'map_meta_cap', array($this, 'flt_post_map_meta_cap'), 5, 4);
+ add_filter( 'user_has_cap', array( $this, 'flt_user_has_cap' ), 98, 3 );
+
+ add_filter( 'map_meta_cap', array( $this, 'flt_limit_others_drafts' ), 10, 4 );
+ }
+
+ if ( is_admin() ) {
+ require_once( dirname(__FILE__).'/admin/admin_rvy.php');
+ new RevisionaryAdmin();
+ }
+
+ add_action( 'wpmu_new_blog', array( $this, 'act_new_blog'), 10, 2 );
+
+ add_action('trashed_post', [$this, 'actTrashedPost']);
+
+ add_action( 'deleted_post', [$this, 'actDeletedPost']);
+
+ if ( rvy_get_option('scheduled_revisions') ) {
+ // users who have edit_published capability for post/page can create a scheduled revision by modifying post date to a future date (without setting "future" status explicitly)
+ add_filter( 'wp_insert_post_data', array($this, 'flt_insert_post_data'), 99, 2 );
+ }
+
+ add_filter( 'wp_insert_post_data', array($this, 'flt_regulate_revision_status'), 100, 2 );
+
+ add_filter('wp_insert_post_data', [$this, 'fltRemoveInvalidPostDataKeys'], 999, 2);
+
+ // REST logging
+ add_filter( 'rest_pre_dispatch', array( $this, 'rest_pre_dispatch' ), 10, 3 );
+
+ // This is needed, implemented for pending revisions only
+ if (!empty($_REQUEST['get_new_revision'])) {
+ add_action('admin_enqueue_scripts', array($this, 'act_new_revision_redirect'));
+ }
+
+ if (!empty($_REQUEST['edit_new_revision'])) {
+ add_action('admin_enqueue_scripts', array($this, 'act_edit_revision_redirect'));
+ }
+
+ add_filter('get_comments_number', array($this, 'flt_get_comments_number'), 10, 2);
+
+ add_action('save_post', array($this, 'actSavePost'), 20, 2);
+ add_action('delete_post', [$this, 'actDeletePost'], 10, 3);
+
+ add_action('post_updated', [$this, 'actUpdateRevision'], 10, 2);
+ add_action('post_updated', [$this, 'actUpdateRevisionFixCommentCount'], 999, 2);
+
+ add_filter('posts_clauses', [$this, 'fltPostsClauses'], 10, 2);
+
+ if (!is_admin()) {
+ add_action('admin_bar_menu', [$this, 'adminToolbarItem'], 100);
+ }
+
+ add_filter('wp_dropdown_pages', [$this, 'fltDropdownPages'], 10, 3);
+
+ if (defined('REVISIONARY_RVY_INIT_ACTION')) {
+ do_action( 'rvy_init', $this );
+ }
+ }
+
+ // Work around unfilterable get_pages() query by replacing the wp_dropdown_pages() return array
+ function fltDropdownPages($output, $parsed_args, $pages) {
+ // ---- Begin PublishPress Modification ---
+ global $wpdb;
+
+ // don't recursively execute this filter
+ remove_filter('wp_dropdown_pages', [$this, 'fltDropdownPages'], 10, 3);
+
+ $parsed_args['echo'] = 0;
+
+ $revision_status_csv = implode("','", array_map('sanitize_key', rvy_revision_statuses()));
+ $parsed_args['exclude'] = $wpdb->get_col("SELECT ID FROM $wpdb->posts WHERE post_mime_type IN ('$revision_status_csv') AND post_type !=''");
+ // ---- End PublishPress Modification ---
+
+ $pages = get_pages( $parsed_args );
+ $output = '';
+ // Back-compat with old system where both id and name were based on $name argument.
+ if ( empty( $parsed_args['id'] ) ) {
+ $parsed_args['id'] = $parsed_args['name'];
+ }
+
+ if ( ! empty( $pages ) ) {
+ $class = '';
+ if ( ! empty( $parsed_args['class'] ) ) {
+ $class = " class='" . esc_attr( $parsed_args['class'] ) . "'";
+ }
+
+ $output = "\n";
+ if ( $parsed_args['show_option_no_change'] ) {
+ $output .= "\t" . $parsed_args['show_option_no_change'] . " \n";
+ }
+ if ( $parsed_args['show_option_none'] ) {
+ $output .= "\t' . $parsed_args['show_option_none'] . " \n";
+ }
+ $output .= walk_page_dropdown_tree( $pages, $parsed_args['depth'], $parsed_args );
+ $output .= " \n";
+ }
+
+ /**
+ * Filters the HTML output of a list of pages as a drop down.
+ *
+ * @since 2.1.0
+ * @since 4.4.0 `$parsed_args` and `$pages` added as arguments.
+ *
+ * @param string $output HTML output for drop down list of pages.
+ * @param array $parsed_args The parsed arguments array. See wp_dropdown_pages()
+ * for information on accepted arguments.
+ * @param WP_Post[] $pages Array of the page objects.
+ */
+ $html = apply_filters( 'wp_dropdown_pages', $output, $parsed_args, $pages );
+
+ if ( $parsed_args['echo'] ) {
+ echo $html;
+ }
+
+ // PublishPress: restore this filter hook
+ add_filter('wp_dropdown_pages', [$this, 'fltDropdownPages'], 10, 3);
+
+ return $html;
+ }
+
+ function adminToolbarItem($admin_bar) {
+ global $post;
+
+ if (!empty($post) && rvy_get_option('pending_revisions') && !rvy_in_revision_workflow($post) && ('revision' != $post->post_type) && rvy_is_supported_post_type($post->post_type)) {
+ $status_obj = get_post_status_object($post->post_status);
+
+ if (!empty($status_obj->public) || !empty($status_obj->private) || rvy_get_option('pending_revision_unpublished')) {
+ if ($type_obj = get_post_type_object($post->post_type)) {
+
+ if (current_user_can('copy_post', $post->ID) && rvy_post_revision_supported($post)) {
+ $admin_bar->add_menu([
+ 'id' => 'rvy-create-revision',
+ 'title' => pp_revisions_status_label('draft-revision', 'submit_short'), // Your menu title
+ 'href' => rvy_admin_url("admin.php?page=rvy-revisions&post={$post->ID}&action=revise&front=1"), // URL
+ 'meta' => [
+ 'target' => '_blank',
+ ]
+ ]
+ );
+ }
+ }
+ }
+ }
+ }
+
+ function configurationLateInit() {
+ $this->setPostTypes();
+ $this->setPostTypesArchive();
+ $this->config_loaded = true;
+ }
+
+ public function fltPostsClauses($clauses, $_wp_query, $args = []) {
+ global $wpdb, $revisionary;
+
+ $defaults = [
+ 'is_revisions_query' => false,
+ 'post_types' => [],
+ 'source_alias' => false,
+ ];
+ $args = array_merge($defaults, (array)$args);
+ foreach (array_keys($defaults) as $var) {
+ $$var = $args[$var];
+ }
+
+ if ($is_revisions_query || !empty($_wp_query->is_revisions_query) || !empty($_wp_query->query['is_revisions_query']) || (!empty($revisionary) && !empty($revisionary->is_revisions_query)) || $_wp_query->is_preview) {
+ return $clauses;
+ }
+
+ // Allow revision retrieval by front end editors / previews
+ if (!is_admin() && (!defined('REST_REQUEST') || ! REST_REQUEST)) {
+ return $clauses;
+ }
+
+ if (empty($clauses['where'])) {
+ $clauses['where'] = '1=1';
+ }
+
+ $src_table = ($source_alias) ? $source_alias : $wpdb->posts;
+ $args['src_table'] = $src_table;
+
+ $revision_status_csv = implode("','", array_map('sanitize_key', rvy_revision_statuses()));
+ $clauses['where'] .= " AND $src_table.post_mime_type NOT IN ('$revision_status_csv')";
+
+ return $clauses;
+ }
+
+ // This is intentionally called twice: once for code that fires on 'init' and then very late on 'init' for types which were registered late on 'init'
+ public function setPostTypes() {
+ $enabled_post_types = get_option('rvy_enabled_post_types', false);
+
+ if (false === $enabled_post_types) {
+ $enabled_post_types = array_fill_keys(
+ get_post_types(['public' => true]), true
+ );
+
+ if (class_exists('WooCommerce')) {
+ $enabled_post_types['product'] = true;
+ $enabled_post_types['order'] = true;
+ }
+
+ if (class_exists('Tribe__Events__Main')) {
+ $enabled_post_types['tribe_events'] = true;
+ }
+
+ if (!defined('REVISIONARY_NO_PRIVATE_TYPES')) {
+ $private_types = array_merge(
+ get_post_types(['public' => false], 'object'),
+ get_post_types(['public' => null], 'object')
+ );
+
+ // by default, enable non-public post types that have type-specific capabilities defined
+ foreach($private_types as $post_type => $type_obj) {
+ if ((!empty($type_obj->cap) && !empty($type_obj->cap->edit_posts) && !in_array($type_obj->cap->edit_posts, ['edit_posts', 'edit_pages']))
+ || defined('REVISIONARY_ENABLE_' . strtoupper($post_type) . '_TYPE')
+ ) {
+ $enabled_post_types[$post_type] = true;
+ }
+ }
+ }
+ }
+
+ $enabled_post_types = apply_filters(
+ 'revisionary_enabled_post_types',
+ array_diff_key(
+ $enabled_post_types,
+ ['attachment' => true, 'tablepress_table' => true, 'acf-field-group' => true, 'acf-field' => true, 'nav_menu_item' => true, 'custom_css' => true, 'customize_changeset' => true, 'wp_block' => true, 'wp_template' => true, 'wp_template_part' => true, 'wp_global_styles' => true, 'wp_navigation' => true]
+ )
+ );
+
+ $this->enabled_post_types = array_merge($this->enabled_post_types, $enabled_post_types);
+
+ unset($this->enabled_post_types['attachment']);
+ $this->enabled_post_types = array_filter($this->enabled_post_types);
+ }
+
+ public function setPostTypesArchive() {
+ global $current_user;
+
+ $enabled_post_types_archive = get_option('rvy_enabled_post_types_archive', false);
+
+ if (false === $enabled_post_types_archive) {
+ $types = get_post_types(['public' => true]);
+
+ $enabled_post_types_archive = array_fill_keys(
+ $types, true
+ );
+
+ if (!defined('REVISIONARY_NO_PRIVATE_TYPES')) {
+ $private_types = array_merge(
+ get_post_types(['public' => false], 'object'),
+ get_post_types(['public' => null], 'object')
+ );
+
+ // by default, enable non-public post types that have type-specific capabilities defined
+ foreach($private_types as $post_type => $type_obj) {
+ if ((!empty($type_obj->cap) && !empty($type_obj->cap->edit_posts) && !in_array($type_obj->cap->edit_posts, ['edit_posts', 'edit_pages']))
+ || defined('REVISIONARY_ENABLE_' . strtoupper($post_type) . '_TYPE')
+ ) {
+ $enabled_post_types_archive[$post_type] = true;
+ }
+ }
+ }
+
+ foreach (array_keys($enabled_post_types_archive) as $post_type) {
+ if (!post_type_supports($post_type, 'revisions')) {
+ unset($enabled_post_types_archive[$post_type]);
+ }
+ }
+
+ if (class_exists('WooCommerce')) {
+ $enabled_post_types_archive['product'] = true;
+ $enabled_post_types_archive['order'] = true;
+ }
+
+ if (class_exists('Tribe__Events__Main')) {
+ $enabled_post_types_archive['tribe_events'] = true;
+ }
+ }
+
+ $enabled_post_types_archive = array_diff_key(
+ $enabled_post_types_archive,
+ [
+ 'attachment' => true,
+ 'tablepress_table' => true,
+ 'acf-field-group' => true,
+ 'acf-field' => true,
+ 'nav_menu_item' => true,
+ 'custom_css' => true,
+ 'customize_changeset' => true,
+ 'wp_block' => true,
+ 'wp_template' => true,
+ 'wp_template_part' => true,
+ 'wp_global_styles' => true,
+ 'wp_navigation' => true,
+ 'product_variation' => true,
+ 'shop_order_refund' => true
+ ]
+ );
+
+ // Remove the post_types that doesn't have a valid object (null)
+ foreach( array_keys( $enabled_post_types_archive ) as $type ) :
+ $type_obj = get_post_type_object( $type );
+ if( ! $type_obj ) :
+ unset( $enabled_post_types_archive[$type] );
+ endif;
+
+ if (
+ (!empty($type_obj->cap->edit_others_posts) && empty($current_user->allcaps[$type_obj->cap->edit_others_posts]))
+ || (!empty($type_obj->cap->edit_published_posts) && empty($current_user->allcaps[$type_obj->cap->edit_published_posts]))
+ ) {
+ unset($enabled_post_types_archive[$type]);
+ }
+ endforeach;
+
+ $this->enabled_post_types_archive = array_merge(
+ $this->enabled_post_types_archive,
+ $enabled_post_types_archive
+ );
+
+ $this->enabled_post_types_archive = apply_filters(
+ 'revisionary_archive_post_types',
+ array_filter($this->enabled_post_types_archive)
+ );
+ }
+
+ function canEditPost($post, $args = []) {
+ global $current_user;
+
+ $args = (array) $args;
+
+ if (is_numeric($post)) {
+ $post = get_post($post);
+ }
+
+ if (!is_object($post)
+ || empty($post->ID)
+ || !$type_obj = get_post_type_object($post->post_type)
+ || !$status_obj = get_post_status_object($post->post_status)
+ ) {
+ return false;
+ }
+
+ if (!empty($args['simple_cap_check']) && (!empty($status_obj->public) || !empty($status_obj->private))) {
+ return isset($type_obj->cap->edit_published_posts) && !empty($current_user->allcaps[$type_obj->cap->edit_published_posts]);
+ } else {
+ static $last_result;
+
+ if (!isset($last_result)) {
+ $last_result = [];
+
+ } elseif (!empty($last_result) && isset($last_result[$post->ID])) {
+ return $last_result[$post->ID];
+ }
+
+ $caps = map_meta_cap('edit_post', $current_user->ID, $post->ID);
+
+ $return = true;
+
+ foreach($caps as $cap) {
+ if (empty($current_user->allcaps[$cap])) {
+ $return = false;
+ break;
+ }
+ }
+
+ $last_result[$post->ID] = $return;
+ return $return;
+ }
+ }
+
+ function actTrashedPost($revision_id) {
+ if (rvy_in_revision_workflow($revision_id, ['include_trash' => true])) {
+ $post_id = rvy_post_id($revision_id);
+
+ if (!$triggered_deletions = get_option('_rvy_trigger_deletion')) {
+ $triggered_deletions = [];
+ }
+
+ $triggered_deletions[$revision_id] = $post_id;
+
+ update_option('_rvy_trigger_deletion', $triggered_deletions);
+ }
+ }
+
+ // On post deletion, clear corresponding _rvy_has_revisions postmeta flag
+ function actDeletedPost($post_id) {
+ delete_post_meta($post_id, '_rvy_has_revisions');
+ }
+
+ function actSavePost($post_id, $post) {
+ if (strtotime($post->post_date_gmt) > agp_time_gmt()) {
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+
+ if (rvy_get_option('revision_publish_cron')) {
+ if (rvy_in_revision_workflow($post_id) && ('future-revision' == $post->post_mime_type)) {
+ rvy_update_next_publish_date(['revision_id' => $post_id]);
+ }
+ } else {
+ rvy_update_next_publish_date();
+ }
+ }
+ }
+
+ // Immediately prior to post deletion, also delete its pending revisions and future revisions (and their meta data)
+ function actDeletePost($post_id) {
+ global $wpdb;
+
+ if (!$post_id) {
+ return;
+ }
+
+ $revision_status_csv = implode("','", array_map('sanitize_key', rvy_revision_statuses()));
+
+ $any_trashed_posts = $wpdb->get_var("SELECT ID FROM $wpdb->posts WHERE post_status = 'trash' AND comment_count > 0 AND post_mime_type IN ('$revision_status_csv') LIMIT 1");
+
+ $trashed_clause = ($any_trashed_posts)
+ ? $wpdb->prepare(
+ " OR (ID IN (SELECT post_id FROM $wpdb->postmeta WHERE meta_key = '_rvy_base_post_id' AND meta_value = %d) AND post_status = 'trash')",
+ $post_id
+ ) : '';
+
+ $post_ids = $wpdb->get_col(
+ $wpdb->prepare(
+ "SELECT ID FROM $wpdb->posts WHERE (post_mime_type IN ('$revision_status_csv') AND comment_count = %d) $trashed_clause",
+ $post_id
+ )
+ );
+
+ foreach($post_ids as $revision_id) {
+ wp_delete_post($revision_id, true);
+ }
+
+ revisionary_refresh_revision_flags($post_id, ['ignore_revision_ids' => $post_ids]);
+
+ $post = get_post($post_id);
+
+ if ($post && rvy_in_revision_workflow($post)) {
+ $wpdb->query(
+ $wpdb->prepare(
+ "DELETE FROM $wpdb->postmeta WHERE post_id = %d",
+ $post_id
+ )
+ );
+
+ $meta_args = ['ignore_revisions' => [$post->ID]];
+
+ if (rvy_get_option('revision_limit_per_post')) {
+ delete_post_meta(rvy_post_id($post->ID), '_rvy_has_revisions');
+ $meta_args['insert_only'] = true;
+ }
+
+ revisionary_refresh_postmeta(rvy_post_id($post->ID), $meta_args);
+ }
+ }
+
+ function actUpdateRevision($post_id, $revision) {
+ if (rvy_in_revision_workflow($revision)
+ && (rvy_get_option('revision_update_notifications'))
+ ) {
+ $published_post = get_post(rvy_post_id($revision));
+
+ if (apply_filters('revisionary_do_revision_notice', !$this->doing_rest, $revision, $published_post)) {
+ if (('future-revision' != $revision->post_mime_type) && rvy_get_option('revision_update_notifications')) {
+ $args = ['update' => true, 'revision_id' => $revision->ID, 'published_post' => $published_post, 'object_type' => $published_post->post_type];
+
+ if ( !empty( $_REQUEST['prev_cc_user'] ) ) {
+ $args['selected_recipients'] = array_map('intval', $_REQUEST['prev_cc_user']);
+ } else {
+ // If the UI that triggered this notification does not support recipient selection, send to default recipients for this post
+ require_once( dirname(__FILE__) . '/revision-workflow_rvy.php' );
+ $result = Rvy_Revision_Workflow_UI::default_notification_recipients($published_post->ID, ['object_type' => $published_post->post_type]);
+ $args['selected_recipients'] = array_keys(array_filter($result['default_ids']));
+ }
+
+ $this->do_notifications('pending-revision', 'pending-revision', (array) $revision, $args);
+ }
+ }
+ }
+ }
+
+ function actUpdateRevisionFixCommentCount($post_id, $revision) {
+ global $wpdb;
+
+ if (rvy_in_revision_workflow($revision)) {
+ if (empty($revision->comment_count)) {
+ if ($main_post_id = get_post_meta($revision->ID, '_rvy_base_post_id', true)) {
+ $wpdb->update($wpdb->posts, ['comment_count' => $main_post_id], ['ID' => $revision->ID]);
+ }
+ }
+ }
+ }
+
+ // Return zero value for revision comments because:
+ // * comments are not supported for revisions
+ // * published post ID is stored to comment_count column is used for query efficiency
+ function flt_get_comments_number($count, $post_id) {
+ if ($post = get_post($post_id)) {
+ if (rvy_in_revision_workflow($post)) {
+ $count = 0;
+ }
+ }
+
+ return $count;
+ }
+
+ function get_last_revision($post_id, $user_id) {
+ require_once(dirname(__FILE__).'/admin/admin-init_rvy.php');
+
+ if ( $revisions = rvy_get_post_revisions( $post_id, '', array( 'order' => 'DESC', 'orderby' => 'ID' ) ) ) { // @todo: retrieve revision_id in block editor js, pass as redirect arg
+ foreach( $revisions as $revision ) {
+ if (rvy_is_post_author($revision, $user_id)) {
+ return $revision;
+ }
+ }
+ }
+
+ return false;
+ }
+
+ function act_new_revision_redirect() {
+ global $current_user, $post;
+
+ if (empty($_REQUEST['get_new_revision'])) {
+ return;
+ }
+
+ $published_post_id = (int) $_REQUEST['get_new_revision'];
+
+ $published_url = get_permalink($published_post_id);
+
+ $revision = $this->get_last_revision($published_post_id, $current_user->ID);
+
+ if ($revision) {
+ $args = [];
+ if (!empty($_REQUEST['nc'])) { // with a specified link target, avoid multiple browser tabs in the same editor instance
+ $args['nc'] = sanitize_key($_REQUEST['nc']);
+ }
+
+ $type_obj = get_post_type_object(get_post_field('post_type', $published_post_id));
+
+ $preview_link = (!empty($type_obj) && !empty($type_obj->public)) ? rvy_preview_url($revision, $args) : admin_url("post.php?post={$published_post_id}&action=edit");
+
+ wp_redirect($preview_link);
+ exit;
+ }
+
+ // If logged user does not have a pending revision of this post, redirect to published permalink
+ wp_redirect($published_url);
+ exit;
+ }
+
+ function act_edit_revision_redirect() {
+ global $current_user, $post;
+
+ if (empty($_REQUEST['edit_new_revision'])) {
+ return;
+ }
+
+ $published_post_id = (!empty($_REQUEST['edit_new_revision'])) ? rvy_post_id($_REQUEST['edit_new_revision']) : rvy_post_id($post->ID);
+ $published_url = get_permalink($published_post_id);
+
+ $revision = $this->get_last_revision($published_post_id, $current_user->ID);
+
+ if ($revision) {
+ $args = [];
+ if (!empty($_REQUEST['nc'])) { // with a specified link target, avoid multiple browser tabs in the same editor instance
+ $args['nc'] = sanitize_key($_REQUEST['nc']);
+ }
+
+ $edit_link = admin_url("post.php?post={$revision->ID}&action=edit&nc={$args['nc']}");
+ wp_redirect($edit_link);
+ exit;
+ }
+
+ // If logged user does not have a pending revision of this post, redirect to published permalink
+ wp_redirect($published_url);
+ exit;
+ }
+
+ // log post type and ID from REST handler for reference by subsequent PP filters
+ function rest_pre_dispatch( $rest_response, $rest_server, $request ) {
+ $this->doing_rest = true;
+
+ require_once( dirname(__FILE__).'/rest_rvy.php' );
+ $this->rest = new Revisionary_REST();
+
+ return $this->rest->pre_dispatch( $rest_response, $rest_server, $request );
+ }
+
+ // prevent revisors from editing other users' regular drafts and pending posts
+ function flt_limit_others_drafts( $caps, $meta_cap, $user_id, $args ) {
+ global $current_user;
+
+ if (!empty($this->skip_filtering)) {
+ return $caps;
+ }
+
+ if ( ! in_array( $meta_cap, array( 'edit_post', 'edit_page' ) ) )
+ return $caps;
+
+ $object_id = ( is_array($args) && ! empty($args[0]) ) ? (int) $args[0] : $args;
+
+ if ( ! $object_id || ! is_scalar($object_id) || ( $object_id < 0 ) || ! rvy_get_option('require_edit_others_drafts') ) {
+ return $caps;
+ }
+
+ if ( $post = get_post( $object_id ) ) {
+ if ( ('revision' != $post->post_type) && ! rvy_in_revision_workflow($post) ) {
+
+ if (empty($this->enabled_post_types[$post->post_type])
+ || !apply_filters('revisionary_require_edit_others_drafts', true, $post->post_type, $post->post_status, $args)) {
+ return $caps;
+ }
+
+ $status_obj = get_post_status_object( $post->post_status );
+
+ if (!rvy_is_post_author($post) && $status_obj && ! $status_obj->public && ! $status_obj->private) {
+
+ $post_type_obj = get_post_type_object( $post->post_type );
+ if (isset($post_type_obj->cap->edit_published_posts) && current_user_can( $post_type_obj->cap->edit_published_posts)) { // don't require any additional caps for sitewide Editors
+ return $caps;
+ }
+
+ static $stati;
+
+ if ( ! isset($stati) ) {
+ $stati = get_post_stati( array( 'internal' => false, 'protected' => true ) );
+ $stati = array_diff( $stati, array( 'future' ) );
+ }
+
+ if ( in_array( $post->post_status, $stati ) ) { // isset check because doing_cap_check property was undefined prior to Permissions 3.3.8
+ if ((!function_exists('presspermit') || (isset(presspermit()->doing_cap_check) && !presspermit()->doing_cap_check)) && $post_type_obj) {
+ if (!empty($post_type_obj->cap->edit_others_posts)) {
+ $caps[] = str_replace('edit_', 'list_', $post_type_obj->cap->edit_others_posts);
+ }
+ }
+
+ if (empty($current_user->allcaps['edit_others_drafts'])) {
+ $caps[] = "edit_others_drafts";
+ }
+ }
+ }
+ }
+ }
+
+ return $caps;
+ }
+
+ function fltStatusChangeCap($caps, $cap, $user_id, $args) {
+ global $current_user;
+
+ if ('copy_post' == $cap) {
+ if (!rvy_get_option('pending_revisions')) {
+ return array_diff_key($caps, [$cap => true]);
+ }
+
+ if (!empty($args[0])) {
+ $post_id = (is_object($args[0])) ? $args[0]->ID : (int) $args[0];
+ } else {
+ $post_id = 0;
+ }
+
+ if (rvy_in_revision_workflow($post_id)) {
+ return array_diff_key($caps, [$cap => true]);
+ }
+
+ $filter_args = [];
+
+ if (!$can_copy = rvy_is_full_editor($post_id)) {
+ if ($_post = get_post($post_id)) {
+ $type_obj = get_post_type_object($_post->post_type);
+ }
+
+ if (!empty($type_obj)) {
+ if (rvy_get_option("copy_posts_capability")) {
+ $base_prop = (rvy_is_post_author($post_id)) ? 'edit_posts' : 'edit_others_posts';
+ $copy_cap_name = str_replace('edit_', 'copy_', $type_obj->cap->$base_prop);
+
+ if (false === strpos($copy_cap_name, 'copy_')) {
+ if ('page' == $_post->post_type) {
+ $copy_cap_name = (rvy_is_post_author($post_id)) ? 'copy_pages' : 'copy_others_pages';
+ } else {
+ $copy_cap_name = (rvy_is_post_author($post_id)) ? 'copy_posts' : 'copy_others_posts';
+ }
+ }
+
+ $can_copy = current_user_can($copy_cap_name);
+ } else {
+ $can_copy = current_user_can($type_obj->cap->edit_posts);
+ }
+
+ $filter_args = compact('type_obj');
+ }
+ }
+
+ // allow PublishPress Permissions to apply 'copy' exceptions
+ if ($can_copy = apply_filters('revisionary_can_copy', $can_copy, $post_id, 'draft', 'draft-revision', $filter_args)
+ || apply_filters('revisionary_can_submit', $can_copy, $post_id, 'pending', 'pending-revision', $filter_args)
+ ) {
+ $caps = ['read'];
+ } else {
+ $caps = array_diff_key($caps, [$cap => true]);
+ }
+
+ } elseif ('set_revision_pending-revision' == $cap) {
+ if (!rvy_get_option('pending_revisions')) {
+ return array_diff_key($caps, [$cap => true]);
+ }
+
+ if (!empty($args[0])) {
+ $post_id = (is_object($args[0])) ? $args[0]->ID : (int) $args[0];
+ } else {
+ $post_id = 0;
+ }
+
+ if (!rvy_in_revision_workflow($post_id)) {
+ return $caps;
+ }
+
+ $filter_args = [];
+
+ if ($can_submit = current_user_can('edit_post', $post_id)) { // require basic editing capabilties for revision ID
+ $main_post_id = rvy_post_id($post_id);
+
+ if (rvy_get_option("revise_posts_capability") && !rvy_is_full_editor($main_post_id)) { // bypass capability check for those with full editing caps on main post
+ if ($_post = get_post($post_id)) {
+ if ($type_obj = get_post_type_object($_post->post_type)) {
+ $base_prop = (rvy_is_post_author($main_post_id)) ? 'edit_posts' : 'edit_others_posts';
+ $submit_cap_name = str_replace('edit_', 'revise_', $type_obj->cap->$base_prop);
+ $can_submit = current_user_can($submit_cap_name);
+ $filter_args = compact('main_post_id', 'type_obj');
+ }
+ }
+ }
+ }
+
+ // allow PublishPress Permissions to apply 'revise' exceptions
+ if ($can_submit = apply_filters('revisionary_can_submit', $can_submit, $post_id, 'pending', 'pending-revision', $filter_args)) {
+ $caps = ['read'];
+ }
+ }
+
+ return $caps;
+ }
+
+ function set_content_roles( $content_roles_obj ) {
+ $this->content_roles = $content_roles_obj;
+
+ if ( ! defined( 'RVY_CONTENT_ROLES' ) ) {
+ define( 'RVY_CONTENT_ROLES', true );
+ }
+ }
+
+
+ function act_new_blog( $blog_id, $user_id ) {
+ rvy_add_revisor_role( $blog_id );
+ }
+
+
+ function flt_post_map_meta_cap($caps, $cap, $user_id, $args) {
+ global $current_user;
+
+ static $busy;
+
+ if (!empty($busy) || !empty($this->skip_filtering)) {
+ return $caps;
+ }
+
+ if (!in_array($cap, array('read_post', 'read_page', 'edit_post', 'edit_page', 'delete_post', 'delete_page'))) {
+ return $caps;
+ }
+
+ if (!empty($args[0])) {
+ $post_id = (is_object($args[0])) ? $args[0]->ID : (int) $args[0];
+ } else {
+ $post_id = 0;
+ }
+
+ if ($post = get_post($post_id)) {
+ if (('inherit' == $post->post_status)
+ || empty($this->enabled_post_types[$post->post_type]) && $this->config_loaded
+ ) {
+ return $caps;
+ }
+ }
+
+ if ($post && (('future-revision' == $post->post_mime_type) || in_array($cap, ['read_post', 'read_page']))) {
+ if (in_array($cap, ['read_post', 'read_page'])) {
+ return $caps;
+ }
+
+ // allow Revisor to view a preview of their scheduled revision
+ if (is_admin() || (defined('REST_REQUEST') && REST_REQUEST) || empty($_REQUEST['preview']) || !empty($_POST) || did_action('template_redirect')) {
+ if ($type_obj = get_post_type_object( $post->post_type )) {
+ if (isset($type_obj->cap->edit_published_posts)) {
+ $check_cap = in_array($cap, ['delete_post', 'delete_page']) ? $type_obj->cap->delete_published_posts : $type_obj->cap->edit_published_posts;
+ return array_merge($caps, [$check_cap]);
+ } else {
+ return $caps;
+ }
+ }
+ }
+ }
+
+ $busy = true;
+
+ $preview_arg = (defined('RVY_PREVIEW_ARG')) ? sanitize_key(constant('RVY_PREVIEW_ARG')) : 'rv_preview';
+
+ if (in_array($cap, ['read_post', 'read_page']) // WP Query imposes edit_post capability requirement for front end viewing of protected statuses
+ || ((!empty($_REQUEST[$preview_arg]) || !empty($_GET['preview'])) && in_array($cap, array('edit_post', 'edit_page')) && did_action('posts_selection') && !did_action('template_redirect'))
+ ) {
+ if ($post && rvy_in_revision_workflow($post)) {
+ $type_obj = get_post_type_object($post->post_type);
+
+ if ($type_obj && !empty($type_obj->cap->edit_others_posts)) {
+ $caps = array_diff($caps, [$type_obj->cap->edit_others_posts, 'do_not_allow']);
+
+ $check_post = $post;
+
+ if ($post->ID <= 0) {
+ if ($check_id = rvy_detect_post_id()) {
+ $check_post = get_post($check_id);
+ }
+ }
+
+ if (rvy_is_post_author($check_post) || rvy_is_post_author(rvy_post_id($check_post->ID)) || rvy_is_full_editor($post)) {
+ $caps []= 'read';
+
+ } elseif (rvy_get_option('revisor_hide_others_revisions')) {
+ $caps []= 'list_others_revisions';
+
+ } else {
+ $caps []= $type_obj->cap->edit_posts;
+ }
+ }
+
+ $busy = false;
+ return $caps;
+ }
+ } elseif (($post_id > 0) && $post && rvy_in_revision_workflow($post)
+ && rvy_get_option('revisor_lock_others_revisions') && !rvy_is_post_author($post) && !rvy_is_full_editor(rvy_post_id($post->ID))
+ ) {
+ if ($type_obj = get_post_type_object( $post->post_type )) {
+ if (in_array($type_obj->cap->edit_others_posts, $caps)) {
+ if ((!empty($type_obj->cap->edit_others_posts) && empty($current_user->allcaps[$type_obj->cap->edit_others_posts]))
+ || (!empty($type_obj->cap->edit_published_posts) && empty($current_user->allcaps[$type_obj->cap->edit_published_posts]))
+ ) {
+ if (!current_user_can('edit_post', rvy_post_id($post_id))) {
+ if (!empty($current_user->allcaps['edit_others_revisions'])) {
+ $caps[] = 'edit_others_revisions';
+ } else {
+ $caps []= 'do_not_allow'; // @todo: implement this within user_has_cap filters?
+ }
+ }
+ }
+ }
+ }
+ }
+
+ if (in_array($cap, array('edit_post', 'edit_page'))) {
+ if ($post && !empty($post->post_status)) {
+ if (!in_array($post->post_status, rvy_filtered_statuses())) {
+ $busy = false;
+ return $caps;
+ }
+ }
+
+ $busy = true;
+
+ // Run reqd_caps array through the filter which is normally used to implicitly grant edit_published cap to Revisors
+ // Applying this adjustment to reqd_caps instead of user caps on 'edit_post' checks allows for better compat with PressPermit and other plugins
+ if ($grant_caps = $this->filter_caps(array(), $caps, array(0 => $cap, 1 => $user_id, 2 => $post_id))) {
+ $caps = array_diff($caps, array_keys(array_filter($grant_caps)));
+
+ if (!$caps) {
+ if ($type_obj = get_post_type_object( $post->post_type )) {
+ $caps = [$type_obj->cap->edit_posts];
+ }
+ }
+ }
+ }
+
+ $busy = false;
+ return $caps;
+ }
+
+ function flt_user_has_cap($wp_blogcaps, $reqd_caps, $args) {
+ return $this->filter_caps($wp_blogcaps, $reqd_caps, $args);
+ }
+
+ private function filter_caps($wp_blogcaps, $reqd_caps, $args, $internal_args = array()) {
+ global $current_user;
+
+ if (!empty($this->skip_filtering) || !rvy_get_option('pending_revisions')) {
+ return $wp_blogcaps;
+ }
+
+ $post_id = ( ! empty($args[2]) ) ? $args[2] : rvy_detect_post_id();
+
+ if (!$post = get_post($post_id)) {
+ if (($post_id == -1) && defined('PRESSPERMIT_PRO_VERSION') && !empty(presspermit()->meta_cap_post)) { // wp_cache_add(-1) does not work for map_meta_cap call on get-revision-diffs ajax call
+ $post = presspermit()->meta_cap_post;
+ }
+ }
+
+ if (empty($post) || (empty($this->enabled_post_types[$post->post_type]) && $this->config_loaded)) {
+ return $wp_blogcaps;
+ }
+
+ if (rvy_in_revision_workflow($post)) {
+ $object_type_obj = get_post_type_object($post->post_type);
+
+ if (('draft-revision' == $post->post_mime_type) && !rvy_is_post_author($post) && rvy_get_option('manage_unsubmitted_capability') && empty($wp_blogcaps['manage_unsubmitted_revisions'])) {
+ unset($wp_blogcaps[$object_type_obj->cap->edit_others_posts]);
+ } else {
+ if (defined('DOING_AJAX') && DOING_AJAX && !empty($_REQUEST['action']) && (false !== strpos(sanitize_key($_REQUEST['action']), 'query-attachments'))) {
+ if ('post' == $post->post_type) {
+ return $wp_blogcaps;
+ }
+ }
+
+ // If edit_others capability is being required for this post type, apply edit_others_revisions capability
+ if (!empty($object_type_obj->cap) && in_array($object_type_obj->cap->edit_others_posts, $reqd_caps)) {
+ if (!empty($current_user->allcaps['edit_others_revisions']) || !rvy_get_option('revisor_lock_others_revisions')) {
+ $wp_blogcaps[$object_type_obj->cap->edit_others_posts] = true;
+
+ } elseif (rvy_get_option('admin_revisions_to_own_posts') && current_user_can('edit_post', rvy_post_id($post_id))) {
+ $wp_blogcaps[$object_type_obj->cap->edit_others_posts] = true;
+ }
+ }
+
+ // Grant edit permission for revision if user can edit main post
+ if (!empty($args[0]) && ('edit_post' == $args[0]) && array_diff($reqd_caps, array_keys(array_filter($wp_blogcaps)))) {
+ $this->skip_filtering = true;
+
+ if (rvy_get_option('admin_revisions_to_own_posts') && current_user_can('edit_post', rvy_post_id($post_id))) {
+ $wp_blogcaps = array_merge($wp_blogcaps, array_fill_keys($reqd_caps, true));
+ }
+
+ $this->skip_filtering = false;
+ }
+ }
+ }
+
+ return $wp_blogcaps;
+ }
+
+ function fltRemoveInvalidPostDataKeys($data, $postarr) {
+ unset($data['filter']);
+ return $data;
+ }
+
+ // If Scheduled Revisions are enabled, don't allow WP to force current post status to future based on publish date
+ function flt_insert_post_data( $data, $postarr ) {
+ if ( ( 'future' == $data['post_status'] ) && ( rvy_is_status_published( $postarr['post_status'] ) ) ) {
+
+ if (!empty($postarr['post_type']) && empty($this->enabled_post_types[$postarr['post_type']])) {
+ return $data;
+ }
+
+ // don't interfere with scheduling of unpublished drafts
+ if ( $stored_status = get_post_field ( 'post_status', rvy_detect_post_id() ) ) {
+ if ( rvy_is_status_published($stored_status) && !rvy_is_status_published($data['post_status']) && ('future' != $data['post_status']) ) {
+ $data['post_status'] = $postarr['post_status'];
+ }
+ }
+ }
+
+ // If this is already a scheduled revision and the date is being modified, update the WP-Cron entry
+ if (rvy_in_revision_workflow($postarr['ID']) && ('future-revision' == $postarr['post_mime_type'])) {
+
+ $current_post_date_gmt = get_post_field('post_date_gmt', $postarr['ID']);
+
+ if ($data['post_date_gmt'] != $current_post_date_gmt) {
+ wp_unschedule_event(strtotime($current_post_date_gmt), 'publish_revision_rvy', [$postarr['ID']]);
+
+ wp_schedule_single_event(strtotime($data['post_date_gmt']), 'publish_revision_rvy', [$postarr['ID']]);
+ }
+ }
+
+ return $data;
+ }
+
+ // @todo: confirm this is still needed
+ function flt_regulate_revision_status($data, $postarr) {
+ // Revisions are not published by wp_update_post() execution; Prevent setting to a non-revision status
+ if (rvy_get_post_meta($postarr['ID'], '_rvy_base_post_id', true) && ('trash' != $data['post_status'])) {
+ if (!$revision = get_post($postarr['ID'])) {
+ return $data;
+ }
+
+ if (empty($this->enabled_post_types[$revision->post_type])) {
+ return $data;
+ }
+
+ if (!rvy_is_revision_status($postarr['post_mime_type']) || !in_array($postarr['post_status'], rvy_revision_base_statuses())) {
+ $revert_status = true;
+
+ } elseif ($revision) {
+ if (($data['post_mime_type'] != $revision->post_mime_type) || ($data['post_status'] != $revision->post_status)
+ && (('future-revision' == $revision->post_mime_type) || ('future-revision' == $postarr['post_mime_type']))
+ ) {
+ $revert_status = true;
+ }
+ }
+
+ if (!empty($revert_status) && rvy_in_revision_workflow($revision)) {
+ $data['post_status'] = $revision->post_status;
+ $data['post_mime_type'] = $revision->post_mime_type;
+ }
+ }
+
+ return $data;
+ }
+
+ function do_notifications( $notification_type, $status, $post_arr, $args ) {
+ global $rvy_workflow_ui;
+ if ( ! isset( $rvy_workflow_ui ) ) {
+ require_once( dirname(__FILE__).'/revision-workflow_rvy.php' );
+ $rvy_workflow_ui = new Rvy_Revision_Workflow_UI();
+ }
+
+ return $rvy_workflow_ui->do_notifications( $notification_type, $status, $post_arr, $args );
+ }
+
+ // Prevent wp_update_comment_count_now() from modifying Pending Revision comment_count field (main post ID)
+ function fltUpdateCommentCountBypass($count, $old, $post_id) {
+ if (rvy_in_revision_workflow($post_id)) {
+ return rvy_post_id($post_id);
+ }
+
+ return $count;
+ }
+
+ function fltIsPreview($is_preview) {
+ if (defined('RVY_PREVIEW_ARG') && RVY_PREVIEW_ARG && !empty($_REQUEST[RVY_PREVIEW_ARG])) {
+ $is_preview = true;
+ }
+
+ return $is_preview;
+ }
+
+ /*
+ * PublishPress Permissions: Make query filtering allow for revision previews
+ */
+ function fltQueryPostStatuses($statuses, $args) {
+ if (((defined('RVY_PREVIEW_ARG') && RVY_PREVIEW_ARG && !empty($_REQUEST[RVY_PREVIEW_ARG])))
+ && !empty($args['required_operation']) && ('edit' == $args['required_operation']) && function_exists('rvy_revision_base_statuses')) {
+ $statuses = array_merge($statuses, array_fill_keys(rvy_revision_base_statuses(), (object)[]));
+ }
+
+ return $statuses;
+ }
+
+ static function applyRevisionLimit($post) {
+ if (!is_object($post) || empty($post->ID)) {
+ return;
+ }
+
+ $post_id = $post->ID;
+
+ /*
+ * If a limit for the number of revisions to keep has been set,
+ * delete the oldest ones.
+ */
+ $revisions_to_keep = wp_revisions_to_keep( $post );
+
+ if ( $revisions_to_keep < 0 ) {
+ return;
+ }
+
+ $revisions = wp_get_post_revisions( $post_id, array( 'order' => 'ASC' ) );
+
+ /**
+ * Filters the revisions to be considered for deletion.
+ *
+ * @since 6.2.0
+ *
+ * @param WP_Post[] $revisions Array of revisions, or an empty array if none.
+ * @param int $post_id The ID of the post to save as a revision.
+ */
+ $revisions = apply_filters(
+ 'wp_save_post_revision_revisions_before_deletion',
+ $revisions,
+ $post_id
+ );
+
+ $delete = count( $revisions ) - $revisions_to_keep;
+
+ if ( $delete < 1 ) {
+ return;
+ }
+
+ $revisions = array_slice( $revisions, 0, $delete );
+
+ for ( $i = 0; isset( $revisions[ $i ] ); $i++ ) {
+ if ( str_contains( $revisions[ $i ]->post_name, 'autosave' ) ) {
+ continue;
+ }
+
+ wp_delete_post_revision( $revisions[ $i ]->ID );
+ }
+ }
+} // end Revisionary class
diff --git a/wp-content/plugins/revisionary/rvy_init-functions.php b/wp-content/plugins/revisionary/rvy_init-functions.php
new file mode 100644
index 000000000..0e78dcb8e
--- /dev/null
+++ b/wp-content/plugins/revisionary/rvy_init-functions.php
@@ -0,0 +1,1532 @@
+ $revision_id]);
+}
+
+function _rvy_rest_prepare($response, $post, $request) {
+ if (!rvy_in_revision_workflow($post)) {
+ return $response;
+ }
+
+ if ($type_obj = get_post_type_object($post->post_type)) {
+ $rest_base = ! empty( $type_obj->rest_base ) ? $type_obj->rest_base : $type_obj->name;
+ $namespace = ! empty( $type_obj->rest_namespace ) ? $type_obj->rest_namespace : 'wp/v2';
+
+ $base = sprintf( '%s/%s', $namespace, $rest_base );
+ $href = rest_url( trailingslashit( $base ) . $post->ID );
+
+ $response->add_link('https://api.w.org/action-publish', $href);
+ }
+
+ return $response;
+}
+
+function rvy_mail_check_buffer($new_msg = [], $args = []) {
+ if (empty($args['log_only'])) {
+ if (!$use_buffer = rvy_get_option('use_notification_buffer')) {
+ return (defined('REVISIONARY_DISABLE_MAIL_LOG'))
+ ? array_fill_keys(['buffer', 'sent_mail', 'send_limits', 'sent_counts', 'new_msg_buffered'], [])
+ : [];
+ }
+ }
+
+ require_once( dirname(__FILE__).'/mail-buffer_rvy.php');
+ return _rvy_mail_check_buffer($new_msg, $args);
+}
+
+function rvy_send_buffered_mail() {
+ require_once( dirname(__FILE__).'/mail-buffer_rvy.php');
+ _rvy_send_buffered_mail();
+}
+
+function rvy_set_notification_buffer_cron() {
+ $cron_timestamp = wp_next_scheduled( 'rvy_mail_buffer_hook' );
+
+ if (rvy_get_option('use_notification_buffer')) {
+ if (!$cron_timestamp) {
+ wp_schedule_event(time(), 'two_minutes', 'rvy_mail_buffer_hook');
+ }
+ } else {
+ wp_unschedule_event($cron_timestamp, 'rvy_mail_buffer_hook');
+ }
+}
+
+function rvy_mail_buffer_cron_interval( $schedules ) {
+ $schedules['two_minutes'] = array(
+ 'interval' => 120,
+ 'display' => esc_html__( 'Every 2 Minutes', 'revisionary' ),
+ );
+
+ return $schedules;
+}
+
+function _revisionary_publish_scheduled_cron($revision_id) {
+ if (is_array($revision_id) && isset($revision_id['revision_id'])) {
+ $revision_id = $revision_id['revision_id'];
+ }
+
+ if (rvy_get_option('scheduled_revisions') && rvy_get_option('scheduled_publish_cron')) {
+ revisionary_publish_scheduled(compact('revision_id'));
+ }
+}
+
+/*=================== End WP-Cron implementation ====================*/
+
+
+function _rvy_existing_schedules_to_cron($prev_use_cron, $use_cron) {
+ if ($use_cron && !$prev_use_cron) {
+ global $wpdb;
+
+ $time_gmt = current_time('mysql', 1);
+
+ $results = $wpdb->get_results(
+ $wpdb->prepare(
+ "SELECT * FROM $wpdb->posts WHERE post_type != 'revision' AND post_status != 'inherit' AND post_mime_type = 'future-revision' AND post_date_gmt > %s ORDER BY post_date_gmt DESC",
+ $time_gmt
+ )
+ );
+
+ foreach($results as $revision) {
+ if (!wp_get_scheduled_event('publish_revision_rvy', ['revision_id' => $revision->ID])) {
+ wp_schedule_single_event(strtotime($revision->post_date_gmt), 'publish_revision_rvy', ['revision_id' => $revision->ID]);
+ }
+ }
+ }
+
+ if (!$use_cron && $prev_use_cron) {
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ rvy_update_next_publish_date();
+ }
+}
+
+
+/*
+ * Revision previews: prevent redirect for non-standard post url
+ */
+function _rvy_no_redirect_filter($redirect, $orig) {
+ global $current_user, $wpdb;
+
+ if (!empty($current_user->ID) && (empty($wpdb) || empty($wpdb->is_404))) {
+ $redirect = $orig;
+ }
+
+ return $redirect;
+}
+
+function rvy_ajax_handler() {
+ global $current_user, $wpdb;
+
+ if (!empty($_REQUEST['rvy_ajax_field']) && !empty($_REQUEST['rvy_ajax_value'])) {
+ if ($post_id = intval($_REQUEST['rvy_ajax_value'])) {
+
+ switch ($_REQUEST['rvy_ajax_field']) {
+ case 'create_revision':
+ if (current_user_can('copy_post', $post_id)) {
+ $time_gmt = (!empty($_REQUEST['rvy_date_selection'])) ? intval($_REQUEST['rvy_date_selection']) : '';
+
+ require_once( dirname(REVISIONARY_FILE).'/revision-creation_rvy.php' );
+ $rvy_creation = new PublishPress\Revisions\RevisionCreation();
+
+ $revision_status = (rvy_get_option('auto_submit_revisions') && current_user_can('edit_post', $post_id)) ? 'pending-revision' : 'draft-revision';
+ $rvy_creation->createRevision($post_id, $revision_status, compact('time_gmt'));
+ }
+ exit;
+
+ case 'submit_revision':
+ // capability check is applied within function to support batch execution without redundant checks
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ rvy_revision_submit($post_id);
+ $check_autosave = true;
+
+ break;
+
+ case 'create_scheduled_revision':
+ if (!empty($_REQUEST['rvy_date_selection'])) {
+ $time_gmt = intval($_REQUEST['rvy_date_selection']);
+
+ if (current_user_can('edit_post', $post_id)) {
+ require_once( dirname(REVISIONARY_FILE).'/revision-creation_rvy.php' );
+ $rvy_creation = new PublishPress\Revisions\RevisionCreation();
+ $rvy_creation->createRevision($post_id, 'future-revision', compact('time_gmt'));
+ }
+ }
+
+ break;
+
+ case 'author_select':
+ if (!empty($_REQUEST['rvy_selection'])) {
+ if (current_user_can('edit_post', $post_id)) {
+ update_post_meta($post_id, '_rvy_author_selection', $_REQUEST['rvy_selection']);
+ }
+ }
+
+ break;
+
+ default:
+ }
+
+ if (('submit_revision' != $_REQUEST['rvy_ajax_field']) && !empty($check_autosave) && !defined('REVISIONARY_IGNORE_REVISION_AUTOSAVE')) {
+ if ($autosave_post = PublishPress\Revisions\Utils::get_post_autosave($post_id, $current_user->ID)) {
+ $main_post = get_post($post_id);
+
+ // If revision autosave is newer than revision post_updated date, copy over post data
+ if (strtotime($autosave_post->post_modified_gmt) > strtotime($main_post->post_modified_gmt)) {
+ $set_post_properties = [
+ 'post_content',
+ 'post_content_filtered',
+ 'post_title',
+ 'post_excerpt',
+ ];
+
+ foreach($set_post_properties as $prop) {
+ if (!empty($autosave_post->$prop)) {
+ $update_data[$prop] = $autosave_post->$prop;
+ }
+ }
+
+ $wpdb->update($wpdb->posts, $update_data, ['ID' => $post_id]);
+
+ $wpdb->delete($wpdb->posts, ['ID' => $autosave_post->ID]);
+ }
+ }
+ }
+
+ exit;
+ }
+
+ }
+
+ if (defined('DOING_AJAX') && DOING_AJAX && isset($_REQUEST['action']) && ('get-revision-diffs' == $_REQUEST['action'])) {
+ require_once( dirname(__FILE__).'/admin/history_rvy.php' );
+ new RevisionaryHistory();
+ }
+}
+
+function rvy_get_post_meta($post_id, $meta_key, $unused = false) {
+ return get_post_meta($post_id, $meta_key, true);
+}
+
+function rvy_update_post_meta($post_id, $meta_key, $meta_value) {
+ global $wpdb, $revisionary;
+
+ if (!empty($revisionary)) {
+ $revisionary->internal_meta_update = true;
+ }
+
+ update_post_meta($post_id, $meta_key, $meta_value);
+
+ if (!empty($revisionary)) {
+ $revisionary->internal_meta_update = true;
+ }
+}
+
+function rvy_delete_post_meta($post_id, $meta_key) {
+ delete_post_meta($post_id, $meta_key);
+}
+
+function rvy_status_registrations() {
+ $block_editor = \PublishPress\Revisions\Utils::isBlockEditorActive();
+
+ $labels = apply_filters('revisionary_status_labels',
+ rvy_get_option('revision_statuses_noun_labels') ?
+ [
+ 'draft-revision' => [
+ 'name' => esc_html__('Working Copy', 'revisionary'),
+ 'submit' => esc_html__('Create Working Copy', 'revisionary'),
+ 'submit_short' => esc_html__('Copy', 'revisionary'),
+ 'submitting' => esc_html__('Creating Working Copy...', 'revisionary'),
+ 'submitted' => esc_html__('Working Copy ready', 'revisionary'),
+ 'approve' => esc_html__('Approve Changes', 'revisionary'),
+ 'approve_short' => esc_html__('Approve', 'revisionary'),
+ 'approving' => esc_html__('Approving Changes...', 'revisionary'),
+ 'publish' => esc_html__('Publish Changes', 'revisionary'),
+ 'save' => esc_html__('Save Revision', 'revisionary'),
+ 'update' => esc_html__('Update Revision', 'revisionary'),
+ 'plural' => esc_html__('Working Copies', 'revisionary'),
+ 'short' => esc_html__('Working Copy', 'revisionary'),
+ 'count' => _n_noop('Working Copies (%d) ', 'Working Copies (%d) ', 'revisionary'), // @todo: confirm API will support a fixed string
+ 'basic' => 'Copy',
+ ],
+
+ 'pending-revision' => [
+ 'name' => esc_html__('Change Request', 'revisionary'),
+ 'submit' => esc_html__('Submit Change Request', 'revisionary'),
+ 'submit_short' => esc_html__('Submit', 'revisionary'),
+ 'submitting' => esc_html__('Submitting Changes...', 'revisionary'),
+ 'submitted' => esc_html__('Changes Submitted', 'revisionary'),
+ 'approve' => esc_html__('Approve Changes', 'revisionary'),
+ 'approve_short' => esc_html__('Approve', 'revisionary'),
+ 'approving' => esc_html__('Approving Changes...', 'revisionary'),
+ 'publish' => esc_html__('Publish Changes', 'revisionary'),
+ 'save' => esc_html__('Save Revision', 'revisionary'),
+ 'update' => esc_html__('Update Revision', 'revisionary'),
+ 'plural' => esc_html__('Change Requests', 'revisionary'),
+ 'short' => esc_html__('Change Request', 'revisionary'),
+ 'count' => _n_noop('Change Requests (%d) ', 'Change Requests (%d) ', 'revisionary'),
+ 'enable' => esc_html__('Enable Change Requests', 'revisionary'),
+ 'basic' => 'Change Request',
+ ],
+
+ 'future-revision' => [
+ 'name' => esc_html__('Scheduled Change', 'revisionary'),
+ 'submit' => esc_html__('Schedule Changes', 'revisionary'),
+ 'submit_short' => esc_html__('Schedule Changes', 'revisionary'),
+ 'submitting' => esc_html__('Scheduling Changes...', 'revisionary'),
+ 'submitted' => esc_html__('Changes are Scheduled.', 'revisionary'),
+ 'approve' => esc_html__('Schedule Changes', 'revisionary'),
+ 'approve_short' => esc_html__('Schedule Changes', 'revisionary'),
+ 'publish' => esc_html__('Publish Changes', 'revisionary'),
+ 'save' => esc_html__('Save Revision', 'revisionary'),
+ 'update' => esc_html__('Update Revision', 'revisionary'),
+ 'plural' => esc_html__('Scheduled Changes', 'revisionary'),
+ 'short' => esc_html__('Scheduled Change', 'revisionary'),
+ 'count' => _n_noop('Scheduled Changes (%d) ', 'Scheduled Changes (%d) ', 'revisionary'),
+ 'basic' => 'Scheduled Change',
+ ],
+ ]
+
+ :
+ [
+ 'draft-revision' => [
+ 'name' => esc_html__('Unsubmitted Revision', 'revisionary'),
+ 'submit' => esc_html__('New Revision', 'revisionary'),
+ 'submit_short' => esc_html__('New Revision', 'revisionary'),
+ 'submitting' => esc_html__('Creating Revision...', 'revisionary'),
+ 'submitted' => ($block_editor) ? esc_html__('The Revision is ready to edit.', 'revisionary') : esc_html__('Revision ready to edit.', 'revisionary'),
+ 'approve' => esc_html__('Approve Revision', 'revisionary'),
+ 'approve_short' => esc_html__('Approve', 'revisionary'),
+ 'publish' => esc_html__('Publish Revision', 'revisionary'),
+ 'save' => esc_html__('Save Revision', 'revisionary'),
+ 'update' => esc_html__('Update Revision', 'revisionary'),
+ 'plural' => esc_html__('Unsubmitted Revisions', 'revisionary'),
+ 'short' => esc_html__('Not Submitted', 'revisionary'),
+ 'count' => _n_noop('Not Submitted for Approval (%s) ', 'Not Submitted for Approval (%s) ', 'revisionary'), // @todo: confirm API will support a fixed string
+ 'basic' => 'Revision',
+ ],
+
+ 'pending-revision' => [
+ 'name' => esc_html__('Submitted Revision', 'revisionary'),
+ 'submit' => esc_html__('Submit Revision', 'revisionary'),
+ 'submit_short' => esc_html__('Submit', 'revisionary'),
+ 'submitting' => esc_html__('Submitting Revision...', 'revisionary'),
+ 'submitted' => ($block_editor) ? esc_html__('The Revision is Submitted', 'revisionary') : esc_html__('Revision Submitted', 'revisionary'),
+ 'approve' => esc_html__('Approve Revision', 'revisionary'),
+ 'approve_short' => esc_html__('Approve', 'revisionary'),
+ 'publish' => esc_html__('Publish Revision', 'revisionary'),
+ 'save' => esc_html__('Save Revision', 'revisionary'),
+ 'update' => esc_html__('Update Revision', 'revisionary'),
+ 'plural' => esc_html__('Submitted Revisions', 'revisionary'),
+ 'short' => esc_html__('Submitted', 'revisionary'),
+ 'count' => _n_noop('Submitted for Approval (%s) ', 'Submitted for Approval (%s) ', 'revisionary'),
+ 'basic' => 'Revision',
+ ],
+
+ 'future-revision' => [
+ 'name' => esc_html__('Scheduled Revision', 'revisionary'),
+ 'submit' => esc_html__('Schedule Revision', 'revisionary'),
+ 'submit_short' => esc_html__('Schedule Revision', 'revisionary'),
+ 'submitting' => esc_html__('Scheduling Revision...', 'revisionary'),
+ 'submitted' => ($block_editor) ? esc_html__('The Revision is Scheduled', 'revisionary') : esc_html__('Revision Scheduled', 'revisionary'),
+ 'approve' => esc_html__('Approve Revision', 'revisionary'),
+ 'approve_short' => esc_html__('Approve', 'revisionary'),
+ 'publish' => esc_html__('Publish Revision', 'revisionary'),
+ 'save' => esc_html__('Save Revision', 'revisionary'),
+ 'update' => esc_html__('Update Revision', 'revisionary'),
+ 'plural' => esc_html__('Scheduled Revisions', 'revisionary'),
+ 'short' => esc_html__('Scheduled', 'revisionary'),
+ 'count' => _n_noop('Scheduled Revision (%s) ', 'Scheduled Revisions (%s) ', 'revisionary'),
+ 'basic' => 'Scheduled Revision',
+ ],
+ ]
+ );
+
+ register_post_status('draft-revision', array(
+ 'label' => $labels['draft-revision']['name'],
+ 'labels' => (object) $labels['draft-revision'],
+ 'protected' => true,
+ 'internal' => true,
+ 'label_count' => $labels['draft-revision']['count'],
+ 'exclude_from_search' => false,
+ 'show_in_admin_all_list' => false,
+ 'show_in_admin_status_list' => false,
+ ));
+
+ register_post_status('pending-revision', array(
+ 'label' => $labels['pending-revision']['name'],
+ 'labels' => (object) $labels['pending-revision'],
+ 'protected' => true,
+ 'internal' => true,
+ 'label_count' => $labels['pending-revision']['count'],
+ 'exclude_from_search' => false,
+ 'show_in_admin_all_list' => false,
+ 'show_in_admin_status_list' => false,
+ ));
+
+ register_post_status('future-revision', array(
+ 'label' => $labels['future-revision']['name'],
+ 'labels' => (object) $labels['future-revision'],
+ 'protected' => true,
+ 'internal' => true,
+ 'label_count' => $labels['future-revision']['count'],
+ 'exclude_from_search' => false,
+ 'show_in_admin_all_list' => false,
+ 'show_in_admin_status_list' => false,
+ ));
+
+ foreach(rvy_get_manageable_types() as $post_type) {
+ add_filter("rest_{$post_type}_collection_params", function($query_params, $post_type = '')
+ {
+ $query_params['status']['items']['enum'] []= 'draft-revision';
+ $query_params['status']['items']['enum'] []= 'pending-revision';
+ $query_params['status']['items']['enum'] []= 'future-revision';
+ return $query_params;
+ }, 999, 2
+ );
+ }
+
+ // WP > 5.3: Don't allow revision statuses to be blocked at the REST API level. Our own filters are sufficient to regulate their usage.
+ add_action( 'rest_api_init', function() {
+ global $wp_post_statuses;
+
+ foreach(rvy_revision_statuses() as $status) {
+ if (isset($wp_post_statuses[$status])) {
+ $wp_post_statuses[$status]->internal = false;
+ }
+ }
+ }, 97
+ );
+
+ add_action( 'rest_api_init', function() {
+ global $wp_post_statuses;
+
+ foreach(rvy_revision_statuses() as $status) {
+ if (isset($wp_post_statuses[$status])) {
+ $wp_post_statuses[$status]->internal = true;
+ }
+ }
+ }, 99
+);
+}
+
+function pp_revisions_status_label($status_name, $label_property) {
+ global $wp_post_statuses;
+
+ if (!empty($wp_post_statuses[$status_name]) && !empty($wp_post_statuses[$status_name]->labels->$label_property)) {
+ return $wp_post_statuses[$status_name]->labels->$label_property;
+ } else {
+ return '';
+ }
+}
+
+function pp_revisions_label($label_name) {
+ static $labels;
+
+ if (empty($labels)) {
+ $labels = apply_filters('revisionary_labels',
+ [
+ 'my_revisions' => (rvy_get_option('revision_statuses_noun_labels'))
+ ? _n_noop('%sMy Copies & Changes%s(%s)', '%sMy Copies & Changes%s(%s)', 'revisionary')
+ : _n_noop('%sMy Revisions%s(%s)', '%sMy Revisions%s(%s)', 'revisionary'),
+
+ 'my_published_posts' => _n_noop('%sRevisions to My Posts%s(%s)', '%sRevisions to My Posts%s(%s)', 'revisionary'),
+
+ 'queue_col_revision' => esc_html__('Revision', 'revisionary'),
+ 'queue_col_revised_by' => esc_html__('Revised By', 'revisionary'),
+ 'queue_col_revision_date' => esc_html__('Revision Date', 'revisionary'),
+ 'queue_col_post_author' => esc_html__('Post Author', 'revisionary'),
+ 'queue_col_published_post' => esc_html__('Published Post', 'revisionary'),
+ 'update_revision' => esc_html__('Update Revision', 'revisionary'),
+
+ 'submit_revision' => (rvy_get_option('revision_statuses_noun_labels'))
+ ? esc_html__('Submit Revision', 'revisionary')
+ : esc_html__('Submit Changes', 'revisionary')
+ ]);
+ }
+
+ return (isset($labels[$label_name])) ? $labels[$label_name] : '';
+}
+
+// WP function is_plugin_active_for_network() is defined in admin
+function rvy_plugin_active_for_network( $plugin ) {
+ if ( ! is_multisite() ) {
+ return false;
+ }
+
+ $plugins = get_site_option( 'active_sitewide_plugins' );
+ if ( isset( $plugins[ $plugin ] ) ) {
+ return true;
+ }
+
+ return false;
+}
+
+function rvy_is_plugin_active($check_plugin_file) {
+ $plugins = (array)get_option('active_plugins');
+ foreach ($plugins as $plugin_file) {
+ if (false !== strpos($plugin_file, $check_plugin_file)) {
+ return $plugin_file;
+ }
+ }
+
+ if (is_multisite()) {
+ $plugins = (array)get_site_option('active_sitewide_plugins');
+
+ // network activated plugin names are array keys
+ foreach (array_keys($plugins) as $plugin_file) {
+ if (false !== strpos($plugin_file, $check_plugin_file)) {
+ return $plugin_file;
+ }
+ }
+ }
+}
+
+function rvy_configuration_late_init() {
+ global $revisionary;
+
+ if (!empty($revisionary)) {
+ $revisionary->configurationLateInit();
+ }
+}
+
+// auto-define the Revisor role to include custom post type capabilities equivalent to those added for post, page in rvy_add_revisor_role()
+function rvy_add_revisor_custom_caps() {
+ if ( ! rvy_get_option( 'revisor_role_add_custom_rolecaps' ) )
+ return;
+
+ global $wp_roles, $revisionary;
+
+ if (empty($revisionary)) {
+ return;
+ }
+
+ $custom_types = array_intersect_key(
+ get_post_types(['_builtin' => false], 'object'),
+ $revisionary->enabled_post_types
+ );
+
+ if ( isset( $wp_roles->roles['revisor'] ) ) {
+ if ($custom_types) {
+ foreach( $custom_types as $post_type => $type_obj ) {
+ $cap = $type_obj->cap;
+ $custom_caps = array_fill_keys( array( $cap->read_private_posts, $cap->edit_posts, $cap->edit_others_posts, "delete_{$post_type}s" ), true );
+
+ if (!empty($type_obj->cap->edit_published_posts)) {
+ $list_published_cap = str_replace('edit_', 'list_', $type_obj->cap->edit_published_posts);
+ $custom_caps[$list_published_cap] = true;
+ }
+
+ if (!empty($type_obj->cap->edit_private_posts)) {
+ $list_private_cap = str_replace('edit_', 'list_', $type_obj->cap->edit_private_posts);
+ $custom_caps[$list_private_cap] = true;
+ }
+
+ $wp_roles->roles['revisor']['capabilities'] = array_merge( $wp_roles->roles['revisor']['capabilities'], $custom_caps );
+ $wp_roles->role_objects['revisor']->capabilities = array_merge( $wp_roles->role_objects['revisor']->capabilities, $custom_caps );
+ }
+ }
+ }
+
+ if ( isset( $wp_roles->roles['contributor'] ) ) {
+ if ($custom_types) {
+ foreach( $custom_types as $post_type => $type_obj ) {
+ $cap = $type_obj->cap;
+ $custom_caps = [];
+
+ if (!empty($type_obj->cap->edit_published_posts)) {
+ $list_published_cap = str_replace('edit_', 'list_', $type_obj->cap->edit_published_posts);
+ $custom_caps[$list_published_cap] = true;
+ }
+
+ if (!empty($type_obj->cap->edit_private_posts)) {
+ $list_private_cap = str_replace('edit_', 'list_', $type_obj->cap->edit_private_posts);
+ $custom_caps[$list_private_cap] = true;
+ }
+
+ $wp_roles->roles['contributor']['capabilities'] = array_merge( $wp_roles->roles['contributor']['capabilities'], $custom_caps );
+ $wp_roles->role_objects['contributor']->capabilities = array_merge( $wp_roles->role_objects['contributor']->capabilities, $custom_caps );
+ }
+ }
+ }
+
+ global $current_user;
+
+ foreach(['contributor', 'revisor'] as $role_name) {
+ if (in_array($role_name, $current_user->roles)) {
+ $current_user->allcaps = array_merge($current_user->allcaps, $wp_roles->role_objects[$role_name]->capabilities);
+ }
+ }
+
+ if (function_exists('presspermit')) {
+ $user = presspermit()->getUser();
+ $user->allcaps = $current_user->allcaps;
+ }
+}
+
+function rvy_detect_post_type() {
+ global $revisionary;
+
+ if ( isset($revisionary) && $revisionary->doing_rest && $revisionary->rest->is_posts_request )
+ return $revisionary->rest->post_type;
+ else
+ return awp_post_type_from_uri();
+}
+
+function rvy_detect_post_id() {
+ global $revisionary;
+
+ if ( isset($revisionary) && $revisionary->doing_rest && $revisionary->rest->is_posts_request ) {
+ $post_id = $revisionary->rest->post_id;
+
+ } elseif ( ! empty( $_GET['post'] ) ) {
+ $post_id = (int) $_GET['post'];
+
+ } elseif ( ! empty( $_POST['post_ID'] ) ) {
+ $post_id = (int) $_POST['post_ID'];
+
+ } elseif ( ! empty( $_REQUEST['post_id'] ) ) {
+ $post_id = (int) $_REQUEST['post_id'];
+
+ } elseif ( ! empty( $_GET['p'] ) ) {
+ $post_id = (int) $_GET['p'];
+
+ } elseif ( ! empty( $_GET['id'] ) ) {
+ $post_id = (int) $_GET['id'];
+
+ } elseif ( ! empty( $_REQUEST['fl_builder_data'] ) && is_array( $_REQUEST['fl_builder_data'] ) && ! empty( $_REQUEST['fl_builder_data']['post_id'] ) ) {
+ $post_id = (int) $_REQUEST['fl_builder_data']['post_id'];
+
+ } elseif ( ! empty( $_GET['page_id'] ) ) {
+ $post_id = (int) $_GET['page_id'];
+
+ } elseif (defined('REST_REQUEST') && REST_REQUEST && isset($_SERVER['REQUEST_URI']) && strpos(esc_url_raw($_SERVER['REQUEST_URI']), 'autosaves')) {
+ require_once( dirname(__FILE__).'/rest_rvy.php' );
+ $post_id = Revisionary_REST::get_id_element(esc_url_raw($_SERVER['REQUEST_URI']), 1);
+
+ } elseif (defined('DOING_AJAX') && DOING_AJAX) {
+ $post_id = apply_filters('revisionary_detect_id', 0, ['is_ajax' => true]);
+
+ } else {
+ $post_id = 0;
+ }
+
+ return $post_id;
+}
+
+function rvy_add_revisor_role( $requested_blog_id = '' ) {
+ global $wp_roles;
+
+ if (defined('REVISIONARY_NO_REVISOR_ROLE')) {
+ return;
+ }
+
+ $wp_role_caps = array(
+ 'read' => true,
+ 'read_private_posts' => true,
+ 'read_private_pages' => true,
+ 'edit_posts' => true,
+ 'delete_posts' => true,
+ 'edit_others_posts' => true,
+ 'edit_pages' => true,
+ 'delete_pages' => true,
+ 'edit_others_pages' => true,
+ 'list_published_posts' => true,
+ 'list_published_pages' => true,
+ 'list_private_posts' => true,
+ 'list_private_pages' => true,
+ 'upload_files' => true,
+ 'level_3' => true,
+ 'level_2' => true,
+ 'level_1' => true,
+ 'level_0' => true
+ );
+
+ $wp_roles->add_role( 'revisor', esc_html__( 'Revisor', 'revisionary' ), $wp_role_caps );
+}
+
+function rvy_apply_role_translation($translations, $text, $context, $domain) {
+ if (('User role' === $context) && ('Revisor' == $text) && ($domain !== 'revisionary')) {
+ return translate_with_gettext_context($text, $context, 'revisionary');
+ }
+
+ return $translations;
+}
+
+function rvy_role_translation_support() {
+ _x('Revisor', 'User role', 'revisionary');
+ add_filter('gettext_with_context', 'rvy_apply_role_translation', 10, 4);
+}
+
+// wrapper function for use with wp_cron hook
+function revisionary_publish_scheduled($args = []) {
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ rvy_publish_scheduled_revisions($args);
+}
+
+function revisionary_refresh_postmeta($post_id, $args = []) {
+ global $wpdb;
+
+ $ignore_revisions = (!empty($args['ignore_revisions'])) ? $args['ignore_revisions'] : [];
+ $ignore_clause = ($ignore_revisions) ? " AND ID NOT IN (" . implode(",", array_map('intval', $ignore_revisions)) . ")" : '';
+
+ if (defined('REVISIONARY_LIMIT_IGNORE_UNSUBMITTED')) {
+ $ignore_clause .= " AND post_mime_type != 'draft-revision'";
+ }
+
+ $revision_status_csv = implode("','", array_map('sanitize_key', rvy_revision_statuses()));
+
+ $has_revisions = $wpdb->get_var(
+ // account for post deletion
+ $wpdb->prepare(
+ "SELECT ID FROM $wpdb->posts WHERE post_mime_type IN ('$revision_status_csv') $ignore_clause AND comment_count = %d LIMIT 1",
+ $post_id
+ )
+ );
+
+ $set_value = !empty($has_revisions);
+
+ $_post = get_post($post_id);
+
+ if ($set_value && (empty($_post) || empty($_post->post_mime_type) || !in_array($_post->post_mime_type, ['draft-revision', 'pending-revision', 'future-revision']))) {
+ rvy_update_post_meta($post_id, '_rvy_has_revisions', $set_value);
+
+ } elseif (empty($args['insert_only'])) { // avoid redundant deletions
+ delete_post_meta($post_id, '_rvy_has_revisions');
+ }
+}
+
+function rvy_post_revision_supported($post) {
+ $post_id = (is_scalar($post)) ? $post : $post->ID;
+
+ if ($post_id) {
+ if (1 === intval(rvy_get_option('revision_limit_per_post'))) {
+ if (rvy_get_post_meta($post_id, '_rvy_has_revisions')) {
+ return false;
+ }
+ }
+ }
+
+ return true;
+}
+
+if (!empty($_REQUEST['rvy_flush_flags'])) {
+ revisionary_refresh_revision_flags();
+}
+
+function revisionary_refresh_revision_flags($published_post_id = 0, $args = []) {
+ global $wpdb;
+
+ $ignore_revision_ids = (!empty($args['ignore_revision_ids'])) ? (array) $args['ignore_revision_ids'] : [];
+
+ $status_csv = implode("','", array_map('sanitize_key', rvy_filtered_statuses()));
+ $revision_base_status_csv = implode("','", array_map('sanitize_key', rvy_revision_base_statuses()));
+
+ $revision_statuses = rvy_revision_statuses();
+
+ if (defined('REVISIONARY_LIMIT_IGNORE_UNSUBMITTED')) {
+ $revision_statuses = array_diff($revision_statuses, ['draft-revision']);
+ }
+
+ $revision_status_csv = implode("','", array_map('sanitize_key', $revision_statuses));
+
+ $query = "SELECT r.comment_count FROM $wpdb->posts r INNER JOIN $wpdb->posts p ON r.comment_count = p.ID"
+ . " WHERE p.post_status IN ('$status_csv') AND r.post_status IN ('$revision_base_status_csv')"
+ . " AND r.post_mime_type IN ('$revision_status_csv') AND p.post_mime_type NOT IN ('$revision_status_csv')";
+
+ if ($published_post_id) {
+ $query = $wpdb->prepare("$query AND p.ID = %d", $published_post_id);
+ }
+
+ if ($ignore_revision_ids) {
+ $ignore_revisions_csv = implode("','", array_map('sanitize_key', $ignore_revision_ids));
+ $query .= " AND r.ID NOT IN ('$ignore_revisions_csv')";
+ }
+
+ $arr_have_revisions = $wpdb->get_col($query);
+
+ $have_revisions = implode("','", array_map('intval', array_unique($arr_have_revisions)));
+
+ if ($ids = $wpdb->get_col("SELECT meta_id FROM $wpdb->postmeta WHERE meta_key = '_rvy_has_revisions' AND post_id NOT IN ('$have_revisions')")) {
+ foreach ($ids as $post_id) {
+ rvy_delete_post_meta($post_id, '_rvy_has_revisions');
+ }
+ }
+
+ $query = "SELECT post_id FROM $wpdb->postmeta WHERE meta_key = '_rvy_has_revisions'";
+
+ if ($published_post_id) {
+ $query = $wpdb->prepare("$query AND post_id = %d", $published_post_id);
+ }
+
+ $have_flag_ids = $wpdb->get_col($query);
+
+ if ($posts_missing_flag = array_diff($arr_have_revisions, $have_flag_ids)) {
+ foreach($posts_missing_flag as $post_id) {
+ rvy_update_post_meta($post_id, '_rvy_has_revisions', true);
+ }
+ }
+
+ if ($posts_invalid_flag = array_diff($have_flag_ids, $arr_have_revisions)) {
+ foreach($posts_missing_flag as $post_id) {
+ rvy_delete_post_meta($post_id, '_rvy_has_revisions');
+ }
+ }
+}
+
+function rvy_refresh_options() {
+ rvy_retrieve_options(true);
+ rvy_retrieve_options(false);
+
+ rvy_refresh_default_options();
+ rvy_refresh_options_sitewide();
+}
+
+function rvy_refresh_options_sitewide() {
+ if ( ! RVY_NETWORK )
+ return;
+
+ global $rvy_options_sitewide;
+ $rvy_options_sitewide = apply_filters( 'options_sitewide_rvy', rvy_default_options_sitewide() ); // establishes which options are set site-wide
+
+ if ( $options_sitewide_reviewed = rvy_get_option( 'options_sitewide_reviewed', true ) ) {
+ $custom_options_sitewide = (array) rvy_get_option( 'options_sitewide', true );
+
+ $unreviewed_default_sitewide = array_diff( array_keys($rvy_options_sitewide), $options_sitewide_reviewed );
+
+ $rvy_options_sitewide = array_fill_keys( array_merge( $custom_options_sitewide, $unreviewed_default_sitewide ), true );
+ }
+
+ $rvy_options_sitewide = array_filter( $rvy_options_sitewide );
+}
+
+function rvy_refresh_default_options() {
+ global $rvy_default_options;
+
+ $rvy_default_options = apply_filters( 'default_options_rvy', rvy_default_options() );
+
+ if ( RVY_NETWORK )
+ rvy_apply_custom_default_options();
+}
+
+function rvy_apply_custom_default_options() {
+ global $wpdb, $rvy_default_options, $rvy_options_sitewide;
+
+ if ( $results = $wpdb->get_results(
+ $wpdb->prepare(
+ "SELECT meta_key, meta_value FROM $wpdb->sitemeta WHERE site_id = %d AND meta_key LIKE 'rvy_default_%'",
+ $wpdb->siteid
+ )
+ ) ) {
+ foreach ( $results as $row ) {
+ $option_basename = str_replace( 'rvy_default_', '', $row->meta_key );
+
+ if ( ! empty( $rvy_options_sitewide[$option_basename] ) )
+ continue; // custom defaults are only for site-specific options
+
+ if( isset( $rvy_default_options[$option_basename] ) )
+ $rvy_default_options[$option_basename] = maybe_unserialize( $row->meta_value );
+ }
+ }
+}
+
+function rvy_delete_option( $option_basename, $sitewide = -1 ) {
+
+ // allow explicit selection of sitewide / non-sitewide scope for better performance and update security
+ if ( -1 === $sitewide ) {
+ global $rvy_options_sitewide;
+ $sitewide = isset( $rvy_options_sitewide ) && ! empty( $rvy_options_sitewide[$option_basename] );
+ }
+
+ if ( $sitewide ) {
+ global $wpdb;
+ $wpdb->query(
+ $wpdb->prepare(
+ "DELETE FROM {$wpdb->sitemeta} WHERE site_id = %s AND meta_key = %s",
+ $wpdb->siteid,
+ "rvy_$option_basename"
+ )
+ );
+ } else
+ delete_option( "rvy_$option_basename" );
+}
+
+function rvy_update_option( $option_basename, $option_val, $sitewide = -1 ) {
+
+ // allow explicit selection of sitewide / non-sitewide scope for better performance and update security
+ if ( -1 === $sitewide ) {
+ global $rvy_options_sitewide;
+ $sitewide = isset( $rvy_options_sitewide ) && ! empty( $rvy_options_sitewide[$option_basename] );
+ }
+
+ if ($sitewide) {
+ update_site_option("rvy_$option_basename", $option_val);
+ } else {
+ update_option("rvy_$option_basename", $option_val);
+ }
+}
+
+function rvy_retrieve_options( $sitewide = false ) {
+ global $wpdb;
+
+ if ( $sitewide ) {
+ if ( ! RVY_NETWORK )
+ return;
+
+ global $rvy_site_options;
+
+ $rvy_site_options = array();
+
+ if ( $results = $wpdb->get_results(
+ $wpdb->prepare(
+ "SELECT meta_key, meta_value FROM $wpdb->sitemeta WHERE site_id = %d AND meta_key LIKE 'rvy_%'",
+ $wpdb->siteid
+ )
+ ) ) {
+ foreach ( $results as $row ) {
+ $rvy_site_options[$row->meta_key] = $row->meta_value;
+ }
+ }
+
+ $rvy_site_options = apply_filters( 'site_options_rvy', $rvy_site_options );
+ return $rvy_site_options;
+
+ } else {
+ global $rvy_blog_options;
+
+ $rvy_blog_options = array();
+
+ if ( $results = $wpdb->get_results("SELECT option_name, option_value FROM $wpdb->options WHERE option_name LIKE 'rvy_%'") ) {
+ foreach ( $results as $row ) {
+ $rvy_blog_options[$row->option_name] = $row->option_value;
+ }
+ }
+
+ $rvy_blog_options = apply_filters( 'options_rvy', $rvy_blog_options );
+ return $rvy_blog_options;
+ }
+}
+
+function rvy_filter_option($option_basename, $args) {
+ return apply_filters("pp_revisions_option_{$option_basename}", rvy_get_option($option_basename), $args);
+}
+
+function rvy_get_option($option_basename, $sitewide = -1, $get_default = false, $args = []) {
+ if (('async_scheduled_publish' == $option_basename) && function_exists('relevanssi_query')) {
+ return false;
+ }
+
+ if (('scheduled_revisions' == $option_basename) && empty($args['bypass_condition_check'])
+ && defined('DISABLE_WP_CRON') && DISABLE_WP_CRON && rvy_get_option('scheduled_publish_cron') && !rvy_get_option('wp_cron_usage_detected') && apply_filters('revisionary_wp_cron_disabled', true)
+ ) {
+ return false;
+ }
+
+ if ( ! $get_default ) {
+ // allow explicit selection of sitewide / non-sitewide scope for better performance and update security
+ if ( -1 === $sitewide ) {
+ global $rvy_options_sitewide;
+ $sitewide = isset( $rvy_options_sitewide ) && ! empty( $rvy_options_sitewide[$option_basename] );
+ }
+
+ if ( $sitewide ) {
+ // this option is set site-wide
+ global $rvy_site_options;
+
+ if ( ! isset($rvy_site_options) )
+ $rvy_site_options = rvy_retrieve_options( true );
+
+ if ( isset($rvy_site_options["rvy_{$option_basename}"]) )
+ $optval = $rvy_site_options["rvy_{$option_basename}"];
+
+ } else {
+ global $rvy_blog_options;
+
+ if ( ! isset($rvy_blog_options) )
+ $rvy_blog_options = rvy_retrieve_options( false );
+
+ if ( isset($rvy_blog_options["rvy_$option_basename"]) )
+ $optval = $rvy_blog_options["rvy_$option_basename"];
+ }
+ }
+
+ if ( ! isset( $optval ) ) {
+ global $rvy_default_options;
+
+ if ( empty( $rvy_default_options ) ) {
+ if ( did_action( 'rvy_init' ) ) // Make sure other plugins have had a chance to apply any filters to default options
+ rvy_refresh_default_options();
+ else {
+ $hardcode_defaults = rvy_default_options();
+ if ( isset($hardcode_defaults[$option_basename]) )
+ $optval = $hardcode_defaults[$option_basename];
+ }
+ }
+
+ if ( ! empty($rvy_default_options) && ! empty( $rvy_default_options[$option_basename] ) )
+ $optval = $rvy_default_options[$option_basename];
+
+ if ( ! isset($optval) )
+ return '';
+ }
+
+ return maybe_unserialize($optval);
+}
+
+function rvy_log_async_request($action) {
+ // the function which performs requested action will clear this entry to confirm that the asynchronous call was effective
+ $requested_actions = get_option( 'requested_remote_actions_rvy' );
+ if ( ! is_array($requested_actions) )
+ $requested_actions = array();
+
+ $requested_actions[$action] = true;
+ update_option( 'requested_remote_actions_rvy', $requested_actions );
+}
+
+function rvy_confirm_async_execution($action) {
+ $requested_actions = get_option( 'requested_remote_actions_rvy' );
+ if ( is_array($requested_actions) && isset($requested_actions[$action]) ) {
+ unset( $requested_actions[$action] );
+ update_option( 'requested_remote_actions_rvy', $requested_actions );
+ } else {
+ exit;
+ }
+}
+
+function is_content_administrator_rvy() {
+ $cap_name = defined( 'SCOPER_CONTENT_ADMIN_CAP' ) ? SCOPER_CONTENT_ADMIN_CAP : 'activate_plugins';
+ return current_user_can( $cap_name );
+}
+
+function rvy_notice( $message, $class = 'error fade' ) {
+ include_once( dirname(__FILE__).'/lib/error_rvy.php');
+ $rvy_err = new RvyError();
+ return $rvy_err->add_notice( $message, compact( 'class' ) );
+}
+
+function rvy_error( $err_slug, $arg2 = '' ) {
+ include_once( dirname(__FILE__).'/lib/error_rvy.php');
+ $rvy_err = new RvyError();
+ $rvy_err->error_notice( $err_slug );
+}
+
+function rvy_check_duplicate_mail($new_msg, $sent_mail, $buffer) {
+ foreach([$sent_mail, $buffer] as $compare_set) {
+ foreach($compare_set as $sent) {
+ foreach(['address', 'title', 'message'] as $field) {
+ if (!isset($new_msg[$field])
+ || !isset($sent[$field])
+ || ($new_msg[$field] != $sent[$field])
+ ) {
+ continue 2;
+ }
+ }
+
+ $min_seconds = (defined('ET_BUILDER_PLUGIN_VERSION') || (false !== stripos(get_template(), 'divi'))) ? 20 : 5;
+
+ // If an identical message was sent or queued to the same recipient less than 5 seconds ago, don't send another
+ if (abs($new_msg['time_gmt'] - $sent['time_gmt']) <= $min_seconds) {
+ return true;
+ }
+ }
+ }
+}
+
+/*
+ * wp_mail() wrapper, also implements mail buffer
+ *
+ * args: ['revision_id' => $revision_id, 'post_id' => $published_post->ID, 'notification_type' => $notification_type, 'notification_class' => $notification_class]
+ */
+function rvy_mail( $address, $title, $message, $args ) {
+ /*
+ * [wp-cron action checks wp_option revisionary_mail_buffer. If wait time has elapsed, send buffered emails (up to limit per minute)]
+ *
+ * If mail is already buffered to wp_option revisionary_mail_buffer, add this email to buffer
+ *
+ * - or -
+ *
+ * Check wp_option array revisionary_sent_mail
+ * - If exceeding daily, hourly or minute limit, add this email to buffer
+ * - If sending, add current timestamp to wp_option array revisionary_sent_mail
+ */
+
+ $send = apply_filters('revisionary_mail', compact('address', 'title', 'message'), $args);
+
+ if (empty($send['address'])) {
+ return;
+ }
+
+ $new_msg = array_merge($send, ['time' => strtotime(current_time( 'mysql' )), 'time_gmt' => time()], $args);
+
+ if (!$buffer_status = rvy_mail_check_buffer($new_msg)) {
+ $buffer_status = (object)[];
+ }
+
+ if (!empty($buffer_status->new_msg_buffered)) {
+ return;
+ }
+
+ $sent_mail = (!empty($buffer_status->sent_mail)) ? $buffer_status->sent_mail : [];
+ $buffer = (!empty($buffer_status->buffer)) ? $buffer_status->buffer : [];
+ if (rvy_check_duplicate_mail($new_msg, $sent_mail, $buffer)) {
+ return;
+ }
+
+ if ( defined( 'RS_DEBUG' ) )
+ $success = wp_mail( $new_msg['address'], $new_msg['title'], $new_msg['message'] );
+ else
+ $success = @wp_mail( $new_msg['address'], $new_msg['title'], $new_msg['message'] );
+
+ if ($success || !defined('REVISIONARY_MAIL_RETRY')) {
+ if (!defined('REVISIONARY_DISABLE_MAIL_LOG')) {
+ if (!isset($buffer_status->sent_mail)) {
+ $buffer_status->sent_mail = [];
+ }
+
+ $buffer_status->sent_mail[]= $new_msg;
+ update_option('revisionary_sent_mail', $buffer_status->sent_mail);
+ }
+ }
+}
+
+function rvy_settings_scripts() {
+ if (defined('PUBLISHPRESS_REVISIONS_PRO_VERSION')) {
+ $suffix = defined('SCRIPT_DEBUG') && SCRIPT_DEBUG ? '.dev' : '';
+ wp_enqueue_script('revisionary-pro-settings', plugins_url('', REVISIONARY_PRO_FILE) . "/includes-pro/settings-pro{$suffix}.js", ['jquery', 'jquery-form'], PUBLISHPRESS_REVISIONS_VERSION, true);
+ }
+}
+
+function rvy_omit_site_options() {
+ rvy_settings_scripts();
+ add_thickbox();
+ include_once( RVY_ABSPATH . '/admin/options.php' );
+ rvy_options( false );
+}
+
+function rvy_wp_api_request() {
+ return ( function_exists('wp_api_request') ) ? wp_api_request() : false;
+}
+
+function rvy_is_status_public( $status ) {
+ if ( $post_status_obj = get_post_status_object( $status ) ) {
+ return ! empty( $post_status_obj->public );
+ }
+
+ return false;
+}
+
+function rvy_is_status_private( $status ) {
+ if ( $post_status_obj = get_post_status_object( $status ) ) {
+ return ! empty( $post_status_obj->private );
+ }
+
+ return false;
+}
+
+function rvy_is_status_published( $status ) {
+ if ( $post_status_obj = get_post_status_object( $status ) ) {
+ return ! empty( $post_status_obj->public ) || ! empty( $post_status_obj->private );
+ }
+
+ return false;
+}
+
+function rvy_halt( $msg, $title = '' ) {
+ if ( ! $title ) {
+ $title = esc_html__( 'Revision Workflow', 'revisionary' );
+ }
+ wp_die( esc_html($msg), esc_html($title), array( 'response' => 200 ) );
+}
+
+function _revisionary_dashboard_dismiss_msg() {
+ $dismissals = get_option( 'revisionary_dismissals' );
+ if ( ! is_array( $dismissals ) )
+ $dismissals = array();
+
+ $msg_id = ( isset( $_REQUEST['msg_id'] ) ) ? sanitize_key($_REQUEST['msg_id']) : 'intro_revisor_role';
+ $dismissals[$msg_id] = true;
+ update_option( 'rvy_dismissals', $dismissals );
+}
+
+function rvy_is_supported_post_type($post_type) {
+ global $revisionary;
+
+ if (empty($revisionary->enabled_post_types[$post_type]) && $revisionary->config_loaded) {
+ return false;
+ }
+
+ $types = rvy_get_manageable_types();
+ return !empty($types[$post_type]);
+}
+
+function rvy_get_manageable_types() {
+ $types = array();
+
+ global $current_user, $revisionary;
+
+ if (empty($revisionary)) {
+ return [];
+ }
+
+ foreach(array_keys($revisionary->enabled_post_types) as $post_type) {
+ $types[$post_type]= $post_type;
+ }
+
+ $types = array_diff_key($types, array('acf-field-group' => true));
+ return apply_filters('revisionary_supported_post_types', $types);
+}
+
+// thanks to GravityForms for the nifty dismissal script
+if (isset($_SERVER['PHP_SELF']) && in_array( basename($_SERVER['PHP_SELF']), array('admin.php', 'admin-ajax.php') ) ) {
+ add_action( 'wp_ajax_rvy_dismiss_msg', '_revisionary_dashboard_dismiss_msg' );
+}
+
+function rvy_is_network_activated($plugin_file = '')
+{
+ if (!$plugin_file && defined('REVISIONARY_FILE')) {
+ $plugin_file = plugin_basename(REVISIONARY_FILE);
+ }
+
+ return (array_key_exists($plugin_file, (array)maybe_unserialize(get_site_option('active_sitewide_plugins'))));
+}
+
+function rvy_init() {
+ global $wp_roles;
+
+ if ( ! isset( $wp_roles->roles['revisor'] ) ) {
+ rvy_add_revisor_role();
+ } else {
+ if (!get_site_transient('revisionary_previous_install')) {
+ set_site_transient('revisionary_previous_install', true, 86400);
+ }
+ }
+
+ rvy_role_translation_support();
+
+ if ( is_admin() ) {
+ require_once(dirname(__FILE__).'/admin/admin-init_rvy.php');
+
+ if (defined('REVISIONARY_BULK_ACTION_EARLY_EXECUTION') || !isset($_REQUEST['action2'])) {
+ rvy_admin_init();
+ } else {
+ // bulk approval fails on some sites due to post types not registered early enough
+ add_action('wp_loaded', 'rvy_admin_init');
+ }
+ } else { // @todo: fix links instead
+ // fill in the missing args for Pending / Scheduled revision preview link from Edit Posts / Pages
+ if ( isset($_SERVER['HTTP_REFERER'])
+ && ( false !== strpos( urldecode(esc_url_raw($_SERVER['HTTP_REFERER'])),'p-admin/edit-pages.php')
+ || false !== strpos( urldecode(esc_url_raw($_SERVER['HTTP_REFERER'])),'p-admin/edit.php') ) ) {
+
+ if ( ! empty($_GET['p']) ) {
+ if ( rvy_get_option( 'scheduled_revisions' ) || rvy_get_option( 'pending_revisions' ) ) {
+ if ( $post = get_post( sanitize_text_field($_GET['p']) ) ) {
+ if (rvy_in_revision_workflow($post)) {
+ $preview_arg = (defined('RVY_PREVIEW_ARG')) ? sanitize_key(constant('RVY_PREVIEW_ARG')) : 'rv_preview';
+ $_GET[$preview_arg] = 1;
+ }
+ }
+ }
+ }
+ // Is this an asynchronous request to publish scheduled revisions?
+ } elseif (!empty($_GET['action']) && ('publish_scheduled_revisions' == $_GET['action']) && rvy_get_option('scheduled_revisions')
+ && !rvy_get_option('scheduled_publish_cron')) {
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ add_action( 'rvy_init', '_rvy_publish_scheduled_revisions' );
+ }
+ }
+
+ if (empty($_GET['action']) || (isset($_GET['action']) && ('publish_scheduled_revisions' != $_GET['action']))) {
+ if (isset($_SERVER['REQUEST_URI']) && ! strpos( esc_url_raw($_SERVER['REQUEST_URI']), 'login.php' ) && rvy_get_option( 'scheduled_revisions' )
+ && !rvy_get_option('scheduled_publish_cron')) {
+
+ // If a previously requested asynchronous request was ineffective, perform the actions now
+ // (this is not executed if the current URI is from a manual publication request with action=publish_scheduled_revisions)
+ if (defined('RVY_SCHEDULED_PUBLISH_FALLBACK')) {
+ $requested_actions = get_option( 'requested_remote_actions_rvy' );
+ if ( is_array( $requested_actions) && ! empty($requested_actions) ) {
+ if ( ! empty($requested_actions['publish_scheduled_revisions']) ) {
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ rvy_publish_scheduled_revisions();
+ unset( $requested_actions['publish_scheduled_revisions'] );
+ }
+
+ update_option( 'requested_remote_actions_rvy', $requested_actions );
+ }
+ }
+
+ $next_publish = get_option( 'rvy_next_rev_publish_gmt' );
+
+ // automatically publish any scheduled revisions whose time has come
+ if ( ! $next_publish || ( agp_time_gmt() >= strtotime( $next_publish ) ) ) {
+ update_option('rvy_next_rev_publish_gmt', '2035-01-01 00:00:00');
+
+ if ( ini_get( 'allow_url_fopen' ) && rvy_get_option('async_scheduled_publish') ) {
+ // asynchronous secondary site call to avoid delays // TODO: pass site key here
+ rvy_log_async_request('publish_scheduled_revisions');
+ $url = site_url( 'index.php?action=publish_scheduled_revisions' );
+ wp_remote_post( $url, array('timeout' => 5, 'blocking' => false, 'sslverify' => apply_filters('https_local_ssl_verify', true)) );
+ } else {
+ // publish scheduled revision now
+ if ( ! defined('DOING_CRON') ) {
+ define( 'DOING_CRON', true );
+ }
+ require_once( dirname(__FILE__).'/admin/revision-action_rvy.php');
+ rvy_publish_scheduled_revisions();
+ }
+ }
+ }
+ }
+
+ require_once( dirname(__FILE__).'/revisionary_main.php');
+
+ global $revisionary;
+ $revisionary = new Revisionary();
+ $revisionary->init();
+}
+
+function rvy_is_full_editor($post, $args = []) {
+ global $current_user, $revisionary;
+
+ if (is_numeric($post)) {
+ $post = get_post($post);
+ }
+
+ if (empty($post) || !is_object($post)) {
+ return false;
+ }
+
+ if (!$type_obj = get_post_type_object($post->post_type)) {
+ return false;
+ }
+
+ $cap = (!empty($type_obj->cap->edit_others_posts)) ? $type_obj->cap->edit_others_posts : $type_obj->cap->edit_posts;
+
+ if (empty($current_user->allcaps[$cap])) {
+ return false;
+ }
+
+ if (!empty($args['check_publish_caps'])) {
+ if (!empty($type_obj->cap->edit_published_posts) && empty($current_user->allcaps[$type_obj->cap->edit_published_posts])) {
+ return false;
+ }
+ } else {
+ if (empty($revisionary)) {
+ return false;
+ }
+
+ return $revisionary->canEditPost($post, ['simple_cap_check' => true]);
+ }
+
+ return true;
+}
+
+function rvy_is_post_author($post, $user = false) {
+ if (!is_object($post)) {
+ if (!$post = get_post($post)) {
+ return false;
+ }
+ }
+
+ if (false === $user) {
+ global $current_user;
+ $user_id = $current_user->ID;
+ } else {
+ $user_id = (is_object($user)) ? $user->ID : $user;
+ }
+
+ if (!empty($post->post_author) && ($post->post_author == $user_id)) {
+ return true;
+
+ } elseif (function_exists('is_multiple_author_for_post') && is_multiple_author_for_post($user_id, $post->ID)) {
+ return true;
+ }
+
+ return false;
+}
+
+function rvy_preview_url($revision, $args = []) {
+ if (is_scalar($revision)) {
+ $revision = get_post($revision);
+ }
+
+ $defaults = ['post_type' => $revision->post_type]; // support preview url for past revisions, which are stored with post_type = 'revision'
+ foreach(array_keys($defaults) as $var) {
+ $$var = (!empty($args[$var])) ? $args[$var] : $defaults[$var];
+ }
+
+ if ('revision' == $post_type) {
+ $post_type = get_post_field('post_type', $revision->post_parent);
+ } else {
+ if ($post_type_obj = get_post_type_object($revision->post_type)) {
+ if (empty($post_type_obj->public) && !defined('FL_BUILDER_VERSION') && !apply_filters('revisionary_private_type_use_preview_url', false, $revision)) { // For non-public types, preview is not available so default to Compare Revisions screen
+ return apply_filters('revisionary_preview_url', rvy_admin_url("revision.php?revision=$revision->ID"), $revision, $args);
+ }
+ }
+ }
+ $post_type = sanitize_key($post_type);
+
+ $link_type = apply_filters(
+ 'revisionary_preview_link_type',
+ rvy_get_option('preview_link_type'),
+ $revision
+ );
+
+ $status_obj = get_post_status_object(get_post_field('post_status', rvy_post_id($revision->ID)));
+ $post_is_published = $status_obj && (!empty($status_obj->public) || !empty($status_obj->private));
+
+ $preview_arg = (defined('RVY_PREVIEW_ARG')) ? sanitize_key(constant('RVY_PREVIEW_ARG')) : 'rv_preview';
+
+ if ('id_only' == $link_type) {
+ // support using ids only if theme or plugins do not tolerate published post url and do not require standard format with revision slug
+ $preview_url = add_query_arg($preview_arg, true, get_post_permalink($revision));
+
+ if ('page' == $post_type) {
+ $preview_url = str_replace('p=', "page_id=", $preview_url);
+ $id_arg = 'page_id';
+ } else {
+ $id_arg = 'p';
+ }
+ } elseif (('revision_slug' == $link_type) || !$post_is_published) {
+ // support using actual revision slug in case theme or plugins do not tolerate published post url
+ $preview_url = add_query_arg($preview_arg, true, get_permalink($revision));
+
+ if ('page' == $post_type) {
+ $preview_url = str_replace('p=', "page_id=", $preview_url);
+ $id_arg = 'page_id';
+ } else {
+ $id_arg = 'p';
+ }
+ } else { // 'published_slug'
+ // default to published post url, appended with 'preview' and page_id args
+ $preview_url = add_query_arg($preview_arg, true, get_permalink(rvy_post_id($revision->ID)));
+ $id_arg = 'page_id';
+ }
+
+ if (strpos($preview_url, "{$id_arg}=")) {
+ $preview_url = remove_query_arg($id_arg, $preview_url);
+ }
+
+ $preview_url = add_query_arg($id_arg, $revision->ID, $preview_url);
+
+ if (!strpos($preview_url, "post_type=")) {
+ $preview_url = add_query_arg('post_type', $post_type, $preview_url);
+ }
+
+ if (!defined('REVISIONARY_PREVIEW_NO_CACHEBUST')) {
+ $preview_url = rvy_nc_url($preview_url);
+ }
+
+ $preview_url = apply_filters('revisionary_preview_url', $preview_url, $revision, $args);
+ $preview_url = remove_query_arg('preview_id', $preview_url);
+
+ return $preview_url;
+}
+
+function rvy_set_ma_post_authors($post_id, $authors)
+{
+ require_once( dirname(__FILE__).'/multiple-authors_rvy.php');
+ _rvy_set_ma_post_authors_custom_field($post_id, $authors);
+
+ $authors = wp_list_pluck($authors, 'term_id');
+ wp_set_object_terms($post_id, $authors, 'author');
+}
+
+function rvy_filtered_statuses($args = []) {
+ $defaults = ['output' => 'names', 'return' => 'array'];
+ $args = array_merge($defaults, $args);
+ foreach (array_keys($defaults) as $var) {
+ $$var = $args[$var];
+ }
+
+ $arr = apply_filters(
+ 'revisionary_main_post_statuses',
+ get_post_stati( ['public' => true, 'private' => true], $output, 'or' ),
+ $output
+ );
+
+ return ('csv' == $return) ? "'" . implode("','", $arr) . "'" : $arr;
+}
+
+// REST API Cache plugin compat
+add_action('init', 'rvy_rest_cache_compat', 9999);
+
+function rvy_rest_cache_compat() {
+ global $wp_post_types;
+
+ if (!isset($_SERVER['REQUEST_URI'])) {
+ return;
+ }
+
+ $uri = esc_url_raw($_SERVER['REQUEST_URI']);
+
+ $rest_cache_active = false;
+ foreach(['rvy_ajax_field', 'rvy_ajax_value'] as $param) {
+ if (strpos($uri, $param)) {
+ $rest_cache_active = true;
+ break;
+ }
+ }
+
+ $rest_cache_active = $rest_cache_active
+ || (strpos($uri, '_locale=user') && strpos($uri, 'wp-json') && strpos($uri, '/posts/') && rvy_is_plugin_active('wp-rest-cache/wp-rest-cache.php'));
+
+ if ($rest_cache_active) {
+ foreach(array_keys($wp_post_types) as $key) {
+ if ((!empty($wp_post_types[$key]->rest_controller_class) && is_string($wp_post_types[$key]->rest_controller_class)) && false !== strpos('WP_Rest_Cache_Plugin', $wp_post_types[$key]->rest_controller_class)) {
+ $wp_post_types[$key]->rest_controller_class = ('attachment' == $key) ? 'WP_REST_Attachments_Controller' : 'WP_REST_Posts_Controller';
+ }
+ }
+ }
+}
+
+// REST API Cache plugin compat
+add_filter('wp_rest_cache/skip_caching', 'rvy_rest_cache_skip');
+
+function rvy_rest_cache_skip($skip) {
+ if (!isset($_SERVER['REQUEST_URI'])) {
+ return;
+ }
+
+ $uri = esc_url_raw($_SERVER['REQUEST_URI']);
+ $uncached_params = array_merge($uncached_params, ['rvy_ajax_field', 'rvy_ajax_value']);
+
+ foreach($uncached_params as $param) {
+ if (strpos($uri, $param)) {
+ $skip = true;
+ break;
+ }
+ }
+
+ return $skip;
+}
+
+function pp_revisions_body_class() {
+ $classes = body_class();
+
+ if (function_exists('rvy_post_id')
+ && ('page' === get_option('show_on_front'))
+ && (rvy_post_id(get_the_ID()) == get_option('page_on_front'))) {
+ $classes []= 'home';
+ }
+
+ return $classes;
+}
\ No newline at end of file
diff --git a/wp-content/plugins/revisionary/rvy_init.php b/wp-content/plugins/revisionary/rvy_init.php
new file mode 100644
index 000000000..27f230013
--- /dev/null
+++ b/wp-content/plugins/revisionary/rvy_init.php
@@ -0,0 +1,156 @@
+ID)) {
+ $return_val = $post_id->ID;
+ }
+ } else {
+ $return_val = $post_id;
+ }
+ }
+
+ return $return_val;
+ }, 10, 2
+ );
+}
+
+if (defined('JREVIEWS_ROOT') && !empty($_REQUEST['preview'])
+&& ((empty($_REQUEST['preview_id']) && empty($_REQUEST['thumbnail_id']))
+|| (!empty($_REQUEST['preview_id']) && rvy_in_revision_workflow((int) $_REQUEST['preview_id']))
+)
+) {
+ require_once('compat_rvy.php');
+ _rvy_jreviews_preview_compat();
+}
+
+// Default early beta testers to same revision status labeling they are already using. They will be directly notified of the new setting.
+$last_ver = get_option('revisionary_last_version');
+
+if (version_compare($last_ver, '3.0-alpha', '>=') && version_compare($last_ver, '3.0-beta7', '<')) {
+ if (!get_option('pp_revisions_beta3_option_sync_done')) {
+ update_option('rvy_revision_statuses_noun_labels', 1);
+ update_option('pp_revisions_beta3_option_sync_done', 1);
+ }
+}
+
+// Revision Edit in Gutenberg: Enable non-Editors to set requested publish date
+add_action('init', function() {
+ global $revisionary;
+
+ foreach(array_keys($revisionary->enabled_post_types) as $post_type) {
+ add_filter("rest_prepare_{$post_type}", '_rvy_rest_prepare', 10, 3);
+ }
+}, 100);
+
+
+// Yoast SEO: Prevent invalid "indexable" maintenance operation on revision creation / submission
+if (defined('WPSEO_VERSION')) {
+ add_filter(
+ 'wpseo_should_save_indexable',
+ function($intend_to_save, $indexable) {
+ if (function_exists('rvy_detect_post_id')) {
+ $post_id = rvy_detect_post_id();
+
+ if ($post_id && rvy_in_revision_workflow($post_id)) {
+ return false;
+ }
+ }
+
+ if (is_object($indexable) && isset($indexable->object_id) && empty($indexable->object_id)) {
+ // WordPress database error Duplicate entry '0' for key 'PRIMARY' for query INSERT INTO `wp_yoast_indexable`
+ return false;
+ }
+
+ return $intend_to_save;
+ },
+ 10, 2);
+}
+
+// Prevent any default filters from screwing with our paging settings
+foreach(['revisions_per_page', 'revision_archive_per_page'] as $option_val) {
+ add_filter("set_screen_option_{$option_val}", function($screen_option, $option, $value ) {return $value;}, 99, 3);
+}
diff --git a/wp-content/plugins/revisionary/submittee_rvy.php b/wp-content/plugins/revisionary/submittee_rvy.php
new file mode 100644
index 000000000..6f63614de
--- /dev/null
+++ b/wp-content/plugins/revisionary/submittee_rvy.php
@@ -0,0 +1,103 @@
+update_page_options( $sitewide, $customize_defaults );
+
+ global $wpdb;
+ $wpdb->query( "UPDATE $wpdb->options SET autoload = 'no' WHERE (option_name LIKE 'rvy_%' OR option_name LIKE 'revisionary_%') AND option_name != 'rvy_next_rev_publish_gmt'" );
+ }
+
+ function default_options( $sitewide = false, $customize_defaults = false ) {
+ check_admin_referer( 'rvy-update-options' );
+
+ $default_prefix = ( $customize_defaults ) ? 'default_' : '';
+
+ if (!empty($_POST['all_options'])) {
+ $reviewed_options = array_map('sanitize_key', explode(',', sanitize_text_field($_POST['all_options'])));
+ foreach ( $reviewed_options as $option_name ) {
+ rvy_delete_option($default_prefix . $option_name, $sitewide );
+ }
+ }
+ }
+
+ function update_sitewide() {
+ check_admin_referer( 'rvy-update-options' );
+
+ $reviewed_options = isset($_POST['rvy_all_movable_options']) ? array_map('sanitize_key', explode(',', $_POST['rvy_all_movable_options'])) : array();
+
+ $options_sitewide = isset($_POST['rvy_options_sitewide']) ? array_map('sanitize_key', (array) $_POST['rvy_options_sitewide']) : array();
+
+ update_site_option( "rvy_options_sitewide_reviewed", $reviewed_options );
+ update_site_option( "rvy_options_sitewide", $options_sitewide );
+ }
+
+ function default_sitewide() {
+ check_admin_referer( 'rvy-update-options' );
+
+ rvy_delete_option( 'options_sitewide', true );
+ rvy_delete_option( 'options_sitewide_reviewed', true );
+ }
+
+ function update_page_options( $sitewide = false, $customize_defaults = false ) {
+ $default_prefix = ( $customize_defaults ) ? 'default_' : '';
+
+ if (!empty($_POST['all_options'])) {
+ $reviewed_options = array_map('sanitize_key', explode(',', sanitize_text_field($_POST['all_options'])));
+
+ foreach ( $reviewed_options as $option_basename ) {
+ if (isset($_POST[$option_basename])) {
+ if (is_array($_POST[$option_basename])) {
+ $value = array_map('sanitize_key', $_POST[$option_basename]);
+ } else {
+ $value = sanitize_key($_POST[$option_basename]);
+ }
+ } else {
+ $value = '';
+ }
+
+ rvy_update_option( $default_prefix . $option_basename, $value, $sitewide );
+ }
+ }
+ }
+}
diff --git a/wp-content/plugins/revisionary/utils.php b/wp-content/plugins/revisionary/utils.php
new file mode 100644
index 000000000..f9a6d4458
--- /dev/null
+++ b/wp-content/plugins/revisionary/utils.php
@@ -0,0 +1,223 @@
+=') || substr($wp_version, 0, 2) === '5.';
+ }
+
+ /**
+ * Based on Edit Flow's \Block_Editor_Compatible::should_apply_compat method.
+ *
+ * @return bool
+ */
+ public static function isBlockEditorActive($postType = false) {
+ global $wp_version;
+
+ // Check if PP Custom Post Statuses lower than v2.4 is installed. It disables Gutenberg.
+ if ( defined('PPS_VERSION') && version_compare(PPS_VERSION, '2.4-beta', '<') ) {
+ return false;
+ }
+
+ if (class_exists('Classic_Editor')) {
+ if (isset($_REQUEST['classic-editor__forget']) && (isset($_REQUEST['classic']) || isset($_REQUEST['classic-editor']))) {
+ return false;
+ } elseif (isset($_REQUEST['classic-editor__forget']) && !isset($_REQUEST['classic']) && !isset($_REQUEST['classic-editor'])) {
+ return true;
+ } elseif (get_option('classic-editor-allow-users') === 'allow') {
+ if ($post_id = rvy_detect_post_id()) {
+ $which = get_post_meta( $post_id, 'classic-editor-remember', true );
+
+ if ('block-editor' == $which) {
+ return true;
+ } elseif ('classic-editor' == $which) {
+ return false;
+ }
+ }
+ }
+ }
+
+ $pluginsState = array(
+ 'classic-editor' => class_exists( 'Classic_Editor' ),
+ 'gutenberg' => function_exists( 'the_gutenberg_project' ),
+ 'gutenberg-ramp' => class_exists('Gutenberg_Ramp'),
+ );
+
+ if (!$postType) {
+ if ( ! $postType = rvy_detect_post_type() ) {
+ $postType = 'page';
+ }
+ }
+
+ if ( $post_type_obj = get_post_type_object( $postType ) ) {
+ if ( empty( $post_type_obj->show_in_rest ) ) {
+ return false;
+ }
+ }
+
+ $conditions = array();
+
+ /**
+ * 5.0:
+ *
+ * Classic editor either disabled or enabled (either via an option or with GET argument).
+ * It's a hairy conditional :(
+ */
+
+ if (version_compare($wp_version, '5.9-beta', '>=')) {
+ remove_action('use_block_editor_for_post_type', '_disable_block_editor_for_navigation_post_type', 10, 2);
+ remove_filter('use_block_editor_for_post_type', '_disable_block_editor_for_navigation_post_type', 10, 2);
+ }
+
+ // Divi: Classic Editor option
+ if (function_exists('et_get_option') && ( 'on' == et_get_option( 'et_enable_classic_editor', 'off' ))) {
+ return false;
+ }
+
+ // phpcs:ignore WordPress.VIP.SuperGlobalInputUsage.AccessDetected, WordPress.Security.NonceVerification.NoNonceVerification
+
+ $_post = get_post(rvy_detect_post_id());
+
+ if (!empty($_post)) {
+ $conditions[] = (self::isWp5() || $pluginsState['gutenberg'])
+ && ! $pluginsState['classic-editor']
+ && ! $pluginsState['gutenberg-ramp']
+ && apply_filters('use_block_editor_for_post_type', true, $postType)
+ && apply_filters('use_block_editor_for_post', true, $_post);
+ }
+
+ $conditions[] = self::isWp5()
+ && $pluginsState['classic-editor']
+ && (get_option('classic-editor-replace') === 'block'
+ && ! isset($_GET['classic-editor__forget']));
+
+ $conditions[] = self::isWp5()
+ && $pluginsState['classic-editor']
+ && (get_option('classic-editor-replace') === 'classic'
+ && isset($_GET['classic-editor__forget']));
+
+ if (!empty($_post)) {
+ $conditions[] = $pluginsState['gutenberg-ramp']
+ && apply_filters('use_block_editor_for_post', true, $_post);
+ }
+
+ if (defined('PP_CAPABILITIES_RESTORE_NAV_TYPE_BLOCK_EDITOR_DISABLE') && version_compare($wp_version, '5.9-beta', '>=')) {
+ add_filter('use_block_editor_for_post_type', '_disable_block_editor_for_navigation_post_type', 10, 2 );
+ }
+
+ // Returns true if at least one condition is true.
+ return count(
+ array_filter($conditions,
+ function ($c) {
+ return (bool)$c;
+ }
+ )
+ ) > 0;
+ }
+
+ /**
+ * Adds slashes only to strings.
+ *
+ * @param mixed $value Value to slash only if string.
+ *
+ * @return string|mixed
+ */
+ public static function addslashes_to_strings_only( $value ) {
+ return \is_string( $value ) ? \addslashes( $value ) : $value;
+ }
+
+ /**
+ * Replaces faulty core wp_slash().
+ *
+ * Until WP 5.5 wp_slash() recursively added slashes not just to strings in array/objects, leading to errors.
+ *
+ * @param mixed $value What to add slashes to.
+ *
+ * @return mixed
+ */
+ public static function recursively_slash_strings( $value ) {
+ return self::map_deep( $value, [ self::class, 'addslashes_to_strings_only' ] );
+ }
+
+ public static function map_deep( $value, $callback ) {
+ if ( is_array( $value ) ) {
+ foreach ( $value as $index => $item ) {
+ $value[ $index ] = self::map_deep( $item, $callback );
+ }
+ } elseif ( is_object( $value ) ) {
+ if ( get_class($value) === "__PHP_Incomplete_Class" ) {
+ return $value;
+ }
+
+ $object_vars = get_object_vars( $value );
+ foreach ( $object_vars as $property_name => $property_value ) {
+ $value->$property_name = self::map_deep( $property_value, $callback );
+ }
+ } else {
+ $value = call_user_func( $callback, $value );
+ }
+
+ return $value;
+ }
+
+ public static function get_post_autosave($post_id, $user_id) {
+ global $wpdb;
+
+ $autosave = $wpdb->get_row(
+ $wpdb->prepare(
+ "SELECT *
+ FROM $wpdb->posts
+ WHERE post_parent = %d
+ AND post_type = 'revision'
+ AND post_status = 'inherit'
+ AND post_name LIKE '%" . intval($post_id) . "-autosave%'
+ AND post_author = %d
+ ORDER BY post_date DESC
+ LIMIT 1",
+ $post_id,
+ $user_id
+ )
+ );
+
+ if ( ! $autosave ) {
+ return false;
+ }
+
+ return get_post($autosave);
+ }
+}
+
diff --git a/wp-content/plugins/tw-episode-importer/api/api.php b/wp-content/plugins/tw-episode-importer/api/api.php
index d51acf57e..fa336de5e 100644
--- a/wp-content/plugins/tw-episode-importer/api/api.php
+++ b/wp-content/plugins/tw-episode-importer/api/api.php
@@ -768,7 +768,7 @@ function ( $p ) use ( $post_type ) {
return $p['type'] === $post_type;
}
) : null;
- $post = is_array( $post ) && ! empty( $post ) ? $post[0] : null;
+ $post = is_array( $post ) && ! empty( $post ) ? array_shift( $post ) : null;
$was_imported = is_array( $post ) && isset( $post['guid'] ) ? $post['guid'] === $guid : false;
$audio_is_different = is_array( $audio ) && isset( $audio['url'] ) && $api_item->_links->enclosure->href !== $audio['url'];
$has_updated_audio = is_array( $audio ) &&
@@ -931,7 +931,7 @@ function tw_episode_importer_get_existing_post_data( $post_type, $guid, $audio_k
// Attempt to get imported audio by guid.
if ( is_null( $audio_post ) ) {
- $audio_id = $wpdb->get_var( $wpdb->prepare( "SELECT ID FROM $wpdb->posts WHERE guid=%s", "http://{$guid}" ) );
+ $audio_id = $wpdb->get_var( $wpdb->prepare( "SELECT ID FROM $wpdb->posts WHERE guid IN (%s, %s)", array( "{$guid}", "http://{$guid}" ) ) );
$audio_post = get_post( $audio_id );
}
diff --git a/wp-content/plugins/tw-previews/preview/ui.php b/wp-content/plugins/tw-previews/preview/ui.php
index ddb42e025..1b93d4caa 100644
--- a/wp-content/plugins/tw-previews/preview/ui.php
+++ b/wp-content/plugins/tw-previews/preview/ui.php
@@ -85,6 +85,7 @@ function tw_previews_preview_post_link( $preview_link, $post ) {
}
}
add_filter( 'preview_post_link', 'tw_previews_preview_post_link', 100000, 2 );
+add_filter( 'revisionary_preview_url', 'tw_previews_preview_post_link', 100000, 2 );
/**
* Render HTML and enqueue scripts for admin page.
diff --git a/wp-content/themes/the-world/functions.php b/wp-content/themes/the-world/functions.php
index 462111d6f..321b9e72f 100644
--- a/wp-content/themes/the-world/functions.php
+++ b/wp-content/themes/the-world/functions.php
@@ -71,8 +71,9 @@ function tw_admin_styles() {
// Hides device options from preview menu dropdown. They are basically useless and confusing.
echo '';
}
endif;
@@ -80,8 +81,7 @@ function tw_admin_styles() {
if ( ! function_exists( 'tw_admin_init_editor_styles' ) ) {
/**
- * Set cookie to store auth token. SHould be an http cookies.
- * Assume frontend will be served from the same domain.
+ * Add editor styles.
*
* @return void
*/