[ギャラリー]に表示されているすべての画像を圧縮してダウンロードリンクとして提供する


13

訪問者に、各ギャラリーページの下部に表示されるZIPファイルとして、フォトギャラリー全体(専用の[ギャラリー]ページに表示される)をダウンロードするオプションを提供したいと思います。-フルサイズの画像を含める必要があります。

デビッド・ウォルシュはここで彼の投稿いくつかのコードを与えましたファイルを圧縮するためのが、Wordpressの機能と統合するのに問題があります。

NextGENギャラリーダウンロードプラグインがあることは承知していますが、ネイティブのワードプレスギャラリー機能を使用しているので、それを使用する立場にはありません。

上記を完了するための別の方法(手動による方法)に関する同様の質問は、ここにあります:添付メディアファイルをダウンロードするプラグイン?

どんな助けも大歓迎です。ありがとう。


専用のギャラリーページとはどういう意味ですか?
-NoBugs

ギャラリーショートコード[gallery columns = "4" link = "file"]のみを表示し、ページ上の他のコンテンツは表示しない標準投稿。開発者を助けるためだけに説明に含めました。
ポールトムソン

回答:


13

まず、画像を取得する必要があります。ギャラリーのすべての画像を取得する方法は、ここで説明されています

WordPressはファイルの解凍に2つのクラスを使用します。PHPビルトインZipArchive()(使用方法はDavid Walshを参照)。そしてPclZip、このクラスはにありwp-admin/includes/class-pclzip.phpます。問題が発生した場合はZipArchive()、PclZipクラスを試してください。

両方を接着する必要があります。あとでサンプルコードを投稿できるかもしれませんが、現在はデスクにいません。

更新

質問は2つの部分に分けることができます。最初の方法は、ギャラリーからすべての画像を取得することです。2つ目は、画像を圧縮し、zipファイルを送信します。
ファイルを圧縮することは少し話題に合わないため、ギャラリーのすべての画像を取得する最初の部分のみを説明します。

他の解決策があるかもしれませんが、この例では、元のギャラリーショートコードをカスタムのコードに置き換えて画像を取得します。その理由は、WordPressがv3.5のギャラリーを少し変更したためです。
3.5より前では、ギャラリーの画像は投稿の添付ファイルです。3.5以降、画像は属性としてショートコードに渡されます。WP3.5以降では、投稿の添付画像を取得できなくなったため、ショートコード属性からリストを取得する必要があります。私の戦略は、元のショートコードをカスタムショートコードに置き換え、属性を取得し、元のショートコードを呼び出してギャラリー出力を取得することです。

ギャラリー関連のものはすべてクラス内にあります。zipファイルを作成するには、ギャラリークラスの出力を入力として受け取る別のクラスを使用できます。クラスと単純なコンストラクターから始めましょう。

class GalleryZip
{
    private static $instance = null;

    public static $images = array();

    public static function get_instance() {
        if ( ! session_id() )
          session_start();

        if ( null === self::$instance )
            self::$instance = new self();

        return self::$instance;
    }

    private final function __construct() {
        remove_shortcode( 'gallery' );
        add_shortcode( 'gallery', array( __CLASS__, 'gallery_zip_shortcode' ) );
    }
}

get_instance()フックを使用して、プラグインの後半でメソッドを呼び出しますplugins_loaded。コンストラクターで、元のショートコードを削除し、カスタムショートコードに置き換えgallery_zip_shortcode()ます。ショートコードコールバックが必要になりました

public static function gallery_zip_shortcode( $atts ) {

    $post = get_post();

    if ( ! function_exists( 'gallery_shortcode' ) )
      require_once ABSPATH . 'wp-includes/media.php';

    self::get_gallery_images_from_shortcode( $post->ID, $atts );
    $output = gallery_shortcode( $atts );

    $gallery_id = count( self::$images[$post->ID] ) - 1;

    $link = sprintf( '<div><a href="#" gallery-id="%d" post-id="%d" class="gallery-zip">%s</a></div>', $gallery_id, $post->ID, __( 'Get as Zip' ) );
    $output .= $link;

    return $output;

}

このメソッドの最初のことは、投稿IDが必要なので、投稿を取得することです。を含めるよりもwp-includes/media.php、このファイルには元のギャラリーショートコードのコールバック関数が含まれています。次に、すべての画像を含む配列を取得するメソッドを呼び出し、元のギャラリーコールバックを呼び出してギャラリー出力を作成し、リンクを作成して、ギャラリー出力にリンクを追加します。画像自体、または画像へのパスはそれぞれクラス変数に格納されます$images。この配列は後で必要になります。
クラス変数に$imageはギャラリー付きの各投稿のエントリが保持されるため、フロントページまたはシングルビューで関数を使用できます。各投稿には複数のギャラリーが存在する可能性があるため、各エントリには各ギャラリーの配列が含まれます。

プラグインの中核は、ショートコードから画像を取得する方法です。

protected static function get_gallery_images_from_shortcode( $id, $atts ) {

    // use the post ID if the attribute 'ids' is not set or empty
    $id = ( ! isset( $atts['ids'] ) || empty( $atts['ids'] ) ) ?
        (int) $id : $atts['ids'];

    $exclude = ( isset( $atts['exclude'] ) && ! empty( $atts['exclude'] ) ) ?
        $atts['exclude'] : '';

    if ( ! isset( self::$images[$id] ) || ! is_array( self::$images[$id] ) )
        self::$images[$id] = array();

    $images = self::get_gallery_images( $id, $exclude );

    array_push( self::$images[$id], $images );

    return $images;

}

最初に、単一の投稿か、投稿IDのリストかを決定します。投稿IDのリストの場合、WP3.5 +のギャラリーを処理します。その後、exclude属性を処理する必要があります。すべての変数を設定した後、最終的にギャラリーから画像を取得できます。取得した画像は$images、後で使用するためにクラス変数にプッシュされます。

protected static function get_gallery_images( $id, $exclude ) {
    $images     = array();
    $query_args = array(
            'post_status'    => 'inherit',
            'post_type'      => 'attachment',
            'post_mime_type' => 'image',
    );

    // handle gallery WP3.5+
    // if $id contains an comma, it is a list of post IDs
    if ( false !== strpos( $id, ',' ) ) {
        $query_args['include'] = $id;
    } elseif ( ! empty( $exclude ) ) {
        // handle excluding posts
        $query_args['post_parent'] = $id;
        $query_args['exclude']     = $exclude;
    } else {
        // handle gallery before WP3.5
        $query_args['post_parent'] = $id;
    }

    $attachments = get_posts( $query_args );

    $img_sizes = array_merge( array( 'full' ), get_intermediate_image_sizes() );

    $img_size = ( in_array( self::IMAGE_SIZE, $img_sizes ) ) ?
            self::IMAGE_SIZE : 'full';

    foreach ( $attachments as $key => $post ) {
        $img = wp_get_attachment_image_src( $post->ID, $img_size, false, false );
        $images[] = sprintf( '%s/%s', dirname( get_attached_file( $post->ID ) ), basename( $img[0] ) );
    }

    return $images;
}

これはプラグインのゴールドです。クエリ引数を使用して配列を設定し、添付ファイルを取得して、取得した添付ファイルを簡単に調べますget_posts()。さまざまなサイズを処理するために、添付画像とURLのストリップを取得します。添付ファイルからパスを取得し、ファイル名と一緒に配置します。配列に$imagesは、すべての画像とギャラリーからのパスが含まれています。

基本的に、あなたの質問はこの時点で答えられます。ただし、画像からzipファイルを作成することもできます。$images最後の方法で配列からzipファイルを作成できます。ただし、このメソッドはギャラリーが表示されるたびに呼び出され、zipファイルの作成には時間がかかる場合があります。ここで作成したzipファイルを要求する人はいないでしょう。これはリソースの無駄です。

どうすればそれを改善できますか?すべての画像をクラス変数に入れたことを覚えています$imagesか?このクラス変数をajaxリクエストに使用できます。ただし、ajaxリクエストは単なるページロードであり、ギャラリーの出力が作成されたときにのみ画像にアクセスできます。別のページがリクエストされた後でも、画像にアクセスできる場所に画像を保存する必要があります。
この例では、セッション変数を使用して配列を画像とともに保存します。セッション変数は、別のページをリロードした後でもアクセスできます。画像を保存するには、shutdownフックにメソッドを登録します。WordPressがページのレンダリングを完了すると、shutdownフックが呼び出されます。この時点で、表示されているすべてのギャラリーからすべての画像を収集しているはずです。画像を保存するだけで、ajaxリクエストでそれらにアクセスできます。

ajaxリクエストがトリガーされると、セッション変数を呼び出して、データからzipファイルを作成します。しかし、これはこの質問のトピックから少し外れています。

完全なプラグインコードでGitHubにリポジトリを作成しました。それが正しい方向にあなたを指し示すことを願っています。


新しい3.5スタイルのギャラリーを扱っている場合、ギャラリーから画像を取得するこの方法は機能しない場合があります。
ミロ

明確にするために、私はWordpress 3.5.1の最新バージョンを使用しています。ヨハネス、デスクトップに次に来たときにサンプルコードを提供できたら、ありがたいです。ありがとう、ポール
ポール・トムソン

ちょっとラルフ、これは素晴らしい作品です!共有してくれてありがとう。ギャラリーの画像を適切に表示するには、ショートコードにlink = "file"を含める必要があります。つまり、[gallery link = "file"]は、コードを介してショートコードが書き直されるため、結果、ギャラリーが正しく表示されません。その引数をコードに修正する方法はありますか?
ポールトムソン

通常、属性は単純に元のショートコードに渡され、変更されません。ギャラリーは通常どおり表示されますが、HTMLが追加されます。私のテスト(20個の標準テーマを使用)では、ギャラリーが正しく表示されます。
Ralf912

@PaulThomson githubリポジトリのいくつかの問題を修正しました。コードはきれいではありませんでした。
-Ralf912

0

一度にギャラリー全体をダウンロードできるというRalfのプラグインのアイデアは気に入っていますが、機能させることができていません。私たちの目的に合った回避策を思いつきました。その方法は、ネイティブWPギャラリーを、テーマのfunctions.phpファイルの最後に配置した独自のWPギャラリーに置き換えdownload.php、アクティブなテーマフォルダーに名前が付けられた次のファイルを追加することです。カスタムギャラリーでは、ファイルの下のリンクがファイルdownload.phpを呼び出します。これにより、ファイルがハードドライブに自動的にダウンロードされます。これを最新のChrome、Firefox、Safariバージョンでテストしましたが、正常に動作します。Twenty Twelveテーマを使用してきましたが、他のテーマでも動作しない理由はありません。

a)の末尾に次を追加しますfunctions.php。これは単にmedia.phpから取得されます

remove_shortcode('gallery');
function gallery_with_download_links($attr) {
    $post = get_post();
    static $instance = 0;
    $instance++;
    if ( ! empty( $attr['ids'] ) ) {
        // 'ids' is explicitly ordered, unless you specify otherwise.
        if ( empty( $attr['orderby'] ) )
            $attr['orderby'] = 'post__in';
        $attr['include'] = $attr['ids'];
    }
    // Allow plugins/themes to override the default gallery template.
    $output = apply_filters('post_gallery', '', $attr);
    if ( $output != '' )
        return $output;
    // We're trusting author input, so let's at least make sure it looks like a valid orderby statement
    if ( isset( $attr['orderby'] ) ) {
        $attr['orderby'] = sanitize_sql_orderby( $attr['orderby'] );
        if ( !$attr['orderby'] )
            unset( $attr['orderby'] );
    }

    extract(shortcode_atts(array(
        'order'      => 'ASC',
        'orderby'    => 'menu_order ID',
        'id'         => $post->ID,
        'itemtag'    => 'dl',
        'icontag'    => 'dt',
        'captiontag' => 'dd',
        'columns'    => 3,
        'size'       => 'thumbnail',
        'include'    => '',
        'exclude'    => ''
    ), $attr));

    $id = intval($id);
    if ( 'RAND' == $order )
        $orderby = 'none';

    if ( !empty($include) ) {
        $_attachments = get_posts( array('include' => $include, 'post_status' => 'inherit', 'post_type' => 'attachment', 'post_mime_type' => 'image', 'order' => $order, 'orderby' => $orderby) );

        $attachments = array();
        foreach ( $_attachments as $key => $val ) {
            $attachments[$val->ID] = $_attachments[$key];
        }
    } elseif ( !empty($exclude) ) {
        $attachments = get_children( array('post_parent' => $id, 'exclude' => $exclude, 'post_status' => 'inherit', 'post_type' => 'attachment', 'post_mime_type' => 'image', 'order' => $order, 'orderby' => $orderby) );
    } else {
        $attachments = get_children( array('post_parent' => $id, 'post_status' => 'inherit', 'post_type' => 'attachment', 'post_mime_type' => 'image', 'order' => $order, 'orderby' => $orderby) );
    }

    if ( empty($attachments) )
        return '';

    if ( is_feed() ) {
        $output = "\n";
        foreach ( $attachments as $att_id => $attachment )
            $output .= wp_get_attachment_link($att_id, $size, true) . "\n";
        return $output;
    }

    $itemtag = tag_escape($itemtag);
    $captiontag = tag_escape($captiontag);
    $icontag = tag_escape($icontag);
    $valid_tags = wp_kses_allowed_html( 'post' );
    if ( ! isset( $valid_tags[ $itemtag ] ) )
        $itemtag = 'dl';
    if ( ! isset( $valid_tags[ $captiontag ] ) )
        $captiontag = 'dd';
    if ( ! isset( $valid_tags[ $icontag ] ) )
        $icontag = 'dt';

    $columns = intval($columns);
    $itemwidth = $columns > 0 ? floor(100/$columns) : 100;
    $float = is_rtl() ? 'right' : 'left';

    $selector = "gallery-{$instance}";

    $gallery_style = $gallery_div = '';
    if ( apply_filters( 'use_default_gallery_style', true ) )
        $gallery_style = "
        <style type='text/css'>
            #{$selector} {
                margin: auto;
            }
            #{$selector} .gallery-item {
                float: {$float};
                margin-top: 10px;
                text-align: center;
                width: {$itemwidth}%;
            }
            #{$selector} img {
                border: 2px solid #cfcfcf;
            }
            #{$selector} .gallery-caption {
                margin-left: 0;
            }
        </style>
        <!-- see gallery_shortcode() in wp-includes/media.php -->";
    $size_class = sanitize_html_class( $size );
    $gallery_div = "<div id='$selector' class='gallery galleryid-{$id} gallery-columns-{$columns} gallery-size-{$size_class}'>";
    $output = apply_filters( 'gallery_style', $gallery_style . "\n\t\t" . $gallery_div );

    $i = 0;
    foreach ( $attachments as $id => $attachment ) {
        $link = isset($attr['link']) && 'file' == $attr['link'] ? wp_get_attachment_link($id, $size, false, false) : wp_get_attachment_link($id, $size, true, false);

        $output .= "<{$itemtag} class='gallery-item'>";
        $output .= "
            <{$icontag} class='gallery-icon'>
                $link
            </{$icontag}>";
        if ( $captiontag && trim($attachment->post_excerpt) ) {
            $output .= "
                <{$captiontag} class='wp-caption-text gallery-caption'>
                " . wptexturize($attachment->post_excerpt) . "
                </{$captiontag}>";
        }
// This is my addon which outputs a link to download the file through download.php . NB your file uri will be public! 
        $output .= '<br/ ><a href="'.get_template_directory_uri().'/download.php?file='.get_attached_file( $id ).'">Download image</a>';
        $output .= "</{$itemtag}>";
        if ( $columns > 0 && ++$i % $columns == 0 )
            $output .= '<br style="clear: both" />';
    }

    $output .= "
            <br style='clear: both;' />
        </div>\n";

    return $output;
}
add_shortcode( 'gallery' , 'gallery_with_download_links' );

b)以下をコピーdownload.phpして、テーマのベースディレクトリで呼び出されるファイルに貼り付けます。

<?php
$file = $_GET['file'];
if (file_exists($file)) {
header('Content-Description: File Transfer');
header('Content-Type: application/octet-stream');
header('Content-Disposition: attachment; filename='.basename($file));
header('Content-Transfer-Encoding: binary');
header('Expires: 0');
header('Cache-Control: must-revalidate');
header('Pragma: public');
header('Content-Length: ' . filesize($file));
ob_clean();
flush();
readfile($file);
exit;
}
?>

c)。ギャラリー内のファイルへのリンクを忘れないでください!! 重要!

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.