Löschen Sie einen Anhang im WP Media Modal-Fenster

15

Ich versuche, eine Option im Fenster für modale WP-Medien zu erstellen, um doppelte Dateien zu erkennen und eine neuere Datei zu löschen, wenn ein älteres Duplikat gefunden wird. Ich habe den folgenden Code (in Verbindung mit dem Filter 'attachment_fields_to_edit'), um die Auswahl einer doppelten Datei aufzuheben und die Originaldatei im Medien-Modal auszuwählen. Was ich tun möchte, ist, wenn ein Benutzer auf die Schaltfläche klickt, die Originaldatei zu löschen (oder sie zumindest im Fenster der Medienbibliothek auszublenden, damit ich sie später löschen kann).

( function( $ ) {

    var _AttachmentDisplay = wp.media.view.Settings.AttachmentDisplay;
    wp.media.view.Settings.AttachmentDisplay = _AttachmentDisplay.extend({
        render: function() {
            _AttachmentDisplay.prototype.render.apply(this, arguments);
            currentselection = this.controller.state().get('selection').first().toJSON();
            selection = this.controller.state().get('selection');

            $('button.dmc').on('click', function(e){

                e.preventDefault();

                var id = $(e.currentTarget).data("id");
                if(currentselection.id == id) {

                    currentattachment = wp.media.attachment(id);
                    selection.remove(currentattachment);

                    console.dir(wp.media.view.Attachment);

                    newattachment = wp.media.attachment($(e.currentTarget).data("original"));
                    selection.add(newattachment);

                }
            });
        }
    });

} )( jQuery );

Die Oberfläche sieht aus wie das angehängte Bild.

Screenshot der Medienüberprüfung

Ich kann in media-views.js in Zeile 5873 sehen, dass es eine Funktion deleteAttachment gibt, die an das Klicken auf .delete-attachment gebunden ist. Wie kann ich bei meinem aktuellen Setup darauf zugreifen, indem ich eine Bild-ID oder ein Anhangsobjekt übergebe?

bcorkins
quelle
4
Ist diese Frage noch offen oder haben Sie schon eine Antwort gefunden?
Engelen
@engelen diese frage haben jetzt ein offenes kopfgeld. Also los :-)
Pieter Goosen

Antworten:

5

Wenn Sie versuchen, eine etwas kanonische (oder zumindest reichhaltige) Antwort zu finden, ist dies das Javascript für wpse142997.jsdas untergeordnete Vorlagenverzeichnis:

jQuery( document ).ready(function() {
    ( function( $ ) {
        var media = wp.media,
            l10n = media.view.l10n = typeof _wpMediaViewsL10n === 'undefined' ? {} : _wpMediaViewsL10n,
            attachments = media.model.Attachments.all,
            attachments_uploaded = [];

        if ( typeof wp.Uploaded === 'undefined') return;

        // Keep track of files uploaded.
        wp.Uploader.queue.on( 'add', function ( attachment ) {
            attachments_uploaded.push( attachment );
        });

        // The Uploader (in wp-includes/js/plupload/wp-plupload.js) resets the queue when all uploads are complete.
        wp.Uploader.queue.on( 'reset', function () {
            var idx, uploaded = attachments_uploaded.slice(0); // Clone
            attachments_uploaded = [];
            for ( idx = 0; idx < uploaded.length; idx++ ) {
                if ( uploaded[idx].get('name').match(/-[0-9]+$/) ) {
                    $.post( ajaxurl, {
                            action: 'wpse142997_is_dup',
                            dup_id: uploaded[idx].id,
                            nonce: wpse142997_params.is_dup_nonce
                        }, function( response ) {
                            var original, dup, dup_view, sidebar, selection;
                            if ( response && !response.error && response.original_id && response.dup_id ) {
                                original = attachments.get( response.original_id );
                                dup = attachments.get( response.dup_id );
                                if ( original && dup ) {
                                    dup.set( 'dup_original', original ); // May be ungood - mostly doing it so can use wp.templates.
                                    dup_view = media.view.Attachment.extend({
                                        tagName:   'div',
                                        className: 'attachment-dmc',
                                        template: media.template('attachment-dmc'),
                                        events: {
                                            'click button.dmc': 'removeDupSelectOriginal'
                                        },
                                        initialize: function() {
                                            this.focusManager = new media.view.FocusManager({
                                                el: this.el
                                            });
                                            media.view.Attachment.prototype.initialize.apply( this, arguments );
                                        },
                                        render: function() {
                                            if ( this.get_dup_original() ) {
                                                media.view.Attachment.prototype.render.apply( this, arguments );
                                                this.focusManager.focus();
                                            }
                                            return this;
                                        },
                                        removeDupSelectOriginal: function( event ) {
                                            var dup_original = this.get_dup_original();
                                            event.preventDefault();

                                            if ( dup_original && confirm( l10n.warnDelete ) ) {
                                                this.model.destroy();
                                                this.controller.state().get('selection').add( dup_original );
                                                this.remove();
                                            }
                                        },
                                        get_dup_original: function () {
                                            var dup_original = this.model.get('dup_original');
                                            return dup_original && attachments.get( dup_original.id ) ? dup_original : null;
                                        }
                                    });
                                    // A hacky way to get the sidebar.
                                    sidebar = media.frame.content.view.views.get('.media-frame-content')[0].sidebar;
                                    selection = sidebar.controller.state().get('selection');
                                    // The sidebar boxes get deleted and recreated on each select - hack into this to do the same.
                                    selection.on( 'selection:single', function ( event ) {
                                        if ( selection.single().get('dup_original') ) {
                                            sidebar.set( 'dmc', new dup_view({
                                                controller: sidebar.controller,
                                                model: selection.single(),
                                                priority: 100
                                            }) );
                                        }
                                    } );
                                    selection.on( 'selection:unsingle', function ( event ) {
                                        sidebar.unset('dmc');
                                    } );
                                    // Refire the select as we missed it (could/should just do the view create code here again).
                                    selection.trigger('selection:single');
                                }
                            }
                        }, 'json'
                    );
                }
            }
        });
    } )( jQuery );
});

Dies ist der functions.php:

function wpse142997_wp_enqueue_scripts() {
    wp_enqueue_script( 'wpse142997', get_stylesheet_directory_uri() . '/wpse142997.js', array( 'jquery', 'media-views' ), '1.0' );
    $params = array(
        'is_dup_nonce' => wp_create_nonce( 'wpse142997_is_dup_submit_' ),
    );
    wp_localize_script( 'wpse142997', 'wpse142997_params', $params );
    ob_start();
    ?>
<style>
.attachment-dmc { float:left; overflow:hidden; position:relative; }
.attachment-dmc div { background-color:#FFEBE7; border:1px solid #CB9495; border-radius:5px; margin-top:16px; padding:6px; }
.attachment-dmc div h3 { margin-top:0; }
.attachment-dmc div h3 span { background-color:#E70000; border-radius:5px; color:white; margin-top:0; padding:0 6px; }
</style>
    <?php
    wp_add_inline_style( 'media-views', str_replace( array( '<style>', '</style>' ), '', ob_get_clean() ) );
}

function wpse142997_print_media_templates() {
?>
<script type="text/html" id="tmpl-attachment-dmc">
    <# if ( data.dup_original ) { #>
        <div>
            <h3><span><?php _e( 'Duplicate file detected' ); ?></span></h3>
            <p>
                <?php _e( 'This file appears to be a duplicate of <a href="{{ data.dup_original.attributes.editLink }}&amp;image-editor" target="_blank">{{ data.dup_original.attributes.filename }}</a> uploaded on {{ data.dup_original.attributes.dateFormatted }}' ); ?>
            </p>
            <button id="run_dmc" class="dmc" name="dmc"><?php _e( 'Remove duplicate and select original' ); ?></button>
        </div>
    <# } #>
</script>
<?php
}

function wpse142997_is_dup() {
    $ret = array( 'error' => false );

    if ( ! check_ajax_referer( 'wpse142997_is_dup_submit_', 'nonce', false /*die*/ ) ) {
        $ret['error'] = __( 'Permission error' );
    } else {
        $dup_id = isset( $_POST['dup_id'] ) ? $_POST['dup_id'] : '';
        if ( ! $dup_id || ! ( $post = get_post( $dup_id ) ) ) {
            $ret['error'] = __( 'Bad dup_id' );
        } else {
            $post_name = preg_replace( '/-[0-9]+$/', '', $post->post_name ); 
            global $wpdb;
            $sql = $wpdb->prepare( 'SELECT ID FROM ' . $wpdb->posts . ' WHERE'
                . ' post_title = %s AND post_type = %s AND post_mime_type = %s AND post_status = %s AND post_name = %s ORDER BY post_date ASC LIMIT 1',
                $post->post_title, $post->post_type, $post->post_mime_type, $post->post_status, $post_name
            );
            if ( $original_id = $wpdb->get_var( $sql ) ) {
                $ret['original_id'] = $original_id;
                $ret['dup_id'] = $dup_id;
            }
        }
    }

    wp_send_json( $ret );
}

add_action( 'admin_enqueue_scripts', 'wpse142997_wp_enqueue_scripts' );
add_action( 'print_media_templates', 'wpse142997_print_media_templates' );
add_action( 'wp_ajax_wpse142997_is_dup', 'wpse142997_is_dup' );

Das Javascript versucht, so weit ich es verstehe, dem WP Media Modal-Weg zu folgen, der nur teilweise ist. Es erstellt eine media.view.Attachmentund verwendet eine wp.templateVorlage. Es gibt ein paar hacky Bits - insbesondere die Sidebar über eine lange Reichweite in das Frame-Objekt zu bekommen, scheint verdächtig (und wurde erst nach langem Stöbern gefunden).

Bonger
quelle
Danke, das ist definitiv das, wonach ich gesucht habe. Was den kanonischen Teil angeht, gibt es nicht viele Gründe, aus denen ich wählen könnte, und ich konnte mir keinen besseren Text selbst ausdenken, also ist dies ziemlich nahe. Ich werde nächste Woche genauer hinschauen.
Nicolai
Ich habe den Code in die Datei functions.php von 2014 kopiert und die js wie beschrieben eingerichtet und kann den doppelten Benachrichtigungsabschnitt immer noch nicht sehen. Welche Aktionen muss ich ausführen, wenn der Code vorhanden ist, um dies anzuzeigen?
KalenGi
Ich nehme an, Sie haben ein doppeltes Bild hochgeladen?! ...
Bonger
4

Sie müssen nur die destroyMethode für das attachmentModell aufrufen . Dadurch wird sowohl der Anhang aus der Medienbibliotheksansicht entfernt als auch ein Ajax-Aufruf an das Backend gesendet, um den Anhang in der Datenbank und alle verknüpften Dateien im Upload-Verzeichnis zu löschen.

Sie müssen den Anhang nicht in JSON konvertieren, um die ID zu erhalten. Sie können die Backbone-Modelle direkt bearbeiten. Das selectionist eine Sammlung von mehreren Anhängen.

( function( $ ) {

    var _AttachmentDisplay = wp.media.view.Settings.AttachmentDisplay;
    wp.media.view.Settings.AttachmentDisplay = _AttachmentDisplay.extend({
        render: function() {
            _AttachmentDisplay.prototype.render.apply(this, arguments);

            $('button.dmc').on('click', $.proxy(function(e){

                e.preventDefault();
                selection = this.controller.state().get('selection');
                firstAttachment = selection.first();

                var id = $(e.currentTarget).data("id");
                if(currentselection.id == id) {

                    selection.remove(firstAttachment);
                    firstAttachment.destroy();

                    console.dir(wp.media.view.Attachment);

                    newattachment = wp.media.attachment($(e.currentTarget).data("original"));
                    selection.add(newattachment);

                }
            }, this));
        }
    });

} )( jQuery );

Ich habe auch einen $ .proxy- Aufruf hinzugefügt , um thisden Click-Event-Callback verwenden zu können.

Fabien Quatravaux
quelle
1
Danke, sieht für mich gut aus. Natürlich sind meine JS-Fähigkeiten und Kenntnisse dieser Teile von WordPress nicht so gut. Ich möchte es auf jeden Fall nächste Woche versuchen. Eine andere Sache, lassen Sie uns so tun, als ob ich nicht viel über WP weiß. Wie kann ich den von Ihnen geposteten Code verwenden?
Nicolai
Da ich keine Zeit gefunden habe, eine Ihrer Lösungen auszuprobieren, habe ich die Antwort von @bonger mit einer Prämie belegt - nur weil es vollständiger zu sein scheint, nochmals vielen Dank für die Antwort.
Nicolai