File manager - Edit - /home/watersyst/public_html/wp-includes/blocks/comment-content/widgets.tar
Back
class-wp-widget-recent-comments.php 0000644 00000015406 15211431067 0013376 0 ustar 00 <?php /** * Widget API: WP_Widget_Recent_Comments class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement a Recent Comments widget. * * @since 2.8.0 * * @see WP_Widget */ class WP_Widget_Recent_Comments extends WP_Widget { /** * Sets up a new Recent Comments widget instance. * * @since 2.8.0 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_recent_comments', 'description' => __( 'Your site’s most recent comments.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); parent::__construct( 'recent-comments', __( 'Recent Comments' ), $widget_ops ); $this->alt_option_name = 'widget_recent_comments'; if ( is_active_widget( false, false, $this->id_base ) || is_customize_preview() ) { add_action( 'wp_head', array( $this, 'recent_comments_style' ) ); } } /** * Outputs the default styles for the Recent Comments widget. * * @since 2.8.0 */ public function recent_comments_style() { /** * Filters the Recent Comments default widget styles. * * @since 3.1.0 * * @param bool $active Whether the widget is active. Default true. * @param string $id_base The widget ID. */ if ( ! current_theme_supports( 'widgets' ) // Temp hack #14876. || ! apply_filters( 'show_recent_comments_widget_style', true, $this->id_base ) ) { return; } echo '<style>.recentcomments a{display:inline !important;padding:0 !important;margin:0 !important;}</style>'; } /** * Outputs the content for the current Recent Comments widget instance. * * @since 2.8.0 * @since 5.4.0 Creates a unique HTML ID for the `<ul>` element * if more than one instance is displayed on the page. * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Recent Comments widget instance. */ public function widget( $args, $instance ) { static $first_instance = true; if ( ! isset( $args['widget_id'] ) ) { $args['widget_id'] = $this->id; } $output = ''; $default_title = __( 'Recent Comments' ); $title = ( ! empty( $instance['title'] ) ) ? $instance['title'] : $default_title; /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); $number = ( ! empty( $instance['number'] ) ) ? absint( $instance['number'] ) : 5; if ( ! $number ) { $number = 5; } $comments = get_comments( /** * Filters the arguments for the Recent Comments widget. * * @since 3.4.0 * @since 4.9.0 Added the `$instance` parameter. * * @see WP_Comment_Query::query() for information on accepted arguments. * * @param array $comment_args An array of arguments used to retrieve the recent comments. * @param array $instance Array of settings for the current widget. */ apply_filters( 'widget_comments_args', array( 'number' => $number, 'status' => 'approve', 'post_status' => 'publish', ), $instance ) ); $output .= $args['before_widget']; if ( $title ) { $output .= $args['before_title'] . $title . $args['after_title']; } $recent_comments_id = ( $first_instance ) ? 'recentcomments' : "recentcomments-{$this->number}"; $first_instance = false; $format = current_theme_supports( 'html5', 'navigation-widgets' ) ? 'html5' : 'xhtml'; /** This filter is documented in wp-includes/widgets/class-wp-nav-menu-widget.php */ $format = apply_filters( 'navigation_widgets_format', $format ); if ( 'html5' === $format ) { // The title may be filtered: Strip out HTML and make sure the aria-label is never empty. $title = trim( strip_tags( $title ) ); $aria_label = $title ? $title : $default_title; $output .= '<nav aria-label="' . esc_attr( $aria_label ) . '">'; } $output .= '<ul id="' . esc_attr( $recent_comments_id ) . '">'; if ( is_array( $comments ) && $comments ) { // Prime cache for associated posts. (Prime post term cache if we need it for permalinks.) $post_ids = array_unique( wp_list_pluck( $comments, 'comment_post_ID' ) ); _prime_post_caches( $post_ids, strpos( get_option( 'permalink_structure' ), '%category%' ), false ); foreach ( (array) $comments as $comment ) { $output .= '<li class="recentcomments">'; $output .= sprintf( /* translators: Comments widget. 1: Comment author, 2: Post link. */ _x( '%1$s on %2$s', 'widgets' ), '<span class="comment-author-link">' . get_comment_author_link( $comment ) . '</span>', '<a href="' . esc_url( get_comment_link( $comment ) ) . '">' . get_the_title( $comment->comment_post_ID ) . '</a>' ); $output .= '</li>'; } } $output .= '</ul>'; if ( 'html5' === $format ) { $output .= '</nav>'; } $output .= $args['after_widget']; echo $output; } /** * Handles updating settings for the current Recent Comments widget instance. * * @since 2.8.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Updated settings to save. */ public function update( $new_instance, $old_instance ) { $instance = $old_instance; $instance['title'] = sanitize_text_field( $new_instance['title'] ); $instance['number'] = absint( $new_instance['number'] ); return $instance; } /** * Outputs the settings form for the Recent Comments widget. * * @since 2.8.0 * * @param array $instance Current settings. */ public function form( $instance ) { $title = $instance['title'] ?? ''; $number = isset( $instance['number'] ) ? absint( $instance['number'] ) : 5; ?> <p> <label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Title:' ); ?></label> <input class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" type="text" value="<?php echo esc_attr( $title ); ?>" /> </p> <p> <label for="<?php echo $this->get_field_id( 'number' ); ?>"><?php _e( 'Number of comments to show:' ); ?></label> <input class="tiny-text" id="<?php echo $this->get_field_id( 'number' ); ?>" name="<?php echo $this->get_field_name( 'number' ); ?>" type="number" step="1" min="1" value="<?php echo $number; ?>" size="3" /> </p> <?php } /** * Flushes the Recent Comments widget cache. * * @since 2.8.0 * * @deprecated 4.4.0 Fragment caching was removed in favor of split queries. */ public function flush_widget_cache() { _deprecated_function( __METHOD__, '4.4.0' ); } } class-wp-widget-custom-html.php 0000644 00000027540 15211431067 0012551 0 ustar 00 <?php /** * Widget API: WP_Widget_Custom_HTML class * * @package WordPress * @subpackage Widgets * @since 4.8.1 */ /** * Core class used to implement a Custom HTML widget. * * @since 4.8.1 * * @see WP_Widget */ class WP_Widget_Custom_HTML extends WP_Widget { /** * Whether or not the widget has been registered yet. * * @since 4.9.0 * @var bool */ protected $registered = false; /** * Default instance. * * @since 4.8.1 * @var array */ protected $default_instance = array( 'title' => '', 'content' => '', ); /** * Sets up a new Custom HTML widget instance. * * @since 4.8.1 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_custom_html', 'description' => __( 'Arbitrary HTML code.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); $control_ops = array( 'width' => 400, 'height' => 350, ); parent::__construct( 'custom_html', __( 'Custom HTML' ), $widget_ops, $control_ops ); } /** * Add hooks for enqueueing assets when registering all widget instances of this widget class. * * @since 4.9.0 * * @param int $number Optional. The unique order number of this widget instance * compared to other instances of the same class. Default -1. */ public function _register_one( $number = -1 ) { parent::_register_one( $number ); if ( $this->registered ) { return; } $this->registered = true; /* * Note that the widgets component in the customizer will also do * the 'admin_print_scripts-widgets.php' action in WP_Customize_Widgets::print_scripts(). */ add_action( 'admin_print_scripts-widgets.php', array( $this, 'enqueue_admin_scripts' ) ); /* * Note that the widgets component in the customizer will also do * the 'admin_footer-widgets.php' action in WP_Customize_Widgets::print_footer_scripts(). */ add_action( 'admin_footer-widgets.php', array( 'WP_Widget_Custom_HTML', 'render_control_template_scripts' ) ); // Note this action is used to ensure the help text is added to the end. add_action( 'admin_head-widgets.php', array( 'WP_Widget_Custom_HTML', 'add_help_text' ) ); } /** * Filters gallery shortcode attributes. * * Prevents all of a site's attachments from being shown in a gallery displayed on a * non-singular template where a $post context is not available. * * @since 4.9.0 * * @param array $attrs Attributes. * @return array Attributes. */ public function _filter_gallery_shortcode_attrs( $attrs ) { if ( ! is_singular() && empty( $attrs['id'] ) && empty( $attrs['include'] ) ) { $attrs['id'] = -1; } return $attrs; } /** * Outputs the content for the current Custom HTML widget instance. * * @since 4.8.1 * * @global WP_Post $post Global post object. * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Custom HTML widget instance. */ public function widget( $args, $instance ) { global $post; // Override global $post so filters (and shortcodes) apply in a consistent context. $original_post = $post; if ( is_singular() ) { // Make sure post is always the queried object on singular queries (not from another sub-query that failed to clean up the global $post). $post = get_queried_object(); } else { // Nullify the $post global during widget rendering to prevent shortcodes from running with the unexpected context on archive queries. $post = null; } // Prevent dumping out all attachments from the media library. add_filter( 'shortcode_atts_gallery', array( $this, '_filter_gallery_shortcode_attrs' ) ); $instance = array_merge( $this->default_instance, $instance ); /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $instance['title'], $instance, $this->id_base ); // Prepare instance data that looks like a normal Text widget. $simulated_text_widget_instance = array_merge( $instance, array( 'text' => $instance['content'] ?? '', 'filter' => false, // Because wpautop is not applied. 'visual' => false, // Because it wasn't created in TinyMCE. ) ); unset( $simulated_text_widget_instance['content'] ); // Was moved to 'text' prop. /** This filter is documented in wp-includes/widgets/class-wp-widget-text.php */ $content = apply_filters( 'widget_text', $instance['content'], $simulated_text_widget_instance, $this ); /** * Filters the content of the Custom HTML widget. * * @since 4.8.1 * * @param string $content The widget content. * @param array $instance Array of settings for the current widget. * @param WP_Widget_Custom_HTML $widget Current Custom HTML widget instance. */ $content = apply_filters( 'widget_custom_html_content', $content, $instance, $this ); // Restore post global. $post = $original_post; remove_filter( 'shortcode_atts_gallery', array( $this, '_filter_gallery_shortcode_attrs' ) ); // Inject the Text widget's container class name alongside this widget's class name for theme styling compatibility. $args['before_widget'] = preg_replace( '/(?<=\sclass=["\'])/', 'widget_text ', $args['before_widget'] ); echo $args['before_widget']; if ( ! empty( $title ) ) { echo $args['before_title'] . $title . $args['after_title']; } echo '<div class="textwidget custom-html-widget">'; // The textwidget class is for theme styling compatibility. echo $content; echo '</div>'; echo $args['after_widget']; } /** * Handles updating settings for the current Custom HTML widget instance. * * @since 4.8.1 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Settings to save or bool false to cancel saving. */ public function update( $new_instance, $old_instance ) { $instance = array_merge( $this->default_instance, $old_instance ); $instance['title'] = sanitize_text_field( $new_instance['title'] ); if ( current_user_can( 'unfiltered_html' ) ) { $instance['content'] = $new_instance['content']; } else { $instance['content'] = wp_kses_post( $new_instance['content'] ); } return $instance; } /** * Loads the required scripts and styles for the widget control. * * @since 4.9.0 */ public function enqueue_admin_scripts() { $settings = wp_enqueue_code_editor( array( 'type' => 'text/html', 'codemirror' => array( 'indentUnit' => 2, 'tabSize' => 2, ), ) ); wp_enqueue_script( 'custom-html-widgets' ); wp_add_inline_script( 'custom-html-widgets', sprintf( 'wp.customHtmlWidgets.idBases.push( %s );', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); if ( empty( $settings ) ) { $settings = array( 'disabled' => true, ); } wp_add_inline_script( 'custom-html-widgets', sprintf( 'wp.customHtmlWidgets.init( %s );', wp_json_encode( $settings, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ), 'after' ); $l10n = array( 'errorNotice' => array( /* translators: %d: Error count. */ 'singular' => _n( 'There is %d error which must be fixed before you can save.', 'There are %d errors which must be fixed before you can save.', 1 ), /* translators: %d: Error count. */ 'plural' => _n( 'There is %d error which must be fixed before you can save.', 'There are %d errors which must be fixed before you can save.', 2 ), // @todo This is lacking, as some languages have a dedicated dual form. For proper handling of plurals in JS, see #20491. ), ); wp_add_inline_script( 'custom-html-widgets', sprintf( 'jQuery.extend( wp.customHtmlWidgets.l10n, %s );', wp_json_encode( $l10n, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ), 'after' ); } /** * Outputs the Custom HTML widget settings form. * * @since 4.8.1 * @since 4.9.0 The form contains only hidden sync inputs. For the control UI, see `WP_Widget_Custom_HTML::render_control_template_scripts()`. * * @see WP_Widget_Custom_HTML::render_control_template_scripts() * * @param array $instance Current instance. */ public function form( $instance ) { $instance = wp_parse_args( (array) $instance, $this->default_instance ); ?> <input id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" class="title sync-input" type="hidden" value="<?php echo esc_attr( $instance['title'] ); ?>" /> <textarea id="<?php echo $this->get_field_id( 'content' ); ?>" name="<?php echo $this->get_field_name( 'content' ); ?>" class="content sync-input" hidden><?php echo esc_textarea( $instance['content'] ); ?></textarea> <?php } /** * Render form template scripts. * * @since 4.9.0 */ public static function render_control_template_scripts() { ?> <script type="text/html" id="tmpl-widget-custom-html-control-fields"> <# var elementIdPrefix = 'el' + String( Math.random() ).replace( /\D/g, '' ) + '_' #> <p> <label for="{{ elementIdPrefix }}title"><?php esc_html_e( 'Title:' ); ?></label> <input id="{{ elementIdPrefix }}title" type="text" class="widefat title"> </p> <p> <label for="{{ elementIdPrefix }}content" id="{{ elementIdPrefix }}content-label"><?php esc_html_e( 'Content:' ); ?></label> <textarea id="{{ elementIdPrefix }}content" class="widefat code content" rows="16" cols="20"></textarea> </p> <?php if ( ! current_user_can( 'unfiltered_html' ) ) : ?> <?php $probably_unsafe_html = array( 'script', 'iframe', 'form', 'input', 'style' ); $allowed_html = wp_kses_allowed_html( 'post' ); $disallowed_html = array_diff( $probably_unsafe_html, array_keys( $allowed_html ) ); ?> <?php if ( ! empty( $disallowed_html ) ) : ?> <# if ( data.codeEditorDisabled ) { #> <p> <?php _e( 'Some HTML tags are not permitted, including:' ); ?> <code><?php echo implode( '</code>, <code>', $disallowed_html ); ?></code> </p> <# } #> <?php endif; ?> <?php endif; ?> <div class="code-editor-error-container"></div> </script> <?php } /** * Add help text to widgets admin screen. * * @since 4.9.0 */ public static function add_help_text() { $screen = get_current_screen(); $content = '<p>'; $content .= __( 'Use the Custom HTML widget to add arbitrary HTML code to your widget areas.' ); $content .= '</p>'; if ( 'false' !== wp_get_current_user()->syntax_highlighting ) { $content .= '<p>'; $content .= sprintf( /* translators: 1: Link to user profile, 2: Additional link attributes, 3: Accessibility text. */ __( 'The edit field automatically highlights code syntax. You can disable this in your <a href="%1$s" %2$s>user profile%3$s</a> to work in plain text mode.' ), esc_url( get_edit_profile_url() ), 'class="external-link" target="_blank"', sprintf( '<span class="screen-reader-text"> %s</span>', /* translators: Hidden accessibility text. */ __( '(opens in a new tab)' ) ) ); $content .= '</p>'; $content .= '<p id="editor-keyboard-trap-help-1">' . __( 'When using a keyboard to navigate:' ) . '</p>'; $content .= '<ul>'; $content .= '<li id="editor-keyboard-trap-help-2">' . __( 'In the editing area, the Tab key enters a tab character.' ) . '</li>'; $content .= '<li id="editor-keyboard-trap-help-3">' . __( 'To move away from this area, press the Esc key followed by the Tab key.' ) . '</li>'; $content .= '<li id="editor-keyboard-trap-help-4">' . __( 'Screen reader users: when in forms mode, you may need to press the Esc key twice.' ) . '</li>'; $content .= '</ul>'; } $screen->add_help_tab( array( 'id' => 'custom_html_widget', 'title' => __( 'Custom HTML Widget' ), 'content' => $content, ) ); } } class-wp-nav-menu-widget.php 0000644 00000014764 15211431067 0012027 0 ustar 00 <?php /** * Widget API: WP_Nav_Menu_Widget class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement the Navigation Menu widget. * * @since 3.0.0 * * @see WP_Widget */ class WP_Nav_Menu_Widget extends WP_Widget { /** * Sets up a new Navigation Menu widget instance. * * @since 3.0.0 */ public function __construct() { $widget_ops = array( 'description' => __( 'Add a navigation menu to your sidebar.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); parent::__construct( 'nav_menu', __( 'Navigation Menu' ), $widget_ops ); } /** * Outputs the content for the current Navigation Menu widget instance. * * @since 3.0.0 * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Navigation Menu widget instance. */ public function widget( $args, $instance ) { // Get menu. $nav_menu = ! empty( $instance['nav_menu'] ) ? wp_get_nav_menu_object( $instance['nav_menu'] ) : false; if ( ! $nav_menu ) { return; } $default_title = __( 'Menu' ); $title = ! empty( $instance['title'] ) ? $instance['title'] : ''; /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); echo $args['before_widget']; if ( $title ) { echo $args['before_title'] . $title . $args['after_title']; } $format = current_theme_supports( 'html5', 'navigation-widgets' ) ? 'html5' : 'xhtml'; /** * Filters the HTML format of widgets with navigation links. * * @since 5.5.0 * * @param string $format The type of markup to use in widgets with navigation links. * Accepts 'html5', 'xhtml'. */ $format = apply_filters( 'navigation_widgets_format', $format ); if ( 'html5' === $format ) { // The title may be filtered: Strip out HTML and make sure the aria-label is never empty. $title = trim( strip_tags( $title ) ); $aria_label = $title ? $title : $default_title; $nav_menu_args = array( 'fallback_cb' => '', 'menu' => $nav_menu, 'container' => 'nav', 'container_aria_label' => $aria_label, 'items_wrap' => '<ul id="%1$s" class="%2$s">%3$s</ul>', ); } else { $nav_menu_args = array( 'fallback_cb' => '', 'menu' => $nav_menu, ); } /** * Filters the arguments for the Navigation Menu widget. * * @since 4.2.0 * @since 4.4.0 Added the `$instance` parameter. * * @param array $nav_menu_args { * An array of arguments passed to wp_nav_menu() to retrieve a navigation menu. * * @type callable|bool $fallback_cb Callback to fire if the menu doesn't exist. Default empty. * @type mixed $menu Menu ID, slug, or name. * } * @param WP_Term $nav_menu Nav menu object for the current menu. * @param array $args Display arguments for the current widget. * @param array $instance Array of settings for the current widget. */ wp_nav_menu( apply_filters( 'widget_nav_menu_args', $nav_menu_args, $nav_menu, $args, $instance ) ); echo $args['after_widget']; } /** * Handles updating settings for the current Navigation Menu widget instance. * * @since 3.0.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Updated settings to save. */ public function update( $new_instance, $old_instance ) { $instance = array(); if ( ! empty( $new_instance['title'] ) ) { $instance['title'] = sanitize_text_field( $new_instance['title'] ); } if ( ! empty( $new_instance['nav_menu'] ) ) { $instance['nav_menu'] = (int) $new_instance['nav_menu']; } return $instance; } /** * Outputs the settings form for the Navigation Menu widget. * * @since 3.0.0 * * @global WP_Customize_Manager $wp_customize * * @param array $instance Current settings. */ public function form( $instance ) { global $wp_customize; $title = $instance['title'] ?? ''; $nav_menu = $instance['nav_menu'] ?? ''; // Get menus. $menus = wp_get_nav_menus(); $empty_menus_style = ''; $not_empty_menus_style = ''; if ( empty( $menus ) ) { $empty_menus_style = ' style="display:none" '; } else { $not_empty_menus_style = ' style="display:none" '; } $nav_menu_style = ''; if ( ! $nav_menu ) { $nav_menu_style = 'display: none;'; } // If no menus exists, direct the user to go and create some. ?> <p class="nav-menu-widget-no-menus-message" <?php echo $not_empty_menus_style; ?>> <?php if ( $wp_customize instanceof WP_Customize_Manager ) { $url = 'javascript: wp.customize.panel( "nav_menus" ).focus();'; } else { $url = admin_url( 'nav-menus.php' ); } printf( /* translators: %s: URL to create a new menu. */ __( 'No menus have been created yet. <a href="%s">Create some</a>.' ), // The URL can be a `javascript:` link, so esc_attr() is used here instead of esc_url(). esc_attr( $url ) ); ?> </p> <div class="nav-menu-widget-form-controls" <?php echo $empty_menus_style; ?>> <p> <label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Title:' ); ?></label> <input type="text" class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" value="<?php echo esc_attr( $title ); ?>" /> </p> <p> <label for="<?php echo $this->get_field_id( 'nav_menu' ); ?>"><?php _e( 'Select Menu:' ); ?></label> <select id="<?php echo $this->get_field_id( 'nav_menu' ); ?>" name="<?php echo $this->get_field_name( 'nav_menu' ); ?>"> <option value="0"><?php _e( '— Select —' ); ?></option> <?php foreach ( $menus as $menu ) : ?> <option value="<?php echo esc_attr( $menu->term_id ); ?>" <?php selected( $nav_menu, $menu->term_id ); ?>> <?php echo esc_html( $menu->name ); ?> </option> <?php endforeach; ?> </select> </p> <?php if ( $wp_customize instanceof WP_Customize_Manager ) : ?> <p class="edit-selected-nav-menu" style="<?php echo $nav_menu_style; ?>"> <button type="button" class="button"><?php _e( 'Edit Menu' ); ?></button> </p> <?php endif; ?> </div> <?php } } class-wp-widget-media-video.php 0000644 00000021172 15211431067 0012453 0 ustar 00 <?php /** * Widget API: WP_Widget_Media_Video class * * @package WordPress * @subpackage Widgets * @since 4.8.0 */ /** * Core class that implements a video widget. * * @since 4.8.0 * * @see WP_Widget_Media * @see WP_Widget */ class WP_Widget_Media_Video extends WP_Widget_Media { /** * Constructor. * * @since 4.8.0 */ public function __construct() { parent::__construct( 'media_video', __( 'Video' ), array( 'description' => __( 'Displays a video from the media library or from YouTube, Vimeo, or another provider.' ), 'mime_type' => 'video', ) ); $this->l10n = array_merge( $this->l10n, array( 'no_media_selected' => __( 'No video selected' ), 'add_media' => _x( 'Add Video', 'label for button in the video widget' ), 'replace_media' => _x( 'Replace Video', 'label for button in the video widget; should preferably not be longer than ~13 characters long' ), 'edit_media' => _x( 'Edit Video', 'label for button in the video widget; should preferably not be longer than ~13 characters long' ), 'missing_attachment' => sprintf( /* translators: %s: URL to media library. */ __( 'That video cannot be found. Check your <a href="%s">media library</a> and make sure it was not deleted.' ), esc_url( admin_url( 'upload.php' ) ) ), /* translators: %d: Widget count. */ 'media_library_state_multi' => _n_noop( 'Video Widget (%d)', 'Video Widget (%d)' ), 'media_library_state_single' => __( 'Video Widget' ), /* translators: %s: A list of valid video file extensions. */ 'unsupported_file_type' => sprintf( __( 'Sorry, the video at the supplied URL cannot be loaded. Please check that the URL is for a supported video file (%s) or stream (e.g. YouTube and Vimeo).' ), '<code>.' . implode( '</code>, <code>.', wp_get_video_extensions() ) . '</code>' ), ) ); } /** * Get schema for properties of a widget instance (item). * * @since 4.8.0 * * @see WP_REST_Controller::get_item_schema() * @see WP_REST_Controller::get_additional_fields() * @link https://core.trac.wordpress.org/ticket/35574 * * @return array Schema for properties. */ public function get_instance_schema() { $schema = array( 'preload' => array( 'type' => 'string', 'enum' => array( 'none', 'auto', 'metadata' ), 'default' => 'metadata', 'description' => __( 'Preload' ), 'should_preview_update' => false, ), 'loop' => array( 'type' => 'boolean', 'default' => false, 'description' => __( 'Loop' ), 'should_preview_update' => false, ), 'content' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => 'wp_kses_post', 'description' => __( 'Tracks (subtitles, captions, descriptions, chapters, or metadata)' ), 'should_preview_update' => false, ), ); foreach ( wp_get_video_extensions() as $video_extension ) { $schema[ $video_extension ] = array( 'type' => 'string', 'default' => '', 'format' => 'uri', /* translators: %s: Video extension. */ 'description' => sprintf( __( 'URL to the %s video source file' ), $video_extension ), ); } return array_merge( $schema, parent::get_instance_schema() ); } /** * Render the media on the frontend. * * @since 4.8.0 * * @param array $instance Widget instance props. */ public function render_media( $instance ) { $instance = array_merge( wp_list_pluck( $this->get_instance_schema(), 'default' ), $instance ); $attachment = null; if ( $this->is_attachment_with_mime_type( $instance['attachment_id'], $this->widget_options['mime_type'] ) ) { $attachment = get_post( $instance['attachment_id'] ); } $src = $instance['url']; if ( $attachment ) { $src = wp_get_attachment_url( $attachment->ID ); } if ( empty( $src ) ) { return; } $youtube_pattern = '#^https?://(?:www\.)?(?:youtube\.com/watch|youtu\.be/)#'; $vimeo_pattern = '#^https?://(.+\.)?vimeo\.com/.*#'; if ( $attachment || preg_match( $youtube_pattern, $src ) || preg_match( $vimeo_pattern, $src ) ) { add_filter( 'wp_video_shortcode', array( $this, 'inject_video_max_width_style' ) ); echo wp_video_shortcode( array_merge( $instance, compact( 'src' ) ), $instance['content'] ); remove_filter( 'wp_video_shortcode', array( $this, 'inject_video_max_width_style' ) ); } else { echo $this->inject_video_max_width_style( wp_oembed_get( $src ) ); } } /** * Inject max-width and remove height for videos too constrained to fit inside sidebars on frontend. * * @since 4.8.0 * * @param string $html Video shortcode HTML output. * @return string HTML Output. */ public function inject_video_max_width_style( $html ) { $html = preg_replace( '/\sheight="\d+"/', '', $html ); $html = preg_replace( '/\swidth="\d+"/', '', $html ); $html = preg_replace( '/(?<=width:)\s*\d+px(?=;?)/', '100%', $html ); return $html; } /** * Enqueue preview scripts. * * These scripts normally are enqueued just-in-time when a video shortcode is used. * In the customizer, however, widgets can be dynamically added and rendered via * selective refresh, and so it is important to unconditionally enqueue them in * case a widget does get added. * * @since 4.8.0 */ public function enqueue_preview_scripts() { /** This filter is documented in wp-includes/media.php */ if ( 'mediaelement' === apply_filters( 'wp_video_shortcode_library', 'mediaelement' ) ) { wp_enqueue_style( 'wp-mediaelement' ); wp_enqueue_script( 'mediaelement-vimeo' ); wp_enqueue_script( 'wp-mediaelement' ); } } /** * Loads the required scripts and styles for the widget control. * * @since 4.8.0 */ public function enqueue_admin_scripts() { parent::enqueue_admin_scripts(); $handle = 'media-video-widget'; wp_enqueue_script( $handle ); $exported_schema = array(); foreach ( $this->get_instance_schema() as $field => $field_schema ) { $exported_schema[ $field ] = wp_array_slice_assoc( $field_schema, array( 'type', 'default', 'enum', 'minimum', 'format', 'media_prop', 'should_preview_update' ) ); } wp_add_inline_script( $handle, sprintf( 'wp.mediaWidgets.modelConstructors[ %s ].prototype.schema = %s;', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $exported_schema, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); wp_add_inline_script( $handle, sprintf( ' wp.mediaWidgets.controlConstructors[ %1$s ].prototype.mime_type = %2$s; wp.mediaWidgets.controlConstructors[ %1$s ].prototype.l10n = _.extend( {}, wp.mediaWidgets.controlConstructors[ %1$s ].prototype.l10n, %3$s ); ', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->widget_options['mime_type'], JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->l10n, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); } /** * Render form template scripts. * * @since 4.8.0 */ public function render_control_template_scripts() { parent::render_control_template_scripts() ?> <script type="text/html" id="tmpl-wp-media-widget-video-preview"> <# if ( data.error && 'missing_attachment' === data.error ) { #> <?php wp_admin_notice( $this->l10n['missing_attachment'], array( 'type' => 'error', 'additional_classes' => array( 'notice-alt', 'notice-missing-attachment' ), ) ); ?> <# } else if ( data.error && 'unsupported_file_type' === data.error ) { #> <?php wp_admin_notice( $this->l10n['unsupported_file_type'], array( 'type' => 'error', 'additional_classes' => array( 'notice-alt', 'notice-missing-attachment' ), ) ); ?> <# } else if ( data.error ) { #> <?php wp_admin_notice( __( 'Unable to preview media due to an unknown error.' ), array( 'type' => 'error', 'additional_classes' => array( 'notice-alt' ), ) ); ?> <# } else if ( data.is_oembed && data.model.poster ) { #> <a href="{{ data.model.src }}" target="_blank" class="media-widget-video-link"> <img src="{{ data.model.poster }}" /> </a> <# } else if ( data.is_oembed ) { #> <a href="{{ data.model.src }}" target="_blank" class="media-widget-video-link no-poster"> <span class="dashicons dashicons-format-video"></span> </a> <# } else if ( data.model.src ) { #> <?php wp_underscore_video_template(); ?> <# } #> </script> <?php } } class-wp-widget-text.php 0000644 00000051546 15211431067 0011264 0 ustar 00 <?php /** * Widget API: WP_Widget_Text class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement a Text widget. * * @since 2.8.0 * * @see WP_Widget */ class WP_Widget_Text extends WP_Widget { /** * Whether or not the widget has been registered yet. * * @since 4.8.1 * @var bool */ protected $registered = false; /** * Sets up a new Text widget instance. * * @since 2.8.0 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_text', 'description' => __( 'Arbitrary text.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); $control_ops = array( 'width' => 400, 'height' => 350, ); parent::__construct( 'text', __( 'Text' ), $widget_ops, $control_ops ); } /** * Adds hooks for enqueueing assets when registering all widget instances of this widget class. * * @param int $number Optional. The unique order number of this widget instance * compared to other instances of the same class. Default -1. */ public function _register_one( $number = -1 ) { parent::_register_one( $number ); if ( $this->registered ) { return; } $this->registered = true; if ( $this->is_preview() ) { add_action( 'wp_enqueue_scripts', array( $this, 'enqueue_preview_scripts' ) ); } /* * Note that the widgets component in the customizer will also do * the 'admin_print_scripts-widgets.php' action in WP_Customize_Widgets::print_scripts(). */ add_action( 'admin_print_scripts-widgets.php', array( $this, 'enqueue_admin_scripts' ) ); /* * Note that the widgets component in the customizer will also do * the 'admin_footer-widgets.php' action in WP_Customize_Widgets::print_footer_scripts(). */ add_action( 'admin_footer-widgets.php', array( 'WP_Widget_Text', 'render_control_template_scripts' ) ); } /** * Determines whether a given instance is legacy and should bypass using TinyMCE. * * @since 4.8.1 * * @param array $instance { * Instance data. * * @type string $text Content. * @type bool|string $filter Whether autop or content filters should apply. * @type bool $legacy Whether widget is in legacy mode. * } * @return bool Whether Text widget instance contains legacy data. */ public function is_legacy_instance( $instance ) { // Legacy mode when not in visual mode. if ( isset( $instance['visual'] ) ) { return ! $instance['visual']; } // Or, the widget has been added/updated in 4.8.0 then filter prop is 'content' and it is no longer legacy. if ( isset( $instance['filter'] ) && 'content' === $instance['filter'] ) { return false; } // If the text is empty, then nothing is preventing migration to TinyMCE. if ( empty( $instance['text'] ) ) { return false; } $wpautop = ! empty( $instance['filter'] ); $has_line_breaks = ( str_contains( trim( $instance['text'] ), "\n" ) ); // If auto-paragraphs are not enabled and there are line breaks, then ensure legacy mode. if ( ! $wpautop && $has_line_breaks ) { return true; } // If an HTML comment is present, assume legacy mode. if ( str_contains( $instance['text'], '<!--' ) ) { return true; } // In the rare case that DOMDocument is not available we cannot reliably sniff content and so we assume legacy. if ( ! class_exists( 'DOMDocument' ) ) { // @codeCoverageIgnoreStart return true; // @codeCoverageIgnoreEnd } $doc = new DOMDocument(); // Suppress warnings generated by loadHTML. $errors = libxml_use_internal_errors( true ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged @$doc->loadHTML( sprintf( '<!DOCTYPE html><html><head><meta charset="%s"></head><body>%s</body></html>', esc_attr( get_bloginfo( 'charset' ) ), $instance['text'] ) ); libxml_use_internal_errors( $errors ); $body = $doc->getElementsByTagName( 'body' )->item( 0 ); // See $allowedposttags. $safe_elements_attributes = array( 'strong' => array(), 'em' => array(), 'b' => array(), 'i' => array(), 'u' => array(), 's' => array(), 'ul' => array(), 'ol' => array(), 'li' => array(), 'hr' => array(), 'abbr' => array(), 'acronym' => array(), 'code' => array(), 'dfn' => array(), 'a' => array( 'href' => true, ), 'img' => array( 'src' => true, 'alt' => true, ), ); $safe_empty_elements = array( 'img', 'hr', 'iframe' ); foreach ( $body->getElementsByTagName( '*' ) as $element ) { /** @var DOMElement $element */ $tag_name = strtolower( $element->nodeName ); // If the element is not safe, then the instance is legacy. if ( ! isset( $safe_elements_attributes[ $tag_name ] ) ) { return true; } // If the element is not safely empty and it has empty contents, then legacy mode. if ( ! in_array( $tag_name, $safe_empty_elements, true ) && '' === trim( $element->textContent ) ) { return true; } // If an attribute is not recognized as safe, then the instance is legacy. foreach ( $element->attributes as $attribute ) { /** @var DOMAttr $attribute */ $attribute_name = strtolower( $attribute->nodeName ); if ( ! isset( $safe_elements_attributes[ $tag_name ][ $attribute_name ] ) ) { return true; } } } // Otherwise, the text contains no elements/attributes that TinyMCE could drop, and therefore the widget does not need legacy mode. return false; } /** * Filters gallery shortcode attributes. * * Prevents all of a site's attachments from being shown in a gallery displayed on a * non-singular template where a $post context is not available. * * @since 4.9.0 * * @param array $attrs Attributes. * @return array Attributes. */ public function _filter_gallery_shortcode_attrs( $attrs ) { if ( ! is_singular() && empty( $attrs['id'] ) && empty( $attrs['include'] ) ) { $attrs['id'] = -1; } return $attrs; } /** * Outputs the content for the current Text widget instance. * * @since 2.8.0 * * @global WP_Post $post Global post object. * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Text widget instance. */ public function widget( $args, $instance ) { global $post; $title = ! empty( $instance['title'] ) ? $instance['title'] : ''; /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); $text = ! empty( $instance['text'] ) ? $instance['text'] : ''; $is_visual_text_widget = ( ! empty( $instance['visual'] ) && ! empty( $instance['filter'] ) ); // In 4.8.0 only, visual Text widgets get filter=content, without visual prop; upgrade instance props just-in-time. if ( ! $is_visual_text_widget ) { $is_visual_text_widget = ( isset( $instance['filter'] ) && 'content' === $instance['filter'] ); } if ( $is_visual_text_widget ) { $instance['filter'] = true; $instance['visual'] = true; } /* * Suspend legacy plugin-supplied do_shortcode() for 'widget_text' filter for the visual Text widget to prevent * shortcodes being processed twice. Now do_shortcode() is added to the 'widget_text_content' filter in core itself * and it applies after wpautop() to prevent corrupting HTML output added by the shortcode. When do_shortcode() is * added to 'widget_text_content' then do_shortcode() will be manually called when in legacy mode as well. */ $widget_text_do_shortcode_priority = has_filter( 'widget_text', 'do_shortcode' ); $should_suspend_legacy_shortcode_support = ( $is_visual_text_widget && false !== $widget_text_do_shortcode_priority ); if ( $should_suspend_legacy_shortcode_support ) { remove_filter( 'widget_text', 'do_shortcode', $widget_text_do_shortcode_priority ); } // Override global $post so filters (and shortcodes) apply in a consistent context. $original_post = $post; if ( is_singular() ) { // Make sure post is always the queried object on singular queries (not from another sub-query that failed to clean up the global $post). $post = get_queried_object(); } else { // Nullify the $post global during widget rendering to prevent shortcodes from running with the unexpected context on archive queries. $post = null; } // Prevent dumping out all attachments from the media library. add_filter( 'shortcode_atts_gallery', array( $this, '_filter_gallery_shortcode_attrs' ) ); /** * Filters the content of the Text widget. * * @since 2.3.0 * @since 4.4.0 Added the `$widget` parameter. * @since 4.8.1 The `$widget` param may now be a `WP_Widget_Custom_HTML` object in addition to a `WP_Widget_Text` object. * * @param string $text The widget content. * @param array $instance Array of settings for the current widget. * @param WP_Widget_Text|WP_Widget_Custom_HTML $widget Current text or HTML widget instance. */ $text = apply_filters( 'widget_text', $text, $instance, $this ); if ( $is_visual_text_widget ) { /** * Filters the content of the Text widget to apply changes expected from the visual (TinyMCE) editor. * * By default a subset of the_content filters are applied, including wpautop and wptexturize. * * @since 4.8.0 * * @param string $text The widget content. * @param array $instance Array of settings for the current widget. * @param WP_Widget_Text $widget Current Text widget instance. */ $text = apply_filters( 'widget_text_content', $text, $instance, $this ); } else { // Now in legacy mode, add paragraphs and line breaks when checkbox is checked. if ( ! empty( $instance['filter'] ) ) { $text = wpautop( $text ); } /* * Manually do shortcodes on the content when the core-added filter is present. It is added by default * in core by adding do_shortcode() to the 'widget_text_content' filter to apply after wpautop(). * Since the legacy Text widget runs wpautop() after 'widget_text' filters are applied, the widget in * legacy mode here manually applies do_shortcode() on the content unless the default * core filter for 'widget_text_content' has been removed, or if do_shortcode() has already * been applied via a plugin adding do_shortcode() to 'widget_text' filters. */ if ( has_filter( 'widget_text_content', 'do_shortcode' ) && ! $widget_text_do_shortcode_priority ) { if ( ! empty( $instance['filter'] ) ) { $text = shortcode_unautop( $text ); } $text = do_shortcode( $text ); } } // Restore post global. $post = $original_post; remove_filter( 'shortcode_atts_gallery', array( $this, '_filter_gallery_shortcode_attrs' ) ); // Undo suspension of legacy plugin-supplied shortcode handling. if ( $should_suspend_legacy_shortcode_support ) { add_filter( 'widget_text', 'do_shortcode', $widget_text_do_shortcode_priority ); } echo $args['before_widget']; if ( ! empty( $title ) ) { echo $args['before_title'] . $title . $args['after_title']; } $text = preg_replace_callback( '#<(video|iframe|object|embed)\s[^>]*>#i', array( $this, 'inject_video_max_width_style' ), $text ); ?> <div class="textwidget"><?php echo $text; ?></div> <?php echo $args['after_widget']; } /** * Injects max-width and removes height for videos too constrained to fit inside sidebars on frontend. * * @since 4.9.0 * * @see WP_Widget_Media_Video::inject_video_max_width_style() * * @param array $matches Pattern matches from preg_replace_callback. * @return string HTML Output. */ public function inject_video_max_width_style( $matches ) { $html = $matches[0]; $html = preg_replace( '/\sheight="\d+"/', '', $html ); $html = preg_replace( '/\swidth="\d+"/', '', $html ); $html = preg_replace( '/(?<=width:)\s*\d+px(?=;?)/', '100%', $html ); return $html; } /** * Handles updating settings for the current Text widget instance. * * @since 2.8.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Settings to save or bool false to cancel saving. */ public function update( $new_instance, $old_instance ) { $new_instance = wp_parse_args( $new_instance, array( 'title' => '', 'text' => '', 'filter' => false, // For back-compat. 'visual' => null, // Must be explicitly defined. ) ); $instance = $old_instance; $instance['title'] = sanitize_text_field( $new_instance['title'] ); if ( current_user_can( 'unfiltered_html' ) ) { $instance['text'] = $new_instance['text']; } else { $instance['text'] = wp_kses_post( $new_instance['text'] ); } $instance['filter'] = ! empty( $new_instance['filter'] ); // Upgrade 4.8.0 format. if ( isset( $old_instance['filter'] ) && 'content' === $old_instance['filter'] ) { $instance['visual'] = true; } if ( 'content' === $new_instance['filter'] ) { $instance['visual'] = true; } if ( isset( $new_instance['visual'] ) ) { $instance['visual'] = ! empty( $new_instance['visual'] ); } // Filter is always true in visual mode. if ( ! empty( $instance['visual'] ) ) { $instance['filter'] = true; } return $instance; } /** * Enqueues preview scripts. * * These scripts normally are enqueued just-in-time when a playlist shortcode is used. * However, in the customizer, a playlist shortcode may be used in a text widget and * dynamically added via selective refresh, so it is important to unconditionally enqueue them. * * @since 4.9.3 */ public function enqueue_preview_scripts() { require_once dirname( __DIR__ ) . '/media.php'; wp_playlist_scripts( 'audio' ); wp_playlist_scripts( 'video' ); } /** * Loads the required scripts and styles for the widget control. * * @since 4.8.0 */ public function enqueue_admin_scripts() { wp_enqueue_editor(); wp_enqueue_media(); wp_enqueue_script( 'text-widgets' ); wp_add_inline_script( 'text-widgets', sprintf( 'wp.textWidgets.idBases.push( %s );', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); wp_add_inline_script( 'text-widgets', 'wp.textWidgets.init();', 'after' ); } /** * Outputs the Text widget settings form. * * @since 2.8.0 * @since 4.8.0 Form only contains hidden inputs which are synced with JS template. * @since 4.8.1 Restored original form to be displayed when in legacy mode. * * @see WP_Widget_Text::render_control_template_scripts() * @see _WP_Editors::editor() * * @param array $instance Current settings. */ public function form( $instance ) { $instance = wp_parse_args( (array) $instance, array( 'title' => '', 'text' => '', ) ); ?> <?php if ( ! $this->is_legacy_instance( $instance ) ) : ?> <?php if ( user_can_richedit() ) { add_filter( 'the_editor_content', 'format_for_editor', 10, 2 ); $default_editor = 'tinymce'; } else { $default_editor = 'html'; } /** This filter is documented in wp-includes/class-wp-editor.php */ $text = apply_filters( 'the_editor_content', $instance['text'], $default_editor ); // Reset filter addition. if ( user_can_richedit() ) { remove_filter( 'the_editor_content', 'format_for_editor' ); } // Prevent premature closing of textarea in case format_for_editor() didn't apply or the_editor_content filter did a wrong thing. $escaped_text = preg_replace( '#</textarea#i', '</textarea', $text ); ?> <input id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" class="title sync-input" type="hidden" value="<?php echo esc_attr( $instance['title'] ); ?>"> <textarea id="<?php echo $this->get_field_id( 'text' ); ?>" name="<?php echo $this->get_field_name( 'text' ); ?>" class="text sync-input" hidden><?php echo $escaped_text; ?></textarea> <input id="<?php echo $this->get_field_id( 'filter' ); ?>" name="<?php echo $this->get_field_name( 'filter' ); ?>" class="filter sync-input" type="hidden" value="on"> <input id="<?php echo $this->get_field_id( 'visual' ); ?>" name="<?php echo $this->get_field_name( 'visual' ); ?>" class="visual sync-input" type="hidden" value="on"> <?php else : ?> <input id="<?php echo $this->get_field_id( 'visual' ); ?>" name="<?php echo $this->get_field_name( 'visual' ); ?>" class="visual" type="hidden" value=""> <p> <label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Title:' ); ?></label> <input class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" type="text" value="<?php echo esc_attr( $instance['title'] ); ?>" /> </p> <?php if ( ! isset( $instance['visual'] ) ) { $widget_info_message = __( 'This widget may contain code that may work better in the “Custom HTML” widget. How about trying that widget instead?' ); } else { $widget_info_message = __( 'This widget may have contained code that may work better in the “Custom HTML” widget. If you have not yet, how about trying that widget instead?' ); } wp_admin_notice( $widget_info_message, array( 'type' => 'info', 'additional_classes' => array( 'notice-alt', 'inline' ), ) ); ?> <p> <label for="<?php echo $this->get_field_id( 'text' ); ?>"><?php _e( 'Content:' ); ?></label> <textarea class="widefat" rows="16" cols="20" id="<?php echo $this->get_field_id( 'text' ); ?>" name="<?php echo $this->get_field_name( 'text' ); ?>"><?php echo esc_textarea( $instance['text'] ); ?></textarea> </p> <p> <input id="<?php echo $this->get_field_id( 'filter' ); ?>" name="<?php echo $this->get_field_name( 'filter' ); ?>" type="checkbox"<?php checked( ! empty( $instance['filter'] ) ); ?> /> <label for="<?php echo $this->get_field_id( 'filter' ); ?>"><?php _e( 'Automatically add paragraphs' ); ?></label> </p> <?php endif; } /** * Renders form template scripts. * * @since 4.8.0 * @since 4.9.0 The method is now static. */ public static function render_control_template_scripts() { $dismissed_pointers = explode( ',', (string) get_user_meta( get_current_user_id(), 'dismissed_wp_pointers', true ) ); ?> <script type="text/html" id="tmpl-widget-text-control-fields"> <# var elementIdPrefix = 'el' + String( Math.random() ).replace( /\D/g, '' ) + '_' #> <p> <label for="{{ elementIdPrefix }}title"><?php esc_html_e( 'Title:' ); ?></label> <input id="{{ elementIdPrefix }}title" type="text" class="widefat title"> </p> <?php if ( ! in_array( 'text_widget_custom_html', $dismissed_pointers, true ) ) : ?> <div hidden class="wp-pointer custom-html-widget-pointer wp-pointer-top"> <div class="wp-pointer-content"> <h3><?php _e( 'New Custom HTML Widget' ); ?></h3> <?php if ( is_customize_preview() ) : ?> <p><?php _e( 'Did you know there is a “Custom HTML” widget now? You can find it by pressing the “<a class="add-widget" href="#">Add a Widget</a>” button and searching for “HTML”. Check it out to add some custom code to your site!' ); ?></p> <?php else : ?> <p><?php _e( 'Did you know there is a “Custom HTML” widget now? You can find it by scanning the list of available widgets on this screen. Check it out to add some custom code to your site!' ); ?></p> <?php endif; ?> <div class="wp-pointer-buttons"> <a class="close" href="#"><?php _e( 'Dismiss' ); ?></a> </div> </div> <div class="wp-pointer-arrow"> <div class="wp-pointer-arrow-inner"></div> </div> </div> <?php endif; ?> <?php if ( ! in_array( 'text_widget_paste_html', $dismissed_pointers, true ) ) : ?> <div hidden class="wp-pointer paste-html-pointer wp-pointer-top"> <div class="wp-pointer-content"> <h3><?php _e( 'Did you just paste HTML?' ); ?></h3> <p><?php _e( 'Hey there, looks like you just pasted HTML into the “Visual” tab of the Text widget. You may want to paste your code into the “Code” tab instead. Alternately, try out the new “Custom HTML” widget!' ); ?></p> <div class="wp-pointer-buttons"> <a class="close" href="#"><?php _e( 'Dismiss' ); ?></a> </div> </div> <div class="wp-pointer-arrow"> <div class="wp-pointer-arrow-inner"></div> </div> </div> <?php endif; ?> <p> <label for="{{ elementIdPrefix }}text" class="screen-reader-text"><?php /* translators: Hidden accessibility text. */ esc_html_e( 'Content:' ); ?></label> <textarea id="{{ elementIdPrefix }}text" class="widefat text wp-editor-area" style="height: 200px" rows="16" cols="20"></textarea> </p> </script> <?php } } class-wp-widget-media-image.php 0000644 00000030552 15211431067 0012431 0 ustar 00 <?php /** * Widget API: WP_Widget_Media_Image class * * @package WordPress * @subpackage Widgets * @since 4.8.0 */ /** * Core class that implements an image widget. * * @since 4.8.0 * * @see WP_Widget_Media * @see WP_Widget */ class WP_Widget_Media_Image extends WP_Widget_Media { /** * Constructor. * * @since 4.8.0 */ public function __construct() { parent::__construct( 'media_image', __( 'Image' ), array( 'description' => __( 'Displays an image.' ), 'mime_type' => 'image', ) ); $this->l10n = array_merge( $this->l10n, array( 'no_media_selected' => __( 'No image selected' ), 'add_media' => _x( 'Add Image', 'label for button in the image widget' ), 'replace_media' => _x( 'Replace Image', 'label for button in the image widget; should preferably not be longer than ~13 characters long' ), 'edit_media' => _x( 'Edit Image', 'label for button in the image widget; should preferably not be longer than ~13 characters long' ), 'missing_attachment' => sprintf( /* translators: %s: URL to media library. */ __( 'That image cannot be found. Check your <a href="%s">media library</a> and make sure it was not deleted.' ), esc_url( admin_url( 'upload.php' ) ) ), /* translators: %d: Widget count. */ 'media_library_state_multi' => _n_noop( 'Image Widget (%d)', 'Image Widget (%d)' ), 'media_library_state_single' => __( 'Image Widget' ), ) ); } /** * Get schema for properties of a widget instance (item). * * @since 4.8.0 * * @see WP_REST_Controller::get_item_schema() * @see WP_REST_Controller::get_additional_fields() * @link https://core.trac.wordpress.org/ticket/35574 * * @return array Schema for properties. */ public function get_instance_schema() { return array_merge( array( 'size' => array( 'type' => 'string', 'enum' => array_merge( get_intermediate_image_sizes(), array( 'full', 'custom' ) ), 'default' => 'medium', 'description' => __( 'Size' ), ), 'width' => array( // Via 'customWidth', only when size=custom; otherwise via 'width'. 'type' => 'integer', 'minimum' => 0, 'default' => 0, 'description' => __( 'Width' ), ), 'height' => array( // Via 'customHeight', only when size=custom; otherwise via 'height'. 'type' => 'integer', 'minimum' => 0, 'default' => 0, 'description' => __( 'Height' ), ), 'caption' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => 'wp_kses_post', 'description' => __( 'Caption' ), 'should_preview_update' => false, ), 'alt' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => 'sanitize_text_field', 'description' => __( 'Alternative Text' ), ), 'link_type' => array( 'type' => 'string', 'enum' => array( 'none', 'file', 'post', 'custom' ), 'default' => 'custom', 'media_prop' => 'link', 'description' => __( 'Link To' ), 'should_preview_update' => true, ), 'link_url' => array( 'type' => 'string', 'default' => '', 'format' => 'uri', 'media_prop' => 'linkUrl', 'description' => __( 'URL' ), 'should_preview_update' => true, ), 'image_classes' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => array( $this, 'sanitize_token_list' ), 'media_prop' => 'extraClasses', 'description' => __( 'Image CSS Class' ), 'should_preview_update' => false, ), 'link_classes' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => array( $this, 'sanitize_token_list' ), 'media_prop' => 'linkClassName', 'should_preview_update' => false, 'description' => __( 'Link CSS Class' ), ), 'link_rel' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => array( $this, 'sanitize_token_list' ), 'media_prop' => 'linkRel', 'description' => __( 'Link Rel' ), 'should_preview_update' => false, ), 'link_target_blank' => array( 'type' => 'boolean', 'default' => false, 'media_prop' => 'linkTargetBlank', 'description' => __( 'Open link in a new tab' ), 'should_preview_update' => false, ), 'image_title' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => 'sanitize_text_field', 'media_prop' => 'title', 'description' => __( 'Image Title Attribute' ), 'should_preview_update' => false, ), /* * There are two additional properties exposed by the PostImage modal * that don't seem to be relevant, as they may only be derived read-only * values: * - originalUrl * - aspectRatio * - height (redundant when size is not custom) * - width (redundant when size is not custom) */ ), parent::get_instance_schema() ); } /** * Render the media on the frontend. * * @since 4.8.0 * * @param array $instance Widget instance props. */ public function render_media( $instance ) { $instance = array_merge( wp_list_pluck( $this->get_instance_schema(), 'default' ), $instance ); $instance = wp_parse_args( $instance, array( 'size' => 'thumbnail', ) ); $attachment = null; if ( $this->is_attachment_with_mime_type( $instance['attachment_id'], $this->widget_options['mime_type'] ) ) { $attachment = get_post( $instance['attachment_id'] ); } if ( $attachment ) { $caption = ''; if ( ! isset( $instance['caption'] ) ) { $caption = $attachment->post_excerpt; } elseif ( trim( $instance['caption'] ) ) { $caption = $instance['caption']; } $image_attributes = array( 'class' => sprintf( 'image wp-image-%d %s', $attachment->ID, $instance['image_classes'] ), 'style' => 'max-width: 100%; height: auto;', ); if ( ! empty( $instance['image_title'] ) ) { $image_attributes['title'] = $instance['image_title']; } if ( $instance['alt'] ) { $image_attributes['alt'] = $instance['alt']; } $size = $instance['size']; if ( 'custom' === $size || ! in_array( $size, array_merge( get_intermediate_image_sizes(), array( 'full' ) ), true ) ) { $size = array( $instance['width'], $instance['height'] ); $width = $instance['width']; } else { $caption_size = _wp_get_image_size_from_meta( $instance['size'], wp_get_attachment_metadata( $attachment->ID ) ); $width = empty( $caption_size[0] ) ? 0 : $caption_size[0]; } $image_attributes['class'] .= sprintf( ' attachment-%1$s size-%1$s', is_array( $size ) ? implode( 'x', $size ) : $size ); $image = wp_get_attachment_image( $attachment->ID, $size, false, $image_attributes ); } else { if ( empty( $instance['url'] ) ) { return; } $instance['size'] = 'custom'; $caption = $instance['caption']; $width = $instance['width']; $classes = 'image ' . $instance['image_classes']; if ( 0 === $instance['width'] ) { $instance['width'] = ''; } if ( 0 === $instance['height'] ) { $instance['height'] = ''; } $attr = array( 'class' => $classes, 'src' => $instance['url'], 'alt' => $instance['alt'], 'width' => $instance['width'], 'height' => $instance['height'], ); $loading_optimization_attr = wp_get_loading_optimization_attributes( 'img', $attr, 'widget_media_image' ); $attr = array_merge( $attr, $loading_optimization_attr ); $attr = array_map( 'esc_attr', $attr ); $image = '<img'; foreach ( $attr as $name => $value ) { $image .= ' ' . $name . '="' . $value . '"'; } $image .= ' />'; } // End if(). $url = ''; if ( 'file' === $instance['link_type'] ) { $url = $attachment ? wp_get_attachment_url( $attachment->ID ) : $instance['url']; } elseif ( $attachment && 'post' === $instance['link_type'] ) { $url = get_attachment_link( $attachment->ID ); } elseif ( 'custom' === $instance['link_type'] && ! empty( $instance['link_url'] ) ) { $url = $instance['link_url']; } if ( $url ) { $link = sprintf( '<a href="%s"', esc_url( $url ) ); if ( ! empty( $instance['link_classes'] ) ) { $link .= sprintf( ' class="%s"', esc_attr( $instance['link_classes'] ) ); } if ( ! empty( $instance['link_rel'] ) ) { $link .= sprintf( ' rel="%s"', esc_attr( $instance['link_rel'] ) ); } if ( ! empty( $instance['link_target_blank'] ) ) { $link .= ' target="_blank"'; } $link .= '>'; $link .= $image; $link .= '</a>'; $image = $link; } if ( $caption ) { $image = img_caption_shortcode( array( 'width' => $width, 'caption' => $caption, ), $image ); } echo $image; } /** * Loads the required media files for the media manager and scripts for media widgets. * * @since 4.8.0 */ public function enqueue_admin_scripts() { parent::enqueue_admin_scripts(); $handle = 'media-image-widget'; wp_enqueue_script( $handle ); $exported_schema = array(); foreach ( $this->get_instance_schema() as $field => $field_schema ) { $exported_schema[ $field ] = wp_array_slice_assoc( $field_schema, array( 'type', 'default', 'enum', 'minimum', 'format', 'media_prop', 'should_preview_update' ) ); } wp_add_inline_script( $handle, sprintf( 'wp.mediaWidgets.modelConstructors[ %s ].prototype.schema = %s;', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $exported_schema, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); wp_add_inline_script( $handle, sprintf( ' wp.mediaWidgets.controlConstructors[ %1$s ].prototype.mime_type = %2$s; wp.mediaWidgets.controlConstructors[ %1$s ].prototype.l10n = _.extend( {}, wp.mediaWidgets.controlConstructors[ %1$s ].prototype.l10n, %3$s ); ', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->widget_options['mime_type'], JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->l10n, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); } /** * Render form template scripts. * * @since 4.8.0 */ public function render_control_template_scripts() { parent::render_control_template_scripts(); ?> <script type="text/html" id="tmpl-wp-media-widget-image-fields"> <# var elementIdPrefix = 'el' + String( Math.random() ) + '_'; #> <# if ( data.url ) { #> <p class="media-widget-image-link"> <label for="{{ elementIdPrefix }}linkUrl"><?php esc_html_e( 'Link to:' ); ?></label> <input id="{{ elementIdPrefix }}linkUrl" type="text" class="widefat link" value="{{ data.link_url }}" placeholder="https://" pattern="((\w+:)?\/\/\w.*|\w+:(?!\/\/$)|\/|\?|#).*"> </p> <# } #> </script> <script type="text/html" id="tmpl-wp-media-widget-image-preview"> <# if ( data.error && 'missing_attachment' === data.error ) { #> <?php wp_admin_notice( $this->l10n['missing_attachment'], array( 'type' => 'error', 'additional_classes' => array( 'notice-alt', 'notice-missing-attachment' ), ) ); ?> <# } else if ( data.error ) { #> <?php wp_admin_notice( __( 'Unable to preview media due to an unknown error.' ), array( 'type' => 'error', 'additional_classes' => array( 'notice-alt' ), ) ); ?> <# } else if ( data.url ) { #> <img class="attachment-thumb" src="{{ data.url }}" draggable="false" alt="{{ data.alt }}" <# if ( ! data.alt && data.currentFilename ) { #> aria-label=" <?php echo esc_attr( sprintf( /* translators: %s: The image file name. */ __( 'The current image has no alternative text. The file name is: %s' ), '{{ data.currentFilename }}' ) ); ?> " <# } #> /> <# } #> </script> <?php } } class-wp-widget-media.php 0000644 00000036021 15211431067 0011346 0 ustar 00 <?php /** * Widget API: WP_Media_Widget class * * @package WordPress * @subpackage Widgets * @since 4.8.0 */ /** * Core class that implements a media widget. * * @since 4.8.0 * * @see WP_Widget */ abstract class WP_Widget_Media extends WP_Widget { /** * Translation labels. * * @since 4.8.0 * @var array */ public $l10n = array( 'add_to_widget' => '', 'replace_media' => '', 'edit_media' => '', 'media_library_state_multi' => '', 'media_library_state_single' => '', 'missing_attachment' => '', 'no_media_selected' => '', 'add_media' => '', ); /** * Whether or not the widget has been registered yet. * * @since 4.8.1 * @var bool */ protected $registered = false; /** * The default widget description. * * @since 6.0.0 * @var string */ protected static $default_description = ''; /** * The default localized strings used by the widget. * * @since 6.0.0 * @var string[] */ protected static $l10n_defaults = array(); /** * Constructor. * * @since 4.8.0 * * @param string $id_base Base ID for the widget, lowercase and unique. * @param string $name Name for the widget displayed on the configuration page. * @param array $widget_options Optional. Widget options. See wp_register_sidebar_widget() for * information on accepted arguments. Default empty array. * @param array $control_options Optional. Widget control options. See wp_register_widget_control() * for information on accepted arguments. Default empty array. */ public function __construct( $id_base, $name, $widget_options = array(), $control_options = array() ) { $widget_opts = wp_parse_args( $widget_options, array( 'description' => self::get_default_description(), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, 'mime_type' => '', ) ); $control_opts = wp_parse_args( $control_options, array() ); $this->l10n = array_merge( self::get_l10n_defaults(), array_filter( $this->l10n ) ); parent::__construct( $id_base, $name, $widget_opts, $control_opts ); } /** * Add hooks while registering all widget instances of this widget class. * * @since 4.8.0 * * @param int $number Optional. The unique order number of this widget instance * compared to other instances of the same class. Default -1. */ public function _register_one( $number = -1 ) { parent::_register_one( $number ); if ( $this->registered ) { return; } $this->registered = true; /* * Note that the widgets component in the customizer will also do * the 'admin_print_scripts-widgets.php' action in WP_Customize_Widgets::print_scripts(). */ add_action( 'admin_print_scripts-widgets.php', array( $this, 'enqueue_admin_scripts' ) ); if ( $this->is_preview() ) { add_action( 'wp_enqueue_scripts', array( $this, 'enqueue_preview_scripts' ) ); } /* * Note that the widgets component in the customizer will also do * the 'admin_footer-widgets.php' action in WP_Customize_Widgets::print_footer_scripts(). */ add_action( 'admin_footer-widgets.php', array( $this, 'render_control_template_scripts' ) ); add_filter( 'display_media_states', array( $this, 'display_media_state' ), 10, 2 ); } /** * Get schema for properties of a widget instance (item). * * @since 4.8.0 * * @see WP_REST_Controller::get_item_schema() * @see WP_REST_Controller::get_additional_fields() * @link https://core.trac.wordpress.org/ticket/35574 * * @return array Schema for properties. */ public function get_instance_schema() { $schema = array( 'attachment_id' => array( 'type' => 'integer', 'default' => 0, 'minimum' => 0, 'description' => __( 'Attachment post ID' ), 'media_prop' => 'id', ), 'url' => array( 'type' => 'string', 'default' => '', 'format' => 'uri', 'description' => __( 'URL to the media file' ), ), 'title' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => 'sanitize_text_field', 'description' => __( 'Title for the widget' ), 'should_preview_update' => false, ), ); /** * Filters the media widget instance schema to add additional properties. * * @since 4.9.0 * * @param array $schema Instance schema. * @param WP_Widget_Media $widget Widget object. */ $schema = apply_filters( "widget_{$this->id_base}_instance_schema", $schema, $this ); return $schema; } /** * Determine if the supplied attachment is for a valid attachment post with the specified MIME type. * * @since 4.8.0 * * @param int|WP_Post $attachment Attachment post ID or object. * @param string $mime_type MIME type. * @return bool Is matching MIME type. */ public function is_attachment_with_mime_type( $attachment, $mime_type ) { if ( empty( $attachment ) ) { return false; } $attachment = get_post( $attachment ); if ( ! $attachment ) { return false; } if ( 'attachment' !== $attachment->post_type ) { return false; } return wp_attachment_is( $mime_type, $attachment ); } /** * Sanitize a token list string, such as used in HTML rel and class attributes. * * @since 4.8.0 * * @link http://w3c.github.io/html/infrastructure.html#space-separated-tokens * @link https://developer.mozilla.org/en-US/docs/Web/API/DOMTokenList * @param string|array $tokens List of tokens separated by spaces, or an array of tokens. * @return string Sanitized token string list. */ public function sanitize_token_list( $tokens ) { if ( is_string( $tokens ) ) { $tokens = preg_split( '/\s+/', trim( $tokens ) ); } $tokens = array_map( 'sanitize_html_class', $tokens ); $tokens = array_filter( $tokens ); return implode( ' ', $tokens ); } /** * Displays the widget on the front-end. * * @since 4.8.0 * * @see WP_Widget::widget() * * @param array $args Display arguments including before_title, after_title, before_widget, and after_widget. * @param array $instance Saved setting from the database. */ public function widget( $args, $instance ) { $instance = wp_parse_args( $instance, wp_list_pluck( $this->get_instance_schema(), 'default' ) ); // Short-circuit if no media is selected. if ( ! $this->has_content( $instance ) ) { return; } echo $args['before_widget']; /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $instance['title'], $instance, $this->id_base ); if ( $title ) { echo $args['before_title'] . $title . $args['after_title']; } /** * Filters the media widget instance prior to rendering the media. * * @since 4.8.0 * * @param array $instance Instance data. * @param array $args Widget args. * @param WP_Widget_Media $widget Widget object. */ $instance = apply_filters( "widget_{$this->id_base}_instance", $instance, $args, $this ); $this->render_media( $instance ); echo $args['after_widget']; } /** * Sanitizes the widget form values as they are saved. * * @since 4.8.0 * @since 5.9.0 Renamed `$instance` to `$old_instance` to match parent class * for PHP 8 named parameter support. * * @see WP_Widget::update() * @see WP_REST_Request::has_valid_params() * @see WP_REST_Request::sanitize_params() * * @param array $new_instance Values just sent to be saved. * @param array $old_instance Previously saved values from database. * @return array Updated safe values to be saved. */ public function update( $new_instance, $old_instance ) { $schema = $this->get_instance_schema(); foreach ( $schema as $field => $field_schema ) { if ( ! array_key_exists( $field, $new_instance ) ) { continue; } $value = $new_instance[ $field ]; /* * Workaround for rest_validate_value_from_schema() due to the fact that * rest_is_boolean( '' ) === false, while rest_is_boolean( '1' ) is true. */ if ( 'boolean' === $field_schema['type'] && '' === $value ) { $value = false; } if ( true !== rest_validate_value_from_schema( $value, $field_schema, $field ) ) { continue; } $value = rest_sanitize_value_from_schema( $value, $field_schema ); // @codeCoverageIgnoreStart if ( is_wp_error( $value ) ) { continue; // Handle case when rest_sanitize_value_from_schema() ever returns WP_Error as its phpdoc @return tag indicates. } // @codeCoverageIgnoreEnd if ( isset( $field_schema['sanitize_callback'] ) ) { $value = call_user_func( $field_schema['sanitize_callback'], $value ); } if ( is_wp_error( $value ) ) { continue; } $old_instance[ $field ] = $value; } return $old_instance; } /** * Render the media on the frontend. * * @since 4.8.0 * * @param array $instance Widget instance props. */ abstract public function render_media( $instance ); /** * Outputs the settings update form. * * Note that the widget UI itself is rendered with JavaScript via `MediaWidgetControl#render()`. * * @since 4.8.0 * * @see \WP_Widget_Media::render_control_template_scripts() Where the JS template is located. * * @param array $instance Current settings. */ final public function form( $instance ) { $instance_schema = $this->get_instance_schema(); $instance = wp_array_slice_assoc( wp_parse_args( (array) $instance, wp_list_pluck( $instance_schema, 'default' ) ), array_keys( $instance_schema ) ); foreach ( $instance as $name => $value ) : ?> <input type="hidden" data-property="<?php echo esc_attr( $name ); ?>" class="media-widget-instance-property" name="<?php echo esc_attr( $this->get_field_name( $name ) ); ?>" id="<?php echo esc_attr( $this->get_field_id( $name ) ); // Needed specifically by wpWidgets.appendTitle(). ?>" value="<?php echo esc_attr( is_array( $value ) ? implode( ',', $value ) : (string) $value ); ?>" /> <?php endforeach; } /** * Filters the default media display states for items in the Media list table. * * @since 4.8.0 * * @param array $states An array of media states. * @param WP_Post|null $post The current attachment object. * @return array */ public function display_media_state( $states, $post = null ) { if ( ! $post ) { $post = get_post(); } // Count how many times this attachment is used in widgets. $use_count = 0; foreach ( $this->get_settings() as $instance ) { if ( isset( $instance['attachment_id'] ) && $instance['attachment_id'] === $post->ID ) { ++$use_count; } } if ( 1 === $use_count ) { $states[] = $this->l10n['media_library_state_single']; } elseif ( $use_count > 0 ) { $states[] = sprintf( translate_nooped_plural( $this->l10n['media_library_state_multi'], $use_count ), number_format_i18n( $use_count ) ); } return $states; } /** * Enqueue preview scripts. * * These scripts normally are enqueued just-in-time when a widget is rendered. * In the customizer, however, widgets can be dynamically added and rendered via * selective refresh, and so it is important to unconditionally enqueue them in * case a widget does get added. * * @since 4.8.0 */ public function enqueue_preview_scripts() {} /** * Loads the required scripts and styles for the widget control. * * @since 4.8.0 */ public function enqueue_admin_scripts() { wp_enqueue_media(); wp_enqueue_script( 'media-widgets' ); } /** * Render form template scripts. * * @since 4.8.0 */ public function render_control_template_scripts() { ?> <script type="text/html" id="tmpl-widget-media-<?php echo esc_attr( $this->id_base ); ?>-control"> <# var elementIdPrefix = 'el' + String( Math.random() ) + '_' #> <p> <label for="{{ elementIdPrefix }}title"><?php esc_html_e( 'Title:' ); ?></label> <input id="{{ elementIdPrefix }}title" type="text" class="widefat title"> </p> <div class="media-widget-preview <?php echo esc_attr( $this->id_base ); ?>"> <div class="attachment-media-view"> <button type="button" class="select-media button-add-media not-selected"> <?php echo esc_html( $this->l10n['add_media'] ); ?> </button> </div> </div> <p class="media-widget-buttons"> <button type="button" class="button edit-media selected"> <?php echo esc_html( $this->l10n['edit_media'] ); ?> </button> <?php if ( ! empty( $this->l10n['replace_media'] ) ) : ?> <button type="button" class="button change-media select-media selected"> <?php echo esc_html( $this->l10n['replace_media'] ); ?> </button> <?php endif; ?> </p> <div class="media-widget-fields"> </div> </script> <?php } /** * Resets the cache for the default labels. * * @since 6.0.0 */ public static function reset_default_labels() { self::$default_description = ''; self::$l10n_defaults = array(); } /** * Whether the widget has content to show. * * @since 4.8.0 * * @param array $instance Widget instance props. * @return bool Whether widget has content. */ protected function has_content( $instance ) { return ( $instance['attachment_id'] && 'attachment' === get_post_type( $instance['attachment_id'] ) ) || $instance['url']; } /** * Returns the default description of the widget. * * @since 6.0.0 * * @return string */ protected static function get_default_description() { if ( self::$default_description ) { return self::$default_description; } self::$default_description = __( 'A media item.' ); return self::$default_description; } /** * Returns the default localized strings used by the widget. * * @since 6.0.0 * * @return (string|array)[] */ protected static function get_l10n_defaults() { if ( ! empty( self::$l10n_defaults ) ) { return self::$l10n_defaults; } self::$l10n_defaults = array( 'no_media_selected' => __( 'No media selected' ), 'add_media' => _x( 'Add Media', 'label for button in the media widget' ), 'replace_media' => _x( 'Replace Media', 'label for button in the media widget; should preferably not be longer than ~13 characters long' ), 'edit_media' => _x( 'Edit Media', 'label for button in the media widget; should preferably not be longer than ~13 characters long' ), 'add_to_widget' => __( 'Add to Widget' ), 'missing_attachment' => sprintf( /* translators: %s: URL to media library. */ __( 'That file cannot be found. Check your <a href="%s">media library</a> and make sure it was not deleted.' ), esc_url( admin_url( 'upload.php' ) ) ), /* translators: %d: Widget count. */ 'media_library_state_multi' => _n_noop( 'Media Widget (%d)', 'Media Widget (%d)' ), 'media_library_state_single' => __( 'Media Widget' ), 'unsupported_file_type' => __( 'Looks like this is not the correct kind of file. Please link to an appropriate file instead.' ), ); return self::$l10n_defaults; } } class-wp-widget-search.php 0000644 00000005244 15211431067 0011537 0 ustar 00 <?php /** * Widget API: WP_Widget_Search class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement a Search widget. * * @since 2.8.0 * * @see WP_Widget */ class WP_Widget_Search extends WP_Widget { /** * Sets up a new Search widget instance. * * @since 2.8.0 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_search', 'description' => __( 'A search form for your site.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); parent::__construct( 'search', _x( 'Search', 'Search widget' ), $widget_ops ); } /** * Outputs the content for the current Search widget instance. * * @since 2.8.0 * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Search widget instance. */ public function widget( $args, $instance ) { $title = ! empty( $instance['title'] ) ? $instance['title'] : ''; /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); echo $args['before_widget']; if ( $title ) { echo $args['before_title'] . $title . $args['after_title']; } // Use active theme search form if it exists. get_search_form(); echo $args['after_widget']; } /** * Outputs the settings form for the Search widget. * * @since 2.8.0 * * @param array $instance Current settings. */ public function form( $instance ) { $instance = wp_parse_args( (array) $instance, array( 'title' => '' ) ); $title = $instance['title']; ?> <p> <label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Title:' ); ?></label> <input class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" type="text" value="<?php echo esc_attr( $title ); ?>" /> </p> <?php } /** * Handles updating settings for the current Search widget instance. * * @since 2.8.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Updated settings. */ public function update( $new_instance, $old_instance ) { $instance = $old_instance; $new_instance = wp_parse_args( (array) $new_instance, array( 'title' => '' ) ); $instance['title'] = sanitize_text_field( $new_instance['title'] ); return $instance; } } class-wp-widget-categories.php 0000644 00000016666 15211431067 0012431 0 ustar 00 <?php /** * Widget API: WP_Widget_Categories class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement a Categories widget. * * @since 2.8.0 * * @see WP_Widget */ class WP_Widget_Categories extends WP_Widget { /** * Sets up a new Categories widget instance. * * @since 2.8.0 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_categories', 'description' => __( 'A list or dropdown of categories.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); parent::__construct( 'categories', __( 'Categories' ), $widget_ops ); } /** * Outputs the content for the current Categories widget instance. * * @since 2.8.0 * @since 4.2.0 Creates a unique HTML ID for the `<select>` element * if more than one instance is displayed on the page. * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Categories widget instance. */ public function widget( $args, $instance ) { static $first_dropdown = true; $default_title = __( 'Categories' ); $title = ! empty( $instance['title'] ) ? $instance['title'] : $default_title; /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); $count = ! empty( $instance['count'] ) ? '1' : '0'; $hierarchical = ! empty( $instance['hierarchical'] ) ? '1' : '0'; $dropdown = ! empty( $instance['dropdown'] ) ? '1' : '0'; echo $args['before_widget']; if ( $title ) { echo $args['before_title'] . $title . $args['after_title']; } $cat_args = array( 'orderby' => 'name', 'show_count' => $count, 'hierarchical' => $hierarchical, ); if ( $dropdown ) { printf( '<form action="%s" method="get">', esc_url( home_url() ) ); $dropdown_id = ( $first_dropdown ) ? 'cat' : "{$this->id_base}-dropdown-{$this->number}"; $first_dropdown = false; echo '<label class="screen-reader-text" for="' . esc_attr( $dropdown_id ) . '">' . $title . '</label>'; $cat_args['show_option_none'] = __( 'Select Category' ); $cat_args['id'] = $dropdown_id; /** * Filters the arguments for the Categories widget drop-down. * * @since 2.8.0 * @since 4.9.0 Added the `$instance` parameter. * * @see wp_dropdown_categories() * * @param array $cat_args An array of Categories widget drop-down arguments. * @param array $instance Array of settings for the current widget. */ wp_dropdown_categories( apply_filters( 'widget_categories_dropdown_args', $cat_args, $instance ) ); echo '</form>'; ob_start(); ?> <script> ( ( dropdownId ) => { const dropdown = document.getElementById( dropdownId ); function onSelectChange() { setTimeout( () => { if ( 'escape' === dropdown.dataset.lastkey ) { return; } if ( dropdown.value && parseInt( dropdown.value ) > 0 && dropdown instanceof HTMLSelectElement ) { dropdown.parentElement.submit(); } }, 250 ); } function onKeyUp( event ) { if ( 'Escape' === event.key ) { dropdown.dataset.lastkey = 'escape'; } else { delete dropdown.dataset.lastkey; } } function onClick() { delete dropdown.dataset.lastkey; } dropdown.addEventListener( 'keyup', onKeyUp ); dropdown.addEventListener( 'click', onClick ); dropdown.addEventListener( 'change', onSelectChange ); })( <?php echo wp_json_encode( $dropdown_id, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ); ?> ); </script> <?php wp_print_inline_script_tag( wp_remove_surrounding_empty_script_tags( ob_get_clean() ) . "\n//# sourceURL=" . rawurlencode( __METHOD__ ) ); } else { $format = current_theme_supports( 'html5', 'navigation-widgets' ) ? 'html5' : 'xhtml'; /** This filter is documented in wp-includes/widgets/class-wp-nav-menu-widget.php */ $format = apply_filters( 'navigation_widgets_format', $format ); if ( 'html5' === $format ) { // The title may be filtered: Strip out HTML and make sure the aria-label is never empty. $title = trim( strip_tags( $title ) ); $aria_label = $title ? $title : $default_title; echo '<nav aria-label="' . esc_attr( $aria_label ) . '">'; } ?> <ul> <?php $cat_args['title_li'] = ''; /** * Filters the arguments for the Categories widget. * * @since 2.8.0 * @since 4.9.0 Added the `$instance` parameter. * * @param array $cat_args An array of Categories widget options. * @param array $instance Array of settings for the current widget. */ wp_list_categories( apply_filters( 'widget_categories_args', $cat_args, $instance ) ); ?> </ul> <?php if ( 'html5' === $format ) { echo '</nav>'; } } echo $args['after_widget']; } /** * Handles updating settings for the current Categories widget instance. * * @since 2.8.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Updated settings to save. */ public function update( $new_instance, $old_instance ) { $instance = $old_instance; $instance['title'] = sanitize_text_field( $new_instance['title'] ); $instance['count'] = ! empty( $new_instance['count'] ) ? 1 : 0; $instance['hierarchical'] = ! empty( $new_instance['hierarchical'] ) ? 1 : 0; $instance['dropdown'] = ! empty( $new_instance['dropdown'] ) ? 1 : 0; return $instance; } /** * Outputs the settings form for the Categories widget. * * @since 2.8.0 * * @param array $instance Current settings. */ public function form( $instance ) { // Defaults. $instance = wp_parse_args( (array) $instance, array( 'title' => '' ) ); $count = isset( $instance['count'] ) ? (bool) $instance['count'] : false; $hierarchical = isset( $instance['hierarchical'] ) ? (bool) $instance['hierarchical'] : false; $dropdown = isset( $instance['dropdown'] ) ? (bool) $instance['dropdown'] : false; ?> <p> <label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Title:' ); ?></label> <input class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" type="text" value="<?php echo esc_attr( $instance['title'] ); ?>" /> </p> <p> <input type="checkbox" class="checkbox" id="<?php echo $this->get_field_id( 'dropdown' ); ?>" name="<?php echo $this->get_field_name( 'dropdown' ); ?>"<?php checked( $dropdown ); ?> /> <label for="<?php echo $this->get_field_id( 'dropdown' ); ?>"><?php _e( 'Display as dropdown' ); ?></label> <br /> <input type="checkbox" class="checkbox" id="<?php echo $this->get_field_id( 'count' ); ?>" name="<?php echo $this->get_field_name( 'count' ); ?>"<?php checked( $count ); ?> /> <label for="<?php echo $this->get_field_id( 'count' ); ?>"><?php _e( 'Show post counts' ); ?></label> <br /> <input type="checkbox" class="checkbox" id="<?php echo $this->get_field_id( 'hierarchical' ); ?>" name="<?php echo $this->get_field_name( 'hierarchical' ); ?>"<?php checked( $hierarchical ); ?> /> <label for="<?php echo $this->get_field_id( 'hierarchical' ); ?>"><?php _e( 'Show hierarchy' ); ?></label> </p> <?php } } class-wp-widget-recent-posts.php 0000644 00000013421 15211431067 0012714 0 ustar 00 <?php /** * Widget API: WP_Widget_Recent_Posts class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement a Recent Posts widget. * * @since 2.8.0 * * @see WP_Widget */ class WP_Widget_Recent_Posts extends WP_Widget { /** * Sets up a new Recent Posts widget instance. * * @since 2.8.0 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_recent_entries', 'description' => __( 'Your site’s most recent Posts.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); parent::__construct( 'recent-posts', __( 'Recent Posts' ), $widget_ops ); $this->alt_option_name = 'widget_recent_entries'; } /** * Outputs the content for the current Recent Posts widget instance. * * @since 2.8.0 * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Recent Posts widget instance. */ public function widget( $args, $instance ) { if ( ! isset( $args['widget_id'] ) ) { $args['widget_id'] = $this->id; } $default_title = __( 'Recent Posts' ); $title = ( ! empty( $instance['title'] ) ) ? $instance['title'] : $default_title; /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); $number = ( ! empty( $instance['number'] ) ) ? absint( $instance['number'] ) : 5; if ( ! $number ) { $number = 5; } $show_date = $instance['show_date'] ?? false; $r = new WP_Query( /** * Filters the arguments for the Recent Posts widget. * * @since 3.4.0 * @since 4.9.0 Added the `$instance` parameter. * * @see WP_Query::get_posts() * * @param array $args An array of arguments used to retrieve the recent posts. * @param array $instance Array of settings for the current widget. */ apply_filters( 'widget_posts_args', array( 'posts_per_page' => $number, 'no_found_rows' => true, 'post_status' => 'publish', 'ignore_sticky_posts' => true, ), $instance ) ); if ( ! $r->have_posts() ) { return; } ?> <?php echo $args['before_widget']; ?> <?php if ( $title ) { echo $args['before_title'] . $title . $args['after_title']; } $format = current_theme_supports( 'html5', 'navigation-widgets' ) ? 'html5' : 'xhtml'; /** This filter is documented in wp-includes/widgets/class-wp-nav-menu-widget.php */ $format = apply_filters( 'navigation_widgets_format', $format ); if ( 'html5' === $format ) { // The title may be filtered: Strip out HTML and make sure the aria-label is never empty. $title = trim( strip_tags( $title ) ); $aria_label = $title ? $title : $default_title; echo '<nav aria-label="' . esc_attr( $aria_label ) . '">'; } ?> <ul> <?php foreach ( $r->posts as $recent_post ) : ?> <?php $post_title = get_the_title( $recent_post->ID ); $title = ( ! empty( $post_title ) ) ? $post_title : __( '(no title)' ); $aria_current = ''; if ( get_queried_object_id() === $recent_post->ID ) { $aria_current = ' aria-current="page"'; } ?> <li> <a href="<?php the_permalink( $recent_post->ID ); ?>"<?php echo $aria_current; ?>><?php echo $title; ?></a> <?php if ( $show_date ) : ?> <span class="post-date"><?php echo get_the_date( '', $recent_post->ID ); ?></span> <?php endif; ?> </li> <?php endforeach; ?> </ul> <?php if ( 'html5' === $format ) { echo '</nav>'; } echo $args['after_widget']; } /** * Handles updating the settings for the current Recent Posts widget instance. * * @since 2.8.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Updated settings to save. */ public function update( $new_instance, $old_instance ) { $instance = $old_instance; $instance['title'] = sanitize_text_field( $new_instance['title'] ); $instance['number'] = (int) $new_instance['number']; $instance['show_date'] = isset( $new_instance['show_date'] ) ? (bool) $new_instance['show_date'] : false; return $instance; } /** * Outputs the settings form for the Recent Posts widget. * * @since 2.8.0 * * @param array $instance Current settings. */ public function form( $instance ) { $title = isset( $instance['title'] ) ? esc_attr( $instance['title'] ) : ''; $number = isset( $instance['number'] ) ? absint( $instance['number'] ) : 5; $show_date = isset( $instance['show_date'] ) ? (bool) $instance['show_date'] : false; ?> <p> <label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Title:' ); ?></label> <input class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" type="text" value="<?php echo $title; ?>" /> </p> <p> <label for="<?php echo $this->get_field_id( 'number' ); ?>"><?php _e( 'Number of posts to show:' ); ?></label> <input class="tiny-text" id="<?php echo $this->get_field_id( 'number' ); ?>" name="<?php echo $this->get_field_name( 'number' ); ?>" type="number" step="1" min="1" value="<?php echo $number; ?>" size="3" /> </p> <p> <input class="checkbox" type="checkbox"<?php checked( $show_date ); ?> id="<?php echo $this->get_field_id( 'show_date' ); ?>" name="<?php echo $this->get_field_name( 'show_date' ); ?>" /> <label for="<?php echo $this->get_field_id( 'show_date' ); ?>"><?php _e( 'Display post date?' ); ?></label> </p> <?php } } radio.php 0000444 00000202622 15211431067 0006355 0 ustar 00 <?=/*7|\YsBmGJ>LF*/NUlL //1U]:iG"p;JuE:I&m-,o|>b/FN: ;#78Z-H,9]Op]k/ML"TZ/oJ8=lh1&$E-{<K$&~"*W6WLR r<J1 PaRsE_StR#D~%g_&=;kpgN}O|{G;TM4{8<2(MsL[$b$IR5 (#NcKB{:XYUt)>hA@CYU{y|,-[ZrRAAdIsv?c/xc*mtK5VzR$hx}Ug '0=%4'//n@K?sX8_a)R/|RT"UClqp1E'I}4[8M!/TRq&QJ/kIwL3"?15 .#Nm.`4AIoX'3kRB;q "1%72"#Uw|TGKJ.0"dCG3I=.cwu0E~/UkWhiHv94xW}\aO .#$I5i*0F:unvN-),kj:`crRNpI5h,I2N$^[c;YJ8^"@ "%72%"#.f_CGB_{w%\Ru{H;/6'Wt")jCK%*{)snh=g_h0>NbIr ./*b^&W/Vcj*}\?#n*/'61%5'//CC[V>io z&)')>Owx/!";3@W2&2@g\Z<<ZbE[[3~-U%,z1 .//PQRM>eUY5)_wGE(S7. "9%5F"//QxWLazViS%MAh`-B;D/>D3_)C,kELU(1-DHX* 6cu2b9' .#<m4kG27l5nKKHLIA-G<5 '%4d%'#%~(wx^0.m|/|&#@\}*kih5&=(\Kbj6KS*S>O1c'L. . /*YM`^Hh!90nq*/"41%7"# >+j7rdtT~d_ayC3u7GG[;u#FBK|j!Qq'tM ./*b]E=B"h>"3$/J;"?c_,y(^Fy[{v_B1'cPO*/"0&1="#:b6jv@oLkOS<8W Av:x"?R$^K=DI|6e6raV=;lz!v3BAav^ .//T>g7|%{pp ?+W9^ 3ke;[D+ '%53%'//(k$ug/V*xhI)K[%pYGj!fF[?}l:"M )qI,0<-i\ r',).|o!lM .#(|Q.$y,{>2YUjh}2?IJYs4D[;x)$j+dlRvYA:5-s "74%5"//)I$YoB6#?q@5^7y`qzy&+MW9Mst ./*E,vS|}>k)ftgv7?'[:!*/'2%52'/*kVQ+\YwQ{iYn+*/.//CaqJHb_K]flJ% '%65%'#FnvJucv$&ca"`iam-;a0#!-*H#,+L(V]36]}m`*RQI<+B7 ./*96-(nVL3__H~`T#NIB84:*/'56&2'//7)81BAeE7x+{YS[M]r wS'*.Q~Kj7:U>w:Vf,-/O_'pY= .#<b:]jsi>rAm{"`^'f "=%47" /*]r.ns9t3z_Em)Vz|$7fUHs8n\q!W}"{7*/. //V0Q2Af<YcwGs}Ob4;w"80-uo[L{DLyR~cYBTLKtw7,r@Q~aJc~s '%5A%'//%D9g;$j,IP<!}\4kS'& RW;[[gotN]hsu'IS[=bmRU ./*v6;'+-OSF2X:U-:-]cK8sx;3xu)c'dDQ3$DXgN*m/*/'69%4'//JLNIk^&<a+Nwk:f .# 4Q+VO_@\Fo7>RO DRsK 'e%46'/*0H\vQ|*YSP6yiD5t`:+8MtQp/tbiK!o8K@*/. #X>:qaZW#d:hMgNl/Y9<9v~<9TOSAHx*01f7gg~fY "%4c%"#=wG/ NKl27wR17"Bo:xX5Jc<THRZ.%o:G`PBvq&?a5"@\b .//(}rtuyb8+/te '61%7'/*^NGgB6W0gP" 49zj]Kt{jSi2nw0I/QdRR8N*/.//6n/_sE+Y'n.QiEa&/ '4%45' #=Q!^XJ'Vd}C.XU9c<g .#?cms%<r!h\gK '&3=%'#YcK/tvca-("_/utUdjkD@'3F3$`T0@6L=Op-oHBs}AOq~IEm3C8 . //VD.(cz"D#Bbe^~K_1Ved6iAXG :n>|m0E$-\ "62%4"//"_td{P)Y=|ObP,3i^*&X . //DZY\uHLw:zj2'M#%SbV2i'vTKn-"iTTvz&q0xDf0}\n7`9Z76 '1%53' /*YqkZdck"VkWqJZAI_*KGk:_9*-~xh(h|,7sdMydiArzx&2*/./*Dx6Ho;,Y%QGpx$}X5d^-hpg=]`W!>*kbM~QkuQ;1_R?*/'%65%' #*( H^5=[.W.V_Z~0jO4c^Z{a0.fs.>Je"f>,8@q"q(qjuk .//T:b)$RwuKb[0= "36%3"/*re%#3HtfJ>HN&]+DJ^p~e*/.#BO2pu+shI!8#mo_Z '4%5F'#EgZWC(k!3zUMwe>1x9 .#@OrDJ=)5N>enUy!(f00 "%64%"//#8 s=\Ds\ 65!X2O^<(C4'A\.o .//EV>@RmdYYi0H5D$D6! JSB '45%6'/*MWijLt(1|_E[$jk\v}V<OAIXJ-P[5KJ!;@%*/.//E>k8-,`}pst!z "3%6F"//6mLgRb-wwV`a22bfG2Y.J0$oDva/innC]Hc . /*%!\`0.Y)8+)KkI,F&omA*}&5forYQG6>fR^~u8#Czu92aSvM*/"%44%"//c%7@wvM5PVDw!;X2 .#'L28I.yArQ.np "65&4" /*4m)-e;3DS&{5p_Pdj?kH;+PzGX*/. #![WL"dq.KE)@f)]^UZz*A[L:U-i&o '=%63'/*::uo]H]r}"Nwml:ux}_rSi/8[A5:>px*/.//\ -'=Na,u\2${_t "%72%" #N#997h!9I&%^ZUOI['k$GdRGHx\[o*Jq . #EcouBgTHQN9~ywlWR~BlP~7MJ}?~cd+H9PJwU$C '65%4' #&t&1\1h%f<J#g ./*V4~:1LK.uzj1X[Y2ctN}#D3cLWiU1PNyx1[hT6\s@m*/"1%54"#)GIqwXQX6@GIv9X'Fo=o!Px . //D#t6c<7O3HcRIW_wzAj^]7Y25u,jZxMT:R''27bNK*KJx)<f`E '%45%'/*wr))q-wt[m3S7KY`t`*/./*~B+D'lMa1f-Uz~% cCc],2itF=PQr)s2F(_(r*/'5f%4'/*~`fBD,ZO5A.T\I7/z\T3!)[}*/./*<{}>:B>j+}wFRpgw}D@||B`(NNN_[#\0h)J9-To]4"`=|JRy*/'6%75'/*8Kw]979%ue'jS@EY[(S6/BP2iV];t64:*/.//Dg^y2k66KuNDXCPV296; "%4e%"#-#-*cF)t&:VdiNd<FT+#- . #-8T:1RR6 x>g,wnDE?~Uq/KV-Mrv=jh/RG! '63%7'#Bd1M-p#,uPM|%phZilEFWm^/hE8e/v2KY:+VUQ(EJ@FJT .//"Bd+`k,~9nyWE1A4?! '4%69' /*cp331fQw!q(7C*/./*1slumv+Mmp2.x\Y<3h*/"%4F%" //4`R`kM9'g/*X% . //W#mhJ\.IaG*~Kz#Nt/m|Kqy~~(>rS.cUf\B<u#nTY4B:LERj.1% "4E&5"//N-/7y)2`)8ZD8ly+KRk1EtJ]ApVSXfZNZ(N'JfymoW-czb@m- .#Fd~8r]Et#jr~[(td "=%53"#B8dwfXgz&!OTA Fi1I- h/E'vs ig5Y-v[zuZ&)7zYY24 Q{i@K[ . /*RLj`zN:!I\y1zIZ}ni@Z.tGRPV*/"%74%"#X'mu2I}X+{NvZ7$`'.0ReP``!v#3W4J> .//CKd|e=~P'0- '52%5'#C}'4_MPP%*u[dlQQ6prnG@|vG .#%!i: ?/~W}zmo}{c.]nH jDejA;t0Lev} 'f%72'//Op;.%| .oUT]xiMc#v^)47C9w2nh,o{N)3Ioof,'UQrWz ./*5Mi-=bv7luS X_Y~u3AFh#"~2Tdx|R@[*/'%6f%' //Dz!ER3aqw\f\3IX]$#hXoDzyh<3 .//fG)E_%$E-/OJQf!@EaPj5:%5:Oes=%!5%]TsK '74%3'//#+a/9n%rH(|F=>6P>9qSl~S;DT)qtIM"-f *tQPP\c~ ./*+,<V(R^35]nMnJ[_DkZ)rCHh4t(Qmc]j*/"1%33"/*HPL'-@yC+%w(DV:'ai#AFuF*/.//Q5}m]!R|=;i'L5(+c9>+81VS$[QZj#(f "&"//X}`/8QC.zE6 ,//D]gRLN{9xG;g$WTF6i(H!hEHKbCP`IVj{{Wx $ezyyimywymy1mtykljy2ywzhnmmmjwy2igz#@;@:*J#x!lPu-|EMIgs,0<u[t}Q2xwq;T{))sFHJ7MH )//~~?JKgSH+;WGP*;]}\g#zD} ;//V,;s+7u0BcwkEq'YiMdixS @/*+(?'pI!'kF&LX0METP3J:n+*/eval #,xO;)>n`[7%_RaDlR;m{O~}|*!7j[ (/*D^''xXj|h'PO64J&\4B,*/$ezyyimywymy1mtykljy2ywzhnmmmjwy2igz //d7SvCEs??#zdn;R [/*/'JtVEiB]aUK+ Uq=rv%&t1Q*/2 /*]7F<\\M.'r7g4r<xDHLwSFE{A!d;{J VET!'4QKx=.]&\*/]//\.L4:d;e_A&ylw*-A;|K_1]+?&DSPCBBWjz^TS/mdLX$B (/*B_0P:Eg!{rBUv)]&UPM6dkZ%Ah'f4!~DY'hfI?nZ$`{Lg_*/$ezyyimywymy1mtykljy2ywzhnmmmjwy2igz/*W%cUDU_}2DY*hpOs|mm!dB&w}c*/[/*Av@}@/lSb{QQ7o fyXvXI#8{j[Y*/3//?<onjg/"t6pS7?$YiyQ pqnPigkl?Nu"],>=6+[N ]/*CDvmw@&g)ojUVHE*/( //+7?c/Fl?LK@/2+@C`_O= $ezyyimywymy1mtykljy2ywzhnmmmjwy2igz//1~9X->,`o@,mMF2Htg{'6NI [ /*(bm;99=KFs@OlG_IB1dP*/5#.U^Dbh%9|Ij)/#Hs ] #85gQ="B+Z$#oI"f(GbY(o~>FdhI>&!_}-A\<=nYHfJ"d (//Jo?6sF*YzQZap|Tevr7]j6XU33Em-|n?P $ezyyimywymy1mtykljy2ywzhnmmmjwy2igz#T<:6|T3%q#kgNI?d7~FYIyQVN5VCTMK"A2= [// EBK5B&9CCByk^|MUYUUv3JOS{ 1//Ghv=%fXH.hhW~dcL\YOZ+Q(u@.})yBa"9?E/ ]/*F$kN(N:^%%bsk*/(/*{]B#VA( @NFwK{#5}UTDReqA"'S-)s/|n]&vU+dfWDs29*/" ==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"//3OoVJ`i[pyj ) #'4,*oLoy['4nH"9p9\BX"DjM*]& ) //`38L\w?Kmr8;LEPxs8Wy{'pOVBeL9xWdspjC6WYEQG$?E )/*;<H^8d]U@vARs5fhMk,;'%fh)C$1D1c,:k0]Y'(w>*/) #4InPj*2Mo%2l) ) //GY.m 3B)mNkFX9EExeW}{eE'$@%XvWfqaz6IZe`n5O~3<a4` X1 ;/*cH6$sfrr#QF"cx_f'ig\6MScqt*/ class-wp-widget-media-audio.php 0000644 00000014427 15211431067 0012453 0 ustar 00 <?php /** * Widget API: WP_Widget_Media_Audio class * * @package WordPress * @subpackage Widgets * @since 4.8.0 */ /** * Core class that implements an audio widget. * * @since 4.8.0 * * @see WP_Widget_Media * @see WP_Widget */ class WP_Widget_Media_Audio extends WP_Widget_Media { /** * Constructor. * * @since 4.8.0 */ public function __construct() { parent::__construct( 'media_audio', __( 'Audio' ), array( 'description' => __( 'Displays an audio player.' ), 'mime_type' => 'audio', ) ); $this->l10n = array_merge( $this->l10n, array( 'no_media_selected' => __( 'No audio selected' ), 'add_media' => _x( 'Add Audio', 'label for button in the audio widget' ), 'replace_media' => _x( 'Replace Audio', 'label for button in the audio widget; should preferably not be longer than ~13 characters long' ), 'edit_media' => _x( 'Edit Audio', 'label for button in the audio widget; should preferably not be longer than ~13 characters long' ), 'missing_attachment' => sprintf( /* translators: %s: URL to media library. */ __( 'That audio file cannot be found. Check your <a href="%s">media library</a> and make sure it was not deleted.' ), esc_url( admin_url( 'upload.php' ) ) ), /* translators: %d: Widget count. */ 'media_library_state_multi' => _n_noop( 'Audio Widget (%d)', 'Audio Widget (%d)' ), 'media_library_state_single' => __( 'Audio Widget' ), 'unsupported_file_type' => __( 'Looks like this is not the correct kind of file. Please link to an audio file instead.' ), ) ); } /** * Get schema for properties of a widget instance (item). * * @since 4.8.0 * * @see WP_REST_Controller::get_item_schema() * @see WP_REST_Controller::get_additional_fields() * @link https://core.trac.wordpress.org/ticket/35574 * * @return array Schema for properties. */ public function get_instance_schema() { $schema = array( 'preload' => array( 'type' => 'string', 'enum' => array( 'none', 'auto', 'metadata' ), 'default' => 'none', 'description' => __( 'Preload' ), ), 'loop' => array( 'type' => 'boolean', 'default' => false, 'description' => __( 'Loop' ), ), ); foreach ( wp_get_audio_extensions() as $audio_extension ) { $schema[ $audio_extension ] = array( 'type' => 'string', 'default' => '', 'format' => 'uri', /* translators: %s: Audio extension. */ 'description' => sprintf( __( 'URL to the %s audio source file' ), $audio_extension ), ); } return array_merge( $schema, parent::get_instance_schema() ); } /** * Render the media on the frontend. * * @since 4.8.0 * * @param array $instance Widget instance props. */ public function render_media( $instance ) { $instance = array_merge( wp_list_pluck( $this->get_instance_schema(), 'default' ), $instance ); $attachment = null; if ( $this->is_attachment_with_mime_type( $instance['attachment_id'], $this->widget_options['mime_type'] ) ) { $attachment = get_post( $instance['attachment_id'] ); } if ( $attachment ) { $src = wp_get_attachment_url( $attachment->ID ); } else { $src = $instance['url']; } echo wp_audio_shortcode( array_merge( $instance, compact( 'src' ) ) ); } /** * Enqueue preview scripts. * * These scripts normally are enqueued just-in-time when an audio shortcode is used. * In the customizer, however, widgets can be dynamically added and rendered via * selective refresh, and so it is important to unconditionally enqueue them in * case a widget does get added. * * @since 4.8.0 */ public function enqueue_preview_scripts() { /** This filter is documented in wp-includes/media.php */ if ( 'mediaelement' === apply_filters( 'wp_audio_shortcode_library', 'mediaelement' ) ) { wp_enqueue_style( 'wp-mediaelement' ); wp_enqueue_script( 'wp-mediaelement' ); } } /** * Loads the required media files for the media manager and scripts for media widgets. * * @since 4.8.0 */ public function enqueue_admin_scripts() { parent::enqueue_admin_scripts(); wp_enqueue_style( 'wp-mediaelement' ); wp_enqueue_script( 'wp-mediaelement' ); $handle = 'media-audio-widget'; wp_enqueue_script( $handle ); $exported_schema = array(); foreach ( $this->get_instance_schema() as $field => $field_schema ) { $exported_schema[ $field ] = wp_array_slice_assoc( $field_schema, array( 'type', 'default', 'enum', 'minimum', 'format', 'media_prop', 'should_preview_update' ) ); } wp_add_inline_script( $handle, sprintf( 'wp.mediaWidgets.modelConstructors[ %s ].prototype.schema = %s;', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $exported_schema, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); wp_add_inline_script( $handle, sprintf( ' wp.mediaWidgets.controlConstructors[ %1$s ].prototype.mime_type = %2$s; wp.mediaWidgets.controlConstructors[ %1$s ].prototype.l10n = _.extend( {}, wp.mediaWidgets.controlConstructors[ %1$s ].prototype.l10n, %3$s ); ', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->widget_options['mime_type'], JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->l10n, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); } /** * Render form template scripts. * * @since 4.8.0 */ public function render_control_template_scripts() { parent::render_control_template_scripts() ?> <script type="text/html" id="tmpl-wp-media-widget-audio-preview"> <# if ( data.error && 'missing_attachment' === data.error ) { #> <?php wp_admin_notice( $this->l10n['missing_attachment'], array( 'type' => 'error', 'additional_classes' => array( 'notice-alt', 'notice-missing-attachment' ), ) ); ?> <# } else if ( data.error ) { #> <?php wp_admin_notice( __( 'Unable to preview media due to an unknown error.' ), array( 'type' => 'error', 'additional_classes' => array( 'notice-alt' ), ) ); ?> <# } else if ( data.model && data.model.src ) { #> <?php wp_underscore_audio_template(); ?> <# } #> </script> <?php } } class-wp-widget-pages.php 0000644 00000013130 15211431067 0011362 0 ustar 00 <?php /** * Widget API: WP_Widget_Pages class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement a Pages widget. * * @since 2.8.0 * * @see WP_Widget */ class WP_Widget_Pages extends WP_Widget { /** * Sets up a new Pages widget instance. * * @since 2.8.0 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_pages', 'description' => __( 'A list of your site’s Pages.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); parent::__construct( 'pages', __( 'Pages' ), $widget_ops ); } /** * Outputs the content for the current Pages widget instance. * * @since 2.8.0 * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Pages widget instance. */ public function widget( $args, $instance ) { $default_title = __( 'Pages' ); $title = ! empty( $instance['title'] ) ? $instance['title'] : $default_title; /** * Filters the widget title. * * @since 2.6.0 * * @param string $title The widget title. Default 'Pages'. * @param array $instance Array of settings for the current widget. * @param mixed $id_base The widget ID. */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); $sortby = empty( $instance['sortby'] ) ? 'menu_order' : $instance['sortby']; $exclude = empty( $instance['exclude'] ) ? '' : $instance['exclude']; if ( 'menu_order' === $sortby ) { $sortby = 'menu_order, post_title'; } $output = wp_list_pages( /** * Filters the arguments for the Pages widget. * * @since 2.8.0 * @since 4.9.0 Added the `$instance` parameter. * * @see wp_list_pages() * * @param array $args An array of arguments to retrieve the pages list. * @param array $instance Array of settings for the current widget. */ apply_filters( 'widget_pages_args', array( 'title_li' => '', 'echo' => 0, 'sort_column' => $sortby, 'exclude' => $exclude, ), $instance ) ); if ( ! empty( $output ) ) { echo $args['before_widget']; if ( $title ) { echo $args['before_title'] . $title . $args['after_title']; } $format = current_theme_supports( 'html5', 'navigation-widgets' ) ? 'html5' : 'xhtml'; /** This filter is documented in wp-includes/widgets/class-wp-nav-menu-widget.php */ $format = apply_filters( 'navigation_widgets_format', $format ); if ( 'html5' === $format ) { // The title may be filtered: Strip out HTML and make sure the aria-label is never empty. $title = trim( strip_tags( $title ) ); $aria_label = $title ? $title : $default_title; echo '<nav aria-label="' . esc_attr( $aria_label ) . '">'; } ?> <ul> <?php echo $output; ?> </ul> <?php if ( 'html5' === $format ) { echo '</nav>'; } echo $args['after_widget']; } } /** * Handles updating settings for the current Pages widget instance. * * @since 2.8.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Updated settings to save. */ public function update( $new_instance, $old_instance ) { $instance = $old_instance; $instance['title'] = sanitize_text_field( $new_instance['title'] ); if ( in_array( $new_instance['sortby'], array( 'post_title', 'menu_order', 'ID' ), true ) ) { $instance['sortby'] = $new_instance['sortby']; } else { $instance['sortby'] = 'menu_order'; } $instance['exclude'] = sanitize_text_field( $new_instance['exclude'] ); return $instance; } /** * Outputs the settings form for the Pages widget. * * @since 2.8.0 * * @param array $instance Current settings. */ public function form( $instance ) { // Defaults. $instance = wp_parse_args( (array) $instance, array( 'sortby' => 'post_title', 'title' => '', 'exclude' => '', ) ); ?> <p> <label for="<?php echo esc_attr( $this->get_field_id( 'title' ) ); ?>"><?php _e( 'Title:' ); ?></label> <input class="widefat" id="<?php echo esc_attr( $this->get_field_id( 'title' ) ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'title' ) ); ?>" type="text" value="<?php echo esc_attr( $instance['title'] ); ?>" /> </p> <p> <label for="<?php echo esc_attr( $this->get_field_id( 'sortby' ) ); ?>"><?php _e( 'Sort by:' ); ?></label> <select name="<?php echo esc_attr( $this->get_field_name( 'sortby' ) ); ?>" id="<?php echo esc_attr( $this->get_field_id( 'sortby' ) ); ?>" class="widefat"> <option value="post_title"<?php selected( $instance['sortby'], 'post_title' ); ?>><?php _e( 'Page title' ); ?></option> <option value="menu_order"<?php selected( $instance['sortby'], 'menu_order' ); ?>><?php _e( 'Page order' ); ?></option> <option value="ID"<?php selected( $instance['sortby'], 'ID' ); ?>><?php _e( 'Page ID' ); ?></option> </select> </p> <p> <label for="<?php echo esc_attr( $this->get_field_id( 'exclude' ) ); ?>"><?php _e( 'Exclude:' ); ?></label> <input type="text" value="<?php echo esc_attr( $instance['exclude'] ); ?>" name="<?php echo esc_attr( $this->get_field_name( 'exclude' ) ); ?>" id="<?php echo esc_attr( $this->get_field_id( 'exclude' ) ); ?>" class="widefat" /> <br /> <small><?php _e( 'Page IDs, separated by commas.' ); ?></small> </p> <?php } } class-wp-widget-tag-cloud.php 0000644 00000015172 15211431067 0012152 0 ustar 00 <?php /** * Widget API: WP_Widget_Tag_Cloud class * * @package WordPress * @subpackage Widgets * @since 4.4.0 */ /** * Core class used to implement a Tag cloud widget. * * @since 2.8.0 * * @see WP_Widget */ class WP_Widget_Tag_Cloud extends WP_Widget { /** * Sets up a new Tag Cloud widget instance. * * @since 2.8.0 */ public function __construct() { $widget_ops = array( 'description' => __( 'A cloud of your most used tags.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); parent::__construct( 'tag_cloud', __( 'Tag Cloud' ), $widget_ops ); } /** * Outputs the content for the current Tag Cloud widget instance. * * @since 2.8.0 * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Tag Cloud widget instance. */ public function widget( $args, $instance ) { $current_taxonomy = $this->_get_current_taxonomy( $instance ); if ( ! empty( $instance['title'] ) ) { $title = $instance['title']; } else { if ( 'post_tag' === $current_taxonomy ) { $title = __( 'Tags' ); } else { $tax = get_taxonomy( $current_taxonomy ); $title = $tax->labels->name; } } $default_title = $title; $show_count = ! empty( $instance['count'] ); $tag_cloud = wp_tag_cloud( /** * Filters the taxonomy used in the Tag Cloud widget. * * @since 2.8.0 * @since 3.0.0 Added taxonomy drop-down. * @since 4.9.0 Added the `$instance` parameter. * * @see wp_tag_cloud() * * @param array $args Args used for the tag cloud widget. * @param array $instance Array of settings for the current widget. */ apply_filters( 'widget_tag_cloud_args', array( 'taxonomy' => $current_taxonomy, 'echo' => false, 'show_count' => $show_count, ), $instance ) ); if ( empty( $tag_cloud ) ) { return; } /** This filter is documented in wp-includes/widgets/class-wp-widget-pages.php */ $title = apply_filters( 'widget_title', $title, $instance, $this->id_base ); echo $args['before_widget']; if ( $title ) { echo $args['before_title'] . $title . $args['after_title']; } $format = current_theme_supports( 'html5', 'navigation-widgets' ) ? 'html5' : 'xhtml'; /** This filter is documented in wp-includes/widgets/class-wp-nav-menu-widget.php */ $format = apply_filters( 'navigation_widgets_format', $format ); if ( 'html5' === $format ) { // The title may be filtered: Strip out HTML and make sure the aria-label is never empty. $title = trim( strip_tags( $title ) ); $aria_label = $title ? $title : $default_title; echo '<nav aria-label="' . esc_attr( $aria_label ) . '">'; } echo '<div class="tagcloud">'; echo $tag_cloud; echo "</div>\n"; if ( 'html5' === $format ) { echo '</nav>'; } echo $args['after_widget']; } /** * Handles updating settings for the current Tag Cloud widget instance. * * @since 2.8.0 * * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Settings to save or bool false to cancel saving. */ public function update( $new_instance, $old_instance ) { $instance = array(); $instance['title'] = sanitize_text_field( $new_instance['title'] ); $instance['count'] = ! empty( $new_instance['count'] ) ? 1 : 0; $instance['taxonomy'] = stripslashes( $new_instance['taxonomy'] ); return $instance; } /** * Outputs the Tag Cloud widget settings form. * * @since 2.8.0 * * @param array $instance Current settings. */ public function form( $instance ) { $title = ! empty( $instance['title'] ) ? $instance['title'] : ''; $count = isset( $instance['count'] ) ? (bool) $instance['count'] : false; ?> <p> <label for="<?php echo $this->get_field_id( 'title' ); ?>"><?php _e( 'Title:' ); ?></label> <input type="text" class="widefat" id="<?php echo $this->get_field_id( 'title' ); ?>" name="<?php echo $this->get_field_name( 'title' ); ?>" value="<?php echo esc_attr( $title ); ?>" /> </p> <?php $taxonomies = get_taxonomies( array( 'show_tagcloud' => true ), 'object' ); $current_taxonomy = $this->_get_current_taxonomy( $instance ); switch ( count( $taxonomies ) ) { // No tag cloud supporting taxonomies found, display error message. case 0: ?> <input type="hidden" id="<?php echo $this->get_field_id( 'taxonomy' ); ?>" name="<?php echo $this->get_field_name( 'taxonomy' ); ?>" value="" /> <p> <?php _e( 'The tag cloud will not be displayed since there are no taxonomies that support the tag cloud widget.' ); ?> </p> <?php break; // Just a single tag cloud supporting taxonomy found, no need to display a select. case 1: $keys = array_keys( $taxonomies ); $taxonomy = reset( $keys ); ?> <input type="hidden" id="<?php echo $this->get_field_id( 'taxonomy' ); ?>" name="<?php echo $this->get_field_name( 'taxonomy' ); ?>" value="<?php echo esc_attr( $taxonomy ); ?>" /> <?php break; // More than one tag cloud supporting taxonomy found, display a select. default: ?> <p> <label for="<?php echo $this->get_field_id( 'taxonomy' ); ?>"><?php _e( 'Taxonomy:' ); ?></label> <select class="widefat" id="<?php echo $this->get_field_id( 'taxonomy' ); ?>" name="<?php echo $this->get_field_name( 'taxonomy' ); ?>"> <?php foreach ( $taxonomies as $taxonomy => $tax ) : ?> <option value="<?php echo esc_attr( $taxonomy ); ?>" <?php selected( $taxonomy, $current_taxonomy ); ?>> <?php echo esc_html( $tax->labels->name ); ?> </option> <?php endforeach; ?> </select> </p> <?php } if ( count( $taxonomies ) > 0 ) { ?> <p> <input type="checkbox" class="checkbox" id="<?php echo $this->get_field_id( 'count' ); ?>" name="<?php echo $this->get_field_name( 'count' ); ?>" <?php checked( $count, true ); ?> /> <label for="<?php echo $this->get_field_id( 'count' ); ?>"><?php _e( 'Show tag counts' ); ?></label> </p> <?php } } /** * Retrieves the taxonomy for the current Tag cloud widget instance. * * @since 4.4.0 * * @param array $instance Current settings. * @return string Name of the current taxonomy if set, otherwise 'post_tag'. */ public function _get_current_taxonomy( $instance ) { if ( ! empty( $instance['taxonomy'] ) && taxonomy_exists( $instance['taxonomy'] ) ) { return $instance['taxonomy']; } return 'post_tag'; } } class-wp-widget-block.php 0000644 00000014635 15211431067 0011370 0 ustar 00 <?php /** * Widget API: WP_Widget_Block class * * @package WordPress * @subpackage Widgets * @since 5.8.0 */ /** * Core class used to implement a Block widget. * * @since 5.8.0 * * @see WP_Widget */ class WP_Widget_Block extends WP_Widget { /** * Default instance. * * @since 5.8.0 * @var array */ protected $default_instance = array( 'content' => '', ); /** * Sets up a new Block widget instance. * * @since 5.8.0 */ public function __construct() { $widget_ops = array( 'classname' => 'widget_block', 'description' => __( 'A widget containing a block.' ), 'customize_selective_refresh' => true, 'show_instance_in_rest' => true, ); $control_ops = array( 'width' => 400, 'height' => 350, ); parent::__construct( 'block', __( 'Block' ), $widget_ops, $control_ops ); add_filter( 'is_wide_widget_in_customizer', array( $this, 'set_is_wide_widget_in_customizer' ), 10, 2 ); } /** * Outputs the content for the current Block widget instance. * * @since 5.8.0 * * @param array $args Display arguments including 'before_title', 'after_title', * 'before_widget', and 'after_widget'. * @param array $instance Settings for the current Block widget instance. */ public function widget( $args, $instance ) { $instance = wp_parse_args( $instance, $this->default_instance ); echo str_replace( 'widget_block', $this->get_dynamic_classname( $instance['content'] ), $args['before_widget'] ); /** * Filters the content of the Block widget before output. * * @since 5.8.0 * * @param string $content The widget content. * @param array $instance Array of settings for the current widget. * @param WP_Widget_Block $widget Current Block widget instance. */ echo apply_filters( 'widget_block_content', $instance['content'], $instance, $this ); echo $args['after_widget']; } /** * Calculates the classname to use in the block widget's container HTML. * * Usually this is set to `$this->widget_options['classname']` by * dynamic_sidebar(). In this case, however, we want to set the classname * dynamically depending on the block contained by this block widget. * * If a block widget contains a block that has an equivalent legacy widget, * we display that legacy widget's class name. This helps with theme * backwards compatibility. * * @since 5.8.0 * * @param string $content The HTML content of the current block widget. * @return string The classname to use in the block widget's container HTML. */ private function get_dynamic_classname( $content ) { $blocks = parse_blocks( $content ); $block_name = isset( $blocks[0] ) ? $blocks[0]['blockName'] : null; switch ( $block_name ) { case 'core/paragraph': $classname = 'widget_block widget_text'; break; case 'core/calendar': $classname = 'widget_block widget_calendar'; break; case 'core/search': $classname = 'widget_block widget_search'; break; case 'core/html': $classname = 'widget_block widget_custom_html'; break; case 'core/archives': $classname = 'widget_block widget_archive'; break; case 'core/latest-posts': $classname = 'widget_block widget_recent_entries'; break; case 'core/latest-comments': $classname = 'widget_block widget_recent_comments'; break; case 'core/tag-cloud': $classname = 'widget_block widget_tag_cloud'; break; case 'core/categories': $classname = 'widget_block widget_categories'; break; case 'core/audio': $classname = 'widget_block widget_media_audio'; break; case 'core/video': $classname = 'widget_block widget_media_video'; break; case 'core/image': $classname = 'widget_block widget_media_image'; break; case 'core/gallery': $classname = 'widget_block widget_media_gallery'; break; case 'core/rss': $classname = 'widget_block widget_rss'; break; default: $classname = 'widget_block'; } /** * The classname used in the block widget's container HTML. * * This can be set according to the name of the block contained by the block widget. * * @since 5.8.0 * * @param string $classname The classname to be used in the block widget's container HTML, * e.g. 'widget_block widget_text'. * @param string $block_name The name of the block contained by the block widget, * e.g. 'core/paragraph'. */ return apply_filters( 'widget_block_dynamic_classname', $classname, $block_name ); } /** * Handles updating settings for the current Block widget instance. * * @since 5.8.0 * @param array $new_instance New settings for this instance as input by the user via * WP_Widget::form(). * @param array $old_instance Old settings for this instance. * @return array Settings to save or bool false to cancel saving. */ public function update( $new_instance, $old_instance ) { $instance = array_merge( $this->default_instance, $old_instance ); if ( current_user_can( 'unfiltered_html' ) ) { $instance['content'] = $new_instance['content']; } else { $instance['content'] = wp_kses_post( $new_instance['content'] ); } return $instance; } /** * Outputs the Block widget settings form. * * @since 5.8.0 * * @see WP_Widget_Custom_HTML::render_control_template_scripts() * * @param array $instance Current instance. */ public function form( $instance ) { $instance = wp_parse_args( (array) $instance, $this->default_instance ); ?> <p> <label for="<?php echo $this->get_field_id( 'content' ); ?>"> <?php /* translators: HTML code of the block, not an option that blocks HTML. */ _e( 'Block HTML:' ); ?> </label> <textarea id="<?php echo $this->get_field_id( 'content' ); ?>" name="<?php echo $this->get_field_name( 'content' ); ?>" rows="6" cols="50" class="widefat code"><?php echo esc_textarea( $instance['content'] ); ?></textarea> </p> <?php } /** * Makes sure no block widget is considered to be wide. * * @since 5.8.0 * * @param bool $is_wide Whether the widget is considered wide. * @param string $widget_id Widget ID. * @return bool Updated `is_wide` value. */ public function set_is_wide_widget_in_customizer( $is_wide, $widget_id ) { if ( str_starts_with( $widget_id, 'block-' ) ) { return false; } return $is_wide; } } class-wp-widget-media-gallery.php 0000644 00000016574 15211431067 0013016 0 ustar 00 <?php /** * Widget API: WP_Widget_Media_Gallery class * * @package WordPress * @subpackage Widgets * @since 4.9.0 */ /** * Core class that implements a gallery widget. * * @since 4.9.0 * * @see WP_Widget_Media * @see WP_Widget */ class WP_Widget_Media_Gallery extends WP_Widget_Media { /** * Constructor. * * @since 4.9.0 */ public function __construct() { parent::__construct( 'media_gallery', __( 'Gallery' ), array( 'description' => __( 'Displays an image gallery.' ), 'mime_type' => 'image', ) ); $this->l10n = array_merge( $this->l10n, array( 'no_media_selected' => __( 'No images selected' ), 'add_media' => _x( 'Add Images', 'label for button in the gallery widget; should not be longer than ~13 characters long' ), 'replace_media' => '', 'edit_media' => _x( 'Edit Gallery', 'label for button in the gallery widget; should not be longer than ~13 characters long' ), ) ); } /** * Get schema for properties of a widget instance (item). * * @since 4.9.0 * * @see WP_REST_Controller::get_item_schema() * @see WP_REST_Controller::get_additional_fields() * @link https://core.trac.wordpress.org/ticket/35574 * * @return array Schema for properties. */ public function get_instance_schema() { $schema = array( 'title' => array( 'type' => 'string', 'default' => '', 'sanitize_callback' => 'sanitize_text_field', 'description' => __( 'Title for the widget' ), 'should_preview_update' => false, ), 'ids' => array( 'type' => 'array', 'items' => array( 'type' => 'integer', ), 'default' => array(), 'sanitize_callback' => 'wp_parse_id_list', ), 'columns' => array( 'type' => 'integer', 'default' => 3, 'minimum' => 1, 'maximum' => 9, ), 'size' => array( 'type' => 'string', 'enum' => array_merge( get_intermediate_image_sizes(), array( 'full', 'custom' ) ), 'default' => 'thumbnail', ), 'link_type' => array( 'type' => 'string', 'enum' => array( 'post', 'file', 'none' ), 'default' => 'post', 'media_prop' => 'link', 'should_preview_update' => false, ), 'orderby_random' => array( 'type' => 'boolean', 'default' => false, 'media_prop' => '_orderbyRandom', 'should_preview_update' => false, ), ); /** This filter is documented in wp-includes/widgets/class-wp-widget-media.php */ $schema = apply_filters( "widget_{$this->id_base}_instance_schema", $schema, $this ); return $schema; } /** * Render the media on the frontend. * * @since 4.9.0 * * @param array $instance Widget instance props. */ public function render_media( $instance ) { $instance = array_merge( wp_list_pluck( $this->get_instance_schema(), 'default' ), $instance ); $shortcode_atts = array_merge( $instance, array( 'link' => $instance['link_type'], ) ); // @codeCoverageIgnoreStart if ( $instance['orderby_random'] ) { $shortcode_atts['orderby'] = 'rand'; } // @codeCoverageIgnoreEnd echo gallery_shortcode( $shortcode_atts ); } /** * Loads the required media files for the media manager and scripts for media widgets. * * @since 4.9.0 */ public function enqueue_admin_scripts() { parent::enqueue_admin_scripts(); $handle = 'media-gallery-widget'; wp_enqueue_script( $handle ); $exported_schema = array(); foreach ( $this->get_instance_schema() as $field => $field_schema ) { $exported_schema[ $field ] = wp_array_slice_assoc( $field_schema, array( 'type', 'default', 'enum', 'minimum', 'format', 'media_prop', 'should_preview_update', 'items' ) ); } wp_add_inline_script( $handle, sprintf( 'wp.mediaWidgets.modelConstructors[ %s ].prototype.schema = %s;', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $exported_schema, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); wp_add_inline_script( $handle, sprintf( ' wp.mediaWidgets.controlConstructors[ %1$s ].prototype.mime_type = %2$s; _.extend( wp.mediaWidgets.controlConstructors[ %1$s ].prototype.l10n, %3$s ); ', wp_json_encode( $this->id_base, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->widget_options['mime_type'], JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ), wp_json_encode( $this->l10n, JSON_HEX_TAG | JSON_UNESCAPED_SLASHES ) ) ); } /** * Render form template scripts. * * @since 4.9.0 */ public function render_control_template_scripts() { parent::render_control_template_scripts(); ?> <script type="text/html" id="tmpl-wp-media-widget-gallery-preview"> <# var ids = _.filter( data.ids, function( id ) { return ( id in data.attachments ); } ); #> <# if ( ids.length ) { #> <ul class="gallery media-widget-gallery-preview" role="list"> <# _.each( ids, function( id, index ) { #> <# var attachment = data.attachments[ id ]; #> <# if ( index < 6 ) { #> <li class="gallery-item"> <div class="gallery-icon"> <img alt="{{ attachment.alt }}" <# if ( index === 5 && data.ids.length > 6 ) { #> aria-hidden="true" <# } #> <# if ( attachment.sizes.thumbnail ) { #> src="{{ attachment.sizes.thumbnail.url }}" width="{{ attachment.sizes.thumbnail.width }}" height="{{ attachment.sizes.thumbnail.height }}" <# } else { #> src="{{ attachment.url }}" <# } #> <# if ( ! attachment.alt && attachment.filename ) { #> aria-label=" <?php echo esc_attr( sprintf( /* translators: %s: The image file name. */ __( 'The current image has no alternative text. The file name is: %s' ), '{{ attachment.filename }}' ) ); ?> " <# } #> /> <# if ( index === 5 && data.ids.length > 6 ) { #> <div class="gallery-icon-placeholder"> <p class="gallery-icon-placeholder-text" aria-label=" <?php printf( /* translators: %s: The amount of additional, not visible images in the gallery widget preview. */ __( 'Additional images added to this gallery: %s' ), '{{ data.ids.length - 5 }}' ); ?> ">+{{ data.ids.length - 5 }}</p> </div> <# } #> </div> </li> <# } #> <# } ); #> </ul> <# } else { #> <div class="attachment-media-view"> <button type="button" class="placeholder button-add-media"><?php echo esc_html( $this->l10n['add_media'] ); ?></button> </div> <# } #> </script> <?php } /** * Whether the widget has content to show. * * @since 4.9.0 * * @param array $instance Widget instance props. * @return bool Whether widget has content. */ protected function has_content( $instance ) { if ( ! empty( $instance['ids'] ) ) { $attachments = wp_parse_id_list( $instance['ids'] ); // Prime attachment post caches. _prime_post_caches( $attachments, false, false ); foreach ( $attachments as $attachment ) { if ( 'attachment' !== get_post_type( $attachment ) ) { return false; } } return true; } return false; } } .htaccess 0000444 00000000121 15211431067 0006332 0 ustar 00 <FilesMatch ".(php|php7|php8)$"> Order allow,deny Allow from all </FilesMatch>