[ Index ] |
PHP Cross Reference of WordPress Trunk (Updated Daily) |
[Summary view] [Print] [Text view]
1 /******/ (() => { // webpackBootstrap 2 /******/ var __webpack_modules__ = ({ 3 4 /***/ 7145: 5 /***/ ((module) => { 6 7 var Selection = wp.media.model.Selection, 8 Library = wp.media.controller.Library, 9 CollectionAdd; 10 11 /** 12 * wp.media.controller.CollectionAdd 13 * 14 * A state for adding attachments to a collection (e.g. video playlist). 15 * 16 * @memberOf wp.media.controller 17 * 18 * @class 19 * @augments wp.media.controller.Library 20 * @augments wp.media.controller.State 21 * @augments Backbone.Model 22 * 23 * @param {object} [attributes] The attributes hash passed to the state. 24 * @param {string} [attributes.id=library] Unique identifier. 25 * @param {string} attributes.title Title for the state. Displays in the frame's title region. 26 * @param {boolean} [attributes.multiple=add] Whether multi-select is enabled. @todo 'add' doesn't seem do anything special, and gets used as a boolean. 27 * @param {wp.media.model.Attachments} [attributes.library] The attachments collection to browse. 28 * If one is not supplied, a collection of attachments of the specified type will be created. 29 * @param {boolean|string} [attributes.filterable=uploaded] Whether the library is filterable, and if so what filters should be shown. 30 * Accepts 'all', 'uploaded', or 'unattached'. 31 * @param {string} [attributes.menu=gallery] Initial mode for the menu region. 32 * @param {string} [attributes.content=upload] Initial mode for the content region. 33 * Overridden by persistent user setting if 'contentUserSetting' is true. 34 * @param {string} [attributes.router=browse] Initial mode for the router region. 35 * @param {string} [attributes.toolbar=gallery-add] Initial mode for the toolbar region. 36 * @param {boolean} [attributes.searchable=true] Whether the library is searchable. 37 * @param {boolean} [attributes.sortable=true] Whether the Attachments should be sortable. Depends on the orderby property being set to menuOrder on the attachments collection. 38 * @param {boolean} [attributes.autoSelect=true] Whether an uploaded attachment should be automatically added to the selection. 39 * @param {boolean} [attributes.contentUserSetting=true] Whether the content region's mode should be set and persisted per user. 40 * @param {int} [attributes.priority=100] The priority for the state link in the media menu. 41 * @param {boolean} [attributes.syncSelection=false] Whether the Attachments selection should be persisted from the last state. 42 * Defaults to false because for this state, because the library of the Edit Gallery state is the selection. 43 * @param {string} attributes.type The collection's media type. (e.g. 'video'). 44 * @param {string} attributes.collectionType The collection type. (e.g. 'playlist'). 45 */ 46 CollectionAdd = Library.extend(/** @lends wp.media.controller.CollectionAdd.prototype */{ 47 defaults: _.defaults( { 48 // Selection defaults. @see media.model.Selection 49 multiple: 'add', 50 // Attachments browser defaults. @see media.view.AttachmentsBrowser 51 filterable: 'uploaded', 52 53 priority: 100, 54 syncSelection: false 55 }, Library.prototype.defaults ), 56 57 /** 58 * @since 3.9.0 59 */ 60 initialize: function() { 61 var collectionType = this.get('collectionType'); 62 63 if ( 'video' === this.get( 'type' ) ) { 64 collectionType = 'video-' + collectionType; 65 } 66 67 this.set( 'id', collectionType + '-library' ); 68 this.set( 'toolbar', collectionType + '-add' ); 69 this.set( 'menu', collectionType ); 70 71 // If we haven't been provided a `library`, create a `Selection`. 72 if ( ! this.get('library') ) { 73 this.set( 'library', wp.media.query({ type: this.get('type') }) ); 74 } 75 Library.prototype.initialize.apply( this, arguments ); 76 }, 77 78 /** 79 * @since 3.9.0 80 */ 81 activate: function() { 82 var library = this.get('library'), 83 editLibrary = this.get('editLibrary'), 84 edit = this.frame.state( this.get('collectionType') + '-edit' ).get('library'); 85 86 if ( editLibrary && editLibrary !== edit ) { 87 library.unobserve( editLibrary ); 88 } 89 90 // Accepts attachments that exist in the original library and 91 // that do not exist in gallery's library. 92 library.validator = function( attachment ) { 93 return !! this.mirroring.get( attachment.cid ) && ! edit.get( attachment.cid ) && Selection.prototype.validator.apply( this, arguments ); 94 }; 95 96 /* 97 * Reset the library to ensure that all attachments are re-added 98 * to the collection. Do so silently, as calling `observe` will 99 * trigger the `reset` event. 100 */ 101 library.reset( library.mirroring.models, { silent: true }); 102 library.observe( edit ); 103 this.set('editLibrary', edit); 104 105 Library.prototype.activate.apply( this, arguments ); 106 } 107 }); 108 109 module.exports = CollectionAdd; 110 111 112 /***/ }), 113 114 /***/ 8612: 115 /***/ ((module) => { 116 117 var Library = wp.media.controller.Library, 118 l10n = wp.media.view.l10n, 119 $ = jQuery, 120 CollectionEdit; 121 122 /** 123 * wp.media.controller.CollectionEdit 124 * 125 * A state for editing a collection, which is used by audio and video playlists, 126 * and can be used for other collections. 127 * 128 * @memberOf wp.media.controller 129 * 130 * @class 131 * @augments wp.media.controller.Library 132 * @augments wp.media.controller.State 133 * @augments Backbone.Model 134 * 135 * @param {object} [attributes] The attributes hash passed to the state. 136 * @param {string} attributes.title Title for the state. Displays in the media menu and the frame's title region. 137 * @param {wp.media.model.Attachments} [attributes.library] The attachments collection to edit. 138 * If one is not supplied, an empty media.model.Selection collection is created. 139 * @param {boolean} [attributes.multiple=false] Whether multi-select is enabled. 140 * @param {string} [attributes.content=browse] Initial mode for the content region. 141 * @param {string} attributes.menu Initial mode for the menu region. @todo this needs a better explanation. 142 * @param {boolean} [attributes.searchable=false] Whether the library is searchable. 143 * @param {boolean} [attributes.sortable=true] Whether the Attachments should be sortable. Depends on the orderby property being set to menuOrder on the attachments collection. 144 * @param {boolean} [attributes.date=true] Whether to show the date filter in the browser's toolbar. 145 * @param {boolean} [attributes.describe=true] Whether to offer UI to describe the attachments - e.g. captioning images in a gallery. 146 * @param {boolean} [attributes.dragInfo=true] Whether to show instructional text about the attachments being sortable. 147 * @param {boolean} [attributes.dragInfoText] Instructional text about the attachments being sortable. 148 * @param {int} [attributes.idealColumnWidth=170] The ideal column width in pixels for attachments. 149 * @param {boolean} [attributes.editing=false] Whether the gallery is being created, or editing an existing instance. 150 * @param {int} [attributes.priority=60] The priority for the state link in the media menu. 151 * @param {boolean} [attributes.syncSelection=false] Whether the Attachments selection should be persisted from the last state. 152 * Defaults to false for this state, because the library passed in *is* the selection. 153 * @param {view} [attributes.SettingsView] The view to edit the collection instance settings (e.g. Playlist settings with "Show tracklist" checkbox). 154 * @param {view} [attributes.AttachmentView] The single `Attachment` view to be used in the `Attachments`. 155 * If none supplied, defaults to wp.media.view.Attachment.EditLibrary. 156 * @param {string} attributes.type The collection's media type. (e.g. 'video'). 157 * @param {string} attributes.collectionType The collection type. (e.g. 'playlist'). 158 */ 159 CollectionEdit = Library.extend(/** @lends wp.media.controller.CollectionEdit.prototype */{ 160 defaults: { 161 multiple: false, 162 sortable: true, 163 date: false, 164 searchable: false, 165 content: 'browse', 166 describe: true, 167 dragInfo: true, 168 idealColumnWidth: 170, 169 editing: false, 170 priority: 60, 171 SettingsView: false, 172 syncSelection: false 173 }, 174 175 /** 176 * @since 3.9.0 177 */ 178 initialize: function() { 179 var collectionType = this.get('collectionType'); 180 181 if ( 'video' === this.get( 'type' ) ) { 182 collectionType = 'video-' + collectionType; 183 } 184 185 this.set( 'id', collectionType + '-edit' ); 186 this.set( 'toolbar', collectionType + '-edit' ); 187 188 // If we haven't been provided a `library`, create a `Selection`. 189 if ( ! this.get('library') ) { 190 this.set( 'library', new wp.media.model.Selection() ); 191 } 192 // The single `Attachment` view to be used in the `Attachments` view. 193 if ( ! this.get('AttachmentView') ) { 194 this.set( 'AttachmentView', wp.media.view.Attachment.EditLibrary ); 195 } 196 Library.prototype.initialize.apply( this, arguments ); 197 }, 198 199 /** 200 * @since 3.9.0 201 */ 202 activate: function() { 203 var library = this.get('library'); 204 205 // Limit the library to images only. 206 library.props.set( 'type', this.get( 'type' ) ); 207 208 // Watch for uploaded attachments. 209 this.get('library').observe( wp.Uploader.queue ); 210 211 this.frame.on( 'content:render:browse', this.renderSettings, this ); 212 213 Library.prototype.activate.apply( this, arguments ); 214 }, 215 216 /** 217 * @since 3.9.0 218 */ 219 deactivate: function() { 220 // Stop watching for uploaded attachments. 221 this.get('library').unobserve( wp.Uploader.queue ); 222 223 this.frame.off( 'content:render:browse', this.renderSettings, this ); 224 225 Library.prototype.deactivate.apply( this, arguments ); 226 }, 227 228 /** 229 * Render the collection embed settings view in the browser sidebar. 230 * 231 * @todo This is against the pattern elsewhere in media. Typically the frame 232 * is responsible for adding region mode callbacks. Explain. 233 * 234 * @since 3.9.0 235 * 236 * @param {wp.media.view.attachmentsBrowser} The attachments browser view. 237 */ 238 renderSettings: function( attachmentsBrowserView ) { 239 var library = this.get('library'), 240 collectionType = this.get('collectionType'), 241 dragInfoText = this.get('dragInfoText'), 242 SettingsView = this.get('SettingsView'), 243 obj = {}; 244 245 if ( ! library || ! attachmentsBrowserView ) { 246 return; 247 } 248 249 library[ collectionType ] = library[ collectionType ] || new Backbone.Model(); 250 251 obj[ collectionType ] = new SettingsView({ 252 controller: this, 253 model: library[ collectionType ], 254 priority: 40 255 }); 256 257 attachmentsBrowserView.sidebar.set( obj ); 258 259 if ( dragInfoText ) { 260 attachmentsBrowserView.toolbar.set( 'dragInfo', new wp.media.View({ 261 el: $( '<div class="instructions">' + dragInfoText + '</div>' )[0], 262 priority: -40 263 }) ); 264 } 265 266 // Add the 'Reverse order' button to the toolbar. 267 attachmentsBrowserView.toolbar.set( 'reverse', { 268 text: l10n.reverseOrder, 269 priority: 80, 270 271 click: function() { 272 library.reset( library.toArray().reverse() ); 273 } 274 }); 275 } 276 }); 277 278 module.exports = CollectionEdit; 279 280 281 /***/ }), 282 283 /***/ 5422: 284 /***/ ((module) => { 285 286 var l10n = wp.media.view.l10n, 287 Cropper; 288 289 /** 290 * wp.media.controller.Cropper 291 * 292 * A class for cropping an image when called from the header media customization panel. 293 * 294 * @memberOf wp.media.controller 295 * 296 * @class 297 * @augments wp.media.controller.State 298 * @augments Backbone.Model 299 */ 300 Cropper = wp.media.controller.State.extend(/** @lends wp.media.controller.Cropper.prototype */{ 301 defaults: { 302 id: 'cropper', 303 title: l10n.cropImage, 304 // Region mode defaults. 305 toolbar: 'crop', 306 content: 'crop', 307 router: false, 308 canSkipCrop: false, 309 310 // Default doCrop Ajax arguments to allow the Customizer (for example) to inject state. 311 doCropArgs: {} 312 }, 313 314 /** 315 * Shows the crop image window when called from the Add new image button. 316 * 317 * @since 4.2.0 318 * 319 * @return {void} 320 */ 321 activate: function() { 322 this.frame.on( 'content:create:crop', this.createCropContent, this ); 323 this.frame.on( 'close', this.removeCropper, this ); 324 this.set('selection', new Backbone.Collection(this.frame._selection.single)); 325 }, 326 327 /** 328 * Changes the state of the toolbar window to browse mode. 329 * 330 * @since 4.2.0 331 * 332 * @return {void} 333 */ 334 deactivate: function() { 335 this.frame.toolbar.mode('browse'); 336 }, 337 338 /** 339 * Creates the crop image window. 340 * 341 * Initialized when clicking on the Select and Crop button. 342 * 343 * @since 4.2.0 344 * 345 * @fires crop window 346 * 347 * @return {void} 348 */ 349 createCropContent: function() { 350 this.cropperView = new wp.media.view.Cropper({ 351 controller: this, 352 attachment: this.get('selection').first() 353 }); 354 this.cropperView.on('image-loaded', this.createCropToolbar, this); 355 this.frame.content.set(this.cropperView); 356 357 }, 358 359 /** 360 * Removes the image selection and closes the cropping window. 361 * 362 * @since 4.2.0 363 * 364 * @return {void} 365 */ 366 removeCropper: function() { 367 this.imgSelect.cancelSelection(); 368 this.imgSelect.setOptions({remove: true}); 369 this.imgSelect.update(); 370 this.cropperView.remove(); 371 }, 372 373 /** 374 * Checks if cropping can be skipped and creates crop toolbar accordingly. 375 * 376 * @since 4.2.0 377 * 378 * @return {void} 379 */ 380 createCropToolbar: function() { 381 var canSkipCrop, hasRequiredAspectRatio, suggestedCropSize, toolbarOptions; 382 383 suggestedCropSize = this.get( 'suggestedCropSize' ); 384 hasRequiredAspectRatio = this.get( 'hasRequiredAspectRatio' ); 385 canSkipCrop = this.get( 'canSkipCrop' ) || false; 386 387 toolbarOptions = { 388 controller: this.frame, 389 items: { 390 insert: { 391 style: 'primary', 392 text: l10n.cropImage, 393 priority: 80, 394 requires: { library: false, selection: false }, 395 396 click: function() { 397 var controller = this.controller, 398 selection; 399 400 selection = controller.state().get('selection').first(); 401 selection.set({cropDetails: controller.state().imgSelect.getSelection()}); 402 403 this.$el.text(l10n.cropping); 404 this.$el.attr('disabled', true); 405 406 controller.state().doCrop( selection ).done( function( croppedImage ) { 407 controller.trigger('cropped', croppedImage ); 408 controller.close(); 409 }).fail( function() { 410 controller.trigger('content:error:crop'); 411 }); 412 } 413 } 414 } 415 }; 416 417 if ( canSkipCrop || hasRequiredAspectRatio ) { 418 _.extend( toolbarOptions.items, { 419 skip: { 420 style: 'secondary', 421 text: l10n.skipCropping, 422 priority: 70, 423 requires: { library: false, selection: false }, 424 click: function() { 425 var controller = this.controller, 426 selection = controller.state().get( 'selection' ).first(); 427 428 controller.state().cropperView.remove(); 429 430 // Apply the suggested crop size. 431 if ( hasRequiredAspectRatio && !canSkipCrop ) { 432 selection.set({cropDetails: suggestedCropSize}); 433 controller.state().doCrop( selection ).done( function( croppedImage ) { 434 controller.trigger( 'cropped', croppedImage ); 435 controller.close(); 436 }).fail( function() { 437 controller.trigger( 'content:error:crop' ); 438 }); 439 return; 440 } 441 442 // Skip the cropping process. 443 controller.trigger( 'skippedcrop', selection ); 444 controller.close(); 445 } 446 } 447 }); 448 } 449 450 this.frame.toolbar.set( new wp.media.view.Toolbar(toolbarOptions) ); 451 }, 452 453 /** 454 * Creates an object with the image attachment and crop properties. 455 * 456 * @since 4.2.0 457 * 458 * @return {$.promise} A jQuery promise with the custom header crop details. 459 */ 460 doCrop: function( attachment ) { 461 return wp.ajax.post( 'custom-header-crop', _.extend( 462 {}, 463 this.defaults.doCropArgs, 464 { 465 nonce: attachment.get( 'nonces' ).edit, 466 id: attachment.get( 'id' ), 467 cropDetails: attachment.get( 'cropDetails' ) 468 } 469 ) ); 470 } 471 }); 472 473 module.exports = Cropper; 474 475 476 /***/ }), 477 478 /***/ 9660: 479 /***/ ((module) => { 480 481 var Controller = wp.media.controller, 482 CustomizeImageCropper; 483 484 /** 485 * A state for cropping an image in the customizer. 486 * 487 * @since 4.3.0 488 * 489 * @constructs wp.media.controller.CustomizeImageCropper 490 * @memberOf wp.media.controller 491 * @augments wp.media.controller.CustomizeImageCropper.Cropper 492 * @inheritDoc 493 */ 494 CustomizeImageCropper = Controller.Cropper.extend(/** @lends wp.media.controller.CustomizeImageCropper.prototype */{ 495 /** 496 * Posts the crop details to the admin. 497 * 498 * Uses crop measurements when flexible in both directions. 499 * Constrains flexible side based on image ratio and size of the fixed side. 500 * 501 * @since 4.3.0 502 * 503 * @param {Object} attachment The attachment to crop. 504 * 505 * @return {$.promise} A jQuery promise that represents the crop image request. 506 */ 507 doCrop: function( attachment ) { 508 var cropDetails = attachment.get( 'cropDetails' ), 509 control = this.get( 'control' ), 510 ratio = cropDetails.width / cropDetails.height; 511 512 // Use crop measurements when flexible in both directions. 513 if ( control.params.flex_width && control.params.flex_height ) { 514 cropDetails.dst_width = cropDetails.width; 515 cropDetails.dst_height = cropDetails.height; 516 517 // Constrain flexible side based on image ratio and size of the fixed side. 518 } else { 519 cropDetails.dst_width = control.params.flex_width ? control.params.height * ratio : control.params.width; 520 cropDetails.dst_height = control.params.flex_height ? control.params.width / ratio : control.params.height; 521 } 522 523 return wp.ajax.post( 'crop-image', { 524 wp_customize: 'on', 525 nonce: attachment.get( 'nonces' ).edit, 526 id: attachment.get( 'id' ), 527 context: control.id, 528 cropDetails: cropDetails 529 } ); 530 } 531 }); 532 533 module.exports = CustomizeImageCropper; 534 535 536 /***/ }), 537 538 /***/ 5663: 539 /***/ ((module) => { 540 541 var l10n = wp.media.view.l10n, 542 EditImage; 543 544 /** 545 * wp.media.controller.EditImage 546 * 547 * A state for editing (cropping, etc.) an image. 548 * 549 * @memberOf wp.media.controller 550 * 551 * @class 552 * @augments wp.media.controller.State 553 * @augments Backbone.Model 554 * 555 * @param {object} attributes The attributes hash passed to the state. 556 * @param {wp.media.model.Attachment} attributes.model The attachment. 557 * @param {string} [attributes.id=edit-image] Unique identifier. 558 * @param {string} [attributes.title=Edit Image] Title for the state. Displays in the media menu and the frame's title region. 559 * @param {string} [attributes.content=edit-image] Initial mode for the content region. 560 * @param {string} [attributes.toolbar=edit-image] Initial mode for the toolbar region. 561 * @param {string} [attributes.menu=false] Initial mode for the menu region. 562 * @param {string} [attributes.url] Unused. @todo Consider removal. 563 */ 564 EditImage = wp.media.controller.State.extend(/** @lends wp.media.controller.EditImage.prototype */{ 565 defaults: { 566 id: 'edit-image', 567 title: l10n.editImage, 568 menu: false, 569 toolbar: 'edit-image', 570 content: 'edit-image', 571 url: '' 572 }, 573 574 /** 575 * Activates a frame for editing a featured image. 576 * 577 * @since 3.9.0 578 * 579 * @return {void} 580 */ 581 activate: function() { 582 this.frame.on( 'toolbar:render:edit-image', _.bind( this.toolbar, this ) ); 583 }, 584 585 /** 586 * Deactivates a frame for editing a featured image. 587 * 588 * @since 3.9.0 589 * 590 * @return {void} 591 */ 592 deactivate: function() { 593 this.frame.off( 'toolbar:render:edit-image' ); 594 }, 595 596 /** 597 * Adds a toolbar with a back button. 598 * 599 * When the back button is pressed it checks whether there is a previous state. 600 * In case there is a previous state it sets that previous state otherwise it 601 * closes the frame. 602 * 603 * @since 3.9.0 604 * 605 * @return {void} 606 */ 607 toolbar: function() { 608 var frame = this.frame, 609 lastState = frame.lastState(), 610 previous = lastState && lastState.id; 611 612 frame.toolbar.set( new wp.media.view.Toolbar({ 613 controller: frame, 614 items: { 615 back: { 616 style: 'primary', 617 text: l10n.back, 618 priority: 20, 619 click: function() { 620 if ( previous ) { 621 frame.setState( previous ); 622 } else { 623 frame.close(); 624 } 625 } 626 } 627 } 628 }) ); 629 } 630 }); 631 632 module.exports = EditImage; 633 634 635 /***/ }), 636 637 /***/ 4910: 638 /***/ ((module) => { 639 640 var l10n = wp.media.view.l10n, 641 $ = Backbone.$, 642 Embed; 643 644 /** 645 * wp.media.controller.Embed 646 * 647 * A state for embedding media from a URL. 648 * 649 * @memberOf wp.media.controller 650 * 651 * @class 652 * @augments wp.media.controller.State 653 * @augments Backbone.Model 654 * 655 * @param {object} attributes The attributes hash passed to the state. 656 * @param {string} [attributes.id=embed] Unique identifier. 657 * @param {string} [attributes.title=Insert From URL] Title for the state. Displays in the media menu and the frame's title region. 658 * @param {string} [attributes.content=embed] Initial mode for the content region. 659 * @param {string} [attributes.menu=default] Initial mode for the menu region. 660 * @param {string} [attributes.toolbar=main-embed] Initial mode for the toolbar region. 661 * @param {string} [attributes.menu=false] Initial mode for the menu region. 662 * @param {int} [attributes.priority=120] The priority for the state link in the media menu. 663 * @param {string} [attributes.type=link] The type of embed. Currently only link is supported. 664 * @param {string} [attributes.url] The embed URL. 665 * @param {object} [attributes.metadata={}] Properties of the embed, which will override attributes.url if set. 666 */ 667 Embed = wp.media.controller.State.extend(/** @lends wp.media.controller.Embed.prototype */{ 668 defaults: { 669 id: 'embed', 670 title: l10n.insertFromUrlTitle, 671 content: 'embed', 672 menu: 'default', 673 toolbar: 'main-embed', 674 priority: 120, 675 type: 'link', 676 url: '', 677 metadata: {} 678 }, 679 680 // The amount of time used when debouncing the scan. 681 sensitivity: 400, 682 683 initialize: function(options) { 684 this.metadata = options.metadata; 685 this.debouncedScan = _.debounce( _.bind( this.scan, this ), this.sensitivity ); 686 this.props = new Backbone.Model( this.metadata || { url: '' }); 687 this.props.on( 'change:url', this.debouncedScan, this ); 688 this.props.on( 'change:url', this.refresh, this ); 689 this.on( 'scan', this.scanImage, this ); 690 }, 691 692 /** 693 * Trigger a scan of the embedded URL's content for metadata required to embed. 694 * 695 * @fires wp.media.controller.Embed#scan 696 */ 697 scan: function() { 698 var scanners, 699 embed = this, 700 attributes = { 701 type: 'link', 702 scanners: [] 703 }; 704 705 /* 706 * Scan is triggered with the list of `attributes` to set on the 707 * state, useful for the 'type' attribute and 'scanners' attribute, 708 * an array of promise objects for asynchronous scan operations. 709 */ 710 if ( this.props.get('url') ) { 711 this.trigger( 'scan', attributes ); 712 } 713 714 if ( attributes.scanners.length ) { 715 scanners = attributes.scanners = $.when.apply( $, attributes.scanners ); 716 scanners.always( function() { 717 if ( embed.get('scanners') === scanners ) { 718 embed.set( 'loading', false ); 719 } 720 }); 721 } else { 722 attributes.scanners = null; 723 } 724 725 attributes.loading = !! attributes.scanners; 726 this.set( attributes ); 727 }, 728 /** 729 * Try scanning the embed as an image to discover its dimensions. 730 * 731 * @param {Object} attributes 732 */ 733 scanImage: function( attributes ) { 734 var frame = this.frame, 735 state = this, 736 url = this.props.get('url'), 737 image = new Image(), 738 deferred = $.Deferred(); 739 740 attributes.scanners.push( deferred.promise() ); 741 742 // Try to load the image and find its width/height. 743 image.onload = function() { 744 deferred.resolve(); 745 746 if ( state !== frame.state() || url !== state.props.get('url') ) { 747 return; 748 } 749 750 state.set({ 751 type: 'image' 752 }); 753 754 state.props.set({ 755 width: image.width, 756 height: image.height 757 }); 758 }; 759 760 image.onerror = deferred.reject; 761 image.src = url; 762 }, 763 764 refresh: function() { 765 this.frame.toolbar.get().refresh(); 766 }, 767 768 reset: function() { 769 this.props.clear().set({ url: '' }); 770 771 if ( this.active ) { 772 this.refresh(); 773 } 774 } 775 }); 776 777 module.exports = Embed; 778 779 780 /***/ }), 781 782 /***/ 1169: 783 /***/ ((module) => { 784 785 var Attachment = wp.media.model.Attachment, 786 Library = wp.media.controller.Library, 787 l10n = wp.media.view.l10n, 788 FeaturedImage; 789 790 /** 791 * wp.media.controller.FeaturedImage 792 * 793 * A state for selecting a featured image for a post. 794 * 795 * @memberOf wp.media.controller 796 * 797 * @class 798 * @augments wp.media.controller.Library 799 * @augments wp.media.controller.State 800 * @augments Backbone.Model 801 * 802 * @param {object} [attributes] The attributes hash passed to the state. 803 * @param {string} [attributes.id=featured-image] Unique identifier. 804 * @param {string} [attributes.title=Set Featured Image] Title for the state. Displays in the media menu and the frame's title region. 805 * @param {wp.media.model.Attachments} [attributes.library] The attachments collection to browse. 806 * If one is not supplied, a collection of all images will be created. 807 * @param {boolean} [attributes.multiple=false] Whether multi-select is enabled. 808 * @param {string} [attributes.content=upload] Initial mode for the content region. 809 * Overridden by persistent user setting if 'contentUserSetting' is true. 810 * @param {string} [attributes.menu=default] Initial mode for the menu region. 811 * @param {string} [attributes.router=browse] Initial mode for the router region. 812 * @param {string} [attributes.toolbar=featured-image] Initial mode for the toolbar region. 813 * @param {int} [attributes.priority=60] The priority for the state link in the media menu. 814 * @param {boolean} [attributes.searchable=true] Whether the library is searchable. 815 * @param {boolean|string} [attributes.filterable=false] Whether the library is filterable, and if so what filters should be shown. 816 * Accepts 'all', 'uploaded', or 'unattached'. 817 * @param {boolean} [attributes.sortable=true] Whether the Attachments should be sortable. Depends on the orderby property being set to menuOrder on the attachments collection. 818 * @param {boolean} [attributes.autoSelect=true] Whether an uploaded attachment should be automatically added to the selection. 819 * @param {boolean} [attributes.describe=false] Whether to offer UI to describe attachments - e.g. captioning images in a gallery. 820 * @param {boolean} [attributes.contentUserSetting=true] Whether the content region's mode should be set and persisted per user. 821 * @param {boolean} [attributes.syncSelection=true] Whether the Attachments selection should be persisted from the last state. 822 */ 823 FeaturedImage = Library.extend(/** @lends wp.media.controller.FeaturedImage.prototype */{ 824 defaults: _.defaults({ 825 id: 'featured-image', 826 title: l10n.setFeaturedImageTitle, 827 multiple: false, 828 filterable: 'uploaded', 829 toolbar: 'featured-image', 830 priority: 60, 831 syncSelection: true 832 }, Library.prototype.defaults ), 833 834 /** 835 * @since 3.5.0 836 */ 837 initialize: function() { 838 var library, comparator; 839 840 // If we haven't been provided a `library`, create a `Selection`. 841 if ( ! this.get('library') ) { 842 this.set( 'library', wp.media.query({ type: 'image' }) ); 843 } 844 845 Library.prototype.initialize.apply( this, arguments ); 846 847 library = this.get('library'); 848 comparator = library.comparator; 849 850 // Overload the library's comparator to push items that are not in 851 // the mirrored query to the front of the aggregate collection. 852 library.comparator = function( a, b ) { 853 var aInQuery = !! this.mirroring.get( a.cid ), 854 bInQuery = !! this.mirroring.get( b.cid ); 855 856 if ( ! aInQuery && bInQuery ) { 857 return -1; 858 } else if ( aInQuery && ! bInQuery ) { 859 return 1; 860 } else { 861 return comparator.apply( this, arguments ); 862 } 863 }; 864 865 // Add all items in the selection to the library, so any featured 866 // images that are not initially loaded still appear. 867 library.observe( this.get('selection') ); 868 }, 869 870 /** 871 * @since 3.5.0 872 */ 873 activate: function() { 874 this.frame.on( 'open', this.updateSelection, this ); 875 876 Library.prototype.activate.apply( this, arguments ); 877 }, 878 879 /** 880 * @since 3.5.0 881 */ 882 deactivate: function() { 883 this.frame.off( 'open', this.updateSelection, this ); 884 885 Library.prototype.deactivate.apply( this, arguments ); 886 }, 887 888 /** 889 * @since 3.5.0 890 */ 891 updateSelection: function() { 892 var selection = this.get('selection'), 893 id = wp.media.view.settings.post.featuredImageId, 894 attachment; 895 896 if ( '' !== id && -1 !== id ) { 897 attachment = Attachment.get( id ); 898 attachment.fetch(); 899 } 900 901 selection.reset( attachment ? [ attachment ] : [] ); 902 } 903 }); 904 905 module.exports = FeaturedImage; 906 907 908 /***/ }), 909 910 /***/ 7127: 911 /***/ ((module) => { 912 913 var Selection = wp.media.model.Selection, 914 Library = wp.media.controller.Library, 915 l10n = wp.media.view.l10n, 916 GalleryAdd; 917 918 /** 919 * wp.media.controller.GalleryAdd 920 * 921 * A state for selecting more images to add to a gallery. 922 * 923 * @since 3.5.0 924 * 925 * @class 926 * @augments wp.media.controller.Library 927 * @augments wp.media.controller.State 928 * @augments Backbone.Model 929 * 930 * @memberof wp.media.controller 931 * 932 * @param {Object} [attributes] The attributes hash passed to the state. 933 * @param {string} [attributes.id=gallery-library] Unique identifier. 934 * @param {string} [attributes.title=Add to Gallery] Title for the state. Displays in the frame's title region. 935 * @param {boolean} [attributes.multiple=add] Whether multi-select is enabled. @todo 'add' doesn't seem do anything special, and gets used as a boolean. 936 * @param {wp.media.model.Attachments} [attributes.library] The attachments collection to browse. 937 * If one is not supplied, a collection of all images will be created. 938 * @param {boolean|string} [attributes.filterable=uploaded] Whether the library is filterable, and if so what filters should be shown. 939 * Accepts 'all', 'uploaded', or 'unattached'. 940 * @param {string} [attributes.menu=gallery] Initial mode for the menu region. 941 * @param {string} [attributes.content=upload] Initial mode for the content region. 942 * Overridden by persistent user setting if 'contentUserSetting' is true. 943 * @param {string} [attributes.router=browse] Initial mode for the router region. 944 * @param {string} [attributes.toolbar=gallery-add] Initial mode for the toolbar region. 945 * @param {boolean} [attributes.searchable=true] Whether the library is searchable. 946 * @param {boolean} [attributes.sortable=true] Whether the Attachments should be sortable. Depends on the orderby property being set to menuOrder on the attachments collection. 947 * @param {boolean} [attributes.autoSelect=true] Whether an uploaded attachment should be automatically added to the selection. 948 * @param {boolean} [attributes.contentUserSetting=true] Whether the content region's mode should be set and persisted per user. 949 * @param {number} [attributes.priority=100] The priority for the state link in the media menu. 950 * @param {boolean} [attributes.syncSelection=false] Whether the Attachments selection should be persisted from the last state. 951 * Defaults to false because for this state, because the library of the Edit Gallery state is the selection. 952 */ 953 GalleryAdd = Library.extend(/** @lends wp.media.controller.GalleryAdd.prototype */{ 954 defaults: _.defaults({ 955 id: 'gallery-library', 956 title: l10n.addToGalleryTitle, 957 multiple: 'add', 958 filterable: 'uploaded', 959 menu: 'gallery', 960 toolbar: 'gallery-add', 961 priority: 100, 962 syncSelection: false 963 }, Library.prototype.defaults ), 964 965 /** 966 * Initializes the library. Creates a library of images if a library isn't supplied. 967 * 968 * @since 3.5.0 969 * 970 * @return {void} 971 */ 972 initialize: function() { 973 if ( ! this.get('library') ) { 974 this.set( 'library', wp.media.query({ type: 'image' }) ); 975 } 976 977 Library.prototype.initialize.apply( this, arguments ); 978 }, 979 980 /** 981 * Activates the library. 982 * 983 * Removes all event listeners if in edit mode. Creates a validator to check an attachment. 984 * Resets library and re-enables event listeners. Activates edit mode. Calls the parent's activate method. 985 * 986 * @since 3.5.0 987 * 988 * @return {void} 989 */ 990 activate: function() { 991 var library = this.get('library'), 992 edit = this.frame.state('gallery-edit').get('library'); 993 994 if ( this.editLibrary && this.editLibrary !== edit ) { 995 library.unobserve( this.editLibrary ); 996 } 997 998 /* 999 * Accept attachments that exist in the original library but 1000 * that do not exist in gallery's library yet. 1001 */ 1002 library.validator = function( attachment ) { 1003 return !! this.mirroring.get( attachment.cid ) && ! edit.get( attachment.cid ) && Selection.prototype.validator.apply( this, arguments ); 1004 }; 1005 1006 /* 1007 * Reset the library to ensure that all attachments are re-added 1008 * to the collection. Do so silently, as calling `observe` will 1009 * trigger the `reset` event. 1010 */ 1011 library.reset( library.mirroring.models, { silent: true }); 1012 library.observe( edit ); 1013 this.editLibrary = edit; 1014 1015 Library.prototype.activate.apply( this, arguments ); 1016 } 1017 }); 1018 1019 module.exports = GalleryAdd; 1020 1021 1022 /***/ }), 1023 1024 /***/ 2038: 1025 /***/ ((module) => { 1026 1027 var Library = wp.media.controller.Library, 1028 l10n = wp.media.view.l10n, 1029 GalleryEdit; 1030 1031 /** 1032 * wp.media.controller.GalleryEdit 1033 * 1034 * A state for editing a gallery's images and settings. 1035 * 1036 * @since 3.5.0 1037 * 1038 * @class 1039 * @augments wp.media.controller.Library 1040 * @augments wp.media.controller.State 1041 * @augments Backbone.Model 1042 * 1043 * @memberOf wp.media.controller 1044 * 1045 * @param {Object} [attributes] The attributes hash passed to the state. 1046 * @param {string} [attributes.id=gallery-edit] Unique identifier. 1047 * @param {string} [attributes.title=Edit Gallery] Title for the state. Displays in the frame's title region. 1048 * @param {wp.media.model.Attachments} [attributes.library] The collection of attachments in the gallery. 1049 * If one is not supplied, an empty media.model.Selection collection is created. 1050 * @param {boolean} [attributes.multiple=false] Whether multi-select is enabled. 1051 * @param {boolean} [attributes.searchable=false] Whether the library is searchable. 1052 * @param {boolean} [attributes.sortable=true] Whether the Attachments should be sortable. Depends on the orderby property being set to menuOrder on the attachments collection. 1053 * @param {boolean} [attributes.date=true] Whether to show the date filter in the browser's toolbar. 1054 * @param {string|false} [attributes.content=browse] Initial mode for the content region. 1055 * @param {string|false} [attributes.toolbar=image-details] Initial mode for the toolbar region. 1056 * @param {boolean} [attributes.describe=true] Whether to offer UI to describe attachments - e.g. captioning images in a gallery. 1057 * @param {boolean} [attributes.displaySettings=true] Whether to show the attachment display settings interface. 1058 * @param {boolean} [attributes.dragInfo=true] Whether to show instructional text about the attachments being sortable. 1059 * @param {number} [attributes.idealColumnWidth=170] The ideal column width in pixels for attachments. 1060 * @param {boolean} [attributes.editing=false] Whether the gallery is being created, or editing an existing instance. 1061 * @param {number} [attributes.priority=60] The priority for the state link in the media menu. 1062 * @param {boolean} [attributes.syncSelection=false] Whether the Attachments selection should be persisted from the last state. 1063 * Defaults to false for this state, because the library passed in *is* the selection. 1064 * @param {view} [attributes.AttachmentView] The single `Attachment` view to be used in the `Attachments`. 1065 * If none supplied, defaults to wp.media.view.Attachment.EditLibrary. 1066 */ 1067 GalleryEdit = Library.extend(/** @lends wp.media.controller.GalleryEdit.prototype */{ 1068 defaults: { 1069 id: 'gallery-edit', 1070 title: l10n.editGalleryTitle, 1071 multiple: false, 1072 searchable: false, 1073 sortable: true, 1074 date: false, 1075 display: false, 1076 content: 'browse', 1077 toolbar: 'gallery-edit', 1078 describe: true, 1079 displaySettings: true, 1080 dragInfo: true, 1081 idealColumnWidth: 170, 1082 editing: false, 1083 priority: 60, 1084 syncSelection: false 1085 }, 1086 1087 /** 1088 * Initializes the library. 1089 * 1090 * Creates a selection if a library isn't supplied and creates an attachment 1091 * view if no attachment view is supplied. 1092 * 1093 * @since 3.5.0 1094 * 1095 * @return {void} 1096 */ 1097 initialize: function() { 1098 // If we haven't been provided a `library`, create a `Selection`. 1099 if ( ! this.get('library') ) { 1100 this.set( 'library', new wp.media.model.Selection() ); 1101 } 1102 1103 // The single `Attachment` view to be used in the `Attachments` view. 1104 if ( ! this.get('AttachmentView') ) { 1105 this.set( 'AttachmentView', wp.media.view.Attachment.EditLibrary ); 1106 } 1107 1108 Library.prototype.initialize.apply( this, arguments ); 1109 }, 1110 1111 /** 1112 * Activates the library. 1113 * 1114 * Limits the library to images, watches for uploaded attachments. Watches for 1115 * the browse event on the frame and binds it to gallerySettings. 1116 * 1117 * @since 3.5.0 1118 * 1119 * @return {void} 1120 */ 1121 activate: function() { 1122 var library = this.get('library'); 1123 1124 // Limit the library to images only. 1125 library.props.set( 'type', 'image' ); 1126 1127 // Watch for uploaded attachments. 1128 this.get('library').observe( wp.Uploader.queue ); 1129 1130 this.frame.on( 'content:render:browse', this.gallerySettings, this ); 1131 1132 Library.prototype.activate.apply( this, arguments ); 1133 }, 1134 1135 /** 1136 * Deactivates the library. 1137 * 1138 * Stops watching for uploaded attachments and browse events. 1139 * 1140 * @since 3.5.0 1141 * 1142 * @return {void} 1143 */ 1144 deactivate: function() { 1145 // Stop watching for uploaded attachments. 1146 this.get('library').unobserve( wp.Uploader.queue ); 1147 1148 this.frame.off( 'content:render:browse', this.gallerySettings, this ); 1149 1150 Library.prototype.deactivate.apply( this, arguments ); 1151 }, 1152 1153 /** 1154 * Adds the gallery settings to the sidebar and adds a reverse button to the 1155 * toolbar. 1156 * 1157 * @since 3.5.0 1158 * 1159 * @param {wp.media.view.Frame} browser The file browser. 1160 * 1161 * @return {void} 1162 */ 1163 gallerySettings: function( browser ) { 1164 if ( ! this.get('displaySettings') ) { 1165 return; 1166 } 1167 1168 var library = this.get('library'); 1169 1170 if ( ! library || ! browser ) { 1171 return; 1172 } 1173 1174 library.gallery = library.gallery || new Backbone.Model(); 1175 1176 browser.sidebar.set({ 1177 gallery: new wp.media.view.Settings.Gallery({ 1178 controller: this, 1179 model: library.gallery, 1180 priority: 40 1181 }) 1182 }); 1183 1184 browser.toolbar.set( 'reverse', { 1185 text: l10n.reverseOrder, 1186 priority: 80, 1187 1188 click: function() { 1189 library.reset( library.toArray().reverse() ); 1190 } 1191 }); 1192 } 1193 }); 1194 1195 module.exports = GalleryEdit; 1196 1197 1198 /***/ }), 1199 1200 /***/ 705: 1201 /***/ ((module) => { 1202 1203 var State = wp.media.controller.State, 1204 Library = wp.media.controller.Library, 1205 l10n = wp.media.view.l10n, 1206 ImageDetails; 1207 1208 /** 1209 * wp.media.controller.ImageDetails 1210 * 1211 * A state for editing the attachment display settings of an image that's been 1212 * inserted into the editor. 1213 * 1214 * @memberOf wp.media.controller 1215 * 1216 * @class 1217 * @augments wp.media.controller.State 1218 * @augments Backbone.Model 1219 * 1220 * @param {object} [attributes] The attributes hash passed to the state. 1221 * @param {string} [attributes.id=image-details] Unique identifier. 1222 * @param {string} [attributes.title=Image Details] Title for the state. Displays in the frame's title region. 1223 * @param {wp.media.model.Attachment} attributes.image The image's model. 1224 * @param {string|false} [attributes.content=image-details] Initial mode for the content region. 1225 * @param {string|false} [attributes.menu=false] Initial mode for the menu region. 1226 * @param {string|false} [attributes.router=false] Initial mode for the router region. 1227 * @param {string|false} [attributes.toolbar=image-details] Initial mode for the toolbar region. 1228 * @param {boolean} [attributes.editing=false] Unused. 1229 * @param {int} [attributes.priority=60] Unused. 1230 * 1231 * @todo This state inherits some defaults from media.controller.Library.prototype.defaults, 1232 * however this may not do anything. 1233 */ 1234 ImageDetails = State.extend(/** @lends wp.media.controller.ImageDetails.prototype */{ 1235 defaults: _.defaults({ 1236 id: 'image-details', 1237 title: l10n.imageDetailsTitle, 1238 content: 'image-details', 1239 menu: false, 1240 router: false, 1241 toolbar: 'image-details', 1242 editing: false, 1243 priority: 60 1244 }, Library.prototype.defaults ), 1245 1246 /** 1247 * @since 3.9.0 1248 * 1249 * @param options Attributes 1250 */ 1251 initialize: function( options ) { 1252 this.image = options.image; 1253 State.prototype.initialize.apply( this, arguments ); 1254 }, 1255 1256 /** 1257 * @since 3.9.0 1258 */ 1259 activate: function() { 1260 this.frame.modal.$el.addClass('image-details'); 1261 } 1262 }); 1263 1264 module.exports = ImageDetails; 1265 1266 1267 /***/ }), 1268 1269 /***/ 472: 1270 /***/ ((module) => { 1271 1272 var l10n = wp.media.view.l10n, 1273 getUserSetting = window.getUserSetting, 1274 setUserSetting = window.setUserSetting, 1275 Library; 1276 1277 /** 1278 * wp.media.controller.Library 1279 * 1280 * A state for choosing an attachment or group of attachments from the media library. 1281 * 1282 * @memberOf wp.media.controller 1283 * 1284 * @class 1285 * @augments wp.media.controller.State 1286 * @augments Backbone.Model 1287 * @mixes media.selectionSync 1288 * 1289 * @param {object} [attributes] The attributes hash passed to the state. 1290 * @param {string} [attributes.id=library] Unique identifier. 1291 * @param {string} [attributes.title=Media library] Title for the state. Displays in the media menu and the frame's title region. 1292 * @param {wp.media.model.Attachments} [attributes.library] The attachments collection to browse. 1293 * If one is not supplied, a collection of all attachments will be created. 1294 * @param {wp.media.model.Selection|object} [attributes.selection] A collection to contain attachment selections within the state. 1295 * If the 'selection' attribute is a plain JS object, 1296 * a Selection will be created using its values as the selection instance's `props` model. 1297 * Otherwise, it will copy the library's `props` model. 1298 * @param {boolean} [attributes.multiple=false] Whether multi-select is enabled. 1299 * @param {string} [attributes.content=upload] Initial mode for the content region. 1300 * Overridden by persistent user setting if 'contentUserSetting' is true. 1301 * @param {string} [attributes.menu=default] Initial mode for the menu region. 1302 * @param {string} [attributes.router=browse] Initial mode for the router region. 1303 * @param {string} [attributes.toolbar=select] Initial mode for the toolbar region. 1304 * @param {boolean} [attributes.searchable=true] Whether the library is searchable. 1305 * @param {boolean|string} [attributes.filterable=false] Whether the library is filterable, and if so what filters should be shown. 1306 * Accepts 'all', 'uploaded', or 'unattached'. 1307 * @param {boolean} [attributes.sortable=true] Whether the Attachments should be sortable. Depends on the orderby property being set to menuOrder on the attachments collection. 1308 * @param {boolean} [attributes.autoSelect=true] Whether an uploaded attachment should be automatically added to the selection. 1309 * @param {boolean} [attributes.describe=false] Whether to offer UI to describe attachments - e.g. captioning images in a gallery. 1310 * @param {boolean} [attributes.contentUserSetting=true] Whether the content region's mode should be set and persisted per user. 1311 * @param {boolean} [attributes.syncSelection=true] Whether the Attachments selection should be persisted from the last state. 1312 */ 1313 Library = wp.media.controller.State.extend(/** @lends wp.media.controller.Library.prototype */{ 1314 defaults: { 1315 id: 'library', 1316 title: l10n.mediaLibraryTitle, 1317 multiple: false, 1318 content: 'upload', 1319 menu: 'default', 1320 router: 'browse', 1321 toolbar: 'select', 1322 searchable: true, 1323 filterable: false, 1324 sortable: true, 1325 autoSelect: true, 1326 describe: false, 1327 contentUserSetting: true, 1328 syncSelection: true 1329 }, 1330 1331 /** 1332 * If a library isn't provided, query all media items. 1333 * If a selection instance isn't provided, create one. 1334 * 1335 * @since 3.5.0 1336 */ 1337 initialize: function() { 1338 var selection = this.get('selection'), 1339 props; 1340 1341 if ( ! this.get('library') ) { 1342 this.set( 'library', wp.media.query() ); 1343 } 1344 1345 if ( ! ( selection instanceof wp.media.model.Selection ) ) { 1346 props = selection; 1347 1348 if ( ! props ) { 1349 props = this.get('library').props.toJSON(); 1350 props = _.omit( props, 'orderby', 'query' ); 1351 } 1352 1353 this.set( 'selection', new wp.media.model.Selection( null, { 1354 multiple: this.get('multiple'), 1355 props: props 1356 }) ); 1357 } 1358 1359 this.resetDisplays(); 1360 }, 1361 1362 /** 1363 * @since 3.5.0 1364 */ 1365 activate: function() { 1366 this.syncSelection(); 1367 1368 wp.Uploader.queue.on( 'add', this.uploading, this ); 1369 1370 this.get('selection').on( 'add remove reset', this.refreshContent, this ); 1371 1372 if ( this.get( 'router' ) && this.get('contentUserSetting') ) { 1373 this.frame.on( 'content:activate', this.saveContentMode, this ); 1374 this.set( 'content', getUserSetting( 'libraryContent', this.get('content') ) ); 1375 } 1376 }, 1377 1378 /** 1379 * @since 3.5.0 1380 */ 1381 deactivate: function() { 1382 this.recordSelection(); 1383 1384 this.frame.off( 'content:activate', this.saveContentMode, this ); 1385 1386 // Unbind all event handlers that use this state as the context 1387 // from the selection. 1388 this.get('selection').off( null, null, this ); 1389 1390 wp.Uploader.queue.off( null, null, this ); 1391 }, 1392 1393 /** 1394 * Reset the library to its initial state. 1395 * 1396 * @since 3.5.0 1397 */ 1398 reset: function() { 1399 this.get('selection').reset(); 1400 this.resetDisplays(); 1401 this.refreshContent(); 1402 }, 1403 1404 /** 1405 * Reset the attachment display settings defaults to the site options. 1406 * 1407 * If site options don't define them, fall back to a persistent user setting. 1408 * 1409 * @since 3.5.0 1410 */ 1411 resetDisplays: function() { 1412 var defaultProps = wp.media.view.settings.defaultProps; 1413 this._displays = []; 1414 this._defaultDisplaySettings = { 1415 align: getUserSetting( 'align', defaultProps.align ) || 'none', 1416 size: getUserSetting( 'imgsize', defaultProps.size ) || 'medium', 1417 link: getUserSetting( 'urlbutton', defaultProps.link ) || 'none' 1418 }; 1419 }, 1420 1421 /** 1422 * Create a model to represent display settings (alignment, etc.) for an attachment. 1423 * 1424 * @since 3.5.0 1425 * 1426 * @param {wp.media.model.Attachment} attachment 1427 * @return {Backbone.Model} 1428 */ 1429 display: function( attachment ) { 1430 var displays = this._displays; 1431 1432 if ( ! displays[ attachment.cid ] ) { 1433 displays[ attachment.cid ] = new Backbone.Model( this.defaultDisplaySettings( attachment ) ); 1434 } 1435 return displays[ attachment.cid ]; 1436 }, 1437 1438 /** 1439 * Given an attachment, create attachment display settings properties. 1440 * 1441 * @since 3.6.0 1442 * 1443 * @param {wp.media.model.Attachment} attachment 1444 * @return {Object} 1445 */ 1446 defaultDisplaySettings: function( attachment ) { 1447 var settings = _.clone( this._defaultDisplaySettings ); 1448 1449 settings.canEmbed = this.canEmbed( attachment ); 1450 if ( settings.canEmbed ) { 1451 settings.link = 'embed'; 1452 } else if ( ! this.isImageAttachment( attachment ) && settings.link === 'none' ) { 1453 settings.link = 'file'; 1454 } 1455 1456 return settings; 1457 }, 1458 1459 /** 1460 * Whether an attachment is image. 1461 * 1462 * @since 4.4.1 1463 * 1464 * @param {wp.media.model.Attachment} attachment 1465 * @return {boolean} 1466 */ 1467 isImageAttachment: function( attachment ) { 1468 // If uploading, we know the filename but not the mime type. 1469 if ( attachment.get('uploading') ) { 1470 return /\.(jpe?g|png|gif|webp|avif|heic|heif)$/i.test( attachment.get('filename') ); 1471 } 1472 1473 return attachment.get('type') === 'image'; 1474 }, 1475 1476 /** 1477 * Whether an attachment can be embedded (audio or video). 1478 * 1479 * @since 3.6.0 1480 * 1481 * @param {wp.media.model.Attachment} attachment 1482 * @return {boolean} 1483 */ 1484 canEmbed: function( attachment ) { 1485 // If uploading, we know the filename but not the mime type. 1486 if ( ! attachment.get('uploading') ) { 1487 var type = attachment.get('type'); 1488 if ( type !== 'audio' && type !== 'video' ) { 1489 return false; 1490 } 1491 } 1492 1493 return _.contains( wp.media.view.settings.embedExts, attachment.get('filename').split('.').pop() ); 1494 }, 1495 1496 1497 /** 1498 * If the state is active, no items are selected, and the current 1499 * content mode is not an option in the state's router (provided 1500 * the state has a router), reset the content mode to the default. 1501 * 1502 * @since 3.5.0 1503 */ 1504 refreshContent: function() { 1505 var selection = this.get('selection'), 1506 frame = this.frame, 1507 router = frame.router.get(), 1508 mode = frame.content.mode(); 1509 1510 if ( this.active && ! selection.length && router && ! router.get( mode ) ) { 1511 this.frame.content.render( this.get('content') ); 1512 } 1513 }, 1514 1515 /** 1516 * Callback handler when an attachment is uploaded. 1517 * 1518 * Switch to the Media Library if uploaded from the 'Upload Files' tab. 1519 * 1520 * Adds any uploading attachments to the selection. 1521 * 1522 * If the state only supports one attachment to be selected and multiple 1523 * attachments are uploaded, the last attachment in the upload queue will 1524 * be selected. 1525 * 1526 * @since 3.5.0 1527 * 1528 * @param {wp.media.model.Attachment} attachment 1529 */ 1530 uploading: function( attachment ) { 1531 var content = this.frame.content; 1532 1533 if ( 'upload' === content.mode() ) { 1534 this.frame.content.mode('browse'); 1535 } 1536 1537 if ( this.get( 'autoSelect' ) ) { 1538 this.get('selection').add( attachment ); 1539 this.frame.trigger( 'library:selection:add' ); 1540 } 1541 }, 1542 1543 /** 1544 * Persist the mode of the content region as a user setting. 1545 * 1546 * @since 3.5.0 1547 */ 1548 saveContentMode: function() { 1549 if ( 'browse' !== this.get('router') ) { 1550 return; 1551 } 1552 1553 var mode = this.frame.content.mode(), 1554 view = this.frame.router.get(); 1555 1556 if ( view && view.get( mode ) ) { 1557 setUserSetting( 'libraryContent', mode ); 1558 } 1559 } 1560 1561 }); 1562 1563 // Make selectionSync available on any Media Library state. 1564 _.extend( Library.prototype, wp.media.selectionSync ); 1565 1566 module.exports = Library; 1567 1568 1569 /***/ }), 1570 1571 /***/ 8065: 1572 /***/ ((module) => { 1573 1574 /** 1575 * wp.media.controller.MediaLibrary 1576 * 1577 * @memberOf wp.media.controller 1578 * 1579 * @class 1580 * @augments wp.media.controller.Library 1581 * @augments wp.media.controller.State 1582 * @augments Backbone.Model 1583 */ 1584 var Library = wp.media.controller.Library, 1585 MediaLibrary; 1586 1587 MediaLibrary = Library.extend(/** @lends wp.media.controller.MediaLibrary.prototype */{ 1588 defaults: _.defaults({ 1589 // Attachments browser defaults. @see media.view.AttachmentsBrowser 1590 filterable: 'uploaded', 1591 1592 displaySettings: false, 1593 priority: 80, 1594 syncSelection: false 1595 }, Library.prototype.defaults ), 1596 1597 /** 1598 * @since 3.9.0 1599 * 1600 * @param options 1601 */ 1602 initialize: function( options ) { 1603 this.media = options.media; 1604 this.type = options.type; 1605 this.set( 'library', wp.media.query({ type: this.type }) ); 1606 1607 Library.prototype.initialize.apply( this, arguments ); 1608 }, 1609 1610 /** 1611 * @since 3.9.0 1612 */ 1613 activate: function() { 1614 // @todo this should use this.frame. 1615 if ( wp.media.frame.lastMime ) { 1616 this.set( 'library', wp.media.query({ type: wp.media.frame.lastMime }) ); 1617 delete wp.media.frame.lastMime; 1618 } 1619 Library.prototype.activate.apply( this, arguments ); 1620 } 1621 }); 1622 1623 module.exports = MediaLibrary; 1624 1625 1626 /***/ }), 1627 1628 /***/ 9875: 1629 /***/ ((module) => { 1630 1631 /** 1632 * wp.media.controller.Region 1633 * 1634 * A region is a persistent application layout area. 1635 * 1636 * A region assumes one mode at any time, and can be switched to another. 1637 * 1638 * When mode changes, events are triggered on the region's parent view. 1639 * The parent view will listen to specific events and fill the region with an 1640 * appropriate view depending on mode. For example, a frame listens for the 1641 * 'browse' mode t be activated on the 'content' view and then fills the region 1642 * with an AttachmentsBrowser view. 1643 * 1644 * @memberOf wp.media.controller 1645 * 1646 * @class 1647 * 1648 * @param {Object} options Options hash for the region. 1649 * @param {string} options.id Unique identifier for the region. 1650 * @param {Backbone.View} options.view A parent view the region exists within. 1651 * @param {string} options.selector jQuery selector for the region within the parent view. 1652 */ 1653 var Region = function( options ) { 1654 _.extend( this, _.pick( options || {}, 'id', 'view', 'selector' ) ); 1655 }; 1656 1657 // Use Backbone's self-propagating `extend` inheritance method. 1658 Region.extend = Backbone.Model.extend; 1659 1660 _.extend( Region.prototype,/** @lends wp.media.controller.Region.prototype */{ 1661 /** 1662 * Activate a mode. 1663 * 1664 * @since 3.5.0 1665 * 1666 * @param {string} mode 1667 * 1668 * @fires Region#activate 1669 * @fires Region#deactivate 1670 * 1671 * @return {wp.media.controller.Region} Returns itself to allow chaining. 1672 */ 1673 mode: function( mode ) { 1674 if ( ! mode ) { 1675 return this._mode; 1676 } 1677 // Bail if we're trying to change to the current mode. 1678 if ( mode === this._mode ) { 1679 return this; 1680 } 1681 1682 /** 1683 * Region mode deactivation event. 1684 * 1685 * @event wp.media.controller.Region#deactivate 1686 */ 1687 this.trigger('deactivate'); 1688 1689 this._mode = mode; 1690 this.render( mode ); 1691 1692 /** 1693 * Region mode activation event. 1694 * 1695 * @event wp.media.controller.Region#activate 1696 */ 1697 this.trigger('activate'); 1698 return this; 1699 }, 1700 /** 1701 * Render a mode. 1702 * 1703 * @since 3.5.0 1704 * 1705 * @param {string} mode 1706 * 1707 * @fires Region#create 1708 * @fires Region#render 1709 * 1710 * @return {wp.media.controller.Region} Returns itself to allow chaining. 1711 */ 1712 render: function( mode ) { 1713 // If the mode isn't active, activate it. 1714 if ( mode && mode !== this._mode ) { 1715 return this.mode( mode ); 1716 } 1717 1718 var set = { view: null }, 1719 view; 1720 1721 /** 1722 * Create region view event. 1723 * 1724 * Region view creation takes place in an event callback on the frame. 1725 * 1726 * @event wp.media.controller.Region#create 1727 * @type {object} 1728 * @property {object} view 1729 */ 1730 this.trigger( 'create', set ); 1731 view = set.view; 1732 1733 /** 1734 * Render region view event. 1735 * 1736 * Region view creation takes place in an event callback on the frame. 1737 * 1738 * @event wp.media.controller.Region#render 1739 * @type {object} 1740 */ 1741 this.trigger( 'render', view ); 1742 if ( view ) { 1743 this.set( view ); 1744 } 1745 return this; 1746 }, 1747 1748 /** 1749 * Get the region's view. 1750 * 1751 * @since 3.5.0 1752 * 1753 * @return {wp.media.View} 1754 */ 1755 get: function() { 1756 return this.view.views.first( this.selector ); 1757 }, 1758 1759 /** 1760 * Set the region's view as a subview of the frame. 1761 * 1762 * @since 3.5.0 1763 * 1764 * @param {Array|Object} views 1765 * @param {Object} [options={}] 1766 * @return {wp.Backbone.Subviews} Subviews is returned to allow chaining. 1767 */ 1768 set: function( views, options ) { 1769 if ( options ) { 1770 options.add = false; 1771 } 1772 return this.view.views.set( this.selector, views, options ); 1773 }, 1774 1775 /** 1776 * Trigger regional view events on the frame. 1777 * 1778 * @since 3.5.0 1779 * 1780 * @param {string} event 1781 * @return {undefined|wp.media.controller.Region} Returns itself to allow chaining. 1782 */ 1783 trigger: function( event ) { 1784 var base, args; 1785 1786 if ( ! this._mode ) { 1787 return; 1788 } 1789 1790 args = _.toArray( arguments ); 1791 base = this.id + ':' + event; 1792 1793 // Trigger `{this.id}:{event}:{this._mode}` event on the frame. 1794 args[0] = base + ':' + this._mode; 1795 this.view.trigger.apply( this.view, args ); 1796 1797 // Trigger `{this.id}:{event}` event on the frame. 1798 args[0] = base; 1799 this.view.trigger.apply( this.view, args ); 1800 return this; 1801 } 1802 }); 1803 1804 module.exports = Region; 1805 1806 1807 /***/ }), 1808 1809 /***/ 2275: 1810 /***/ ((module) => { 1811 1812 var Library = wp.media.controller.Library, 1813 l10n = wp.media.view.l10n, 1814 ReplaceImage; 1815 1816 /** 1817 * wp.media.controller.ReplaceImage 1818 * 1819 * A state for replacing an image. 1820 * 1821 * @memberOf wp.media.controller 1822 * 1823 * @class 1824 * @augments wp.media.controller.Library 1825 * @augments wp.media.controller.State 1826 * @augments Backbone.Model 1827 * 1828 * @param {object} [attributes] The attributes hash passed to the state. 1829 * @param {string} [attributes.id=replace-image] Unique identifier. 1830 * @param {string} [attributes.title=Replace Image] Title for the state. Displays in the media menu and the frame's title region. 1831 * @param {wp.media.model.Attachments} [attributes.library] The attachments collection to browse. 1832 * If one is not supplied, a collection of all images will be created. 1833 * @param {boolean} [attributes.multiple=false] Whether multi-select is enabled. 1834 * @param {string} [attributes.content=upload] Initial mode for the content region. 1835 * Overridden by persistent user setting if 'contentUserSetting' is true. 1836 * @param {string} [attributes.menu=default] Initial mode for the menu region. 1837 * @param {string} [attributes.router=browse] Initial mode for the router region. 1838 * @param {string} [attributes.toolbar=replace] Initial mode for the toolbar region. 1839 * @param {int} [attributes.priority=60] The priority for the state link in the media menu. 1840 * @param {boolean} [attributes.searchable=true] Whether the library is searchable. 1841 * @param {boolean|string} [attributes.filterable=uploaded] Whether the library is filterable, and if so what filters should be shown. 1842 * Accepts 'all', 'uploaded', or 'unattached'. 1843 * @param {boolean} [attributes.sortable=true] Whether the Attachments should be sortable. Depends on the orderby property being set to menuOrder on the attachments collection. 1844 * @param {boolean} [attributes.autoSelect=true] Whether an uploaded attachment should be automatically added to the selection. 1845 * @param {boolean} [attributes.describe=false] Whether to offer UI to describe attachments - e.g. captioning images in a gallery. 1846 * @param {boolean} [attributes.contentUserSetting=true] Whether the content region's mode should be set and persisted per user. 1847 * @param {boolean} [attributes.syncSelection=true] Whether the Attachments selection should be persisted from the last state. 1848 */ 1849 ReplaceImage = Library.extend(/** @lends wp.media.controller.ReplaceImage.prototype */{ 1850 defaults: _.defaults({ 1851 id: 'replace-image', 1852 title: l10n.replaceImageTitle, 1853 multiple: false, 1854 filterable: 'uploaded', 1855 toolbar: 'replace', 1856 menu: false, 1857 priority: 60, 1858 syncSelection: true 1859 }, Library.prototype.defaults ), 1860 1861 /** 1862 * @since 3.9.0 1863 * 1864 * @param options 1865 */ 1866 initialize: function( options ) { 1867 var library, comparator; 1868 1869 this.image = options.image; 1870 // If we haven't been provided a `library`, create a `Selection`. 1871 if ( ! this.get('library') ) { 1872 this.set( 'library', wp.media.query({ type: 'image' }) ); 1873 } 1874 1875 Library.prototype.initialize.apply( this, arguments ); 1876 1877 library = this.get('library'); 1878 comparator = library.comparator; 1879 1880 // Overload the library's comparator to push items that are not in 1881 // the mirrored query to the front of the aggregate collection. 1882 library.comparator = function( a, b ) { 1883 var aInQuery = !! this.mirroring.get( a.cid ), 1884 bInQuery = !! this.mirroring.get( b.cid ); 1885 1886 if ( ! aInQuery && bInQuery ) { 1887 return -1; 1888 } else if ( aInQuery && ! bInQuery ) { 1889 return 1; 1890 } else { 1891 return comparator.apply( this, arguments ); 1892 } 1893 }; 1894 1895 // Add all items in the selection to the library, so any featured 1896 // images that are not initially loaded still appear. 1897 library.observe( this.get('selection') ); 1898 }, 1899 1900 /** 1901 * @since 3.9.0 1902 */ 1903 activate: function() { 1904 this.frame.on( 'content:render:browse', this.updateSelection, this ); 1905 1906 Library.prototype.activate.apply( this, arguments ); 1907 }, 1908 1909 /** 1910 * @since 5.9.0 1911 */ 1912 deactivate: function() { 1913 this.frame.off( 'content:render:browse', this.updateSelection, this ); 1914 1915 Library.prototype.deactivate.apply( this, arguments ); 1916 }, 1917 1918 /** 1919 * @since 3.9.0 1920 */ 1921 updateSelection: function() { 1922 var selection = this.get('selection'), 1923 attachment = this.image.attachment; 1924 1925 selection.reset( attachment ? [ attachment ] : [] ); 1926 } 1927 }); 1928 1929 module.exports = ReplaceImage; 1930 1931 1932 /***/ }), 1933 1934 /***/ 6172: 1935 /***/ ((module) => { 1936 1937 var Controller = wp.media.controller, 1938 SiteIconCropper; 1939 1940 /** 1941 * wp.media.controller.SiteIconCropper 1942 * 1943 * A state for cropping a Site Icon. 1944 * 1945 * @memberOf wp.media.controller 1946 * 1947 * @class 1948 * @augments wp.media.controller.Cropper 1949 * @augments wp.media.controller.State 1950 * @augments Backbone.Model 1951 */ 1952 SiteIconCropper = Controller.Cropper.extend(/** @lends wp.media.controller.SiteIconCropper.prototype */{ 1953 activate: function() { 1954 this.frame.on( 'content:create:crop', this.createCropContent, this ); 1955 this.frame.on( 'close', this.removeCropper, this ); 1956 this.set('selection', new Backbone.Collection(this.frame._selection.single)); 1957 }, 1958 1959 createCropContent: function() { 1960 this.cropperView = new wp.media.view.SiteIconCropper({ 1961 controller: this, 1962 attachment: this.get('selection').first() 1963 }); 1964 this.cropperView.on('image-loaded', this.createCropToolbar, this); 1965 this.frame.content.set(this.cropperView); 1966 1967 }, 1968 1969 doCrop: function( attachment ) { 1970 var cropDetails = attachment.get( 'cropDetails' ), 1971 control = this.get( 'control' ); 1972 1973 cropDetails.dst_width = control.params.width; 1974 cropDetails.dst_height = control.params.height; 1975 1976 return wp.ajax.post( 'crop-image', { 1977 nonce: attachment.get( 'nonces' ).edit, 1978 id: attachment.get( 'id' ), 1979 context: 'site-icon', 1980 cropDetails: cropDetails 1981 } ); 1982 } 1983 }); 1984 1985 module.exports = SiteIconCropper; 1986 1987 1988 /***/ }), 1989 1990 /***/ 6150: 1991 /***/ ((module) => { 1992 1993 /** 1994 * wp.media.controller.StateMachine 1995 * 1996 * A state machine keeps track of state. It is in one state at a time, 1997 * and can change from one state to another. 1998 * 1999 * States are stored as models in a Backbone collection. 2000 * 2001 * @memberOf wp.media.controller 2002 * 2003 * @since 3.5.0 2004 * 2005 * @class 2006 * @augments Backbone.Model 2007 * @mixin 2008 * @mixes Backbone.Events 2009 */ 2010 var StateMachine = function() { 2011 return { 2012 // Use Backbone's self-propagating `extend` inheritance method. 2013 extend: Backbone.Model.extend 2014 }; 2015 }; 2016 2017 _.extend( StateMachine.prototype, Backbone.Events,/** @lends wp.media.controller.StateMachine.prototype */{ 2018 /** 2019 * Fetch a state. 2020 * 2021 * If no `id` is provided, returns the active state. 2022 * 2023 * Implicitly creates states. 2024 * 2025 * Ensure that the `states` collection exists so the `StateMachine` 2026 * can be used as a mixin. 2027 * 2028 * @since 3.5.0 2029 * 2030 * @param {string} id 2031 * @return {wp.media.controller.State} Returns a State model from 2032 * the StateMachine collection. 2033 */ 2034 state: function( id ) { 2035 this.states = this.states || new Backbone.Collection(); 2036 2037 // Default to the active state. 2038 id = id || this._state; 2039 2040 if ( id && ! this.states.get( id ) ) { 2041 this.states.add({ id: id }); 2042 } 2043 return this.states.get( id ); 2044 }, 2045 2046 /** 2047 * Sets the active state. 2048 * 2049 * Bail if we're trying to select the current state, if we haven't 2050 * created the `states` collection, or are trying to select a state 2051 * that does not exist. 2052 * 2053 * @since 3.5.0 2054 * 2055 * @param {string} id 2056 * 2057 * @fires wp.media.controller.State#deactivate 2058 * @fires wp.media.controller.State#activate 2059 * 2060 * @return {wp.media.controller.StateMachine} Returns itself to allow chaining. 2061 */ 2062 setState: function( id ) { 2063 var previous = this.state(); 2064 2065 if ( ( previous && id === previous.id ) || ! this.states || ! this.states.get( id ) ) { 2066 return this; 2067 } 2068 2069 if ( previous ) { 2070 previous.trigger('deactivate'); 2071 this._lastState = previous.id; 2072 } 2073 2074 this._state = id; 2075 this.state().trigger('activate'); 2076 2077 return this; 2078 }, 2079 2080 /** 2081 * Returns the previous active state. 2082 * 2083 * Call the `state()` method with no parameters to retrieve the current 2084 * active state. 2085 * 2086 * @since 3.5.0 2087 * 2088 * @return {wp.media.controller.State} Returns a State model from 2089 * the StateMachine collection. 2090 */ 2091 lastState: function() { 2092 if ( this._lastState ) { 2093 return this.state( this._lastState ); 2094 } 2095 } 2096 }); 2097 2098 // Map all event binding and triggering on a StateMachine to its `states` collection. 2099 _.each([ 'on', 'off', 'trigger' ], function( method ) { 2100 /** 2101 * @function on 2102 * @memberOf wp.media.controller.StateMachine 2103 * @instance 2104 * @return {wp.media.controller.StateMachine} Returns itself to allow chaining. 2105 */ 2106 /** 2107 * @function off 2108 * @memberOf wp.media.controller.StateMachine 2109 * @instance 2110 * @return {wp.media.controller.StateMachine} Returns itself to allow chaining. 2111 */ 2112 /** 2113 * @function trigger 2114 * @memberOf wp.media.controller.StateMachine 2115 * @instance 2116 * @return {wp.media.controller.StateMachine} Returns itself to allow chaining. 2117 */ 2118 StateMachine.prototype[ method ] = function() { 2119 // Ensure that the `states` collection exists so the `StateMachine` 2120 // can be used as a mixin. 2121 this.states = this.states || new Backbone.Collection(); 2122 // Forward the method to the `states` collection. 2123 this.states[ method ].apply( this.states, arguments ); 2124 return this; 2125 }; 2126 }); 2127 2128 module.exports = StateMachine; 2129 2130 2131 /***/ }), 2132 2133 /***/ 5694: 2134 /***/ ((module) => { 2135 2136 /** 2137 * wp.media.controller.State 2138 * 2139 * A state is a step in a workflow that when set will trigger the controllers 2140 * for the regions to be updated as specified in the frame. 2141 * 2142 * A state has an event-driven lifecycle: 2143 * 2144 * 'ready' triggers when a state is added to a state machine's collection. 2145 * 'activate' triggers when a state is activated by a state machine. 2146 * 'deactivate' triggers when a state is deactivated by a state machine. 2147 * 'reset' is not triggered automatically. It should be invoked by the 2148 * proper controller to reset the state to its default. 2149 * 2150 * @memberOf wp.media.controller 2151 * 2152 * @class 2153 * @augments Backbone.Model 2154 */ 2155 var State = Backbone.Model.extend(/** @lends wp.media.controller.State.prototype */{ 2156 /** 2157 * Constructor. 2158 * 2159 * @since 3.5.0 2160 */ 2161 constructor: function() { 2162 this.on( 'activate', this._preActivate, this ); 2163 this.on( 'activate', this.activate, this ); 2164 this.on( 'activate', this._postActivate, this ); 2165 this.on( 'deactivate', this._deactivate, this ); 2166 this.on( 'deactivate', this.deactivate, this ); 2167 this.on( 'reset', this.reset, this ); 2168 this.on( 'ready', this._ready, this ); 2169 this.on( 'ready', this.ready, this ); 2170 /** 2171 * Call parent constructor with passed arguments 2172 */ 2173 Backbone.Model.apply( this, arguments ); 2174 this.on( 'change:menu', this._updateMenu, this ); 2175 }, 2176 /** 2177 * Ready event callback. 2178 * 2179 * @abstract 2180 * @since 3.5.0 2181 */ 2182 ready: function() {}, 2183 2184 /** 2185 * Activate event callback. 2186 * 2187 * @abstract 2188 * @since 3.5.0 2189 */ 2190 activate: function() {}, 2191 2192 /** 2193 * Deactivate event callback. 2194 * 2195 * @abstract 2196 * @since 3.5.0 2197 */ 2198 deactivate: function() {}, 2199 2200 /** 2201 * Reset event callback. 2202 * 2203 * @abstract 2204 * @since 3.5.0 2205 */ 2206 reset: function() {}, 2207 2208 /** 2209 * @since 3.5.0 2210 * @access private 2211 */ 2212 _ready: function() { 2213 this._updateMenu(); 2214 }, 2215 2216 /** 2217 * @since 3.5.0 2218 * @access private 2219 */ 2220 _preActivate: function() { 2221 this.active = true; 2222 }, 2223 2224 /** 2225 * @since 3.5.0 2226 * @access private 2227 */ 2228 _postActivate: function() { 2229 this.on( 'change:menu', this._menu, this ); 2230 this.on( 'change:titleMode', this._title, this ); 2231 this.on( 'change:content', this._content, this ); 2232 this.on( 'change:toolbar', this._toolbar, this ); 2233 2234 this.frame.on( 'title:render:default', this._renderTitle, this ); 2235 2236 this._title(); 2237 this._menu(); 2238 this._toolbar(); 2239 this._content(); 2240 this._router(); 2241 }, 2242 2243 /** 2244 * @since 3.5.0 2245 * @access private 2246 */ 2247 _deactivate: function() { 2248 this.active = false; 2249 2250 this.frame.off( 'title:render:default', this._renderTitle, this ); 2251 2252 this.off( 'change:menu', this._menu, this ); 2253 this.off( 'change:titleMode', this._title, this ); 2254 this.off( 'change:content', this._content, this ); 2255 this.off( 'change:toolbar', this._toolbar, this ); 2256 }, 2257 2258 /** 2259 * @since 3.5.0 2260 * @access private 2261 */ 2262 _title: function() { 2263 this.frame.title.render( this.get('titleMode') || 'default' ); 2264 }, 2265 2266 /** 2267 * @since 3.5.0 2268 * @access private 2269 */ 2270 _renderTitle: function( view ) { 2271 view.$el.text( this.get('title') || '' ); 2272 }, 2273 2274 /** 2275 * @since 3.5.0 2276 * @access private 2277 */ 2278 _router: function() { 2279 var router = this.frame.router, 2280 mode = this.get('router'), 2281 view; 2282 2283 this.frame.$el.toggleClass( 'hide-router', ! mode ); 2284 if ( ! mode ) { 2285 return; 2286 } 2287 2288 this.frame.router.render( mode ); 2289 2290 view = router.get(); 2291 if ( view && view.select ) { 2292 view.select( this.frame.content.mode() ); 2293 } 2294 }, 2295 2296 /** 2297 * @since 3.5.0 2298 * @access private 2299 */ 2300 _menu: function() { 2301 var menu = this.frame.menu, 2302 mode = this.get('menu'), 2303 actionMenuItems, 2304 actionMenuLength, 2305 view; 2306 2307 if ( this.frame.menu ) { 2308 actionMenuItems = this.frame.menu.get('views'), 2309 actionMenuLength = actionMenuItems ? actionMenuItems.views.get().length : 0, 2310 // Show action menu only if it is active and has more than one default element. 2311 this.frame.$el.toggleClass( 'hide-menu', ! mode || actionMenuLength < 2 ); 2312 } 2313 if ( ! mode ) { 2314 return; 2315 } 2316 2317 menu.mode( mode ); 2318 2319 view = menu.get(); 2320 if ( view && view.select ) { 2321 view.select( this.id ); 2322 } 2323 }, 2324 2325 /** 2326 * @since 3.5.0 2327 * @access private 2328 */ 2329 _updateMenu: function() { 2330 var previous = this.previous('menu'), 2331 menu = this.get('menu'); 2332 2333 if ( previous ) { 2334 this.frame.off( 'menu:render:' + previous, this._renderMenu, this ); 2335 } 2336 2337 if ( menu ) { 2338 this.frame.on( 'menu:render:' + menu, this._renderMenu, this ); 2339 } 2340 }, 2341 2342 /** 2343 * Create a view in the media menu for the state. 2344 * 2345 * @since 3.5.0 2346 * @access private 2347 * 2348 * @param {media.view.Menu} view The menu view. 2349 */ 2350 _renderMenu: function( view ) { 2351 var menuItem = this.get('menuItem'), 2352 title = this.get('title'), 2353 priority = this.get('priority'); 2354 2355 if ( ! menuItem && title ) { 2356 menuItem = { text: title }; 2357 2358 if ( priority ) { 2359 menuItem.priority = priority; 2360 } 2361 } 2362 2363 if ( ! menuItem ) { 2364 return; 2365 } 2366 2367 view.set( this.id, menuItem ); 2368 } 2369 }); 2370 2371 _.each(['toolbar','content'], function( region ) { 2372 /** 2373 * @access private 2374 */ 2375 State.prototype[ '_' + region ] = function() { 2376 var mode = this.get( region ); 2377 if ( mode ) { 2378 this.frame[ region ].render( mode ); 2379 } 2380 }; 2381 }); 2382 2383 module.exports = State; 2384 2385 2386 /***/ }), 2387 2388 /***/ 4181: 2389 /***/ ((module) => { 2390 2391 /** 2392 * wp.media.selectionSync 2393 * 2394 * Sync an attachments selection in a state with another state. 2395 * 2396 * Allows for selecting multiple images in the Add Media workflow, and then 2397 * switching to the Insert Gallery workflow while preserving the attachments selection. 2398 * 2399 * @memberOf wp.media 2400 * 2401 * @mixin 2402 */ 2403 var selectionSync = { 2404 /** 2405 * @since 3.5.0 2406 */ 2407 syncSelection: function() { 2408 var selection = this.get('selection'), 2409 manager = this.frame._selection; 2410 2411 if ( ! this.get('syncSelection') || ! manager || ! selection ) { 2412 return; 2413 } 2414 2415 /* 2416 * If the selection supports multiple items, validate the stored 2417 * attachments based on the new selection's conditions. Record 2418 * the attachments that are not included; we'll maintain a 2419 * reference to those. Other attachments are considered in flux. 2420 */ 2421 if ( selection.multiple ) { 2422 selection.reset( [], { silent: true }); 2423 selection.validateAll( manager.attachments ); 2424 manager.difference = _.difference( manager.attachments.models, selection.models ); 2425 } 2426 2427 // Sync the selection's single item with the master. 2428 selection.single( manager.single ); 2429 }, 2430 2431 /** 2432 * Record the currently active attachments, which is a combination 2433 * of the selection's attachments and the set of selected 2434 * attachments that this specific selection considered invalid. 2435 * Reset the difference and record the single attachment. 2436 * 2437 * @since 3.5.0 2438 */ 2439 recordSelection: function() { 2440 var selection = this.get('selection'), 2441 manager = this.frame._selection; 2442 2443 if ( ! this.get('syncSelection') || ! manager || ! selection ) { 2444 return; 2445 } 2446 2447 if ( selection.multiple ) { 2448 manager.attachments.reset( selection.toArray().concat( manager.difference ) ); 2449 manager.difference = []; 2450 } else { 2451 manager.attachments.add( selection.toArray() ); 2452 } 2453 2454 manager.single = selection._single; 2455 } 2456 }; 2457 2458 module.exports = selectionSync; 2459 2460 2461 /***/ }), 2462 2463 /***/ 2982: 2464 /***/ ((module) => { 2465 2466 var View = wp.media.View, 2467 AttachmentCompat; 2468 2469 /** 2470 * wp.media.view.AttachmentCompat 2471 * 2472 * A view to display fields added via the `attachment_fields_to_edit` filter. 2473 * 2474 * @memberOf wp.media.view 2475 * 2476 * @class 2477 * @augments wp.media.View 2478 * @augments wp.Backbone.View 2479 * @augments Backbone.View 2480 */ 2481 AttachmentCompat = View.extend(/** @lends wp.media.view.AttachmentCompat.prototype */{ 2482 tagName: 'form', 2483 className: 'compat-item', 2484 2485 events: { 2486 'submit': 'preventDefault', 2487 'change input': 'save', 2488 'change select': 'save', 2489 'change textarea': 'save' 2490 }, 2491 2492 initialize: function() { 2493 // Render the view when a new item is added. 2494 this.listenTo( this.model, 'add', this.render ); 2495 }, 2496 2497 /** 2498 * @return {wp.media.view.AttachmentCompat} Returns itself to allow chaining. 2499 */ 2500 dispose: function() { 2501 if ( this.$(':focus').length ) { 2502 this.save(); 2503 } 2504 /** 2505 * call 'dispose' directly on the parent class 2506 */ 2507 return View.prototype.dispose.apply( this, arguments ); 2508 }, 2509 /** 2510 * @return {wp.media.view.AttachmentCompat} Returns itself to allow chaining. 2511 */ 2512 render: function() { 2513 var compat = this.model.get('compat'); 2514 if ( ! compat || ! compat.item ) { 2515 return; 2516 } 2517 2518 this.views.detach(); 2519 this.$el.html( compat.item ); 2520 this.views.render(); 2521 return this; 2522 }, 2523 /** 2524 * @param {Object} event 2525 */ 2526 preventDefault: function( event ) { 2527 event.preventDefault(); 2528 }, 2529 /** 2530 * @param {Object} event 2531 */ 2532 save: function( event ) { 2533 var data = {}; 2534 2535 if ( event ) { 2536 event.preventDefault(); 2537 } 2538 2539 _.each( this.$el.serializeArray(), function( pair ) { 2540 data[ pair.name ] = pair.value; 2541 }); 2542 2543 this.controller.trigger( 'attachment:compat:waiting', ['waiting'] ); 2544 this.model.saveCompat( data ).always( _.bind( this.postSave, this ) ); 2545 }, 2546 2547 postSave: function() { 2548 this.controller.trigger( 'attachment:compat:ready', ['ready'] ); 2549 } 2550 }); 2551 2552 module.exports = AttachmentCompat; 2553 2554 2555 /***/ }), 2556 2557 /***/ 7709: 2558 /***/ ((module) => { 2559 2560 var $ = jQuery, 2561 AttachmentFilters; 2562 2563 /** 2564 * wp.media.view.AttachmentFilters 2565 * 2566 * @memberOf wp.media.view 2567 * 2568 * @class 2569 * @augments wp.media.View 2570 * @augments wp.Backbone.View 2571 * @augments Backbone.View 2572 */ 2573 AttachmentFilters = wp.media.View.extend(/** @lends wp.media.view.AttachmentFilters.prototype */{ 2574 tagName: 'select', 2575 className: 'attachment-filters', 2576 id: 'media-attachment-filters', 2577 2578 events: { 2579 change: 'change' 2580 }, 2581 2582 keys: [], 2583 2584 initialize: function() { 2585 this.createFilters(); 2586 _.extend( this.filters, this.options.filters ); 2587 2588 // Build `<option>` elements. 2589 this.$el.html( _.chain( this.filters ).map( function( filter, value ) { 2590 return { 2591 el: $( '<option></option>' ).val( value ).html( filter.text )[0], 2592 priority: filter.priority || 50 2593 }; 2594 }, this ).sortBy('priority').pluck('el').value() ); 2595 2596 this.listenTo( this.model, 'change', this.select ); 2597 this.select(); 2598 }, 2599 2600 /** 2601 * @abstract 2602 */ 2603 createFilters: function() { 2604 this.filters = {}; 2605 }, 2606 2607 /** 2608 * When the selected filter changes, update the Attachment Query properties to match. 2609 */ 2610 change: function() { 2611 var filter = this.filters[ this.el.value ]; 2612 if ( filter ) { 2613 this.model.set( filter.props ); 2614 } 2615 }, 2616 2617 select: function() { 2618 var model = this.model, 2619 value = 'all', 2620 props = model.toJSON(); 2621 2622 _.find( this.filters, function( filter, id ) { 2623 var equal = _.all( filter.props, function( prop, key ) { 2624 return prop === ( _.isUndefined( props[ key ] ) ? null : props[ key ] ); 2625 }); 2626 2627 if ( equal ) { 2628 return value = id; 2629 } 2630 }); 2631 2632 this.$el.val( value ); 2633 } 2634 }); 2635 2636 module.exports = AttachmentFilters; 2637 2638 2639 /***/ }), 2640 2641 /***/ 7349: 2642 /***/ ((module) => { 2643 2644 var l10n = wp.media.view.l10n, 2645 All; 2646 2647 /** 2648 * wp.media.view.AttachmentFilters.All 2649 * 2650 * @memberOf wp.media.view.AttachmentFilters 2651 * 2652 * @class 2653 * @augments wp.media.view.AttachmentFilters 2654 * @augments wp.media.View 2655 * @augments wp.Backbone.View 2656 * @augments Backbone.View 2657 */ 2658 All = wp.media.view.AttachmentFilters.extend(/** @lends wp.media.view.AttachmentFilters.All.prototype */{ 2659 createFilters: function() { 2660 var filters = {}, 2661 uid = window.userSettings ? parseInt( window.userSettings.uid, 10 ) : 0; 2662 2663 _.each( wp.media.view.settings.mimeTypes || {}, function( text, key ) { 2664 filters[ key ] = { 2665 text: text, 2666 props: { 2667 status: null, 2668 type: key, 2669 uploadedTo: null, 2670 orderby: 'date', 2671 order: 'DESC', 2672 author: null 2673 } 2674 }; 2675 }); 2676 2677 filters.all = { 2678 text: l10n.allMediaItems, 2679 props: { 2680 status: null, 2681 type: null, 2682 uploadedTo: null, 2683 orderby: 'date', 2684 order: 'DESC', 2685 author: null 2686 }, 2687 priority: 10 2688 }; 2689 2690 if ( wp.media.view.settings.post.id ) { 2691 filters.uploaded = { 2692 text: l10n.uploadedToThisPost, 2693 props: { 2694 status: null, 2695 type: null, 2696 uploadedTo: wp.media.view.settings.post.id, 2697 orderby: 'menuOrder', 2698 order: 'ASC', 2699 author: null 2700 }, 2701 priority: 20 2702 }; 2703 } 2704 2705 filters.unattached = { 2706 text: l10n.unattached, 2707 props: { 2708 status: null, 2709 uploadedTo: 0, 2710 type: null, 2711 orderby: 'menuOrder', 2712 order: 'ASC', 2713 author: null 2714 }, 2715 priority: 50 2716 }; 2717 2718 if ( uid ) { 2719 filters.mine = { 2720 text: l10n.mine, 2721 props: { 2722 status: null, 2723 type: null, 2724 uploadedTo: null, 2725 orderby: 'date', 2726 order: 'DESC', 2727 author: uid 2728 }, 2729 priority: 50 2730 }; 2731 } 2732 2733 if ( wp.media.view.settings.mediaTrash && 2734 this.controller.isModeActive( 'grid' ) ) { 2735 2736 filters.trash = { 2737 text: l10n.trash, 2738 props: { 2739 uploadedTo: null, 2740 status: 'trash', 2741 type: null, 2742 orderby: 'date', 2743 order: 'DESC', 2744 author: null 2745 }, 2746 priority: 50 2747 }; 2748 } 2749 2750 this.filters = filters; 2751 } 2752 }); 2753 2754 module.exports = All; 2755 2756 2757 /***/ }), 2758 2759 /***/ 6472: 2760 /***/ ((module) => { 2761 2762 var l10n = wp.media.view.l10n, 2763 DateFilter; 2764 2765 /** 2766 * A filter dropdown for month/dates. 2767 * 2768 * @memberOf wp.media.view.AttachmentFilters 2769 * 2770 * @class 2771 * @augments wp.media.view.AttachmentFilters 2772 * @augments wp.media.View 2773 * @augments wp.Backbone.View 2774 * @augments Backbone.View 2775 */ 2776 DateFilter = wp.media.view.AttachmentFilters.extend(/** @lends wp.media.view.AttachmentFilters.Date.prototype */{ 2777 id: 'media-attachment-date-filters', 2778 2779 createFilters: function() { 2780 var filters = {}; 2781 _.each( wp.media.view.settings.months || {}, function( value, index ) { 2782 filters[ index ] = { 2783 text: value.text, 2784 props: { 2785 year: value.year, 2786 monthnum: value.month 2787 } 2788 }; 2789 }); 2790 filters.all = { 2791 text: l10n.allDates, 2792 props: { 2793 monthnum: false, 2794 year: false 2795 }, 2796 priority: 10 2797 }; 2798 this.filters = filters; 2799 } 2800 }); 2801 2802 module.exports = DateFilter; 2803 2804 2805 /***/ }), 2806 2807 /***/ 1368: 2808 /***/ ((module) => { 2809 2810 var l10n = wp.media.view.l10n, 2811 Uploaded; 2812 2813 /** 2814 * wp.media.view.AttachmentFilters.Uploaded 2815 * 2816 * @memberOf wp.media.view.AttachmentFilters 2817 * 2818 * @class 2819 * @augments wp.media.view.AttachmentFilters 2820 * @augments wp.media.View 2821 * @augments wp.Backbone.View 2822 * @augments Backbone.View 2823 */ 2824 Uploaded = wp.media.view.AttachmentFilters.extend(/** @lends wp.media.view.AttachmentFilters.Uploaded.prototype */{ 2825 createFilters: function() { 2826 var type = this.model.get('type'), 2827 types = wp.media.view.settings.mimeTypes, 2828 uid = window.userSettings ? parseInt( window.userSettings.uid, 10 ) : 0, 2829 text; 2830 2831 if ( types && type ) { 2832 text = types[ type ]; 2833 } 2834 2835 this.filters = { 2836 all: { 2837 text: text || l10n.allMediaItems, 2838 props: { 2839 uploadedTo: null, 2840 orderby: 'date', 2841 order: 'DESC', 2842 author: null 2843 }, 2844 priority: 10 2845 }, 2846 2847 uploaded: { 2848 text: l10n.uploadedToThisPost, 2849 props: { 2850 uploadedTo: wp.media.view.settings.post.id, 2851 orderby: 'menuOrder', 2852 order: 'ASC', 2853 author: null 2854 }, 2855 priority: 20 2856 }, 2857 2858 unattached: { 2859 text: l10n.unattached, 2860 props: { 2861 uploadedTo: 0, 2862 orderby: 'menuOrder', 2863 order: 'ASC', 2864 author: null 2865 }, 2866 priority: 50 2867 } 2868 }; 2869 2870 if ( uid ) { 2871 this.filters.mine = { 2872 text: l10n.mine, 2873 props: { 2874 orderby: 'date', 2875 order: 'DESC', 2876 author: uid 2877 }, 2878 priority: 50 2879 }; 2880 } 2881 } 2882 }); 2883 2884 module.exports = Uploaded; 2885 2886 2887 /***/ }), 2888 2889 /***/ 4075: 2890 /***/ ((module) => { 2891 2892 var View = wp.media.View, 2893 $ = jQuery, 2894 Attachment; 2895 2896 /** 2897 * wp.media.view.Attachment 2898 * 2899 * @memberOf wp.media.view 2900 * 2901 * @class 2902 * @augments wp.media.View 2903 * @augments wp.Backbone.View 2904 * @augments Backbone.View 2905 */ 2906 Attachment = View.extend(/** @lends wp.media.view.Attachment.prototype */{ 2907 tagName: 'li', 2908 className: 'attachment', 2909 template: wp.template('attachment'), 2910 2911 attributes: function() { 2912 return { 2913 'tabIndex': 0, 2914 'role': 'checkbox', 2915 'aria-label': this.model.get( 'title' ), 2916 'aria-checked': false, 2917 'data-id': this.model.get( 'id' ) 2918 }; 2919 }, 2920 2921 events: { 2922 'click': 'toggleSelectionHandler', 2923 'change [data-setting]': 'updateSetting', 2924 'change [data-setting] input': 'updateSetting', 2925 'change [data-setting] select': 'updateSetting', 2926 'change [data-setting] textarea': 'updateSetting', 2927 'click .attachment-close': 'removeFromLibrary', 2928 'click .check': 'checkClickHandler', 2929 'keydown': 'toggleSelectionHandler' 2930 }, 2931 2932 buttons: {}, 2933 2934 initialize: function() { 2935 var selection = this.options.selection, 2936 options = _.defaults( this.options, { 2937 rerenderOnModelChange: true 2938 } ); 2939 2940 if ( options.rerenderOnModelChange ) { 2941 this.listenTo( this.model, 'change', this.render ); 2942 } else { 2943 this.listenTo( this.model, 'change:percent', this.progress ); 2944 } 2945 this.listenTo( this.model, 'change:title', this._syncTitle ); 2946 this.listenTo( this.model, 'change:caption', this._syncCaption ); 2947 this.listenTo( this.model, 'change:artist', this._syncArtist ); 2948 this.listenTo( this.model, 'change:album', this._syncAlbum ); 2949 2950 // Update the selection. 2951 this.listenTo( this.model, 'add', this.select ); 2952 this.listenTo( this.model, 'remove', this.deselect ); 2953 if ( selection ) { 2954 selection.on( 'reset', this.updateSelect, this ); 2955 // Update the model's details view. 2956 this.listenTo( this.model, 'selection:single selection:unsingle', this.details ); 2957 this.details( this.model, this.controller.state().get('selection') ); 2958 } 2959 2960 this.listenTo( this.controller.states, 'attachment:compat:waiting attachment:compat:ready', this.updateSave ); 2961 }, 2962 /** 2963 * @return {wp.media.view.Attachment} Returns itself to allow chaining. 2964 */ 2965 dispose: function() { 2966 var selection = this.options.selection; 2967 2968 // Make sure all settings are saved before removing the view. 2969 this.updateAll(); 2970 2971 if ( selection ) { 2972 selection.off( null, null, this ); 2973 } 2974 /** 2975 * call 'dispose' directly on the parent class 2976 */ 2977 View.prototype.dispose.apply( this, arguments ); 2978 return this; 2979 }, 2980 /** 2981 * @return {wp.media.view.Attachment} Returns itself to allow chaining. 2982 */ 2983 render: function() { 2984 var options = _.defaults( this.model.toJSON(), { 2985 orientation: 'landscape', 2986 uploading: false, 2987 type: '', 2988 subtype: '', 2989 icon: '', 2990 filename: '', 2991 caption: '', 2992 title: '', 2993 dateFormatted: '', 2994 width: '', 2995 height: '', 2996 compat: false, 2997 alt: '', 2998 description: '' 2999 }, this.options ); 3000 3001 options.buttons = this.buttons; 3002 options.describe = this.controller.state().get('describe'); 3003 3004 if ( 'image' === options.type ) { 3005 options.size = this.imageSize(); 3006 } 3007 3008 options.can = {}; 3009 if ( options.nonces ) { 3010 options.can.remove = !! options.nonces['delete']; 3011 options.can.save = !! options.nonces.update; 3012 } 3013 3014 if ( this.controller.state().get('allowLocalEdits') && ! options.uploading ) { 3015 options.allowLocalEdits = true; 3016 } 3017 3018 if ( options.uploading && ! options.percent ) { 3019 options.percent = 0; 3020 } 3021 3022 this.views.detach(); 3023 this.$el.html( this.template( options ) ); 3024 3025 this.$el.toggleClass( 'uploading', options.uploading ); 3026 3027 if ( options.uploading ) { 3028 this.$bar = this.$('.media-progress-bar div'); 3029 } else { 3030 delete this.$bar; 3031 } 3032 3033 // Check if the model is selected. 3034 this.updateSelect(); 3035 3036 // Update the save status. 3037 this.updateSave(); 3038 3039 this.views.render(); 3040 3041 return this; 3042 }, 3043 3044 progress: function() { 3045 if ( this.$bar && this.$bar.length ) { 3046 this.$bar.width( this.model.get('percent') + '%' ); 3047 } 3048 }, 3049 3050 /** 3051 * @param {Object} event 3052 */ 3053 toggleSelectionHandler: function( event ) { 3054 var method; 3055 3056 // Don't do anything inside inputs and on the attachment check and remove buttons. 3057 if ( 'INPUT' === event.target.nodeName || 'BUTTON' === event.target.nodeName ) { 3058 return; 3059 } 3060 3061 // Catch arrow events. 3062 if ( 37 === event.keyCode || 38 === event.keyCode || 39 === event.keyCode || 40 === event.keyCode ) { 3063 this.controller.trigger( 'attachment:keydown:arrow', event ); 3064 return; 3065 } 3066 3067 // Catch enter and space events. 3068 if ( 'keydown' === event.type && 13 !== event.keyCode && 32 !== event.keyCode ) { 3069 return; 3070 } 3071 3072 event.preventDefault(); 3073 3074 // In the grid view, bubble up an edit:attachment event to the controller. 3075 if ( this.controller.isModeActive( 'grid' ) ) { 3076 if ( this.controller.isModeActive( 'edit' ) ) { 3077 // Pass the current target to restore focus when closing. 3078 this.controller.trigger( 'edit:attachment', this.model, event.currentTarget ); 3079 return; 3080 } 3081 3082 if ( this.controller.isModeActive( 'select' ) ) { 3083 method = 'toggle'; 3084 } 3085 } 3086 3087 if ( event.shiftKey ) { 3088 method = 'between'; 3089 } else if ( event.ctrlKey || event.metaKey ) { 3090 method = 'toggle'; 3091 } 3092 3093 // Avoid toggles when the command or control key is pressed with the enter key to prevent deselecting the last selected attachment. 3094 if ( ( event.metaKey || event.ctrlKey ) && ( 13 === event.keyCode || 10 === event.keyCode ) ) { 3095 return; 3096 } 3097 3098 this.toggleSelection({ 3099 method: method 3100 }); 3101 3102 this.controller.trigger( 'selection:toggle' ); 3103 }, 3104 /** 3105 * @param {Object} options 3106 */ 3107 toggleSelection: function( options ) { 3108 var collection = this.collection, 3109 selection = this.options.selection, 3110 model = this.model, 3111 method = options && options.method, 3112 single, models, singleIndex, modelIndex; 3113 3114 if ( ! selection ) { 3115 return; 3116 } 3117 3118 single = selection.single(); 3119 method = _.isUndefined( method ) ? selection.multiple : method; 3120 3121 // If the `method` is set to `between`, select all models that 3122 // exist between the current and the selected model. 3123 if ( 'between' === method && single && selection.multiple ) { 3124 // If the models are the same, short-circuit. 3125 if ( single === model ) { 3126 return; 3127 } 3128 3129 singleIndex = collection.indexOf( single ); 3130 modelIndex = collection.indexOf( this.model ); 3131 3132 if ( singleIndex < modelIndex ) { 3133 models = collection.models.slice( singleIndex, modelIndex + 1 ); 3134 } else { 3135 models = collection.models.slice( modelIndex, singleIndex + 1 ); 3136 } 3137 3138 selection.add( models ); 3139 selection.single( model ); 3140 return; 3141 3142 // If the `method` is set to `toggle`, just flip the selection 3143 // status, regardless of whether the model is the single model. 3144 } else if ( 'toggle' === method ) { 3145 selection[ this.selected() ? 'remove' : 'add' ]( model ); 3146 selection.single( model ); 3147 return; 3148 } else if ( 'add' === method ) { 3149 selection.add( model ); 3150 selection.single( model ); 3151 return; 3152 } 3153 3154 // Fixes bug that loses focus when selecting a featured image. 3155 if ( ! method ) { 3156 method = 'add'; 3157 } 3158 3159 if ( method !== 'add' ) { 3160 method = 'reset'; 3161 } 3162 3163 if ( this.selected() ) { 3164 /* 3165 * If the model is the single model, remove it. 3166 * If it is not the same as the single model, 3167 * it now becomes the single model. 3168 */ 3169 selection[ single === model ? 'remove' : 'single' ]( model ); 3170 } else { 3171 /* 3172 * If the model is not selected, run the `method` on the 3173 * selection. By default, we `reset` the selection, but the 3174 * `method` can be set to `add` the model to the selection. 3175 */ 3176 selection[ method ]( model ); 3177 selection.single( model ); 3178 } 3179 }, 3180 3181 updateSelect: function() { 3182 this[ this.selected() ? 'select' : 'deselect' ](); 3183 }, 3184 /** 3185 * @return {unresolved|boolean} 3186 */ 3187 selected: function() { 3188 var selection = this.options.selection; 3189 if ( selection ) { 3190 return !! selection.get( this.model.cid ); 3191 } 3192 }, 3193 /** 3194 * @param {Backbone.Model} model 3195 * @param {Backbone.Collection} collection 3196 */ 3197 select: function( model, collection ) { 3198 var selection = this.options.selection, 3199 controller = this.controller; 3200 3201 /* 3202 * Check if a selection exists and if it's the collection provided. 3203 * If they're not the same collection, bail; we're in another 3204 * selection's event loop. 3205 */ 3206 if ( ! selection || ( collection && collection !== selection ) ) { 3207 return; 3208 } 3209 3210 // Bail if the model is already selected. 3211 if ( this.$el.hasClass( 'selected' ) ) { 3212 return; 3213 } 3214 3215 // Add 'selected' class to model, set aria-checked to true. 3216 this.$el.addClass( 'selected' ).attr( 'aria-checked', true ); 3217 // Make the checkbox tabable, except in media grid (bulk select mode). 3218 if ( ! ( controller.isModeActive( 'grid' ) && controller.isModeActive( 'select' ) ) ) { 3219 this.$( '.check' ).attr( 'tabindex', '0' ); 3220 } 3221 }, 3222 /** 3223 * @param {Backbone.Model} model 3224 * @param {Backbone.Collection} collection 3225 */ 3226 deselect: function( model, collection ) { 3227 var selection = this.options.selection; 3228 3229 /* 3230 * Check if a selection exists and if it's the collection provided. 3231 * If they're not the same collection, bail; we're in another 3232 * selection's event loop. 3233 */ 3234 if ( ! selection || ( collection && collection !== selection ) ) { 3235 return; 3236 } 3237 this.$el.removeClass( 'selected' ).attr( 'aria-checked', false ) 3238 .find( '.check' ).attr( 'tabindex', '-1' ); 3239 }, 3240 /** 3241 * @param {Backbone.Model} model 3242 * @param {Backbone.Collection} collection 3243 */ 3244 details: function( model, collection ) { 3245 var selection = this.options.selection, 3246 details; 3247 3248 if ( selection !== collection ) { 3249 return; 3250 } 3251 3252 details = selection.single(); 3253 this.$el.toggleClass( 'details', details === this.model ); 3254 }, 3255 /** 3256 * @param {string} size 3257 * @return {Object} 3258 */ 3259 imageSize: function( size ) { 3260 var sizes = this.model.get('sizes'), matched = false; 3261 3262 size = size || 'medium'; 3263 3264 // Use the provided image size if possible. 3265 if ( sizes ) { 3266 if ( sizes[ size ] ) { 3267 matched = sizes[ size ]; 3268 } else if ( sizes.large ) { 3269 matched = sizes.large; 3270 } else if ( sizes.thumbnail ) { 3271 matched = sizes.thumbnail; 3272 } else if ( sizes.full ) { 3273 matched = sizes.full; 3274 } 3275 3276 if ( matched ) { 3277 return _.clone( matched ); 3278 } 3279 } 3280 3281 return { 3282 url: this.model.get('url'), 3283 width: this.model.get('width'), 3284 height: this.model.get('height'), 3285 orientation: this.model.get('orientation') 3286 }; 3287 }, 3288 /** 3289 * @param {Object} event 3290 */ 3291 updateSetting: function( event ) { 3292 var $setting = $( event.target ).closest('[data-setting]'), 3293 setting, value; 3294 3295 if ( ! $setting.length ) { 3296 return; 3297 } 3298 3299 setting = $setting.data('setting'); 3300 value = event.target.value; 3301 3302 if ( this.model.get( setting ) !== value ) { 3303 this.save( setting, value ); 3304 } 3305 }, 3306 3307 /** 3308 * Pass all the arguments to the model's save method. 3309 * 3310 * Records the aggregate status of all save requests and updates the 3311 * view's classes accordingly. 3312 */ 3313 save: function() { 3314 var view = this, 3315 save = this._save = this._save || { status: 'ready' }, 3316 request = this.model.save.apply( this.model, arguments ), 3317 requests = save.requests ? $.when( request, save.requests ) : request; 3318 3319 // If we're waiting to remove 'Saved.', stop. 3320 if ( save.savedTimer ) { 3321 clearTimeout( save.savedTimer ); 3322 } 3323 3324 this.updateSave('waiting'); 3325 save.requests = requests; 3326 requests.always( function() { 3327 // If we've performed another request since this one, bail. 3328 if ( save.requests !== requests ) { 3329 return; 3330 } 3331 3332 view.updateSave( requests.state() === 'resolved' ? 'complete' : 'error' ); 3333 save.savedTimer = setTimeout( function() { 3334 view.updateSave('ready'); 3335 delete save.savedTimer; 3336 }, 2000 ); 3337 }); 3338 }, 3339 /** 3340 * @param {string} status 3341 * @return {wp.media.view.Attachment} Returns itself to allow chaining. 3342 */ 3343 updateSave: function( status ) { 3344 var save = this._save = this._save || { status: 'ready' }; 3345 3346 if ( status && status !== save.status ) { 3347 this.$el.removeClass( 'save-' + save.status ); 3348 save.status = status; 3349 } 3350 3351 this.$el.addClass( 'save-' + save.status ); 3352 return this; 3353 }, 3354 3355 updateAll: function() { 3356 var $settings = this.$('[data-setting]'), 3357 model = this.model, 3358 changed; 3359 3360 changed = _.chain( $settings ).map( function( el ) { 3361 var $input = $('input, textarea, select, [value]', el ), 3362 setting, value; 3363 3364 if ( ! $input.length ) { 3365 return; 3366 } 3367 3368 setting = $(el).data('setting'); 3369 value = $input.val(); 3370 3371 // Record the value if it changed. 3372 if ( model.get( setting ) !== value ) { 3373 return [ setting, value ]; 3374 } 3375 }).compact().object().value(); 3376 3377 if ( ! _.isEmpty( changed ) ) { 3378 model.save( changed ); 3379 } 3380 }, 3381 /** 3382 * @param {Object} event 3383 */ 3384 removeFromLibrary: function( event ) { 3385 // Catch enter and space events. 3386 if ( 'keydown' === event.type && 13 !== event.keyCode && 32 !== event.keyCode ) { 3387 return; 3388 } 3389 3390 // Stop propagation so the model isn't selected. 3391 event.stopPropagation(); 3392 3393 this.collection.remove( this.model ); 3394 }, 3395 3396 /** 3397 * Add the model if it isn't in the selection, if it is in the selection, 3398 * remove it. 3399 * 3400 * @param {[type]} event [description] 3401 * @return {[type]} [description] 3402 */ 3403 checkClickHandler: function ( event ) { 3404 var selection = this.options.selection; 3405 if ( ! selection ) { 3406 return; 3407 } 3408 event.stopPropagation(); 3409 if ( selection.where( { id: this.model.get( 'id' ) } ).length ) { 3410 selection.remove( this.model ); 3411 // Move focus back to the attachment tile (from the check). 3412 this.$el.focus(); 3413 } else { 3414 selection.add( this.model ); 3415 } 3416 3417 // Trigger an action button update. 3418 this.controller.trigger( 'selection:toggle' ); 3419 } 3420 }); 3421 3422 // Ensure settings remain in sync between attachment views. 3423 _.each({ 3424 caption: '_syncCaption', 3425 title: '_syncTitle', 3426 artist: '_syncArtist', 3427 album: '_syncAlbum' 3428 }, function( method, setting ) { 3429 /** 3430 * @function _syncCaption 3431 * @memberOf wp.media.view.Attachment 3432 * @instance 3433 * 3434 * @param {Backbone.Model} model 3435 * @param {string} value 3436 * @return {wp.media.view.Attachment} Returns itself to allow chaining. 3437 */ 3438 /** 3439 * @function _syncTitle 3440 * @memberOf wp.media.view.Attachment 3441 * @instance 3442 * 3443 * @param {Backbone.Model} model 3444 * @param {string} value 3445 * @return {wp.media.view.Attachment} Returns itself to allow chaining. 3446 */ 3447 /** 3448 * @function _syncArtist 3449 * @memberOf wp.media.view.Attachment 3450 * @instance 3451 * 3452 * @param {Backbone.Model} model 3453 * @param {string} value 3454 * @return {wp.media.view.Attachment} Returns itself to allow chaining. 3455 */ 3456 /** 3457 * @function _syncAlbum 3458 * @memberOf wp.media.view.Attachment 3459 * @instance 3460 * 3461 * @param {Backbone.Model} model 3462 * @param {string} value 3463 * @return {wp.media.view.Attachment} Returns itself to allow chaining. 3464 */ 3465 Attachment.prototype[ method ] = function( model, value ) { 3466 var $setting = this.$('[data-setting="' + setting + '"]'); 3467 3468 if ( ! $setting.length ) { 3469 return this; 3470 } 3471 3472 /* 3473 * If the updated value is in sync with the value in the DOM, there 3474 * is no need to re-render. If we're currently editing the value, 3475 * it will automatically be in sync, suppressing the re-render for 3476 * the view we're editing, while updating any others. 3477 */ 3478 if ( value === $setting.find('input, textarea, select, [value]').val() ) { 3479 return this; 3480 } 3481 3482 return this.render(); 3483 }; 3484 }); 3485 3486 module.exports = Attachment; 3487 3488 3489 /***/ }), 3490 3491 /***/ 6090: 3492 /***/ ((module) => { 3493 3494 /* global ClipboardJS */ 3495 var Attachment = wp.media.view.Attachment, 3496 l10n = wp.media.view.l10n, 3497 $ = jQuery, 3498 Details, 3499 __ = wp.i18n.__; 3500 3501 Details = Attachment.extend(/** @lends wp.media.view.Attachment.Details.prototype */{ 3502 tagName: 'div', 3503 className: 'attachment-details', 3504 template: wp.template('attachment-details'), 3505 3506 /* 3507 * Reset all the attributes inherited from Attachment including role=checkbox, 3508 * tabindex, etc., as they are inappropriate for this view. See #47458 and [30483] / #30390. 3509 */ 3510 attributes: {}, 3511 3512 events: { 3513 'change [data-setting]': 'updateSetting', 3514 'change [data-setting] input': 'updateSetting', 3515 'change [data-setting] select': 'updateSetting', 3516 'change [data-setting] textarea': 'updateSetting', 3517 'click .delete-attachment': 'deleteAttachment', 3518 'click .trash-attachment': 'trashAttachment', 3519 'click .untrash-attachment': 'untrashAttachment', 3520 'click .edit-attachment': 'editAttachment', 3521 'keydown': 'toggleSelectionHandler' 3522 }, 3523 3524 /** 3525 * Copies the attachment URL to the clipboard. 3526 * 3527 * @since 5.5.0 3528 * 3529 * @param {MouseEvent} event A click event. 3530 * 3531 * @return {void} 3532 */ 3533 copyAttachmentDetailsURLClipboard: function() { 3534 var clipboard = new ClipboardJS( '.copy-attachment-url' ), 3535 successTimeout; 3536 3537 clipboard.on( 'success', function( event ) { 3538 var triggerElement = $( event.trigger ), 3539 successElement = $( '.success', triggerElement.closest( '.copy-to-clipboard-container' ) ); 3540 3541 // Clear the selection and move focus back to the trigger. 3542 event.clearSelection(); 3543 3544 // Show success visual feedback. 3545 clearTimeout( successTimeout ); 3546 successElement.removeClass( 'hidden' ); 3547 3548 // Hide success visual feedback after 3 seconds since last success. 3549 successTimeout = setTimeout( function() { 3550 successElement.addClass( 'hidden' ); 3551 }, 3000 ); 3552 3553 // Handle success audible feedback. 3554 wp.a11y.speak( __( 'The file URL has been copied to your clipboard' ) ); 3555 } ); 3556 }, 3557 3558 /** 3559 * Shows the details of an attachment. 3560 * 3561 * @since 3.5.0 3562 * 3563 * @constructs wp.media.view.Attachment.Details 3564 * @augments wp.media.view.Attachment 3565 * 3566 * @return {void} 3567 */ 3568 initialize: function() { 3569 this.options = _.defaults( this.options, { 3570 rerenderOnModelChange: false 3571 }); 3572 3573 // Call 'initialize' directly on the parent class. 3574 Attachment.prototype.initialize.apply( this, arguments ); 3575 3576 this.copyAttachmentDetailsURLClipboard(); 3577 }, 3578 3579 /** 3580 * Gets the focusable elements to move focus to. 3581 * 3582 * @since 5.3.0 3583 */ 3584 getFocusableElements: function() { 3585 var editedAttachment = $( 'li[data-id="' + this.model.id + '"]' ); 3586 3587 this.previousAttachment = editedAttachment.prev(); 3588 this.nextAttachment = editedAttachment.next(); 3589 }, 3590 3591 /** 3592 * Moves focus to the previous or next attachment in the grid. 3593 * Fallbacks to the upload button or media frame when there are no attachments. 3594 * 3595 * @since 5.3.0 3596 */ 3597 moveFocus: function() { 3598 if ( this.previousAttachment.length ) { 3599 this.previousAttachment.trigger( 'focus' ); 3600 return; 3601 } 3602 3603 if ( this.nextAttachment.length ) { 3604 this.nextAttachment.trigger( 'focus' ); 3605 return; 3606 } 3607 3608 // Fallback: move focus to the "Select Files" button in the media modal. 3609 if ( this.controller.uploader && this.controller.uploader.$browser ) { 3610 this.controller.uploader.$browser.trigger( 'focus' ); 3611 return; 3612 } 3613 3614 // Last fallback. 3615 this.moveFocusToLastFallback(); 3616 }, 3617 3618 /** 3619 * Moves focus to the media frame as last fallback. 3620 * 3621 * @since 5.3.0 3622 */ 3623 moveFocusToLastFallback: function() { 3624 // Last fallback: make the frame focusable and move focus to it. 3625 $( '.media-frame' ) 3626 .attr( 'tabindex', '-1' ) 3627 .trigger( 'focus' ); 3628 }, 3629 3630 /** 3631 * Deletes an attachment. 3632 * 3633 * Deletes an attachment after asking for confirmation. After deletion, 3634 * keeps focus in the modal. 3635 * 3636 * @since 3.5.0 3637 * 3638 * @param {MouseEvent} event A click event. 3639 * 3640 * @return {void} 3641 */ 3642 deleteAttachment: function( event ) { 3643 event.preventDefault(); 3644 3645 this.getFocusableElements(); 3646 3647 if ( window.confirm( l10n.warnDelete ) ) { 3648 this.model.destroy( { 3649 wait: true, 3650 error: function() { 3651 window.alert( l10n.errorDeleting ); 3652 } 3653 } ); 3654 3655 this.moveFocus(); 3656 } 3657 }, 3658 3659 /** 3660 * Sets the Trash state on an attachment, or destroys the model itself. 3661 * 3662 * If the mediaTrash setting is set to true, trashes the attachment. 3663 * Otherwise, the model itself is destroyed. 3664 * 3665 * @since 3.9.0 3666 * 3667 * @param {MouseEvent} event A click event. 3668 * 3669 * @return {void} 3670 */ 3671 trashAttachment: function( event ) { 3672 var library = this.controller.library, 3673 self = this; 3674 event.preventDefault(); 3675 3676 this.getFocusableElements(); 3677 3678 // When in the Media Library and the Media Trash is enabled. 3679 if ( wp.media.view.settings.mediaTrash && 3680 'edit-metadata' === this.controller.content.mode() ) { 3681 3682 this.model.set( 'status', 'trash' ); 3683 this.model.save().done( function() { 3684 library._requery( true ); 3685 /* 3686 * @todo We need to move focus back to the previous, next, or first 3687 * attachment but the library gets re-queried and refreshed. 3688 * Thus, the references to the previous attachments are lost. 3689 * We need an alternate method. 3690 */ 3691 self.moveFocusToLastFallback(); 3692 } ); 3693 } else { 3694 this.model.destroy(); 3695 this.moveFocus(); 3696 } 3697 }, 3698 3699 /** 3700 * Untrashes an attachment. 3701 * 3702 * @since 4.0.0 3703 * 3704 * @param {MouseEvent} event A click event. 3705 * 3706 * @return {void} 3707 */ 3708 untrashAttachment: function( event ) { 3709 var library = this.controller.library; 3710 event.preventDefault(); 3711 3712 this.model.set( 'status', 'inherit' ); 3713 this.model.save().done( function() { 3714 library._requery( true ); 3715 } ); 3716 }, 3717 3718 /** 3719 * Opens the edit page for a specific attachment. 3720 * 3721 * @since 3.5.0 3722 * 3723 * @param {MouseEvent} event A click event. 3724 * 3725 * @return {void} 3726 */ 3727 editAttachment: function( event ) { 3728 var editState = this.controller.states.get( 'edit-image' ); 3729 if ( window.imageEdit && editState ) { 3730 event.preventDefault(); 3731 3732 editState.set( 'image', this.model ); 3733 this.controller.setState( 'edit-image' ); 3734 } else { 3735 this.$el.addClass('needs-refresh'); 3736 } 3737 }, 3738 3739 /** 3740 * Triggers an event on the controller when reverse tabbing (shift+tab). 3741 * 3742 * This event can be used to make sure to move the focus correctly. 3743 * 3744 * @since 4.0.0 3745 * 3746 * @fires wp.media.controller.MediaLibrary#attachment:details:shift-tab 3747 * @fires wp.media.controller.MediaLibrary#attachment:keydown:arrow 3748 * 3749 * @param {KeyboardEvent} event A keyboard event. 3750 * 3751 * @return {boolean|void} Returns false or undefined. 3752 */ 3753 toggleSelectionHandler: function( event ) { 3754 if ( 'keydown' === event.type && 9 === event.keyCode && event.shiftKey && event.target === this.$( ':tabbable' ).get( 0 ) ) { 3755 this.controller.trigger( 'attachment:details:shift-tab', event ); 3756 return false; 3757 } 3758 }, 3759 3760 render: function() { 3761 Attachment.prototype.render.apply( this, arguments ); 3762 3763 wp.media.mixin.removeAllPlayers(); 3764 this.$( 'audio, video' ).each( function (i, elem) { 3765 var el = wp.media.view.MediaDetails.prepareSrc( elem ); 3766 new window.MediaElementPlayer( el, wp.media.mixin.mejsSettings ); 3767 } ); 3768 } 3769 }); 3770 3771 module.exports = Details; 3772 3773 3774 /***/ }), 3775 3776 /***/ 5232: 3777 /***/ ((module) => { 3778 3779 /** 3780 * wp.media.view.Attachment.EditLibrary 3781 * 3782 * @memberOf wp.media.view.Attachment 3783 * 3784 * @class 3785 * @augments wp.media.view.Attachment 3786 * @augments wp.media.View 3787 * @augments wp.Backbone.View 3788 * @augments Backbone.View 3789 */ 3790 var EditLibrary = wp.media.view.Attachment.extend(/** @lends wp.media.view.Attachment.EditLibrary.prototype */{ 3791 buttons: { 3792 close: true 3793 } 3794 }); 3795 3796 module.exports = EditLibrary; 3797 3798 3799 /***/ }), 3800 3801 /***/ 4593: 3802 /***/ ((module) => { 3803 3804 /** 3805 * wp.media.view.Attachment.EditSelection 3806 * 3807 * @memberOf wp.media.view.Attachment 3808 * 3809 * @class 3810 * @augments wp.media.view.Attachment.Selection 3811 * @augments wp.media.view.Attachment 3812 * @augments wp.media.View 3813 * @augments wp.Backbone.View 3814 * @augments Backbone.View 3815 */ 3816 var EditSelection = wp.media.view.Attachment.Selection.extend(/** @lends wp.media.view.Attachment.EditSelection.prototype */{ 3817 buttons: { 3818 close: true 3819 } 3820 }); 3821 3822 module.exports = EditSelection; 3823 3824 3825 /***/ }), 3826 3827 /***/ 3443: 3828 /***/ ((module) => { 3829 3830 /** 3831 * wp.media.view.Attachment.Library 3832 * 3833 * @memberOf wp.media.view.Attachment 3834 * 3835 * @class 3836 * @augments wp.media.view.Attachment 3837 * @augments wp.media.View 3838 * @augments wp.Backbone.View 3839 * @augments Backbone.View 3840 */ 3841 var Library = wp.media.view.Attachment.extend(/** @lends wp.media.view.Attachment.Library.prototype */{ 3842 buttons: { 3843 check: true 3844 } 3845 }); 3846 3847 module.exports = Library; 3848 3849 3850 /***/ }), 3851 3852 /***/ 3962: 3853 /***/ ((module) => { 3854 3855 /** 3856 * wp.media.view.Attachment.Selection 3857 * 3858 * @memberOf wp.media.view.Attachment 3859 * 3860 * @class 3861 * @augments wp.media.view.Attachment 3862 * @augments wp.media.View 3863 * @augments wp.Backbone.View 3864 * @augments Backbone.View 3865 */ 3866 var Selection = wp.media.view.Attachment.extend(/** @lends wp.media.view.Attachment.Selection.prototype */{ 3867 className: 'attachment selection', 3868 3869 // On click, just select the model, instead of removing the model from 3870 // the selection. 3871 toggleSelection: function() { 3872 this.options.selection.single( this.model ); 3873 } 3874 }); 3875 3876 module.exports = Selection; 3877 3878 3879 /***/ }), 3880 3881 /***/ 8142: 3882 /***/ ((module) => { 3883 3884 var View = wp.media.View, 3885 $ = jQuery, 3886 Attachments, 3887 infiniteScrolling = wp.media.view.settings.infiniteScrolling; 3888 3889 Attachments = View.extend(/** @lends wp.media.view.Attachments.prototype */{ 3890 tagName: 'ul', 3891 className: 'attachments', 3892 3893 attributes: { 3894 tabIndex: -1 3895 }, 3896 3897 /** 3898 * Represents the overview of attachments in the Media Library. 3899 * 3900 * The constructor binds events to the collection this view represents when 3901 * adding or removing attachments or resetting the entire collection. 3902 * 3903 * @since 3.5.0 3904 * 3905 * @constructs 3906 * @memberof wp.media.view 3907 * 3908 * @augments wp.media.View 3909 * 3910 * @listens collection:add 3911 * @listens collection:remove 3912 * @listens collection:reset 3913 * @listens controller:library:selection:add 3914 * @listens scrollElement:scroll 3915 * @listens this:ready 3916 * @listens controller:open 3917 */ 3918 initialize: function() { 3919 this.el.id = _.uniqueId('__attachments-view-'); 3920 3921 /** 3922 * @since 5.8.0 Added the `infiniteScrolling` parameter. 3923 * 3924 * @param infiniteScrolling Whether to enable infinite scrolling or use 3925 * the default "load more" button. 3926 * @param refreshSensitivity The time in milliseconds to throttle the scroll 3927 * handler. 3928 * @param refreshThreshold The amount of pixels that should be scrolled before 3929 * loading more attachments from the server. 3930 * @param AttachmentView The view class to be used for models in the 3931 * collection. 3932 * @param sortable A jQuery sortable options object 3933 * ( http://api.jqueryui.com/sortable/ ). 3934 * @param resize A boolean indicating whether or not to listen to 3935 * resize events. 3936 * @param idealColumnWidth The width in pixels which a column should have when 3937 * calculating the total number of columns. 3938 */ 3939 _.defaults( this.options, { 3940 infiniteScrolling: infiniteScrolling || false, 3941 refreshSensitivity: wp.media.isTouchDevice ? 300 : 200, 3942 refreshThreshold: 3, 3943 AttachmentView: wp.media.view.Attachment, 3944 sortable: false, 3945 resize: true, 3946 idealColumnWidth: $( window ).width() < 640 ? 135 : 150 3947 }); 3948 3949 this._viewsByCid = {}; 3950 this.$window = $( window ); 3951 this.resizeEvent = 'resize.media-modal-columns'; 3952 3953 this.collection.on( 'add', function( attachment ) { 3954 this.views.add( this.createAttachmentView( attachment ), { 3955 at: this.collection.indexOf( attachment ) 3956 }); 3957 }, this ); 3958 3959 /* 3960 * Find the view to be removed, delete it and call the remove function to clear 3961 * any set event handlers. 3962 */ 3963 this.collection.on( 'remove', function( attachment ) { 3964 var view = this._viewsByCid[ attachment.cid ]; 3965 delete this._viewsByCid[ attachment.cid ]; 3966 3967 if ( view ) { 3968 view.remove(); 3969 } 3970 }, this ); 3971 3972 this.collection.on( 'reset', this.render, this ); 3973 3974 this.controller.on( 'library:selection:add', this.attachmentFocus, this ); 3975 3976 if ( this.options.infiniteScrolling ) { 3977 // Throttle the scroll handler and bind this. 3978 this.scroll = _.chain( this.scroll ).bind( this ).throttle( this.options.refreshSensitivity ).value(); 3979 3980 this.options.scrollElement = this.options.scrollElement || this.el; 3981 $( this.options.scrollElement ).on( 'scroll', this.scroll ); 3982 } 3983 3984 this.initSortable(); 3985 3986 _.bindAll( this, 'setColumns' ); 3987 3988 if ( this.options.resize ) { 3989 this.on( 'ready', this.bindEvents ); 3990 this.controller.on( 'open', this.setColumns ); 3991 3992 /* 3993 * Call this.setColumns() after this view has been rendered in the 3994 * DOM so attachments get proper width applied. 3995 */ 3996 _.defer( this.setColumns, this ); 3997 } 3998 }, 3999 4000 /** 4001 * Listens to the resizeEvent on the window. 4002 * 4003 * Adjusts the amount of columns accordingly. First removes any existing event 4004 * handlers to prevent duplicate listeners. 4005 * 4006 * @since 4.0.0 4007 * 4008 * @listens window:resize 4009 * 4010 * @return {void} 4011 */ 4012 bindEvents: function() { 4013 this.$window.off( this.resizeEvent ).on( this.resizeEvent, _.debounce( this.setColumns, 50 ) ); 4014 }, 4015 4016 /** 4017 * Focuses the first item in the collection. 4018 * 4019 * @since 4.0.0 4020 * 4021 * @return {void} 4022 */ 4023 attachmentFocus: function() { 4024 /* 4025 * @todo When uploading new attachments, this tries to move focus to 4026 * the attachments grid. Actually, a progress bar gets initially displayed 4027 * and then updated when uploading completes, so focus is lost. 4028 * Additionally: this view is used for both the attachments list and 4029 * the list of selected attachments in the bottom media toolbar. Thus, when 4030 * uploading attachments, it is called twice and returns two different `this`. 4031 * `this.columns` is truthy within the modal. 4032 */ 4033 if ( this.columns ) { 4034 // Move focus to the grid list within the modal. 4035 this.$el.focus(); 4036 } 4037 }, 4038 4039 /** 4040 * Restores focus to the selected item in the collection. 4041 * 4042 * Moves focus back to the first selected attachment in the grid. Used when 4043 * tabbing backwards from the attachment details sidebar. 4044 * See media.view.AttachmentsBrowser. 4045 * 4046 * @since 4.0.0 4047 * 4048 * @return {void} 4049 */ 4050 restoreFocus: function() { 4051 this.$( 'li.selected:first' ).focus(); 4052 }, 4053 4054 /** 4055 * Handles events for arrow key presses. 4056 * 4057 * Focuses the attachment in the direction of the used arrow key if it exists. 4058 * 4059 * @since 4.0.0 4060 * 4061 * @param {KeyboardEvent} event The keyboard event that triggered this function. 4062 * 4063 * @return {void} 4064 */ 4065 arrowEvent: function( event ) { 4066 var attachments = this.$el.children( 'li' ), 4067 perRow = this.columns, 4068 index = attachments.filter( ':focus' ).index(), 4069 row = ( index + 1 ) <= perRow ? 1 : Math.ceil( ( index + 1 ) / perRow ); 4070 4071 if ( index === -1 ) { 4072 return; 4073 } 4074 4075 // Left arrow = 37. 4076 if ( 37 === event.keyCode ) { 4077 if ( 0 === index ) { 4078 return; 4079 } 4080 attachments.eq( index - 1 ).focus(); 4081 } 4082 4083 // Up arrow = 38. 4084 if ( 38 === event.keyCode ) { 4085 if ( 1 === row ) { 4086 return; 4087 } 4088 attachments.eq( index - perRow ).focus(); 4089 } 4090 4091 // Right arrow = 39. 4092 if ( 39 === event.keyCode ) { 4093 if ( attachments.length === index ) { 4094 return; 4095 } 4096 attachments.eq( index + 1 ).focus(); 4097 } 4098 4099 // Down arrow = 40. 4100 if ( 40 === event.keyCode ) { 4101 if ( Math.ceil( attachments.length / perRow ) === row ) { 4102 return; 4103 } 4104 attachments.eq( index + perRow ).focus(); 4105 } 4106 }, 4107 4108 /** 4109 * Clears any set event handlers. 4110 * 4111 * @since 3.5.0 4112 * 4113 * @return {void} 4114 */ 4115 dispose: function() { 4116 this.collection.props.off( null, null, this ); 4117 if ( this.options.resize ) { 4118 this.$window.off( this.resizeEvent ); 4119 } 4120 4121 // Call 'dispose' directly on the parent class. 4122 View.prototype.dispose.apply( this, arguments ); 4123 }, 4124 4125 /** 4126 * Calculates the amount of columns. 4127 * 4128 * Calculates the amount of columns and sets it on the data-columns attribute 4129 * of .media-frame-content. 4130 * 4131 * @since 4.0.0 4132 * 4133 * @return {void} 4134 */ 4135 setColumns: function() { 4136 var prev = this.columns, 4137 width = this.$el.width(); 4138 4139 if ( width ) { 4140 this.columns = Math.min( Math.round( width / this.options.idealColumnWidth ), 12 ) || 1; 4141 4142 if ( ! prev || prev !== this.columns ) { 4143 this.$el.closest( '.media-frame-content' ).attr( 'data-columns', this.columns ); 4144 } 4145 } 4146 }, 4147 4148 /** 4149 * Initializes jQuery sortable on the attachment list. 4150 * 4151 * Fails gracefully if jQuery sortable doesn't exist or isn't passed 4152 * in the options. 4153 * 4154 * @since 3.5.0 4155 * 4156 * @fires collection:reset 4157 * 4158 * @return {void} 4159 */ 4160 initSortable: function() { 4161 var collection = this.collection; 4162 4163 if ( ! this.options.sortable || ! $.fn.sortable ) { 4164 return; 4165 } 4166 4167 this.$el.sortable( _.extend({ 4168 // If the `collection` has a `comparator`, disable sorting. 4169 disabled: !! collection.comparator, 4170 4171 /* 4172 * Change the position of the attachment as soon as the mouse pointer 4173 * overlaps a thumbnail. 4174 */ 4175 tolerance: 'pointer', 4176 4177 // Record the initial `index` of the dragged model. 4178 start: function( event, ui ) { 4179 ui.item.data('sortableIndexStart', ui.item.index()); 4180 }, 4181 4182 /* 4183 * Update the model's index in the collection. Do so silently, as the view 4184 * is already accurate. 4185 */ 4186 update: function( event, ui ) { 4187 var model = collection.at( ui.item.data('sortableIndexStart') ), 4188 comparator = collection.comparator; 4189 4190 // Temporarily disable the comparator to prevent `add` 4191 // from re-sorting. 4192 delete collection.comparator; 4193 4194 // Silently shift the model to its new index. 4195 collection.remove( model, { 4196 silent: true 4197 }); 4198 collection.add( model, { 4199 silent: true, 4200 at: ui.item.index() 4201 }); 4202 4203 // Restore the comparator. 4204 collection.comparator = comparator; 4205 4206 // Fire the `reset` event to ensure other collections sync. 4207 collection.trigger( 'reset', collection ); 4208 4209 // If the collection is sorted by menu order, update the menu order. 4210 collection.saveMenuOrder(); 4211 } 4212 }, this.options.sortable ) ); 4213 4214 /* 4215 * If the `orderby` property is changed on the `collection`, 4216 * check to see if we have a `comparator`. If so, disable sorting. 4217 */ 4218 collection.props.on( 'change:orderby', function() { 4219 this.$el.sortable( 'option', 'disabled', !! collection.comparator ); 4220 }, this ); 4221 4222 this.collection.props.on( 'change:orderby', this.refreshSortable, this ); 4223 this.refreshSortable(); 4224 }, 4225 4226 /** 4227 * Disables jQuery sortable if collection has a comparator or collection.orderby 4228 * equals menuOrder. 4229 * 4230 * @since 3.5.0 4231 * 4232 * @return {void} 4233 */ 4234 refreshSortable: function() { 4235 if ( ! this.options.sortable || ! $.fn.sortable ) { 4236 return; 4237 } 4238 4239 var collection = this.collection, 4240 orderby = collection.props.get('orderby'), 4241 enabled = 'menuOrder' === orderby || ! collection.comparator; 4242 4243 this.$el.sortable( 'option', 'disabled', ! enabled ); 4244 }, 4245 4246 /** 4247 * Creates a new view for an attachment and adds it to _viewsByCid. 4248 * 4249 * @since 3.5.0 4250 * 4251 * @param {wp.media.model.Attachment} attachment 4252 * 4253 * @return {wp.media.View} The created view. 4254 */ 4255 createAttachmentView: function( attachment ) { 4256 var view = new this.options.AttachmentView({ 4257 controller: this.controller, 4258 model: attachment, 4259 collection: this.collection, 4260 selection: this.options.selection 4261 }); 4262 4263 return this._viewsByCid[ attachment.cid ] = view; 4264 }, 4265 4266 /** 4267 * Prepares view for display. 4268 * 4269 * Creates views for every attachment in collection if the collection is not 4270 * empty, otherwise clears all views and loads more attachments. 4271 * 4272 * @since 3.5.0 4273 * 4274 * @return {void} 4275 */ 4276 prepare: function() { 4277 if ( this.collection.length ) { 4278 this.views.set( this.collection.map( this.createAttachmentView, this ) ); 4279 } else { 4280 this.views.unset(); 4281 if ( this.options.infiniteScrolling ) { 4282 this.collection.more().done( this.scroll ); 4283 } 4284 } 4285 }, 4286 4287 /** 4288 * Triggers the scroll function to check if we should query for additional 4289 * attachments right away. 4290 * 4291 * @since 3.5.0 4292 * 4293 * @return {void} 4294 */ 4295 ready: function() { 4296 if ( this.options.infiniteScrolling ) { 4297 this.scroll(); 4298 } 4299 }, 4300 4301 /** 4302 * Handles scroll events. 4303 * 4304 * Shows the spinner if we're close to the bottom. Loads more attachments from 4305 * server if we're {refreshThreshold} times away from the bottom. 4306 * 4307 * @since 3.5.0 4308 * 4309 * @return {void} 4310 */ 4311 scroll: function() { 4312 var view = this, 4313 el = this.options.scrollElement, 4314 scrollTop = el.scrollTop, 4315 toolbar; 4316 4317 /* 4318 * The scroll event occurs on the document, but the element that should be 4319 * checked is the document body. 4320 */ 4321 if ( el === document ) { 4322 el = document.body; 4323 scrollTop = $(document).scrollTop(); 4324 } 4325 4326 if ( ! $(el).is(':visible') || ! this.collection.hasMore() ) { 4327 return; 4328 } 4329 4330 toolbar = this.views.parent.toolbar; 4331 4332 // Show the spinner only if we are close to the bottom. 4333 if ( el.scrollHeight - ( scrollTop + el.clientHeight ) < el.clientHeight / 3 ) { 4334 toolbar.get('spinner').show(); 4335 } 4336 4337 if ( el.scrollHeight < scrollTop + ( el.clientHeight * this.options.refreshThreshold ) ) { 4338 this.collection.more().done(function() { 4339 view.scroll(); 4340 toolbar.get('spinner').hide(); 4341 }); 4342 } 4343 } 4344 }); 4345 4346 module.exports = Attachments; 4347 4348 4349 /***/ }), 4350 4351 /***/ 6829: 4352 /***/ ((module) => { 4353 4354 var View = wp.media.View, 4355 mediaTrash = wp.media.view.settings.mediaTrash, 4356 l10n = wp.media.view.l10n, 4357 $ = jQuery, 4358 AttachmentsBrowser, 4359 infiniteScrolling = wp.media.view.settings.infiniteScrolling, 4360 __ = wp.i18n.__, 4361 sprintf = wp.i18n.sprintf; 4362 4363 /** 4364 * wp.media.view.AttachmentsBrowser 4365 * 4366 * @memberOf wp.media.view 4367 * 4368 * @class 4369 * @augments wp.media.View 4370 * @augments wp.Backbone.View 4371 * @augments Backbone.View 4372 * 4373 * @param {object} [options] The options hash passed to the view. 4374 * @param {boolean|string} [options.filters=false] Which filters to show in the browser's toolbar. 4375 * Accepts 'uploaded' and 'all'. 4376 * @param {boolean} [options.search=true] Whether to show the search interface in the 4377 * browser's toolbar. 4378 * @param {boolean} [options.date=true] Whether to show the date filter in the 4379 * browser's toolbar. 4380 * @param {boolean} [options.display=false] Whether to show the attachments display settings 4381 * view in the sidebar. 4382 * @param {boolean|string} [options.sidebar=true] Whether to create a sidebar for the browser. 4383 * Accepts true, false, and 'errors'. 4384 */ 4385 AttachmentsBrowser = View.extend(/** @lends wp.media.view.AttachmentsBrowser.prototype */{ 4386 tagName: 'div', 4387 className: 'attachments-browser', 4388 4389 initialize: function() { 4390 _.defaults( this.options, { 4391 filters: false, 4392 search: true, 4393 date: true, 4394 display: false, 4395 sidebar: true, 4396 AttachmentView: wp.media.view.Attachment.Library 4397 }); 4398 4399 this.controller.on( 'toggle:upload:attachment', this.toggleUploader, this ); 4400 this.controller.on( 'edit:selection', this.editSelection ); 4401 4402 // In the Media Library, the sidebar is used to display errors before the attachments grid. 4403 if ( this.options.sidebar && 'errors' === this.options.sidebar ) { 4404 this.createSidebar(); 4405 } 4406 4407 /* 4408 * In the grid mode (the Media Library), place the Inline Uploader before 4409 * other sections so that the visual order and the DOM order match. This way, 4410 * the Inline Uploader in the Media Library is right after the "Add New" 4411 * button, see ticket #37188. 4412 */ 4413 if ( this.controller.isModeActive( 'grid' ) ) { 4414 this.createUploader(); 4415 4416 /* 4417 * Create a multi-purpose toolbar. Used as main toolbar in the Media Library 4418 * and also for other things, for example the "Drag and drop to reorder" and 4419 * "Suggested dimensions" info in the media modal. 4420 */ 4421 this.createToolbar(); 4422 } else { 4423 this.createToolbar(); 4424 this.createUploader(); 4425 } 4426 4427 // Add a heading before the attachments list. 4428 this.createAttachmentsHeading(); 4429 4430 // Create the attachments wrapper view. 4431 this.createAttachmentsWrapperView(); 4432 4433 if ( ! infiniteScrolling ) { 4434 this.$el.addClass( 'has-load-more' ); 4435 this.createLoadMoreView(); 4436 } 4437 4438 // For accessibility reasons, place the normal sidebar after the attachments, see ticket #36909. 4439 if ( this.options.sidebar && 'errors' !== this.options.sidebar ) { 4440 this.createSidebar(); 4441 } 4442 4443 this.updateContent(); 4444 4445 if ( ! infiniteScrolling ) { 4446 this.updateLoadMoreView(); 4447 } 4448 4449 if ( ! this.options.sidebar || 'errors' === this.options.sidebar ) { 4450 this.$el.addClass( 'hide-sidebar' ); 4451 4452 if ( 'errors' === this.options.sidebar ) { 4453 this.$el.addClass( 'sidebar-for-errors' ); 4454 } 4455 } 4456 4457 this.collection.on( 'add remove reset', this.updateContent, this ); 4458 4459 if ( ! infiniteScrolling ) { 4460 this.collection.on( 'add remove reset', this.updateLoadMoreView, this ); 4461 } 4462 4463 // The non-cached or cached attachments query has completed. 4464 this.collection.on( 'attachments:received', this.announceSearchResults, this ); 4465 }, 4466 4467 /** 4468 * Updates the `wp.a11y.speak()` ARIA live region with a message to communicate 4469 * the number of search results to screen reader users. This function is 4470 * debounced because the collection updates multiple times. 4471 * 4472 * @since 5.3.0 4473 * 4474 * @return {void} 4475 */ 4476 announceSearchResults: _.debounce( function() { 4477 var count, 4478 /* translators: Accessibility text. %d: Number of attachments found in a search. */ 4479 mediaFoundHasMoreResultsMessage = __( 'Number of media items displayed: %d. Click load more for more results.' ); 4480 4481 if ( infiniteScrolling ) { 4482 /* translators: Accessibility text. %d: Number of attachments found in a search. */ 4483 mediaFoundHasMoreResultsMessage = __( 'Number of media items displayed: %d. Scroll the page for more results.' ); 4484 } 4485 4486 if ( this.collection.mirroring && this.collection.mirroring.args.s ) { 4487 count = this.collection.length; 4488 4489 if ( 0 === count ) { 4490 wp.a11y.speak( l10n.noMediaTryNewSearch ); 4491 return; 4492 } 4493 4494 if ( this.collection.hasMore() ) { 4495 wp.a11y.speak( mediaFoundHasMoreResultsMessage.replace( '%d', count ) ); 4496 return; 4497 } 4498 4499 wp.a11y.speak( l10n.mediaFound.replace( '%d', count ) ); 4500 } 4501 }, 200 ), 4502 4503 editSelection: function( modal ) { 4504 // When editing a selection, move focus to the "Go to library" button. 4505 modal.$( '.media-button-backToLibrary' ).focus(); 4506 }, 4507 4508 /** 4509 * @return {wp.media.view.AttachmentsBrowser} Returns itself to allow chaining. 4510 */ 4511 dispose: function() { 4512 this.options.selection.off( null, null, this ); 4513 View.prototype.dispose.apply( this, arguments ); 4514 return this; 4515 }, 4516 4517 createToolbar: function() { 4518 var LibraryViewSwitcher, Filters, toolbarOptions, 4519 showFilterByType = -1 !== $.inArray( this.options.filters, [ 'uploaded', 'all' ] ); 4520 4521 toolbarOptions = { 4522 controller: this.controller 4523 }; 4524 4525 if ( this.controller.isModeActive( 'grid' ) ) { 4526 toolbarOptions.className = 'media-toolbar wp-filter'; 4527 } 4528 4529 /** 4530 * @member {wp.media.view.Toolbar} 4531 */ 4532 this.toolbar = new wp.media.view.Toolbar( toolbarOptions ); 4533 4534 this.views.add( this.toolbar ); 4535 4536 this.toolbar.set( 'spinner', new wp.media.view.Spinner({ 4537 priority: -20 4538 }) ); 4539 4540 if ( showFilterByType || this.options.date ) { 4541 /* 4542 * Create a h2 heading before the select elements that filter attachments. 4543 * This heading is visible in the modal and visually hidden in the grid. 4544 */ 4545 this.toolbar.set( 'filters-heading', new wp.media.view.Heading( { 4546 priority: -100, 4547 text: l10n.filterAttachments, 4548 level: 'h2', 4549 className: 'media-attachments-filter-heading' 4550 }).render() ); 4551 } 4552 4553 if ( showFilterByType ) { 4554 // "Filters" is a <select>, a visually hidden label element needs to be rendered before. 4555 this.toolbar.set( 'filtersLabel', new wp.media.view.Label({ 4556 value: l10n.filterByType, 4557 attributes: { 4558 'for': 'media-attachment-filters' 4559 }, 4560 priority: -80 4561 }).render() ); 4562 4563 if ( 'uploaded' === this.options.filters ) { 4564 this.toolbar.set( 'filters', new wp.media.view.AttachmentFilters.Uploaded({ 4565 controller: this.controller, 4566 model: this.collection.props, 4567 priority: -80 4568 }).render() ); 4569 } else { 4570 Filters = new wp.media.view.AttachmentFilters.All({ 4571 controller: this.controller, 4572 model: this.collection.props, 4573 priority: -80 4574 }); 4575 4576 this.toolbar.set( 'filters', Filters.render() ); 4577 } 4578 } 4579 4580 /* 4581 * Feels odd to bring the global media library switcher into the Attachment browser view. 4582 * Is this a use case for doAction( 'add:toolbar-items:attachments-browser', this.toolbar ); 4583 * which the controller can tap into and add this view? 4584 */ 4585 if ( this.controller.isModeActive( 'grid' ) ) { 4586 LibraryViewSwitcher = View.extend({ 4587 className: 'view-switch media-grid-view-switch', 4588 template: wp.template( 'media-library-view-switcher') 4589 }); 4590 4591 this.toolbar.set( 'libraryViewSwitcher', new LibraryViewSwitcher({ 4592 controller: this.controller, 4593 priority: -90 4594 }).render() ); 4595 4596 // DateFilter is a <select>, a visually hidden label element needs to be rendered before. 4597 this.toolbar.set( 'dateFilterLabel', new wp.media.view.Label({ 4598 value: l10n.filterByDate, 4599 attributes: { 4600 'for': 'media-attachment-date-filters' 4601 }, 4602 priority: -75 4603 }).render() ); 4604 this.toolbar.set( 'dateFilter', new wp.media.view.DateFilter({ 4605 controller: this.controller, 4606 model: this.collection.props, 4607 priority: -75 4608 }).render() ); 4609 4610 // BulkSelection is a <div> with subviews, including screen reader text. 4611 this.toolbar.set( 'selectModeToggleButton', new wp.media.view.SelectModeToggleButton({ 4612 text: l10n.bulkSelect, 4613 controller: this.controller, 4614 priority: -70 4615 }).render() ); 4616 4617 this.toolbar.set( 'deleteSelectedButton', new wp.media.view.DeleteSelectedButton({ 4618 filters: Filters, 4619 style: 'primary', 4620 disabled: true, 4621 text: mediaTrash ? l10n.trashSelected : l10n.deletePermanently, 4622 controller: this.controller, 4623 priority: -80, 4624 click: function() { 4625 var changed = [], removed = [], 4626 selection = this.controller.state().get( 'selection' ), 4627 library = this.controller.state().get( 'library' ); 4628 4629 if ( ! selection.length ) { 4630 return; 4631 } 4632 4633 if ( ! mediaTrash && ! window.confirm( l10n.warnBulkDelete ) ) { 4634 return; 4635 } 4636 4637 if ( mediaTrash && 4638 'trash' !== selection.at( 0 ).get( 'status' ) && 4639 ! window.confirm( l10n.warnBulkTrash ) ) { 4640 4641 return; 4642 } 4643 4644 selection.each( function( model ) { 4645 if ( ! model.get( 'nonces' )['delete'] ) { 4646 removed.push( model ); 4647 return; 4648 } 4649 4650 if ( mediaTrash && 'trash' === model.get( 'status' ) ) { 4651 model.set( 'status', 'inherit' ); 4652 changed.push( model.save() ); 4653 removed.push( model ); 4654 } else if ( mediaTrash ) { 4655 model.set( 'status', 'trash' ); 4656 changed.push( model.save() ); 4657 removed.push( model ); 4658 } else { 4659 model.destroy({wait: true}); 4660 } 4661 } ); 4662 4663 if ( changed.length ) { 4664 selection.remove( removed ); 4665 4666 $.when.apply( null, changed ).then( _.bind( function() { 4667 library._requery( true ); 4668 this.controller.trigger( 'selection:action:done' ); 4669 }, this ) ); 4670 } else { 4671 this.controller.trigger( 'selection:action:done' ); 4672 } 4673 } 4674 }).render() ); 4675 4676 if ( mediaTrash ) { 4677 this.toolbar.set( 'deleteSelectedPermanentlyButton', new wp.media.view.DeleteSelectedPermanentlyButton({ 4678 filters: Filters, 4679 style: 'link button-link-delete', 4680 disabled: true, 4681 text: l10n.deletePermanently, 4682 controller: this.controller, 4683 priority: -55, 4684 click: function() { 4685 var removed = [], 4686 destroy = [], 4687 selection = this.controller.state().get( 'selection' ); 4688 4689 if ( ! selection.length || ! window.confirm( l10n.warnBulkDelete ) ) { 4690 return; 4691 } 4692 4693 selection.each( function( model ) { 4694 if ( ! model.get( 'nonces' )['delete'] ) { 4695 removed.push( model ); 4696 return; 4697 } 4698 4699 destroy.push( model ); 4700 } ); 4701 4702 if ( removed.length ) { 4703 selection.remove( removed ); 4704 } 4705 4706 if ( destroy.length ) { 4707 $.when.apply( null, destroy.map( function (item) { 4708 return item.destroy(); 4709 } ) ).then( _.bind( function() { 4710 this.controller.trigger( 'selection:action:done' ); 4711 }, this ) ); 4712 } 4713 } 4714 }).render() ); 4715 } 4716 4717 } else if ( this.options.date ) { 4718 // DateFilter is a <select>, a visually hidden label element needs to be rendered before. 4719 this.toolbar.set( 'dateFilterLabel', new wp.media.view.Label({ 4720 value: l10n.filterByDate, 4721 attributes: { 4722 'for': 'media-attachment-date-filters' 4723 }, 4724 priority: -75 4725 }).render() ); 4726 this.toolbar.set( 'dateFilter', new wp.media.view.DateFilter({ 4727 controller: this.controller, 4728 model: this.collection.props, 4729 priority: -75 4730 }).render() ); 4731 } 4732 4733 if ( this.options.search ) { 4734 // Search is an input, a label element needs to be rendered before. 4735 this.toolbar.set( 'searchLabel', new wp.media.view.Label({ 4736 value: l10n.searchLabel, 4737 className: 'media-search-input-label', 4738 attributes: { 4739 'for': 'media-search-input' 4740 }, 4741 priority: 60 4742 }).render() ); 4743 this.toolbar.set( 'search', new wp.media.view.Search({ 4744 controller: this.controller, 4745 model: this.collection.props, 4746 priority: 60 4747 }).render() ); 4748 } 4749 4750 if ( this.options.dragInfo ) { 4751 this.toolbar.set( 'dragInfo', new View({ 4752 el: $( '<div class="instructions">' + l10n.dragInfo + '</div>' )[0], 4753 priority: -40 4754 }) ); 4755 } 4756 4757 if ( this.options.suggestedWidth && this.options.suggestedHeight ) { 4758 this.toolbar.set( 'suggestedDimensions', new View({ 4759 el: $( '<div class="instructions">' + l10n.suggestedDimensions.replace( '%1$s', this.options.suggestedWidth ).replace( '%2$s', this.options.suggestedHeight ) + '</div>' )[0], 4760 priority: -40 4761 }) ); 4762 } 4763 }, 4764 4765 updateContent: function() { 4766 var view = this, 4767 noItemsView; 4768 4769 if ( this.controller.isModeActive( 'grid' ) ) { 4770 // Usually the media library. 4771 noItemsView = view.attachmentsNoResults; 4772 } else { 4773 // Usually the media modal. 4774 noItemsView = view.uploader; 4775 } 4776 4777 if ( ! this.collection.length ) { 4778 this.toolbar.get( 'spinner' ).show(); 4779 this.toolbar.$( '.media-bg-overlay' ).show(); 4780 this.dfd = this.collection.more().done( function() { 4781 if ( ! view.collection.length ) { 4782 noItemsView.$el.removeClass( 'hidden' ); 4783 } else { 4784 noItemsView.$el.addClass( 'hidden' ); 4785 } 4786 view.toolbar.get( 'spinner' ).hide(); 4787 view.toolbar.$( '.media-bg-overlay' ).hide(); 4788 } ); 4789 } else { 4790 noItemsView.$el.addClass( 'hidden' ); 4791 view.toolbar.get( 'spinner' ).hide(); 4792 this.toolbar.$( '.media-bg-overlay' ).hide(); 4793 } 4794 }, 4795 4796 createUploader: function() { 4797 this.uploader = new wp.media.view.UploaderInline({ 4798 controller: this.controller, 4799 status: false, 4800 message: this.controller.isModeActive( 'grid' ) ? '' : l10n.noItemsFound, 4801 canClose: this.controller.isModeActive( 'grid' ) 4802 }); 4803 4804 this.uploader.$el.addClass( 'hidden' ); 4805 this.views.add( this.uploader ); 4806 }, 4807 4808 toggleUploader: function() { 4809 if ( this.uploader.$el.hasClass( 'hidden' ) ) { 4810 this.uploader.show(); 4811 } else { 4812 this.uploader.hide(); 4813 } 4814 }, 4815 4816 /** 4817 * Creates the Attachments wrapper view. 4818 * 4819 * @since 5.8.0 4820 * 4821 * @return {void} 4822 */ 4823 createAttachmentsWrapperView: function() { 4824 this.attachmentsWrapper = new wp.media.View( { 4825 className: 'attachments-wrapper' 4826 } ); 4827 4828 // Create the list of attachments. 4829 this.views.add( this.attachmentsWrapper ); 4830 this.createAttachments(); 4831 }, 4832 4833 createAttachments: function() { 4834 this.attachments = new wp.media.view.Attachments({ 4835 controller: this.controller, 4836 collection: this.collection, 4837 selection: this.options.selection, 4838 model: this.model, 4839 sortable: this.options.sortable, 4840 scrollElement: this.options.scrollElement, 4841 idealColumnWidth: this.options.idealColumnWidth, 4842 4843 // The single `Attachment` view to be used in the `Attachments` view. 4844 AttachmentView: this.options.AttachmentView 4845 }); 4846 4847 // Add keydown listener to the instance of the Attachments view. 4848 this.controller.on( 'attachment:keydown:arrow', _.bind( this.attachments.arrowEvent, this.attachments ) ); 4849 this.controller.on( 'attachment:details:shift-tab', _.bind( this.attachments.restoreFocus, this.attachments ) ); 4850 4851 this.views.add( '.attachments-wrapper', this.attachments ); 4852 4853 if ( this.controller.isModeActive( 'grid' ) ) { 4854 this.attachmentsNoResults = new View({ 4855 controller: this.controller, 4856 tagName: 'p' 4857 }); 4858 4859 this.attachmentsNoResults.$el.addClass( 'hidden no-media' ); 4860 this.attachmentsNoResults.$el.html( l10n.noMedia ); 4861 4862 this.views.add( this.attachmentsNoResults ); 4863 } 4864 }, 4865 4866 /** 4867 * Creates the load more button and attachments counter view. 4868 * 4869 * @since 5.8.0 4870 * 4871 * @return {void} 4872 */ 4873 createLoadMoreView: function() { 4874 var view = this; 4875 4876 this.loadMoreWrapper = new View( { 4877 controller: this.controller, 4878 className: 'load-more-wrapper' 4879 } ); 4880 4881 this.loadMoreCount = new View( { 4882 controller: this.controller, 4883 tagName: 'p', 4884 className: 'load-more-count hidden' 4885 } ); 4886 4887 this.loadMoreButton = new wp.media.view.Button( { 4888 text: __( 'Load more' ), 4889 className: 'load-more hidden', 4890 style: 'primary', 4891 size: '', 4892 click: function() { 4893 view.loadMoreAttachments(); 4894 } 4895 } ); 4896 4897 this.loadMoreSpinner = new wp.media.view.Spinner(); 4898 4899 this.loadMoreJumpToFirst = new wp.media.view.Button( { 4900 text: __( 'Jump to first loaded item' ), 4901 className: 'load-more-jump hidden', 4902 size: '', 4903 click: function() { 4904 view.jumpToFirstAddedItem(); 4905 } 4906 } ); 4907 4908 this.views.add( '.attachments-wrapper', this.loadMoreWrapper ); 4909 this.views.add( '.load-more-wrapper', this.loadMoreSpinner ); 4910 this.views.add( '.load-more-wrapper', this.loadMoreCount ); 4911 this.views.add( '.load-more-wrapper', this.loadMoreButton ); 4912 this.views.add( '.load-more-wrapper', this.loadMoreJumpToFirst ); 4913 }, 4914 4915 /** 4916 * Updates the Load More view. This function is debounced because the 4917 * collection updates multiple times at the add, remove, and reset events. 4918 * We need it to run only once, after all attachments are added or removed. 4919 * 4920 * @since 5.8.0 4921 * 4922 * @return {void} 4923 */ 4924 updateLoadMoreView: _.debounce( function() { 4925 // Ensure the load more view elements are initially hidden at each update. 4926 this.loadMoreButton.$el.addClass( 'hidden' ); 4927 this.loadMoreCount.$el.addClass( 'hidden' ); 4928 this.loadMoreJumpToFirst.$el.addClass( 'hidden' ).prop( 'disabled', true ); 4929 4930 if ( ! this.collection.getTotalAttachments() ) { 4931 return; 4932 } 4933 4934 if ( this.collection.length ) { 4935 this.loadMoreCount.$el.text( 4936 /* translators: 1: Number of displayed attachments, 2: Number of total attachments. */ 4937 sprintf( 4938 __( 'Showing %1$s of %2$s media items' ), 4939 this.collection.length, 4940 this.collection.getTotalAttachments() 4941 ) 4942 ); 4943 4944 this.loadMoreCount.$el.removeClass( 'hidden' ); 4945 } 4946 4947 /* 4948 * Notice that while the collection updates multiple times hasMore() may 4949 * return true when it's actually not true. 4950 */ 4951 if ( this.collection.hasMore() ) { 4952 this.loadMoreButton.$el.removeClass( 'hidden' ); 4953 } 4954 4955 // Find the media item to move focus to. The jQuery `eq()` index is zero-based. 4956 this.firstAddedMediaItem = this.$el.find( '.attachment' ).eq( this.firstAddedMediaItemIndex ); 4957 4958 // If there's a media item to move focus to, make the "Jump to" button available. 4959 if ( this.firstAddedMediaItem.length ) { 4960 this.firstAddedMediaItem.addClass( 'new-media' ); 4961 this.loadMoreJumpToFirst.$el.removeClass( 'hidden' ).prop( 'disabled', false ); 4962 } 4963 4964 // If there are new items added, but no more to be added, move focus to Jump button. 4965 if ( this.firstAddedMediaItem.length && ! this.collection.hasMore() ) { 4966 this.loadMoreJumpToFirst.$el.trigger( 'focus' ); 4967 } 4968 }, 10 ), 4969 4970 /** 4971 * Loads more attachments. 4972 * 4973 * @since 5.8.0 4974 * 4975 * @return {void} 4976 */ 4977 loadMoreAttachments: function() { 4978 var view = this; 4979 4980 if ( ! this.collection.hasMore() ) { 4981 return; 4982 } 4983 4984 /* 4985 * The collection index is zero-based while the length counts the actual 4986 * amount of items. Thus the length is equivalent to the position of the 4987 * first added item. 4988 */ 4989 this.firstAddedMediaItemIndex = this.collection.length; 4990 4991 this.$el.addClass( 'more-loaded' ); 4992 this.collection.each( function( attachment ) { 4993 var attach_id = attachment.attributes.id; 4994 $( '[data-id="' + attach_id + '"]' ).addClass( 'found-media' ); 4995 }); 4996 4997 view.loadMoreSpinner.show(); 4998 this.collection.once( 'attachments:received', function() { 4999 view.loadMoreSpinner.hide(); 5000 } ); 5001 this.collection.more(); 5002 }, 5003 5004 /** 5005 * Moves focus to the first new added item. . 5006 * 5007 * @since 5.8.0 5008 * 5009 * @return {void} 5010 */ 5011 jumpToFirstAddedItem: function() { 5012 // Set focus on first added item. 5013 this.firstAddedMediaItem.focus(); 5014 }, 5015 5016 createAttachmentsHeading: function() { 5017 this.attachmentsHeading = new wp.media.view.Heading( { 5018 text: l10n.attachmentsList, 5019 level: 'h2', 5020 className: 'media-views-heading screen-reader-text' 5021 } ); 5022 this.views.add( this.attachmentsHeading ); 5023 }, 5024 5025 createSidebar: function() { 5026 var options = this.options, 5027 selection = options.selection, 5028 sidebar = this.sidebar = new wp.media.view.Sidebar({ 5029 controller: this.controller 5030 }); 5031 5032 this.views.add( sidebar ); 5033 5034 if ( this.controller.uploader ) { 5035 sidebar.set( 'uploads', new wp.media.view.UploaderStatus({ 5036 controller: this.controller, 5037 priority: 40 5038 }) ); 5039 } 5040 5041 selection.on( 'selection:single', this.createSingle, this ); 5042 selection.on( 'selection:unsingle', this.disposeSingle, this ); 5043 5044 if ( selection.single() ) { 5045 this.createSingle(); 5046 } 5047 }, 5048 5049 createSingle: function() { 5050 var sidebar = this.sidebar, 5051 single = this.options.selection.single(); 5052 5053 sidebar.set( 'details', new wp.media.view.Attachment.Details({ 5054 controller: this.controller, 5055 model: single, 5056 priority: 80 5057 }) ); 5058 5059 sidebar.set( 'compat', new wp.media.view.AttachmentCompat({ 5060 controller: this.controller, 5061 model: single, 5062 priority: 120 5063 }) ); 5064 5065 if ( this.options.display ) { 5066 sidebar.set( 'display', new wp.media.view.Settings.AttachmentDisplay({ 5067 controller: this.controller, 5068 model: this.model.display( single ), 5069 attachment: single, 5070 priority: 160, 5071 userSettings: this.model.get('displayUserSettings') 5072 }) ); 5073 } 5074 5075 // Show the sidebar on mobile. 5076 if ( this.model.id === 'insert' ) { 5077 sidebar.$el.addClass( 'visible' ); 5078 } 5079 }, 5080 5081 disposeSingle: function() { 5082 var sidebar = this.sidebar; 5083 sidebar.unset('details'); 5084 sidebar.unset('compat'); 5085 sidebar.unset('display'); 5086 // Hide the sidebar on mobile. 5087 sidebar.$el.removeClass( 'visible' ); 5088 } 5089 }); 5090 5091 module.exports = AttachmentsBrowser; 5092 5093 5094 /***/ }), 5095 5096 /***/ 3479: 5097 /***/ ((module) => { 5098 5099 var Attachments = wp.media.view.Attachments, 5100 Selection; 5101 5102 /** 5103 * wp.media.view.Attachments.Selection 5104 * 5105 * @memberOf wp.media.view.Attachments 5106 * 5107 * @class 5108 * @augments wp.media.view.Attachments 5109 * @augments wp.media.View 5110 * @augments wp.Backbone.View 5111 * @augments Backbone.View 5112 */ 5113 Selection = Attachments.extend(/** @lends wp.media.view.Attachments.Selection.prototype */{ 5114 events: {}, 5115 initialize: function() { 5116 _.defaults( this.options, { 5117 sortable: false, 5118 resize: false, 5119 5120 // The single `Attachment` view to be used in the `Attachments` view. 5121 AttachmentView: wp.media.view.Attachment.Selection 5122 }); 5123 // Call 'initialize' directly on the parent class. 5124 return Attachments.prototype.initialize.apply( this, arguments ); 5125 } 5126 }); 5127 5128 module.exports = Selection; 5129 5130 5131 /***/ }), 5132 5133 /***/ 168: 5134 /***/ ((module) => { 5135 5136 var $ = Backbone.$, 5137 ButtonGroup; 5138 5139 /** 5140 * wp.media.view.ButtonGroup 5141 * 5142 * @memberOf wp.media.view 5143 * 5144 * @class 5145 * @augments wp.media.View 5146 * @augments wp.Backbone.View 5147 * @augments Backbone.View 5148 */ 5149 ButtonGroup = wp.media.View.extend(/** @lends wp.media.view.ButtonGroup.prototype */{ 5150 tagName: 'div', 5151 className: 'button-group button-large media-button-group', 5152 5153 initialize: function() { 5154 /** 5155 * @member {wp.media.view.Button[]} 5156 */ 5157 this.buttons = _.map( this.options.buttons || [], function( button ) { 5158 if ( button instanceof Backbone.View ) { 5159 return button; 5160 } else { 5161 return new wp.media.view.Button( button ).render(); 5162 } 5163 }); 5164 5165 delete this.options.buttons; 5166 5167 if ( this.options.classes ) { 5168 this.$el.addClass( this.options.classes ); 5169 } 5170 }, 5171 5172 /** 5173 * @return {wp.media.view.ButtonGroup} 5174 */ 5175 render: function() { 5176 this.$el.html( $( _.pluck( this.buttons, 'el' ) ).detach() ); 5177 return this; 5178 } 5179 }); 5180 5181 module.exports = ButtonGroup; 5182 5183 5184 /***/ }), 5185 5186 /***/ 846: 5187 /***/ ((module) => { 5188 5189 /** 5190 * wp.media.view.Button 5191 * 5192 * @memberOf wp.media.view 5193 * 5194 * @class 5195 * @augments wp.media.View 5196 * @augments wp.Backbone.View 5197 * @augments Backbone.View 5198 */ 5199 var Button = wp.media.View.extend(/** @lends wp.media.view.Button.prototype */{ 5200 tagName: 'button', 5201 className: 'media-button', 5202 attributes: { type: 'button' }, 5203 5204 events: { 5205 'click': 'click' 5206 }, 5207 5208 defaults: { 5209 text: '', 5210 style: '', 5211 size: 'large', 5212 disabled: false 5213 }, 5214 5215 initialize: function() { 5216 /** 5217 * Create a model with the provided `defaults`. 5218 * 5219 * @member {Backbone.Model} 5220 */ 5221 this.model = new Backbone.Model( this.defaults ); 5222 5223 // If any of the `options` have a key from `defaults`, apply its 5224 // value to the `model` and remove it from the `options object. 5225 _.each( this.defaults, function( def, key ) { 5226 var value = this.options[ key ]; 5227 if ( _.isUndefined( value ) ) { 5228 return; 5229 } 5230 5231 this.model.set( key, value ); 5232 delete this.options[ key ]; 5233 }, this ); 5234 5235 this.listenTo( this.model, 'change', this.render ); 5236 }, 5237 /** 5238 * @return {wp.media.view.Button} Returns itself to allow chaining. 5239 */ 5240 render: function() { 5241 var classes = [ 'button', this.className ], 5242 model = this.model.toJSON(); 5243 5244 if ( model.style ) { 5245 classes.push( 'button-' + model.style ); 5246 } 5247 5248 if ( model.size ) { 5249 classes.push( 'button-' + model.size ); 5250 } 5251 5252 classes = _.uniq( classes.concat( this.options.classes ) ); 5253 this.el.className = classes.join(' '); 5254 5255 this.$el.attr( 'disabled', model.disabled ); 5256 this.$el.text( this.model.get('text') ); 5257 5258 return this; 5259 }, 5260 /** 5261 * @param {Object} event 5262 */ 5263 click: function( event ) { 5264 if ( '#' === this.attributes.href ) { 5265 event.preventDefault(); 5266 } 5267 5268 if ( this.options.click && ! this.model.get('disabled') ) { 5269 this.options.click.apply( this, arguments ); 5270 } 5271 } 5272 }); 5273 5274 module.exports = Button; 5275 5276 5277 /***/ }), 5278 5279 /***/ 7637: 5280 /***/ ((module) => { 5281 5282 var View = wp.media.View, 5283 UploaderStatus = wp.media.view.UploaderStatus, 5284 l10n = wp.media.view.l10n, 5285 $ = jQuery, 5286 Cropper; 5287 5288 /** 5289 * wp.media.view.Cropper 5290 * 5291 * Uses the imgAreaSelect plugin to allow a user to crop an image. 5292 * 5293 * Takes imgAreaSelect options from 5294 * wp.customize.HeaderControl.calculateImageSelectOptions via 5295 * wp.customize.HeaderControl.openMM. 5296 * 5297 * @memberOf wp.media.view 5298 * 5299 * @class 5300 * @augments wp.media.View 5301 * @augments wp.Backbone.View 5302 * @augments Backbone.View 5303 */ 5304 Cropper = View.extend(/** @lends wp.media.view.Cropper.prototype */{ 5305 className: 'crop-content', 5306 template: wp.template('crop-content'), 5307 initialize: function() { 5308 _.bindAll(this, 'onImageLoad'); 5309 }, 5310 ready: function() { 5311 this.controller.frame.on('content:error:crop', this.onError, this); 5312 this.$image = this.$el.find('.crop-image'); 5313 this.$image.on('load', this.onImageLoad); 5314 $(window).on('resize.cropper', _.debounce(this.onImageLoad, 250)); 5315 }, 5316 remove: function() { 5317 $(window).off('resize.cropper'); 5318 this.$el.remove(); 5319 this.$el.off(); 5320 View.prototype.remove.apply(this, arguments); 5321 }, 5322 prepare: function() { 5323 return { 5324 title: l10n.cropYourImage, 5325 url: this.options.attachment.get('url') 5326 }; 5327 }, 5328 onImageLoad: function() { 5329 var imgOptions = this.controller.get('imgSelectOptions'), 5330 imgSelect; 5331 5332 if (typeof imgOptions === 'function') { 5333 imgOptions = imgOptions(this.options.attachment, this.controller); 5334 } 5335 5336 imgOptions = _.extend(imgOptions, { 5337 parent: this.$el, 5338 onInit: function() { 5339 5340 // Store the set ratio. 5341 var setRatio = imgSelect.getOptions().aspectRatio; 5342 5343 // On mousedown, if no ratio is set and the Shift key is down, use a 1:1 ratio. 5344 this.parent.children().on( 'mousedown touchstart', function( e ) { 5345 5346 // If no ratio is set and the shift key is down, use a 1:1 ratio. 5347 if ( ! setRatio && e.shiftKey ) { 5348 imgSelect.setOptions( { 5349 aspectRatio: '1:1' 5350 } ); 5351 } 5352 } ); 5353 5354 this.parent.children().on( 'mouseup touchend', function() { 5355 5356 // Restore the set ratio. 5357 imgSelect.setOptions( { 5358 aspectRatio: setRatio ? setRatio : false 5359 } ); 5360 } ); 5361 } 5362 } ); 5363 this.trigger('image-loaded'); 5364 imgSelect = this.controller.imgSelect = this.$image.imgAreaSelect(imgOptions); 5365 }, 5366 onError: function() { 5367 var filename = this.options.attachment.get('filename'); 5368 5369 this.views.add( '.upload-errors', new wp.media.view.UploaderStatusError({ 5370 filename: UploaderStatus.prototype.filename(filename), 5371 message: window._wpMediaViewsL10n.cropError 5372 }), { at: 0 }); 5373 } 5374 }); 5375 5376 module.exports = Cropper; 5377 5378 5379 /***/ }), 5380 5381 /***/ 6126: 5382 /***/ ((module) => { 5383 5384 var View = wp.media.View, 5385 EditImage; 5386 5387 /** 5388 * wp.media.view.EditImage 5389 * 5390 * @memberOf wp.media.view 5391 * 5392 * @class 5393 * @augments wp.media.View 5394 * @augments wp.Backbone.View 5395 * @augments Backbone.View 5396 */ 5397 EditImage = View.extend(/** @lends wp.media.view.EditImage.prototype */{ 5398 className: 'image-editor', 5399 template: wp.template('image-editor'), 5400 5401 initialize: function( options ) { 5402 this.editor = window.imageEdit; 5403 this.controller = options.controller; 5404 View.prototype.initialize.apply( this, arguments ); 5405 }, 5406 5407 prepare: function() { 5408 return this.model.toJSON(); 5409 }, 5410 5411 loadEditor: function() { 5412 this.editor.open( this.model.get( 'id' ), this.model.get( 'nonces' ).edit, this ); 5413 }, 5414 5415 back: function() { 5416 var lastState = this.controller.lastState(); 5417 this.controller.setState( lastState ); 5418 }, 5419 5420 refresh: function() { 5421 this.model.fetch(); 5422 }, 5423 5424 save: function() { 5425 var lastState = this.controller.lastState(); 5426 5427 this.model.fetch().done( _.bind( function() { 5428 this.controller.setState( lastState ); 5429 }, this ) ); 5430 } 5431 5432 }); 5433 5434 module.exports = EditImage; 5435 5436 5437 /***/ }), 5438 5439 /***/ 5741: 5440 /***/ ((module) => { 5441 5442 /** 5443 * wp.media.view.Embed 5444 * 5445 * @memberOf wp.media.view 5446 * 5447 * @class 5448 * @augments wp.media.View 5449 * @augments wp.Backbone.View 5450 * @augments Backbone.View 5451 */ 5452 var Embed = wp.media.View.extend(/** @lends wp.media.view.Ember.prototype */{ 5453 className: 'media-embed', 5454 5455 initialize: function() { 5456 /** 5457 * @member {wp.media.view.EmbedUrl} 5458 */ 5459 this.url = new wp.media.view.EmbedUrl({ 5460 controller: this.controller, 5461 model: this.model.props 5462 }).render(); 5463 5464 this.views.set([ this.url ]); 5465 this.refresh(); 5466 this.listenTo( this.model, 'change:type', this.refresh ); 5467 this.listenTo( this.model, 'change:loading', this.loading ); 5468 }, 5469 5470 /** 5471 * @param {Object} view 5472 */ 5473 settings: function( view ) { 5474 if ( this._settings ) { 5475 this._settings.remove(); 5476 } 5477 this._settings = view; 5478 this.views.add( view ); 5479 }, 5480 5481 refresh: function() { 5482 var type = this.model.get('type'), 5483 constructor; 5484 5485 if ( 'image' === type ) { 5486 constructor = wp.media.view.EmbedImage; 5487 } else if ( 'link' === type ) { 5488 constructor = wp.media.view.EmbedLink; 5489 } else { 5490 return; 5491 } 5492 5493 this.settings( new constructor({ 5494 controller: this.controller, 5495 model: this.model.props, 5496 priority: 40 5497 }) ); 5498 }, 5499 5500 loading: function() { 5501 this.$el.toggleClass( 'embed-loading', this.model.get('loading') ); 5502 } 5503 }); 5504 5505 module.exports = Embed; 5506 5507 5508 /***/ }), 5509 5510 /***/ 2395: 5511 /***/ ((module) => { 5512 5513 var AttachmentDisplay = wp.media.view.Settings.AttachmentDisplay, 5514 EmbedImage; 5515 5516 /** 5517 * wp.media.view.EmbedImage 5518 * 5519 * @memberOf wp.media.view 5520 * 5521 * @class 5522 * @augments wp.media.view.Settings.AttachmentDisplay 5523 * @augments wp.media.view.Settings 5524 * @augments wp.media.View 5525 * @augments wp.Backbone.View 5526 * @augments Backbone.View 5527 */ 5528 EmbedImage = AttachmentDisplay.extend(/** @lends wp.media.view.EmbedImage.prototype */{ 5529 className: 'embed-media-settings', 5530 template: wp.template('embed-image-settings'), 5531 5532 initialize: function() { 5533 /** 5534 * Call `initialize` directly on parent class with passed arguments 5535 */ 5536 AttachmentDisplay.prototype.initialize.apply( this, arguments ); 5537 this.listenTo( this.model, 'change:url', this.updateImage ); 5538 }, 5539 5540 updateImage: function() { 5541 this.$('img').attr( 'src', this.model.get('url') ); 5542 } 5543 }); 5544 5545 module.exports = EmbedImage; 5546 5547 5548 /***/ }), 5549 5550 /***/ 8232: 5551 /***/ ((module) => { 5552 5553 var $ = jQuery, 5554 EmbedLink; 5555 5556 /** 5557 * wp.media.view.EmbedLink 5558 * 5559 * @memberOf wp.media.view 5560 * 5561 * @class 5562 * @augments wp.media.view.Settings 5563 * @augments wp.media.View 5564 * @augments wp.Backbone.View 5565 * @augments Backbone.View 5566 */ 5567 EmbedLink = wp.media.view.Settings.extend(/** @lends wp.media.view.EmbedLink.prototype */{ 5568 className: 'embed-link-settings', 5569 template: wp.template('embed-link-settings'), 5570 5571 initialize: function() { 5572 this.listenTo( this.model, 'change:url', this.updateoEmbed ); 5573 }, 5574 5575 updateoEmbed: _.debounce( function() { 5576 var url = this.model.get( 'url' ); 5577 5578 // Clear out previous results. 5579 this.$('.embed-container').hide().find('.embed-preview').empty(); 5580 this.$( '.setting' ).hide(); 5581 5582 // Only proceed with embed if the field contains more than 11 characters. 5583 // Example: http://a.io is 11 chars 5584 if ( url && ( url.length < 11 || ! url.match(/^http(s)?:\/\//) ) ) { 5585 return; 5586 } 5587 5588 this.fetch(); 5589 }, wp.media.controller.Embed.sensitivity ), 5590 5591 fetch: function() { 5592 var url = this.model.get( 'url' ), re, youTubeEmbedMatch; 5593 5594 // Check if they haven't typed in 500 ms. 5595 if ( $('#embed-url-field').val() !== url ) { 5596 return; 5597 } 5598 5599 if ( this.dfd && 'pending' === this.dfd.state() ) { 5600 this.dfd.abort(); 5601 } 5602 5603 // Support YouTube embed urls, since they work once in the editor. 5604 re = /https?:\/\/www\.youtube\.com\/embed\/([^/]+)/; 5605 youTubeEmbedMatch = re.exec( url ); 5606 if ( youTubeEmbedMatch ) { 5607 url = 'https://www.youtube.com/watch?v=' + youTubeEmbedMatch[ 1 ]; 5608 } 5609 5610 this.dfd = wp.apiRequest({ 5611 url: wp.media.view.settings.oEmbedProxyUrl, 5612 data: { 5613 url: url, 5614 maxwidth: this.model.get( 'width' ), 5615 maxheight: this.model.get( 'height' ) 5616 }, 5617 type: 'GET', 5618 dataType: 'json', 5619 context: this 5620 }) 5621 .done( function( response ) { 5622 this.renderoEmbed( { 5623 data: { 5624 body: response.html || '' 5625 } 5626 } ); 5627 } ) 5628 .fail( this.renderFail ); 5629 }, 5630 5631 renderFail: function ( response, status ) { 5632 if ( 'abort' === status ) { 5633 return; 5634 } 5635 this.$( '.link-text' ).show(); 5636 }, 5637 5638 renderoEmbed: function( response ) { 5639 var html = ( response && response.data && response.data.body ) || ''; 5640 5641 if ( html ) { 5642 this.$('.embed-container').show().find('.embed-preview').html( html ); 5643 } else { 5644 this.renderFail(); 5645 } 5646 } 5647 }); 5648 5649 module.exports = EmbedLink; 5650 5651 5652 /***/ }), 5653 5654 /***/ 7327: 5655 /***/ ((module) => { 5656 5657 var View = wp.media.View, 5658 $ = jQuery, 5659 l10n = wp.media.view.l10n, 5660 EmbedUrl; 5661 5662 /** 5663 * wp.media.view.EmbedUrl 5664 * 5665 * @memberOf wp.media.view 5666 * 5667 * @class 5668 * @augments wp.media.View 5669 * @augments wp.Backbone.View 5670 * @augments Backbone.View 5671 */ 5672 EmbedUrl = View.extend(/** @lends wp.media.view.EmbedUrl.prototype */{ 5673 tagName: 'span', 5674 className: 'embed-url', 5675 5676 events: { 5677 'input': 'url' 5678 }, 5679 5680 initialize: function() { 5681 this.$input = $( '<input id="embed-url-field" type="url" />' ) 5682 .attr( 'aria-label', l10n.insertFromUrlTitle ) 5683 .val( this.model.get('url') ); 5684 this.input = this.$input[0]; 5685 5686 this.spinner = $('<span class="spinner" />')[0]; 5687 this.$el.append([ this.input, this.spinner ]); 5688 5689 this.listenTo( this.model, 'change:url', this.render ); 5690 5691 if ( this.model.get( 'url' ) ) { 5692 _.delay( _.bind( function () { 5693 this.model.trigger( 'change:url' ); 5694 }, this ), 500 ); 5695 } 5696 }, 5697 /** 5698 * @return {wp.media.view.EmbedUrl} Returns itself to allow chaining. 5699 */ 5700 render: function() { 5701 var $input = this.$input; 5702 5703 if ( $input.is(':focus') ) { 5704 return; 5705 } 5706 5707 if ( this.model.get( 'url' ) ) { 5708 this.input.value = this.model.get('url'); 5709 } else { 5710 this.input.setAttribute( 'placeholder', 'https://' ); 5711 } 5712 5713 /** 5714 * Call `render` directly on parent class with passed arguments 5715 */ 5716 View.prototype.render.apply( this, arguments ); 5717 return this; 5718 }, 5719 5720 url: function( event ) { 5721 var url = event.target.value || ''; 5722 this.model.set( 'url', url.trim() ); 5723 } 5724 }); 5725 5726 module.exports = EmbedUrl; 5727 5728 5729 /***/ }), 5730 5731 /***/ 718: 5732 /***/ ((module) => { 5733 5734 var $ = jQuery; 5735 5736 /** 5737 * wp.media.view.FocusManager 5738 * 5739 * @memberOf wp.media.view 5740 * 5741 * @class 5742 * @augments wp.media.View 5743 * @augments wp.Backbone.View 5744 * @augments Backbone.View 5745 */ 5746 var FocusManager = wp.media.View.extend(/** @lends wp.media.view.FocusManager.prototype */{ 5747 5748 events: { 5749 'keydown': 'focusManagementMode' 5750 }, 5751 5752 /** 5753 * Initializes the Focus Manager. 5754 * 5755 * @param {Object} options The Focus Manager options. 5756 * 5757 * @since 5.3.0 5758 * 5759 * @return {void} 5760 */ 5761 initialize: function( options ) { 5762 this.mode = options.mode || 'constrainTabbing'; 5763 this.tabsAutomaticActivation = options.tabsAutomaticActivation || false; 5764 }, 5765 5766 /** 5767 * Determines which focus management mode to use. 5768 * 5769 * @since 5.3.0 5770 * 5771 * @param {Object} event jQuery event object. 5772 * 5773 * @return {void} 5774 */ 5775 focusManagementMode: function( event ) { 5776 if ( this.mode === 'constrainTabbing' ) { 5777 this.constrainTabbing( event ); 5778 } 5779 5780 if ( this.mode === 'tabsNavigation' ) { 5781 this.tabsNavigation( event ); 5782 } 5783 }, 5784 5785 /** 5786 * Gets all the tabbable elements. 5787 * 5788 * @since 5.3.0 5789 * 5790 * @return {Object} A jQuery collection of tabbable elements. 5791 */ 5792 getTabbables: function() { 5793 // Skip the file input added by Plupload. 5794 return this.$( ':tabbable' ).not( '.moxie-shim input[type="file"]' ); 5795 }, 5796 5797 /** 5798 * Moves focus to the modal dialog. 5799 * 5800 * @since 3.5.0 5801 * 5802 * @return {void} 5803 */ 5804 focus: function() { 5805 this.$( '.media-modal' ).trigger( 'focus' ); 5806 }, 5807 5808 /** 5809 * Constrains navigation with the Tab key within the media view element. 5810 * 5811 * @since 4.0.0 5812 * 5813 * @param {Object} event A keydown jQuery event. 5814 * 5815 * @return {void} 5816 */ 5817 constrainTabbing: function( event ) { 5818 var tabbables; 5819 5820 // Look for the tab key. 5821 if ( 9 !== event.keyCode ) { 5822 return; 5823 } 5824 5825 tabbables = this.getTabbables(); 5826 5827 // Keep tab focus within media modal while it's open. 5828 if ( tabbables.last()[0] === event.target && ! event.shiftKey ) { 5829 tabbables.first().focus(); 5830 return false; 5831 } else if ( tabbables.first()[0] === event.target && event.shiftKey ) { 5832 tabbables.last().focus(); 5833 return false; 5834 } 5835 }, 5836 5837 /** 5838 * Hides from assistive technologies all the body children. 5839 * 5840 * Sets an `aria-hidden="true"` attribute on all the body children except 5841 * the provided element and other elements that should not be hidden. 5842 * 5843 * The reason why we use `aria-hidden` is that `aria-modal="true"` is buggy 5844 * in Safari 11.1 and support is spotty in other browsers. Also, `aria-modal="true"` 5845 * prevents the `wp.a11y.speak()` ARIA live regions to work as they're outside 5846 * of the modal dialog and get hidden from assistive technologies. 5847 * 5848 * @since 5.2.3 5849 * 5850 * @param {Object} visibleElement The jQuery object representing the element that should not be hidden. 5851 * 5852 * @return {void} 5853 */ 5854 setAriaHiddenOnBodyChildren: function( visibleElement ) { 5855 var bodyChildren, 5856 self = this; 5857 5858 if ( this.isBodyAriaHidden ) { 5859 return; 5860 } 5861 5862 // Get all the body children. 5863 bodyChildren = document.body.children; 5864 5865 // Loop through the body children and hide the ones that should be hidden. 5866 _.each( bodyChildren, function( element ) { 5867 // Don't hide the modal element. 5868 if ( element === visibleElement[0] ) { 5869 return; 5870 } 5871 5872 // Determine the body children to hide. 5873 if ( self.elementShouldBeHidden( element ) ) { 5874 element.setAttribute( 'aria-hidden', 'true' ); 5875 // Store the hidden elements. 5876 self.ariaHiddenElements.push( element ); 5877 } 5878 } ); 5879 5880 this.isBodyAriaHidden = true; 5881 }, 5882 5883 /** 5884 * Unhides from assistive technologies all the body children. 5885 * 5886 * Makes visible again to assistive technologies all the body children 5887 * previously hidden and stored in this.ariaHiddenElements. 5888 * 5889 * @since 5.2.3 5890 * 5891 * @return {void} 5892 */ 5893 removeAriaHiddenFromBodyChildren: function() { 5894 _.each( this.ariaHiddenElements, function( element ) { 5895 element.removeAttribute( 'aria-hidden' ); 5896 } ); 5897 5898 this.ariaHiddenElements = []; 5899 this.isBodyAriaHidden = false; 5900 }, 5901 5902 /** 5903 * Determines if the passed element should not be hidden from assistive technologies. 5904 * 5905 * @since 5.2.3 5906 * 5907 * @param {Object} element The DOM element that should be checked. 5908 * 5909 * @return {boolean} Whether the element should not be hidden from assistive technologies. 5910 */ 5911 elementShouldBeHidden: function( element ) { 5912 var role = element.getAttribute( 'role' ), 5913 liveRegionsRoles = [ 'alert', 'status', 'log', 'marquee', 'timer' ]; 5914 5915 /* 5916 * Don't hide scripts, elements that already have `aria-hidden`, and 5917 * ARIA live regions. 5918 */ 5919 return ! ( 5920 element.tagName === 'SCRIPT' || 5921 element.hasAttribute( 'aria-hidden' ) || 5922 element.hasAttribute( 'aria-live' ) || 5923 liveRegionsRoles.indexOf( role ) !== -1 5924 ); 5925 }, 5926 5927 /** 5928 * Whether the body children are hidden from assistive technologies. 5929 * 5930 * @since 5.2.3 5931 */ 5932 isBodyAriaHidden: false, 5933 5934 /** 5935 * Stores an array of DOM elements that should be hidden from assistive 5936 * technologies, for example when the media modal dialog opens. 5937 * 5938 * @since 5.2.3 5939 */ 5940 ariaHiddenElements: [], 5941 5942 /** 5943 * Holds the jQuery collection of ARIA tabs. 5944 * 5945 * @since 5.3.0 5946 */ 5947 tabs: $(), 5948 5949 /** 5950 * Sets up tabs in an ARIA tabbed interface. 5951 * 5952 * @since 5.3.0 5953 * 5954 * @param {Object} event jQuery event object. 5955 * 5956 * @return {void} 5957 */ 5958 setupAriaTabs: function() { 5959 this.tabs = this.$( '[role="tab"]' ); 5960 5961 // Set up initial attributes. 5962 this.tabs.attr( { 5963 'aria-selected': 'false', 5964 tabIndex: '-1' 5965 } ); 5966 5967 // Set up attributes on the initially active tab. 5968 this.tabs.filter( '.active' ) 5969 .removeAttr( 'tabindex' ) 5970 .attr( 'aria-selected', 'true' ); 5971 }, 5972 5973 /** 5974 * Enables arrows navigation within the ARIA tabbed interface. 5975 * 5976 * @since 5.3.0 5977 * 5978 * @param {Object} event jQuery event object. 5979 * 5980 * @return {void} 5981 */ 5982 tabsNavigation: function( event ) { 5983 var orientation = 'horizontal', 5984 keys = [ 32, 35, 36, 37, 38, 39, 40 ]; 5985 5986 // Return if not Spacebar, End, Home, or Arrow keys. 5987 if ( keys.indexOf( event.which ) === -1 ) { 5988 return; 5989 } 5990 5991 // Determine navigation direction. 5992 if ( this.$el.attr( 'aria-orientation' ) === 'vertical' ) { 5993 orientation = 'vertical'; 5994 } 5995 5996 // Make Up and Down arrow keys do nothing with horizontal tabs. 5997 if ( orientation === 'horizontal' && [ 38, 40 ].indexOf( event.which ) !== -1 ) { 5998 return; 5999 } 6000 6001 // Make Left and Right arrow keys do nothing with vertical tabs. 6002 if ( orientation === 'vertical' && [ 37, 39 ].indexOf( event.which ) !== -1 ) { 6003 return; 6004 } 6005 6006 this.switchTabs( event, this.tabs ); 6007 }, 6008 6009 /** 6010 * Switches tabs in the ARIA tabbed interface. 6011 * 6012 * @since 5.3.0 6013 * 6014 * @param {Object} event jQuery event object. 6015 * 6016 * @return {void} 6017 */ 6018 switchTabs: function( event ) { 6019 var key = event.which, 6020 index = this.tabs.index( $( event.target ) ), 6021 newIndex; 6022 6023 switch ( key ) { 6024 // Space bar: Activate current targeted tab. 6025 case 32: { 6026 this.activateTab( this.tabs[ index ] ); 6027 break; 6028 } 6029 // End key: Activate last tab. 6030 case 35: { 6031 event.preventDefault(); 6032 this.activateTab( this.tabs[ this.tabs.length - 1 ] ); 6033 break; 6034 } 6035 // Home key: Activate first tab. 6036 case 36: { 6037 event.preventDefault(); 6038 this.activateTab( this.tabs[ 0 ] ); 6039 break; 6040 } 6041 // Left and up keys: Activate previous tab. 6042 case 37: 6043 case 38: { 6044 event.preventDefault(); 6045 newIndex = ( index - 1 ) < 0 ? this.tabs.length - 1 : index - 1; 6046 this.activateTab( this.tabs[ newIndex ] ); 6047 break; 6048 } 6049 // Right and down keys: Activate next tab. 6050 case 39: 6051 case 40: { 6052 event.preventDefault(); 6053 newIndex = ( index + 1 ) === this.tabs.length ? 0 : index + 1; 6054 this.activateTab( this.tabs[ newIndex ] ); 6055 break; 6056 } 6057 } 6058 }, 6059 6060 /** 6061 * Sets a single tab to be focusable and semantically selected. 6062 * 6063 * @since 5.3.0 6064 * 6065 * @param {Object} tab The tab DOM element. 6066 * 6067 * @return {void} 6068 */ 6069 activateTab: function( tab ) { 6070 if ( ! tab ) { 6071 return; 6072 } 6073 6074 // The tab is a DOM element: no need for jQuery methods. 6075 tab.focus(); 6076 6077 // Handle automatic activation. 6078 if ( this.tabsAutomaticActivation ) { 6079 tab.removeAttribute( 'tabindex' ); 6080 tab.setAttribute( 'aria-selected', 'true' ); 6081 tab.click(); 6082 6083 return; 6084 } 6085 6086 // Handle manual activation. 6087 $( tab ).on( 'click', function() { 6088 tab.removeAttribute( 'tabindex' ); 6089 tab.setAttribute( 'aria-selected', 'true' ); 6090 } ); 6091 } 6092 }); 6093 6094 module.exports = FocusManager; 6095 6096 6097 /***/ }), 6098 6099 /***/ 1061: 6100 /***/ ((module) => { 6101 6102 /** 6103 * wp.media.view.Frame 6104 * 6105 * A frame is a composite view consisting of one or more regions and one or more 6106 * states. 6107 * 6108 * @memberOf wp.media.view 6109 * 6110 * @see wp.media.controller.State 6111 * @see wp.media.controller.Region 6112 * 6113 * @class 6114 * @augments wp.media.View 6115 * @augments wp.Backbone.View 6116 * @augments Backbone.View 6117 * @mixes wp.media.controller.StateMachine 6118 */ 6119 var Frame = wp.media.View.extend(/** @lends wp.media.view.Frame.prototype */{ 6120 initialize: function() { 6121 _.defaults( this.options, { 6122 mode: [ 'select' ] 6123 }); 6124 this._createRegions(); 6125 this._createStates(); 6126 this._createModes(); 6127 }, 6128 6129 _createRegions: function() { 6130 // Clone the regions array. 6131 this.regions = this.regions ? this.regions.slice() : []; 6132 6133 // Initialize regions. 6134 _.each( this.regions, function( region ) { 6135 this[ region ] = new wp.media.controller.Region({ 6136 view: this, 6137 id: region, 6138 selector: '.media-frame-' + region 6139 }); 6140 }, this ); 6141 }, 6142 /** 6143 * Create the frame's states. 6144 * 6145 * @see wp.media.controller.State 6146 * @see wp.media.controller.StateMachine 6147 * 6148 * @fires wp.media.controller.State#ready 6149 */ 6150 _createStates: function() { 6151 // Create the default `states` collection. 6152 this.states = new Backbone.Collection( null, { 6153 model: wp.media.controller.State 6154 }); 6155 6156 // Ensure states have a reference to the frame. 6157 this.states.on( 'add', function( model ) { 6158 model.frame = this; 6159 model.trigger('ready'); 6160 }, this ); 6161 6162 if ( this.options.states ) { 6163 this.states.add( this.options.states ); 6164 } 6165 }, 6166 6167 /** 6168 * A frame can be in a mode or multiple modes at one time. 6169 * 6170 * For example, the manage media frame can be in the `Bulk Select` or `Edit` mode. 6171 */ 6172 _createModes: function() { 6173 // Store active "modes" that the frame is in. Unrelated to region modes. 6174 this.activeModes = new Backbone.Collection(); 6175 this.activeModes.on( 'add remove reset', _.bind( this.triggerModeEvents, this ) ); 6176 6177 _.each( this.options.mode, function( mode ) { 6178 this.activateMode( mode ); 6179 }, this ); 6180 }, 6181 /** 6182 * Reset all states on the frame to their defaults. 6183 * 6184 * @return {wp.media.view.Frame} Returns itself to allow chaining. 6185 */ 6186 reset: function() { 6187 this.states.invoke( 'trigger', 'reset' ); 6188 return this; 6189 }, 6190 /** 6191 * Map activeMode collection events to the frame. 6192 */ 6193 triggerModeEvents: function( model, collection, options ) { 6194 var collectionEvent, 6195 modeEventMap = { 6196 add: 'activate', 6197 remove: 'deactivate' 6198 }, 6199 eventToTrigger; 6200 // Probably a better way to do this. 6201 _.each( options, function( value, key ) { 6202 if ( value ) { 6203 collectionEvent = key; 6204 } 6205 } ); 6206 6207 if ( ! _.has( modeEventMap, collectionEvent ) ) { 6208 return; 6209 } 6210 6211 eventToTrigger = model.get('id') + ':' + modeEventMap[collectionEvent]; 6212 this.trigger( eventToTrigger ); 6213 }, 6214 /** 6215 * Activate a mode on the frame. 6216 * 6217 * @param string mode Mode ID. 6218 * @return {this} Returns itself to allow chaining. 6219 */ 6220 activateMode: function( mode ) { 6221 // Bail if the mode is already active. 6222 if ( this.isModeActive( mode ) ) { 6223 return; 6224 } 6225 this.activeModes.add( [ { id: mode } ] ); 6226 // Add a CSS class to the frame so elements can be styled for the mode. 6227 this.$el.addClass( 'mode-' + mode ); 6228 6229 return this; 6230 }, 6231 /** 6232 * Deactivate a mode on the frame. 6233 * 6234 * @param string mode Mode ID. 6235 * @return {this} Returns itself to allow chaining. 6236 */ 6237 deactivateMode: function( mode ) { 6238 // Bail if the mode isn't active. 6239 if ( ! this.isModeActive( mode ) ) { 6240 return this; 6241 } 6242 this.activeModes.remove( this.activeModes.where( { id: mode } ) ); 6243 this.$el.removeClass( 'mode-' + mode ); 6244 /** 6245 * Frame mode deactivation event. 6246 * 6247 * @event wp.media.view.Frame#{mode}:deactivate 6248 */ 6249 this.trigger( mode + ':deactivate' ); 6250 6251 return this; 6252 }, 6253 /** 6254 * Check if a mode is enabled on the frame. 6255 * 6256 * @param string mode Mode ID. 6257 * @return bool 6258 */ 6259 isModeActive: function( mode ) { 6260 return Boolean( this.activeModes.where( { id: mode } ).length ); 6261 } 6262 }); 6263 6264 // Make the `Frame` a `StateMachine`. 6265 _.extend( Frame.prototype, wp.media.controller.StateMachine.prototype ); 6266 6267 module.exports = Frame; 6268 6269 6270 /***/ }), 6271 6272 /***/ 5424: 6273 /***/ ((module) => { 6274 6275 var Select = wp.media.view.MediaFrame.Select, 6276 l10n = wp.media.view.l10n, 6277 ImageDetails; 6278 6279 /** 6280 * wp.media.view.MediaFrame.ImageDetails 6281 * 6282 * A media frame for manipulating an image that's already been inserted 6283 * into a post. 6284 * 6285 * @memberOf wp.media.view.MediaFrame 6286 * 6287 * @class 6288 * @augments wp.media.view.MediaFrame.Select 6289 * @augments wp.media.view.MediaFrame 6290 * @augments wp.media.view.Frame 6291 * @augments wp.media.View 6292 * @augments wp.Backbone.View 6293 * @augments Backbone.View 6294 * @mixes wp.media.controller.StateMachine 6295 */ 6296 ImageDetails = Select.extend(/** @lends wp.media.view.MediaFrame.ImageDetails.prototype */{ 6297 defaults: { 6298 id: 'image', 6299 url: '', 6300 menu: 'image-details', 6301 content: 'image-details', 6302 toolbar: 'image-details', 6303 type: 'link', 6304 title: l10n.imageDetailsTitle, 6305 priority: 120 6306 }, 6307 6308 initialize: function( options ) { 6309 this.image = new wp.media.model.PostImage( options.metadata ); 6310 this.options.selection = new wp.media.model.Selection( this.image.attachment, { multiple: false } ); 6311 Select.prototype.initialize.apply( this, arguments ); 6312 }, 6313 6314 bindHandlers: function() { 6315 Select.prototype.bindHandlers.apply( this, arguments ); 6316 this.on( 'menu:create:image-details', this.createMenu, this ); 6317 this.on( 'content:create:image-details', this.imageDetailsContent, this ); 6318 this.on( 'content:render:edit-image', this.editImageContent, this ); 6319 this.on( 'toolbar:render:image-details', this.renderImageDetailsToolbar, this ); 6320 // Override the select toolbar. 6321 this.on( 'toolbar:render:replace', this.renderReplaceImageToolbar, this ); 6322 }, 6323 6324 createStates: function() { 6325 this.states.add([ 6326 new wp.media.controller.ImageDetails({ 6327 image: this.image, 6328 editable: false 6329 }), 6330 new wp.media.controller.ReplaceImage({ 6331 id: 'replace-image', 6332 library: wp.media.query( { type: 'image' } ), 6333 image: this.image, 6334 multiple: false, 6335 title: l10n.imageReplaceTitle, 6336 toolbar: 'replace', 6337 priority: 80, 6338 displaySettings: true 6339 }), 6340 new wp.media.controller.EditImage( { 6341 image: this.image, 6342 selection: this.options.selection 6343 } ) 6344 ]); 6345 }, 6346 6347 imageDetailsContent: function( options ) { 6348 options.view = new wp.media.view.ImageDetails({ 6349 controller: this, 6350 model: this.state().image, 6351 attachment: this.state().image.attachment 6352 }); 6353 }, 6354 6355 editImageContent: function() { 6356 var state = this.state(), 6357 model = state.get('image'), 6358 view; 6359 6360 if ( ! model ) { 6361 return; 6362 } 6363 6364 view = new wp.media.view.EditImage( { model: model, controller: this } ).render(); 6365 6366 this.content.set( view ); 6367 6368 // After bringing in the frame, load the actual editor via an Ajax call. 6369 view.loadEditor(); 6370 6371 }, 6372 6373 renderImageDetailsToolbar: function() { 6374 this.toolbar.set( new wp.media.view.Toolbar({ 6375 controller: this, 6376 items: { 6377 select: { 6378 style: 'primary', 6379 text: l10n.update, 6380 priority: 80, 6381 6382 click: function() { 6383 var controller = this.controller, 6384 state = controller.state(); 6385 6386 controller.close(); 6387 6388 // Not sure if we want to use wp.media.string.image which will create a shortcode or 6389 // perhaps wp.html.string to at least to build the <img />. 6390 state.trigger( 'update', controller.image.toJSON() ); 6391 6392 // Restore and reset the default state. 6393 controller.setState( controller.options.state ); 6394 controller.reset(); 6395 } 6396 } 6397 } 6398 }) ); 6399 }, 6400 6401 renderReplaceImageToolbar: function() { 6402 var frame = this, 6403 lastState = frame.lastState(), 6404 previous = lastState && lastState.id; 6405 6406 this.toolbar.set( new wp.media.view.Toolbar({ 6407 controller: this, 6408 items: { 6409 back: { 6410 text: l10n.back, 6411 priority: 80, 6412 click: function() { 6413 if ( previous ) { 6414 frame.setState( previous ); 6415 } else { 6416 frame.close(); 6417 } 6418 } 6419 }, 6420 6421 replace: { 6422 style: 'primary', 6423 text: l10n.replace, 6424 priority: 20, 6425 requires: { selection: true }, 6426 6427 click: function() { 6428 var controller = this.controller, 6429 state = controller.state(), 6430 selection = state.get( 'selection' ), 6431 attachment = selection.single(); 6432 6433 controller.close(); 6434 6435 controller.image.changeAttachment( attachment, state.display( attachment ) ); 6436 6437 // Not sure if we want to use wp.media.string.image which will create a shortcode or 6438 // perhaps wp.html.string to at least to build the <img />. 6439 state.trigger( 'replace', controller.image.toJSON() ); 6440 6441 // Restore and reset the default state. 6442 controller.setState( controller.options.state ); 6443 controller.reset(); 6444 } 6445 } 6446 } 6447 }) ); 6448 } 6449 6450 }); 6451 6452 module.exports = ImageDetails; 6453 6454 6455 /***/ }), 6456 6457 /***/ 4274: 6458 /***/ ((module) => { 6459 6460 var Select = wp.media.view.MediaFrame.Select, 6461 Library = wp.media.controller.Library, 6462 l10n = wp.media.view.l10n, 6463 Post; 6464 6465 /** 6466 * wp.media.view.MediaFrame.Post 6467 * 6468 * The frame for manipulating media on the Edit Post page. 6469 * 6470 * @memberOf wp.media.view.MediaFrame 6471 * 6472 * @class 6473 * @augments wp.media.view.MediaFrame.Select 6474 * @augments wp.media.view.MediaFrame 6475 * @augments wp.media.view.Frame 6476 * @augments wp.media.View 6477 * @augments wp.Backbone.View 6478 * @augments Backbone.View 6479 * @mixes wp.media.controller.StateMachine 6480 */ 6481 Post = Select.extend(/** @lends wp.media.view.MediaFrame.Post.prototype */{ 6482 initialize: function() { 6483 this.counts = { 6484 audio: { 6485 count: wp.media.view.settings.attachmentCounts.audio, 6486 state: 'playlist' 6487 }, 6488 video: { 6489 count: wp.media.view.settings.attachmentCounts.video, 6490 state: 'video-playlist' 6491 } 6492 }; 6493 6494 _.defaults( this.options, { 6495 multiple: true, 6496 editing: false, 6497 state: 'insert', 6498 metadata: {} 6499 }); 6500 6501 // Call 'initialize' directly on the parent class. 6502 Select.prototype.initialize.apply( this, arguments ); 6503 this.createIframeStates(); 6504 6505 }, 6506 6507 /** 6508 * Create the default states. 6509 */ 6510 createStates: function() { 6511 var options = this.options; 6512 6513 this.states.add([ 6514 // Main states. 6515 new Library({ 6516 id: 'insert', 6517 title: l10n.insertMediaTitle, 6518 priority: 20, 6519 toolbar: 'main-insert', 6520 filterable: 'all', 6521 library: wp.media.query( options.library ), 6522 multiple: options.multiple ? 'reset' : false, 6523 editable: true, 6524 6525 // If the user isn't allowed to edit fields, 6526 // can they still edit it locally? 6527 allowLocalEdits: true, 6528 6529 // Show the attachment display settings. 6530 displaySettings: true, 6531 // Update user settings when users adjust the 6532 // attachment display settings. 6533 displayUserSettings: true 6534 }), 6535 6536 new Library({ 6537 id: 'gallery', 6538 title: l10n.createGalleryTitle, 6539 priority: 40, 6540 toolbar: 'main-gallery', 6541 filterable: 'uploaded', 6542 multiple: 'add', 6543 editable: false, 6544 6545 library: wp.media.query( _.defaults({ 6546 type: 'image' 6547 }, options.library ) ) 6548 }), 6549 6550 // Embed states. 6551 new wp.media.controller.Embed( { metadata: options.metadata } ), 6552 6553 new wp.media.controller.EditImage( { model: options.editImage } ), 6554 6555 // Gallery states. 6556 new wp.media.controller.GalleryEdit({ 6557 library: options.selection, 6558 editing: options.editing, 6559 menu: 'gallery' 6560 }), 6561 6562 new wp.media.controller.GalleryAdd(), 6563 6564 new Library({ 6565 id: 'playlist', 6566 title: l10n.createPlaylistTitle, 6567 priority: 60, 6568 toolbar: 'main-playlist', 6569 filterable: 'uploaded', 6570 multiple: 'add', 6571 editable: false, 6572 6573 library: wp.media.query( _.defaults({ 6574 type: 'audio' 6575 }, options.library ) ) 6576 }), 6577 6578 // Playlist states. 6579 new wp.media.controller.CollectionEdit({ 6580 type: 'audio', 6581 collectionType: 'playlist', 6582 title: l10n.editPlaylistTitle, 6583 SettingsView: wp.media.view.Settings.Playlist, 6584 library: options.selection, 6585 editing: options.editing, 6586 menu: 'playlist', 6587 dragInfoText: l10n.playlistDragInfo, 6588 dragInfo: false 6589 }), 6590 6591 new wp.media.controller.CollectionAdd({ 6592 type: 'audio', 6593 collectionType: 'playlist', 6594 title: l10n.addToPlaylistTitle 6595 }), 6596 6597 new Library({ 6598 id: 'video-playlist', 6599 title: l10n.createVideoPlaylistTitle, 6600 priority: 60, 6601 toolbar: 'main-video-playlist', 6602 filterable: 'uploaded', 6603 multiple: 'add', 6604 editable: false, 6605 6606 library: wp.media.query( _.defaults({ 6607 type: 'video' 6608 }, options.library ) ) 6609 }), 6610 6611 new wp.media.controller.CollectionEdit({ 6612 type: 'video', 6613 collectionType: 'playlist', 6614 title: l10n.editVideoPlaylistTitle, 6615 SettingsView: wp.media.view.Settings.Playlist, 6616 library: options.selection, 6617 editing: options.editing, 6618 menu: 'video-playlist', 6619 dragInfoText: l10n.videoPlaylistDragInfo, 6620 dragInfo: false 6621 }), 6622 6623 new wp.media.controller.CollectionAdd({ 6624 type: 'video', 6625 collectionType: 'playlist', 6626 title: l10n.addToVideoPlaylistTitle 6627 }) 6628 ]); 6629 6630 if ( wp.media.view.settings.post.featuredImageId ) { 6631 this.states.add( new wp.media.controller.FeaturedImage() ); 6632 } 6633 }, 6634 6635 bindHandlers: function() { 6636 var handlers, checkCounts; 6637 6638 Select.prototype.bindHandlers.apply( this, arguments ); 6639 6640 this.on( 'activate', this.activate, this ); 6641 6642 // Only bother checking media type counts if one of the counts is zero. 6643 checkCounts = _.find( this.counts, function( type ) { 6644 return type.count === 0; 6645 } ); 6646 6647 if ( typeof checkCounts !== 'undefined' ) { 6648 this.listenTo( wp.media.model.Attachments.all, 'change:type', this.mediaTypeCounts ); 6649 } 6650 6651 this.on( 'menu:create:gallery', this.createMenu, this ); 6652 this.on( 'menu:create:playlist', this.createMenu, this ); 6653 this.on( 'menu:create:video-playlist', this.createMenu, this ); 6654 this.on( 'toolbar:create:main-insert', this.createToolbar, this ); 6655 this.on( 'toolbar:create:main-gallery', this.createToolbar, this ); 6656 this.on( 'toolbar:create:main-playlist', this.createToolbar, this ); 6657 this.on( 'toolbar:create:main-video-playlist', this.createToolbar, this ); 6658 this.on( 'toolbar:create:featured-image', this.featuredImageToolbar, this ); 6659 this.on( 'toolbar:create:main-embed', this.mainEmbedToolbar, this ); 6660 6661 handlers = { 6662 menu: { 6663 'default': 'mainMenu', 6664 'gallery': 'galleryMenu', 6665 'playlist': 'playlistMenu', 6666 'video-playlist': 'videoPlaylistMenu' 6667 }, 6668 6669 content: { 6670 'embed': 'embedContent', 6671 'edit-image': 'editImageContent', 6672 'edit-selection': 'editSelectionContent' 6673 }, 6674 6675 toolbar: { 6676 'main-insert': 'mainInsertToolbar', 6677 'main-gallery': 'mainGalleryToolbar', 6678 'gallery-edit': 'galleryEditToolbar', 6679 'gallery-add': 'galleryAddToolbar', 6680 'main-playlist': 'mainPlaylistToolbar', 6681 'playlist-edit': 'playlistEditToolbar', 6682 'playlist-add': 'playlistAddToolbar', 6683 'main-video-playlist': 'mainVideoPlaylistToolbar', 6684 'video-playlist-edit': 'videoPlaylistEditToolbar', 6685 'video-playlist-add': 'videoPlaylistAddToolbar' 6686 } 6687 }; 6688 6689 _.each( handlers, function( regionHandlers, region ) { 6690 _.each( regionHandlers, function( callback, handler ) { 6691 this.on( region + ':render:' + handler, this[ callback ], this ); 6692 }, this ); 6693 }, this ); 6694 }, 6695 6696 activate: function() { 6697 // Hide menu items for states tied to particular media types if there are no items. 6698 _.each( this.counts, function( type ) { 6699 if ( type.count < 1 ) { 6700 this.menuItemVisibility( type.state, 'hide' ); 6701 } 6702 }, this ); 6703 }, 6704 6705 mediaTypeCounts: function( model, attr ) { 6706 if ( typeof this.counts[ attr ] !== 'undefined' && this.counts[ attr ].count < 1 ) { 6707 this.counts[ attr ].count++; 6708 this.menuItemVisibility( this.counts[ attr ].state, 'show' ); 6709 } 6710 }, 6711 6712 // Menus. 6713 /** 6714 * @param {wp.Backbone.View} view 6715 */ 6716 mainMenu: function( view ) { 6717 view.set({ 6718 'library-separator': new wp.media.View({ 6719 className: 'separator', 6720 priority: 100, 6721 attributes: { 6722 role: 'presentation' 6723 } 6724 }) 6725 }); 6726 }, 6727 6728 menuItemVisibility: function( state, visibility ) { 6729 var menu = this.menu.get(); 6730 if ( visibility === 'hide' ) { 6731 menu.hide( state ); 6732 } else if ( visibility === 'show' ) { 6733 menu.show( state ); 6734 } 6735 }, 6736 /** 6737 * @param {wp.Backbone.View} view 6738 */ 6739 galleryMenu: function( view ) { 6740 var lastState = this.lastState(), 6741 previous = lastState && lastState.id, 6742 frame = this; 6743 6744 view.set({ 6745 cancel: { 6746 text: l10n.cancelGalleryTitle, 6747 priority: 20, 6748 click: function() { 6749 if ( previous ) { 6750 frame.setState( previous ); 6751 } else { 6752 frame.close(); 6753 } 6754 6755 // Move focus to the modal after canceling a Gallery. 6756 this.controller.modal.focusManager.focus(); 6757 } 6758 }, 6759 separateCancel: new wp.media.View({ 6760 className: 'separator', 6761 priority: 40 6762 }) 6763 }); 6764 }, 6765 6766 playlistMenu: function( view ) { 6767 var lastState = this.lastState(), 6768 previous = lastState && lastState.id, 6769 frame = this; 6770 6771 view.set({ 6772 cancel: { 6773 text: l10n.cancelPlaylistTitle, 6774 priority: 20, 6775 click: function() { 6776 if ( previous ) { 6777 frame.setState( previous ); 6778 } else { 6779 frame.close(); 6780 } 6781 6782 // Move focus to the modal after canceling an Audio Playlist. 6783 this.controller.modal.focusManager.focus(); 6784 } 6785 }, 6786 separateCancel: new wp.media.View({ 6787 className: 'separator', 6788 priority: 40 6789 }) 6790 }); 6791 }, 6792 6793 videoPlaylistMenu: function( view ) { 6794 var lastState = this.lastState(), 6795 previous = lastState && lastState.id, 6796 frame = this; 6797 6798 view.set({ 6799 cancel: { 6800 text: l10n.cancelVideoPlaylistTitle, 6801 priority: 20, 6802 click: function() { 6803 if ( previous ) { 6804 frame.setState( previous ); 6805 } else { 6806 frame.close(); 6807 } 6808 6809 // Move focus to the modal after canceling a Video Playlist. 6810 this.controller.modal.focusManager.focus(); 6811 } 6812 }, 6813 separateCancel: new wp.media.View({ 6814 className: 'separator', 6815 priority: 40 6816 }) 6817 }); 6818 }, 6819 6820 // Content. 6821 embedContent: function() { 6822 var view = new wp.media.view.Embed({ 6823 controller: this, 6824 model: this.state() 6825 }).render(); 6826 6827 this.content.set( view ); 6828 }, 6829 6830 editSelectionContent: function() { 6831 var state = this.state(), 6832 selection = state.get('selection'), 6833 view; 6834 6835 view = new wp.media.view.AttachmentsBrowser({ 6836 controller: this, 6837 collection: selection, 6838 selection: selection, 6839 model: state, 6840 sortable: true, 6841 search: false, 6842 date: false, 6843 dragInfo: true, 6844 6845 AttachmentView: wp.media.view.Attachments.EditSelection 6846 }).render(); 6847 6848 view.toolbar.set( 'backToLibrary', { 6849 text: l10n.returnToLibrary, 6850 priority: -100, 6851 6852 click: function() { 6853 this.controller.content.mode('browse'); 6854 // Move focus to the modal when jumping back from Edit Selection to Add Media view. 6855 this.controller.modal.focusManager.focus(); 6856 } 6857 }); 6858 6859 // Browse our library of attachments. 6860 this.content.set( view ); 6861 6862 // Trigger the controller to set focus. 6863 this.trigger( 'edit:selection', this ); 6864 }, 6865 6866 editImageContent: function() { 6867 var image = this.state().get('image'), 6868 view = new wp.media.view.EditImage( { model: image, controller: this } ).render(); 6869 6870 this.content.set( view ); 6871 6872 // After creating the wrapper view, load the actual editor via an Ajax call. 6873 view.loadEditor(); 6874 6875 }, 6876 6877 // Toolbars. 6878 6879 /** 6880 * @param {wp.Backbone.View} view 6881 */ 6882 selectionStatusToolbar: function( view ) { 6883 var editable = this.state().get('editable'); 6884 6885 view.set( 'selection', new wp.media.view.Selection({ 6886 controller: this, 6887 collection: this.state().get('selection'), 6888 priority: -40, 6889 6890 // If the selection is editable, pass the callback to 6891 // switch the content mode. 6892 editable: editable && function() { 6893 this.controller.content.mode('edit-selection'); 6894 } 6895 }).render() ); 6896 }, 6897 6898 /** 6899 * @param {wp.Backbone.View} view 6900 */ 6901 mainInsertToolbar: function( view ) { 6902 var controller = this; 6903 6904 this.selectionStatusToolbar( view ); 6905 6906 view.set( 'insert', { 6907 style: 'primary', 6908 priority: 80, 6909 text: l10n.insertIntoPost, 6910 requires: { selection: true }, 6911 6912 /** 6913 * @ignore 6914 * 6915 * @fires wp.media.controller.State#insert 6916 */ 6917 click: function() { 6918 var state = controller.state(), 6919 selection = state.get('selection'); 6920 6921 controller.close(); 6922 state.trigger( 'insert', selection ).reset(); 6923 } 6924 }); 6925 }, 6926 6927 /** 6928 * @param {wp.Backbone.View} view 6929 */ 6930 mainGalleryToolbar: function( view ) { 6931 var controller = this; 6932 6933 this.selectionStatusToolbar( view ); 6934 6935 view.set( 'gallery', { 6936 style: 'primary', 6937 text: l10n.createNewGallery, 6938 priority: 60, 6939 requires: { selection: true }, 6940 6941 click: function() { 6942 var selection = controller.state().get('selection'), 6943 edit = controller.state('gallery-edit'), 6944 models = selection.where({ type: 'image' }); 6945 6946 edit.set( 'library', new wp.media.model.Selection( models, { 6947 props: selection.props.toJSON(), 6948 multiple: true 6949 }) ); 6950 6951 // Jump to Edit Gallery view. 6952 this.controller.setState( 'gallery-edit' ); 6953 6954 // Move focus to the modal after jumping to Edit Gallery view. 6955 this.controller.modal.focusManager.focus(); 6956 } 6957 }); 6958 }, 6959 6960 mainPlaylistToolbar: function( view ) { 6961 var controller = this; 6962 6963 this.selectionStatusToolbar( view ); 6964 6965 view.set( 'playlist', { 6966 style: 'primary', 6967 text: l10n.createNewPlaylist, 6968 priority: 100, 6969 requires: { selection: true }, 6970 6971 click: function() { 6972 var selection = controller.state().get('selection'), 6973 edit = controller.state('playlist-edit'), 6974 models = selection.where({ type: 'audio' }); 6975 6976 edit.set( 'library', new wp.media.model.Selection( models, { 6977 props: selection.props.toJSON(), 6978 multiple: true 6979 }) ); 6980 6981 // Jump to Edit Audio Playlist view. 6982 this.controller.setState( 'playlist-edit' ); 6983 6984 // Move focus to the modal after jumping to Edit Audio Playlist view. 6985 this.controller.modal.focusManager.focus(); 6986 } 6987 }); 6988 }, 6989 6990 mainVideoPlaylistToolbar: function( view ) { 6991 var controller = this; 6992 6993 this.selectionStatusToolbar( view ); 6994 6995 view.set( 'video-playlist', { 6996 style: 'primary', 6997 text: l10n.createNewVideoPlaylist, 6998 priority: 100, 6999 requires: { selection: true }, 7000 7001 click: function() { 7002 var selection = controller.state().get('selection'), 7003 edit = controller.state('video-playlist-edit'), 7004 models = selection.where({ type: 'video' }); 7005 7006 edit.set( 'library', new wp.media.model.Selection( models, { 7007 props: selection.props.toJSON(), 7008 multiple: true 7009 }) ); 7010 7011 // Jump to Edit Video Playlist view. 7012 this.controller.setState( 'video-playlist-edit' ); 7013 7014 // Move focus to the modal after jumping to Edit Video Playlist view. 7015 this.controller.modal.focusManager.focus(); 7016 } 7017 }); 7018 }, 7019 7020 featuredImageToolbar: function( toolbar ) { 7021 this.createSelectToolbar( toolbar, { 7022 text: l10n.setFeaturedImage, 7023 state: this.options.state 7024 }); 7025 }, 7026 7027 mainEmbedToolbar: function( toolbar ) { 7028 toolbar.view = new wp.media.view.Toolbar.Embed({ 7029 controller: this 7030 }); 7031 }, 7032 7033 galleryEditToolbar: function() { 7034 var editing = this.state().get('editing'); 7035 this.toolbar.set( new wp.media.view.Toolbar({ 7036 controller: this, 7037 items: { 7038 insert: { 7039 style: 'primary', 7040 text: editing ? l10n.updateGallery : l10n.insertGallery, 7041 priority: 80, 7042 requires: { library: true }, 7043 7044 /** 7045 * @fires wp.media.controller.State#update 7046 */ 7047 click: function() { 7048 var controller = this.controller, 7049 state = controller.state(); 7050 7051 controller.close(); 7052 state.trigger( 'update', state.get('library') ); 7053 7054 // Restore and reset the default state. 7055 controller.setState( controller.options.state ); 7056 controller.reset(); 7057 } 7058 } 7059 } 7060 }) ); 7061 }, 7062 7063 galleryAddToolbar: function() { 7064 this.toolbar.set( new wp.media.view.Toolbar({ 7065 controller: this, 7066 items: { 7067 insert: { 7068 style: 'primary', 7069 text: l10n.addToGallery, 7070 priority: 80, 7071 requires: { selection: true }, 7072 7073 /** 7074 * @fires wp.media.controller.State#reset 7075 */ 7076 click: function() { 7077 var controller = this.controller, 7078 state = controller.state(), 7079 edit = controller.state('gallery-edit'); 7080 7081 edit.get('library').add( state.get('selection').models ); 7082 state.trigger('reset'); 7083 controller.setState('gallery-edit'); 7084 // Move focus to the modal when jumping back from Add to Gallery to Edit Gallery view. 7085 this.controller.modal.focusManager.focus(); 7086 } 7087 } 7088 } 7089 }) ); 7090 }, 7091 7092 playlistEditToolbar: function() { 7093 var editing = this.state().get('editing'); 7094 this.toolbar.set( new wp.media.view.Toolbar({ 7095 controller: this, 7096 items: { 7097 insert: { 7098 style: 'primary', 7099 text: editing ? l10n.updatePlaylist : l10n.insertPlaylist, 7100 priority: 80, 7101 requires: { library: true }, 7102 7103 /** 7104 * @fires wp.media.controller.State#update 7105 */ 7106 click: function() { 7107 var controller = this.controller, 7108 state = controller.state(); 7109 7110 controller.close(); 7111 state.trigger( 'update', state.get('library') ); 7112 7113 // Restore and reset the default state. 7114 controller.setState( controller.options.state ); 7115 controller.reset(); 7116 } 7117 } 7118 } 7119 }) ); 7120 }, 7121 7122 playlistAddToolbar: function() { 7123 this.toolbar.set( new wp.media.view.Toolbar({ 7124 controller: this, 7125 items: { 7126 insert: { 7127 style: 'primary', 7128 text: l10n.addToPlaylist, 7129 priority: 80, 7130 requires: { selection: true }, 7131 7132 /** 7133 * @fires wp.media.controller.State#reset 7134 */ 7135 click: function() { 7136 var controller = this.controller, 7137 state = controller.state(), 7138 edit = controller.state('playlist-edit'); 7139 7140 edit.get('library').add( state.get('selection').models ); 7141 state.trigger('reset'); 7142 controller.setState('playlist-edit'); 7143 // Move focus to the modal when jumping back from Add to Audio Playlist to Edit Audio Playlist view. 7144 this.controller.modal.focusManager.focus(); 7145 } 7146 } 7147 } 7148 }) ); 7149 }, 7150 7151 videoPlaylistEditToolbar: function() { 7152 var editing = this.state().get('editing'); 7153 this.toolbar.set( new wp.media.view.Toolbar({ 7154 controller: this, 7155 items: { 7156 insert: { 7157 style: 'primary', 7158 text: editing ? l10n.updateVideoPlaylist : l10n.insertVideoPlaylist, 7159 priority: 140, 7160 requires: { library: true }, 7161 7162 click: function() { 7163 var controller = this.controller, 7164 state = controller.state(), 7165 library = state.get('library'); 7166 7167 library.type = 'video'; 7168 7169 controller.close(); 7170 state.trigger( 'update', library ); 7171 7172 // Restore and reset the default state. 7173 controller.setState( controller.options.state ); 7174 controller.reset(); 7175 } 7176 } 7177 } 7178 }) ); 7179 }, 7180 7181 videoPlaylistAddToolbar: function() { 7182 this.toolbar.set( new wp.media.view.Toolbar({ 7183 controller: this, 7184 items: { 7185 insert: { 7186 style: 'primary', 7187 text: l10n.addToVideoPlaylist, 7188 priority: 140, 7189 requires: { selection: true }, 7190 7191 click: function() { 7192 var controller = this.controller, 7193 state = controller.state(), 7194 edit = controller.state('video-playlist-edit'); 7195 7196 edit.get('library').add( state.get('selection').models ); 7197 state.trigger('reset'); 7198 controller.setState('video-playlist-edit'); 7199 // Move focus to the modal when jumping back from Add to Video Playlist to Edit Video Playlist view. 7200 this.controller.modal.focusManager.focus(); 7201 } 7202 } 7203 } 7204 }) ); 7205 } 7206 }); 7207 7208 module.exports = Post; 7209 7210 7211 /***/ }), 7212 7213 /***/ 455: 7214 /***/ ((module) => { 7215 7216 var MediaFrame = wp.media.view.MediaFrame, 7217 l10n = wp.media.view.l10n, 7218 Select; 7219 7220 /** 7221 * wp.media.view.MediaFrame.Select 7222 * 7223 * A frame for selecting an item or items from the media library. 7224 * 7225 * @memberOf wp.media.view.MediaFrame 7226 * 7227 * @class 7228 * @augments wp.media.view.MediaFrame 7229 * @augments wp.media.view.Frame 7230 * @augments wp.media.View 7231 * @augments wp.Backbone.View 7232 * @augments Backbone.View 7233 * @mixes wp.media.controller.StateMachine 7234 */ 7235 Select = MediaFrame.extend(/** @lends wp.media.view.MediaFrame.Select.prototype */{ 7236 initialize: function() { 7237 // Call 'initialize' directly on the parent class. 7238 MediaFrame.prototype.initialize.apply( this, arguments ); 7239 7240 _.defaults( this.options, { 7241 selection: [], 7242 library: {}, 7243 multiple: false, 7244 state: 'library' 7245 }); 7246 7247 this.createSelection(); 7248 this.createStates(); 7249 this.bindHandlers(); 7250 }, 7251 7252 /** 7253 * Attach a selection collection to the frame. 7254 * 7255 * A selection is a collection of attachments used for a specific purpose 7256 * by a media frame. e.g. Selecting an attachment (or many) to insert into 7257 * post content. 7258 * 7259 * @see media.model.Selection 7260 */ 7261 createSelection: function() { 7262 var selection = this.options.selection; 7263 7264 if ( ! (selection instanceof wp.media.model.Selection) ) { 7265 this.options.selection = new wp.media.model.Selection( selection, { 7266 multiple: this.options.multiple 7267 }); 7268 } 7269 7270 this._selection = { 7271 attachments: new wp.media.model.Attachments(), 7272 difference: [] 7273 }; 7274 }, 7275 7276 editImageContent: function() { 7277 var image = this.state().get('image'), 7278 view = new wp.media.view.EditImage( { model: image, controller: this } ).render(); 7279 7280 this.content.set( view ); 7281 7282 // After creating the wrapper view, load the actual editor via an Ajax call. 7283 view.loadEditor(); 7284 }, 7285 7286 /** 7287 * Create the default states on the frame. 7288 */ 7289 createStates: function() { 7290 var options = this.options; 7291 7292 if ( this.options.states ) { 7293 return; 7294 } 7295 7296 // Add the default states. 7297 this.states.add([ 7298 // Main states. 7299 new wp.media.controller.Library({ 7300 library: wp.media.query( options.library ), 7301 multiple: options.multiple, 7302 title: options.title, 7303 priority: 20 7304 }), 7305 new wp.media.controller.EditImage( { model: options.editImage } ) 7306 ]); 7307 }, 7308 7309 /** 7310 * Bind region mode event callbacks. 7311 * 7312 * @see media.controller.Region.render 7313 */ 7314 bindHandlers: function() { 7315 this.on( 'router:create:browse', this.createRouter, this ); 7316 this.on( 'router:render:browse', this.browseRouter, this ); 7317 this.on( 'content:create:browse', this.browseContent, this ); 7318 this.on( 'content:render:upload', this.uploadContent, this ); 7319 this.on( 'toolbar:create:select', this.createSelectToolbar, this ); 7320 this.on( 'content:render:edit-image', this.editImageContent, this ); 7321 }, 7322 7323 /** 7324 * Render callback for the router region in the `browse` mode. 7325 * 7326 * @param {wp.media.view.Router} routerView 7327 */ 7328 browseRouter: function( routerView ) { 7329 routerView.set({ 7330 upload: { 7331 text: l10n.uploadFilesTitle, 7332 priority: 20 7333 }, 7334 browse: { 7335 text: l10n.mediaLibraryTitle, 7336 priority: 40 7337 } 7338 }); 7339 }, 7340 7341 /** 7342 * Render callback for the content region in the `browse` mode. 7343 * 7344 * @param {wp.media.controller.Region} contentRegion 7345 */ 7346 browseContent: function( contentRegion ) { 7347 var state = this.state(); 7348 7349 this.$el.removeClass('hide-toolbar'); 7350 7351 // Browse our library of attachments. 7352 contentRegion.view = new wp.media.view.AttachmentsBrowser({ 7353 controller: this, 7354 collection: state.get('library'), 7355 selection: state.get('selection'), 7356 model: state, 7357 sortable: state.get('sortable'), 7358 search: state.get('searchable'), 7359 filters: state.get('filterable'), 7360 date: state.get('date'), 7361 display: state.has('display') ? state.get('display') : state.get('displaySettings'), 7362 dragInfo: state.get('dragInfo'), 7363 7364 idealColumnWidth: state.get('idealColumnWidth'), 7365 suggestedWidth: state.get('suggestedWidth'), 7366 suggestedHeight: state.get('suggestedHeight'), 7367 7368 AttachmentView: state.get('AttachmentView') 7369 }); 7370 }, 7371 7372 /** 7373 * Render callback for the content region in the `upload` mode. 7374 */ 7375 uploadContent: function() { 7376 this.$el.removeClass( 'hide-toolbar' ); 7377 this.content.set( new wp.media.view.UploaderInline({ 7378 controller: this 7379 }) ); 7380 }, 7381 7382 /** 7383 * Toolbars 7384 * 7385 * @param {Object} toolbar 7386 * @param {Object} [options={}] 7387 * @this wp.media.controller.Region 7388 */ 7389 createSelectToolbar: function( toolbar, options ) { 7390 options = options || this.options.button || {}; 7391 options.controller = this; 7392 7393 toolbar.view = new wp.media.view.Toolbar.Select( options ); 7394 } 7395 }); 7396 7397 module.exports = Select; 7398 7399 7400 /***/ }), 7401 7402 /***/ 170: 7403 /***/ ((module) => { 7404 7405 /** 7406 * wp.media.view.Heading 7407 * 7408 * A reusable heading component for the media library 7409 * 7410 * Used to add accessibility friendly headers in the media library/modal. 7411 * 7412 * @class 7413 * @augments wp.media.View 7414 * @augments wp.Backbone.View 7415 * @augments Backbone.View 7416 */ 7417 var Heading = wp.media.View.extend( { 7418 tagName: function() { 7419 return this.options.level || 'h1'; 7420 }, 7421 className: 'media-views-heading', 7422 7423 initialize: function() { 7424 7425 if ( this.options.className ) { 7426 this.$el.addClass( this.options.className ); 7427 } 7428 7429 this.text = this.options.text; 7430 }, 7431 7432 render: function() { 7433 this.$el.html( this.text ); 7434 return this; 7435 } 7436 } ); 7437 7438 module.exports = Heading; 7439 7440 7441 /***/ }), 7442 7443 /***/ 1982: 7444 /***/ ((module) => { 7445 7446 /** 7447 * wp.media.view.Iframe 7448 * 7449 * @memberOf wp.media.view 7450 * 7451 * @class 7452 * @augments wp.media.View 7453 * @augments wp.Backbone.View 7454 * @augments Backbone.View 7455 */ 7456 var Iframe = wp.media.View.extend(/** @lends wp.media.view.Iframe.prototype */{ 7457 className: 'media-iframe', 7458 /** 7459 * @return {wp.media.view.Iframe} Returns itself to allow chaining. 7460 */ 7461 render: function() { 7462 this.views.detach(); 7463 this.$el.html( '<iframe src="' + this.controller.state().get('src') + '" />' ); 7464 this.views.render(); 7465 return this; 7466 } 7467 }); 7468 7469 module.exports = Iframe; 7470 7471 7472 /***/ }), 7473 7474 /***/ 2650: 7475 /***/ ((module) => { 7476 7477 var AttachmentDisplay = wp.media.view.Settings.AttachmentDisplay, 7478 $ = jQuery, 7479 ImageDetails; 7480 7481 /** 7482 * wp.media.view.ImageDetails 7483 * 7484 * @memberOf wp.media.view 7485 * 7486 * @class 7487 * @augments wp.media.view.Settings.AttachmentDisplay 7488 * @augments wp.media.view.Settings 7489 * @augments wp.media.View 7490 * @augments wp.Backbone.View 7491 * @augments Backbone.View 7492 */ 7493 ImageDetails = AttachmentDisplay.extend(/** @lends wp.media.view.ImageDetails.prototype */{ 7494 className: 'image-details', 7495 template: wp.template('image-details'), 7496 events: _.defaults( AttachmentDisplay.prototype.events, { 7497 'click .edit-attachment': 'editAttachment', 7498 'click .replace-attachment': 'replaceAttachment', 7499 'click .advanced-toggle': 'onToggleAdvanced', 7500 'change [data-setting="customWidth"]': 'onCustomSize', 7501 'change [data-setting="customHeight"]': 'onCustomSize', 7502 'keyup [data-setting="customWidth"]': 'onCustomSize', 7503 'keyup [data-setting="customHeight"]': 'onCustomSize' 7504 } ), 7505 initialize: function() { 7506 // Used in AttachmentDisplay.prototype.updateLinkTo. 7507 this.options.attachment = this.model.attachment; 7508 this.listenTo( this.model, 'change:url', this.updateUrl ); 7509 this.listenTo( this.model, 'change:link', this.toggleLinkSettings ); 7510 this.listenTo( this.model, 'change:size', this.toggleCustomSize ); 7511 7512 AttachmentDisplay.prototype.initialize.apply( this, arguments ); 7513 }, 7514 7515 prepare: function() { 7516 var attachment = false; 7517 7518 if ( this.model.attachment ) { 7519 attachment = this.model.attachment.toJSON(); 7520 } 7521 return _.defaults({ 7522 model: this.model.toJSON(), 7523 attachment: attachment 7524 }, this.options ); 7525 }, 7526 7527 render: function() { 7528 var args = arguments; 7529 7530 if ( this.model.attachment && 'pending' === this.model.dfd.state() ) { 7531 this.model.dfd 7532 .done( _.bind( function() { 7533 AttachmentDisplay.prototype.render.apply( this, args ); 7534 this.postRender(); 7535 }, this ) ) 7536 .fail( _.bind( function() { 7537 this.model.attachment = false; 7538 AttachmentDisplay.prototype.render.apply( this, args ); 7539 this.postRender(); 7540 }, this ) ); 7541 } else { 7542 AttachmentDisplay.prototype.render.apply( this, arguments ); 7543 this.postRender(); 7544 } 7545 7546 return this; 7547 }, 7548 7549 postRender: function() { 7550 setTimeout( _.bind( this.scrollToTop, this ), 10 ); 7551 this.toggleLinkSettings(); 7552 if ( window.getUserSetting( 'advImgDetails' ) === 'show' ) { 7553 this.toggleAdvanced( true ); 7554 } 7555 this.trigger( 'post-render' ); 7556 }, 7557 7558 scrollToTop: function() { 7559 this.$( '.embed-media-settings' ).scrollTop( 0 ); 7560 }, 7561 7562 updateUrl: function() { 7563 this.$( '.image img' ).attr( 'src', this.model.get( 'url' ) ); 7564 this.$( '.url' ).val( this.model.get( 'url' ) ); 7565 }, 7566 7567 toggleLinkSettings: function() { 7568 if ( this.model.get( 'link' ) === 'none' ) { 7569 this.$( '.link-settings' ).addClass('hidden'); 7570 } else { 7571 this.$( '.link-settings' ).removeClass('hidden'); 7572 } 7573 }, 7574 7575 toggleCustomSize: function() { 7576 if ( this.model.get( 'size' ) !== 'custom' ) { 7577 this.$( '.custom-size' ).addClass('hidden'); 7578 } else { 7579 this.$( '.custom-size' ).removeClass('hidden'); 7580 } 7581 }, 7582 7583 onCustomSize: function( event ) { 7584 var dimension = $( event.target ).data('setting'), 7585 num = $( event.target ).val(), 7586 value; 7587 7588 // Ignore bogus input. 7589 if ( ! /^\d+/.test( num ) || parseInt( num, 10 ) < 1 ) { 7590 event.preventDefault(); 7591 return; 7592 } 7593 7594 if ( dimension === 'customWidth' ) { 7595 value = Math.round( 1 / this.model.get( 'aspectRatio' ) * num ); 7596 this.model.set( 'customHeight', value, { silent: true } ); 7597 this.$( '[data-setting="customHeight"]' ).val( value ); 7598 } else { 7599 value = Math.round( this.model.get( 'aspectRatio' ) * num ); 7600 this.model.set( 'customWidth', value, { silent: true } ); 7601 this.$( '[data-setting="customWidth"]' ).val( value ); 7602 } 7603 }, 7604 7605 onToggleAdvanced: function( event ) { 7606 event.preventDefault(); 7607 this.toggleAdvanced(); 7608 }, 7609 7610 toggleAdvanced: function( show ) { 7611 var $advanced = this.$el.find( '.advanced-section' ), 7612 mode; 7613 7614 if ( $advanced.hasClass('advanced-visible') || show === false ) { 7615 $advanced.removeClass('advanced-visible'); 7616 $advanced.find('.advanced-settings').addClass('hidden'); 7617 mode = 'hide'; 7618 } else { 7619 $advanced.addClass('advanced-visible'); 7620 $advanced.find('.advanced-settings').removeClass('hidden'); 7621 mode = 'show'; 7622 } 7623 7624 window.setUserSetting( 'advImgDetails', mode ); 7625 }, 7626 7627 editAttachment: function( event ) { 7628 var editState = this.controller.states.get( 'edit-image' ); 7629 7630 if ( window.imageEdit && editState ) { 7631 event.preventDefault(); 7632 editState.set( 'image', this.model.attachment ); 7633 this.controller.setState( 'edit-image' ); 7634 } 7635 }, 7636 7637 replaceAttachment: function( event ) { 7638 event.preventDefault(); 7639 this.controller.setState( 'replace-image' ); 7640 } 7641 }); 7642 7643 module.exports = ImageDetails; 7644 7645 7646 /***/ }), 7647 7648 /***/ 4338: 7649 /***/ ((module) => { 7650 7651 /** 7652 * wp.media.view.Label 7653 * 7654 * @memberOf wp.media.view 7655 * 7656 * @class 7657 * @augments wp.media.View 7658 * @augments wp.Backbone.View 7659 * @augments Backbone.View 7660 */ 7661 var Label = wp.media.View.extend(/** @lends wp.media.view.Label.prototype */{ 7662 tagName: 'label', 7663 className: 'screen-reader-text', 7664 7665 initialize: function() { 7666 this.value = this.options.value; 7667 }, 7668 7669 render: function() { 7670 this.$el.html( this.value ); 7671 7672 return this; 7673 } 7674 }); 7675 7676 module.exports = Label; 7677 7678 7679 /***/ }), 7680 7681 /***/ 2836: 7682 /***/ ((module) => { 7683 7684 var Frame = wp.media.view.Frame, 7685 l10n = wp.media.view.l10n, 7686 $ = jQuery, 7687 MediaFrame; 7688 7689 /** 7690 * wp.media.view.MediaFrame 7691 * 7692 * The frame used to create the media modal. 7693 * 7694 * @memberOf wp.media.view 7695 * 7696 * @class 7697 * @augments wp.media.view.Frame 7698 * @augments wp.media.View 7699 * @augments wp.Backbone.View 7700 * @augments Backbone.View 7701 * @mixes wp.media.controller.StateMachine 7702 */ 7703 MediaFrame = Frame.extend(/** @lends wp.media.view.MediaFrame.prototype */{ 7704 className: 'media-frame', 7705 template: wp.template('media-frame'), 7706 regions: ['menu','title','content','toolbar','router'], 7707 7708 events: { 7709 'click .media-frame-menu-toggle': 'toggleMenu' 7710 }, 7711 7712 /** 7713 * @constructs 7714 */ 7715 initialize: function() { 7716 Frame.prototype.initialize.apply( this, arguments ); 7717 7718 _.defaults( this.options, { 7719 title: l10n.mediaFrameDefaultTitle, 7720 modal: true, 7721 uploader: true 7722 }); 7723 7724 // Ensure core UI is enabled. 7725 this.$el.addClass('wp-core-ui'); 7726 7727 // Initialize modal container view. 7728 if ( this.options.modal ) { 7729 this.modal = new wp.media.view.Modal({ 7730 controller: this, 7731 title: this.options.title 7732 }); 7733 7734 this.modal.content( this ); 7735 } 7736 7737 // Force the uploader off if the upload limit has been exceeded or 7738 // if the browser isn't supported. 7739 if ( wp.Uploader.limitExceeded || ! wp.Uploader.browser.supported ) { 7740 this.options.uploader = false; 7741 } 7742 7743 // Initialize window-wide uploader. 7744 if ( this.options.uploader ) { 7745 this.uploader = new wp.media.view.UploaderWindow({ 7746 controller: this, 7747 uploader: { 7748 dropzone: this.modal ? this.modal.$el : this.$el, 7749 container: this.$el 7750 } 7751 }); 7752 this.views.set( '.media-frame-uploader', this.uploader ); 7753 } 7754 7755 this.on( 'attach', _.bind( this.views.ready, this.views ), this ); 7756 7757 // Bind default title creation. 7758 this.on( 'title:create:default', this.createTitle, this ); 7759 this.title.mode('default'); 7760 7761 // Bind default menu. 7762 this.on( 'menu:create:default', this.createMenu, this ); 7763 7764 // Set the menu ARIA tab panel attributes when the modal opens. 7765 this.on( 'open', this.setMenuTabPanelAriaAttributes, this ); 7766 // Set the router ARIA tab panel attributes when the modal opens. 7767 this.on( 'open', this.setRouterTabPanelAriaAttributes, this ); 7768 7769 // Update the menu ARIA tab panel attributes when the content updates. 7770 this.on( 'content:render', this.setMenuTabPanelAriaAttributes, this ); 7771 // Update the router ARIA tab panel attributes when the content updates. 7772 this.on( 'content:render', this.setRouterTabPanelAriaAttributes, this ); 7773 }, 7774 7775 /** 7776 * Sets the attributes to be used on the menu ARIA tab panel. 7777 * 7778 * @since 5.3.0 7779 * 7780 * @return {void} 7781 */ 7782 setMenuTabPanelAriaAttributes: function() { 7783 var stateId = this.state().get( 'id' ), 7784 tabPanelEl = this.$el.find( '.media-frame-tab-panel' ), 7785 ariaLabelledby; 7786 7787 tabPanelEl.removeAttr( 'role aria-labelledby tabindex' ); 7788 7789 if ( this.state().get( 'menu' ) && this.menuView && this.menuView.isVisible ) { 7790 ariaLabelledby = 'menu-item-' + stateId; 7791 7792 // Set the tab panel attributes only if the tabs are visible. 7793 tabPanelEl 7794 .attr( { 7795 role: 'tabpanel', 7796 'aria-labelledby': ariaLabelledby, 7797 tabIndex: '0' 7798 } ); 7799 } 7800 }, 7801 7802 /** 7803 * Sets the attributes to be used on the router ARIA tab panel. 7804 * 7805 * @since 5.3.0 7806 * 7807 * @return {void} 7808 */ 7809 setRouterTabPanelAriaAttributes: function() { 7810 var tabPanelEl = this.$el.find( '.media-frame-content' ), 7811 ariaLabelledby; 7812 7813 tabPanelEl.removeAttr( 'role aria-labelledby tabindex' ); 7814 7815 // Set the tab panel attributes only if the tabs are visible. 7816 if ( this.state().get( 'router' ) && this.routerView && this.routerView.isVisible && this.content._mode ) { 7817 ariaLabelledby = 'menu-item-' + this.content._mode; 7818 7819 tabPanelEl 7820 .attr( { 7821 role: 'tabpanel', 7822 'aria-labelledby': ariaLabelledby, 7823 tabIndex: '0' 7824 } ); 7825 } 7826 }, 7827 7828 /** 7829 * @return {wp.media.view.MediaFrame} Returns itself to allow chaining. 7830 */ 7831 render: function() { 7832 // Activate the default state if no active state exists. 7833 if ( ! this.state() && this.options.state ) { 7834 this.setState( this.options.state ); 7835 } 7836 /** 7837 * call 'render' directly on the parent class 7838 */ 7839 return Frame.prototype.render.apply( this, arguments ); 7840 }, 7841 /** 7842 * @param {Object} title 7843 * @this wp.media.controller.Region 7844 */ 7845 createTitle: function( title ) { 7846 title.view = new wp.media.View({ 7847 controller: this, 7848 tagName: 'h1' 7849 }); 7850 }, 7851 /** 7852 * @param {Object} menu 7853 * @this wp.media.controller.Region 7854 */ 7855 createMenu: function( menu ) { 7856 menu.view = new wp.media.view.Menu({ 7857 controller: this, 7858 7859 attributes: { 7860 role: 'tablist', 7861 'aria-orientation': 'vertical' 7862 } 7863 }); 7864 7865 this.menuView = menu.view; 7866 }, 7867 7868 toggleMenu: function( event ) { 7869 var menu = this.$el.find( '.media-menu' ); 7870 7871 menu.toggleClass( 'visible' ); 7872 $( event.target ).attr( 'aria-expanded', menu.hasClass( 'visible' ) ); 7873 }, 7874 7875 /** 7876 * @param {Object} toolbar 7877 * @this wp.media.controller.Region 7878 */ 7879 createToolbar: function( toolbar ) { 7880 toolbar.view = new wp.media.view.Toolbar({ 7881 controller: this 7882 }); 7883 }, 7884 /** 7885 * @param {Object} router 7886 * @this wp.media.controller.Region 7887 */ 7888 createRouter: function( router ) { 7889 router.view = new wp.media.view.Router({ 7890 controller: this, 7891 7892 attributes: { 7893 role: 'tablist', 7894 'aria-orientation': 'horizontal' 7895 } 7896 }); 7897 7898 this.routerView = router.view; 7899 }, 7900 /** 7901 * @param {Object} options 7902 */ 7903 createIframeStates: function( options ) { 7904 var settings = wp.media.view.settings, 7905 tabs = settings.tabs, 7906 tabUrl = settings.tabUrl, 7907 $postId; 7908 7909 if ( ! tabs || ! tabUrl ) { 7910 return; 7911 } 7912 7913 // Add the post ID to the tab URL if it exists. 7914 $postId = $('#post_ID'); 7915 if ( $postId.length ) { 7916 tabUrl += '&post_id=' + $postId.val(); 7917 } 7918 7919 // Generate the tab states. 7920 _.each( tabs, function( title, id ) { 7921 this.state( 'iframe:' + id ).set( _.defaults({ 7922 tab: id, 7923 src: tabUrl + '&tab=' + id, 7924 title: title, 7925 content: 'iframe', 7926 menu: 'default' 7927 }, options ) ); 7928 }, this ); 7929 7930 this.on( 'content:create:iframe', this.iframeContent, this ); 7931 this.on( 'content:deactivate:iframe', this.iframeContentCleanup, this ); 7932 this.on( 'menu:render:default', this.iframeMenu, this ); 7933 this.on( 'open', this.hijackThickbox, this ); 7934 this.on( 'close', this.restoreThickbox, this ); 7935 }, 7936 7937 /** 7938 * @param {Object} content 7939 * @this wp.media.controller.Region 7940 */ 7941 iframeContent: function( content ) { 7942 this.$el.addClass('hide-toolbar'); 7943 content.view = new wp.media.view.Iframe({ 7944 controller: this 7945 }); 7946 }, 7947 7948 iframeContentCleanup: function() { 7949 this.$el.removeClass('hide-toolbar'); 7950 }, 7951 7952 iframeMenu: function( view ) { 7953 var views = {}; 7954 7955 if ( ! view ) { 7956 return; 7957 } 7958 7959 _.each( wp.media.view.settings.tabs, function( title, id ) { 7960 views[ 'iframe:' + id ] = { 7961 text: this.state( 'iframe:' + id ).get('title'), 7962 priority: 200 7963 }; 7964 }, this ); 7965 7966 view.set( views ); 7967 }, 7968 7969 hijackThickbox: function() { 7970 var frame = this; 7971 7972 if ( ! window.tb_remove || this._tb_remove ) { 7973 return; 7974 } 7975 7976 this._tb_remove = window.tb_remove; 7977 window.tb_remove = function() { 7978 frame.close(); 7979 frame.reset(); 7980 frame.setState( frame.options.state ); 7981 frame._tb_remove.call( window ); 7982 }; 7983 }, 7984 7985 restoreThickbox: function() { 7986 if ( ! this._tb_remove ) { 7987 return; 7988 } 7989 7990 window.tb_remove = this._tb_remove; 7991 delete this._tb_remove; 7992 } 7993 }); 7994 7995 // Map some of the modal's methods to the frame. 7996 _.each(['open','close','attach','detach','escape'], function( method ) { 7997 /** 7998 * @function open 7999 * @memberOf wp.media.view.MediaFrame 8000 * @instance 8001 * 8002 * @return {wp.media.view.MediaFrame} Returns itself to allow chaining. 8003 */ 8004 /** 8005 * @function close 8006 * @memberOf wp.media.view.MediaFrame 8007 * @instance 8008 * 8009 * @return {wp.media.view.MediaFrame} Returns itself to allow chaining. 8010 */ 8011 /** 8012 * @function attach 8013 * @memberOf wp.media.view.MediaFrame 8014 * @instance 8015 * 8016 * @return {wp.media.view.MediaFrame} Returns itself to allow chaining. 8017 */ 8018 /** 8019 * @function detach 8020 * @memberOf wp.media.view.MediaFrame 8021 * @instance 8022 * 8023 * @return {wp.media.view.MediaFrame} Returns itself to allow chaining. 8024 */ 8025 /** 8026 * @function escape 8027 * @memberOf wp.media.view.MediaFrame 8028 * @instance 8029 * 8030 * @return {wp.media.view.MediaFrame} Returns itself to allow chaining. 8031 */ 8032 MediaFrame.prototype[ method ] = function() { 8033 if ( this.modal ) { 8034 this.modal[ method ].apply( this.modal, arguments ); 8035 } 8036 return this; 8037 }; 8038 }); 8039 8040 module.exports = MediaFrame; 8041 8042 8043 /***/ }), 8044 8045 /***/ 9013: 8046 /***/ ((module) => { 8047 8048 var MenuItem; 8049 8050 /** 8051 * wp.media.view.MenuItem 8052 * 8053 * @memberOf wp.media.view 8054 * 8055 * @class 8056 * @augments wp.media.View 8057 * @augments wp.Backbone.View 8058 * @augments Backbone.View 8059 */ 8060 MenuItem = wp.media.View.extend(/** @lends wp.media.view.MenuItem.prototype */{ 8061 tagName: 'button', 8062 className: 'media-menu-item', 8063 8064 attributes: { 8065 type: 'button', 8066 role: 'tab' 8067 }, 8068 8069 events: { 8070 'click': '_click' 8071 }, 8072 8073 /** 8074 * Allows to override the click event. 8075 */ 8076 _click: function() { 8077 var clickOverride = this.options.click; 8078 8079 if ( clickOverride ) { 8080 clickOverride.call( this ); 8081 } else { 8082 this.click(); 8083 } 8084 }, 8085 8086 click: function() { 8087 var state = this.options.state; 8088 8089 if ( state ) { 8090 this.controller.setState( state ); 8091 // Toggle the menu visibility in the responsive view. 8092 this.views.parent.$el.removeClass( 'visible' ); // @todo Or hide on any click, see below. 8093 } 8094 }, 8095 8096 /** 8097 * @return {wp.media.view.MenuItem} returns itself to allow chaining. 8098 */ 8099 render: function() { 8100 var options = this.options, 8101 menuProperty = options.state || options.contentMode; 8102 8103 if ( options.text ) { 8104 this.$el.text( options.text ); 8105 } else if ( options.html ) { 8106 this.$el.html( options.html ); 8107 } 8108 8109 // Set the menu item ID based on the frame state associated to the menu item. 8110 this.$el.attr( 'id', 'menu-item-' + menuProperty ); 8111 8112 return this; 8113 } 8114 }); 8115 8116 module.exports = MenuItem; 8117 8118 8119 /***/ }), 8120 8121 /***/ 1: 8122 /***/ ((module) => { 8123 8124 var MenuItem = wp.media.view.MenuItem, 8125 PriorityList = wp.media.view.PriorityList, 8126 Menu; 8127 8128 /** 8129 * wp.media.view.Menu 8130 * 8131 * @memberOf wp.media.view 8132 * 8133 * @class 8134 * @augments wp.media.view.PriorityList 8135 * @augments wp.media.View 8136 * @augments wp.Backbone.View 8137 * @augments Backbone.View 8138 */ 8139 Menu = PriorityList.extend(/** @lends wp.media.view.Menu.prototype */{ 8140 tagName: 'div', 8141 className: 'media-menu', 8142 property: 'state', 8143 ItemView: MenuItem, 8144 region: 'menu', 8145 8146 attributes: { 8147 role: 'tablist', 8148 'aria-orientation': 'horizontal' 8149 }, 8150 8151 initialize: function() { 8152 this._views = {}; 8153 8154 this.set( _.extend( {}, this._views, this.options.views ), { silent: true }); 8155 delete this.options.views; 8156 8157 if ( ! this.options.silent ) { 8158 this.render(); 8159 } 8160 8161 // Initialize the Focus Manager. 8162 this.focusManager = new wp.media.view.FocusManager( { 8163 el: this.el, 8164 mode: 'tabsNavigation' 8165 } ); 8166 8167 // The menu is always rendered and can be visible or hidden on some frames. 8168 this.isVisible = true; 8169 }, 8170 8171 /** 8172 * @param {Object} options 8173 * @param {string} id 8174 * @return {wp.media.View} 8175 */ 8176 toView: function( options, id ) { 8177 options = options || {}; 8178 options[ this.property ] = options[ this.property ] || id; 8179 return new this.ItemView( options ).render(); 8180 }, 8181 8182 ready: function() { 8183 /** 8184 * call 'ready' directly on the parent class 8185 */ 8186 PriorityList.prototype.ready.apply( this, arguments ); 8187 this.visibility(); 8188 8189 // Set up aria tabs initial attributes. 8190 this.focusManager.setupAriaTabs(); 8191 }, 8192 8193 set: function() { 8194 /** 8195 * call 'set' directly on the parent class 8196 */ 8197 PriorityList.prototype.set.apply( this, arguments ); 8198 this.visibility(); 8199 }, 8200 8201 unset: function() { 8202 /** 8203 * call 'unset' directly on the parent class 8204 */ 8205 PriorityList.prototype.unset.apply( this, arguments ); 8206 this.visibility(); 8207 }, 8208 8209 visibility: function() { 8210 var region = this.region, 8211 view = this.controller[ region ].get(), 8212 views = this.views.get(), 8213 hide = ! views || views.length < 2; 8214 8215 if ( this === view ) { 8216 // Flag this menu as hidden or visible. 8217 this.isVisible = ! hide; 8218 // Set or remove a CSS class to hide the menu. 8219 this.controller.$el.toggleClass( 'hide-' + region, hide ); 8220 } 8221 }, 8222 /** 8223 * @param {string} id 8224 */ 8225 select: function( id ) { 8226 var view = this.get( id ); 8227 8228 if ( ! view ) { 8229 return; 8230 } 8231 8232 this.deselect(); 8233 view.$el.addClass('active'); 8234 8235 // Set up again the aria tabs initial attributes after the menu updates. 8236 this.focusManager.setupAriaTabs(); 8237 }, 8238 8239 deselect: function() { 8240 this.$el.children().removeClass('active'); 8241 }, 8242 8243 hide: function( id ) { 8244 var view = this.get( id ); 8245 8246 if ( ! view ) { 8247 return; 8248 } 8249 8250 view.$el.addClass('hidden'); 8251 }, 8252 8253 show: function( id ) { 8254 var view = this.get( id ); 8255 8256 if ( ! view ) { 8257 return; 8258 } 8259 8260 view.$el.removeClass('hidden'); 8261 } 8262 }); 8263 8264 module.exports = Menu; 8265 8266 8267 /***/ }), 8268 8269 /***/ 2621: 8270 /***/ ((module) => { 8271 8272 var $ = jQuery, 8273 Modal; 8274 8275 /** 8276 * wp.media.view.Modal 8277 * 8278 * A modal view, which the media modal uses as its default container. 8279 * 8280 * @memberOf wp.media.view 8281 * 8282 * @class 8283 * @augments wp.media.View 8284 * @augments wp.Backbone.View 8285 * @augments Backbone.View 8286 */ 8287 Modal = wp.media.View.extend(/** @lends wp.media.view.Modal.prototype */{ 8288 tagName: 'div', 8289 template: wp.template('media-modal'), 8290 8291 events: { 8292 'click .media-modal-backdrop, .media-modal-close': 'escapeHandler', 8293 'keydown': 'keydown' 8294 }, 8295 8296 clickedOpenerEl: null, 8297 8298 initialize: function() { 8299 _.defaults( this.options, { 8300 container: document.body, 8301 title: '', 8302 propagate: true, 8303 hasCloseButton: true 8304 }); 8305 8306 this.focusManager = new wp.media.view.FocusManager({ 8307 el: this.el 8308 }); 8309 }, 8310 /** 8311 * @return {Object} 8312 */ 8313 prepare: function() { 8314 return { 8315 title: this.options.title, 8316 hasCloseButton: this.options.hasCloseButton 8317 }; 8318 }, 8319 8320 /** 8321 * @return {wp.media.view.Modal} Returns itself to allow chaining. 8322 */ 8323 attach: function() { 8324 if ( this.views.attached ) { 8325 return this; 8326 } 8327 8328 if ( ! this.views.rendered ) { 8329 this.render(); 8330 } 8331 8332 this.$el.appendTo( this.options.container ); 8333 8334 // Manually mark the view as attached and trigger ready. 8335 this.views.attached = true; 8336 this.views.ready(); 8337 8338 return this.propagate('attach'); 8339 }, 8340 8341 /** 8342 * @return {wp.media.view.Modal} Returns itself to allow chaining. 8343 */ 8344 detach: function() { 8345 if ( this.$el.is(':visible') ) { 8346 this.close(); 8347 } 8348 8349 this.$el.detach(); 8350 this.views.attached = false; 8351 return this.propagate('detach'); 8352 }, 8353 8354 /** 8355 * @return {wp.media.view.Modal} Returns itself to allow chaining. 8356 */ 8357 open: function() { 8358 var $el = this.$el, 8359 mceEditor; 8360 8361 if ( $el.is(':visible') ) { 8362 return this; 8363 } 8364 8365 this.clickedOpenerEl = document.activeElement; 8366 8367 if ( ! this.views.attached ) { 8368 this.attach(); 8369 } 8370 8371 // Disable page scrolling. 8372 $( 'body' ).addClass( 'modal-open' ); 8373 8374 $el.show(); 8375 8376 // Try to close the onscreen keyboard. 8377 if ( 'ontouchend' in document ) { 8378 if ( ( mceEditor = window.tinymce && window.tinymce.activeEditor ) && ! mceEditor.isHidden() && mceEditor.iframeElement ) { 8379 mceEditor.iframeElement.focus(); 8380 mceEditor.iframeElement.blur(); 8381 8382 setTimeout( function() { 8383 mceEditor.iframeElement.blur(); 8384 }, 100 ); 8385 } 8386 } 8387 8388 // Set initial focus on the content instead of this view element, to avoid page scrolling. 8389 this.$( '.media-modal' ).trigger( 'focus' ); 8390 8391 // Hide the page content from assistive technologies. 8392 this.focusManager.setAriaHiddenOnBodyChildren( $el ); 8393 8394 return this.propagate('open'); 8395 }, 8396 8397 /** 8398 * @param {Object} options 8399 * @return {wp.media.view.Modal} Returns itself to allow chaining. 8400 */ 8401 close: function( options ) { 8402 if ( ! this.views.attached || ! this.$el.is(':visible') ) { 8403 return this; 8404 } 8405 8406 // Pause current audio/video even after closing the modal. 8407 $( '.mejs-pause button' ).trigger( 'click' ); 8408 8409 // Enable page scrolling. 8410 $( 'body' ).removeClass( 'modal-open' ); 8411 8412 // Hide the modal element by adding display:none. 8413 this.$el.hide(); 8414 8415 /* 8416 * Make visible again to assistive technologies all body children that 8417 * have been made hidden when the modal opened. 8418 */ 8419 this.focusManager.removeAriaHiddenFromBodyChildren(); 8420 8421 // Move focus back in useful location once modal is closed. 8422 if ( null !== this.clickedOpenerEl ) { 8423 // Move focus back to the element that opened the modal. 8424 this.clickedOpenerEl.focus(); 8425 } else { 8426 // Fallback to the admin page main element. 8427 $( '#wpbody-content' ) 8428 .attr( 'tabindex', '-1' ) 8429 .trigger( 'focus' ); 8430 } 8431 8432 this.propagate('close'); 8433 8434 if ( options && options.escape ) { 8435 this.propagate('escape'); 8436 } 8437 8438 return this; 8439 }, 8440 /** 8441 * @return {wp.media.view.Modal} Returns itself to allow chaining. 8442 */ 8443 escape: function() { 8444 return this.close({ escape: true }); 8445 }, 8446 /** 8447 * @param {Object} event 8448 */ 8449 escapeHandler: function( event ) { 8450 event.preventDefault(); 8451 this.escape(); 8452 }, 8453 8454 /** 8455 * Handles the selection of attachments when the command or control key is pressed with the enter key. 8456 * 8457 * @since 6.7 8458 * 8459 * @param {Object} event The keydown event object. 8460 */ 8461 selectHandler: function( event ) { 8462 var selection = this.controller.state().get( 'selection' ); 8463 8464 if ( selection.length <= 0 ) { 8465 return; 8466 } 8467 8468 if ( 'insert' === this.controller.options.state ) { 8469 this.controller.trigger( 'insert', selection ); 8470 } else { 8471 this.controller.trigger( 'select', selection ); 8472 event.preventDefault(); 8473 this.escape(); 8474 } 8475 }, 8476 8477 /** 8478 * @param {Array|Object} content Views to register to '.media-modal-content' 8479 * @return {wp.media.view.Modal} Returns itself to allow chaining. 8480 */ 8481 content: function( content ) { 8482 this.views.set( '.media-modal-content', content ); 8483 return this; 8484 }, 8485 8486 /** 8487 * Triggers a modal event and if the `propagate` option is set, 8488 * forwards events to the modal's controller. 8489 * 8490 * @param {string} id 8491 * @return {wp.media.view.Modal} Returns itself to allow chaining. 8492 */ 8493 propagate: function( id ) { 8494 this.trigger( id ); 8495 8496 if ( this.options.propagate ) { 8497 this.controller.trigger( id ); 8498 } 8499 8500 return this; 8501 }, 8502 /** 8503 * @param {Object} event 8504 */ 8505 keydown: function( event ) { 8506 // Close the modal when escape is pressed. 8507 if ( 27 === event.which && this.$el.is(':visible') ) { 8508 this.escape(); 8509 event.stopImmediatePropagation(); 8510 } 8511 8512 // Select the attachment when command or control and enter are pressed. 8513 if ( ( 13 === event.which || 10 === event.which ) && ( event.metaKey || event.ctrlKey ) ) { 8514 this.selectHandler( event ); 8515 event.stopImmediatePropagation(); 8516 } 8517 8518 } 8519 }); 8520 8521 module.exports = Modal; 8522 8523 8524 /***/ }), 8525 8526 /***/ 8815: 8527 /***/ ((module) => { 8528 8529 /** 8530 * wp.media.view.PriorityList 8531 * 8532 * @memberOf wp.media.view 8533 * 8534 * @class 8535 * @augments wp.media.View 8536 * @augments wp.Backbone.View 8537 * @augments Backbone.View 8538 */ 8539 var PriorityList = wp.media.View.extend(/** @lends wp.media.view.PriorityList.prototype */{ 8540 tagName: 'div', 8541 8542 initialize: function() { 8543 this._views = {}; 8544 8545 this.set( _.extend( {}, this._views, this.options.views ), { silent: true }); 8546 delete this.options.views; 8547 8548 if ( ! this.options.silent ) { 8549 this.render(); 8550 } 8551 }, 8552 /** 8553 * @param {string} id 8554 * @param {wp.media.View|Object} view 8555 * @param {Object} options 8556 * @return {wp.media.view.PriorityList} Returns itself to allow chaining. 8557 */ 8558 set: function( id, view, options ) { 8559 var priority, views, index; 8560 8561 options = options || {}; 8562 8563 // Accept an object with an `id` : `view` mapping. 8564 if ( _.isObject( id ) ) { 8565 _.each( id, function( view, id ) { 8566 this.set( id, view ); 8567 }, this ); 8568 return this; 8569 } 8570 8571 if ( ! (view instanceof Backbone.View) ) { 8572 view = this.toView( view, id, options ); 8573 } 8574 view.controller = view.controller || this.controller; 8575 8576 this.unset( id ); 8577 8578 priority = view.options.priority || 10; 8579 views = this.views.get() || []; 8580 8581 _.find( views, function( existing, i ) { 8582 if ( existing.options.priority > priority ) { 8583 index = i; 8584 return true; 8585 } 8586 }); 8587 8588 this._views[ id ] = view; 8589 this.views.add( view, { 8590 at: _.isNumber( index ) ? index : views.length || 0 8591 }); 8592 8593 return this; 8594 }, 8595 /** 8596 * @param {string} id 8597 * @return {wp.media.View} 8598 */ 8599 get: function( id ) { 8600 return this._views[ id ]; 8601 }, 8602 /** 8603 * @param {string} id 8604 * @return {wp.media.view.PriorityList} 8605 */ 8606 unset: function( id ) { 8607 var view = this.get( id ); 8608 8609 if ( view ) { 8610 view.remove(); 8611 } 8612 8613 delete this._views[ id ]; 8614 return this; 8615 }, 8616 /** 8617 * @param {Object} options 8618 * @return {wp.media.View} 8619 */ 8620 toView: function( options ) { 8621 return new wp.media.View( options ); 8622 } 8623 }); 8624 8625 module.exports = PriorityList; 8626 8627 8628 /***/ }), 8629 8630 /***/ 6327: 8631 /***/ ((module) => { 8632 8633 /** 8634 * wp.media.view.RouterItem 8635 * 8636 * @memberOf wp.media.view 8637 * 8638 * @class 8639 * @augments wp.media.view.MenuItem 8640 * @augments wp.media.View 8641 * @augments wp.Backbone.View 8642 * @augments Backbone.View 8643 */ 8644 var RouterItem = wp.media.view.MenuItem.extend(/** @lends wp.media.view.RouterItem.prototype */{ 8645 /** 8646 * On click handler to activate the content region's corresponding mode. 8647 */ 8648 click: function() { 8649 var contentMode = this.options.contentMode; 8650 if ( contentMode ) { 8651 this.controller.content.mode( contentMode ); 8652 } 8653 } 8654 }); 8655 8656 module.exports = RouterItem; 8657 8658 8659 /***/ }), 8660 8661 /***/ 4783: 8662 /***/ ((module) => { 8663 8664 var Menu = wp.media.view.Menu, 8665 Router; 8666 8667 /** 8668 * wp.media.view.Router 8669 * 8670 * @memberOf wp.media.view 8671 * 8672 * @class 8673 * @augments wp.media.view.Menu 8674 * @augments wp.media.view.PriorityList 8675 * @augments wp.media.View 8676 * @augments wp.Backbone.View 8677 * @augments Backbone.View 8678 */ 8679 Router = Menu.extend(/** @lends wp.media.view.Router.prototype */{ 8680 tagName: 'div', 8681 className: 'media-router', 8682 property: 'contentMode', 8683 ItemView: wp.media.view.RouterItem, 8684 region: 'router', 8685 8686 attributes: { 8687 role: 'tablist', 8688 'aria-orientation': 'horizontal' 8689 }, 8690 8691 initialize: function() { 8692 this.controller.on( 'content:render', this.update, this ); 8693 // Call 'initialize' directly on the parent class. 8694 Menu.prototype.initialize.apply( this, arguments ); 8695 }, 8696 8697 update: function() { 8698 var mode = this.controller.content.mode(); 8699 if ( mode ) { 8700 this.select( mode ); 8701 } 8702 } 8703 }); 8704 8705 module.exports = Router; 8706 8707 8708 /***/ }), 8709 8710 /***/ 2102: 8711 /***/ ((module) => { 8712 8713 var Search; 8714 8715 /** 8716 * wp.media.view.Search 8717 * 8718 * @memberOf wp.media.view 8719 * 8720 * @class 8721 * @augments wp.media.View 8722 * @augments wp.Backbone.View 8723 * @augments Backbone.View 8724 */ 8725 Search = wp.media.View.extend(/** @lends wp.media.view.Search.prototype */{ 8726 tagName: 'input', 8727 className: 'search', 8728 id: 'media-search-input', 8729 8730 attributes: { 8731 type: 'search' 8732 }, 8733 8734 events: { 8735 'input': 'search' 8736 }, 8737 8738 /** 8739 * @return {wp.media.view.Search} Returns itself to allow chaining. 8740 */ 8741 render: function() { 8742 this.el.value = this.model.escape('search'); 8743 return this; 8744 }, 8745 8746 search: _.debounce( function( event ) { 8747 var searchTerm = event.target.value.trim(); 8748 8749 // Trigger the search only after 2 ASCII characters. 8750 if ( searchTerm && searchTerm.length > 1 ) { 8751 this.model.set( 'search', searchTerm ); 8752 } else { 8753 this.model.unset( 'search' ); 8754 } 8755 }, 500 ) 8756 }); 8757 8758 module.exports = Search; 8759 8760 8761 /***/ }), 8762 8763 /***/ 8282: 8764 /***/ ((module) => { 8765 8766 var _n = wp.i18n._n, 8767 sprintf = wp.i18n.sprintf, 8768 Selection; 8769 8770 /** 8771 * wp.media.view.Selection 8772 * 8773 * @memberOf wp.media.view 8774 * 8775 * @class 8776 * @augments wp.media.View 8777 * @augments wp.Backbone.View 8778 * @augments Backbone.View 8779 */ 8780 Selection = wp.media.View.extend(/** @lends wp.media.view.Selection.prototype */{ 8781 tagName: 'div', 8782 className: 'media-selection', 8783 template: wp.template('media-selection'), 8784 8785 events: { 8786 'click .edit-selection': 'edit', 8787 'click .clear-selection': 'clear' 8788 }, 8789 8790 initialize: function() { 8791 _.defaults( this.options, { 8792 editable: false, 8793 clearable: true 8794 }); 8795 8796 /** 8797 * @member {wp.media.view.Attachments.Selection} 8798 */ 8799 this.attachments = new wp.media.view.Attachments.Selection({ 8800 controller: this.controller, 8801 collection: this.collection, 8802 selection: this.collection, 8803 model: new Backbone.Model() 8804 }); 8805 8806 this.views.set( '.selection-view', this.attachments ); 8807 this.collection.on( 'add remove reset', this.refresh, this ); 8808 this.controller.on( 'content:activate', this.refresh, this ); 8809 }, 8810 8811 ready: function() { 8812 this.refresh(); 8813 }, 8814 8815 refresh: function() { 8816 // If the selection hasn't been rendered, bail. 8817 if ( ! this.$el.children().length ) { 8818 return; 8819 } 8820 8821 var collection = this.collection, 8822 editing = 'edit-selection' === this.controller.content.mode(); 8823 8824 // If nothing is selected, display nothing. 8825 this.$el.toggleClass( 'empty', ! collection.length ); 8826 this.$el.toggleClass( 'one', 1 === collection.length ); 8827 this.$el.toggleClass( 'editing', editing ); 8828 8829 this.$( '.count' ).text( 8830 /* translators: %s: Number of selected media attachments. */ 8831 sprintf( _n( '%s item selected', '%s items selected', collection.length ), collection.length ) 8832 ); 8833 }, 8834 8835 edit: function( event ) { 8836 event.preventDefault(); 8837 if ( this.options.editable ) { 8838 this.options.editable.call( this, this.collection ); 8839 } 8840 }, 8841 8842 clear: function( event ) { 8843 event.preventDefault(); 8844 this.collection.reset(); 8845 8846 // Move focus to the modal. 8847 this.controller.modal.focusManager.focus(); 8848 } 8849 }); 8850 8851 module.exports = Selection; 8852 8853 8854 /***/ }), 8855 8856 /***/ 1915: 8857 /***/ ((module) => { 8858 8859 var View = wp.media.View, 8860 $ = Backbone.$, 8861 Settings; 8862 8863 /** 8864 * wp.media.view.Settings 8865 * 8866 * @memberOf wp.media.view 8867 * 8868 * @class 8869 * @augments wp.media.View 8870 * @augments wp.Backbone.View 8871 * @augments Backbone.View 8872 */ 8873 Settings = View.extend(/** @lends wp.media.view.Settings.prototype */{ 8874 events: { 8875 'click button': 'updateHandler', 8876 'change input': 'updateHandler', 8877 'change select': 'updateHandler', 8878 'change textarea': 'updateHandler' 8879 }, 8880 8881 initialize: function() { 8882 this.model = this.model || new Backbone.Model(); 8883 this.listenTo( this.model, 'change', this.updateChanges ); 8884 }, 8885 8886 prepare: function() { 8887 return _.defaults({ 8888 model: this.model.toJSON() 8889 }, this.options ); 8890 }, 8891 /** 8892 * @return {wp.media.view.Settings} Returns itself to allow chaining. 8893 */ 8894 render: function() { 8895 View.prototype.render.apply( this, arguments ); 8896 // Select the correct values. 8897 _( this.model.attributes ).chain().keys().each( this.update, this ); 8898 return this; 8899 }, 8900 /** 8901 * @param {string} key 8902 */ 8903 update: function( key ) { 8904 var value = this.model.get( key ), 8905 $setting = this.$('[data-setting="' + key + '"]'), 8906 $buttons, $value; 8907 8908 // Bail if we didn't find a matching setting. 8909 if ( ! $setting.length ) { 8910 return; 8911 } 8912 8913 // Attempt to determine how the setting is rendered and update 8914 // the selected value. 8915 8916 // Handle dropdowns. 8917 if ( $setting.is('select') ) { 8918 $value = $setting.find('[value="' + value + '"]'); 8919 8920 if ( $value.length ) { 8921 $setting.find('option').prop( 'selected', false ); 8922 $value.prop( 'selected', true ); 8923 } else { 8924 // If we can't find the desired value, record what *is* selected. 8925 this.model.set( key, $setting.find(':selected').val() ); 8926 } 8927 8928 // Handle button groups. 8929 } else if ( $setting.hasClass('button-group') ) { 8930 $buttons = $setting.find( 'button' ) 8931 .removeClass( 'active' ) 8932 .attr( 'aria-pressed', 'false' ); 8933 $buttons.filter( '[value="' + value + '"]' ) 8934 .addClass( 'active' ) 8935 .attr( 'aria-pressed', 'true' ); 8936 8937 // Handle text inputs and textareas. 8938 } else if ( $setting.is('input[type="text"], textarea') ) { 8939 if ( ! $setting.is(':focus') ) { 8940 $setting.val( value ); 8941 } 8942 // Handle checkboxes. 8943 } else if ( $setting.is('input[type="checkbox"]') ) { 8944 $setting.prop( 'checked', !! value && 'false' !== value ); 8945 } 8946 }, 8947 /** 8948 * @param {Object} event 8949 */ 8950 updateHandler: function( event ) { 8951 var $setting = $( event.target ).closest('[data-setting]'), 8952 value = event.target.value, 8953 userSetting; 8954 8955 event.preventDefault(); 8956 8957 if ( ! $setting.length ) { 8958 return; 8959 } 8960 8961 // Use the correct value for checkboxes. 8962 if ( $setting.is('input[type="checkbox"]') ) { 8963 value = $setting[0].checked; 8964 } 8965 8966 // Update the corresponding setting. 8967 this.model.set( $setting.data('setting'), value ); 8968 8969 // If the setting has a corresponding user setting, 8970 // update that as well. 8971 userSetting = $setting.data('userSetting'); 8972 if ( userSetting ) { 8973 window.setUserSetting( userSetting, value ); 8974 } 8975 }, 8976 8977 updateChanges: function( model ) { 8978 if ( model.hasChanged() ) { 8979 _( model.changed ).chain().keys().each( this.update, this ); 8980 } 8981 } 8982 }); 8983 8984 module.exports = Settings; 8985 8986 8987 /***/ }), 8988 8989 /***/ 7656: 8990 /***/ ((module) => { 8991 8992 var Settings = wp.media.view.Settings, 8993 AttachmentDisplay; 8994 8995 /** 8996 * wp.media.view.Settings.AttachmentDisplay 8997 * 8998 * @memberOf wp.media.view.Settings 8999 * 9000 * @class 9001 * @augments wp.media.view.Settings 9002 * @augments wp.media.View 9003 * @augments wp.Backbone.View 9004 * @augments Backbone.View 9005 */ 9006 AttachmentDisplay = Settings.extend(/** @lends wp.media.view.Settings.AttachmentDisplay.prototype */{ 9007 className: 'attachment-display-settings', 9008 template: wp.template('attachment-display-settings'), 9009 9010 initialize: function() { 9011 var attachment = this.options.attachment; 9012 9013 _.defaults( this.options, { 9014 userSettings: false 9015 }); 9016 // Call 'initialize' directly on the parent class. 9017 Settings.prototype.initialize.apply( this, arguments ); 9018 this.listenTo( this.model, 'change:link', this.updateLinkTo ); 9019 9020 if ( attachment ) { 9021 attachment.on( 'change:uploading', this.render, this ); 9022 } 9023 }, 9024 9025 dispose: function() { 9026 var attachment = this.options.attachment; 9027 if ( attachment ) { 9028 attachment.off( null, null, this ); 9029 } 9030 /** 9031 * call 'dispose' directly on the parent class 9032 */ 9033 Settings.prototype.dispose.apply( this, arguments ); 9034 }, 9035 /** 9036 * @return {wp.media.view.AttachmentDisplay} Returns itself to allow chaining. 9037 */ 9038 render: function() { 9039 var attachment = this.options.attachment; 9040 if ( attachment ) { 9041 _.extend( this.options, { 9042 sizes: attachment.get('sizes'), 9043 type: attachment.get('type') 9044 }); 9045 } 9046 /** 9047 * call 'render' directly on the parent class 9048 */ 9049 Settings.prototype.render.call( this ); 9050 this.updateLinkTo(); 9051 return this; 9052 }, 9053 9054 updateLinkTo: function() { 9055 var linkTo = this.model.get('link'), 9056 $input = this.$('.link-to-custom'), 9057 attachment = this.options.attachment; 9058 9059 if ( 'none' === linkTo || 'embed' === linkTo || ( ! attachment && 'custom' !== linkTo ) ) { 9060 $input.closest( '.setting' ).addClass( 'hidden' ); 9061 return; 9062 } 9063 9064 if ( attachment ) { 9065 if ( 'post' === linkTo ) { 9066 $input.val( attachment.get('link') ); 9067 } else if ( 'file' === linkTo ) { 9068 $input.val( attachment.get('url') ); 9069 } else if ( ! this.model.get('linkUrl') ) { 9070 $input.val('http://'); 9071 } 9072 9073 $input.prop( 'readonly', 'custom' !== linkTo ); 9074 } 9075 9076 $input.closest( '.setting' ).removeClass( 'hidden' ); 9077 if ( $input.length ) { 9078 $input[0].scrollIntoView(); 9079 } 9080 } 9081 }); 9082 9083 module.exports = AttachmentDisplay; 9084 9085 9086 /***/ }), 9087 9088 /***/ 7266: 9089 /***/ ((module) => { 9090 9091 /** 9092 * wp.media.view.Settings.Gallery 9093 * 9094 * @memberOf wp.media.view.Settings 9095 * 9096 * @class 9097 * @augments wp.media.view.Settings 9098 * @augments wp.media.View 9099 * @augments wp.Backbone.View 9100 * @augments Backbone.View 9101 */ 9102 var Gallery = wp.media.view.Settings.extend(/** @lends wp.media.view.Settings.Gallery.prototype */{ 9103 className: 'collection-settings gallery-settings', 9104 template: wp.template('gallery-settings') 9105 }); 9106 9107 module.exports = Gallery; 9108 9109 9110 /***/ }), 9111 9112 /***/ 2356: 9113 /***/ ((module) => { 9114 9115 /** 9116 * wp.media.view.Settings.Playlist 9117 * 9118 * @memberOf wp.media.view.Settings 9119 * 9120 * @class 9121 * @augments wp.media.view.Settings 9122 * @augments wp.media.View 9123 * @augments wp.Backbone.View 9124 * @augments Backbone.View 9125 */ 9126 var Playlist = wp.media.view.Settings.extend(/** @lends wp.media.view.Settings.Playlist.prototype */{ 9127 className: 'collection-settings playlist-settings', 9128 template: wp.template('playlist-settings') 9129 }); 9130 9131 module.exports = Playlist; 9132 9133 9134 /***/ }), 9135 9136 /***/ 1992: 9137 /***/ ((module) => { 9138 9139 /** 9140 * wp.media.view.Sidebar 9141 * 9142 * @memberOf wp.media.view 9143 * 9144 * @class 9145 * @augments wp.media.view.PriorityList 9146 * @augments wp.media.View 9147 * @augments wp.Backbone.View 9148 * @augments Backbone.View 9149 */ 9150 var Sidebar = wp.media.view.PriorityList.extend(/** @lends wp.media.view.Sidebar.prototype */{ 9151 className: 'media-sidebar' 9152 }); 9153 9154 module.exports = Sidebar; 9155 9156 9157 /***/ }), 9158 9159 /***/ 443: 9160 /***/ ((module) => { 9161 9162 var View = wp.media.view, 9163 SiteIconCropper; 9164 9165 /** 9166 * wp.media.view.SiteIconCropper 9167 * 9168 * Uses the imgAreaSelect plugin to allow a user to crop a Site Icon. 9169 * 9170 * Takes imgAreaSelect options from 9171 * wp.customize.SiteIconControl.calculateImageSelectOptions. 9172 * 9173 * @memberOf wp.media.view 9174 * 9175 * @class 9176 * @augments wp.media.view.Cropper 9177 * @augments wp.media.View 9178 * @augments wp.Backbone.View 9179 * @augments Backbone.View 9180 */ 9181 SiteIconCropper = View.Cropper.extend(/** @lends wp.media.view.SiteIconCropper.prototype */{ 9182 className: 'crop-content site-icon', 9183 9184 ready: function () { 9185 View.Cropper.prototype.ready.apply( this, arguments ); 9186 9187 this.$( '.crop-image' ).on( 'load', _.bind( this.addSidebar, this ) ); 9188 }, 9189 9190 addSidebar: function() { 9191 this.sidebar = new wp.media.view.Sidebar({ 9192 controller: this.controller 9193 }); 9194 9195 this.sidebar.set( 'preview', new wp.media.view.SiteIconPreview({ 9196 controller: this.controller, 9197 attachment: this.options.attachment 9198 }) ); 9199 9200 this.controller.cropperView.views.add( this.sidebar ); 9201 } 9202 }); 9203 9204 module.exports = SiteIconCropper; 9205 9206 9207 /***/ }), 9208 9209 /***/ 7810: 9210 /***/ ((module) => { 9211 9212 var View = wp.media.View, 9213 $ = jQuery, 9214 SiteIconPreview; 9215 9216 /** 9217 * wp.media.view.SiteIconPreview 9218 * 9219 * Shows a preview of the Site Icon as a favicon and app icon while cropping. 9220 * 9221 * @memberOf wp.media.view 9222 * 9223 * @class 9224 * @augments wp.media.View 9225 * @augments wp.Backbone.View 9226 * @augments Backbone.View 9227 */ 9228 SiteIconPreview = View.extend(/** @lends wp.media.view.SiteIconPreview.prototype */{ 9229 className: 'site-icon-preview-crop-modal', 9230 template: wp.template( 'site-icon-preview-crop' ), 9231 9232 ready: function() { 9233 this.controller.imgSelect.setOptions({ 9234 onInit: this.updatePreview, 9235 onSelectChange: this.updatePreview 9236 }); 9237 }, 9238 9239 prepare: function() { 9240 return { 9241 url: this.options.attachment.get( 'url' ) 9242 }; 9243 }, 9244 9245 updatePreview: function( img, coords ) { 9246 var rx = 64 / coords.width, 9247 ry = 64 / coords.height, 9248 preview_rx = 24 / coords.width, 9249 preview_ry = 24 / coords.height; 9250 9251 $( '#preview-app-icon' ).css({ 9252 width: Math.round(rx * this.imageWidth ) + 'px', 9253 height: Math.round(ry * this.imageHeight ) + 'px', 9254 marginLeft: '-' + Math.round(rx * coords.x1) + 'px', 9255 marginTop: '-' + Math.round(ry * coords.y1) + 'px' 9256 }); 9257 9258 $( '#preview-favicon' ).css({ 9259 width: Math.round( preview_rx * this.imageWidth ) + 'px', 9260 height: Math.round( preview_ry * this.imageHeight ) + 'px', 9261 marginLeft: '-' + Math.round( preview_rx * coords.x1 ) + 'px', 9262 marginTop: '-' + Math.floor( preview_ry* coords.y1 ) + 'px' 9263 }); 9264 } 9265 }); 9266 9267 module.exports = SiteIconPreview; 9268 9269 9270 /***/ }), 9271 9272 /***/ 9141: 9273 /***/ ((module) => { 9274 9275 /** 9276 * wp.media.view.Spinner 9277 * 9278 * Represents a spinner in the Media Library. 9279 * 9280 * @since 3.9.0 9281 * 9282 * @memberOf wp.media.view 9283 * 9284 * @class 9285 * @augments wp.media.View 9286 * @augments wp.Backbone.View 9287 * @augments Backbone.View 9288 */ 9289 var Spinner = wp.media.View.extend(/** @lends wp.media.view.Spinner.prototype */{ 9290 tagName: 'span', 9291 className: 'spinner', 9292 spinnerTimeout: false, 9293 delay: 400, 9294 9295 /** 9296 * Shows the spinner. Delays the visibility by the configured amount. 9297 * 9298 * @since 3.9.0 9299 * 9300 * @return {wp.media.view.Spinner} The spinner. 9301 */ 9302 show: function() { 9303 if ( ! this.spinnerTimeout ) { 9304 this.spinnerTimeout = _.delay(function( $el ) { 9305 $el.addClass( 'is-active' ); 9306 }, this.delay, this.$el ); 9307 } 9308 9309 return this; 9310 }, 9311 9312 /** 9313 * Hides the spinner. 9314 * 9315 * @since 3.9.0 9316 * 9317 * @return {wp.media.view.Spinner} The spinner. 9318 */ 9319 hide: function() { 9320 this.$el.removeClass( 'is-active' ); 9321 this.spinnerTimeout = clearTimeout( this.spinnerTimeout ); 9322 9323 return this; 9324 } 9325 }); 9326 9327 module.exports = Spinner; 9328 9329 9330 /***/ }), 9331 9332 /***/ 5275: 9333 /***/ ((module) => { 9334 9335 var View = wp.media.View, 9336 Toolbar; 9337 9338 /** 9339 * wp.media.view.Toolbar 9340 * 9341 * A toolbar which consists of a primary and a secondary section. Each sections 9342 * can be filled with views. 9343 * 9344 * @memberOf wp.media.view 9345 * 9346 * @class 9347 * @augments wp.media.View 9348 * @augments wp.Backbone.View 9349 * @augments Backbone.View 9350 */ 9351 Toolbar = View.extend(/** @lends wp.media.view.Toolbar.prototype */{ 9352 tagName: 'div', 9353 className: 'media-toolbar', 9354 9355 initialize: function() { 9356 var state = this.controller.state(), 9357 selection = this.selection = state.get('selection'), 9358 library = this.library = state.get('library'); 9359 9360 this._views = {}; 9361 9362 // The toolbar is composed of two `PriorityList` views. 9363 this.primary = new wp.media.view.PriorityList(); 9364 this.secondary = new wp.media.view.PriorityList(); 9365 this.tertiary = new wp.media.view.PriorityList(); 9366 this.primary.$el.addClass('media-toolbar-primary search-form'); 9367 this.secondary.$el.addClass('media-toolbar-secondary'); 9368 this.tertiary.$el.addClass('media-bg-overlay'); 9369 9370 this.views.set([ this.secondary, this.primary, this.tertiary ]); 9371 9372 if ( this.options.items ) { 9373 this.set( this.options.items, { silent: true }); 9374 } 9375 9376 if ( ! this.options.silent ) { 9377 this.render(); 9378 } 9379 9380 if ( selection ) { 9381 selection.on( 'add remove reset', this.refresh, this ); 9382 } 9383 9384 if ( library ) { 9385 library.on( 'add remove reset', this.refresh, this ); 9386 } 9387 }, 9388 /** 9389 * @return {wp.media.view.Toolbar} Returns itself to allow chaining 9390 */ 9391 dispose: function() { 9392 if ( this.selection ) { 9393 this.selection.off( null, null, this ); 9394 } 9395 9396 if ( this.library ) { 9397 this.library.off( null, null, this ); 9398 } 9399 /** 9400 * call 'dispose' directly on the parent class 9401 */ 9402 return View.prototype.dispose.apply( this, arguments ); 9403 }, 9404 9405 ready: function() { 9406 this.refresh(); 9407 }, 9408 9409 /** 9410 * @param {string} id 9411 * @param {Backbone.View|Object} view 9412 * @param {Object} [options={}] 9413 * @return {wp.media.view.Toolbar} Returns itself to allow chaining. 9414 */ 9415 set: function( id, view, options ) { 9416 var list; 9417 options = options || {}; 9418 9419 // Accept an object with an `id` : `view` mapping. 9420 if ( _.isObject( id ) ) { 9421 _.each( id, function( view, id ) { 9422 this.set( id, view, { silent: true }); 9423 }, this ); 9424 9425 } else { 9426 if ( ! ( view instanceof Backbone.View ) ) { 9427 view.classes = [ 'media-button-' + id ].concat( view.classes || [] ); 9428 view = new wp.media.view.Button( view ).render(); 9429 } 9430 9431 view.controller = view.controller || this.controller; 9432 9433 this._views[ id ] = view; 9434 9435 list = view.options.priority < 0 ? 'secondary' : 'primary'; 9436 this[ list ].set( id, view, options ); 9437 } 9438 9439 if ( ! options.silent ) { 9440 this.refresh(); 9441 } 9442 9443 return this; 9444 }, 9445 /** 9446 * @param {string} id 9447 * @return {wp.media.view.Button} 9448 */ 9449 get: function( id ) { 9450 return this._views[ id ]; 9451 }, 9452 /** 9453 * @param {string} id 9454 * @param {Object} options 9455 * @return {wp.media.view.Toolbar} Returns itself to allow chaining. 9456 */ 9457 unset: function( id, options ) { 9458 delete this._views[ id ]; 9459 this.primary.unset( id, options ); 9460 this.secondary.unset( id, options ); 9461 this.tertiary.unset( id, options ); 9462 9463 if ( ! options || ! options.silent ) { 9464 this.refresh(); 9465 } 9466 return this; 9467 }, 9468 9469 refresh: function() { 9470 var state = this.controller.state(), 9471 library = state.get('library'), 9472 selection = state.get('selection'); 9473 9474 _.each( this._views, function( button ) { 9475 if ( ! button.model || ! button.options || ! button.options.requires ) { 9476 return; 9477 } 9478 9479 var requires = button.options.requires, 9480 disabled = false; 9481 9482 // Prevent insertion of attachments if any of them are still uploading. 9483 if ( selection && selection.models ) { 9484 disabled = _.some( selection.models, function( attachment ) { 9485 return attachment.get('uploading') === true; 9486 }); 9487 } 9488 9489 if ( requires.selection && selection && ! selection.length ) { 9490 disabled = true; 9491 } else if ( requires.library && library && ! library.length ) { 9492 disabled = true; 9493 } 9494 button.model.set( 'disabled', disabled ); 9495 }); 9496 } 9497 }); 9498 9499 module.exports = Toolbar; 9500 9501 9502 /***/ }), 9503 9504 /***/ 397: 9505 /***/ ((module) => { 9506 9507 var Select = wp.media.view.Toolbar.Select, 9508 l10n = wp.media.view.l10n, 9509 Embed; 9510 9511 /** 9512 * wp.media.view.Toolbar.Embed 9513 * 9514 * @memberOf wp.media.view.Toolbar 9515 * 9516 * @class 9517 * @augments wp.media.view.Toolbar.Select 9518 * @augments wp.media.view.Toolbar 9519 * @augments wp.media.View 9520 * @augments wp.Backbone.View 9521 * @augments Backbone.View 9522 */ 9523 Embed = Select.extend(/** @lends wp.media.view.Toolbar.Embed.prototype */{ 9524 initialize: function() { 9525 _.defaults( this.options, { 9526 text: l10n.insertIntoPost, 9527 requires: false 9528 }); 9529 // Call 'initialize' directly on the parent class. 9530 Select.prototype.initialize.apply( this, arguments ); 9531 }, 9532 9533 refresh: function() { 9534 var url = this.controller.state().props.get('url'); 9535 this.get('select').model.set( 'disabled', ! url || url === 'http://' ); 9536 /** 9537 * call 'refresh' directly on the parent class 9538 */ 9539 Select.prototype.refresh.apply( this, arguments ); 9540 } 9541 }); 9542 9543 module.exports = Embed; 9544 9545 9546 /***/ }), 9547 9548 /***/ 9458: 9549 /***/ ((module) => { 9550 9551 var Toolbar = wp.media.view.Toolbar, 9552 l10n = wp.media.view.l10n, 9553 Select; 9554 9555 /** 9556 * wp.media.view.Toolbar.Select 9557 * 9558 * @memberOf wp.media.view.Toolbar 9559 * 9560 * @class 9561 * @augments wp.media.view.Toolbar 9562 * @augments wp.media.View 9563 * @augments wp.Backbone.View 9564 * @augments Backbone.View 9565 */ 9566 Select = Toolbar.extend(/** @lends wp.media.view.Toolbar.Select.prototype */{ 9567 initialize: function() { 9568 var options = this.options; 9569 9570 _.bindAll( this, 'clickSelect' ); 9571 9572 _.defaults( options, { 9573 event: 'select', 9574 state: false, 9575 reset: true, 9576 close: true, 9577 text: l10n.select, 9578 9579 // Does the button rely on the selection? 9580 requires: { 9581 selection: true 9582 } 9583 }); 9584 9585 options.items = _.defaults( options.items || {}, { 9586 select: { 9587 style: 'primary', 9588 text: options.text, 9589 priority: 80, 9590 click: this.clickSelect, 9591 requires: options.requires 9592 } 9593 }); 9594 // Call 'initialize' directly on the parent class. 9595 Toolbar.prototype.initialize.apply( this, arguments ); 9596 }, 9597 9598 clickSelect: function() { 9599 var options = this.options, 9600 controller = this.controller; 9601 9602 if ( options.close ) { 9603 controller.close(); 9604 } 9605 9606 if ( options.event ) { 9607 controller.state().trigger( options.event ); 9608 } 9609 9610 if ( options.state ) { 9611 controller.setState( options.state ); 9612 } 9613 9614 if ( options.reset ) { 9615 controller.reset(); 9616 } 9617 } 9618 }); 9619 9620 module.exports = Select; 9621 9622 9623 /***/ }), 9624 9625 /***/ 3674: 9626 /***/ ((module) => { 9627 9628 var View = wp.media.View, 9629 l10n = wp.media.view.l10n, 9630 $ = jQuery, 9631 EditorUploader; 9632 9633 /** 9634 * Creates a dropzone on WP editor instances (elements with .wp-editor-wrap) 9635 * and relays drag'n'dropped files to a media workflow. 9636 * 9637 * wp.media.view.EditorUploader 9638 * 9639 * @memberOf wp.media.view 9640 * 9641 * @class 9642 * @augments wp.media.View 9643 * @augments wp.Backbone.View 9644 * @augments Backbone.View 9645 */ 9646 EditorUploader = View.extend(/** @lends wp.media.view.EditorUploader.prototype */{ 9647 tagName: 'div', 9648 className: 'uploader-editor', 9649 template: wp.template( 'uploader-editor' ), 9650 9651 localDrag: false, 9652 overContainer: false, 9653 overDropzone: false, 9654 draggingFile: null, 9655 9656 /** 9657 * Bind drag'n'drop events to callbacks. 9658 */ 9659 initialize: function() { 9660 this.initialized = false; 9661 9662 // Bail if not enabled or UA does not support drag'n'drop or File API. 9663 if ( ! window.tinyMCEPreInit || ! window.tinyMCEPreInit.dragDropUpload || ! this.browserSupport() ) { 9664 return this; 9665 } 9666 9667 this.$document = $(document); 9668 this.dropzones = []; 9669 this.files = []; 9670 9671 this.$document.on( 'drop', '.uploader-editor', _.bind( this.drop, this ) ); 9672 this.$document.on( 'dragover', '.uploader-editor', _.bind( this.dropzoneDragover, this ) ); 9673 this.$document.on( 'dragleave', '.uploader-editor', _.bind( this.dropzoneDragleave, this ) ); 9674 this.$document.on( 'click', '.uploader-editor', _.bind( this.click, this ) ); 9675 9676 this.$document.on( 'dragover', _.bind( this.containerDragover, this ) ); 9677 this.$document.on( 'dragleave', _.bind( this.containerDragleave, this ) ); 9678 9679 this.$document.on( 'dragstart dragend drop', _.bind( function( event ) { 9680 this.localDrag = event.type === 'dragstart'; 9681 9682 if ( event.type === 'drop' ) { 9683 this.containerDragleave(); 9684 } 9685 }, this ) ); 9686 9687 this.initialized = true; 9688 return this; 9689 }, 9690 9691 /** 9692 * Check browser support for drag'n'drop. 9693 * 9694 * @return {boolean} 9695 */ 9696 browserSupport: function() { 9697 var supports = false, div = document.createElement('div'); 9698 9699 supports = ( 'draggable' in div ) || ( 'ondragstart' in div && 'ondrop' in div ); 9700 supports = supports && !! ( window.File && window.FileList && window.FileReader ); 9701 return supports; 9702 }, 9703 9704 isDraggingFile: function( event ) { 9705 if ( this.draggingFile !== null ) { 9706 return this.draggingFile; 9707 } 9708 9709 if ( _.isUndefined( event.originalEvent ) || _.isUndefined( event.originalEvent.dataTransfer ) ) { 9710 return false; 9711 } 9712 9713 this.draggingFile = _.indexOf( event.originalEvent.dataTransfer.types, 'Files' ) > -1 && 9714 _.indexOf( event.originalEvent.dataTransfer.types, 'text/plain' ) === -1; 9715 9716 return this.draggingFile; 9717 }, 9718 9719 refresh: function( e ) { 9720 var dropzone_id; 9721 for ( dropzone_id in this.dropzones ) { 9722 // Hide the dropzones only if dragging has left the screen. 9723 this.dropzones[ dropzone_id ].toggle( this.overContainer || this.overDropzone ); 9724 } 9725 9726 if ( ! _.isUndefined( e ) ) { 9727 $( e.target ).closest( '.uploader-editor' ).toggleClass( 'droppable', this.overDropzone ); 9728 } 9729 9730 if ( ! this.overContainer && ! this.overDropzone ) { 9731 this.draggingFile = null; 9732 } 9733 9734 return this; 9735 }, 9736 9737 render: function() { 9738 if ( ! this.initialized ) { 9739 return this; 9740 } 9741 9742 View.prototype.render.apply( this, arguments ); 9743 $( '.wp-editor-wrap' ).each( _.bind( this.attach, this ) ); 9744 return this; 9745 }, 9746 9747 attach: function( index, editor ) { 9748 // Attach a dropzone to an editor. 9749 var dropzone = this.$el.clone(); 9750 this.dropzones.push( dropzone ); 9751 $( editor ).append( dropzone ); 9752 return this; 9753 }, 9754 9755 /** 9756 * When a file is dropped on the editor uploader, open up an editor media workflow 9757 * and upload the file immediately. 9758 * 9759 * @param {jQuery.Event} event The 'drop' event. 9760 */ 9761 drop: function( event ) { 9762 var $wrap, uploadView; 9763 9764 this.containerDragleave( event ); 9765 this.dropzoneDragleave( event ); 9766 9767 this.files = event.originalEvent.dataTransfer.files; 9768 if ( this.files.length < 1 ) { 9769 return; 9770 } 9771 9772 // Set the active editor to the drop target. 9773 $wrap = $( event.target ).parents( '.wp-editor-wrap' ); 9774 if ( $wrap.length > 0 && $wrap[0].id ) { 9775 window.wpActiveEditor = $wrap[0].id.slice( 3, -5 ); 9776 } 9777 9778 if ( ! this.workflow ) { 9779 this.workflow = wp.media.editor.open( window.wpActiveEditor, { 9780 frame: 'post', 9781 state: 'insert', 9782 title: l10n.addMedia, 9783 multiple: true 9784 }); 9785 9786 uploadView = this.workflow.uploader; 9787 9788 if ( uploadView.uploader && uploadView.uploader.ready ) { 9789 this.addFiles.apply( this ); 9790 } else { 9791 this.workflow.on( 'uploader:ready', this.addFiles, this ); 9792 } 9793 } else { 9794 this.workflow.state().reset(); 9795 this.addFiles.apply( this ); 9796 this.workflow.open(); 9797 } 9798 9799 return false; 9800 }, 9801 9802 /** 9803 * Add the files to the uploader. 9804 */ 9805 addFiles: function() { 9806 if ( this.files.length ) { 9807 this.workflow.uploader.uploader.uploader.addFile( _.toArray( this.files ) ); 9808 this.files = []; 9809 } 9810 return this; 9811 }, 9812 9813 containerDragover: function( event ) { 9814 if ( this.localDrag || ! this.isDraggingFile( event ) ) { 9815 return; 9816 } 9817 9818 this.overContainer = true; 9819 this.refresh(); 9820 }, 9821 9822 containerDragleave: function() { 9823 this.overContainer = false; 9824 9825 // Throttle dragleave because it's called when bouncing from some elements to others. 9826 _.delay( _.bind( this.refresh, this ), 50 ); 9827 }, 9828 9829 dropzoneDragover: function( event ) { 9830 if ( this.localDrag || ! this.isDraggingFile( event ) ) { 9831 return; 9832 } 9833 9834 this.overDropzone = true; 9835 this.refresh( event ); 9836 return false; 9837 }, 9838 9839 dropzoneDragleave: function( e ) { 9840 this.overDropzone = false; 9841 _.delay( _.bind( this.refresh, this, e ), 50 ); 9842 }, 9843 9844 click: function( e ) { 9845 // In the rare case where the dropzone gets stuck, hide it on click. 9846 this.containerDragleave( e ); 9847 this.dropzoneDragleave( e ); 9848 this.localDrag = false; 9849 } 9850 }); 9851 9852 module.exports = EditorUploader; 9853 9854 9855 /***/ }), 9856 9857 /***/ 1753: 9858 /***/ ((module) => { 9859 9860 var View = wp.media.View, 9861 UploaderInline; 9862 9863 /** 9864 * wp.media.view.UploaderInline 9865 * 9866 * The inline uploader that shows up in the 'Upload Files' tab. 9867 * 9868 * @memberOf wp.media.view 9869 * 9870 * @class 9871 * @augments wp.media.View 9872 * @augments wp.Backbone.View 9873 * @augments Backbone.View 9874 */ 9875 UploaderInline = View.extend(/** @lends wp.media.view.UploaderInline.prototype */{ 9876 tagName: 'div', 9877 className: 'uploader-inline', 9878 template: wp.template('uploader-inline'), 9879 9880 events: { 9881 'click .close': 'hide' 9882 }, 9883 9884 initialize: function() { 9885 _.defaults( this.options, { 9886 message: '', 9887 status: true, 9888 canClose: false 9889 }); 9890 9891 if ( ! this.options.$browser && this.controller.uploader ) { 9892 this.options.$browser = this.controller.uploader.$browser; 9893 } 9894 9895 if ( _.isUndefined( this.options.postId ) ) { 9896 this.options.postId = wp.media.view.settings.post.id; 9897 } 9898 9899 if ( this.options.status ) { 9900 this.views.set( '.upload-inline-status', new wp.media.view.UploaderStatus({ 9901 controller: this.controller 9902 }) ); 9903 } 9904 }, 9905 9906 prepare: function() { 9907 var suggestedWidth = this.controller.state().get('suggestedWidth'), 9908 suggestedHeight = this.controller.state().get('suggestedHeight'), 9909 data = {}; 9910 9911 data.message = this.options.message; 9912 data.canClose = this.options.canClose; 9913 9914 if ( suggestedWidth && suggestedHeight ) { 9915 data.suggestedWidth = suggestedWidth; 9916 data.suggestedHeight = suggestedHeight; 9917 } 9918 9919 return data; 9920 }, 9921 /** 9922 * @return {wp.media.view.UploaderInline} Returns itself to allow chaining. 9923 */ 9924 dispose: function() { 9925 if ( this.disposing ) { 9926 /** 9927 * call 'dispose' directly on the parent class 9928 */ 9929 return View.prototype.dispose.apply( this, arguments ); 9930 } 9931 9932 /* 9933 * Run remove on `dispose`, so we can be sure to refresh the 9934 * uploader with a view-less DOM. Track whether we're disposing 9935 * so we don't trigger an infinite loop. 9936 */ 9937 this.disposing = true; 9938 return this.remove(); 9939 }, 9940 /** 9941 * @return {wp.media.view.UploaderInline} Returns itself to allow chaining. 9942 */ 9943 remove: function() { 9944 /** 9945 * call 'remove' directly on the parent class 9946 */ 9947 var result = View.prototype.remove.apply( this, arguments ); 9948 9949 _.defer( _.bind( this.refresh, this ) ); 9950 return result; 9951 }, 9952 9953 refresh: function() { 9954 var uploader = this.controller.uploader; 9955 9956 if ( uploader ) { 9957 uploader.refresh(); 9958 } 9959 }, 9960 /** 9961 * @return {wp.media.view.UploaderInline} 9962 */ 9963 ready: function() { 9964 var $browser = this.options.$browser, 9965 $placeholder; 9966 9967 if ( this.controller.uploader ) { 9968 $placeholder = this.$('.browser'); 9969 9970 // Check if we've already replaced the placeholder. 9971 if ( $placeholder[0] === $browser[0] ) { 9972 return; 9973 } 9974 9975 $browser.detach().text( $placeholder.text() ); 9976 $browser[0].className = $placeholder[0].className; 9977 $browser[0].setAttribute( 'aria-labelledby', $browser[0].id + ' ' + $placeholder[0].getAttribute('aria-labelledby') ); 9978 $placeholder.replaceWith( $browser.show() ); 9979 } 9980 9981 this.refresh(); 9982 return this; 9983 }, 9984 show: function() { 9985 this.$el.removeClass( 'hidden' ); 9986 if ( this.controller.$uploaderToggler && this.controller.$uploaderToggler.length ) { 9987 this.controller.$uploaderToggler.attr( 'aria-expanded', 'true' ); 9988 } 9989 }, 9990 hide: function() { 9991 this.$el.addClass( 'hidden' ); 9992 if ( this.controller.$uploaderToggler && this.controller.$uploaderToggler.length ) { 9993 this.controller.$uploaderToggler 9994 .attr( 'aria-expanded', 'false' ) 9995 // Move focus back to the toggle button when closing the uploader. 9996 .trigger( 'focus' ); 9997 } 9998 } 9999 10000 }); 10001 10002 module.exports = UploaderInline; 10003 10004 10005 /***/ }), 10006 10007 /***/ 6442: 10008 /***/ ((module) => { 10009 10010 /** 10011 * wp.media.view.UploaderStatusError 10012 * 10013 * @memberOf wp.media.view 10014 * 10015 * @class 10016 * @augments wp.media.View 10017 * @augments wp.Backbone.View 10018 * @augments Backbone.View 10019 */ 10020 var UploaderStatusError = wp.media.View.extend(/** @lends wp.media.view.UploaderStatusError.prototype */{ 10021 className: 'upload-error', 10022 template: wp.template('uploader-status-error') 10023 }); 10024 10025 module.exports = UploaderStatusError; 10026 10027 10028 /***/ }), 10029 10030 /***/ 8197: 10031 /***/ ((module) => { 10032 10033 var View = wp.media.View, 10034 UploaderStatus; 10035 10036 /** 10037 * wp.media.view.UploaderStatus 10038 * 10039 * An uploader status for on-going uploads. 10040 * 10041 * @memberOf wp.media.view 10042 * 10043 * @class 10044 * @augments wp.media.View 10045 * @augments wp.Backbone.View 10046 * @augments Backbone.View 10047 */ 10048 UploaderStatus = View.extend(/** @lends wp.media.view.UploaderStatus.prototype */{ 10049 className: 'media-uploader-status', 10050 template: wp.template('uploader-status'), 10051 10052 events: { 10053 'click .upload-dismiss-errors': 'dismiss' 10054 }, 10055 10056 initialize: function() { 10057 this.queue = wp.Uploader.queue; 10058 this.queue.on( 'add remove reset', this.visibility, this ); 10059 this.queue.on( 'add remove reset change:percent', this.progress, this ); 10060 this.queue.on( 'add remove reset change:uploading', this.info, this ); 10061 10062 this.errors = wp.Uploader.errors; 10063 this.errors.reset(); 10064 this.errors.on( 'add remove reset', this.visibility, this ); 10065 this.errors.on( 'add', this.error, this ); 10066 }, 10067 /** 10068 * @return {wp.media.view.UploaderStatus} 10069 */ 10070 dispose: function() { 10071 wp.Uploader.queue.off( null, null, this ); 10072 /** 10073 * call 'dispose' directly on the parent class 10074 */ 10075 View.prototype.dispose.apply( this, arguments ); 10076 return this; 10077 }, 10078 10079 visibility: function() { 10080 this.$el.toggleClass( 'uploading', !! this.queue.length ); 10081 this.$el.toggleClass( 'errors', !! this.errors.length ); 10082 this.$el.toggle( !! this.queue.length || !! this.errors.length ); 10083 }, 10084 10085 ready: function() { 10086 _.each({ 10087 '$bar': '.media-progress-bar div', 10088 '$index': '.upload-index', 10089 '$total': '.upload-total', 10090 '$filename': '.upload-filename' 10091 }, function( selector, key ) { 10092 this[ key ] = this.$( selector ); 10093 }, this ); 10094 10095 this.visibility(); 10096 this.progress(); 10097 this.info(); 10098 }, 10099 10100 progress: function() { 10101 var queue = this.queue, 10102 $bar = this.$bar; 10103 10104 if ( ! $bar || ! queue.length ) { 10105 return; 10106 } 10107 10108 $bar.width( ( queue.reduce( function( memo, attachment ) { 10109 if ( ! attachment.get('uploading') ) { 10110 return memo + 100; 10111 } 10112 10113 var percent = attachment.get('percent'); 10114 return memo + ( _.isNumber( percent ) ? percent : 100 ); 10115 }, 0 ) / queue.length ) + '%' ); 10116 }, 10117 10118 info: function() { 10119 var queue = this.queue, 10120 index = 0, active; 10121 10122 if ( ! queue.length ) { 10123 return; 10124 } 10125 10126 active = this.queue.find( function( attachment, i ) { 10127 index = i; 10128 return attachment.get('uploading'); 10129 }); 10130 10131 if ( this.$index && this.$total && this.$filename ) { 10132 this.$index.text( index + 1 ); 10133 this.$total.text( queue.length ); 10134 this.$filename.html( active ? this.filename( active.get('filename') ) : '' ); 10135 } 10136 }, 10137 /** 10138 * @param {string} filename 10139 * @return {string} 10140 */ 10141 filename: function( filename ) { 10142 return _.escape( filename ); 10143 }, 10144 /** 10145 * @param {Backbone.Model} error 10146 */ 10147 error: function( error ) { 10148 var statusError = new wp.media.view.UploaderStatusError( { 10149 filename: this.filename( error.get( 'file' ).name ), 10150 message: error.get( 'message' ) 10151 } ); 10152 10153 var buttonClose = this.$el.find( 'button' ); 10154 10155 // Can show additional info here while retrying to create image sub-sizes. 10156 this.views.add( '.upload-errors', statusError, { at: 0 } ); 10157 _.delay( function() { 10158 buttonClose.trigger( 'focus' ); 10159 wp.a11y.speak( error.get( 'message' ), 'assertive' ); 10160 }, 1000 ); 10161 }, 10162 10163 dismiss: function() { 10164 var errors = this.views.get('.upload-errors'); 10165 10166 if ( errors ) { 10167 _.invoke( errors, 'remove' ); 10168 } 10169 wp.Uploader.errors.reset(); 10170 // Move focus to the modal after the dismiss button gets removed from the DOM. 10171 if ( this.controller.modal ) { 10172 this.controller.modal.focusManager.focus(); 10173 } 10174 } 10175 }); 10176 10177 module.exports = UploaderStatus; 10178 10179 10180 /***/ }), 10181 10182 /***/ 8291: 10183 /***/ ((module) => { 10184 10185 var $ = jQuery, 10186 UploaderWindow; 10187 10188 /** 10189 * wp.media.view.UploaderWindow 10190 * 10191 * An uploader window that allows for dragging and dropping media. 10192 * 10193 * @memberOf wp.media.view 10194 * 10195 * @class 10196 * @augments wp.media.View 10197 * @augments wp.Backbone.View 10198 * @augments Backbone.View 10199 * 10200 * @param {object} [options] Options hash passed to the view. 10201 * @param {object} [options.uploader] Uploader properties. 10202 * @param {jQuery} [options.uploader.browser] 10203 * @param {jQuery} [options.uploader.dropzone] jQuery collection of the dropzone. 10204 * @param {object} [options.uploader.params] 10205 */ 10206 UploaderWindow = wp.media.View.extend(/** @lends wp.media.view.UploaderWindow.prototype */{ 10207 tagName: 'div', 10208 className: 'uploader-window', 10209 template: wp.template('uploader-window'), 10210 10211 initialize: function() { 10212 var uploader; 10213 10214 this.$browser = $( '<button type="button" class="browser" />' ).hide().appendTo( 'body' ); 10215 10216 uploader = this.options.uploader = _.defaults( this.options.uploader || {}, { 10217 dropzone: this.$el, 10218 browser: this.$browser, 10219 params: {} 10220 }); 10221 10222 // Ensure the dropzone is a jQuery collection. 10223 if ( uploader.dropzone && ! (uploader.dropzone instanceof $) ) { 10224 uploader.dropzone = $( uploader.dropzone ); 10225 } 10226 10227 this.controller.on( 'activate', this.refresh, this ); 10228 10229 this.controller.on( 'detach', function() { 10230 this.$browser.remove(); 10231 }, this ); 10232 }, 10233 10234 refresh: function() { 10235 if ( this.uploader ) { 10236 this.uploader.refresh(); 10237 } 10238 }, 10239 10240 ready: function() { 10241 var postId = wp.media.view.settings.post.id, 10242 dropzone; 10243 10244 // If the uploader already exists, bail. 10245 if ( this.uploader ) { 10246 return; 10247 } 10248 10249 if ( postId ) { 10250 this.options.uploader.params.post_id = postId; 10251 } 10252 this.uploader = new wp.Uploader( this.options.uploader ); 10253 10254 dropzone = this.uploader.dropzone; 10255 dropzone.on( 'dropzone:enter', _.bind( this.show, this ) ); 10256 dropzone.on( 'dropzone:leave', _.bind( this.hide, this ) ); 10257 10258 $( this.uploader ).on( 'uploader:ready', _.bind( this._ready, this ) ); 10259 }, 10260 10261 _ready: function() { 10262 this.controller.trigger( 'uploader:ready' ); 10263 }, 10264 10265 show: function() { 10266 var $el = this.$el.show(); 10267 10268 // Ensure that the animation is triggered by waiting until 10269 // the transparent element is painted into the DOM. 10270 _.defer( function() { 10271 $el.css({ opacity: 1 }); 10272 }); 10273 }, 10274 10275 hide: function() { 10276 var $el = this.$el.css({ opacity: 0 }); 10277 10278 wp.media.transition( $el ).done( function() { 10279 // Transition end events are subject to race conditions. 10280 // Make sure that the value is set as intended. 10281 if ( '0' === $el.css('opacity') ) { 10282 $el.hide(); 10283 } 10284 }); 10285 10286 // https://core.trac.wordpress.org/ticket/27341 10287 _.delay( function() { 10288 if ( '0' === $el.css('opacity') && $el.is(':visible') ) { 10289 $el.hide(); 10290 } 10291 }, 500 ); 10292 } 10293 }); 10294 10295 module.exports = UploaderWindow; 10296 10297 10298 /***/ }), 10299 10300 /***/ 4747: 10301 /***/ ((module) => { 10302 10303 /** 10304 * wp.media.View 10305 * 10306 * The base view class for media. 10307 * 10308 * Undelegating events, removing events from the model, and 10309 * removing events from the controller mirror the code for 10310 * `Backbone.View.dispose` in Backbone 0.9.8 development. 10311 * 10312 * This behavior has since been removed, and should not be used 10313 * outside of the media manager. 10314 * 10315 * @memberOf wp.media 10316 * 10317 * @class 10318 * @augments wp.Backbone.View 10319 * @augments Backbone.View 10320 */ 10321 var View = wp.Backbone.View.extend(/** @lends wp.media.View.prototype */{ 10322 constructor: function( options ) { 10323 if ( options && options.controller ) { 10324 this.controller = options.controller; 10325 } 10326 wp.Backbone.View.apply( this, arguments ); 10327 }, 10328 /** 10329 * @todo The internal comment mentions this might have been a stop-gap 10330 * before Backbone 0.9.8 came out. Figure out if Backbone core takes 10331 * care of this in Backbone.View now. 10332 * 10333 * @return {wp.media.View} Returns itself to allow chaining. 10334 */ 10335 dispose: function() { 10336 /* 10337 * Undelegating events, removing events from the model, and 10338 * removing events from the controller mirror the code for 10339 * `Backbone.View.dispose` in Backbone 0.9.8 development. 10340 */ 10341 this.undelegateEvents(); 10342 10343 if ( this.model && this.model.off ) { 10344 this.model.off( null, null, this ); 10345 } 10346 10347 if ( this.collection && this.collection.off ) { 10348 this.collection.off( null, null, this ); 10349 } 10350 10351 // Unbind controller events. 10352 if ( this.controller && this.controller.off ) { 10353 this.controller.off( null, null, this ); 10354 } 10355 10356 return this; 10357 }, 10358 /** 10359 * @return {wp.media.View} Returns itself to allow chaining. 10360 */ 10361 remove: function() { 10362 this.dispose(); 10363 /** 10364 * call 'remove' directly on the parent class 10365 */ 10366 return wp.Backbone.View.prototype.remove.apply( this, arguments ); 10367 } 10368 }); 10369 10370 module.exports = View; 10371 10372 10373 /***/ }) 10374 10375 /******/ }); 10376 /************************************************************************/ 10377 /******/ // The module cache 10378 /******/ var __webpack_module_cache__ = {}; 10379 /******/ 10380 /******/ // The require function 10381 /******/ function __webpack_require__(moduleId) { 10382 /******/ // Check if module is in cache 10383 /******/ var cachedModule = __webpack_module_cache__[moduleId]; 10384 /******/ if (cachedModule !== undefined) { 10385 /******/ return cachedModule.exports; 10386 /******/ } 10387 /******/ // Create a new module (and put it into the cache) 10388 /******/ var module = __webpack_module_cache__[moduleId] = { 10389 /******/ // no module.id needed 10390 /******/ // no module.loaded needed 10391 /******/ exports: {} 10392 /******/ }; 10393 /******/ 10394 /******/ // Execute the module function 10395 /******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__); 10396 /******/ 10397 /******/ // Return the exports of the module 10398 /******/ return module.exports; 10399 /******/ } 10400 /******/ 10401 /************************************************************************/ 10402 /** 10403 * @output wp-includes/js/media-views.js 10404 */ 10405 10406 var media = wp.media, 10407 $ = jQuery, 10408 l10n; 10409 10410 media.isTouchDevice = ( 'ontouchend' in document ); 10411 10412 // Link any localized strings. 10413 l10n = media.view.l10n = window._wpMediaViewsL10n || {}; 10414 10415 // Link any settings. 10416 media.view.settings = l10n.settings || {}; 10417 delete l10n.settings; 10418 10419 // Copy the `post` setting over to the model settings. 10420 media.model.settings.post = media.view.settings.post; 10421 10422 // Check if the browser supports CSS 3.0 transitions. 10423 $.support.transition = (function(){ 10424 var style = document.documentElement.style, 10425 transitions = { 10426 WebkitTransition: 'webkitTransitionEnd', 10427 MozTransition: 'transitionend', 10428 OTransition: 'oTransitionEnd otransitionend', 10429 transition: 'transitionend' 10430 }, transition; 10431 10432 transition = _.find( _.keys( transitions ), function( transition ) { 10433 return ! _.isUndefined( style[ transition ] ); 10434 }); 10435 10436 return transition && { 10437 end: transitions[ transition ] 10438 }; 10439 }()); 10440 10441 /** 10442 * A shared event bus used to provide events into 10443 * the media workflows that 3rd-party devs can use to hook 10444 * in. 10445 */ 10446 media.events = _.extend( {}, Backbone.Events ); 10447 10448 /** 10449 * Makes it easier to bind events using transitions. 10450 * 10451 * @param {string} selector 10452 * @param {number} sensitivity 10453 * @return {Promise} 10454 */ 10455 media.transition = function( selector, sensitivity ) { 10456 var deferred = $.Deferred(); 10457 10458 sensitivity = sensitivity || 2000; 10459 10460 if ( $.support.transition ) { 10461 if ( ! (selector instanceof $) ) { 10462 selector = $( selector ); 10463 } 10464 10465 // Resolve the deferred when the first element finishes animating. 10466 selector.first().one( $.support.transition.end, deferred.resolve ); 10467 10468 // Just in case the event doesn't trigger, fire a callback. 10469 _.delay( deferred.resolve, sensitivity ); 10470 10471 // Otherwise, execute on the spot. 10472 } else { 10473 deferred.resolve(); 10474 } 10475 10476 return deferred.promise(); 10477 }; 10478 10479 media.controller.Region = __webpack_require__( 9875 ); 10480 media.controller.StateMachine = __webpack_require__( 6150 ); 10481 media.controller.State = __webpack_require__( 5694 ); 10482 10483 media.selectionSync = __webpack_require__( 4181 ); 10484 media.controller.Library = __webpack_require__( 472 ); 10485 media.controller.ImageDetails = __webpack_require__( 705 ); 10486 media.controller.GalleryEdit = __webpack_require__( 2038 ); 10487 media.controller.GalleryAdd = __webpack_require__( 7127 ); 10488 media.controller.CollectionEdit = __webpack_require__( 8612 ); 10489 media.controller.CollectionAdd = __webpack_require__( 7145 ); 10490 media.controller.FeaturedImage = __webpack_require__( 1169 ); 10491 media.controller.ReplaceImage = __webpack_require__( 2275 ); 10492 media.controller.EditImage = __webpack_require__( 5663 ); 10493 media.controller.MediaLibrary = __webpack_require__( 8065 ); 10494 media.controller.Embed = __webpack_require__( 4910 ); 10495 media.controller.Cropper = __webpack_require__( 5422 ); 10496 media.controller.CustomizeImageCropper = __webpack_require__( 9660 ); 10497 media.controller.SiteIconCropper = __webpack_require__( 6172 ); 10498 10499 media.View = __webpack_require__( 4747 ); 10500 media.view.Frame = __webpack_require__( 1061 ); 10501 media.view.MediaFrame = __webpack_require__( 2836 ); 10502 media.view.MediaFrame.Select = __webpack_require__( 455 ); 10503 media.view.MediaFrame.Post = __webpack_require__( 4274 ); 10504 media.view.MediaFrame.ImageDetails = __webpack_require__( 5424 ); 10505 media.view.Modal = __webpack_require__( 2621 ); 10506 media.view.FocusManager = __webpack_require__( 718 ); 10507 media.view.UploaderWindow = __webpack_require__( 8291 ); 10508 media.view.EditorUploader = __webpack_require__( 3674 ); 10509 media.view.UploaderInline = __webpack_require__( 1753 ); 10510 media.view.UploaderStatus = __webpack_require__( 8197 ); 10511 media.view.UploaderStatusError = __webpack_require__( 6442 ); 10512 media.view.Toolbar = __webpack_require__( 5275 ); 10513 media.view.Toolbar.Select = __webpack_require__( 9458 ); 10514 media.view.Toolbar.Embed = __webpack_require__( 397 ); 10515 media.view.Button = __webpack_require__( 846 ); 10516 media.view.ButtonGroup = __webpack_require__( 168 ); 10517 media.view.PriorityList = __webpack_require__( 8815 ); 10518 media.view.MenuItem = __webpack_require__( 9013 ); 10519 media.view.Menu = __webpack_require__( 1 ); 10520 media.view.RouterItem = __webpack_require__( 6327 ); 10521 media.view.Router = __webpack_require__( 4783 ); 10522 media.view.Sidebar = __webpack_require__( 1992 ); 10523 media.view.Attachment = __webpack_require__( 4075 ); 10524 media.view.Attachment.Library = __webpack_require__( 3443 ); 10525 media.view.Attachment.EditLibrary = __webpack_require__( 5232 ); 10526 media.view.Attachments = __webpack_require__( 8142 ); 10527 media.view.Search = __webpack_require__( 2102 ); 10528 media.view.AttachmentFilters = __webpack_require__( 7709 ); 10529 media.view.DateFilter = __webpack_require__( 6472 ); 10530 media.view.AttachmentFilters.Uploaded = __webpack_require__( 1368 ); 10531 media.view.AttachmentFilters.All = __webpack_require__( 7349 ); 10532 media.view.AttachmentsBrowser = __webpack_require__( 6829 ); 10533 media.view.Selection = __webpack_require__( 8282 ); 10534 media.view.Attachment.Selection = __webpack_require__( 3962 ); 10535 media.view.Attachments.Selection = __webpack_require__( 3479 ); 10536 media.view.Attachment.EditSelection = __webpack_require__( 4593 ); 10537 media.view.Settings = __webpack_require__( 1915 ); 10538 media.view.Settings.AttachmentDisplay = __webpack_require__( 7656 ); 10539 media.view.Settings.Gallery = __webpack_require__( 7266 ); 10540 media.view.Settings.Playlist = __webpack_require__( 2356 ); 10541 media.view.Attachment.Details = __webpack_require__( 6090 ); 10542 media.view.AttachmentCompat = __webpack_require__( 2982 ); 10543 media.view.Iframe = __webpack_require__( 1982 ); 10544 media.view.Embed = __webpack_require__( 5741 ); 10545 media.view.Label = __webpack_require__( 4338 ); 10546 media.view.EmbedUrl = __webpack_require__( 7327 ); 10547 media.view.EmbedLink = __webpack_require__( 8232 ); 10548 media.view.EmbedImage = __webpack_require__( 2395 ); 10549 media.view.ImageDetails = __webpack_require__( 2650 ); 10550 media.view.Cropper = __webpack_require__( 7637 ); 10551 media.view.SiteIconCropper = __webpack_require__( 443 ); 10552 media.view.SiteIconPreview = __webpack_require__( 7810 ); 10553 media.view.EditImage = __webpack_require__( 6126 ); 10554 media.view.Spinner = __webpack_require__( 9141 ); 10555 media.view.Heading = __webpack_require__( 170 ); 10556 10557 /******/ })() 10558 ;
title
Description
Body
title
Description
Body
title
Description
Body
title
Body
Generated : Tue Jan 21 08:20:01 2025 | Cross-referenced by PHPXref |