[ Index ] |
PHP Cross Reference of WordPress Trunk (Updated Daily) |
[Summary view] [Print] [Text view]
1 /** 2 * Functions for ajaxified updates, deletions and installs inside the WordPress admin. 3 * 4 * @version 4.2.0 5 * @output wp-admin/js/updates.js 6 */ 7 8 /* global pagenow, _wpThemeSettings */ 9 10 /** 11 * @param {jQuery} $ jQuery object. 12 * @param {object} wp WP object. 13 * @param {object} settings WP Updates settings. 14 * @param {string} settings.ajax_nonce Ajax nonce. 15 * @param {object=} settings.plugins Base names of plugins in their different states. 16 * @param {Array} settings.plugins.all Base names of all plugins. 17 * @param {Array} settings.plugins.active Base names of active plugins. 18 * @param {Array} settings.plugins.inactive Base names of inactive plugins. 19 * @param {Array} settings.plugins.upgrade Base names of plugins with updates available. 20 * @param {Array} settings.plugins.recently_activated Base names of recently activated plugins. 21 * @param {Array} settings.plugins['auto-update-enabled'] Base names of plugins set to auto-update. 22 * @param {Array} settings.plugins['auto-update-disabled'] Base names of plugins set to not auto-update. 23 * @param {object=} settings.themes Slugs of themes in their different states. 24 * @param {Array} settings.themes.all Slugs of all themes. 25 * @param {Array} settings.themes.upgrade Slugs of themes with updates available. 26 * @param {Arrat} settings.themes.disabled Slugs of disabled themes. 27 * @param {Array} settings.themes['auto-update-enabled'] Slugs of themes set to auto-update. 28 * @param {Array} settings.themes['auto-update-disabled'] Slugs of themes set to not auto-update. 29 * @param {object=} settings.totals Combined information for available update counts. 30 * @param {number} settings.totals.count Holds the amount of available updates. 31 */ 32 (function( $, wp, settings ) { 33 var $document = $( document ), 34 __ = wp.i18n.__, 35 _x = wp.i18n._x, 36 _n = wp.i18n._n, 37 _nx = wp.i18n._nx, 38 sprintf = wp.i18n.sprintf; 39 40 wp = wp || {}; 41 42 /** 43 * The WP Updates object. 44 * 45 * @since 4.2.0 46 * 47 * @namespace wp.updates 48 */ 49 wp.updates = {}; 50 51 /** 52 * Removed in 5.5.0, needed for back-compatibility. 53 * 54 * @since 4.2.0 55 * @deprecated 5.5.0 56 * 57 * @type {object} 58 */ 59 wp.updates.l10n = { 60 searchResults: '', 61 searchResultsLabel: '', 62 noPlugins: '', 63 noItemsSelected: '', 64 updating: '', 65 pluginUpdated: '', 66 themeUpdated: '', 67 update: '', 68 updateNow: '', 69 pluginUpdateNowLabel: '', 70 updateFailedShort: '', 71 updateFailed: '', 72 pluginUpdatingLabel: '', 73 pluginUpdatedLabel: '', 74 pluginUpdateFailedLabel: '', 75 updatingMsg: '', 76 updatedMsg: '', 77 updateCancel: '', 78 beforeunload: '', 79 installNow: '', 80 pluginInstallNowLabel: '', 81 installing: '', 82 pluginInstalled: '', 83 themeInstalled: '', 84 installFailedShort: '', 85 installFailed: '', 86 pluginInstallingLabel: '', 87 themeInstallingLabel: '', 88 pluginInstalledLabel: '', 89 themeInstalledLabel: '', 90 pluginInstallFailedLabel: '', 91 themeInstallFailedLabel: '', 92 installingMsg: '', 93 installedMsg: '', 94 importerInstalledMsg: '', 95 aysDelete: '', 96 aysDeleteUninstall: '', 97 aysBulkDelete: '', 98 aysBulkDeleteThemes: '', 99 deleting: '', 100 deleteFailed: '', 101 pluginDeleted: '', 102 themeDeleted: '', 103 livePreview: '', 104 activatePlugin: '', 105 activateTheme: '', 106 activatePluginLabel: '', 107 activateThemeLabel: '', 108 activateImporter: '', 109 activateImporterLabel: '', 110 unknownError: '', 111 connectionError: '', 112 nonceError: '', 113 pluginsFound: '', 114 noPluginsFound: '', 115 autoUpdatesEnable: '', 116 autoUpdatesEnabling: '', 117 autoUpdatesEnabled: '', 118 autoUpdatesDisable: '', 119 autoUpdatesDisabling: '', 120 autoUpdatesDisabled: '', 121 autoUpdatesError: '' 122 }; 123 124 wp.updates.l10n = window.wp.deprecateL10nObject( 'wp.updates.l10n', wp.updates.l10n, '5.5.0' ); 125 126 /** 127 * User nonce for ajax calls. 128 * 129 * @since 4.2.0 130 * 131 * @type {string} 132 */ 133 wp.updates.ajaxNonce = settings.ajax_nonce; 134 135 /** 136 * Current search term. 137 * 138 * @since 4.6.0 139 * 140 * @type {string} 141 */ 142 wp.updates.searchTerm = ''; 143 144 /** 145 * Minimum number of characters before an ajax search is fired. 146 * 147 * @since 6.7.0 148 * 149 * @type {number} 150 */ 151 wp.updates.searchMinCharacters = 2; 152 153 /** 154 * Whether filesystem credentials need to be requested from the user. 155 * 156 * @since 4.2.0 157 * 158 * @type {bool} 159 */ 160 wp.updates.shouldRequestFilesystemCredentials = false; 161 162 /** 163 * Filesystem credentials to be packaged along with the request. 164 * 165 * @since 4.2.0 166 * @since 4.6.0 Added `available` property to indicate whether credentials have been provided. 167 * 168 * @type {Object} 169 * @property {Object} filesystemCredentials.ftp Holds FTP credentials. 170 * @property {string} filesystemCredentials.ftp.host FTP host. Default empty string. 171 * @property {string} filesystemCredentials.ftp.username FTP user name. Default empty string. 172 * @property {string} filesystemCredentials.ftp.password FTP password. Default empty string. 173 * @property {string} filesystemCredentials.ftp.connectionType Type of FTP connection. 'ssh', 'ftp', or 'ftps'. 174 * Default empty string. 175 * @property {Object} filesystemCredentials.ssh Holds SSH credentials. 176 * @property {string} filesystemCredentials.ssh.publicKey The public key. Default empty string. 177 * @property {string} filesystemCredentials.ssh.privateKey The private key. Default empty string. 178 * @property {string} filesystemCredentials.fsNonce Filesystem credentials form nonce. 179 * @property {bool} filesystemCredentials.available Whether filesystem credentials have been provided. 180 * Default 'false'. 181 */ 182 wp.updates.filesystemCredentials = { 183 ftp: { 184 host: '', 185 username: '', 186 password: '', 187 connectionType: '' 188 }, 189 ssh: { 190 publicKey: '', 191 privateKey: '' 192 }, 193 fsNonce: '', 194 available: false 195 }; 196 197 /** 198 * Whether we're waiting for an Ajax request to complete. 199 * 200 * @since 4.2.0 201 * @since 4.6.0 More accurately named `ajaxLocked`. 202 * 203 * @type {bool} 204 */ 205 wp.updates.ajaxLocked = false; 206 207 /** 208 * Admin notice template. 209 * 210 * @since 4.6.0 211 * 212 * @type {function} 213 */ 214 wp.updates.adminNotice = wp.template( 'wp-updates-admin-notice' ); 215 216 /** 217 * Update queue. 218 * 219 * If the user tries to update a plugin while an update is 220 * already happening, it can be placed in this queue to perform later. 221 * 222 * @since 4.2.0 223 * @since 4.6.0 More accurately named `queue`. 224 * 225 * @type {Array.object} 226 */ 227 wp.updates.queue = []; 228 229 /** 230 * Holds a jQuery reference to return focus to when exiting the request credentials modal. 231 * 232 * @since 4.2.0 233 * 234 * @type {jQuery} 235 */ 236 wp.updates.$elToReturnFocusToFromCredentialsModal = undefined; 237 238 /** 239 * Adds or updates an admin notice. 240 * 241 * @since 4.6.0 242 * 243 * @param {Object} data 244 * @param {*=} data.selector Optional. Selector of an element to be replaced with the admin notice. 245 * @param {string=} data.id Optional. Unique id that will be used as the notice's id attribute. 246 * @param {string=} data.className Optional. Class names that will be used in the admin notice. 247 * @param {string=} data.message Optional. The message displayed in the notice. 248 * @param {number=} data.successes Optional. The amount of successful operations. 249 * @param {number=} data.errors Optional. The amount of failed operations. 250 * @param {Array=} data.errorMessages Optional. Error messages of failed operations. 251 * 252 */ 253 wp.updates.addAdminNotice = function( data ) { 254 var $notice = $( data.selector ), 255 $headerEnd = $( '.wp-header-end' ), 256 $adminNotice; 257 258 delete data.selector; 259 $adminNotice = wp.updates.adminNotice( data ); 260 261 // Check if this admin notice already exists. 262 if ( ! $notice.length ) { 263 $notice = $( '#' + data.id ); 264 } 265 266 if ( $notice.length ) { 267 $notice.replaceWith( $adminNotice ); 268 } else if ( $headerEnd.length ) { 269 $headerEnd.after( $adminNotice ); 270 } else { 271 if ( 'customize' === pagenow ) { 272 $( '.customize-themes-notifications' ).append( $adminNotice ); 273 } else { 274 $( '.wrap' ).find( '> h1' ).after( $adminNotice ); 275 } 276 } 277 278 $document.trigger( 'wp-updates-notice-added' ); 279 }; 280 281 /** 282 * Handles Ajax requests to WordPress. 283 * 284 * @since 4.6.0 285 * 286 * @param {string} action The type of Ajax request ('update-plugin', 'install-theme', etc). 287 * @param {Object} data Data that needs to be passed to the ajax callback. 288 * @return {$.promise} A jQuery promise that represents the request, 289 * decorated with an abort() method. 290 */ 291 wp.updates.ajax = function( action, data ) { 292 var options = {}; 293 294 if ( wp.updates.ajaxLocked ) { 295 wp.updates.queue.push( { 296 action: action, 297 data: data 298 } ); 299 300 // Return a Deferred object so callbacks can always be registered. 301 return $.Deferred(); 302 } 303 304 wp.updates.ajaxLocked = true; 305 306 if ( data.success ) { 307 options.success = data.success; 308 delete data.success; 309 } 310 311 if ( data.error ) { 312 options.error = data.error; 313 delete data.error; 314 } 315 316 options.data = _.extend( data, { 317 action: action, 318 _ajax_nonce: wp.updates.ajaxNonce, 319 _fs_nonce: wp.updates.filesystemCredentials.fsNonce, 320 username: wp.updates.filesystemCredentials.ftp.username, 321 password: wp.updates.filesystemCredentials.ftp.password, 322 hostname: wp.updates.filesystemCredentials.ftp.hostname, 323 connection_type: wp.updates.filesystemCredentials.ftp.connectionType, 324 public_key: wp.updates.filesystemCredentials.ssh.publicKey, 325 private_key: wp.updates.filesystemCredentials.ssh.privateKey 326 } ); 327 328 return wp.ajax.send( options ).always( wp.updates.ajaxAlways ); 329 }; 330 331 /** 332 * Actions performed after every Ajax request. 333 * 334 * @since 4.6.0 335 * 336 * @param {Object} response 337 * @param {Array=} response.debug Optional. Debug information. 338 * @param {string=} response.errorCode Optional. Error code for an error that occurred. 339 */ 340 wp.updates.ajaxAlways = function( response ) { 341 if ( ! response.errorCode || 'unable_to_connect_to_filesystem' !== response.errorCode ) { 342 wp.updates.ajaxLocked = false; 343 wp.updates.queueChecker(); 344 } 345 346 if ( 'undefined' !== typeof response.debug && window.console && window.console.log ) { 347 _.map( response.debug, function( message ) { 348 // Remove all HTML tags and write a message to the console. 349 window.console.log( wp.sanitize.stripTagsAndEncodeText( message ) ); 350 } ); 351 } 352 }; 353 354 /** 355 * Refreshes update counts everywhere on the screen. 356 * 357 * @since 4.7.0 358 */ 359 wp.updates.refreshCount = function() { 360 var $adminBarUpdates = $( '#wp-admin-bar-updates' ), 361 $dashboardNavMenuUpdateCount = $( 'a[href="update-core.php"] .update-plugins' ), 362 $pluginsNavMenuUpdateCount = $( 'a[href="plugins.php"] .update-plugins' ), 363 $appearanceNavMenuUpdateCount = $( 'a[href="themes.php"] .update-plugins' ), 364 itemCount; 365 366 $adminBarUpdates.find( '.ab-label' ).text( settings.totals.counts.total ); 367 $adminBarUpdates.find( '.updates-available-text' ).text( 368 sprintf( 369 /* translators: %s: Total number of updates available. */ 370 _n( '%s update available', '%s updates available', settings.totals.counts.total ), 371 settings.totals.counts.total 372 ) 373 ); 374 375 // Remove the update count from the toolbar if it's zero. 376 if ( 0 === settings.totals.counts.total ) { 377 $adminBarUpdates.find( '.ab-label' ).parents( 'li' ).remove(); 378 } 379 380 // Update the "Updates" menu item. 381 $dashboardNavMenuUpdateCount.each( function( index, element ) { 382 element.className = element.className.replace( /count-\d+/, 'count-' + settings.totals.counts.total ); 383 } ); 384 if ( settings.totals.counts.total > 0 ) { 385 $dashboardNavMenuUpdateCount.find( '.update-count' ).text( settings.totals.counts.total ); 386 } else { 387 $dashboardNavMenuUpdateCount.remove(); 388 } 389 390 // Update the "Plugins" menu item. 391 $pluginsNavMenuUpdateCount.each( function( index, element ) { 392 element.className = element.className.replace( /count-\d+/, 'count-' + settings.totals.counts.plugins ); 393 } ); 394 if ( settings.totals.counts.total > 0 ) { 395 $pluginsNavMenuUpdateCount.find( '.plugin-count' ).text( settings.totals.counts.plugins ); 396 } else { 397 $pluginsNavMenuUpdateCount.remove(); 398 } 399 400 // Update the "Appearance" menu item. 401 $appearanceNavMenuUpdateCount.each( function( index, element ) { 402 element.className = element.className.replace( /count-\d+/, 'count-' + settings.totals.counts.themes ); 403 } ); 404 if ( settings.totals.counts.total > 0 ) { 405 $appearanceNavMenuUpdateCount.find( '.theme-count' ).text( settings.totals.counts.themes ); 406 } else { 407 $appearanceNavMenuUpdateCount.remove(); 408 } 409 410 // Update list table filter navigation. 411 if ( 'plugins' === pagenow || 'plugins-network' === pagenow ) { 412 itemCount = settings.totals.counts.plugins; 413 } else if ( 'themes' === pagenow || 'themes-network' === pagenow ) { 414 itemCount = settings.totals.counts.themes; 415 } 416 417 if ( itemCount > 0 ) { 418 $( '.subsubsub .upgrade .count' ).text( '(' + itemCount + ')' ); 419 } else { 420 $( '.subsubsub .upgrade' ).remove(); 421 $( '.subsubsub li:last' ).html( function() { return $( this ).children(); } ); 422 } 423 }; 424 425 /** 426 * Sends a message from a modal to the main screen to update buttons in plugin cards. 427 * 428 * @since 6.5.0 429 * 430 * @param {Object} data An object of data to use for the button. 431 * @param {string} data.slug The plugin's slug. 432 * @param {string} data.text The text to use for the button. 433 * @param {string} data.ariaLabel The value for the button's aria-label attribute. An empty string removes the attribute. 434 * @param {string=} data.status Optional. An identifier for the status. 435 * @param {string=} data.removeClasses Optional. A space-separated list of classes to remove from the button. 436 * @param {string=} data.addClasses Optional. A space-separated list of classes to add to the button. 437 * @param {string=} data.href Optional. The button's URL. 438 * @param {string=} data.pluginName Optional. The plugin's name. 439 * @param {string=} data.plugin Optional. The plugin file, relative to the plugins directory. 440 */ 441 wp.updates.setCardButtonStatus = function( data ) { 442 var target = window.parent === window ? null : window.parent; 443 444 $.support.postMessage = !! window.postMessage; 445 if ( false !== $.support.postMessage && null !== target && -1 === window.parent.location.pathname.indexOf( 'index.php' ) ) { 446 target.postMessage( JSON.stringify( data ), window.location.origin ); 447 } 448 }; 449 450 /** 451 * Decrements the update counts throughout the various menus. 452 * 453 * This includes the toolbar, the "Updates" menu item and the menu items 454 * for plugins and themes. 455 * 456 * @since 3.9.0 457 * 458 * @param {string} type The type of item that was updated or deleted. 459 * Can be 'plugin', 'theme'. 460 */ 461 wp.updates.decrementCount = function( type ) { 462 settings.totals.counts.total = Math.max( --settings.totals.counts.total, 0 ); 463 464 if ( 'plugin' === type ) { 465 settings.totals.counts.plugins = Math.max( --settings.totals.counts.plugins, 0 ); 466 } else if ( 'theme' === type ) { 467 settings.totals.counts.themes = Math.max( --settings.totals.counts.themes, 0 ); 468 } 469 470 wp.updates.refreshCount( type ); 471 }; 472 473 /** 474 * Sends an Ajax request to the server to update a plugin. 475 * 476 * @since 4.2.0 477 * @since 4.6.0 More accurately named `updatePlugin`. 478 * 479 * @param {Object} args Arguments. 480 * @param {string} args.plugin Plugin basename. 481 * @param {string} args.slug Plugin slug. 482 * @param {updatePluginSuccess=} args.success Optional. Success callback. Default: wp.updates.updatePluginSuccess 483 * @param {updatePluginError=} args.error Optional. Error callback. Default: wp.updates.updatePluginError 484 * @return {$.promise} A jQuery promise that represents the request, 485 * decorated with an abort() method. 486 */ 487 wp.updates.updatePlugin = function( args ) { 488 var $updateRow, $card, $message, message, 489 $adminBarUpdates = $( '#wp-admin-bar-updates' ), 490 buttonText = __( 'Updating...' ), 491 isPluginInstall = 'plugin-install' === pagenow || 'plugin-install-network' === pagenow; 492 493 args = _.extend( { 494 success: wp.updates.updatePluginSuccess, 495 error: wp.updates.updatePluginError 496 }, args ); 497 498 if ( 'plugins' === pagenow || 'plugins-network' === pagenow ) { 499 $updateRow = $( 'tr[data-plugin="' + args.plugin + '"]' ); 500 $message = $updateRow.find( '.update-message' ).removeClass( 'notice-error' ).addClass( 'updating-message notice-warning' ).find( 'p' ); 501 message = sprintf( 502 /* translators: %s: Plugin name and version. */ 503 _x( 'Updating %s...', 'plugin' ), 504 $updateRow.find( '.plugin-title strong' ).text() 505 ); 506 } else if ( isPluginInstall ) { 507 $card = $( '.plugin-card-' + args.slug + ', #plugin-information-footer' ); 508 $message = $card.find( '.update-now' ).addClass( 'updating-message' ); 509 message = sprintf( 510 /* translators: %s: Plugin name and version. */ 511 _x( 'Updating %s...', 'plugin' ), 512 $message.data( 'name' ) 513 ); 514 515 // Remove previous error messages, if any. 516 $card.removeClass( 'plugin-card-update-failed' ).find( '.notice.notice-error' ).remove(); 517 } 518 519 $adminBarUpdates.addClass( 'spin' ); 520 521 if ( $message.html() !== __( 'Updating...' ) ) { 522 $message.data( 'originaltext', $message.html() ); 523 } 524 525 $message 526 .attr( 'aria-label', message ) 527 .text( buttonText ); 528 529 $document.trigger( 'wp-plugin-updating', args ); 530 531 if ( isPluginInstall && 'plugin-information-footer' === $card.attr( 'id' ) ) { 532 wp.updates.setCardButtonStatus( 533 { 534 status: 'updating-plugin', 535 slug: args.slug, 536 addClasses: 'updating-message', 537 text: buttonText, 538 ariaLabel: message 539 } 540 ); 541 } 542 543 return wp.updates.ajax( 'update-plugin', args ); 544 }; 545 546 /** 547 * Updates the UI appropriately after a successful plugin update. 548 * 549 * @since 4.2.0 550 * @since 4.6.0 More accurately named `updatePluginSuccess`. 551 * @since 5.5.0 Auto-update "time to next update" text cleared. 552 * 553 * @param {Object} response Response from the server. 554 * @param {string} response.slug Slug of the plugin to be updated. 555 * @param {string} response.plugin Basename of the plugin to be updated. 556 * @param {string} response.pluginName Name of the plugin to be updated. 557 * @param {string} response.oldVersion Old version of the plugin. 558 * @param {string} response.newVersion New version of the plugin. 559 */ 560 wp.updates.updatePluginSuccess = function( response ) { 561 var $pluginRow, $updateMessage, newText, 562 $adminBarUpdates = $( '#wp-admin-bar-updates' ), 563 buttonText = _x( 'Updated!', 'plugin' ), 564 ariaLabel = sprintf( 565 /* translators: %s: Plugin name and version. */ 566 _x( '%s updated!', 'plugin' ), 567 response.pluginName 568 ); 569 570 if ( 'plugins' === pagenow || 'plugins-network' === pagenow ) { 571 $pluginRow = $( 'tr[data-plugin="' + response.plugin + '"]' ) 572 .removeClass( 'update is-enqueued' ) 573 .addClass( 'updated' ); 574 $updateMessage = $pluginRow.find( '.update-message' ) 575 .removeClass( 'updating-message notice-warning' ) 576 .addClass( 'updated-message notice-success' ).find( 'p' ); 577 578 // Update the version number in the row. 579 newText = $pluginRow.find( '.plugin-version-author-uri' ).html().replace( response.oldVersion, response.newVersion ); 580 $pluginRow.find( '.plugin-version-author-uri' ).html( newText ); 581 582 // Clear the "time to next auto-update" text. 583 $pluginRow.find( '.auto-update-time' ).empty(); 584 } else if ( 'plugin-install' === pagenow || 'plugin-install-network' === pagenow ) { 585 $updateMessage = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ).find( '.update-now' ) 586 .removeClass( 'updating-message' ) 587 .addClass( 'button-disabled updated-message' ); 588 } 589 590 $adminBarUpdates.removeClass( 'spin' ); 591 592 $updateMessage 593 .attr( 'aria-label', ariaLabel ) 594 .text( buttonText ); 595 596 wp.a11y.speak( __( 'Update completed successfully.' ) ); 597 598 if ( 'plugin_install_from_iframe' !== $updateMessage.attr( 'id' ) ) { 599 wp.updates.decrementCount( 'plugin' ); 600 } else { 601 wp.updates.setCardButtonStatus( 602 { 603 status: 'updated-plugin', 604 slug: response.slug, 605 removeClasses: 'updating-message', 606 addClasses: 'button-disabled updated-message', 607 text: buttonText, 608 ariaLabel: ariaLabel 609 } 610 ); 611 } 612 613 $document.trigger( 'wp-plugin-update-success', response ); 614 }; 615 616 /** 617 * Updates the UI appropriately after a failed plugin update. 618 * 619 * @since 4.2.0 620 * @since 4.6.0 More accurately named `updatePluginError`. 621 * 622 * @param {Object} response Response from the server. 623 * @param {string} response.slug Slug of the plugin to be updated. 624 * @param {string} response.plugin Basename of the plugin to be updated. 625 * @param {string=} response.pluginName Optional. Name of the plugin to be updated. 626 * @param {string} response.errorCode Error code for the error that occurred. 627 * @param {string} response.errorMessage The error that occurred. 628 */ 629 wp.updates.updatePluginError = function( response ) { 630 var $pluginRow, $card, $message, errorMessage, buttonText, ariaLabel, 631 $adminBarUpdates = $( '#wp-admin-bar-updates' ); 632 633 if ( ! wp.updates.isValidResponse( response, 'update' ) ) { 634 return; 635 } 636 637 if ( wp.updates.maybeHandleCredentialError( response, 'update-plugin' ) ) { 638 return; 639 } 640 641 errorMessage = sprintf( 642 /* translators: %s: Error string for a failed update. */ 643 __( 'Update failed: %s' ), 644 response.errorMessage 645 ); 646 647 if ( 'plugins' === pagenow || 'plugins-network' === pagenow ) { 648 $pluginRow = $( 'tr[data-plugin="' + response.plugin + '"]' ).removeClass( 'is-enqueued' ); 649 650 if ( response.plugin ) { 651 $message = $( 'tr[data-plugin="' + response.plugin + '"]' ).find( '.update-message' ); 652 } else { 653 $message = $( 'tr[data-slug="' + response.slug + '"]' ).find( '.update-message' ); 654 } 655 $message.removeClass( 'updating-message notice-warning' ).addClass( 'notice-error' ).find( 'p' ).html( errorMessage ); 656 657 if ( response.pluginName ) { 658 $message.find( 'p' ) 659 .attr( 660 'aria-label', 661 sprintf( 662 /* translators: %s: Plugin name and version. */ 663 _x( '%s update failed.', 'plugin' ), 664 response.pluginName 665 ) 666 ); 667 } else { 668 $message.find( 'p' ).removeAttr( 'aria-label' ); 669 } 670 } else if ( 'plugin-install' === pagenow || 'plugin-install-network' === pagenow ) { 671 buttonText = __( 'Update failed.' ); 672 673 $card = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ) 674 .append( wp.updates.adminNotice( { 675 className: 'update-message notice-error notice-alt is-dismissible', 676 message: errorMessage 677 } ) ); 678 679 if ( $card.hasClass( 'plugin-card-' + response.slug ) ) { 680 $card.addClass( 'plugin-card-update-failed' ); 681 } 682 683 $card.find( '.update-now' ) 684 .text( buttonText ) 685 .removeClass( 'updating-message' ); 686 687 if ( response.pluginName ) { 688 ariaLabel = sprintf( 689 /* translators: %s: Plugin name and version. */ 690 _x( '%s update failed.', 'plugin' ), 691 response.pluginName 692 ); 693 694 $card.find( '.update-now' ).attr( 'aria-label', ariaLabel ); 695 } else { 696 ariaLabel = ''; 697 $card.find( '.update-now' ).removeAttr( 'aria-label' ); 698 } 699 700 $card.on( 'click', '.notice.is-dismissible .notice-dismiss', function() { 701 702 // Use same delay as the total duration of the notice fadeTo + slideUp animation. 703 setTimeout( function() { 704 $card 705 .removeClass( 'plugin-card-update-failed' ) 706 .find( '.column-name a' ).trigger( 'focus' ); 707 708 $card.find( '.update-now' ) 709 .attr( 'aria-label', false ) 710 .text( __( 'Update Now' ) ); 711 }, 200 ); 712 } ); 713 } 714 715 $adminBarUpdates.removeClass( 'spin' ); 716 717 wp.a11y.speak( errorMessage, 'assertive' ); 718 719 if ( 'plugin-information-footer' === $card.attr('id' ) ) { 720 wp.updates.setCardButtonStatus( 721 { 722 status: 'plugin-update-failed', 723 slug: response.slug, 724 removeClasses: 'updating-message', 725 text: buttonText, 726 ariaLabel: ariaLabel 727 } 728 ); 729 } 730 731 $document.trigger( 'wp-plugin-update-error', response ); 732 }; 733 734 /** 735 * Sends an Ajax request to the server to install a plugin. 736 * 737 * @since 4.6.0 738 * 739 * @param {Object} args Arguments. 740 * @param {string} args.slug Plugin identifier in the WordPress.org Plugin repository. 741 * @param {installPluginSuccess=} args.success Optional. Success callback. Default: wp.updates.installPluginSuccess 742 * @param {installPluginError=} args.error Optional. Error callback. Default: wp.updates.installPluginError 743 * @return {$.promise} A jQuery promise that represents the request, 744 * decorated with an abort() method. 745 */ 746 wp.updates.installPlugin = function( args ) { 747 var $card = $( '.plugin-card-' + args.slug + ', #plugin-information-footer' ), 748 $message = $card.find( '.install-now' ), 749 buttonText = __( 'Installing...' ), 750 ariaLabel; 751 752 args = _.extend( { 753 success: wp.updates.installPluginSuccess, 754 error: wp.updates.installPluginError 755 }, args ); 756 757 if ( 'import' === pagenow ) { 758 $message = $( '[data-slug="' + args.slug + '"]' ); 759 } 760 761 if ( $message.html() !== __( 'Installing...' ) ) { 762 $message.data( 'originaltext', $message.html() ); 763 } 764 765 ariaLabel = sprintf( 766 /* translators: %s: Plugin name and version. */ 767 _x( 'Installing %s...', 'plugin' ), 768 $message.data( 'name' ) 769 ); 770 771 $message 772 .addClass( 'updating-message' ) 773 .attr( 'aria-label', ariaLabel ) 774 .text( buttonText ); 775 776 wp.a11y.speak( __( 'Installing... please wait.' ) ); 777 778 // Remove previous error messages, if any. 779 $card.removeClass( 'plugin-card-install-failed' ).find( '.notice.notice-error' ).remove(); 780 781 $document.trigger( 'wp-plugin-installing', args ); 782 783 if ( 'plugin-information-footer' === $message.parent().attr( 'id' ) ) { 784 wp.updates.setCardButtonStatus( 785 { 786 status: 'installing-plugin', 787 slug: args.slug, 788 addClasses: 'updating-message', 789 text: buttonText, 790 ariaLabel: ariaLabel 791 } 792 ); 793 } 794 795 return wp.updates.ajax( 'install-plugin', args ); 796 }; 797 798 /** 799 * Updates the UI appropriately after a successful plugin install. 800 * 801 * @since 4.6.0 802 * 803 * @param {Object} response Response from the server. 804 * @param {string} response.slug Slug of the installed plugin. 805 * @param {string} response.pluginName Name of the installed plugin. 806 * @param {string} response.activateUrl URL to activate the just installed plugin. 807 */ 808 wp.updates.installPluginSuccess = function( response ) { 809 var $message = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ).find( '.install-now' ), 810 buttonText = _x( 'Installed!', 'plugin' ), 811 ariaLabel = sprintf( 812 /* translators: %s: Plugin name and version. */ 813 _x( '%s installed!', 'plugin' ), 814 response.pluginName 815 ); 816 817 $message 818 .removeClass( 'updating-message' ) 819 .addClass( 'updated-message installed button-disabled' ) 820 .attr( 'aria-label', ariaLabel ) 821 .text( buttonText ); 822 823 wp.a11y.speak( __( 'Installation completed successfully.' ) ); 824 825 $document.trigger( 'wp-plugin-install-success', response ); 826 827 if ( response.activateUrl ) { 828 setTimeout( function() { 829 wp.updates.checkPluginDependencies( { 830 slug: response.slug 831 } ); 832 }, 1000 ); 833 } 834 835 if ( 'plugin-information-footer' === $message.parent().attr( 'id' ) ) { 836 wp.updates.setCardButtonStatus( 837 { 838 status: 'installed-plugin', 839 slug: response.slug, 840 removeClasses: 'updating-message', 841 addClasses: 'updated-message installed button-disabled', 842 text: buttonText, 843 ariaLabel: ariaLabel 844 } 845 ); 846 } 847 }; 848 849 /** 850 * Updates the UI appropriately after a failed plugin install. 851 * 852 * @since 4.6.0 853 * 854 * @param {Object} response Response from the server. 855 * @param {string} response.slug Slug of the plugin to be installed. 856 * @param {string=} response.pluginName Optional. Name of the plugin to be installed. 857 * @param {string} response.errorCode Error code for the error that occurred. 858 * @param {string} response.errorMessage The error that occurred. 859 */ 860 wp.updates.installPluginError = function( response ) { 861 var $card = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ), 862 $button = $card.find( '.install-now' ), 863 buttonText = __( 'Installation failed.' ), 864 ariaLabel = sprintf( 865 /* translators: %s: Plugin name and version. */ 866 _x( '%s installation failed', 'plugin' ), 867 $button.data( 'name' ) 868 ), 869 errorMessage; 870 871 if ( ! wp.updates.isValidResponse( response, 'install' ) ) { 872 return; 873 } 874 875 if ( wp.updates.maybeHandleCredentialError( response, 'install-plugin' ) ) { 876 return; 877 } 878 879 errorMessage = sprintf( 880 /* translators: %s: Error string for a failed installation. */ 881 __( 'Installation failed: %s' ), 882 response.errorMessage 883 ); 884 885 $card 886 .addClass( 'plugin-card-update-failed' ) 887 .append( '<div class="notice notice-error notice-alt is-dismissible" role="alert"><p>' + errorMessage + '</p></div>' ); 888 889 $card.on( 'click', '.notice.is-dismissible .notice-dismiss', function() { 890 891 // Use same delay as the total duration of the notice fadeTo + slideUp animation. 892 setTimeout( function() { 893 $card 894 .removeClass( 'plugin-card-update-failed' ) 895 .find( '.column-name a' ).trigger( 'focus' ); 896 }, 200 ); 897 } ); 898 899 $button 900 .removeClass( 'updating-message' ).addClass( 'button-disabled' ) 901 .attr( 'aria-label', ariaLabel ) 902 .text( buttonText ); 903 904 wp.a11y.speak( errorMessage, 'assertive' ); 905 906 wp.updates.setCardButtonStatus( 907 { 908 status: 'plugin-install-failed', 909 slug: response.slug, 910 removeClasses: 'updating-message', 911 addClasses: 'button-disabled', 912 text: buttonText, 913 ariaLabel: ariaLabel 914 } 915 ); 916 917 $document.trigger( 'wp-plugin-install-error', response ); 918 }; 919 920 /** 921 * Sends an Ajax request to the server to check a plugin's dependencies. 922 * 923 * @since 6.5.0 924 * 925 * @param {Object} args Arguments. 926 * @param {string} args.slug Plugin identifier in the WordPress.org Plugin repository. 927 * @param {checkPluginDependenciesSuccess=} args.success Optional. Success callback. Default: wp.updates.checkPluginDependenciesSuccess 928 * @param {checkPluginDependenciesError=} args.error Optional. Error callback. Default: wp.updates.checkPluginDependenciesError 929 * @return {$.promise} A jQuery promise that represents the request, 930 * decorated with an abort() method. 931 */ 932 wp.updates.checkPluginDependencies = function( args ) { 933 args = _.extend( { 934 success: wp.updates.checkPluginDependenciesSuccess, 935 error: wp.updates.checkPluginDependenciesError 936 }, args ); 937 938 wp.a11y.speak( __( 'Checking plugin dependencies... please wait.' ) ); 939 $document.trigger( 'wp-checking-plugin-dependencies', args ); 940 941 return wp.updates.ajax( 'check_plugin_dependencies', args ); 942 }; 943 944 /** 945 * Updates the UI appropriately after a successful plugin dependencies check. 946 * 947 * @since 6.5.0 948 * 949 * @param {Object} response Response from the server. 950 * @param {string} response.slug Slug of the checked plugin. 951 * @param {string} response.pluginName Name of the checked plugin. 952 * @param {string} response.plugin The plugin file, relative to the plugins directory. 953 * @param {string} response.activateUrl URL to activate the just checked plugin. 954 */ 955 wp.updates.checkPluginDependenciesSuccess = function( response ) { 956 var $message = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ).find( '.install-now' ), 957 buttonText, ariaLabel; 958 959 // Transform the 'Install' button into an 'Activate' button. 960 $message 961 .removeClass( 'install-now installed button-disabled updated-message' ) 962 .addClass( 'activate-now button-primary' ) 963 .attr( 'href', response.activateUrl ); 964 965 wp.a11y.speak( __( 'Plugin dependencies check completed successfully.' ) ); 966 $document.trigger( 'wp-check-plugin-dependencies-success', response ); 967 968 if ( 'plugins-network' === pagenow ) { 969 buttonText = _x( 'Network Activate', 'plugin' ); 970 ariaLabel = sprintf( 971 /* translators: %s: Plugin name. */ 972 _x( 'Network Activate %s', 'plugin' ), 973 response.pluginName 974 ); 975 976 $message 977 .attr( 'aria-label', ariaLabel ) 978 .text( buttonText ); 979 } else { 980 buttonText = _x( 'Activate', 'plugin' ); 981 ariaLabel = sprintf( 982 /* translators: %s: Plugin name. */ 983 _x( 'Activate %s', 'plugin' ), 984 response.pluginName 985 ); 986 987 $message 988 .attr( 'aria-label', ariaLabel ) 989 .attr( 'data-name', response.pluginName ) 990 .attr( 'data-slug', response.slug ) 991 .attr( 'data-plugin', response.plugin ) 992 .text( buttonText ); 993 } 994 995 if ( 'plugin-information-footer' === $message.parent().attr( 'id' ) ) { 996 wp.updates.setCardButtonStatus( 997 { 998 status: 'dependencies-check-success', 999 slug: response.slug, 1000 removeClasses: 'install-now installed button-disabled updated-message', 1001 addClasses: 'activate-now button-primary', 1002 text: buttonText, 1003 ariaLabel: ariaLabel, 1004 pluginName: response.pluginName, 1005 plugin: response.plugin, 1006 href: response.activateUrl 1007 } 1008 ); 1009 } 1010 }; 1011 1012 /** 1013 * Updates the UI appropriately after a failed plugin dependencies check. 1014 * 1015 * @since 6.5.0 1016 * 1017 * @param {Object} response Response from the server. 1018 * @param {string} response.slug Slug of the plugin to be checked. 1019 * @param {string=} response.pluginName Optional. Name of the plugin to be checked. 1020 * @param {string} response.errorCode Error code for the error that occurred. 1021 * @param {string} response.errorMessage The error that occurred. 1022 */ 1023 wp.updates.checkPluginDependenciesError = function( response ) { 1024 var $message = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ).find( '.install-now' ), 1025 buttonText = _x( 'Activate', 'plugin' ), 1026 ariaLabel = sprintf( 1027 /* translators: 1: Plugin name, 2. The reason the plugin cannot be activated. */ 1028 _x( 'Cannot activate %1$s. %2$s', 'plugin' ), 1029 response.pluginName, 1030 response.errorMessage 1031 ), 1032 errorMessage; 1033 1034 if ( ! wp.updates.isValidResponse( response, 'check-dependencies' ) ) { 1035 return; 1036 } 1037 1038 errorMessage = sprintf( 1039 /* translators: %s: Error string for a failed activation. */ 1040 __( 'Activation failed: %s' ), 1041 response.errorMessage 1042 ); 1043 1044 wp.a11y.speak( errorMessage, 'assertive' ); 1045 $document.trigger( 'wp-check-plugin-dependencies-error', response ); 1046 1047 $message 1048 .removeClass( 'install-now installed updated-message' ) 1049 .addClass( 'activate-now button-primary' ) 1050 .attr( 'aria-label', ariaLabel ) 1051 .text( buttonText ); 1052 1053 if ( 'plugin-information-footer' === $message.parent().attr('id' ) ) { 1054 wp.updates.setCardButtonStatus( 1055 { 1056 status: 'dependencies-check-failed', 1057 slug: response.slug, 1058 removeClasses: 'install-now installed updated-message', 1059 addClasses: 'activate-now button-primary', 1060 text: buttonText, 1061 ariaLabel: ariaLabel 1062 } 1063 ); 1064 } 1065 }; 1066 1067 /** 1068 * Sends an Ajax request to the server to activate a plugin. 1069 * 1070 * @since 6.5.0 1071 * 1072 * @param {Object} args Arguments. 1073 * @param {string} args.name The name of the plugin. 1074 * @param {string} args.slug Plugin identifier in the WordPress.org Plugin repository. 1075 * @param {string} args.plugin The plugin file, relative to the plugins directory. 1076 * @param {activatePluginSuccess=} args.success Optional. Success callback. Default: wp.updates.activatePluginSuccess 1077 * @param {activatePluginError=} args.error Optional. Error callback. Default: wp.updates.activatePluginError 1078 * @return {$.promise} A jQuery promise that represents the request, 1079 * decorated with an abort() method. 1080 */ 1081 wp.updates.activatePlugin = function( args ) { 1082 var $message = $( '.plugin-card-' + args.slug + ', #plugin-information-footer' ).find( '.activate-now, .activating-message' ); 1083 1084 args = _.extend( { 1085 success: wp.updates.activatePluginSuccess, 1086 error: wp.updates.activatePluginError 1087 }, args ); 1088 1089 wp.a11y.speak( __( 'Activating... please wait.' ) ); 1090 $document.trigger( 'wp-activating-plugin', args ); 1091 1092 if ( 'plugin-information-footer' === $message.parent().attr( 'id' ) ) { 1093 wp.updates.setCardButtonStatus( 1094 { 1095 status: 'activating-plugin', 1096 slug: args.slug, 1097 removeClasses: 'installed updated-message button-primary', 1098 addClasses: 'activating-message', 1099 text: __( 'Activating...' ), 1100 ariaLabel: sprintf( 1101 /* translators: %s: Plugin name. */ 1102 _x( 'Activating %s', 'plugin' ), 1103 args.name 1104 ) 1105 } 1106 ); 1107 } 1108 1109 return wp.updates.ajax( 'activate-plugin', args ); 1110 }; 1111 1112 /** 1113 * Updates the UI appropriately after a successful plugin activation. 1114 * 1115 * @since 6.5.0 1116 * 1117 * @param {Object} response Response from the server. 1118 * @param {string} response.slug Slug of the activated plugin. 1119 * @param {string} response.pluginName Name of the activated plugin. 1120 * @param {string} response.plugin The plugin file, relative to the plugins directory. 1121 */ 1122 wp.updates.activatePluginSuccess = function( response ) { 1123 var $message = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ).find( '.activating-message' ), 1124 buttonText = _x( 'Activated!', 'plugin' ), 1125 ariaLabel = sprintf( 1126 /* translators: %s: The plugin name. */ 1127 '%s activated successfully.', 1128 response.pluginName 1129 ); 1130 1131 wp.a11y.speak( __( 'Activation completed successfully.' ) ); 1132 $document.trigger( 'wp-plugin-activate-success', response ); 1133 1134 $message 1135 .removeClass( 'activating-message' ) 1136 .addClass( 'activated-message button-disabled' ) 1137 .attr( 'aria-label', ariaLabel ) 1138 .text( buttonText ); 1139 1140 if ( 'plugin-information-footer' === $message.parent().attr( 'id' ) ) { 1141 wp.updates.setCardButtonStatus( 1142 { 1143 status: 'activated-plugin', 1144 slug: response.slug, 1145 removeClasses: 'activating-message', 1146 addClasses: 'activated-message button-disabled', 1147 text: buttonText, 1148 ariaLabel: ariaLabel 1149 } 1150 ); 1151 } 1152 1153 setTimeout( function() { 1154 $message.removeClass( 'activated-message' ) 1155 .text( _x( 'Active', 'plugin' ) ); 1156 1157 if ( 'plugin-information-footer' === $message.parent().attr( 'id' ) ) { 1158 wp.updates.setCardButtonStatus( 1159 { 1160 status: 'plugin-active', 1161 slug: response.slug, 1162 removeClasses: 'activated-message', 1163 text: _x( 'Active', 'plugin' ), 1164 ariaLabel: sprintf( 1165 /* translators: %s: The plugin name. */ 1166 '%s is active.', 1167 response.pluginName 1168 ) 1169 } 1170 ); 1171 } 1172 }, 1000 ); 1173 }; 1174 1175 /** 1176 * Updates the UI appropriately after a failed plugin activation. 1177 * 1178 * @since 6.5.0 1179 * 1180 * @param {Object} response Response from the server. 1181 * @param {string} response.slug Slug of the plugin to be activated. 1182 * @param {string=} response.pluginName Optional. Name of the plugin to be activated. 1183 * @param {string} response.errorCode Error code for the error that occurred. 1184 * @param {string} response.errorMessage The error that occurred. 1185 */ 1186 wp.updates.activatePluginError = function( response ) { 1187 var $message = $( '.plugin-card-' + response.slug + ', #plugin-information-footer' ).find( '.activating-message' ), 1188 buttonText = __( 'Activation failed.' ), 1189 ariaLabel = sprintf( 1190 /* translators: %s: Plugin name. */ 1191 _x( '%s activation failed', 'plugin' ), 1192 response.pluginName 1193 ), 1194 errorMessage; 1195 1196 if ( ! wp.updates.isValidResponse( response, 'activate' ) ) { 1197 return; 1198 } 1199 1200 errorMessage = sprintf( 1201 /* translators: %s: Error string for a failed activation. */ 1202 __( 'Activation failed: %s' ), 1203 response.errorMessage 1204 ); 1205 1206 wp.a11y.speak( errorMessage, 'assertive' ); 1207 $document.trigger( 'wp-plugin-activate-error', response ); 1208 1209 $message 1210 .removeClass( 'install-now installed activating-message' ) 1211 .addClass( 'button-disabled' ) 1212 .attr( 'aria-label', ariaLabel ) 1213 .text( buttonText ); 1214 1215 if ( 'plugin-information-footer' === $message.parent().attr( 'id' ) ) { 1216 wp.updates.setCardButtonStatus( 1217 { 1218 status: 'plugin-activation-failed', 1219 slug: response.slug, 1220 removeClasses: 'install-now installed activating-message', 1221 addClasses: 'button-disabled', 1222 text: buttonText, 1223 ariaLabel: ariaLabel 1224 } 1225 ); 1226 } 1227 }; 1228 1229 /** 1230 * Updates the UI appropriately after a successful importer install. 1231 * 1232 * @since 4.6.0 1233 * 1234 * @param {Object} response Response from the server. 1235 * @param {string} response.slug Slug of the installed plugin. 1236 * @param {string} response.pluginName Name of the installed plugin. 1237 * @param {string} response.activateUrl URL to activate the just installed plugin. 1238 */ 1239 wp.updates.installImporterSuccess = function( response ) { 1240 wp.updates.addAdminNotice( { 1241 id: 'install-success', 1242 className: 'notice-success is-dismissible', 1243 message: sprintf( 1244 /* translators: %s: Activation URL. */ 1245 __( 'Importer installed successfully. <a href="%s">Run importer</a>' ), 1246 response.activateUrl + '&from=import' 1247 ) 1248 } ); 1249 1250 $( '[data-slug="' + response.slug + '"]' ) 1251 .removeClass( 'install-now updating-message' ) 1252 .addClass( 'activate-now' ) 1253 .attr({ 1254 'href': response.activateUrl + '&from=import', 1255 'aria-label':sprintf( 1256 /* translators: %s: Importer name. */ 1257 __( 'Run %s' ), 1258 response.pluginName 1259 ) 1260 }) 1261 .text( __( 'Run Importer' ) ); 1262 1263 wp.a11y.speak( __( 'Installation completed successfully.' ) ); 1264 1265 $document.trigger( 'wp-importer-install-success', response ); 1266 }; 1267 1268 /** 1269 * Updates the UI appropriately after a failed importer install. 1270 * 1271 * @since 4.6.0 1272 * 1273 * @param {Object} response Response from the server. 1274 * @param {string} response.slug Slug of the plugin to be installed. 1275 * @param {string=} response.pluginName Optional. Name of the plugin to be installed. 1276 * @param {string} response.errorCode Error code for the error that occurred. 1277 * @param {string} response.errorMessage The error that occurred. 1278 */ 1279 wp.updates.installImporterError = function( response ) { 1280 var errorMessage = sprintf( 1281 /* translators: %s: Error string for a failed installation. */ 1282 __( 'Installation failed: %s' ), 1283 response.errorMessage 1284 ), 1285 $installLink = $( '[data-slug="' + response.slug + '"]' ), 1286 pluginName = $installLink.data( 'name' ); 1287 1288 if ( ! wp.updates.isValidResponse( response, 'install' ) ) { 1289 return; 1290 } 1291 1292 if ( wp.updates.maybeHandleCredentialError( response, 'install-plugin' ) ) { 1293 return; 1294 } 1295 1296 wp.updates.addAdminNotice( { 1297 id: response.errorCode, 1298 className: 'notice-error is-dismissible', 1299 message: errorMessage 1300 } ); 1301 1302 $installLink 1303 .removeClass( 'updating-message' ) 1304 .attr( 1305 'aria-label', 1306 sprintf( 1307 /* translators: %s: Plugin name. */ 1308 _x( 'Install %s now', 'plugin' ), 1309 pluginName 1310 ) 1311 ) 1312 .text( _x( 'Install Now', 'plugin' ) ); 1313 1314 wp.a11y.speak( errorMessage, 'assertive' ); 1315 1316 $document.trigger( 'wp-importer-install-error', response ); 1317 }; 1318 1319 /** 1320 * Sends an Ajax request to the server to delete a plugin. 1321 * 1322 * @since 4.6.0 1323 * 1324 * @param {Object} args Arguments. 1325 * @param {string} args.plugin Basename of the plugin to be deleted. 1326 * @param {string} args.slug Slug of the plugin to be deleted. 1327 * @param {deletePluginSuccess=} args.success Optional. Success callback. Default: wp.updates.deletePluginSuccess 1328 * @param {deletePluginError=} args.error Optional. Error callback. Default: wp.updates.deletePluginError 1329 * @return {$.promise} A jQuery promise that represents the request, 1330 * decorated with an abort() method. 1331 */ 1332 wp.updates.deletePlugin = function( args ) { 1333 var $link = $( '[data-plugin="' + args.plugin + '"]' ).find( '.row-actions a.delete' ); 1334 1335 args = _.extend( { 1336 success: wp.updates.deletePluginSuccess, 1337 error: wp.updates.deletePluginError 1338 }, args ); 1339 1340 if ( $link.html() !== __( 'Deleting...' ) ) { 1341 $link 1342 .data( 'originaltext', $link.html() ) 1343 .text( __( 'Deleting...' ) ); 1344 } 1345 1346 wp.a11y.speak( __( 'Deleting...' ) ); 1347 1348 $document.trigger( 'wp-plugin-deleting', args ); 1349 1350 return wp.updates.ajax( 'delete-plugin', args ); 1351 }; 1352 1353 /** 1354 * Updates the UI appropriately after a successful plugin deletion. 1355 * 1356 * @since 4.6.0 1357 * 1358 * @param {Object} response Response from the server. 1359 * @param {string} response.slug Slug of the plugin that was deleted. 1360 * @param {string} response.plugin Base name of the plugin that was deleted. 1361 * @param {string} response.pluginName Name of the plugin that was deleted. 1362 */ 1363 wp.updates.deletePluginSuccess = function( response ) { 1364 1365 // Removes the plugin and updates rows. 1366 $( '[data-plugin="' + response.plugin + '"]' ).css( { backgroundColor: '#faafaa' } ).fadeOut( 350, function() { 1367 var $form = $( '#bulk-action-form' ), 1368 $views = $( '.subsubsub' ), 1369 $pluginRow = $( this ), 1370 $currentView = $views.find( '[aria-current="page"]' ), 1371 $itemsCount = $( '.displaying-num' ), 1372 columnCount = $form.find( 'thead th:not(.hidden), thead td' ).length, 1373 pluginDeletedRow = wp.template( 'item-deleted-row' ), 1374 /** 1375 * Plugins Base names of plugins in their different states. 1376 * 1377 * @type {Object} 1378 */ 1379 plugins = settings.plugins, 1380 remainingCount; 1381 1382 // Add a success message after deleting a plugin. 1383 if ( ! $pluginRow.hasClass( 'plugin-update-tr' ) ) { 1384 $pluginRow.after( 1385 pluginDeletedRow( { 1386 slug: response.slug, 1387 plugin: response.plugin, 1388 colspan: columnCount, 1389 name: response.pluginName 1390 } ) 1391 ); 1392 } 1393 1394 $pluginRow.remove(); 1395 1396 // Remove plugin from update count. 1397 if ( -1 !== _.indexOf( plugins.upgrade, response.plugin ) ) { 1398 plugins.upgrade = _.without( plugins.upgrade, response.plugin ); 1399 wp.updates.decrementCount( 'plugin' ); 1400 } 1401 1402 // Remove from views. 1403 if ( -1 !== _.indexOf( plugins.inactive, response.plugin ) ) { 1404 plugins.inactive = _.without( plugins.inactive, response.plugin ); 1405 if ( plugins.inactive.length ) { 1406 $views.find( '.inactive .count' ).text( '(' + plugins.inactive.length + ')' ); 1407 } else { 1408 $views.find( '.inactive' ).remove(); 1409 } 1410 } 1411 1412 if ( -1 !== _.indexOf( plugins.active, response.plugin ) ) { 1413 plugins.active = _.without( plugins.active, response.plugin ); 1414 if ( plugins.active.length ) { 1415 $views.find( '.active .count' ).text( '(' + plugins.active.length + ')' ); 1416 } else { 1417 $views.find( '.active' ).remove(); 1418 } 1419 } 1420 1421 if ( -1 !== _.indexOf( plugins.recently_activated, response.plugin ) ) { 1422 plugins.recently_activated = _.without( plugins.recently_activated, response.plugin ); 1423 if ( plugins.recently_activated.length ) { 1424 $views.find( '.recently_activated .count' ).text( '(' + plugins.recently_activated.length + ')' ); 1425 } else { 1426 $views.find( '.recently_activated' ).remove(); 1427 } 1428 } 1429 1430 if ( -1 !== _.indexOf( plugins['auto-update-enabled'], response.plugin ) ) { 1431 plugins['auto-update-enabled'] = _.without( plugins['auto-update-enabled'], response.plugin ); 1432 if ( plugins['auto-update-enabled'].length ) { 1433 $views.find( '.auto-update-enabled .count' ).text( '(' + plugins['auto-update-enabled'].length + ')' ); 1434 } else { 1435 $views.find( '.auto-update-enabled' ).remove(); 1436 } 1437 } 1438 1439 if ( -1 !== _.indexOf( plugins['auto-update-disabled'], response.plugin ) ) { 1440 plugins['auto-update-disabled'] = _.without( plugins['auto-update-disabled'], response.plugin ); 1441 if ( plugins['auto-update-disabled'].length ) { 1442 $views.find( '.auto-update-disabled .count' ).text( '(' + plugins['auto-update-disabled'].length + ')' ); 1443 } else { 1444 $views.find( '.auto-update-disabled' ).remove(); 1445 } 1446 } 1447 1448 plugins.all = _.without( plugins.all, response.plugin ); 1449 1450 if ( plugins.all.length ) { 1451 $views.find( '.all .count' ).text( '(' + plugins.all.length + ')' ); 1452 } else { 1453 $form.find( '.tablenav' ).css( { visibility: 'hidden' } ); 1454 $views.find( '.all' ).remove(); 1455 1456 if ( ! $form.find( 'tr.no-items' ).length ) { 1457 $form.find( '#the-list' ).append( '<tr class="no-items"><td class="colspanchange" colspan="' + columnCount + '">' + __( 'No plugins are currently available.' ) + '</td></tr>' ); 1458 } 1459 } 1460 1461 if ( $itemsCount.length && $currentView.length ) { 1462 remainingCount = plugins[ $currentView.parent( 'li' ).attr('class') ].length; 1463 $itemsCount.text( 1464 sprintf( 1465 /* translators: %s: The remaining number of plugins. */ 1466 _nx( '%s item', '%s items', remainingCount, 'plugin/plugins' ), 1467 remainingCount 1468 ) 1469 ); 1470 } 1471 } ); 1472 1473 wp.a11y.speak( _x( 'Deleted!', 'plugin' ) ); 1474 1475 $document.trigger( 'wp-plugin-delete-success', response ); 1476 }; 1477 1478 /** 1479 * Updates the UI appropriately after a failed plugin deletion. 1480 * 1481 * @since 4.6.0 1482 * 1483 * @param {Object} response Response from the server. 1484 * @param {string} response.slug Slug of the plugin to be deleted. 1485 * @param {string} response.plugin Base name of the plugin to be deleted 1486 * @param {string=} response.pluginName Optional. Name of the plugin to be deleted. 1487 * @param {string} response.errorCode Error code for the error that occurred. 1488 * @param {string} response.errorMessage The error that occurred. 1489 */ 1490 wp.updates.deletePluginError = function( response ) { 1491 var $plugin, $pluginUpdateRow, 1492 pluginUpdateRow = wp.template( 'item-update-row' ), 1493 noticeContent = wp.updates.adminNotice( { 1494 className: 'update-message notice-error notice-alt', 1495 message: response.errorMessage 1496 } ); 1497 1498 if ( response.plugin ) { 1499 $plugin = $( 'tr.inactive[data-plugin="' + response.plugin + '"]' ); 1500 $pluginUpdateRow = $plugin.siblings( '[data-plugin="' + response.plugin + '"]' ); 1501 } else { 1502 $plugin = $( 'tr.inactive[data-slug="' + response.slug + '"]' ); 1503 $pluginUpdateRow = $plugin.siblings( '[data-slug="' + response.slug + '"]' ); 1504 } 1505 1506 if ( ! wp.updates.isValidResponse( response, 'delete' ) ) { 1507 return; 1508 } 1509 1510 if ( wp.updates.maybeHandleCredentialError( response, 'delete-plugin' ) ) { 1511 return; 1512 } 1513 1514 // Add a plugin update row if it doesn't exist yet. 1515 if ( ! $pluginUpdateRow.length ) { 1516 $plugin.addClass( 'update' ).after( 1517 pluginUpdateRow( { 1518 slug: response.slug, 1519 plugin: response.plugin || response.slug, 1520 colspan: $( '#bulk-action-form' ).find( 'thead th:not(.hidden), thead td' ).length, 1521 content: noticeContent 1522 } ) 1523 ); 1524 } else { 1525 1526 // Remove previous error messages, if any. 1527 $pluginUpdateRow.find( '.notice-error' ).remove(); 1528 1529 $pluginUpdateRow.find( '.plugin-update' ).append( noticeContent ); 1530 } 1531 1532 $document.trigger( 'wp-plugin-delete-error', response ); 1533 }; 1534 1535 /** 1536 * Sends an Ajax request to the server to update a theme. 1537 * 1538 * @since 4.6.0 1539 * 1540 * @param {Object} args Arguments. 1541 * @param {string} args.slug Theme stylesheet. 1542 * @param {updateThemeSuccess=} args.success Optional. Success callback. Default: wp.updates.updateThemeSuccess 1543 * @param {updateThemeError=} args.error Optional. Error callback. Default: wp.updates.updateThemeError 1544 * @return {$.promise} A jQuery promise that represents the request, 1545 * decorated with an abort() method. 1546 */ 1547 wp.updates.updateTheme = function( args ) { 1548 var $notice; 1549 1550 args = _.extend( { 1551 success: wp.updates.updateThemeSuccess, 1552 error: wp.updates.updateThemeError 1553 }, args ); 1554 1555 if ( 'themes-network' === pagenow ) { 1556 $notice = $( '[data-slug="' + args.slug + '"]' ).find( '.update-message' ).removeClass( 'notice-error' ).addClass( 'updating-message notice-warning' ).find( 'p' ); 1557 1558 } else if ( 'customize' === pagenow ) { 1559 1560 // Update the theme details UI. 1561 $notice = $( '[data-slug="' + args.slug + '"].notice' ).removeClass( 'notice-large' ); 1562 1563 $notice.find( 'h3' ).remove(); 1564 1565 // Add the top-level UI, and update both. 1566 $notice = $notice.add( $( '#customize-control-installed_theme_' + args.slug ).find( '.update-message' ) ); 1567 $notice = $notice.addClass( 'updating-message' ).find( 'p' ); 1568 1569 } else { 1570 $notice = $( '#update-theme' ).closest( '.notice' ).removeClass( 'notice-large' ); 1571 1572 $notice.find( 'h3' ).remove(); 1573 1574 $notice = $notice.add( $( '[data-slug="' + args.slug + '"]' ).find( '.update-message' ) ); 1575 $notice = $notice.addClass( 'updating-message' ).find( 'p' ); 1576 } 1577 1578 if ( $notice.html() !== __( 'Updating...' ) ) { 1579 $notice.data( 'originaltext', $notice.html() ); 1580 } 1581 1582 wp.a11y.speak( __( 'Updating... please wait.' ) ); 1583 $notice.text( __( 'Updating...' ) ); 1584 1585 $document.trigger( 'wp-theme-updating', args ); 1586 1587 return wp.updates.ajax( 'update-theme', args ); 1588 }; 1589 1590 /** 1591 * Updates the UI appropriately after a successful theme update. 1592 * 1593 * @since 4.6.0 1594 * @since 5.5.0 Auto-update "time to next update" text cleared. 1595 * 1596 * @param {Object} response 1597 * @param {string} response.slug Slug of the theme to be updated. 1598 * @param {Object} response.theme Updated theme. 1599 * @param {string} response.oldVersion Old version of the theme. 1600 * @param {string} response.newVersion New version of the theme. 1601 */ 1602 wp.updates.updateThemeSuccess = function( response ) { 1603 var isModalOpen = $( 'body.modal-open' ).length, 1604 $theme = $( '[data-slug="' + response.slug + '"]' ), 1605 updatedMessage = { 1606 className: 'updated-message notice-success notice-alt', 1607 message: _x( 'Updated!', 'theme' ) 1608 }, 1609 $notice, newText; 1610 1611 if ( 'customize' === pagenow ) { 1612 $theme = $( '.updating-message' ).siblings( '.theme-name' ); 1613 1614 if ( $theme.length ) { 1615 1616 // Update the version number in the row. 1617 newText = $theme.html().replace( response.oldVersion, response.newVersion ); 1618 $theme.html( newText ); 1619 } 1620 1621 $notice = $( '.theme-info .notice' ).add( wp.customize.control( 'installed_theme_' + response.slug ).container.find( '.theme' ).find( '.update-message' ) ); 1622 } else if ( 'themes-network' === pagenow ) { 1623 $notice = $theme.find( '.update-message' ); 1624 1625 // Update the version number in the row. 1626 newText = $theme.find( '.theme-version-author-uri' ).html().replace( response.oldVersion, response.newVersion ); 1627 $theme.find( '.theme-version-author-uri' ).html( newText ); 1628 1629 // Clear the "time to next auto-update" text. 1630 $theme.find( '.auto-update-time' ).empty(); 1631 } else { 1632 $notice = $( '.theme-info .notice' ).add( $theme.find( '.update-message' ) ); 1633 1634 // Focus on Customize button after updating. 1635 if ( isModalOpen ) { 1636 $( '.load-customize:visible' ).trigger( 'focus' ); 1637 $( '.theme-info .theme-autoupdate' ).find( '.auto-update-time' ).empty(); 1638 } else { 1639 $theme.find( '.load-customize' ).trigger( 'focus' ); 1640 } 1641 } 1642 1643 wp.updates.addAdminNotice( _.extend( { selector: $notice }, updatedMessage ) ); 1644 wp.a11y.speak( __( 'Update completed successfully.' ) ); 1645 1646 wp.updates.decrementCount( 'theme' ); 1647 1648 $document.trigger( 'wp-theme-update-success', response ); 1649 1650 // Show updated message after modal re-rendered. 1651 if ( isModalOpen && 'customize' !== pagenow ) { 1652 $( '.theme-info .theme-author' ).after( wp.updates.adminNotice( updatedMessage ) ); 1653 } 1654 }; 1655 1656 /** 1657 * Updates the UI appropriately after a failed theme update. 1658 * 1659 * @since 4.6.0 1660 * 1661 * @param {Object} response Response from the server. 1662 * @param {string} response.slug Slug of the theme to be updated. 1663 * @param {string} response.errorCode Error code for the error that occurred. 1664 * @param {string} response.errorMessage The error that occurred. 1665 */ 1666 wp.updates.updateThemeError = function( response ) { 1667 var $theme = $( '[data-slug="' + response.slug + '"]' ), 1668 errorMessage = sprintf( 1669 /* translators: %s: Error string for a failed update. */ 1670 __( 'Update failed: %s' ), 1671 response.errorMessage 1672 ), 1673 $notice; 1674 1675 if ( ! wp.updates.isValidResponse( response, 'update' ) ) { 1676 return; 1677 } 1678 1679 if ( wp.updates.maybeHandleCredentialError( response, 'update-theme' ) ) { 1680 return; 1681 } 1682 1683 if ( 'customize' === pagenow ) { 1684 $theme = wp.customize.control( 'installed_theme_' + response.slug ).container.find( '.theme' ); 1685 } 1686 1687 if ( 'themes-network' === pagenow ) { 1688 $notice = $theme.find( '.update-message ' ); 1689 } else { 1690 $notice = $( '.theme-info .notice' ).add( $theme.find( '.notice' ) ); 1691 1692 $( 'body.modal-open' ).length ? $( '.load-customize:visible' ).trigger( 'focus' ) : $theme.find( '.load-customize' ).trigger( 'focus'); 1693 } 1694 1695 wp.updates.addAdminNotice( { 1696 selector: $notice, 1697 className: 'update-message notice-error notice-alt is-dismissible', 1698 message: errorMessage 1699 } ); 1700 1701 wp.a11y.speak( errorMessage ); 1702 1703 $document.trigger( 'wp-theme-update-error', response ); 1704 }; 1705 1706 /** 1707 * Sends an Ajax request to the server to install a theme. 1708 * 1709 * @since 4.6.0 1710 * 1711 * @param {Object} args 1712 * @param {string} args.slug Theme stylesheet. 1713 * @param {installThemeSuccess=} args.success Optional. Success callback. Default: wp.updates.installThemeSuccess 1714 * @param {installThemeError=} args.error Optional. Error callback. Default: wp.updates.installThemeError 1715 * @return {$.promise} A jQuery promise that represents the request, 1716 * decorated with an abort() method. 1717 */ 1718 wp.updates.installTheme = function( args ) { 1719 var $message = $( '.theme-install[data-slug="' + args.slug + '"]' ); 1720 1721 args = _.extend( { 1722 success: wp.updates.installThemeSuccess, 1723 error: wp.updates.installThemeError 1724 }, args ); 1725 1726 $message.addClass( 'updating-message' ); 1727 $message.parents( '.theme' ).addClass( 'focus' ); 1728 if ( $message.html() !== __( 'Installing...' ) ) { 1729 $message.data( 'originaltext', $message.html() ); 1730 } 1731 1732 $message 1733 .attr( 1734 'aria-label', 1735 sprintf( 1736 /* translators: %s: Theme name and version. */ 1737 _x( 'Installing %s...', 'theme' ), 1738 $message.data( 'name' ) 1739 ) 1740 ) 1741 .text( __( 'Installing...' ) ); 1742 1743 wp.a11y.speak( __( 'Installing... please wait.' ) ); 1744 1745 // Remove previous error messages, if any. 1746 $( '.install-theme-info, [data-slug="' + args.slug + '"]' ).removeClass( 'theme-install-failed' ).find( '.notice.notice-error' ).remove(); 1747 1748 $document.trigger( 'wp-theme-installing', args ); 1749 1750 return wp.updates.ajax( 'install-theme', args ); 1751 }; 1752 1753 /** 1754 * Updates the UI appropriately after a successful theme install. 1755 * 1756 * @since 4.6.0 1757 * 1758 * @param {Object} response Response from the server. 1759 * @param {string} response.slug Slug of the theme to be installed. 1760 * @param {string} response.customizeUrl URL to the Customizer for the just installed theme. 1761 * @param {string} response.activateUrl URL to activate the just installed theme. 1762 */ 1763 wp.updates.installThemeSuccess = function( response ) { 1764 var $card = $( '.wp-full-overlay-header, [data-slug=' + response.slug + ']' ), 1765 $message; 1766 1767 $document.trigger( 'wp-theme-install-success', response ); 1768 1769 $message = $card.find( '.button-primary' ) 1770 .removeClass( 'updating-message' ) 1771 .addClass( 'updated-message disabled' ) 1772 .attr( 1773 'aria-label', 1774 sprintf( 1775 /* translators: %s: Theme name and version. */ 1776 _x( '%s installed!', 'theme' ), 1777 response.themeName 1778 ) 1779 ) 1780 .text( _x( 'Installed!', 'theme' ) ); 1781 1782 wp.a11y.speak( __( 'Installation completed successfully.' ) ); 1783 1784 setTimeout( function() { 1785 1786 if ( response.activateUrl ) { 1787 1788 // Transform the 'Install' button into an 'Activate' button. 1789 $message 1790 .attr( 'href', response.activateUrl ) 1791 .removeClass( 'theme-install updated-message disabled' ) 1792 .addClass( 'activate' ); 1793 1794 if ( 'themes-network' === pagenow ) { 1795 $message 1796 .attr( 1797 'aria-label', 1798 sprintf( 1799 /* translators: %s: Theme name. */ 1800 _x( 'Network Activate %s', 'theme' ), 1801 response.themeName 1802 ) 1803 ) 1804 .text( __( 'Network Enable' ) ); 1805 } else { 1806 $message 1807 .attr( 1808 'aria-label', 1809 sprintf( 1810 /* translators: %s: Theme name. */ 1811 _x( 'Activate %s', 'theme' ), 1812 response.themeName 1813 ) 1814 ) 1815 .text( _x( 'Activate', 'theme' ) ); 1816 } 1817 } 1818 1819 if ( response.customizeUrl ) { 1820 1821 // Transform the 'Preview' button into a 'Live Preview' button. 1822 $message.siblings( '.preview' ).replaceWith( function () { 1823 return $( '<a>' ) 1824 .attr( 'href', response.customizeUrl ) 1825 .addClass( 'button load-customize' ) 1826 .text( __( 'Live Preview' ) ); 1827 } ); 1828 } 1829 }, 1000 ); 1830 }; 1831 1832 /** 1833 * Updates the UI appropriately after a failed theme install. 1834 * 1835 * @since 4.6.0 1836 * 1837 * @param {Object} response Response from the server. 1838 * @param {string} response.slug Slug of the theme to be installed. 1839 * @param {string} response.errorCode Error code for the error that occurred. 1840 * @param {string} response.errorMessage The error that occurred. 1841 */ 1842 wp.updates.installThemeError = function( response ) { 1843 var $card, $button, 1844 errorMessage = sprintf( 1845 /* translators: %s: Error string for a failed installation. */ 1846 __( 'Installation failed: %s' ), 1847 response.errorMessage 1848 ), 1849 $message = wp.updates.adminNotice( { 1850 className: 'update-message notice-error notice-alt', 1851 message: errorMessage 1852 } ); 1853 1854 if ( ! wp.updates.isValidResponse( response, 'install' ) ) { 1855 return; 1856 } 1857 1858 if ( wp.updates.maybeHandleCredentialError( response, 'install-theme' ) ) { 1859 return; 1860 } 1861 1862 if ( 'customize' === pagenow ) { 1863 if ( $document.find( 'body' ).hasClass( 'modal-open' ) ) { 1864 $button = $( '.theme-install[data-slug="' + response.slug + '"]' ); 1865 $card = $( '.theme-overlay .theme-info' ).prepend( $message ); 1866 } else { 1867 $button = $( '.theme-install[data-slug="' + response.slug + '"]' ); 1868 $card = $button.closest( '.theme' ).addClass( 'theme-install-failed' ).append( $message ); 1869 } 1870 wp.customize.notifications.remove( 'theme_installing' ); 1871 } else { 1872 if ( $document.find( 'body' ).hasClass( 'full-overlay-active' ) ) { 1873 $button = $( '.theme-install[data-slug="' + response.slug + '"]' ); 1874 $card = $( '.install-theme-info' ).prepend( $message ); 1875 } else { 1876 $card = $( '[data-slug="' + response.slug + '"]' ).removeClass( 'focus' ).addClass( 'theme-install-failed' ).append( $message ); 1877 $button = $card.find( '.theme-install' ); 1878 } 1879 } 1880 1881 $button 1882 .removeClass( 'updating-message' ) 1883 .attr( 1884 'aria-label', 1885 sprintf( 1886 /* translators: %s: Theme name and version. */ 1887 _x( '%s installation failed', 'theme' ), 1888 $button.data( 'name' ) 1889 ) 1890 ) 1891 .text( __( 'Installation failed.' ) ); 1892 1893 wp.a11y.speak( errorMessage, 'assertive' ); 1894 1895 $document.trigger( 'wp-theme-install-error', response ); 1896 }; 1897 1898 /** 1899 * Sends an Ajax request to the server to delete a theme. 1900 * 1901 * @since 4.6.0 1902 * 1903 * @param {Object} args 1904 * @param {string} args.slug Theme stylesheet. 1905 * @param {deleteThemeSuccess=} args.success Optional. Success callback. Default: wp.updates.deleteThemeSuccess 1906 * @param {deleteThemeError=} args.error Optional. Error callback. Default: wp.updates.deleteThemeError 1907 * @return {$.promise} A jQuery promise that represents the request, 1908 * decorated with an abort() method. 1909 */ 1910 wp.updates.deleteTheme = function( args ) { 1911 var $button; 1912 1913 if ( 'themes' === pagenow ) { 1914 $button = $( '.theme-actions .delete-theme' ); 1915 } else if ( 'themes-network' === pagenow ) { 1916 $button = $( '[data-slug="' + args.slug + '"]' ).find( '.row-actions a.delete' ); 1917 } 1918 1919 args = _.extend( { 1920 success: wp.updates.deleteThemeSuccess, 1921 error: wp.updates.deleteThemeError 1922 }, args ); 1923 1924 if ( $button && $button.html() !== __( 'Deleting...' ) ) { 1925 $button 1926 .data( 'originaltext', $button.html() ) 1927 .text( __( 'Deleting...' ) ); 1928 } 1929 1930 wp.a11y.speak( __( 'Deleting...' ) ); 1931 1932 // Remove previous error messages, if any. 1933 $( '.theme-info .update-message' ).remove(); 1934 1935 $document.trigger( 'wp-theme-deleting', args ); 1936 1937 return wp.updates.ajax( 'delete-theme', args ); 1938 }; 1939 1940 /** 1941 * Updates the UI appropriately after a successful theme deletion. 1942 * 1943 * @since 4.6.0 1944 * 1945 * @param {Object} response Response from the server. 1946 * @param {string} response.slug Slug of the theme that was deleted. 1947 */ 1948 wp.updates.deleteThemeSuccess = function( response ) { 1949 var $themeRows = $( '[data-slug="' + response.slug + '"]' ); 1950 1951 if ( 'themes-network' === pagenow ) { 1952 1953 // Removes the theme and updates rows. 1954 $themeRows.css( { backgroundColor: '#faafaa' } ).fadeOut( 350, function() { 1955 var $views = $( '.subsubsub' ), 1956 $themeRow = $( this ), 1957 themes = settings.themes, 1958 deletedRow = wp.template( 'item-deleted-row' ); 1959 1960 if ( ! $themeRow.hasClass( 'plugin-update-tr' ) ) { 1961 $themeRow.after( 1962 deletedRow( { 1963 slug: response.slug, 1964 colspan: $( '#bulk-action-form' ).find( 'thead th:not(.hidden), thead td' ).length, 1965 name: $themeRow.find( '.theme-title strong' ).text() 1966 } ) 1967 ); 1968 } 1969 1970 $themeRow.remove(); 1971 1972 // Remove theme from update count. 1973 if ( -1 !== _.indexOf( themes.upgrade, response.slug ) ) { 1974 themes.upgrade = _.without( themes.upgrade, response.slug ); 1975 wp.updates.decrementCount( 'theme' ); 1976 } 1977 1978 // Remove from views. 1979 if ( -1 !== _.indexOf( themes.disabled, response.slug ) ) { 1980 themes.disabled = _.without( themes.disabled, response.slug ); 1981 if ( themes.disabled.length ) { 1982 $views.find( '.disabled .count' ).text( '(' + themes.disabled.length + ')' ); 1983 } else { 1984 $views.find( '.disabled' ).remove(); 1985 } 1986 } 1987 1988 if ( -1 !== _.indexOf( themes['auto-update-enabled'], response.slug ) ) { 1989 themes['auto-update-enabled'] = _.without( themes['auto-update-enabled'], response.slug ); 1990 if ( themes['auto-update-enabled'].length ) { 1991 $views.find( '.auto-update-enabled .count' ).text( '(' + themes['auto-update-enabled'].length + ')' ); 1992 } else { 1993 $views.find( '.auto-update-enabled' ).remove(); 1994 } 1995 } 1996 1997 if ( -1 !== _.indexOf( themes['auto-update-disabled'], response.slug ) ) { 1998 themes['auto-update-disabled'] = _.without( themes['auto-update-disabled'], response.slug ); 1999 if ( themes['auto-update-disabled'].length ) { 2000 $views.find( '.auto-update-disabled .count' ).text( '(' + themes['auto-update-disabled'].length + ')' ); 2001 } else { 2002 $views.find( '.auto-update-disabled' ).remove(); 2003 } 2004 } 2005 2006 themes.all = _.without( themes.all, response.slug ); 2007 2008 // There is always at least one theme available. 2009 $views.find( '.all .count' ).text( '(' + themes.all.length + ')' ); 2010 } ); 2011 } 2012 2013 // DecrementCount from update count. 2014 if ( 'themes' === pagenow ) { 2015 var theme = _.find( _wpThemeSettings.themes, { id: response.slug } ); 2016 if ( theme.hasUpdate ) { 2017 wp.updates.decrementCount( 'theme' ); 2018 } 2019 } 2020 2021 wp.a11y.speak( _x( 'Deleted!', 'theme' ) ); 2022 2023 $document.trigger( 'wp-theme-delete-success', response ); 2024 }; 2025 2026 /** 2027 * Updates the UI appropriately after a failed theme deletion. 2028 * 2029 * @since 4.6.0 2030 * 2031 * @param {Object} response Response from the server. 2032 * @param {string} response.slug Slug of the theme to be deleted. 2033 * @param {string} response.errorCode Error code for the error that occurred. 2034 * @param {string} response.errorMessage The error that occurred. 2035 */ 2036 wp.updates.deleteThemeError = function( response ) { 2037 var $themeRow = $( 'tr.inactive[data-slug="' + response.slug + '"]' ), 2038 $button = $( '.theme-actions .delete-theme' ), 2039 updateRow = wp.template( 'item-update-row' ), 2040 $updateRow = $themeRow.siblings( '#' + response.slug + '-update' ), 2041 errorMessage = sprintf( 2042 /* translators: %s: Error string for a failed deletion. */ 2043 __( 'Deletion failed: %s' ), 2044 response.errorMessage 2045 ), 2046 $message = wp.updates.adminNotice( { 2047 className: 'update-message notice-error notice-alt', 2048 message: errorMessage 2049 } ); 2050 2051 if ( wp.updates.maybeHandleCredentialError( response, 'delete-theme' ) ) { 2052 return; 2053 } 2054 2055 if ( 'themes-network' === pagenow ) { 2056 if ( ! $updateRow.length ) { 2057 $themeRow.addClass( 'update' ).after( 2058 updateRow( { 2059 slug: response.slug, 2060 colspan: $( '#bulk-action-form' ).find( 'thead th:not(.hidden), thead td' ).length, 2061 content: $message 2062 } ) 2063 ); 2064 } else { 2065 // Remove previous error messages, if any. 2066 $updateRow.find( '.notice-error' ).remove(); 2067 $updateRow.find( '.plugin-update' ).append( $message ); 2068 } 2069 } else { 2070 $( '.theme-info .theme-description' ).before( $message ); 2071 } 2072 2073 $button.html( $button.data( 'originaltext' ) ); 2074 2075 wp.a11y.speak( errorMessage, 'assertive' ); 2076 2077 $document.trigger( 'wp-theme-delete-error', response ); 2078 }; 2079 2080 /** 2081 * Adds the appropriate callback based on the type of action and the current page. 2082 * 2083 * @since 4.6.0 2084 * @private 2085 * 2086 * @param {Object} data Ajax payload. 2087 * @param {string} action The type of request to perform. 2088 * @return {Object} The Ajax payload with the appropriate callbacks. 2089 */ 2090 wp.updates._addCallbacks = function( data, action ) { 2091 if ( 'import' === pagenow && 'install-plugin' === action ) { 2092 data.success = wp.updates.installImporterSuccess; 2093 data.error = wp.updates.installImporterError; 2094 } 2095 2096 return data; 2097 }; 2098 2099 /** 2100 * Pulls available jobs from the queue and runs them. 2101 * 2102 * @since 4.2.0 2103 * @since 4.6.0 Can handle multiple job types. 2104 */ 2105 wp.updates.queueChecker = function() { 2106 var job; 2107 2108 if ( wp.updates.ajaxLocked || ! wp.updates.queue.length ) { 2109 return; 2110 } 2111 2112 job = wp.updates.queue.shift(); 2113 2114 // Handle a queue job. 2115 switch ( job.action ) { 2116 case 'install-plugin': 2117 wp.updates.installPlugin( job.data ); 2118 break; 2119 2120 case 'update-plugin': 2121 wp.updates.updatePlugin( job.data ); 2122 break; 2123 2124 case 'delete-plugin': 2125 wp.updates.deletePlugin( job.data ); 2126 break; 2127 2128 case 'install-theme': 2129 wp.updates.installTheme( job.data ); 2130 break; 2131 2132 case 'update-theme': 2133 wp.updates.updateTheme( job.data ); 2134 break; 2135 2136 case 'delete-theme': 2137 wp.updates.deleteTheme( job.data ); 2138 break; 2139 2140 default: 2141 break; 2142 } 2143 }; 2144 2145 /** 2146 * Requests the users filesystem credentials if they aren't already known. 2147 * 2148 * @since 4.2.0 2149 * 2150 * @param {Event=} event Optional. Event interface. 2151 */ 2152 wp.updates.requestFilesystemCredentials = function( event ) { 2153 if ( false === wp.updates.filesystemCredentials.available ) { 2154 /* 2155 * After exiting the credentials request modal, 2156 * return the focus to the element triggering the request. 2157 */ 2158 if ( event && ! wp.updates.$elToReturnFocusToFromCredentialsModal ) { 2159 wp.updates.$elToReturnFocusToFromCredentialsModal = $( event.target ); 2160 } 2161 2162 wp.updates.ajaxLocked = true; 2163 wp.updates.requestForCredentialsModalOpen(); 2164 } 2165 }; 2166 2167 /** 2168 * Requests the users filesystem credentials if needed and there is no lock. 2169 * 2170 * @since 4.6.0 2171 * 2172 * @param {Event=} event Optional. Event interface. 2173 */ 2174 wp.updates.maybeRequestFilesystemCredentials = function( event ) { 2175 if ( wp.updates.shouldRequestFilesystemCredentials && ! wp.updates.ajaxLocked ) { 2176 wp.updates.requestFilesystemCredentials( event ); 2177 } 2178 }; 2179 2180 /** 2181 * Keydown handler for the request for credentials modal. 2182 * 2183 * Closes the modal when the escape key is pressed and 2184 * constrains keyboard navigation to inside the modal. 2185 * 2186 * @since 4.2.0 2187 * 2188 * @param {Event} event Event interface. 2189 */ 2190 wp.updates.keydown = function( event ) { 2191 if ( 27 === event.keyCode ) { 2192 wp.updates.requestForCredentialsModalCancel(); 2193 } else if ( 9 === event.keyCode ) { 2194 2195 // #upgrade button must always be the last focus-able element in the dialog. 2196 if ( 'upgrade' === event.target.id && ! event.shiftKey ) { 2197 $( '#hostname' ).trigger( 'focus' ); 2198 2199 event.preventDefault(); 2200 } else if ( 'hostname' === event.target.id && event.shiftKey ) { 2201 $( '#upgrade' ).trigger( 'focus' ); 2202 2203 event.preventDefault(); 2204 } 2205 } 2206 }; 2207 2208 /** 2209 * Opens the request for credentials modal. 2210 * 2211 * @since 4.2.0 2212 */ 2213 wp.updates.requestForCredentialsModalOpen = function() { 2214 var $modal = $( '#request-filesystem-credentials-dialog' ); 2215 2216 $( 'body' ).addClass( 'modal-open' ); 2217 $modal.show(); 2218 $modal.find( 'input:enabled:first' ).trigger( 'focus' ); 2219 $modal.on( 'keydown', wp.updates.keydown ); 2220 }; 2221 2222 /** 2223 * Closes the request for credentials modal. 2224 * 2225 * @since 4.2.0 2226 */ 2227 wp.updates.requestForCredentialsModalClose = function() { 2228 $( '#request-filesystem-credentials-dialog' ).hide(); 2229 $( 'body' ).removeClass( 'modal-open' ); 2230 2231 if ( wp.updates.$elToReturnFocusToFromCredentialsModal ) { 2232 wp.updates.$elToReturnFocusToFromCredentialsModal.trigger( 'focus' ); 2233 } 2234 }; 2235 2236 /** 2237 * Takes care of the steps that need to happen when the modal is canceled out. 2238 * 2239 * @since 4.2.0 2240 * @since 4.6.0 Triggers an event for callbacks to listen to and add their actions. 2241 */ 2242 wp.updates.requestForCredentialsModalCancel = function() { 2243 2244 // Not ajaxLocked and no queue means we already have cleared things up. 2245 if ( ! wp.updates.ajaxLocked && ! wp.updates.queue.length ) { 2246 return; 2247 } 2248 2249 _.each( wp.updates.queue, function( job ) { 2250 $document.trigger( 'credential-modal-cancel', job ); 2251 } ); 2252 2253 // Remove the lock, and clear the queue. 2254 wp.updates.ajaxLocked = false; 2255 wp.updates.queue = []; 2256 2257 wp.updates.requestForCredentialsModalClose(); 2258 }; 2259 2260 /** 2261 * Displays an error message in the request for credentials form. 2262 * 2263 * @since 4.2.0 2264 * 2265 * @param {string} message Error message. 2266 */ 2267 wp.updates.showErrorInCredentialsForm = function( message ) { 2268 var $filesystemForm = $( '#request-filesystem-credentials-form' ); 2269 2270 // Remove any existing error. 2271 $filesystemForm.find( '.notice' ).remove(); 2272 $filesystemForm.find( '#request-filesystem-credentials-title' ).after( '<div class="notice notice-alt notice-error" role="alert"><p>' + message + '</p></div>' ); 2273 }; 2274 2275 /** 2276 * Handles credential errors and runs events that need to happen in that case. 2277 * 2278 * @since 4.2.0 2279 * 2280 * @param {Object} response Ajax response. 2281 * @param {string} action The type of request to perform. 2282 */ 2283 wp.updates.credentialError = function( response, action ) { 2284 2285 // Restore callbacks. 2286 response = wp.updates._addCallbacks( response, action ); 2287 2288 wp.updates.queue.unshift( { 2289 action: action, 2290 2291 /* 2292 * Not cool that we're depending on response for this data. 2293 * This would feel more whole in a view all tied together. 2294 */ 2295 data: response 2296 } ); 2297 2298 wp.updates.filesystemCredentials.available = false; 2299 wp.updates.showErrorInCredentialsForm( response.errorMessage ); 2300 wp.updates.requestFilesystemCredentials(); 2301 }; 2302 2303 /** 2304 * Handles credentials errors if it could not connect to the filesystem. 2305 * 2306 * @since 4.6.0 2307 * 2308 * @param {Object} response Response from the server. 2309 * @param {string} response.errorCode Error code for the error that occurred. 2310 * @param {string} response.errorMessage The error that occurred. 2311 * @param {string} action The type of request to perform. 2312 * @return {boolean} Whether there is an error that needs to be handled or not. 2313 */ 2314 wp.updates.maybeHandleCredentialError = function( response, action ) { 2315 if ( wp.updates.shouldRequestFilesystemCredentials && response.errorCode && 'unable_to_connect_to_filesystem' === response.errorCode ) { 2316 wp.updates.credentialError( response, action ); 2317 return true; 2318 } 2319 2320 return false; 2321 }; 2322 2323 /** 2324 * Validates an Ajax response to ensure it's a proper object. 2325 * 2326 * If the response deems to be invalid, an admin notice is being displayed. 2327 * 2328 * @param {(Object|string)} response Response from the server. 2329 * @param {function=} response.always Optional. Callback for when the Deferred is resolved or rejected. 2330 * @param {string=} response.statusText Optional. Status message corresponding to the status code. 2331 * @param {string=} response.responseText Optional. Request response as text. 2332 * @param {string} action Type of action the response is referring to. Can be 'delete', 2333 * 'update' or 'install'. 2334 */ 2335 wp.updates.isValidResponse = function( response, action ) { 2336 var error = __( 'Something went wrong.' ), 2337 errorMessage; 2338 2339 // Make sure the response is a valid data object and not a Promise object. 2340 if ( _.isObject( response ) && ! _.isFunction( response.always ) ) { 2341 return true; 2342 } 2343 2344 if ( _.isString( response ) && '-1' === response ) { 2345 error = __( 'An error has occurred. Please reload the page and try again.' ); 2346 } else if ( _.isString( response ) ) { 2347 error = response; 2348 } else if ( 'undefined' !== typeof response.readyState && 0 === response.readyState ) { 2349 error = __( 'Connection lost or the server is busy. Please try again later.' ); 2350 } else if ( _.isString( response.responseText ) && '' !== response.responseText ) { 2351 error = response.responseText; 2352 } else if ( _.isString( response.statusText ) ) { 2353 error = response.statusText; 2354 } 2355 2356 switch ( action ) { 2357 case 'update': 2358 /* translators: %s: Error string for a failed update. */ 2359 errorMessage = __( 'Update failed: %s' ); 2360 break; 2361 2362 case 'install': 2363 /* translators: %s: Error string for a failed installation. */ 2364 errorMessage = __( 'Installation failed: %s' ); 2365 break; 2366 2367 case 'check-dependencies': 2368 /* translators: %s: Error string for a failed dependencies check. */ 2369 errorMessage = __( 'Dependencies check failed: %s' ); 2370 break; 2371 2372 case 'activate': 2373 /* translators: %s: Error string for a failed activation. */ 2374 errorMessage = __( 'Activation failed: %s' ); 2375 break; 2376 2377 case 'delete': 2378 /* translators: %s: Error string for a failed deletion. */ 2379 errorMessage = __( 'Deletion failed: %s' ); 2380 break; 2381 } 2382 2383 // Messages are escaped, remove HTML tags to make them more readable. 2384 error = error.replace( /<[\/a-z][^<>]*>/gi, '' ); 2385 errorMessage = errorMessage.replace( '%s', error ); 2386 2387 // Add admin notice. 2388 wp.updates.addAdminNotice( { 2389 id: 'unknown_error', 2390 className: 'notice-error is-dismissible', 2391 message: _.escape( errorMessage ) 2392 } ); 2393 2394 // Remove the lock, and clear the queue. 2395 wp.updates.ajaxLocked = false; 2396 wp.updates.queue = []; 2397 2398 // Change buttons of all running updates. 2399 $( '.button.updating-message' ) 2400 .removeClass( 'updating-message' ) 2401 .removeAttr( 'aria-label' ) 2402 .prop( 'disabled', true ) 2403 .text( __( 'Update failed.' ) ); 2404 2405 $( '.updating-message:not(.button):not(.thickbox)' ) 2406 .removeClass( 'updating-message notice-warning' ) 2407 .addClass( 'notice-error' ) 2408 .find( 'p' ) 2409 .removeAttr( 'aria-label' ) 2410 .text( errorMessage ); 2411 2412 wp.a11y.speak( errorMessage, 'assertive' ); 2413 2414 return false; 2415 }; 2416 2417 /** 2418 * Potentially adds an AYS to a user attempting to leave the page. 2419 * 2420 * If an update is on-going and a user attempts to leave the page, 2421 * opens an "Are you sure?" alert. 2422 * 2423 * @since 4.2.0 2424 */ 2425 wp.updates.beforeunload = function() { 2426 if ( wp.updates.ajaxLocked ) { 2427 return __( 'Updates may not complete if you navigate away from this page.' ); 2428 } 2429 }; 2430 2431 $( function() { 2432 var $pluginFilter = $( '#plugin-filter, #plugin-information-footer' ), 2433 $bulkActionForm = $( '#bulk-action-form' ), 2434 $filesystemForm = $( '#request-filesystem-credentials-form' ), 2435 $filesystemModal = $( '#request-filesystem-credentials-dialog' ), 2436 $pluginSearch = $( '.plugins-php .wp-filter-search' ), 2437 $pluginInstallSearch = $( '.plugin-install-php .wp-filter-search' ); 2438 2439 settings = _.extend( settings, window._wpUpdatesItemCounts || {} ); 2440 2441 if ( settings.totals ) { 2442 wp.updates.refreshCount(); 2443 } 2444 2445 /* 2446 * Whether a user needs to submit filesystem credentials. 2447 * 2448 * This is based on whether the form was output on the page server-side. 2449 * 2450 * @see {wp_print_request_filesystem_credentials_modal() in PHP} 2451 */ 2452 wp.updates.shouldRequestFilesystemCredentials = $filesystemModal.length > 0; 2453 2454 /** 2455 * File system credentials form submit noop-er / handler. 2456 * 2457 * @since 4.2.0 2458 */ 2459 $filesystemModal.on( 'submit', 'form', function( event ) { 2460 event.preventDefault(); 2461 2462 // Persist the credentials input by the user for the duration of the page load. 2463 wp.updates.filesystemCredentials.ftp.hostname = $( '#hostname' ).val(); 2464 wp.updates.filesystemCredentials.ftp.username = $( '#username' ).val(); 2465 wp.updates.filesystemCredentials.ftp.password = $( '#password' ).val(); 2466 wp.updates.filesystemCredentials.ftp.connectionType = $( 'input[name="connection_type"]:checked' ).val(); 2467 wp.updates.filesystemCredentials.ssh.publicKey = $( '#public_key' ).val(); 2468 wp.updates.filesystemCredentials.ssh.privateKey = $( '#private_key' ).val(); 2469 wp.updates.filesystemCredentials.fsNonce = $( '#_fs_nonce' ).val(); 2470 wp.updates.filesystemCredentials.available = true; 2471 2472 // Unlock and invoke the queue. 2473 wp.updates.ajaxLocked = false; 2474 wp.updates.queueChecker(); 2475 2476 wp.updates.requestForCredentialsModalClose(); 2477 } ); 2478 2479 /** 2480 * Closes the request credentials modal when clicking the 'Cancel' button or outside of the modal. 2481 * 2482 * @since 4.2.0 2483 */ 2484 $filesystemModal.on( 'click', '[data-js-action="close"], .notification-dialog-background', wp.updates.requestForCredentialsModalCancel ); 2485 2486 /** 2487 * Hide SSH fields when not selected. 2488 * 2489 * @since 4.2.0 2490 */ 2491 $filesystemForm.on( 'change', 'input[name="connection_type"]', function() { 2492 $( '#ssh-keys' ).toggleClass( 'hidden', ( 'ssh' !== $( this ).val() ) ); 2493 } ).trigger( 'change' ); 2494 2495 /** 2496 * Handles events after the credential modal was closed. 2497 * 2498 * @since 4.6.0 2499 * 2500 * @param {Event} event Event interface. 2501 * @param {string} job The install/update.delete request. 2502 */ 2503 $document.on( 'credential-modal-cancel', function( event, job ) { 2504 var $updatingMessage = $( '.updating-message' ), 2505 $message, originalText; 2506 2507 if ( 'import' === pagenow ) { 2508 $updatingMessage.removeClass( 'updating-message' ); 2509 } else if ( 'plugins' === pagenow || 'plugins-network' === pagenow ) { 2510 if ( 'update-plugin' === job.action ) { 2511 $message = $( 'tr[data-plugin="' + job.data.plugin + '"]' ).find( '.update-message' ); 2512 } else if ( 'delete-plugin' === job.action ) { 2513 $message = $( '[data-plugin="' + job.data.plugin + '"]' ).find( '.row-actions a.delete' ); 2514 } 2515 } else if ( 'themes' === pagenow || 'themes-network' === pagenow ) { 2516 if ( 'update-theme' === job.action ) { 2517 $message = $( '[data-slug="' + job.data.slug + '"]' ).find( '.update-message' ); 2518 } else if ( 'delete-theme' === job.action && 'themes-network' === pagenow ) { 2519 $message = $( '[data-slug="' + job.data.slug + '"]' ).find( '.row-actions a.delete' ); 2520 } else if ( 'delete-theme' === job.action && 'themes' === pagenow ) { 2521 $message = $( '.theme-actions .delete-theme' ); 2522 } 2523 } else { 2524 $message = $updatingMessage; 2525 } 2526 2527 if ( $message && $message.hasClass( 'updating-message' ) ) { 2528 originalText = $message.data( 'originaltext' ); 2529 2530 if ( 'undefined' === typeof originalText ) { 2531 originalText = $( '<p>' ).html( $message.find( 'p' ).data( 'originaltext' ) ); 2532 } 2533 2534 $message 2535 .removeClass( 'updating-message' ) 2536 .html( originalText ); 2537 2538 if ( 'plugin-install' === pagenow || 'plugin-install-network' === pagenow ) { 2539 if ( 'update-plugin' === job.action ) { 2540 $message.attr( 2541 'aria-label', 2542 sprintf( 2543 /* translators: %s: Plugin name and version. */ 2544 _x( 'Update %s now', 'plugin' ), 2545 $message.data( 'name' ) 2546 ) 2547 ); 2548 } else if ( 'install-plugin' === job.action ) { 2549 $message.attr( 2550 'aria-label', 2551 sprintf( 2552 /* translators: %s: Plugin name. */ 2553 _x( 'Install %s now', 'plugin' ), 2554 $message.data( 'name' ) 2555 ) 2556 ); 2557 } 2558 } 2559 } 2560 2561 wp.a11y.speak( __( 'Update canceled.' ) ); 2562 } ); 2563 2564 /** 2565 * Click handler for plugin updates in List Table view. 2566 * 2567 * @since 4.2.0 2568 * 2569 * @param {Event} event Event interface. 2570 */ 2571 $bulkActionForm.on( 'click', '[data-plugin] .update-link', function( event ) { 2572 var $message = $( event.target ), 2573 $pluginRow = $message.parents( 'tr' ); 2574 2575 event.preventDefault(); 2576 2577 if ( $message.hasClass( 'updating-message' ) || $message.hasClass( 'button-disabled' ) ) { 2578 return; 2579 } 2580 2581 wp.updates.maybeRequestFilesystemCredentials( event ); 2582 2583 // Return the user to the input box of the plugin's table row after closing the modal. 2584 wp.updates.$elToReturnFocusToFromCredentialsModal = $pluginRow.find( '.check-column input' ); 2585 wp.updates.updatePlugin( { 2586 plugin: $pluginRow.data( 'plugin' ), 2587 slug: $pluginRow.data( 'slug' ) 2588 } ); 2589 } ); 2590 2591 /** 2592 * Click handler for plugin updates in plugin install view. 2593 * 2594 * @since 4.2.0 2595 * 2596 * @param {Event} event Event interface. 2597 */ 2598 $pluginFilter.on( 'click', '.update-now', function( event ) { 2599 var $button = $( event.target ); 2600 event.preventDefault(); 2601 2602 if ( $button.hasClass( 'updating-message' ) || $button.hasClass( 'button-disabled' ) ) { 2603 return; 2604 } 2605 2606 wp.updates.maybeRequestFilesystemCredentials( event ); 2607 2608 wp.updates.updatePlugin( { 2609 plugin: $button.data( 'plugin' ), 2610 slug: $button.data( 'slug' ) 2611 } ); 2612 } ); 2613 2614 /** 2615 * Click handler for plugin installs in plugin install view. 2616 * 2617 * @since 4.6.0 2618 * 2619 * @param {Event} event Event interface. 2620 */ 2621 $pluginFilter.on( 'click', '.install-now', function( event ) { 2622 var $button = $( event.target ); 2623 event.preventDefault(); 2624 2625 if ( $button.hasClass( 'updating-message' ) || $button.hasClass( 'button-disabled' ) ) { 2626 return; 2627 } 2628 2629 if ( wp.updates.shouldRequestFilesystemCredentials && ! wp.updates.ajaxLocked ) { 2630 wp.updates.requestFilesystemCredentials( event ); 2631 2632 $document.on( 'credential-modal-cancel', function() { 2633 var $message = $( '.install-now.updating-message' ); 2634 2635 $message 2636 .removeClass( 'updating-message' ) 2637 .text( _x( 'Install Now', 'plugin' ) ); 2638 2639 wp.a11y.speak( __( 'Update canceled.' ) ); 2640 } ); 2641 } 2642 2643 wp.updates.installPlugin( { 2644 slug: $button.data( 'slug' ) 2645 } ); 2646 } ); 2647 2648 /** 2649 * Click handler for plugin activations in plugin activation modal view. 2650 * 2651 * @since 6.5.0 2652 * @since 6.5.4 Redirect the parent window to the activation URL. 2653 * 2654 * @param {Event} event Event interface. 2655 */ 2656 $document.on( 'click', '#plugin-information-footer .activate-now', function( event ) { 2657 event.preventDefault(); 2658 window.parent.location.href = $( event.target ).attr( 'href' ); 2659 }); 2660 2661 /** 2662 * Click handler for importer plugins installs in the Import screen. 2663 * 2664 * @since 4.6.0 2665 * 2666 * @param {Event} event Event interface. 2667 */ 2668 $document.on( 'click', '.importer-item .install-now', function( event ) { 2669 var $button = $( event.target ), 2670 pluginName = $( this ).data( 'name' ); 2671 2672 event.preventDefault(); 2673 2674 if ( $button.hasClass( 'updating-message' ) ) { 2675 return; 2676 } 2677 2678 if ( wp.updates.shouldRequestFilesystemCredentials && ! wp.updates.ajaxLocked ) { 2679 wp.updates.requestFilesystemCredentials( event ); 2680 2681 $document.on( 'credential-modal-cancel', function() { 2682 2683 $button 2684 .removeClass( 'updating-message' ) 2685 .attr( 2686 'aria-label', 2687 sprintf( 2688 /* translators: %s: Plugin name. */ 2689 _x( 'Install %s now', 'plugin' ), 2690 pluginName 2691 ) 2692 ) 2693 .text( _x( 'Install Now', 'plugin' ) ); 2694 2695 wp.a11y.speak( __( 'Update canceled.' ) ); 2696 } ); 2697 } 2698 2699 wp.updates.installPlugin( { 2700 slug: $button.data( 'slug' ), 2701 pagenow: pagenow, 2702 success: wp.updates.installImporterSuccess, 2703 error: wp.updates.installImporterError 2704 } ); 2705 } ); 2706 2707 /** 2708 * Click handler for plugin deletions. 2709 * 2710 * @since 4.6.0 2711 * 2712 * @param {Event} event Event interface. 2713 */ 2714 $bulkActionForm.on( 'click', '[data-plugin] a.delete', function( event ) { 2715 var $pluginRow = $( event.target ).parents( 'tr' ), 2716 confirmMessage; 2717 2718 if ( $pluginRow.hasClass( 'is-uninstallable' ) ) { 2719 confirmMessage = sprintf( 2720 /* translators: %s: Plugin name. */ 2721 __( 'Are you sure you want to delete %s and its data?' ), 2722 $pluginRow.find( '.plugin-title strong' ).text() 2723 ); 2724 } else { 2725 confirmMessage = sprintf( 2726 /* translators: %s: Plugin name. */ 2727 __( 'Are you sure you want to delete %s?' ), 2728 $pluginRow.find( '.plugin-title strong' ).text() 2729 ); 2730 } 2731 2732 event.preventDefault(); 2733 2734 if ( ! window.confirm( confirmMessage ) ) { 2735 return; 2736 } 2737 2738 wp.updates.maybeRequestFilesystemCredentials( event ); 2739 2740 wp.updates.deletePlugin( { 2741 plugin: $pluginRow.data( 'plugin' ), 2742 slug: $pluginRow.data( 'slug' ) 2743 } ); 2744 2745 } ); 2746 2747 /** 2748 * Click handler for theme updates. 2749 * 2750 * @since 4.6.0 2751 * 2752 * @param {Event} event Event interface. 2753 */ 2754 $document.on( 'click', '.themes-php.network-admin .update-link', function( event ) { 2755 var $message = $( event.target ), 2756 $themeRow = $message.parents( 'tr' ); 2757 2758 event.preventDefault(); 2759 2760 if ( $message.hasClass( 'updating-message' ) || $message.hasClass( 'button-disabled' ) ) { 2761 return; 2762 } 2763 2764 wp.updates.maybeRequestFilesystemCredentials( event ); 2765 2766 // Return the user to the input box of the theme's table row after closing the modal. 2767 wp.updates.$elToReturnFocusToFromCredentialsModal = $themeRow.find( '.check-column input' ); 2768 wp.updates.updateTheme( { 2769 slug: $themeRow.data( 'slug' ) 2770 } ); 2771 } ); 2772 2773 /** 2774 * Click handler for theme deletions. 2775 * 2776 * @since 4.6.0 2777 * 2778 * @param {Event} event Event interface. 2779 */ 2780 $document.on( 'click', '.themes-php.network-admin a.delete', function( event ) { 2781 var $themeRow = $( event.target ).parents( 'tr' ), 2782 confirmMessage = sprintf( 2783 /* translators: %s: Theme name. */ 2784 __( 'Are you sure you want to delete %s?' ), 2785 $themeRow.find( '.theme-title strong' ).text() 2786 ); 2787 2788 event.preventDefault(); 2789 2790 if ( ! window.confirm( confirmMessage ) ) { 2791 return; 2792 } 2793 2794 wp.updates.maybeRequestFilesystemCredentials( event ); 2795 2796 wp.updates.deleteTheme( { 2797 slug: $themeRow.data( 'slug' ) 2798 } ); 2799 } ); 2800 2801 /** 2802 * Bulk action handler for plugins and themes. 2803 * 2804 * Handles both deletions and updates. 2805 * 2806 * @since 4.6.0 2807 * 2808 * @param {Event} event Event interface. 2809 */ 2810 $bulkActionForm.on( 'click', '[type="submit"]:not([name="clear-recent-list"])', function( event ) { 2811 var bulkAction = $( event.target ).siblings( 'select' ).val(), 2812 itemsSelected = $bulkActionForm.find( 'input[name="checked[]"]:checked' ), 2813 success = 0, 2814 error = 0, 2815 errorMessages = [], 2816 type, action; 2817 2818 // Determine which type of item we're dealing with. 2819 switch ( pagenow ) { 2820 case 'plugins': 2821 case 'plugins-network': 2822 type = 'plugin'; 2823 break; 2824 2825 case 'themes-network': 2826 type = 'theme'; 2827 break; 2828 2829 default: 2830 return; 2831 } 2832 2833 // Bail if there were no items selected. 2834 if ( ! itemsSelected.length ) { 2835 bulkAction = false; 2836 } 2837 2838 // Determine the type of request we're dealing with. 2839 switch ( bulkAction ) { 2840 case 'update-selected': 2841 action = bulkAction.replace( 'selected', type ); 2842 break; 2843 2844 case 'delete-selected': 2845 var confirmMessage = 'plugin' === type ? 2846 __( 'Are you sure you want to delete the selected plugins and their data?' ) : 2847 __( 'Caution: These themes may be active on other sites in the network. Are you sure you want to proceed?' ); 2848 2849 if ( ! window.confirm( confirmMessage ) ) { 2850 event.preventDefault(); 2851 return; 2852 } 2853 2854 action = bulkAction.replace( 'selected', type ); 2855 break; 2856 2857 default: 2858 return; 2859 } 2860 2861 wp.updates.maybeRequestFilesystemCredentials( event ); 2862 2863 event.preventDefault(); 2864 2865 // Un-check the bulk checkboxes. 2866 $bulkActionForm.find( '.manage-column [type="checkbox"]' ).prop( 'checked', false ); 2867 2868 $document.trigger( 'wp-' + type + '-bulk-' + bulkAction, itemsSelected ); 2869 2870 // Find all the checkboxes which have been checked. 2871 itemsSelected.each( function( index, element ) { 2872 var $checkbox = $( element ), 2873 $itemRow = $checkbox.parents( 'tr' ); 2874 2875 // Only add update-able items to the update queue. 2876 if ( 'update-selected' === bulkAction && ( ! $itemRow.hasClass( 'update' ) || $itemRow.find( 'notice-error' ).length ) ) { 2877 2878 // Un-check the box. 2879 $checkbox.prop( 'checked', false ); 2880 return; 2881 } 2882 2883 // Don't add items to the update queue again, even if the user clicks the update button several times. 2884 if ( 'update-selected' === bulkAction && $itemRow.hasClass( 'is-enqueued' ) ) { 2885 return; 2886 } 2887 2888 $itemRow.addClass( 'is-enqueued' ); 2889 2890 // Add it to the queue. 2891 wp.updates.queue.push( { 2892 action: action, 2893 data: { 2894 plugin: $itemRow.data( 'plugin' ), 2895 slug: $itemRow.data( 'slug' ) 2896 } 2897 } ); 2898 } ); 2899 2900 // Display bulk notification for updates of any kind. 2901 $document.on( 'wp-plugin-update-success wp-plugin-update-error wp-theme-update-success wp-theme-update-error', function( event, response ) { 2902 var $itemRow = $( '[data-slug="' + response.slug + '"]' ), 2903 $bulkActionNotice, itemName; 2904 2905 if ( 'wp-' + response.update + '-update-success' === event.type ) { 2906 success++; 2907 } else { 2908 itemName = response.pluginName ? response.pluginName : $itemRow.find( '.column-primary strong' ).text(); 2909 2910 error++; 2911 errorMessages.push( itemName + ': ' + response.errorMessage ); 2912 } 2913 2914 $itemRow.find( 'input[name="checked[]"]:checked' ).prop( 'checked', false ); 2915 2916 wp.updates.adminNotice = wp.template( 'wp-bulk-updates-admin-notice' ); 2917 2918 var successMessage = null; 2919 2920 if ( success ) { 2921 if ( 'plugin' === response.update ) { 2922 successMessage = sprintf( 2923 /* translators: %s: Number of plugins. */ 2924 _n( '%s plugin successfully updated.', '%s plugins successfully updated.', success ), 2925 success 2926 ); 2927 } else { 2928 successMessage = sprintf( 2929 /* translators: %s: Number of themes. */ 2930 _n( '%s theme successfully updated.', '%s themes successfully updated.', success ), 2931 success 2932 ); 2933 } 2934 } 2935 2936 var errorMessage = null; 2937 2938 if ( error ) { 2939 errorMessage = sprintf( 2940 /* translators: %s: Number of failed updates. */ 2941 _n( '%s update failed.', '%s updates failed.', error ), 2942 error 2943 ); 2944 } 2945 2946 wp.updates.addAdminNotice( { 2947 id: 'bulk-action-notice', 2948 className: 'bulk-action-notice', 2949 successMessage: successMessage, 2950 errorMessage: errorMessage, 2951 errorMessages: errorMessages, 2952 type: response.update 2953 } ); 2954 2955 $bulkActionNotice = $( '#bulk-action-notice' ).on( 'click', 'button', function() { 2956 // $( this ) is the clicked button, no need to get it again. 2957 $( this ) 2958 .toggleClass( 'bulk-action-errors-collapsed' ) 2959 .attr( 'aria-expanded', ! $( this ).hasClass( 'bulk-action-errors-collapsed' ) ); 2960 // Show the errors list. 2961 $bulkActionNotice.find( '.bulk-action-errors' ).toggleClass( 'hidden' ); 2962 } ); 2963 2964 if ( error > 0 && ! wp.updates.queue.length ) { 2965 $( 'html, body' ).animate( { scrollTop: 0 } ); 2966 } 2967 } ); 2968 2969 // Reset admin notice template after #bulk-action-notice was added. 2970 $document.on( 'wp-updates-notice-added', function() { 2971 wp.updates.adminNotice = wp.template( 'wp-updates-admin-notice' ); 2972 } ); 2973 2974 // Check the queue, now that the event handlers have been added. 2975 wp.updates.queueChecker(); 2976 } ); 2977 2978 if ( $pluginInstallSearch.length ) { 2979 $pluginInstallSearch.attr( 'aria-describedby', 'live-search-desc' ); 2980 } 2981 2982 // Track the previous search string length. 2983 var previousSearchStringLength = 0; 2984 wp.updates.shouldSearch = function( searchStringLength ) { 2985 var shouldSearch = searchStringLength >= wp.updates.searchMinCharacters || 2986 previousSearchStringLength > wp.updates.searchMinCharacters; 2987 previousSearchStringLength = searchStringLength; 2988 return shouldSearch; 2989 }; 2990 2991 /** 2992 * Handles changes to the plugin search box on the new-plugin page, 2993 * searching the repository dynamically. 2994 * 2995 * @since 4.6.0 2996 */ 2997 $pluginInstallSearch.on( 'keyup input', _.debounce( function( event, eventtype ) { 2998 var $searchTab = $( '.plugin-install-search' ), data, searchLocation, 2999 searchStringLength = $pluginInstallSearch.val().length; 3000 3001 data = { 3002 _ajax_nonce: wp.updates.ajaxNonce, 3003 s: encodeURIComponent( event.target.value ), 3004 tab: 'search', 3005 type: $( '#typeselector' ).val(), 3006 pagenow: pagenow 3007 }; 3008 searchLocation = location.href.split( '?' )[ 0 ] + '?' + $.param( _.omit( data, [ '_ajax_nonce', 'pagenow' ] ) ); 3009 3010 // Set the autocomplete attribute, turning off autocomplete 1 character before ajax search kicks in. 3011 if ( wp.updates.shouldSearch( searchStringLength ) ) { 3012 $pluginInstallSearch.attr( 'autocomplete', 'off' ); 3013 } else { 3014 $pluginInstallSearch.attr( 'autocomplete', 'on' ); 3015 return; 3016 } 3017 3018 // Clear on escape. 3019 if ( 'keyup' === event.type && 27 === event.which ) { 3020 event.target.value = ''; 3021 } 3022 3023 if ( wp.updates.searchTerm === data.s && 'typechange' !== eventtype ) { 3024 return; 3025 } else { 3026 $pluginFilter.empty(); 3027 wp.updates.searchTerm = data.s; 3028 } 3029 3030 if ( window.history && window.history.replaceState ) { 3031 window.history.replaceState( null, '', searchLocation ); 3032 } 3033 3034 if ( ! $searchTab.length ) { 3035 $searchTab = $( '<li class="plugin-install-search" />' ) 3036 .append( $( '<a />', { 3037 'class': 'current', 3038 'href': searchLocation, 3039 'text': __( 'Search Results' ) 3040 } ) ); 3041 3042 $( '.wp-filter .filter-links .current' ) 3043 .removeClass( 'current' ) 3044 .parents( '.filter-links' ) 3045 .prepend( $searchTab ); 3046 3047 $pluginFilter.prev( 'p' ).remove(); 3048 $( '.plugins-popular-tags-wrapper' ).remove(); 3049 } 3050 3051 if ( 'undefined' !== typeof wp.updates.searchRequest ) { 3052 wp.updates.searchRequest.abort(); 3053 } 3054 $( 'body' ).addClass( 'loading-content' ); 3055 3056 wp.updates.searchRequest = wp.ajax.post( 'search-install-plugins', data ).done( function( response ) { 3057 $( 'body' ).removeClass( 'loading-content' ); 3058 $pluginFilter.append( response.items ); 3059 delete wp.updates.searchRequest; 3060 3061 if ( 0 === response.count ) { 3062 wp.a11y.speak( __( 'You do not appear to have any plugins available at this time.' ) ); 3063 } else { 3064 wp.a11y.speak( 3065 sprintf( 3066 /* translators: %s: Number of plugins. */ 3067 __( 'Number of plugins found: %d' ), 3068 response.count 3069 ) 3070 ); 3071 } 3072 } ); 3073 }, 1000 ) ); 3074 3075 if ( $pluginSearch.length ) { 3076 $pluginSearch.attr( 'aria-describedby', 'live-search-desc' ); 3077 3078 } 3079 3080 /** 3081 * Handles changes to the plugin search box on the Installed Plugins screen, 3082 * searching the plugin list dynamically. 3083 * 3084 * @since 4.6.0 3085 */ 3086 $pluginSearch.on( 'keyup input', _.debounce( function( event ) { 3087 var data = { 3088 _ajax_nonce: wp.updates.ajaxNonce, 3089 s: encodeURIComponent( event.target.value ), 3090 pagenow: pagenow, 3091 plugin_status: 'all' 3092 }, 3093 queryArgs, 3094 searchStringLength = $pluginSearch.val().length; 3095 3096 // Set the autocomplete attribute, turning off autocomplete 1 character before ajax search kicks in. 3097 if ( wp.updates.shouldSearch( searchStringLength ) ) { 3098 $pluginSearch.attr( 'autocomplete', 'off' ); 3099 } else { 3100 $pluginSearch.attr( 'autocomplete', 'on' ); 3101 return; 3102 } 3103 3104 // Clear on escape. 3105 if ( 'keyup' === event.type && 27 === event.which ) { 3106 event.target.value = ''; 3107 } 3108 3109 if ( wp.updates.searchTerm === data.s ) { 3110 return; 3111 } else { 3112 wp.updates.searchTerm = data.s; 3113 } 3114 3115 queryArgs = _.object( _.compact( _.map( location.search.slice( 1 ).split( '&' ), function( item ) { 3116 if ( item ) return item.split( '=' ); 3117 } ) ) ); 3118 3119 data.plugin_status = queryArgs.plugin_status || 'all'; 3120 3121 if ( window.history && window.history.replaceState ) { 3122 window.history.replaceState( null, '', location.href.split( '?' )[ 0 ] + '?s=' + data.s + '&plugin_status=' + data.plugin_status ); 3123 } 3124 3125 if ( 'undefined' !== typeof wp.updates.searchRequest ) { 3126 wp.updates.searchRequest.abort(); 3127 } 3128 3129 $bulkActionForm.empty(); 3130 $( 'body' ).addClass( 'loading-content' ); 3131 $( '.subsubsub .current' ).removeClass( 'current' ); 3132 3133 wp.updates.searchRequest = wp.ajax.post( 'search-plugins', data ).done( function( response ) { 3134 3135 // Can we just ditch this whole subtitle business? 3136 var $subTitle = $( '<span />' ).addClass( 'subtitle' ).html( 3137 sprintf( 3138 /* translators: %s: Search query. */ 3139 __( 'Search results for: %s' ), 3140 '<strong>' + _.escape( decodeURIComponent( data.s ) ) + '</strong>' 3141 ) ), 3142 $oldSubTitle = $( '.wrap .subtitle' ); 3143 3144 if ( ! data.s.length ) { 3145 $oldSubTitle.remove(); 3146 $( '.subsubsub .' + data.plugin_status + ' a' ).addClass( 'current' ); 3147 } else if ( $oldSubTitle.length ) { 3148 $oldSubTitle.replaceWith( $subTitle ); 3149 } else { 3150 $( '.wp-header-end' ).before( $subTitle ); 3151 } 3152 3153 $( 'body' ).removeClass( 'loading-content' ); 3154 $bulkActionForm.append( response.items ); 3155 delete wp.updates.searchRequest; 3156 3157 if ( 0 === response.count ) { 3158 wp.a11y.speak( __( 'No plugins found. Try a different search.' ) ); 3159 } else { 3160 wp.a11y.speak( 3161 sprintf( 3162 /* translators: %s: Number of plugins. */ 3163 __( 'Number of plugins found: %d' ), 3164 response.count 3165 ) 3166 ); 3167 } 3168 } ); 3169 }, 500 ) ); 3170 3171 /** 3172 * Trigger a search event when the search form gets submitted. 3173 * 3174 * @since 4.6.0 3175 */ 3176 $document.on( 'submit', '.search-plugins', function( event ) { 3177 event.preventDefault(); 3178 3179 $( 'input.wp-filter-search' ).trigger( 'input' ); 3180 } ); 3181 3182 /** 3183 * Trigger a search event when the "Try Again" button is clicked. 3184 * 3185 * @since 4.9.0 3186 */ 3187 $document.on( 'click', '.try-again', function( event ) { 3188 event.preventDefault(); 3189 $pluginInstallSearch.trigger( 'input' ); 3190 } ); 3191 3192 /** 3193 * Trigger a search event when the search type gets changed. 3194 * 3195 * @since 4.6.0 3196 */ 3197 $( '#typeselector' ).on( 'change', function() { 3198 var $search = $( 'input[name="s"]' ); 3199 3200 if ( $search.val().length ) { 3201 $search.trigger( 'input', 'typechange' ); 3202 } 3203 } ); 3204 3205 /** 3206 * Click handler for updating a plugin from the details modal on `plugin-install.php`. 3207 * 3208 * @since 4.2.0 3209 * 3210 * @param {Event} event Event interface. 3211 */ 3212 $( '#plugin_update_from_iframe' ).on( 'click', function( event ) { 3213 var target = window.parent === window ? null : window.parent, 3214 update; 3215 3216 $.support.postMessage = !! window.postMessage; 3217 3218 if ( false === $.support.postMessage || null === target || -1 !== window.parent.location.pathname.indexOf( 'update-core.php' ) ) { 3219 return; 3220 } 3221 3222 event.preventDefault(); 3223 3224 update = { 3225 action: 'update-plugin', 3226 data: { 3227 plugin: $( this ).data( 'plugin' ), 3228 slug: $( this ).data( 'slug' ) 3229 } 3230 }; 3231 3232 target.postMessage( JSON.stringify( update ), window.location.origin ); 3233 } ); 3234 3235 /** 3236 * Handles postMessage events. 3237 * 3238 * @since 4.2.0 3239 * @since 4.6.0 Switched `update-plugin` action to use the queue. 3240 * 3241 * @param {Event} event Event interface. 3242 */ 3243 $( window ).on( 'message', function( event ) { 3244 var originalEvent = event.originalEvent, 3245 expectedOrigin = document.location.protocol + '//' + document.location.host, 3246 message; 3247 3248 if ( originalEvent.origin !== expectedOrigin ) { 3249 return; 3250 } 3251 3252 try { 3253 message = JSON.parse( originalEvent.data ); 3254 } catch ( e ) { 3255 return; 3256 } 3257 3258 if ( ! message ) { 3259 return; 3260 } 3261 3262 if ( 3263 'undefined' !== typeof message.status && 3264 'undefined' !== typeof message.slug && 3265 'undefined' !== typeof message.text && 3266 'undefined' !== typeof message.ariaLabel 3267 ) { 3268 var $card = $( '.plugin-card-' + message.slug ), 3269 $message = $card.find( '[data-slug="' + message.slug + '"]' ); 3270 3271 if ( 'undefined' !== typeof message.removeClasses ) { 3272 $message.removeClass( message.removeClasses ); 3273 } 3274 3275 if ( 'undefined' !== typeof message.addClasses ) { 3276 $message.addClass( message.addClasses ); 3277 } 3278 3279 if ( '' === message.ariaLabel ) { 3280 $message.removeAttr( 'aria-label' ); 3281 } else { 3282 $message.attr( 'aria-label', message.ariaLabel ); 3283 } 3284 3285 if ( 'dependencies-check-success' === message.status ) { 3286 $message 3287 .attr( 'data-name', message.pluginName ) 3288 .attr( 'data-slug', message.slug ) 3289 .attr( 'data-plugin', message.plugin ) 3290 .attr( 'href', message.href ); 3291 } 3292 3293 $message.text( message.text ); 3294 } 3295 3296 if ( 'undefined' === typeof message.action ) { 3297 return; 3298 } 3299 3300 switch ( message.action ) { 3301 3302 // Called from `wp-admin/includes/class-wp-upgrader-skins.php`. 3303 case 'decrementUpdateCount': 3304 /** @property {string} message.upgradeType */ 3305 wp.updates.decrementCount( message.upgradeType ); 3306 break; 3307 3308 case 'install-plugin': 3309 case 'update-plugin': 3310 if ( 'undefined' === typeof message.data || 'undefined' === typeof message.data.slug ) { 3311 return; 3312 } 3313 3314 message.data = wp.updates._addCallbacks( message.data, message.action ); 3315 3316 wp.updates.queue.push( message ); 3317 wp.updates.queueChecker(); 3318 break; 3319 } 3320 } ); 3321 3322 /** 3323 * Adds a callback to display a warning before leaving the page. 3324 * 3325 * @since 4.2.0 3326 */ 3327 $( window ).on( 'beforeunload', wp.updates.beforeunload ); 3328 3329 /** 3330 * Prevents the page form scrolling when activating auto-updates with the Spacebar key. 3331 * 3332 * @since 5.5.0 3333 */ 3334 $document.on( 'keydown', '.column-auto-updates .toggle-auto-update, .theme-overlay .toggle-auto-update', function( event ) { 3335 if ( 32 === event.which ) { 3336 event.preventDefault(); 3337 } 3338 } ); 3339 3340 /** 3341 * Click and keyup handler for enabling and disabling plugin and theme auto-updates. 3342 * 3343 * These controls can be either links or buttons. When JavaScript is enabled, 3344 * we want them to behave like buttons. An ARIA role `button` is added via 3345 * the JavaScript that targets elements with the CSS class `aria-button-if-js`. 3346 * 3347 * @since 5.5.0 3348 */ 3349 $document.on( 'click keyup', '.column-auto-updates .toggle-auto-update, .theme-overlay .toggle-auto-update', function( event ) { 3350 var data, asset, type, $parent, 3351 $toggler = $( this ), 3352 action = $toggler.attr( 'data-wp-action' ), 3353 $label = $toggler.find( '.label' ); 3354 3355 if ( 'keyup' === event.type && 32 !== event.which ) { 3356 return; 3357 } 3358 3359 if ( 'themes' !== pagenow ) { 3360 $parent = $toggler.closest( '.column-auto-updates' ); 3361 } else { 3362 $parent = $toggler.closest( '.theme-autoupdate' ); 3363 } 3364 3365 event.preventDefault(); 3366 3367 // Prevent multiple simultaneous requests. 3368 if ( $toggler.attr( 'data-doing-ajax' ) === 'yes' ) { 3369 return; 3370 } 3371 3372 $toggler.attr( 'data-doing-ajax', 'yes' ); 3373 3374 switch ( pagenow ) { 3375 case 'plugins': 3376 case 'plugins-network': 3377 type = 'plugin'; 3378 asset = $toggler.closest( 'tr' ).attr( 'data-plugin' ); 3379 break; 3380 case 'themes-network': 3381 type = 'theme'; 3382 asset = $toggler.closest( 'tr' ).attr( 'data-slug' ); 3383 break; 3384 case 'themes': 3385 type = 'theme'; 3386 asset = $toggler.attr( 'data-slug' ); 3387 break; 3388 } 3389 3390 // Clear any previous errors. 3391 $parent.find( '.notice.notice-error' ).addClass( 'hidden' ); 3392 3393 // Show loading status. 3394 if ( 'enable' === action ) { 3395 $label.text( __( 'Enabling...' ) ); 3396 } else { 3397 $label.text( __( 'Disabling...' ) ); 3398 } 3399 3400 $toggler.find( '.dashicons-update' ).removeClass( 'hidden' ); 3401 3402 data = { 3403 action: 'toggle-auto-updates', 3404 _ajax_nonce: settings.ajax_nonce, 3405 state: action, 3406 type: type, 3407 asset: asset 3408 }; 3409 3410 $.post( window.ajaxurl, data ) 3411 .done( function( response ) { 3412 var $enabled, $disabled, enabledNumber, disabledNumber, errorMessage, 3413 href = $toggler.attr( 'href' ); 3414 3415 if ( ! response.success ) { 3416 // if WP returns 0 for response (which can happen in a few cases), 3417 // output the general error message since we won't have response.data.error. 3418 if ( response.data && response.data.error ) { 3419 errorMessage = response.data.error; 3420 } else { 3421 errorMessage = __( 'The request could not be completed.' ); 3422 } 3423 3424 $parent.find( '.notice.notice-error' ).removeClass( 'hidden' ).find( 'p' ).text( errorMessage ); 3425 wp.a11y.speak( errorMessage, 'assertive' ); 3426 return; 3427 } 3428 3429 // Update the counts in the enabled/disabled views if on a screen 3430 // with a list table. 3431 if ( 'themes' !== pagenow ) { 3432 $enabled = $( '.auto-update-enabled span' ); 3433 $disabled = $( '.auto-update-disabled span' ); 3434 enabledNumber = parseInt( $enabled.text().replace( /[^\d]+/g, '' ), 10 ) || 0; 3435 disabledNumber = parseInt( $disabled.text().replace( /[^\d]+/g, '' ), 10 ) || 0; 3436 3437 switch ( action ) { 3438 case 'enable': 3439 ++enabledNumber; 3440 --disabledNumber; 3441 break; 3442 case 'disable': 3443 --enabledNumber; 3444 ++disabledNumber; 3445 break; 3446 } 3447 3448 enabledNumber = Math.max( 0, enabledNumber ); 3449 disabledNumber = Math.max( 0, disabledNumber ); 3450 3451 $enabled.text( '(' + enabledNumber + ')' ); 3452 $disabled.text( '(' + disabledNumber + ')' ); 3453 } 3454 3455 if ( 'enable' === action ) { 3456 // The toggler control can be either a link or a button. 3457 if ( $toggler[ 0 ].hasAttribute( 'href' ) ) { 3458 href = href.replace( 'action=enable-auto-update', 'action=disable-auto-update' ); 3459 $toggler.attr( 'href', href ); 3460 } 3461 $toggler.attr( 'data-wp-action', 'disable' ); 3462 3463 $label.text( __( 'Disable auto-updates' ) ); 3464 $parent.find( '.auto-update-time' ).removeClass( 'hidden' ); 3465 wp.a11y.speak( __( 'Auto-updates enabled' ) ); 3466 } else { 3467 // The toggler control can be either a link or a button. 3468 if ( $toggler[ 0 ].hasAttribute( 'href' ) ) { 3469 href = href.replace( 'action=disable-auto-update', 'action=enable-auto-update' ); 3470 $toggler.attr( 'href', href ); 3471 } 3472 $toggler.attr( 'data-wp-action', 'enable' ); 3473 3474 $label.text( __( 'Enable auto-updates' ) ); 3475 $parent.find( '.auto-update-time' ).addClass( 'hidden' ); 3476 wp.a11y.speak( __( 'Auto-updates disabled' ) ); 3477 } 3478 3479 $document.trigger( 'wp-auto-update-setting-changed', { state: action, type: type, asset: asset } ); 3480 } ) 3481 .fail( function() { 3482 $parent.find( '.notice.notice-error' ) 3483 .removeClass( 'hidden' ) 3484 .find( 'p' ) 3485 .text( __( 'The request could not be completed.' ) ); 3486 3487 wp.a11y.speak( __( 'The request could not be completed.' ), 'assertive' ); 3488 } ) 3489 .always( function() { 3490 $toggler.removeAttr( 'data-doing-ajax' ).find( '.dashicons-update' ).addClass( 'hidden' ); 3491 } ); 3492 } 3493 ); 3494 } ); 3495 })( jQuery, window.wp, window._wpUpdatesSettings );
title
Description
Body
title
Description
Body
title
Description
Body
title
Body
Generated : Tue Jan 21 08:20:01 2025 | Cross-referenced by PHPXref |