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