Server IP : 15.235.198.142 / Your IP : 216.73.216.190 Web Server : Apache/2.4.58 (Ubuntu) System : Linux ballsack 6.8.0-45-generic #45-Ubuntu SMP PREEMPT_DYNAMIC Fri Aug 30 12:02:04 UTC 2024 x86_64 User : www-data ( 33) PHP Version : 8.3.6 Disable Function : NONE MySQL : OFF | cURL : ON | WGET : ON | Perl : ON | Python : OFF | Sudo : ON | Pkexec : OFF Directory : /var/www/rhodeworks/wp-content/plugins/nextgen-gallery/src/Legacy/admin/ |
Upload File : |
<?php use Imagely\NGG\DataMappers\Gallery as GalleryMapper; use Imagely\NGG\DataMappers\Image as ImageMapper; use Imagely\NGG\DataStorage\Manager as StorageManager; use Imagely\NGG\DataStorage\Sanitizer; use Imagely\NGG\Settings\Settings; use Imagely\NGG\Util\{ Router, Security }; // Exit if accessed directly. if ( ! defined( 'ABSPATH' ) ) { exit; } class nggManageGallery { public $mode = 'main'; public $gid = false; public $gallery = null; public $pid = false; public $base_page = 'admin.php?page=nggallery-manage-gallery'; public $search_result = false; // initiate the manage page. public function __construct() { // GET variables. if ( isset( $_GET['gid'] ) ) { $this->gid = (int) $_GET['gid']; $this->gallery = GalleryMapper::get_instance()->find( $this->gid, true ); } if ( isset( $_GET['pid'] ) ) { $this->pid = (int) $_GET['pid']; } if ( isset( $_GET['mode'] ) ) { $this->mode = trim( sanitize_text_field( wp_unslash( $_GET['mode'] ) ) ); } // Check for pagination request, avoid post process of other submit button, exclude search results. // Nonce verification not necessary here: we are only determining which page to view, whose ID is always // cast to an integer. NextGEN has historically used POST in places it shouldn't, such as pagination here. // // phpcs:disable WordPress.Security.NonceVerification.Missing if ( isset( $_POST['post_paged'] ) && ! isset( $_GET['s'] ) ) { if ( $_GET['paged'] != $_POST['post_paged'] ) { $_GET['paged'] = absint( $_POST['post_paged'] ); return; } } // phpcs:enable WordPress.Security.NonceVerification.Missing // Should be only called via manage galleries overview. if ( isset( $_POST['nggpage'] ) && 'manage-galleries' === $_POST['nggpage'] && wp_verify_nonce( sanitize_key( wp_unslash( $_POST['_wpnonce'] ) ), 'ngg_bulkgallery' ) ) { $this->post_processor_galleries(); } // Should be only called via a edit single gallery page. if ( isset( $_POST['nggpage'] ) && 'manage-images' === $_POST['nggpage'] && wp_verify_nonce( sanitize_key( wp_unslash( $_POST['_wpnonce'] ) ), 'ngg_updategallery' ) ) { $this->post_processor_images(); } // Look for other POST process. if ( ! empty( $_POST ) || ! empty( $_GET ) ) { $this->processor(); } M_NextGen_Admin::emit_do_notices_action(); } public function controller() { switch ( $this->mode ) { case 'sort': include_once __DIR__ . '/manage-sort.php'; nggallery_sortorder( $this->gid ); break; case 'edit': $this->setup_gallery_fields(); $this->setup_image_rows(); include_once __DIR__ . '/manage-images.php'; nggallery_picturelist( $this ); break; case 'main': default: include_once __DIR__ . '/manage-galleries.php'; nggallery_manage_gallery_main(); break; } } public function processor() { global $ngg, $nggdb; // Delete a picture. if ( $this->mode == 'delpic' ) { // TODO:Remove also Tag reference. check_admin_referer( 'ngg_delpicture' ); $image = $nggdb->find_image( $this->pid ); if ( $image ) { do_action( 'ngg_delete_picture', $this->pid, $image ); if ( $ngg->options['deleteImg'] ) { $storage = StorageManager::get_instance(); $storage->delete_image( $this->pid ); } $mapper = ImageMapper::get_instance(); $result = $mapper->destroy( $this->pid ); if ( $result ) { nggGallery::show_message( __( 'Picture', 'nggallery' ) . ' \'' . $this->pid . '\' ' . __( 'deleted successfully', 'nggallery' ) ); } } $this->mode = 'edit'; // show pictures. } // Recover picture from backup. if ( $this->mode == 'recoverpic' ) { check_admin_referer( 'ngg_recoverpicture' ); // bring back the old image. nggAdmin::recover_image( $this->pid ); nggGallery::show_message( __( 'Operation successful. Please clear your browser cache.', 'nggallery' ) ); $this->mode = 'edit'; // show pictures. } // will be called after a ajax operation. if ( isset( $_POST['ajax_callback'] ) ) { if ( $_POST['ajax_callback'] == 1 ) { nggGallery::show_message( __( 'Operation successful. Please clear your browser cache.', 'nggallery' ) ); } } // show sort order. if ( isset( $_POST['sortGallery'] ) ) { $this->mode = 'sort'; } if ( isset( $_GET['s'] ) ) { $this->search_images(); } } public function setup_image_rows() { add_filter( 'ngg_manage_images_row', [ &$this, 'render_image_row' ], 10, 2 ); add_filter( 'ngg_manage_images_column_1_header', [ &$this, 'render_image_column_1_header' ] ); add_filter( 'ngg_manage_images_column_1_content', [ &$this, 'render_image_column_1' ], 10, 2 ); add_filter( 'ngg_manage_images_column_2_header', [ &$this, 'render_image_column_2_header' ] ); add_filter( 'ngg_manage_images_column_2_content', [ &$this, 'render_image_column_2' ], 10, 2 ); add_filter( 'ngg_manage_images_column_3_header', [ &$this, 'render_image_column_3_header' ] ); add_filter( 'ngg_manage_images_column_3_content', [ &$this, 'render_image_column_3' ], 10, 2 ); add_filter( 'ngg_manage_images_column_4_header', [ &$this, 'render_image_column_4_header' ] ); add_filter( 'ngg_manage_images_column_4_content', [ &$this, 'render_image_column_4' ], 10, 2 ); add_filter( 'ngg_manage_images_column_5_header', [ &$this, 'render_image_column_5_header' ] ); add_filter( 'ngg_manage_images_column_5_content', [ &$this, 'render_image_column_5' ], 10, 2 ); add_filter( 'ngg_manage_images_column_6_header', [ &$this, 'render_image_column_6_header' ] ); add_filter( 'ngg_manage_images_column_6_content', [ &$this, 'render_image_column_6' ], 10, 2 ); } public function render_image_column_1_header() { return '<input type="checkbox" id="cb-select-all-1" onclick="checkAll(document.getElementById(\'updategallery\'));">'; } public function render_image_column_2_header() { return __( 'ID', 'nggallery' ); } public function render_image_column_3_header() { return __( 'Thumbnail', 'nggallery' ); } public function render_image_column_4_header() { return __( 'Filename', 'nggallery' ); } public function render_image_column_5_header() { return __( 'Alt & Title Text / Description', 'nggallery' ); } public function render_image_column_6_header() { return __( 'Tags', 'nggallery' ); } public function render_image_column_1( $output = '', $picture = [] ) { return "<input type='checkbox' name='doaction[]' value='{$picture->pid}'/>"; } public function render_image_column_2( $output = '', $picture = [] ) { return $picture->pid; } public function render_image_column_3( $output = '', $picture = [] ) { $image_url = add_query_arg( 'i', mt_rand(), $picture->imageURL ); $thumb_url = add_query_arg( 'i', mt_rand(), $picture->thumbURL ); $filename = esc_attr( $picture->filename ); $output = []; $output[] = "<a href='{$image_url}' class='thickbox' title='{$filename}'>"; $output[] = "<img class='thumb' src='{$thumb_url}' id='thumb{$picture->pid}'/>"; $output[] = '</a>'; return ( $output = implode( "\n", $output ) ); } public function render_image_column_4( $output = '', $picture = [] ) { $image_url = Router::esc_url( $picture->imageURL ); $filename = esc_attr( $picture->filename ); $caption = esc_html( ( empty( $picture->alttext ) ? $picture->filename : $picture->alttext ) ); $data_caption = esc_attr( $caption ); $caption = \Imagely\NGG\Display\I18N::ngg_plain_text_alt_title_attributes( $caption ); $caption = esc_attr( $caption ); $date = mysql2date( get_option( 'date_format' ), $picture->imagedate ); $width = $picture->meta_data['width']; $height = $picture->meta_data['height']; $pixels = "{$width} x {$height} pixels"; $excluded = checked( $picture->exclude, 1, false ); $exclude_label = __( 'Exclude ?', 'nggallery' ); $output = []; $output[] = "<div><strong><a href='{$image_url}' class='thickbox' title='{$caption}' data-view-title='{$data_caption}'>{$filename}</a></strong></div>"; $output[] = '<div class="meta">' . esc_html( $date ) . '</div>'; $output[] = "<div class='meta'>{$pixels}</div>"; $output[] = "<label for='exclude_{$picture->pid}'>"; $output[] = "<input type='checkbox' id='exclude_{$picture->pid}' value='1' name='images[{$picture->pid}][exclude]' {$excluded}/> {$exclude_label}"; $output[] = '</label>'; return ( $output = implode( "\n", $output ) ); } public function render_image_column_5( $output = '', $picture = [] ) { $alttext = \Imagely\NGG\Display\I18N::ngg_sanitize_text_alt_title_desc( $picture->alttext ); $desc = \Imagely\NGG\Display\I18N::ngg_sanitize_text_alt_title_desc( $picture->description ); $output = []; $output[] = "<input title='Alt/Title Text' type='text' name='images[{$picture->pid}][alttext]' value='{$alttext}'/>"; $output[] = "<textarea title='Description' rows='3' name='images[$picture->pid][description]'>{$desc}</textarea>"; return ( $output = implode( "\n", $output ) ); } public function render_image_column_6( $output = '', $picture = [] ) { global $wp_version; $fields = version_compare( $wp_version, '4.6', '<=' ) ? 'fields=names' : [ 'fields' => 'names' ]; $tags = wp_get_object_terms( $picture->pid, 'ngg_tag', $fields ); if ( is_array( $tags ) ) { $tags = implode( ', ', $tags ); } $tags = esc_html( $tags ); return "<textarea rows='4' name='images[{$picture->pid}][tags]'>{$tags}</textarea>"; } public function render_image_row( $picture, $counter ) { // Get number of columns. $class = ! ( $counter % 2 == 0 ) ? '' : 'alternate'; $columns = apply_filters( 'ngg_manage_images_number_of_columns', 6 ); // Get the valid row actions. $actions = []; $row_actions = apply_filters( 'ngg_manage_images_row_actions', [ 'view' => [ &$this, 'render_view_action_link' ], 'meta' => [ &$this, 'render_meta_action_link' ], 'custom_thumb' => [ &$this, 'render_custom_thumb_action_link' ], 'rotate' => [ &$this, 'render_rotate_action_link' ], 'recover' => [ &$this, 'render_recover_action_link' ], 'delete' => [ &$this, 'render_delete_action_link' ], ] ); foreach ( $row_actions as $id => $callback ) { if ( is_callable( $callback ) ) { $result = call_user_func( $callback, $id, $picture ); if ( $result ) { $actions[] = $result; } } } // Output row columns. echo "<tr class='{$class} iedit' valign='top'>"; for ( $i = 1; $i <= $columns; $i++ ) { $rowspan = $i > 4 ? "rowspan='2'" : ''; echo "<td class='column column-{$i}' {$rowspan}>"; echo apply_filters( "ngg_manage_images_column_{$i}_content", '', $picture ); echo '</td>'; } echo '</tr>'; // Actions row. echo "<tr class='{$class} row_actions'>"; echo '<td colspan="2"></td>'; echo "<td colspan='" . ( $columns - 2 ) . "'>"; echo "<div class='row-actions'>"; echo implode( ' | ', $actions ); echo '</div>'; echo '</td>'; echo '</tr>'; } public function render_view_action_link( $id, $picture ) { $image_url = Router::esc_url( $picture->imageURL ); $label = esc_html__( 'View', 'nggallery' ); $alt_text = empty( $picture->alttext ) ? $picture->filename : $picture->alttext; $data_title = esc_attr( __( 'View', 'nggallery' ) ) . ' "' . esc_attr( $alt_text ) . '"'; $alt_text = \Imagely\NGG\Display\I18N::ngg_plain_text_alt_title_attributes( $alt_text ); $title = esc_attr( __( 'View', 'nggallery' ) ) . ' "' . esc_attr( $alt_text ) . '"'; return "<a href='{$image_url}' class='thickbox' title='{$title}' data-view-title='{$data_title}'>{$label}</a>"; } public function render_meta_action_link( $id, $picture ) { $url = Router::esc_url( NGGALLERY_URLPATH . 'admin/showmeta.php?id=' . $picture->pid . '&nonce=' . \wp_create_nonce( 'ngg_meta_popup' ) ); $title = esc_attr__( 'Show meta data', 'nggallery' ); $label = esc_html__( 'Meta', 'nggallery' ); return "<a href='{$url}' class='ngg-dialog' title='{$title}'>{$label}</a>"; } public function render_custom_thumb_action_link( $id, $picture ) { $url = Router::esc_url( NGGALLERY_URLPATH . 'admin/edit-thumbnail.php?id=' . $picture->pid . '&nonce=' . \wp_create_nonce( 'ngg_edit_thumbnail' ) ); $title = esc_attr__( 'Customize thumbnail', 'nggallery' ); $label = esc_html__( 'Edit thumb', 'nggallery' ); return "<a href='{$url}' class='ngg-dialog' title='{$title}'>{$label}</a>"; } public function render_rotate_action_link( $id, $picture ) { $url = Router::esc_url( NGGALLERY_URLPATH . 'admin/rotate.php?id=' . $picture->pid . '&nonce=' . \wp_create_nonce( 'ngg_edit_rotation' ) ); $title = esc_attr__( 'Rotate', 'nggallery' ); $label = esc_html__( 'Rotate', 'nggallery' ); return "<a href='{$url}' class='ngg-dialog' title='{$title}'>{$label}</a>"; } public function render_recover_action_link( $id, $picture ) { if ( ! file_exists( $picture->imagePath . '_backup' ) ) { return false; } $url = wp_nonce_url( "admin.php?page=nggallery-manage-gallery&mode=recoverpic&gid={$picture->galleryid}&pid={$picture->pid}", 'ngg_recoverpicture' ); $title = esc_attr__( 'Recover image from backup', 'nggallery' ); $label = esc_html__( 'Recover', 'nggallery' ); $question = __( 'Recover', 'nggallery' ); $alttext = empty( $picture->alttext ) ? $picture->filename : $picture->alttext; $alttext = Sanitizer::strip_html( html_entity_decode( $alttext ), true ); $alttext = htmlentities( $alttext, ENT_QUOTES | ENT_HTML401 ); $alttext = \Imagely\NGG\Display\I18N::ngg_plain_text_alt_title_attributes( $alttext ); $alttext = esc_attr( $alttext ); // Event handler is found in nextgen_admin_page.js. return "<a href='{$url}' class='confirmrecover' data-question='{$question}' data-text='{$alttext}' title='{$title}'>{$label}</a>"; } public function render_delete_action_link( $id, $picture ) { $url = wp_nonce_url( "admin.php?page=nggallery-manage-gallery&mode=delpic&gid={$picture->galleryid}&pid={$picture->pid}", 'ngg_delpicture' ); $title = esc_attr__( 'Delete image', 'nggallery' ); $label = esc_html__( 'Delete', 'nggallery' ); $question = __( 'Delete', 'nggallery' ); $alttext = empty( $picture->alttext ) ? $picture->filename : $picture->alttext; $alttext = Sanitizer::strip_html( html_entity_decode( $alttext ), true ); $alttext = htmlentities( $alttext, ENT_QUOTES | ENT_HTML401 ); $alttext = \Imagely\NGG\Display\I18N::ngg_plain_text_alt_title_attributes( $alttext ); $alttext = esc_attr( $alttext ); // Event handler is found in nextgen_admin_page.js. return "<a href='{$url}' class='submitdelete delete' data-question='{$question}' data-text='{$alttext}' title='{$title}'>{$label}</a>"; } public function render_image_row_header() { $columns = apply_filters( 'ngg_manage_images_number_of_columns', 6 ); echo '<tr>'; for ( $i = 1; $i <= $columns; $i++ ) { echo "<th class='column column-{$i}'>"; echo apply_filters( 'ngg_manage_images_column_' . $i . '_header', "Column #{$i}" ); echo '</th>'; } echo '</tr>'; } public function setup_gallery_fields() { add_filter( 'ngg_manage_gallery_fields', [ &$this, 'default_gallery_fields' ], 10, 2 ); } public function default_gallery_fields( $fields = [], $gallery = null ) { $fields['left'] = [ 'title' => [ 'callback' => [ &$this, 'render_gallery_title_field' ], 'label' => __( 'Title:', 'nggallery' ), 'tooltip' => null, 'id' => 'gallery_title', ], 'description' => [ 'callback' => [ &$this, 'render_gallery_desc_field' ], 'label' => __( 'Description:', 'nggallery' ), 'tooltip' => null, 'id' => 'gallery_desc', ], 'path' => [ 'callback' => [ &$this, 'render_gallery_path_field' ], 'label' => __( 'Gallery path:', 'nggallery' ), 'tooltip' => null, 'id' => 'gallery_path', ], 'gallery_author' => [ 'callback' => [ &$this, 'render_gallery_author_field' ], 'label' => __( 'Author', 'nggallery' ), 'tooltip' => null, 'id' => 'gallery_author', ], ]; $fields['right'] = [ 'page_link_to' => [ 'callback' => [ &$this, 'render_gallery_link_to_page_field' ], 'label' => __( 'Link to page:', 'nggallery' ), 'tooltip' => __( 'Albums will link this gallery to the selected page', 'nggallery' ), 'id' => 'gallery_page_link_to', ], 'preview_image' => [ 'callback' => [ &$this, 'render_gallery_preview_image_field' ], 'label' => __( 'Preview image:', 'nggallery' ), 'tooltip' => null, 'id' => 'gallery_preview_image', ], 'create_page' => [ 'callback' => [ &$this, 'render_gallery_create_page_field' ], 'label' => __( 'Create new page:', 'nggallery' ), 'tooltip' => null, 'id' => 'gallery_create_new_page', ], ]; return $fields; } public function render_gallery_field_label_column( $text, $for, $tooltip = null ) { $for = esc_attr( $for ); if ( ! empty( $tooltip ) ) { $tooltip = "title='{$tooltip}' class='tooltip'"; } echo "<td><label {$tooltip} for='{$for}'>{$text}</label></td>"; } public function render_gallery_fields() { // Get the gallery entity. $gallery = GalleryMapper::get_instance()->find( $this->gid ); // Get fields. $fields = apply_filters( 'ngg_manage_gallery_fields', [], $gallery ); $left = isset( $fields['left'] ) ? $fields['left'] : []; $right = isset( $fields['right'] ) ? $fields['right'] : []; // Output table. echo '<table id="gallery_fields">'; $number_of_fields = max( count( $left ), count( $right ) ); $left_keys = array_keys( $left ); $right_keys = array_keys( $right ); for ( $i = 0; $i < $number_of_fields; $i++ ) { // Start row. echo '<tr>'; // Left column. if ( isset( $left_keys[ $i ] ) ) { extract( $left[ $left_keys[ $i ] ] ); // Label. $this->render_gallery_field_label_column( $label, $id, $tooltip ); // Input field. if ( is_callable( $callback ) ) { echo '<td>'; call_user_func( $callback, $gallery ); echo '</td>'; } elseif ( WP_DEBUG ) { echo "<p>Could not render {$left_keys[$i]} field. No callback exists</p>"; } } else { $output[] = '<td colspan="2"></td>'; } // Right column. if ( isset( $right_keys[ $i ] ) ) { extract( $right[ $right_keys[ $i ] ] ); // Label. $this->render_gallery_field_label_column( $label, $id, $tooltip ); // Input field.. if ( is_callable( $callback ) ) { echo '<td>'; call_user_func( $callback, $gallery ); echo '</td>'; } elseif ( WP_DEBUG ) { echo "<p>Could not render {$right_keys[$i]} field. No callback exists</p>"; } } else { $output[] = '<td colspan="2"></td>'; } // End. echo '</tr>'; } echo '</table>'; } public function render_gallery_title_field( $gallery ) { include 'templates/manage_gallery/gallery_title_field.php'; } public function render_gallery_desc_field( $gallery ) { include 'templates/manage_gallery/gallery_desc_field.php'; } public function render_gallery_path_field( $gallery ) { include 'templates/manage_gallery/gallery_path_field.php'; } public function render_gallery_author_field( $gallery ) { $user = get_userdata( $gallery->author ); $author = isset( $user->display_name ) ? $user->display_name : $user->user_nicename; include 'templates/manage_gallery/gallery_author_field.php'; } public function render_gallery_link_to_page_field( $gallery ) { $pages = get_pages(); include 'templates/manage_gallery/gallery_link_to_page_field.php'; } public function render_gallery_preview_image_field( $gallery ) { $images = []; foreach ( ImageMapper::get_instance()->find_all( [ 'galleryid = %s', $gallery->{$gallery->id_field} ] ) as $image ) { $images[ $image->{$image->id_field} ] = "[{$image->{$image->id_field}}] {$image->filename}"; } include 'templates/manage_gallery/gallery_preview_image_field.php'; } public function render_gallery_create_page_field( $gallery ) { $pages = get_pages(); include 'templates/manage_gallery/gallery_create_page_field.php'; } public function post_processor_galleries() { global $ngg; check_admin_referer( 'ngg_bulkgallery' ); // bulk update in a single gallery. if ( isset( $_POST['bulkaction'] ) && isset( $_POST['doaction'] ) ) { switch ( $_POST['bulkaction'] ) { case 'no_action'; // No action. break; case 'recover_images': // Recover images from backup. // A prefix 'gallery_' will first fetch all ids from the selected galleries. nggAdmin::do_ajax_operation( 'gallery_recover_image', $_POST['doaction'], __( 'Recover from backup', 'nggallery' ) ); break; case 'set_watermark': // Set watermark // A prefix 'gallery_' will first fetch all ids from the selected galleries. nggAdmin::do_ajax_operation( 'gallery_set_watermark', $_POST['doaction'], __( 'Set watermark', 'nggallery' ) ); break; case 'import_meta': // Import Metadata // A prefix 'gallery_' will first fetch all ids from the selected galleries. nggAdmin::do_ajax_operation( 'gallery_import_metadata', $_POST['doaction'], __( 'Import metadata', 'nggallery' ) ); break; case 'delete_gallery': // Delete gallery. if ( is_array( $_POST['doaction'] ) ) { $deleted = false; $mapper = GalleryMapper::get_instance(); foreach ( $_POST['doaction'] as $id ) { $gallery = $mapper->find( $id ); if ( $gallery->path == '../' || false !== strpos( $gallery->path, '/../' ) ) { nggGallery::show_message( sprintf( __( 'One or more "../" in Gallery paths could be unsafe and NextGen Gallery will not delete gallery %s automatically', 'nggallery' ), $gallery->{$gallery->id_field} ) ); } elseif ( $mapper->destroy( $id, true ) ) { $deleted = true; } } if ( $deleted ) { nggGallery::show_message( esc_html__( 'Gallery deleted successfully ', 'nggallery' ) ); } } break; } } if ( isset( $_POST['addgallery'] ) && isset( $_POST['galleryname'] ) ) { check_admin_referer( 'ngg_bulkgallery' ); if ( ! nggGallery::current_user_can( 'NextGEN Add new gallery' ) ) { wp_die( esc_html_e( 'Cheatin’ uh?', 'nggallery' ) ); } // get the default path for a new gallery. $newgallery = $_POST['galleryname']; if ( ! empty( $newgallery ) ) { $gallery_mapper = GalleryMapper::get_instance(); $gallery = $gallery_mapper->create( [ 'title' => $newgallery ] ); if ( $gallery->save() && ! isset( $_REQUEST['attach_to_post'] ) ) { $url = admin_url() . 'admin.php?page=nggallery-manage-gallery&mode=edit&gid=' . $gallery->gid; $message = sprintf( __( 'Gallery successfully created. <a href="%s" target="_blank">Manage gallery</a>', 'nggallery' ), $url ); nggGallery::show_message( $message, 'gallery_created_msg' ); } } do_action( 'ngg_update_addgallery_page' ); } if ( isset( $_POST['TB_bulkaction'] ) && isset( $_POST['TB_ResizeImages'] ) ) { // save the new values for the next operation. $ngg->options['imgWidth'] = (int) $_POST['imgWidth']; $ngg->options['imgHeight'] = (int) $_POST['imgHeight']; // What is in the case the user has no if cap 'NextGEN Change options' ? Check feedback. update_option( 'ngg_options', $ngg->options ); $gallery_ids = explode( ',', $_POST['TB_imagelist'] ); // A prefix 'gallery_' will first fetch all ids from the selected galleries. nggAdmin::do_ajax_operation( 'gallery_resize_image', $gallery_ids, __( 'Resize images', 'nggallery' ) ); } if ( isset( $_POST['TB_bulkaction'] ) && isset( $_POST['TB_NewThumbnail'] ) ) { // save the new values for the next operation. $settings = Settings::get_instance(); $settings->set( 'thumbwidth', (int) $_POST['thumbwidth'] ); $settings->set( 'thumbheight', (int) $_POST['thumbheight'] ); $settings->set( 'thumbfix', isset( $_POST['thumbfix'] ) ); $settings->save(); ngg_refreshSavedSettings(); // What is in the case the user has no if cap 'NextGEN Change options' ? Check feedback. $gallery_ids = explode( ',', $_POST['TB_imagelist'] ); // A prefix 'gallery_' will first fetch all ids from the selected galleries. nggAdmin::do_ajax_operation( 'gallery_create_thumbnail', $gallery_ids, __( 'Create new thumbnails', 'nggallery' ) ); } } public function post_processor_images() { global $wpdb, $ngg, $nggdb; check_admin_referer( 'ngg_updategallery' ); // bulk update in a single gallery. if ( isset( $_POST['bulkaction'] ) && isset( $_POST['doaction'] ) ) { check_admin_referer( 'ngg_updategallery' ); switch ( $_POST['bulkaction'] ) { case 'no_action'; break; case 'rotate_cw': nggAdmin::do_ajax_operation( 'rotate_cw', $_POST['doaction'], __( 'Rotate images', 'nggallery' ) ); break; case 'rotate_ccw': nggAdmin::do_ajax_operation( 'rotate_ccw', $_POST['doaction'], __( 'Rotate images', 'nggallery' ) ); break; case 'recover_images': nggAdmin::do_ajax_operation( 'recover_image', $_POST['doaction'], __( 'Recover from backup', 'nggallery' ) ); break; case 'set_watermark': nggAdmin::do_ajax_operation( 'set_watermark', $_POST['doaction'], __( 'Set watermark', 'nggallery' ) ); break; case 'strip_orientation_tag': nggAdmin::do_ajax_operation( 'strip_orientation_tag', $_POST['doaction'], __( 'Remove EXIF Orientation', 'nggallery' ) ); break; case 'delete_images': if ( is_array( $_POST['doaction'] ) ) { foreach ( $_POST['doaction'] as $imageID ) { $image = $nggdb->find_image( $imageID ); if ( $image ) { do_action( 'ngg_delete_picture', $image->pid, $image ); if ( $ngg->options['deleteImg'] ) { $storage = StorageManager::get_instance(); $storage->delete_image( $image->pid ); } $delete_pic = ImageMapper::get_instance()->destroy( $image->pid ); } } if ( $delete_pic ) { nggGallery::show_message( __( 'Pictures deleted successfully ', 'nggallery' ) ); } } break; case 'import_meta': nggAdmin::do_ajax_operation( 'import_metadata', $_POST['doaction'], __( 'Import metadata', 'nggallery' ) ); break; } } if ( isset( $_POST['TB_bulkaction'] ) && isset( $_POST['TB_ResizeImages'] ) ) { // save the new values for the next operation. $ngg->options['imgWidth'] = (int) $_POST['imgWidth']; $ngg->options['imgHeight'] = (int) $_POST['imgHeight']; update_option( 'ngg_options', $ngg->options ); $pic_ids = explode( ',', $_POST['TB_imagelist'] ); nggAdmin::do_ajax_operation( 'resize_image', $pic_ids, __( 'Resize images', 'nggallery' ) ); } if ( isset( $_POST['TB_bulkaction'] ) && isset( $_POST['TB_NewThumbnail'] ) ) { // save the new values for the next operation. $settings = Settings::get_instance(); $settings->set( 'thumbwidth', (int) $_POST['thumbwidth'] ); $settings->set( 'thumbheight', (int) $_POST['thumbheight'] ); $settings->set( 'thumbfix', isset( $_POST['thumbfix'] ) ); $settings->save(); ngg_refreshSavedSettings(); $pic_ids = explode( ',', $_POST['TB_imagelist'] ); nggAdmin::do_ajax_operation( 'create_thumbnail', $pic_ids, __( 'Create new thumbnails', 'nggallery' ) ); } if ( isset( $_POST['TB_bulkaction'] ) && isset( $_POST['TB_SelectGallery'] ) ) { $pic_ids = explode( ',', $_POST['TB_imagelist'] ); $dest_gid = (int) $_POST['dest_gid']; switch ( $_POST['TB_bulkaction'] ) { case 'copy_to': $destination = GalleryMapper::get_instance()->find( $dest_gid ); $new_ids = StorageManager::get_instance()->copy_images( $pic_ids, $dest_gid ); if ( ! empty( $new_ids ) ) { $admin_url = admin_url(); $title = esc_html( $destination->title ); $link = "<a href='{$admin_url}admin.php?page=nggallery-manage-gallery&mode=edit&gid={$destination->gid}'>{$title}</a>"; nggGallery::show_message( sprintf( __( 'Copied %1$s picture(s) to gallery: %2$s .', 'nggallery' ), count( $new_ids ), $link ) ); } else { nggGallery::show_error( __( 'Failed to copy images', 'nggallery' ) ); } break; case 'move_to': $destination = GalleryMapper::get_instance()->find( $dest_gid ); $new_ids = StorageManager::get_instance()->move_images( $pic_ids, $dest_gid ); if ( ! empty( $new_ids ) ) { $admin_url = admin_url(); $title = esc_html( $destination->title ); $link = "<a href='{$admin_url}admin.php?page=nggallery-manage-gallery&mode=edit&gid={$destination->gid}'>{$title}</a>"; nggGallery::show_message( sprintf( __( 'Moved %1$s picture(s) to gallery: %2$s .', 'nggallery' ), count( $new_ids ), $link ) ); } else { nggGallery::show_error( __( 'Failed to move images', 'nggallery' ) ); } break; } } if ( isset( $_POST['TB_bulkaction'] ) && isset( $_POST['TB_EditTags'] ) ) { // do tags update. // get the images list. $pic_ids = explode( ',', $_POST['TB_imagelist'] ); $taglist = explode( ',', $_POST['taglist'] ); $taglist = array_map( 'trim', $taglist ); if ( is_array( $pic_ids ) ) { foreach ( $pic_ids as $pic_id ) { // which action should be performed ? switch ( $_POST['TB_bulkaction'] ) { case 'no_action'; // No action. break; case 'overwrite_tags': // Overwrite tags. wp_set_object_terms( $pic_id, $taglist, 'ngg_tag' ); break; case 'add_tags': // Add / append tags. wp_set_object_terms( $pic_id, $taglist, 'ngg_tag', true ); break; case 'delete_tags': // Delete tags. $oldtags = wp_get_object_terms( $pic_id, 'ngg_tag', 'fields=names' ); // get the slugs, to vaoid case sensitive problems. $slugarray = array_map( 'sanitize_title', $taglist ); $oldtags = array_map( 'sanitize_title', $oldtags ); // compare them and return the diff. $newtags = array_diff( $oldtags, $slugarray ); wp_set_object_terms( $pic_id, $newtags, 'ngg_tag' ); break; } } nggGallery::show_message( __( 'Tags changed', 'nggallery' ) ); } } if ( isset( $_POST['updatepictures'] ) && wp_verify_nonce( sanitize_text_field( wp_unslash( $_POST['_wpnonce'] ) ), 'ngg_updategallery' ) ) { // Update pictures. $success = false; if ( nggGallery::current_user_can( 'NextGEN Edit gallery options' ) && ! isset( $_GET['s'] ) ) { $tags = [ '<a>', '<abbr>', '<acronym>', '<address>', '<b>', '<base>', '<basefont>', '<big>', '<blockquote>', '<br>', '<br/>', '<caption>', '<center>', '<cite>', '<code>', '<col>', '<colgroup>', '<dd>', '<del>', '<dfn>', '<dir>', '<div>', '<dl>', '<dt>', '<em>', '<fieldset>', '<font>', '<h1>', '<h2>', '<h3>', '<h4>', '<h5>', '<h6>', '<hr>', '<i>', '<img>', '<ins>', '<label>', '<legend>', '<li>', '<menu>', '<noframes>', '<noscript>', '<ol>', '<optgroup>', '<option>', '<p>', '<pre>', '<q>', '<s>', '<samp>', '<select>', '<small>', '<span>', '<strike>', '<strong>', '<sub>', '<sup>', '<table>', '<tbody>', '<td>', '<tfoot>', '<th>', '<thead>', '<tr>', '<tt>', '<u>', '<ul>' ]; $fields = [ 'title', 'galdesc' ]; // Sanitize fields. foreach ( $fields as $field ) { $html = stripslashes( $_POST[ $field ] ); $html = preg_replace( '/\\s+on\\w+=(["\']).*?\\1/i', '', $html ); $html = preg_replace( '/(<\/[^>]+?>)(<[^>\/][^>]*?>)/', '$1 $2', $html ); $html = strip_tags( $html, implode( '', $tags ) ); $_POST[ $field ] = $html; } $mapper = GalleryMapper::get_instance(); // Update the gallery. if ( ! $this->gallery ) { $this->gallery = $mapper->find( $this->gid, true ); } if ( $this->gallery ) { $excludes = [ '_wpnonce', '_wp_http_referer', 'nggpage' ]; foreach ( $_POST as $key => $value ) { // Yet another IIS hack: gallery paths can be mangled into \\wp-content\\blah\\ which causes // later errors when validating the gallery path: just automatically replace \\ with / here. if ( $key === 'path' ) { $value = str_replace( '\\\\', '/', $value ); } if ( ! in_array( $key, $excludes, true ) ) { $this->gallery->$key = $value; } } $mapper->save( $this->gallery ); if ( ! $this->gallery->is_valid() ) { foreach ( $this->gallery->validation() as $property => $errors ) { foreach ( $errors as $error ) { nggGallery::show_error( $error ); } } } wp_cache_delete( $this->gid, 'ngg_gallery' ); $success = $this->gallery->is_valid(); } } $pictures_updated = $this->update_pictures(); if ( $success || $pictures_updated >= 1 ) { // Hook for other plugin to update the fields. do_action( 'ngg_update_gallery', $this->gid, $_POST ); nggGallery::show_message( __( 'Updated successfully', 'nggallery' ) ); } } // Rescan folder. if ( isset( $_POST['scanfolder'] ) ) { $gallerypath = $wpdb->get_var( $wpdb->prepare( "SELECT `path` FROM `{$wpdb->nggallery}` WHERE `gid` = %d", [ $this->gid, ] ) ); nggAdmin::import_gallery( $gallerypath, $this->gid ); } // Add a new page. if ( isset( $_POST['addnewpage'] ) ) { $parent_id = esc_attr( $_POST['parent_id'] ); $gallery_title = esc_attr( $_POST['title'] ); $mapper = GalleryMapper::get_instance(); $gallery = $mapper->find( $this->gid ); $gallery_name = $gallery->name; // Create a WP page. global $user_ID; $page['post_type'] = 'page'; $page['post_content'] = apply_filters( 'ngg_add_page_shortcode', '[ngg src="galleries" display="basic_thumbnail" ids="' . $this->gid . '"]' ); $page['post_parent'] = $parent_id; $page['post_author'] = $user_ID; $page['post_status'] = 'publish'; $page['post_title'] = $gallery_title == '' ? $gallery_name : $gallery_title; $page = apply_filters( 'ngg_add_new_page', $page, $this->gid ); $gallery_pageid = wp_insert_post( $page ); if ( $gallery_pageid != 0 ) { $gallery->pageid = $gallery_pageid; $mapper->save( $gallery ); nggGallery::show_message( __( 'New gallery page ID', 'nggallery' ) . ' ' . $gallery_pageid . ' -> <strong>' . $gallery_title . '</strong> ' . __( 'created', 'nggallery' ) ); } do_action( 'ngg_gallery_addnewpage', $this->gid ); } } public function can_user_manage_gallery() { $retval = false; if ( $this->gallery && wp_get_current_user()->ID == $this->gallery->author ) { $retval = true; } elseif ( Security::is_allowed( 'nextgen_edit_gallery_unowned' ) ) { $retval = true; } return $retval; } public function update_pictures() { $updated = 0; if ( ! $this->can_user_manage_gallery() ) { return $updated; } if ( isset( $_POST['images'] ) && is_array( $_POST['images'] ) && isset( $_POST['_wpnonce'] ) && wp_verify_nonce( sanitize_text_field( wp_unslash( $_POST['_wpnonce'] ) ), 'ngg_updategallery' ) ) { $image_mapper = ImageMapper::get_instance(); foreach ( $_POST['images'] as $pid => $data ) { if ( ! isset( $data['exclude'] ) ) { $data['exclude'] = 0; } if ( ( $image = $image_mapper->find( $pid ) ) ) { // Strip slashes from title/description/alttext fields. if ( isset( $data['description'] ) ) { $data['description'] = \Imagely\NGG\Display\I18N::ngg_sanitize_text_alt_title_desc( $data['description'] ); } if ( isset( $data['alttext'] ) ) { $data['alttext'] = \Imagely\NGG\Display\I18N::ngg_sanitize_text_alt_title_desc( $data['alttext'] ); } if ( isset( $data['title'] ) ) { $data['title'] = \Imagely\NGG\Display\I18N::ngg_sanitize_text_alt_title_desc( $data['title'] ); } // Generate new slug if the alttext has changed. if ( isset( $data['alttext'] ) && $image->alttext != $data['alttext'] ) { $data['slug'] = null; // will cause a new slug to be generated. } // Update all fields. foreach ( $data as $key => $value ) { $image->$key = $value; } if ( $image_mapper->save( $image ) ) { $updated += 1; // Update the tags for the image. if ( isset( $data['tags'] ) ) { $tags = $data['tags']; if ( ! is_array( $tags ) ) { $tags = explode( ',', $tags ); } foreach ( $tags as &$tag ) { $tag = trim( $tag ); } wp_set_object_terms( $image->{$image->id_field}, $tags, 'ngg_tag' ); } // remove from cache. wp_cache_delete( $image->pid, 'ngg_image' ); // hook for other plugins after image is updated. do_action( 'ngg_image_updated', $image ); } } } // Determine if any WP terms have been orphaned and clean them up. global $wpdb; $results = $wpdb->get_col( "SELECT t.`term_id` FROM `{$wpdb->term_taxonomy}` tt LEFT JOIN `{$wpdb->terms}` t ON tt.`term_id` = t.`term_id` WHERE tt.`taxonomy` = 'ngg_tag' AND tt.`count` <= 0" ); if ( ! empty( $results ) ) { foreach ( $results as $term_id ) { $term_id = apply_filters( 'ngg_pre_delete_unused_term_id', $term_id ); if ( ! empty( $term_id ) ) { wp_delete_term( $term_id, 'ngg_tag' ); } } } } return $updated; } public function search_images() { global $nggdb; if ( empty( $_GET['s'] ) ) { return; } // on what ever reason I need to set again the query var. set_query_var( 's', $_GET['s'] ); $request = get_search_query(); // look now for the images. $search_for_images = (array) $nggdb->search_for_images( $request ); $search_for_tags = (array) nggTags::find_images_for_tags( $request, 'ASC' ); // Merge the two arrays and deduplicate. $merged = array_merge( $search_for_images, $search_for_tags ); $this->search_result = []; foreach ( $merged as $result ) { if ( ! in_array( $result, $this->search_result ) ) { $this->search_result[] = $result; } } // TODO: Currently we didn't support a proper pagination. $nggdb->paged['total_objects'] = $nggdb->paged['objects_per_page'] = count( $this->search_result ); $nggdb->paged['max_objects_per_page'] = 1; // show pictures page. $this->mode = 'edit'; } /** * Display the pagination. * * @since 1.8.0 * @author taken from WP core (see includes/class-wp-list-table.php) * @return string echo the html pagination bar */ public function pagination( $which, $current, $total_items, $per_page ) { $total_pages = ( $per_page > 0 ) ? ceil( $total_items / $per_page ) : 1; $output = '<span class="displaying-num">' . sprintf( _n( '1 item', '%s items', $total_items ), number_format_i18n( $total_items ) ) . '</span>'; $current_url = ( is_ssl() ? 'https://' : 'http://' ) . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI']; $current_url = remove_query_arg( [ 'hotkeys_highlight_last', 'hotkeys_highlight_first' ], $current_url ); $page_links = []; $disable_first = $disable_last = ''; if ( $current == 1 ) { $disable_first = ' disabled'; } if ( $current == $total_pages ) { $disable_last = ' disabled'; } $page_links[] = sprintf( "<a class='%s' title='%s' href='%s'>%s</a>", 'first-page' . $disable_first, esc_attr__( 'Go to the first page' ), Router::esc_url( remove_query_arg( 'paged', $current_url ) ), '«' ); $page_links[] = sprintf( "<a class='%s' title='%s' href='%s'>%s</a>", 'prev-page' . $disable_first, esc_attr__( 'Go to the previous page', 'nggallery' ), Router::esc_url( add_query_arg( 'paged', max( 1, $current - 1 ), $current_url ) ), '‹' ); if ( 'bottom' == $which ) { $html_current_page = $current; } else { $html_current_page = sprintf( "<input class='current-page' title='%s' type='text' name='%s' value='%s' size='%d' />", esc_attr__( 'Current page', 'nggallery' ), esc_attr__( 'post_paged', 'nggallery' ), $current, strlen( $total_pages ) ); } $html_total_pages = sprintf( "<span class='total-pages'>%s</span>", number_format_i18n( $total_pages ) ); $page_links[] = '<span class="paging-input">' . sprintf( _x( '%1$s of %2$s', 'paging' ), $html_current_page, $html_total_pages ) . '</span>'; $page_links[] = sprintf( "<a class='%s' title='%s' href='%s'>%s</a>", 'next-page' . $disable_last, esc_attr__( 'Go to the next page', 'nggallery' ), Router::esc_url( add_query_arg( 'paged', min( $total_pages, $current + 1 ), $current_url ) ), '›' ); $page_links[] = sprintf( "<a class='%s' title='%s' href='%s'>%s</a>", 'last-page' . $disable_last, esc_attr__( 'Go to the last page', 'nggallery' ), Router::esc_url( add_query_arg( 'paged', $total_pages, $current_url ) ), '»' ); $output .= "\n<span class='pagination-links'>" . join( "\n", $page_links ) . '</span>'; if ( $total_pages ) { $page_class = $total_pages < 2 ? ' one-page' : ''; } else { $page_class = ' no-pages'; } $pagination = "<div class='tablenav-pages{$page_class}'>$output</div>"; echo $pagination; return $pagination; } }