压缩[图库]中显示的所有图像,并提供下载链接


13

我想为访问者提供一个选项,可以下载整个照片库(显示在专用[画廊]页面上),并以ZIP文件的形式显示在每个画廊页面的底部。-必须包含完整尺寸的图像。

大卫·沃尔什(David Walsh)在此处发布一些代码来压缩文件,但是我很难将其与Wordpress函数集成在一起。

我知道有一个NextGEN Gallery下载插件,但是由于我使用本地wordpress gallery功能,因此无法使用该插件。

可以在此处找到类似的问题,用一种替代的方法(手动方法)来完成上述操作:可以下载附件的媒体文件的插件吗?

任何帮助将不胜感激。谢谢。


专门的画廊页面是什么意思?
2013年

标准帖子,仅显示画廊简码[gallery columns =“ 4” link =“ file”],页面上没有其他内容。我将它包括在描述中,以防万一它对开发人员有所帮助。
汤姆森

Answers:


14

首先,您必须获取图像。这里介绍如何获取画廊的所有图像。

WordPress使用两个类来解压缩文件。PHP中的代码ZipArchive()(参见David Walsh)。和PclZip,您可以在中找到此类wp-admin/includes/class-pclzip.php。如果您遇到问题,请ZipArchive()尝试PclZip类。

现在,您只需要将两者粘合在一起即可。也许以后我可以发布一些示例代码,目前我不在办公桌前。

更新资料

您的问题可以分为两部分。第一个是从图库中获取所有图像。第二个是压缩图像并发送压缩文件。
我只解释第一部分,获取画廊的所有图像,因为压缩文件有点不合时宜。

也许还有其他解决方案,但是在此示例中,我将原始的画廊短代码替换为自定义的画廊短代码以获取图像。原因是,WordPress稍微改变了v3.5中的图库。
在3.5之前,画廊的图像是帖子的附件。3.5之后,图像将作为属性传递到短代码。由于WP3.5,我们无法再获取帖子的附件图像,因此我们必须从shortcode属性中获取列表。我的策略是用自定义简码替换原始简码,获取属性并调用原始简码以获取图库输出。

所有画廊相关的东西都在一个类中。要创建一个zip文件,我们可以使用另一个类,该类将gallery类的输出作为输入。让我们从一个类和一个简单的构造函数开始。

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()稍后将在带有hook的插件中调用该方法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。比起include wp-includes/media.php,此文件包含原始库短代码的回调函数。现在,我们调用一个方法来获取一个包含所有图像的数组,通过调用原始的画廊回调来创建画廊输出,创建一个链接并将该链接附加到画廊输出。图像本身以及图像的路径都存储在class变量中$images,稍后我们需要此数组。
class变量$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属性。设置完所有变量后,我们终于可以从图库中获取图像。检索到的图像将被放入var类中,$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()并遍历检索到的附件。为了处理不同的大小,我们获得了附件图像和网址条。从附件中获取路径,并将其与文件名放在一起。$images现在在阵列中是所有图像及其来自图库的路径。

基本上,您的问题在这一点上得到了回答。但是您也想从图像创建一个zip文件。您可以$images在最后一个方法中从数组创建一个zip文件。但是,每次显示画廊时都会调用此方法,创建zip文件可能需要一段时间。也许没有人会要求您在此处创建的zip文件,这会浪费资源。

我们如何做得更好?您还记得我将所有图像都放在class变量中$images吗?我们可以将此类var用于ajax请求。但是ajax请求只是另一个页面加载,我们只有在创建画廊的输出后才能访问图像。我们必须将图像保存在一个位置,即使在另一个页面请求之后也可以访问它们。
在这个例子中,我使用一个会话变量来存储带有图像的数组。即使在另一个页面重新加载后,也可以访问会话变量。为了存储图像,我用shutdown钩子注册了一个方法。WordPress完成呈现页面后,shutdown将调用该挂钩。此时,我们应该已经从所有显示的画廊中收集了所有图像。我们只存储图像,并可以在ajax请求中访问它们。

触发ajax请求时,我们调用会话var并从数据创建一个zip文件。但这对于这个问题来说有点离题。

在GitHub上创建了带有完整插件代码的存储库。我希望它能为您指明正确的方向。


请注意,如果您要处理的是新的3.5样式画廊,则从画廊检索图像的方法可能不适合您。
米洛(Milo)

为了澄清我正在使用最新版本的Wordpress 3.5.1。Johannes,如果您下次在桌面上能够提供一些示例代码,我将不胜感激。保罗,谢谢
汤姆森

嗨,拉尔夫,这真是太棒了!感谢分享。为了使我的图库图像正确显示,我必须在短代码中包含link =“ file”,即:[gallery link =“ file”],因为通过您的代码重写了我的短代码,我认为这被排除在外了。结果我的画廊无法正确显示。有什么办法可以将该参数固定到您的代码中吗?
汤姆森

通常,属性只是简单地传递到原始的简码,并且不会被修改。画廊将正常显示,但会附加一些HTML。在我的测试(具有十二个standrad主题)中,画廊正确显示。
Ralf912

@PaulThomson我修复了github存储库中的一些问题。代码不干净。
Ralf912

0

我喜欢Ralf的插件可以一次性下载整个画廊的想法,但我一直无法使其工作。我想出了一种适合我们目的的解决方法。该方法是用您自己的WP Gallery替换本机WP Gallery,将其放在主题functions.php文件的末尾,然后将以下文件添加download.php到活动主题文件夹中。在自定义图片库中,该文件下的链接称为文件download.php,它会自动将您的文件下载到硬盘驱动器。我已经在最新的Chrome,Firefox和Safari版本上对此进行了测试,并且效果很好。一直在使用“十二十二”主题,但是没有理由也不能在其他主题上使用。

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)。不要忘记链接到图库中的文件!!重要!


我认为这不是一个好主意,使用download.php$_GET参数,我可以从您的Web文件夹(也包括网络共享)中下载我想要的所有文件,例如wp-config.php
Diego Betto
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.