HEX
Server: Apache
System: Linux pdx1-shared-a1-38 6.6.104-grsec-jammy+ #3 SMP Tue Sep 16 00:28:11 UTC 2025 x86_64
User: mmickelson (3396398)
PHP: 8.1.31
Disabled: NONE
Upload Files
File: /home/mmickelson/martyknows.com/wp-content/plugins/enhanced-media-library/js/eml-media-grid.js
window.wp = window.wp || {};
window.wpCookies = window.wpCookies || {};
window.eml = window.eml || { l10n: {} };



( function( $, _ ) {

    var media = wp.media,
        mediaTrash = media.view.settings.mediaTrash,
        l10n = media.view.l10n,
        original = {},
        newEvents = {};



    _.extend( eml.l10n, wpuxss_eml_media_grid_l10n );



    _.extend( media.view.Attachment.Library.prototype, {

        buttons: {
            check  : true,
            edit   : true,
            remove : false, // TODO: consider 'delete' button
            attach : false // TODO: consider 'attach' button
        }
    });



    newEvents = { 'click .edit': 'emlEditAttachment' };
    _.extend( newEvents, media.view.Attachment.prototype.events);

    _.extend( media.view.Attachment.prototype, {

        template:  media.template('attachment-grid-view'),

        events: newEvents,

        emlEditAttachment: function( event ) {

            if ( this.controller.isModeActive( 'eml-grid' ) ) {

                this.controller.trigger( 'edit:attachment', this.model);

                event.stopPropagation();
                return;
            }
        }
    });



    _.extend( media.view.Attachment.Details.prototype, {

        editAttachment: function( event ) {

            if ( this.controller.isModeActive( 'eml-grid' ) ) {

                event.preventDefault();
                this.controller.trigger( 'edit:attachment', this.model);
            }
        }
    });


    
    newEvents = { 'click .eml-toggle-collapse': 'toggleCollapse' };
    _.extend( newEvents, media.view.Attachment.Details.prototype.events);

    media.view.emlGridAttachmentDetails = media.view.Attachment.Details.extend({

        events: newEvents,

        render: function() {

            media.view.Attachment.Details.prototype.render.apply( this, arguments );

            // collapse
            this.$el.find('.setting, #alt-text-description').wrapAll( '<div class="eml-collapse" />' );
            this.$el.find('.compat-meta').before( '<a class="eml-toggle-collapse" href="javascript:;">'+eml.l10n.more_details+'</a>' );

            this.toggleCollapse();
        },

        toggleCollapse: function( event ) {

            var collapsed = ( this.controller._attachmentDetailsCollapsed === 'true' ),
                secure = ( 'https:' === window.location.protocol );


            if ( typeof event !== 'undefined' && 'eml-toggle-collapse' === event.currentTarget.className ) {

                this.$el.find('.eml-collapse').toggle( 300 );
                collapsed = ! collapsed;
            }
            else if ( collapsed ) {
                this.$el.find('.eml-collapse').hide();
            }
            else {
                this.$el.find('.eml-collapse').show();
            }

            this.$el.find('.eml-toggle-collapse').html(function(i, html) {
                return ! collapsed ? eml.l10n.less_details+' \u2191' : eml.l10n.more_details+' \u2193';
            });

            this.controller._attachmentDetailsCollapsed = collapsed ? 'true' : 'false';
            wpCookies.set( 'eml-details-collapsed', collapsed, 30 * 24 * 60 * 60, false, false, secure );
        }
    });



    /**
     * media.view.emlGridSelection
     *
     */
    media.view.emlGridSelection = media.view.Selection.extend({

        template:  media.template('eml-media-selection')
    });



    /**
     * wp.media.view.emlSelectAllButton
     *
     * Used in the bulk mode only right now
     */
    media.view.emlSelectAllButton = media.view.Button.extend({

        initialize: function() {
            media.view.Button.prototype.initialize.apply( this, arguments );
            this.controller.state().get( 'selection' ).on( 'add remove reset', this.toggleDisabled, this );
        },

        toggleDisabled: function() {
            this.$el.toggleClass( 'hidden', !! this.controller.state().get( 'selection' ).length );
            this.model.set( 'disabled', !! this.controller.state().get( 'selection' ).length );
        },

        render: function() {
            media.view.Button.prototype.render.apply( this, arguments );
            this.toggleDisabled();
            return this;
        },

        click: function() {
            this.controller.selectAll();
        }
    });



    /**
     * wp.media.view.emlDeselectButton
     *
     */
    media.view.emlDeselectButton = media.view.Button.extend({

        initialize: function() {
            media.view.Button.prototype.initialize.apply( this, arguments );
            this.controller.state().get( 'selection' ).on( 'add remove reset', this.toggleDisabled, this );
        },

        toggleDisabled: function() {
            this.$el.toggleClass( 'hidden', ! this.controller.state().get( 'selection' ).length );
            this.model.set( 'disabled', ! this.controller.state().get( 'selection' ).length );
        },

        render: function() {
            media.view.Button.prototype.render.apply( this, arguments );
            this.toggleDisabled();
            return this;
        },

        click: function() {

            var selection = this.controller.state().get( 'selection' );

            selection.reset();

            // Keep focus inside media modal
            if ( this.controller.modal ) {
                this.controller.modal.focusManager.focus();
            }
        }
    });



    /**
     * wp.media.view.emlDeleteSelectedButton
     *
     */
    media.view.emlDeleteSelectedButton = media.view.DeleteSelectedButton.extend({

        initialize: function() {
            media.view.Button.prototype.initialize.apply( this, arguments );
            if ( this.options.filters ) {
                this.options.filters.model.on( 'change', this.filterChange, this );
            }
            this.controller.state().get( 'selection' ).on( 'add remove reset', this.toggleDisabled, this );
        },

        toggleDisabled: function() {
            this.$el.toggleClass( 'hidden', ! this.controller.state().get( 'selection' ).length );
            this.model.set( 'disabled', ! this.controller.state().get( 'selection' ).length );
        },

        render: function() {
            media.view.Button.prototype.render.apply( this, arguments );
            this.toggleDisabled();
            return this;
        },

        click: function() {

            var changed = [], removed = [], action = 'delete',
                selection = this.controller.state().get( 'selection' ),
                library = this.controller.state().get( 'library' );


            if ( typeof this.controller._bulk !== 'undefined' ) {

                if ( mediaTrash ) {
                    action = 'trash' === selection.at( 0 ).get( 'status' ) ? 'restore' : 'trash';
                }

                this.controller.bulk( action );
            }
            else {

                if ( ! selection.length ) {
                    return;
                }

                if ( ! mediaTrash && ! window.confirm( l10n.warnBulkDelete ) ) {
                    return;
                }

                if ( mediaTrash &&
                    'trash' !== selection.at( 0 ).get( 'status' ) &&
                    ! window.confirm( l10n.warnBulkTrash ) ) {

                    return;
                }

                selection.each( function( model ) {
                    if ( ! model.get( 'nonces' )['delete'] ) {
                        removed.push( model );
                        return;
                    }

                    if ( mediaTrash && 'trash' === model.get( 'status' ) ) {
                        model.set( 'status', 'inherit' );
                        changed.push( model.save() );
                        removed.push( model );
                    } else if ( mediaTrash ) {
                        model.set( 'status', 'trash' );
                        changed.push( model.save() );
                        removed.push( model );
                    } else {
                        model.destroy({wait: true});
                    }
                } );

                if ( changed.length ) {
                    selection.remove( removed );

                    $.when.apply( null, changed ).then( _.bind( function() {
                        library._requery( true );
                        this.controller.trigger( 'selection:action:done' );
                    }, this ) );
                } else {
                    this.controller.trigger( 'selection:action:done' );
                }
            }
        }
    });



    /**
     * wp.media.view.emlDeleteSelectedPermanentlyButton
     *
     */
    media.view.emlDeleteSelectedPermanentlyButton = media.view.emlDeleteSelectedButton.extend({

        filterChange: function( model ) {
            this.canShow = ( 'trash' === model.get( 'status' ) );
        },

        toggleDisabled: function() {
            this.$el.toggleClass( 'hidden', ! this.canShow );
            this.model.set( 'disabled', ! this.canShow );
        },

        click: function() {

            if ( typeof this.controller._bulk !== 'undefined' ) {
                this.controller.bulk( 'delete' );
            }
            else {

                var removed = [],
                    destroy = [],
                    selection = this.controller.state().get( 'selection' );

                if ( ! selection.length || ! window.confirm( l10n.warnBulkDelete ) ) {
                    return;
                }

                selection.each( function( model ) {
                    if ( ! model.get( 'nonces' )['delete'] ) {
                        removed.push( model );
                        return;
                    }

                    destroy.push( model );
                } );

                if ( removed.length ) {
                    selection.remove( removed );
                }

                if ( destroy.length ) {
                    $.when.apply( null, destroy.map( function (item) {
                        return item.destroy();
                    } ) ).then( _.bind( function() {
                        this.controller.trigger( 'selection:action:done' );
                    }, this ) );
                }
            }
        }
    });



    /**
     * wp.media.view.MediaFrame.emlGrid
     *
     */
    media.view.MediaFrame.emlGrid = media.view.MediaFrame.Select.extend({

        _attachmentDetailsCollapsed: wpCookies.get( 'eml-details-collapsed' ) || 'true',

        initialize: function() {

            _.defaults( this.options, {
                title    : '',
                modal    : false,

                selection: [],
                library:   {}, // Options hash for the query to the media library.
                uploader:  true,

                multiple : true, 
                state    : 'library',
                mode     : [ 'eml-grid', 'edit' ]
            });

            // this.$body = $( document.body );
            // this.$window = $( window );
            // this.$adminBar = $( '#wpadminbar' );
            this.$uploaderToggler = $( '.page-title-action' )
                .attr( 'aria-expanded', 'false' )
                .on( 'click', _.bind( this.addNewClickHandler, this ) );

            // Ensure core and media grid view UI is enabled.
            this.$el.addClass('wp-core-ui');

            this.gridRouter = new media.view.MediaFrame.Manage.Router();

            // Call 'initialize' directly on the parent class.
            media.view.MediaFrame.Select.prototype.initialize.apply( this, arguments );

            // Append the frame view directly the supplied container.
            this.$el.appendTo( this.options.container );

            this.bindRegionModeHandlers();
            this.render();
            this.bindSearchHandler();

            media.frames.browse = this;
        },

        createStates: function() {

            var options = this.options;

            if ( this.options.states ) {
                return;
            }

            this.states.add([

                new media.controller.Library({
                    library            : media.query( options.library ),
                    title              : options.title,
                    multiple           : options.multiple ? 'reset' : false,

                    content            : 'browse',
                    toolbar            : 'bulk-edit',
                    menu               : false,
                    router             : false,

                    // contentUserSetting : false, //true,

                    // searchable         : true,
                    filterable         : 'all',

                    // autoSelect         : false, // true,
                    // editable:   true,
                })
            ]);
        },

        bindRegionModeHandlers: function() {

            this.on( 'select:activate', this.bindKeydown, this );
            this.on( 'select:deactivate', this.unbindKeydown, this );

            this.on( 'toolbar:create:bulk-edit', this.createToolbar, this );
            this.on( 'toolbar:render:bulk-edit', this.selectionStatusToolbar, this );
            this.on( 'edit:attachment', this.openEditAttachmentModal, this );
        },

        bindSearchHandler: function() {

            var search = this.$( '#media-search-input' ),
                searchView = this.browserView.toolbar.get( 'search' ).$el,
                listMode = this.$( '.view-list' ),


                input  = _.throttle( function (e) {

                    var val = $( e.currentTarget ).val(),
                        url = val ? '?search=' + val : '';

                    this.gridRouter.navigate( this.gridRouter.baseUrl( url ), { replace: true } );

                }, 1000 );

            // Update the URL when entering search string (at most once per second).
            search.on( 'input', _.bind( input, this ) );

            this.gridRouter
                .on( 'route:search', function () {
                    var href = window.location.href;
                    if ( href.indexOf( 'mode=' ) > -1 ) {
                        href = href.replace( /mode=[^&]+/g, 'mode=list' );
                    } else {
                        href += href.indexOf( '?' ) > -1 ? '&mode=list' : '?mode=list';
                    }
                    href = href.replace( 'search=', 's=' );
                    listMode.prop( 'href', href );
                })
                .on( 'route:reset', function() {
                    searchView.val( '' ).trigger( 'input' );
                });
        },

        handleKeydown: function( e ) {
            if ( 27 === e.which ) {
                e.preventDefault();
                this.deactivateMode( 'select' ).activateMode( 'edit' );
            }
        },

        bindKeydown: function() {
            this.$body.on( 'keydown.select', _.bind( this.handleKeydown, this ) );
        },

        unbindKeydown: function() {
            this.$body.off( 'keydown.select' );
        },

        selectionStatusToolbar: function( view ) {

            view.set( 'selection', new media.view.emlGridSelection({
                controller: this,
                collection: this.state().get('selection'),
                priority:   -40,
            }).render() );
        },

        addNewClickHandler: function( event ) {

            event.preventDefault();
            this.trigger( 'toggle:upload:attachment' );
        },

        browseContent: function( contentRegion ) {

            var state = this.state();

            this.$el.removeClass('hide-toolbar');

            // Browse our library of attachments.
            this.browserView = contentRegion.view = new media.view.AttachmentsBrowser({
                controller: this,
                collection: state.get('library'),
                selection:  state.get('selection'),
                model:      state,
                sortable:   state.get('sortable'),
                search:     state.get('searchable'),
                filters:    state.get('filterable'),
                date:       state.get('date'),
                display:    state.has('display') ? state.get('display') : state.get('displaySettings'),
                dragInfo:   state.get('dragInfo'),

                idealColumnWidth: state.get('idealColumnWidth'),
                suggestedWidth:   state.get('suggestedWidth'),
                suggestedHeight:  state.get('suggestedHeight'),

                AttachmentView: state.get('AttachmentView')
            });

            this.browserView.on( 'ready', _.bind( this.bindDeferred, this ) );
        },

        bindDeferred: function() {

            if ( ! this.browserView.dfd ) {
                return;
            }
            this.browserView.dfd.done( _.bind( this.startHistory, this ) );
        },

        startHistory: function() {
            
            // Verify pushState support and activate.
            if ( window.history && window.history.pushState ) {
                if ( Backbone.History.started ) {
                    Backbone.history.stop();
                }
                Backbone.history.start( {
                    root: window._wpMediaGridSettings.adminUrl,
                    pushState: true
                } );
            }
        },

        openEditAttachmentModal: function( model ) {

            media.frames.edit = wp.media( {
                frame:       'edit-attachments',
                controller:  this,
                library:     this.state().get('library'),
                model:       model
            } );
        }
    });



    original.UploaderInline = {

        show: media.view.UploaderInline.prototype.show,
        hide: media.view.UploaderInline.prototype.hide
    };

    _.extend( media.view.UploaderInline.prototype, {

        show: function() {

            var attachments;

            original.UploaderInline.show.apply( this, arguments );

            if ( this.controller.browserView ) {
                attachments = this.controller.browserView.attachmentsWrapper || this.controller.browserView.attachments;
                attachments.$el.css( 'top', this.$el.outerHeight() + 20 + 'px' );
            }
        },

        hide: function() {

            var attachments;

            original.UploaderInline.hide.apply( this, arguments );

            if ( this.controller.browserView ) {
                attachments = this.controller.browserView.attachmentsWrapper || this.controller.browserView.attachments;
                attachments.$el.css( 'top', 0 );
            }
        }
    });




    original.controllerLibrary = {

        beforeUpload: media.controller.Library.prototype.beforeUpload
    };

    _.extend( media.controller.Library.prototype, {

        beforeUpload: function() {

            original.controllerLibrary.beforeUpload.apply( this, arguments );
            this.frame.browserView.uploader.hide();
        }
    });



    //TODO: move to PHP side
    $('body').addClass('eml-grid');


})( jQuery, _ );