-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathrevision-notes.php
209 lines (168 loc) · 6.53 KB
/
revision-notes.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
<?php
/**
* Plugin Name: Revision Notes
* Plugin URI: http://wordpress.org/plugins/revision-notes/
* Description: Add a note explaining the changes you're about to save. It's like commit messages, except for your WordPress content.
* Version: 2.0
* Author: Helen Hou-Sandí
* Author URI: https://helen.blog/
* Text Domain: revision-notes
* License: GPLv2
* License URI: https://www.gnu.org/licenses/gpl-2.0.html
*/
defined( 'WPINC' ) or die;
// Cheapo namespace; this isn't a real object or very testable (right now).
class HHS_Revision_Notes {
public function __construct() {
add_action( 'init', array( $this, 'init' ), 99 );
}
public function init() {
add_action( 'post_submitbox_misc_actions', array( $this, 'edit_field' ) );
add_action( 'save_post', array( $this, 'save_post' ), 10, 2 );
// As of 5.6, revisions are created too early for REST requests
// so move the revision saving to after the post is fully inserted including meta.
// This is fine for the classic editor too.
// For future-proofing and compat, check to make sure it's hooked on already.
if ( has_action( 'post_updated', 'wp_save_post_revision' ) ) {
remove_action( 'post_updated', 'wp_save_post_revision', 10, 1 );
add_action( 'wp_after_insert_post', array( $this, 'maybe_save_revision' ), 10, 3 );
}
add_filter( 'wp_prepare_revision_for_js', array( $this, 'wp_prepare_revision_for_js' ), 10, 2 );
add_filter( 'wp_post_revision_title_expanded', array( $this, 'wp_post_revision_title_expanded' ), 10, 2 );
// Use post_type_supports() to make showing/hiding of the field easy for devs.
// By default we'll show it for any post type that has an edit UI.
$post_types = get_post_types( array( 'show_ui' => true ) );
if ( ! empty( $post_types ) ) {
foreach ( $post_types as $post_type ) {
add_post_type_support( $post_type, 'revision-notes' );
add_action( "manage_edit-{$post_type}_columns", array( $this, 'add_column' ) );
add_action( "manage_{$post_type}_posts_custom_column", array( $this, 'show_column' ), 10, 2 );
register_rest_field(
$post_type,
'revision_note',
array(
'get_callback' => '__return_empty_string',
'update_callback' => function ( $value, $post ) {
update_post_meta( $post->ID, 'revision_note', $value );
},
'schema' => array(
'type' => 'string',
),
)
);
}
}
add_action( 'enqueue_block_editor_assets', array( $this, 'block_editor_assets' ) );
}
public function edit_field() {
$post = get_post();
if ( ! $post ) {
return;
}
if ( ! post_type_supports( $post->post_type, 'revision-notes' ) ) {
return;
}
wp_nonce_field( 'hhs-revision-notes-save', 'hhs_revision_notes_nonce' );
?>
<div class="misc-pub-section revision-note">
<label><?php _e( 'Revision note (optional)', 'revision-notes' ); ?>
<input name="hhs_revision_note" type="text" class="widefat" maxlength="100" />
</label>
<p class="description"><?php _e( 'Enter a brief note about this change', 'revision-notes' ); ?></p>
</div>
<?php
}
public function save_post( $post_id, $post ) {
// Classic editor handling of $_POST value from metabox
// Technically the else case would also handle revisions in classic editor,
// but let's just leave this as-is for now.
if ( ! empty( $_POST['hhs_revision_note'] ) ) {
// verify nonce
if ( ! isset( $_POST['hhs_revision_notes_nonce'] ) ||
! wp_verify_nonce( $_POST['hhs_revision_notes_nonce'], 'hhs-revision-notes-save' )
) {
return;
}
if ( defined( 'DOING_AUTOSAVE' ) && DOING_AUTOSAVE ) {
return;
}
if ( ! current_user_can( 'edit_post', $post_id ) ) {
return;
}
$note = wp_strip_all_tags( $_POST['hhs_revision_note'] );
// Save the note as meta on the revision itself.
// save_post actually runs a second time on the parent post,
// so it will also be stored as the latest note in the parent post's meta.
update_metadata( 'post', $post_id, 'revision_note', $note );
} elseif ( 'revision' === get_post_type( $post ) ) {
// Block editor/non-nonce handling
// Regular post meta is handled via REST API, but we still have to do the revision storage.
// No cap checking, but it's coming from the post itself anyway,
// so the biggest potential issue is if revisions are being created in other instances,
// they might end up with duplicate / inaccurate revision notes.
// This relies on the revision being saved after the parent as specified by core!
$parent = wp_is_post_revision( $post );
$note = get_post_meta( $parent, 'revision_note', true );
if ( ! empty( $note ) ) {
update_metadata( 'post', $post->ID, 'revision_note', wp_slash( $note ) );
}
}
}
public function maybe_save_revision( $post_id, $post, $update ) {
if ( ! $update ) {
return;
}
wp_save_post_revision( $post_id );
}
public function wp_prepare_revision_for_js( $data, $revision ) {
$note = esc_html( get_metadata( 'post', $revision->ID, 'revision_note', true ) );
if ( ! empty( $note ) ) {
/* Translators: 1: revision note; 2: time ago; */
$data['timeAgo'] = sprintf( __( 'Note: %1$s - %2$s', 'revision-notes' ), $note, $data['timeAgo'] );
}
return $data;
}
public function wp_post_revision_title_expanded( $text, $revision ) {
// Some safeguards in case this is being called by somebody else for something else.
// We may want to do these checks elsewhere so that this function can still be used
// in other contexts should a developer want to do so.
if ( ! is_admin() ) {
return $text;
}
$screen = get_current_screen();
if ( 'post' !== $screen->base ) {
return $text;
}
$note = get_metadata( 'post', $revision->ID, 'revision_note', true );
if ( ! empty( $note ) ) {
$text .= ' — <em>' . esc_html( $note ) . '</em>';
}
return $text;
}
public function add_column( $columns ) {
$columns['revision_note'] = __( 'Latest Revision Note', 'revision-notes' );
return $columns;
}
public function show_column( $column, $post_id ) {
if ( 'revision_note' !== $column ) {
return;
}
$note = get_post_meta( $post_id, 'revision_note', true );
if ( empty( $note ) ) {
return;
}
echo esc_html( $note );
}
public function block_editor_assets() {
$script_asset = require 'build/index.asset.php';
wp_enqueue_script(
'revision-notes-block-editor',
plugin_dir_url( __FILE__ ) . 'build/index.js',
$script_asset['dependencies'],
$script_asset['version']
);
}
}
// This is a global so it can be accessed by others for things like hooks.
// It could be a singleton too, I guess.
$hhs_revision_notes = new HHS_Revision_Notes();