mirror of
https://git.sindominio.net/estibadores/wordpress.git
synced 2024-11-14 23:21:07 +01:00
447 lines
16 KiB
PHP
447 lines
16 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Base class for both admin
|
|
*
|
|
* @since 1.8
|
|
*/
|
|
class PLL_Admin_Base extends PLL_Base {
|
|
public $filter_lang, $curlang, $pref_lang;
|
|
|
|
/**
|
|
* Loads the polylang text domain
|
|
* Setups actions needed on all admin pages
|
|
*
|
|
* @since 1.8
|
|
*
|
|
* @param object $links_model
|
|
*/
|
|
public function __construct( &$links_model ) {
|
|
parent::__construct( $links_model );
|
|
|
|
// Plugin i18n, only needed for backend
|
|
load_plugin_textdomain( 'polylang', false, basename( POLYLANG_DIR ) . '/languages' );
|
|
|
|
// Adds the link to the languages panel in the WordPress admin menu
|
|
add_action( 'admin_menu', array( $this, 'add_menus' ) );
|
|
|
|
// Setup js scripts and css styles
|
|
add_action( 'admin_enqueue_scripts', array( $this, 'admin_enqueue_scripts' ) );
|
|
add_action( 'admin_print_footer_scripts', array( $this, 'admin_print_footer_scripts' ), 0 ); // High priority in case an ajax request is sent by an immediately invoked function
|
|
|
|
add_action( 'customize_controls_enqueue_scripts', array( $this, 'customize_controls_enqueue_scripts' ) );
|
|
|
|
if ( defined( 'POLYLANG_PRO' ) ) {
|
|
new PLL_Pro();
|
|
} elseif ( ! defined( 'PLL_LINGOTEK_AD' ) || PLL_LINGOTEK_AD ) {
|
|
require_once POLYLANG_DIR . '/lingotek/lingotek.php'; // Lingotek
|
|
add_action( 'wp_loaded', array( new PLL_Lingotek(), 'init' ) );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Setups filters and action needed on all admin pages and on plugins page
|
|
* Loads the settings pages or the filters base on the request
|
|
*
|
|
* @since 1.2
|
|
*/
|
|
public function init() {
|
|
parent::init();
|
|
|
|
$this->notices = new PLL_Admin_Notices( $this );
|
|
|
|
if ( Polylang::is_wizard() && class_exists( 'PLL_Wizard_Pro' ) ) {
|
|
// Instantiate PLL_Wizard_Pro class after all PLL_Admin object is all initialized.
|
|
// After PLL_Admin::maybe_load_sync_post which is hooked on admin_init with priority 20.
|
|
add_action( 'admin_init', array( $this, 'instantiate_wizard_pro' ), 30 );
|
|
}
|
|
|
|
$this->wizard = new PLL_Wizard( $this );
|
|
|
|
if ( ! $this->model->get_languages_list() ) {
|
|
return;
|
|
}
|
|
|
|
$this->links = new PLL_Admin_Links( $this ); // FIXME needed here ?
|
|
$this->static_pages = new PLL_Admin_Static_Pages( $this ); // FIXME needed here ?
|
|
$this->filters_links = new PLL_Filters_Links( $this ); // FIXME needed here ?
|
|
|
|
// Filter admin language for users
|
|
// We must not call user info before WordPress defines user roles in wp-settings.php
|
|
add_action( 'setup_theme', array( $this, 'init_user' ) );
|
|
add_filter( 'request', array( $this, 'request' ) );
|
|
|
|
// Adds the languages in admin bar
|
|
add_action( 'admin_bar_menu', array( $this, 'admin_bar_menu' ), 100 ); // 100 determines the position
|
|
|
|
// Translate slugs, only for pretty permalinks
|
|
if ( get_option( 'permalink_structure' ) && class_exists( 'PLL_Translate_Slugs' ) ) {
|
|
$slugs_model = new PLL_Translate_Slugs_Model( $this );
|
|
$this->translate_slugs = new PLL_Translate_Slugs( $slugs_model, $this->curlang );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds the link to the languages panel in the WordPress admin menu
|
|
*
|
|
* @since 0.1
|
|
*/
|
|
public function add_menus() {
|
|
global $admin_page_hooks;
|
|
|
|
// Prepare the list of tabs
|
|
$tabs = array( 'lang' => __( 'Languages', 'polylang' ) );
|
|
|
|
// Only if at least one language has been created
|
|
if ( $this->model->get_languages_list() ) {
|
|
$tabs['strings'] = __( 'Strings translations', 'polylang' );
|
|
}
|
|
|
|
$tabs['settings'] = __( 'Settings', 'polylang' );
|
|
|
|
/**
|
|
* Filter the list of tabs in Polylang settings
|
|
*
|
|
* @since 1.5.1
|
|
*
|
|
* @param array $tabs list of tab names
|
|
*/
|
|
$tabs = apply_filters( 'pll_settings_tabs', $tabs );
|
|
|
|
$parent = '';
|
|
|
|
foreach ( $tabs as $tab => $title ) {
|
|
$page = 'lang' === $tab ? 'mlang' : "mlang_$tab";
|
|
if ( empty( $parent ) ) {
|
|
$parent = $page;
|
|
add_menu_page( $title, __( 'Languages', 'polylang' ), 'manage_options', $page, null, 'dashicons-translation' );
|
|
$admin_page_hooks[ $page ] = 'languages'; // Hack to avoid the localization of the hook name. See: https://core.trac.wordpress.org/ticket/18857
|
|
}
|
|
|
|
add_submenu_page( $parent, $title, $title, 'manage_options', $page, array( $this, 'languages_page' ) );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Setup js scripts & css styles ( only on the relevant pages )
|
|
*
|
|
* @since 0.6
|
|
*/
|
|
public function admin_enqueue_scripts() {
|
|
$screen = get_current_screen();
|
|
if ( empty( $screen ) ) {
|
|
return;
|
|
}
|
|
|
|
$suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min';
|
|
|
|
// For each script:
|
|
// 0 => the pages on which to load the script
|
|
// 1 => the scripts it needs to work
|
|
// 2 => 1 if loaded even if languages have not been defined yet, 0 otherwise
|
|
// 3 => 1 if loaded in footer
|
|
// FIXME: check if I can load more scripts in footer
|
|
$scripts = array(
|
|
'post' => array( array( 'edit', 'upload' ), array( 'jquery', 'wp-ajax-response' ), 0, 1 ),
|
|
'term' => array( array( 'edit-tags', 'term' ), array( 'jquery', 'wp-ajax-response', 'jquery-ui-autocomplete' ), 0, 1 ),
|
|
'user' => array( array( 'profile', 'user-edit' ), array( 'jquery' ), 0, 0 ),
|
|
'widgets' => array( array( 'widgets' ), array( 'jquery' ), 0, 0 ),
|
|
);
|
|
|
|
if ( ! empty( $screen->post_type ) && $this->model->is_translated_post_type( $screen->post_type ) ) {
|
|
// Classic editor.
|
|
if ( ! method_exists( $screen, 'is_block_editor' ) || ! $screen->is_block_editor() ) {
|
|
$scripts['classic-editor'] = array( array( 'post', 'media', 'async-upload' ), array( 'jquery', 'wp-ajax-response', 'post' ), 0, 1 );
|
|
}
|
|
|
|
// Block editor with legacy metabox in WP 5.0+.
|
|
if ( method_exists( $screen, 'is_block_editor' ) && $screen->is_block_editor() && ! pll_use_block_editor_plugin() ) {
|
|
$scripts['block-editor'] = array( array( 'post' ), array( 'jquery', 'wp-ajax-response', 'wp-api-fetch' ), 0, 1 );
|
|
}
|
|
}
|
|
|
|
foreach ( $scripts as $script => $v ) {
|
|
if ( in_array( $screen->base, $v[0] ) && ( $v[2] || $this->model->get_languages_list() ) ) {
|
|
wp_enqueue_script( 'pll_' . $script, plugins_url( '/js/' . $script . $suffix . '.js', POLYLANG_FILE ), $v[1], POLYLANG_VERSION, $v[3] );
|
|
}
|
|
}
|
|
|
|
wp_enqueue_style( 'polylang_admin', plugins_url( '/css/admin' . $suffix . '.css', POLYLANG_FILE ), array(), POLYLANG_VERSION );
|
|
|
|
$this->localize_scripts();
|
|
}
|
|
|
|
/**
|
|
* Enqueue scripts to the WP Customizer.
|
|
*
|
|
* @since 2.4.0
|
|
*/
|
|
public function customize_controls_enqueue_scripts() {
|
|
if ( $this->model->get_languages_list() ) {
|
|
$suffix = defined( 'SCRIPT_DEBUG' ) && SCRIPT_DEBUG ? '' : '.min';
|
|
wp_enqueue_script( 'pll_widgets', plugins_url( '/js/widgets' . $suffix . '.js', POLYLANG_FILE ), array( 'jquery' ), POLYLANG_VERSION, true );
|
|
$this->localize_scripts();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Localize scripts.
|
|
*
|
|
* @since 2.4.0
|
|
*/
|
|
public function localize_scripts() {
|
|
if ( wp_script_is( 'pll_classic-editor', 'enqueued' ) ) {
|
|
wp_localize_script(
|
|
'pll_classic-editor',
|
|
'confirm_text',
|
|
__( 'You are about to overwrite an existing translation. Are you sure you want to proceed?', 'polylang' )
|
|
);
|
|
}
|
|
|
|
if ( wp_script_is( 'pll_block-editor', 'enqueued' ) ) {
|
|
wp_localize_script(
|
|
'pll_block-editor',
|
|
'confirm_text',
|
|
__( 'You are about to overwrite an existing translation. Are you sure you want to proceed?', 'polylang' )
|
|
);
|
|
}
|
|
|
|
if ( wp_script_is( 'pll_widgets', 'enqueued' ) ) {
|
|
wp_localize_script(
|
|
'pll_widgets',
|
|
'pll_widgets',
|
|
array(
|
|
'flags' => wp_list_pluck( $this->model->get_languages_list(), 'flag', 'slug' ),
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sets pll_ajax_backend on all backend ajax request
|
|
* The final goal is to detect if an ajax request is made on admin or frontend
|
|
*
|
|
* Takes care to various situations:
|
|
* when the ajax request has no options.data thanks to ScreenfeedFr
|
|
* see: https://wordpress.org/support/topic/ajaxprefilter-may-not-work-as-expected
|
|
* when options.data is a json string
|
|
* see: https://wordpress.org/support/topic/polylang-breaking-third-party-ajax-requests-on-admin-panels
|
|
* when options.data is an empty string (GET request with the method 'load')
|
|
* see: https://wordpress.org/support/topic/invalid-url-during-wordpress-new-dashboard-widget-operation
|
|
*
|
|
* @since 1.4
|
|
*/
|
|
public function admin_print_footer_scripts() {
|
|
global $post_ID, $tag_ID;
|
|
|
|
$params = array( 'pll_ajax_backend' => 1 );
|
|
if ( ! empty( $post_ID ) ) {
|
|
$params = array_merge( $params, array( 'pll_post_id' => (int) $post_ID ) );
|
|
}
|
|
|
|
if ( ! empty( $tag_ID ) ) {
|
|
$params = array_merge( $params, array( 'pll_term_id' => (int) $tag_ID ) );
|
|
}
|
|
|
|
$str = http_build_query( $params );
|
|
$arr = wp_json_encode( $params );
|
|
?>
|
|
<script type="text/javascript">
|
|
if (typeof jQuery != 'undefined') {
|
|
(function($){
|
|
$.ajaxPrefilter(function (options, originalOptions, jqXHR) {
|
|
if ( -1 != options.url.indexOf( ajaxurl ) || -1 != ajaxurl.indexOf( options.url ) ) {
|
|
if ( 'undefined' === typeof options.data ) {
|
|
options.data = ( 'get' === options.type.toLowerCase() ) ? '<?php echo $str; // phpcs:ignore WordPress.Security.EscapeOutput ?>' : <?php echo $arr; // phpcs:ignore WordPress.Security.EscapeOutput ?>;
|
|
} else {
|
|
if ( 'string' === typeof options.data ) {
|
|
if ( '' === options.data && 'get' === options.type.toLowerCase() ) {
|
|
options.url = options.url+'&<?php echo $str; // phpcs:ignore WordPress.Security.EscapeOutput ?>';
|
|
} else {
|
|
try {
|
|
var o = $.parseJSON(options.data);
|
|
o = $.extend(o, <?php echo $arr; // phpcs:ignore WordPress.Security.EscapeOutput ?>);
|
|
options.data = JSON.stringify(o);
|
|
}
|
|
catch(e) {
|
|
options.data = '<?php echo $str; // phpcs:ignore WordPress.Security.EscapeOutput ?>&'+options.data;
|
|
}
|
|
}
|
|
} else {
|
|
options.data = $.extend(options.data, <?php echo $arr; // phpcs:ignore WordPress.Security.EscapeOutput ?>);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
})(jQuery)
|
|
}
|
|
</script>
|
|
<?php
|
|
}
|
|
|
|
/**
|
|
* Sets the admin current language, used to filter the content
|
|
*
|
|
* @since 2.0
|
|
*/
|
|
public function set_current_language() {
|
|
$this->curlang = $this->filter_lang;
|
|
|
|
// Edit Post
|
|
if ( isset( $_REQUEST['pll_post_id'] ) && $lang = $this->model->post->get_language( (int) $_REQUEST['pll_post_id'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
$this->curlang = $lang;
|
|
} elseif ( 'post.php' === $GLOBALS['pagenow'] && isset( $_GET['post'] ) && $lang = $this->model->post->get_language( (int) $_GET['post'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
$this->curlang = $lang;
|
|
} elseif ( 'post-new.php' === $GLOBALS['pagenow'] && ( empty( $_GET['post_type'] ) || $this->model->is_translated_post_type( sanitize_key( $_GET['post_type'] ) ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
$this->curlang = empty( $_GET['new_lang'] ) ? $this->pref_lang : $this->model->get_language( sanitize_key( $_GET['new_lang'] ) ); // phpcs:ignore WordPress.Security.NonceVerification
|
|
}
|
|
|
|
// Edit Term
|
|
// FIXME 'edit-tags.php' for backward compatibility with WP < 4.5
|
|
elseif ( in_array( $GLOBALS['pagenow'], array( 'edit-tags.php', 'term.php' ) ) && isset( $_GET['tag_ID'] ) && $lang = $this->model->term->get_language( (int) $_GET['tag_ID'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
$this->curlang = $lang;
|
|
} elseif ( isset( $_REQUEST['pll_term_id'] ) && $lang = $this->model->term->get_language( (int) $_REQUEST['pll_term_id'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
$this->curlang = $lang;
|
|
} elseif ( 'edit-tags.php' === $GLOBALS['pagenow'] && isset( $_GET['taxonomy'] ) && $this->model->is_translated_taxonomy( sanitize_key( $_GET['taxonomy'] ) ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
if ( ! empty( $_GET['new_lang'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
$this->curlang = $this->model->get_language( sanitize_key( $_GET['new_lang'] ) ); // phpcs:ignore WordPress.Security.NonceVerification
|
|
} elseif ( empty( $this->curlang ) ) {
|
|
$this->curlang = $this->pref_lang;
|
|
}
|
|
}
|
|
|
|
// Ajax
|
|
if ( wp_doing_ajax() && ! empty( $_REQUEST['lang'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
$this->curlang = $this->model->get_language( sanitize_key( $_REQUEST['lang'] ) ); // phpcs:ignore WordPress.Security.NonceVerification
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Defines the backend language and the admin language filter based on user preferences
|
|
*
|
|
* @since 1.2.3
|
|
*/
|
|
public function init_user() {
|
|
// Language for admin language filter: may be empty
|
|
// $_GET['lang'] is numeric when editing a language, not when selecting a new language in the filter
|
|
// We intentionally don't use a nonce to update the language filter
|
|
if ( ! wp_doing_ajax() && ! empty( $_GET['lang'] ) && ! is_numeric( sanitize_key( $_GET['lang'] ) ) && current_user_can( 'edit_user', $user_id = get_current_user_id() ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
|
update_user_meta( $user_id, 'pll_filter_content', ( $lang = $this->model->get_language( sanitize_key( $_GET['lang'] ) ) ) ? $lang->slug : '' ); // phpcs:ignore WordPress.Security.NonceVerification
|
|
}
|
|
|
|
$this->filter_lang = $this->model->get_language( get_user_meta( get_current_user_id(), 'pll_filter_content', true ) );
|
|
|
|
// Set preferred language for use when saving posts and terms: must not be empty
|
|
$this->pref_lang = empty( $this->filter_lang ) ? $this->model->get_language( $this->options['default_lang'] ) : $this->filter_lang;
|
|
|
|
/**
|
|
* Filter the preferred language on admin side
|
|
* The preferred language is used for example to determine the language of a new post
|
|
*
|
|
* @since 1.2.3
|
|
*
|
|
* @param object $pref_lang preferred language
|
|
*/
|
|
$this->pref_lang = apply_filters( 'pll_admin_preferred_language', $this->pref_lang );
|
|
|
|
$this->set_current_language();
|
|
|
|
// Inform that the admin language has been set
|
|
// Only if the admin language is one of the Polylang defined language
|
|
if ( $curlang = $this->model->get_language( get_user_locale() ) ) {
|
|
/** This action is documented in frontend/choose-lang.php */
|
|
do_action( 'pll_language_defined', $curlang->slug, $curlang );
|
|
} else {
|
|
/** This action is documented in include/class-polylang.php */
|
|
do_action( 'pll_no_language_defined' ); // to load overridden textdomains
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Avoids parsing a tax query when all languages are requested
|
|
* Fixes https://wordpress.org/support/topic/notice-undefined-offset-0-in-wp-includesqueryphp-on-line-3877 introduced in WP 4.1
|
|
*
|
|
* @see the suggestion of @boonebgorges, https://core.trac.wordpress.org/ticket/31246
|
|
*
|
|
* @since 1.6.5
|
|
*
|
|
* @param array $qvars
|
|
* @return array
|
|
*/
|
|
public function request( $qvars ) {
|
|
if ( isset( $qvars['lang'] ) && 'all' === $qvars['lang'] ) {
|
|
unset( $qvars['lang'] );
|
|
}
|
|
|
|
return $qvars;
|
|
}
|
|
|
|
/**
|
|
* Adds the languages list in admin bar for the admin languages filter
|
|
*
|
|
* @since 0.9
|
|
*
|
|
* @param object $wp_admin_bar
|
|
*/
|
|
public function admin_bar_menu( $wp_admin_bar ) {
|
|
$all_item = (object) array(
|
|
'slug' => 'all',
|
|
'name' => __( 'Show all languages', 'polylang' ),
|
|
'flag' => '<span class="ab-icon"></span>',
|
|
);
|
|
|
|
$selected = empty( $this->filter_lang ) ? $all_item : $this->filter_lang;
|
|
|
|
$title = sprintf(
|
|
'<span class="ab-label"%1$s><span class="screen-reader-text">%2$s</span>%3$s</span>',
|
|
'all' === $selected->slug ? '' : sprintf( ' lang="%s"', esc_attr( $selected->get_locale( 'display' ) ) ),
|
|
__( 'Filters content by language', 'polylang' ),
|
|
esc_html( $selected->name )
|
|
);
|
|
|
|
/**
|
|
* Filters the admin languages filter submenu items
|
|
*
|
|
* @since 2.6
|
|
*
|
|
* @param array $items The admin languages filter submenu items.
|
|
*/
|
|
$items = apply_filters( 'pll_admin_languages_filter', array_merge( array( $all_item ), $this->model->get_languages_list() ) );
|
|
|
|
if ( ! empty( $items ) ) {
|
|
$wp_admin_bar->add_menu(
|
|
array(
|
|
'id' => 'languages',
|
|
'title' => $selected->flag . $title,
|
|
'href' => esc_url( add_query_arg( 'lang', $selected->slug, remove_query_arg( 'paged' ) ) ),
|
|
'meta' => array( 'title' => __( 'Filters content by language', 'polylang' ) ),
|
|
)
|
|
);
|
|
}
|
|
|
|
foreach ( $items as $lang ) {
|
|
if ( $selected->slug === $lang->slug ) {
|
|
continue;
|
|
}
|
|
|
|
$wp_admin_bar->add_menu(
|
|
array(
|
|
'parent' => 'languages',
|
|
'id' => $lang->slug,
|
|
'title' => $lang->flag . esc_html( $lang->name ),
|
|
'href' => esc_url( add_query_arg( 'lang', $lang->slug, remove_query_arg( 'paged' ) ) ),
|
|
'meta' => 'all' === $lang->slug ? array() : array( 'lang' => esc_attr( $lang->get_locale( 'display' ) ) ),
|
|
)
|
|
);
|
|
}
|
|
}
|
|
/**
|
|
* Instantiate PLL_Wizard_Pro class.
|
|
*
|
|
* @since 2.7
|
|
*/
|
|
public function instantiate_wizard_pro() {
|
|
$this->wizard_pro = new PLL_Wizard_Pro( $this );
|
|
}
|
|
}
|