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