forked from beezwax/WP-Publish-to-Apple-News
-
Notifications
You must be signed in to change notification settings - Fork 71
/
Copy pathclass-admin-apple-notice.php
478 lines (417 loc) · 13.5 KB
/
class-admin-apple-notice.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
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
<?php
/**
* Publish to Apple News Admin: Admin_Apple_Notice class
*
* @package Apple_News
*/
/**
* A class to manage Publish to Apple News notices in the WP admin.
*
* @since 0.6.0
*/
class Admin_Apple_Notice {
/**
* Key for admin notices.
*
* @access public
*/
const KEY = 'apple_news_notice';
/**
* Clear one or more admin notices.
*
* Accepts an array of notification objects to clear. First performs a
* check to ensure that the notification objects match, then clears them
* according to the following rules:
*
* - If a notification is set to be dismissible, sets the dismissed flag
* to true.
* - If a notification is not set to be dismissible, removes the
* notification entirely.
*
* @access public
*
* @param array $notifications Notification objects to clear.
* @return bool True if notifications were cleared, false if they were not.
*/
public static function clear( $notifications ) {
// Ensure a user is logged in before proceeding.
$current_user = get_current_user_id();
if ( empty( $current_user ) ) {
return false;
}
// Ensure we were given a non-empty array of notifications to clear.
if ( empty( $notifications ) || ! is_array( $notifications ) ) {
return false;
}
// Ensure the user has notices to compare to.
$notices = self::get_user_meta( $current_user );
if ( empty( $notices ) || ! is_array( $notices ) ) {
return false;
}
// Sort and JSON-encode the removal array to simplify comparison.
$notifications = array_map(
function ( $value ) {
ksort( $value );
return wp_json_encode( $value );
},
$notifications
);
// Build an updated array of notices.
$updated = false;
$notices = array_filter(
array_map(
function ( $notice ) use ( $notifications, &$updated ) { // phpcs:ignore WordPressVIPMinimum.Variables.VariableAnalysis.UnusedVariable
ksort( $notice );
if ( in_array( wp_json_encode( $notice ), $notifications, true ) ) {
$updated = true;
if ( ! empty( $notice['dismissible'] ) && true === $notice['dismissible'] ) {
$notice['dismissed'] = true;
} else {
return null;
}
}
return $notice;
},
$notices
)
);
// If there are no changes, bail out.
if ( ! $updated ) {
return false;
}
return false !== self::update_user_meta( $current_user, $notices );
}
/**
* Add an error message.
*
* @param string $message The message to be displayed.
* @param int $user_id The user ID for which to display the message.
* @param bool $dismissible Whether the message is dismissible (dismissed state stored in DB).
* @access public
*/
public static function error( $message, $user_id = null, $dismissible = false ) {
self::message( $message, 'error', $user_id, $dismissible );
}
/**
* Get the admin notice(s).
*
* @access public
*/
public static function get() {
$notices = self::get_user_meta( get_current_user_id() );
return ( ! empty( $notices ) && is_array( $notices ) ) ? $notices : [];
}
/**
* Check if any notices exist for the current user.
*
* @access public
* @return bool True if the user has notices, false otherwise.
*/
public static function has_notice() {
$messages = self::get_user_meta( get_current_user_id() );
return ! empty( $messages );
}
/**
* Add an info message.
*
* @param string $message The message to be displayed.
* @param int $user_id The user ID for which to display the message.
* @param bool $dismissible Whether the message is dismissible (dismissed state stored in DB).
* @access public
*/
public static function info( $message, $user_id = null, $dismissible = false ) {
self::message( $message, 'warning', $user_id, $dismissible );
}
/**
* Add a notice message to be displayed.
*
* @param string $message The message to be displayed.
* @param string $type The type of message to display.
* @param int $user_id The user ID for which to display the message.
* @param bool $dismissible Whether the message is dismissible (dismissed state stored in DB).
* @access public
*/
public static function message( $message, $type, $user_id = null, $dismissible = false ) {
// Default to the current user, if no ID was specified.
if ( empty( $user_id ) ) {
$user_id = get_current_user_id();
}
// Sanitize values.
$message = wp_kses_post( $message );
$type = sanitize_text_field( $type );
// Pull usermeta and see if the message already exists.
$messages = self::get_user_meta( $user_id );
if ( ! empty( $messages ) && is_array( $messages ) ) {
foreach ( $messages as $message_check ) {
if ( ! empty( $message_check['message'] )
&& $message === $message_check['message']
&& ! empty( $message_check['type'] )
&& $type === $message_check['type']
) {
return;
}
}
}
// Add the message to usermeta for later display.
self::add_user_meta(
$user_id,
[
'dismissible' => $dismissible,
'dismissed' => false,
'message' => $message,
'type' => $type,
'timestamp' => time(),
]
);
}
/**
* Registers assets used by meta boxes.
*
* @access public
*/
public static function register_assets() {
// Enqueue notice script.
wp_enqueue_script(
'apple_news_notices_js',
plugin_dir_url( __FILE__ ) . '../assets/js/notices.js',
[],
Apple_News::$version,
true
);
}
/**
* Show the admin notice(s).
*
* @access public
*/
public static function show() {
// If we are on a single page and the block editor is in use, don't trigger this functionality.
if ( apple_news_block_editor_is_active_for_post() ) {
return;
}
// Check for notices.
$notices = self::get_user_meta( get_current_user_id() );
if ( empty( $notices ) || ! is_array( $notices ) ) {
return;
}
// Keep track of an updated list of notices to save to the DB, if necessary.
$updated_notices = [];
// Show the notices.
foreach ( $notices as $notice ) {
// If the notice doesn't have a message (for some reason), skip it.
if ( empty( $notice['message'] ) ) {
continue;
}
// If a type isn't specified, default to 'updated'.
$type = isset( $notice['type'] ) ? $notice['type'] : 'updated';
// Only display notices that aren't dismissed.
if ( empty( $notice['dismissed'] ) ) {
self::show_notice( $notice['message'], $type );
}
// If the notice is dismissible, ensure it persists in the DB.
if ( ! empty( $notice['dismissible'] ) ) {
$updated_notices[] = $notice;
}
}
// Delete the notices in the DB if they have changed.
if ( $notices !== $updated_notices ) {
self::update_user_meta( get_current_user_id(), $updated_notices );
}
}
/**
* Add a success message.
*
* @param string $message The message to be displayed.
* @param int $user_id The user ID for which to display the message.
* @param bool $dismissible Whether the message is dismissible (dismissed state stored in DB).
* @access public
*/
public static function success( $message, $user_id = null, $dismissible = false ) {
self::message( $message, 'success', $user_id, $dismissible );
}
/**
* A WP-AJAX handler for a click on a dismiss notice button.
*/
public static function wp_ajax_dismiss_notice() {
// Check the nonce.
check_admin_referer( 'apple_news_dismiss_notice', 'nonce' );
// Safely extract message and type for comparison.
$message = isset( $_POST['message'] )
? wp_kses_post( wp_unslash( $_POST['message'] ) )
: '';
$type = isset( $_POST['type'] )
? wp_kses_post( wp_unslash( $_POST['type'] ) )
: '';
// Pull the list of notices for this user.
$update = false;
$notices = self::get_user_meta( get_current_user_id() );
foreach ( $notices as &$notice ) {
if ( isset( $notice['message'] )
&& $message === $notice['message']
&& isset( $notice['type'] )
&& $type === $notice['type']
) {
$notice['dismissed'] = true;
$update = true;
}
}
// Save the updated notice config to the database.
if ( $update ) {
self::update_user_meta( get_current_user_id(), $notices );
}
}
/**
* Handle adding user meta across potential hosting platforms.
*
* @param int $user_id The user ID to use when adding meta.
* @param array $value The value to add.
* @access private
* @return int|bool Meta ID if new, true on update, false otherwise.
*/
private static function add_user_meta( $user_id, $value ) {
// We can't use add_user_meta because there is no equivalent on VIP.
// Instead manage values within the same variable for consistency.
$values = self::get_user_meta( $user_id );
if ( empty( $values ) ) {
$values = [];
}
// Add the new value.
$values[] = $value;
// Save using the appropriate method.
if ( defined( 'WPCOM_IS_VIP_ENV' ) && true === WPCOM_IS_VIP_ENV ) {
return update_user_attribute( $user_id, self::KEY, $values );
} else {
return update_user_meta( $user_id, self::KEY, $values ); // phpcs:ignore WordPress.VIP.RestrictedFunctions.user_meta_update_user_meta
}
}
/**
* Handle deleting user meta across potential hosting platforms.
*
* @param int $user_id The user ID for which to delete meta.
* @access private
* @return bool True on success, false on failure.
*/
private static function delete_user_meta( $user_id ) {
if ( defined( 'WPCOM_IS_VIP_ENV' ) && true === WPCOM_IS_VIP_ENV ) {
return delete_user_attribute( $user_id, self::KEY );
} else {
return delete_user_meta( $user_id, self::KEY ); // phpcs:ignore WordPress.VIP.RestrictedFunctions.user_meta_delete_user_meta
}
}
/**
* Handle getting user meta across potential hosting platforms.
*
* @param int $user_id The user ID for which to retrieve meta.
* @access private
* @return array An array of values for the key.
*/
private static function get_user_meta( $user_id ) {
// Negotiate meta value.
if ( defined( 'WPCOM_IS_VIP_ENV' ) && true === WPCOM_IS_VIP_ENV && ! defined( 'VIP_GO_APP_ENVIRONMENT' ) && function_exists( 'get_user_attribute' ) ) {
$meta_value = get_user_attribute( $user_id, self::KEY );
} else {
$meta_value = get_user_meta( $user_id, self::KEY, true ); // phpcs:ignore WordPress.VIP.RestrictedFunctions.user_meta_get_user_meta
}
return ( ! empty( $meta_value ) ) ? $meta_value : [];
}
/**
* Display the admin notice template.
*
* @param string $message The message to display.
* @param string $type The message type.
* @access private
*/
private static function show_notice( $message, $type ) {
// Format messages a little nicer.
$message = str_replace( '|', '<br />', $message );
$message_array = explode( ':', $message );
if ( 2 === count( $message_array ) ) {
/**
* If it's not 2, it's too unclear how to proceed.
* Try to split the second param on commas.
*/
$errors = explode( ',', $message_array[1] );
if ( count( $errors ) > 1 ) {
// If there isn't more than one error, this isn't worth it.
$errors_formatted = implode( '<br />', array_map( 'trim', $errors ) );
$message = sprintf(
'%s:<br />%s',
$message_array[0],
$errors_formatted
);
}
}
// Add the support tagline to errors.
if ( 'error' === $type ) {
$message .= Apple_News::get_support_info();
}
/**
* Allows you to change any success, info, or error notice that appears in
* the dashboard.
*
* @param string $message The message to be displayed.
* @param string $type The type of message being displayed.
*/
$message = apply_filters( 'apple_news_notice_message', $message, $type );
// Load the partial for the notice.
include __DIR__ . '/partials/notice.php';
}
/**
* Handle updating user meta across potential hosting platforms.
*
* @param int $user_id The user ID to use when updating meta.
* @param array $values The new values to use.
* @access private
* @return int|bool Meta ID if new, true on update, false otherwise.
*/
private static function update_user_meta( $user_id, $values ) {
// Delete existing user meta.
self::delete_user_meta( $user_id );
// Only add meta back if new values were provided.
if ( ! empty( $values ) ) {
// Save using the appropriate method.
if ( defined( 'WPCOM_IS_VIP_ENV' ) && true === WPCOM_IS_VIP_ENV ) {
return update_user_attribute( $user_id, self::KEY, $values );
} else {
return update_user_meta( $user_id, self::KEY, $values ); // phpcs:ignore WordPress.VIP.RestrictedFunctions.user_meta_update_user_meta
}
}
return true;
}
/**
* Checks if the current user has access to see the notifications and returns them if true.
*
* @param array $object The requested object.
* @return array
*/
public static function get_if_allowed( $object ) {
$notifications = [];
// Ensure we have an object type.
if ( empty( $object['type'] ) ) {
return $notifications;
}
// This functionality is only available to logged in users.
if ( ! is_user_logged_in() ) {
return $notifications;
}
// Ensure current user has the appropriate publish permission.
if ( ! current_user_can(
/** This filter is documented in admin/class-admin-apple-post-sync.php */
apply_filters( 'apple_news_publish_capability', Apple_News::get_capability_for_post_type( 'publish_posts', $object['type'] ) )
) ) {
return $notifications;
}
// Get notifications.
$notifications = self::get();
self::clear( $notifications );
$notifications = array_values(
array_filter(
$notifications,
function ( $notification ) {
return empty( $notification['dismissible'] );
}
)
);
return $notifications;
}
}