mirror of
https://git.sindominio.net/estibadores/wordpress.git
synced 2024-11-23 11:01:07 +01:00
411 lines
12 KiB
PHP
411 lines
12 KiB
PHP
|
<?php
|
||
|
|
||
|
if ( ! class_exists( 'WP_List_Table' ) ) {
|
||
|
require_once ABSPATH . 'wp-admin/includes/class-wp-list-table.php'; // since WP 3.1
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* A class to create the strings translations table
|
||
|
* Thanks to Matt Van Andel ( http://www.mattvanandel.com ) for its plugin "Custom List Table Example" !
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*/
|
||
|
class PLL_Table_String extends WP_List_Table {
|
||
|
protected $languages, $strings, $groups, $selected_group;
|
||
|
|
||
|
/**
|
||
|
* Constructor
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*
|
||
|
* @param array $languages list of languages
|
||
|
*/
|
||
|
public function __construct( $languages ) {
|
||
|
parent::__construct(
|
||
|
array(
|
||
|
'plural' => 'Strings translations', // Do not translate ( used for css class )
|
||
|
'ajax' => false,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$this->languages = $languages;
|
||
|
$this->strings = PLL_Admin_Strings::get_strings();
|
||
|
$this->groups = array_unique( wp_list_pluck( $this->strings, 'context' ) );
|
||
|
|
||
|
$this->selected_group = -1;
|
||
|
|
||
|
if ( ! empty( $_GET['group'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
||
|
$group = sanitize_text_field( wp_unslash( $_GET['group'] ) ); // phpcs:ignore WordPress.Security.NonceVerification
|
||
|
if ( in_array( $group, $this->groups ) ) {
|
||
|
$this->selected_group = $group;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
add_action( 'mlang_action_string-translation', array( $this, 'save_translations' ) );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Displays the item information in a column ( default case )
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*
|
||
|
* @param array $item
|
||
|
* @param string $column_name
|
||
|
* @return string
|
||
|
*/
|
||
|
public function column_default( $item, $column_name ) {
|
||
|
return $item[ $column_name ];
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Displays the checkbox in first column
|
||
|
*
|
||
|
* @since 1.1
|
||
|
*
|
||
|
* @param array $item
|
||
|
* @return string
|
||
|
*/
|
||
|
public function column_cb( $item ) {
|
||
|
return sprintf(
|
||
|
'<label class="screen-reader-text" for="cb-select-%1$s">%2$s</label><input id="cb-select-%1$s" type="checkbox" name="strings[]" value="%1$s" %3$s />',
|
||
|
esc_attr( $item['row'] ),
|
||
|
/* translators: accessibility text, %s is a string potentially in any language */
|
||
|
sprintf( __( 'Select %s', 'polylang' ), format_to_edit( $item['string'] ) ),
|
||
|
empty( $item['icl'] ) ? 'disabled' : '' // Only strings registered with WPML API can be removed
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Displays the string to translate
|
||
|
*
|
||
|
* @since 1.0
|
||
|
*
|
||
|
* @param array $item
|
||
|
* @return string
|
||
|
*/
|
||
|
public function column_string( $item ) {
|
||
|
return format_to_edit( $item['string'] ); // Don't interpret special chars for the string column
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Displays the translations to edit
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*
|
||
|
* @param array $item
|
||
|
* @return string
|
||
|
*/
|
||
|
public function column_translations( $item ) {
|
||
|
$languages = array_combine( wp_list_pluck( $this->languages, 'slug' ), wp_list_pluck( $this->languages, 'name' ) );
|
||
|
$out = '';
|
||
|
|
||
|
foreach ( $item['translations'] as $key => $translation ) {
|
||
|
$input_type = $item['multiline'] ?
|
||
|
'<textarea name="translation[%1$s][%2$s]" id="%1$s-%2$s">%4$s</textarea>' :
|
||
|
'<input type="text" name="translation[%1$s][%2$s]" id="%1$s-%2$s" value="%4$s" />';
|
||
|
$out .= sprintf(
|
||
|
'<div class="translation"><label for="%1$s-%2$s">%3$s</label>' . $input_type . '</div>' . "\n",
|
||
|
esc_attr( $key ),
|
||
|
esc_attr( $item['row'] ),
|
||
|
esc_html( $languages[ $key ] ),
|
||
|
format_to_edit( $translation ) // Don't interpret special chars
|
||
|
);
|
||
|
}
|
||
|
|
||
|
return $out;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the list of columns
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*
|
||
|
* @return array the list of column titles
|
||
|
*/
|
||
|
public function get_columns() {
|
||
|
return array(
|
||
|
'cb' => '<input type="checkbox" />', // Checkbox
|
||
|
'string' => esc_html__( 'String', 'polylang' ),
|
||
|
'name' => esc_html__( 'Name', 'polylang' ),
|
||
|
'context' => esc_html__( 'Group', 'polylang' ),
|
||
|
'translations' => esc_html__( 'Translations', 'polylang' ),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the list of sortable columns
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
public function get_sortable_columns() {
|
||
|
return array(
|
||
|
'string' => array( 'string', false ),
|
||
|
'name' => array( 'name', false ),
|
||
|
'context' => array( 'context', false ),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the name of the default primary column.
|
||
|
*
|
||
|
* @since 2.1
|
||
|
*
|
||
|
* @return string Name of the default primary column, in this case, 'string'.
|
||
|
*/
|
||
|
protected function get_default_primary_column_name() {
|
||
|
return 'string';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Search for a string in translations. Case insensitive.
|
||
|
*
|
||
|
* @since 2.6
|
||
|
*
|
||
|
* @param array $mos An array of PLL_MO objects
|
||
|
* @param string $s Searched string
|
||
|
* @return array Found strings
|
||
|
*/
|
||
|
protected function search_in_translations( $mos, $s ) {
|
||
|
$founds = array();
|
||
|
|
||
|
foreach ( $mos as $mo ) {
|
||
|
foreach ( wp_list_pluck( $mo->entries, 'translations' ) as $string => $translation ) {
|
||
|
if ( false !== stripos( $translation[0], $s ) ) {
|
||
|
$founds[] = $string;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return array_unique( $founds );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sort items
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*
|
||
|
* @param object $a The first object to compare
|
||
|
* @param object $b The second object to compare
|
||
|
* @return int -1 or 1 if $a is considered to be respectively less than or greater than $b.
|
||
|
*/
|
||
|
protected function usort_reorder( $a, $b ) {
|
||
|
if ( ! empty( $_GET['orderby'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification
|
||
|
$orderby = sanitize_key( $_GET['orderby'] ); // phpcs:ignore WordPress.Security.NonceVerification
|
||
|
if ( isset( $a[ $orderby ], $b[ $orderby ] ) ) {
|
||
|
$result = strcmp( $a[ $orderby ], $b[ $orderby ] ); // Determine sort order
|
||
|
return ( empty( $_GET['order'] ) || 'asc' === $_GET['order'] ) ? $result : -$result; // phpcs:ignore WordPress.Security.NonceVerification
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Prepares the list of items for displaying
|
||
|
*
|
||
|
* @since 0.6
|
||
|
*/
|
||
|
public function prepare_items() {
|
||
|
// Is admin language filter active?
|
||
|
if ( $lg = get_user_meta( get_current_user_id(), 'pll_filter_content', true ) ) {
|
||
|
$languages = wp_list_filter( $this->languages, array( 'slug' => $lg ) );
|
||
|
} else {
|
||
|
$languages = $this->languages;
|
||
|
}
|
||
|
|
||
|
// Load translations
|
||
|
$mo = array();
|
||
|
foreach ( $languages as $language ) {
|
||
|
$mo[ $language->slug ] = new PLL_MO();
|
||
|
$mo[ $language->slug ]->import_from_db( $language );
|
||
|
}
|
||
|
|
||
|
$data = $this->strings;
|
||
|
|
||
|
// Filter by selected group
|
||
|
if ( -1 !== $this->selected_group ) {
|
||
|
$data = wp_list_filter( $data, array( 'context' => $this->selected_group ) );
|
||
|
}
|
||
|
|
||
|
// Filter by searched string
|
||
|
$s = empty( $_GET['s'] ) ? '' : wp_unslash( $_GET['s'] ); // phpcs:ignore WordPress.Security.NonceVerification, WordPress.Security.ValidatedSanitizedInput
|
||
|
|
||
|
if ( ! empty( $s ) ) {
|
||
|
// Search in translations
|
||
|
$in_translations = $this->search_in_translations( $mo, $s );
|
||
|
|
||
|
foreach ( $data as $key => $row ) {
|
||
|
if ( stripos( $row['name'], $s ) === false && stripos( $row['string'], $s ) === false && ! in_array( $row['string'], $in_translations ) ) {
|
||
|
unset( $data[ $key ] );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Sorting
|
||
|
uasort( $data, array( $this, 'usort_reorder' ) );
|
||
|
|
||
|
// Paging
|
||
|
$per_page = $this->get_items_per_page( 'pll_strings_per_page' );
|
||
|
$this->_column_headers = array( $this->get_columns(), array(), $this->get_sortable_columns() );
|
||
|
|
||
|
$total_items = count( $data );
|
||
|
$this->items = array_slice( $data, ( $this->get_pagenum() - 1 ) * $per_page, $per_page, true );
|
||
|
|
||
|
$this->set_pagination_args(
|
||
|
array(
|
||
|
'total_items' => $total_items,
|
||
|
'per_page' => $per_page,
|
||
|
'total_pages' => ceil( $total_items / $per_page ),
|
||
|
)
|
||
|
);
|
||
|
|
||
|
// Translate strings
|
||
|
// Kept for the end as it is a slow process
|
||
|
foreach ( $languages as $language ) {
|
||
|
foreach ( $this->items as $key => $row ) {
|
||
|
$this->items[ $key ]['translations'][ $language->slug ] = $mo[ $language->slug ]->translate( $row['string'] );
|
||
|
$this->items[ $key ]['row'] = $key; // Store the row number for convenience
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the list of possible bulk actions
|
||
|
*
|
||
|
* @since 1.1
|
||
|
*
|
||
|
* @return array
|
||
|
*/
|
||
|
public function get_bulk_actions() {
|
||
|
return array( 'delete' => __( 'Delete', 'polylang' ) );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the current action selected from the bulk actions dropdown.
|
||
|
* overrides parent function to avoid submit button to trigger bulk actions
|
||
|
*
|
||
|
* @since 1.8
|
||
|
*
|
||
|
* @return string|false The action name or False if no action was selected
|
||
|
*/
|
||
|
public function current_action() {
|
||
|
return empty( $_POST['submit'] ) ? parent::current_action() : false; // phpcs:ignore WordPress.Security.NonceVerification
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Displays the dropdown list to filter strings per group
|
||
|
*
|
||
|
* @since 1.1
|
||
|
*
|
||
|
* @param string $which only 'top' is supported
|
||
|
*/
|
||
|
public function extra_tablenav( $which ) {
|
||
|
if ( 'top' !== $which ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
echo '<div class="alignleft actions">';
|
||
|
printf(
|
||
|
'<label class="screen-reader-text" for="select-group" >%s</label>',
|
||
|
/* translators: accessibility text */
|
||
|
esc_html__( 'Filter by group', 'polylang' )
|
||
|
);
|
||
|
echo '<select id="select-group" name="group">' . "\n";
|
||
|
printf(
|
||
|
'<option value="-1"%s>%s</option>' . "\n",
|
||
|
selected( $this->group_selected, -1, false ),
|
||
|
esc_html__( 'View all groups', 'polylang' )
|
||
|
);
|
||
|
|
||
|
foreach ( $this->groups as $group ) {
|
||
|
printf(
|
||
|
'<option value="%s"%s>%s</option>' . "\n",
|
||
|
esc_attr( urlencode( $group ) ),
|
||
|
selected( $this->selected_group, $group, false ),
|
||
|
esc_html( $group )
|
||
|
);
|
||
|
}
|
||
|
echo '</select>' . "\n";
|
||
|
|
||
|
submit_button( __( 'Filter', 'polylang' ), 'button', 'filter_action', false, array( 'id' => 'post-query-submit' ) );
|
||
|
echo '</div>';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Saves the strings translations in DB
|
||
|
* Optionaly clean the DB
|
||
|
*
|
||
|
* @since 1.9
|
||
|
*/
|
||
|
public function save_translations() {
|
||
|
check_admin_referer( 'string-translation', '_wpnonce_string-translation' );
|
||
|
|
||
|
if ( ! empty( $_POST['submit'] ) ) {
|
||
|
foreach ( $this->languages as $language ) {
|
||
|
if ( empty( $_POST['translation'][ $language->slug ] ) ) { // In case the language filter is active ( thanks to John P. Bloch )
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$mo = new PLL_MO();
|
||
|
$mo->import_from_db( $language );
|
||
|
|
||
|
$translations = array_map( 'trim', wp_unslash( $_POST['translation'][ $language->slug ] ) ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized
|
||
|
foreach ( $translations as $key => $translation ) {
|
||
|
/**
|
||
|
* Filter the string translation before it is saved in DB
|
||
|
* Allows to sanitize strings registered with pll_register_string
|
||
|
*
|
||
|
* @since 1.6
|
||
|
* @since 2.7 The translation passed to the filter is unslashed.
|
||
|
*
|
||
|
* @param string $translation The string translation.
|
||
|
* @param string $name The name as defined in pll_register_string.
|
||
|
* @param string $context The context as defined in pll_register_string.
|
||
|
*/
|
||
|
$translation = apply_filters( 'pll_sanitize_string_translation', $translation, $this->strings[ $key ]['name'], $this->strings[ $key ]['context'] );
|
||
|
$mo->add_entry( $mo->make_entry( $this->strings[ $key ]['string'], $translation ) );
|
||
|
}
|
||
|
|
||
|
// Clean database ( removes all strings which were registered some day but are no more )
|
||
|
if ( ! empty( $_POST['clean'] ) ) {
|
||
|
$new_mo = new PLL_MO();
|
||
|
|
||
|
foreach ( $this->strings as $string ) {
|
||
|
$new_mo->add_entry( $mo->make_entry( $string['string'], $mo->translate( $string['string'] ) ) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
isset( $new_mo ) ? $new_mo->export_to_db( $language ) : $mo->export_to_db( $language );
|
||
|
}
|
||
|
|
||
|
add_settings_error( 'general', 'pll_strings_translations_updated', __( 'Translations updated.', 'polylang' ), 'updated' );
|
||
|
|
||
|
/**
|
||
|
* Fires after the strings translations are saved in DB
|
||
|
*
|
||
|
* @since 1.2
|
||
|
*/
|
||
|
do_action( 'pll_save_strings_translations' );
|
||
|
}
|
||
|
|
||
|
// Unregisters strings registered through WPML API
|
||
|
if ( $this->current_action() === 'delete' && ! empty( $_POST['strings'] ) && function_exists( 'icl_unregister_string' ) ) {
|
||
|
foreach ( array_map( 'sanitize_key', $_POST['strings'] ) as $key ) {
|
||
|
icl_unregister_string( $this->strings[ $key ]['context'], $this->strings[ $key ]['name'] );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// To refresh the page ( possible thanks to the $_GET['noheader']=true )
|
||
|
$args = array_intersect_key( $_REQUEST, array_flip( array( 's', 'paged', 'group' ) ) );
|
||
|
if ( ! empty( $_GET['paged'] ) && ! empty( $_POST['submit'] ) ) {
|
||
|
$args['paged'] = (int) $_GET['paged']; // Don't rely on $_REQUEST['paged'] or $_POST['paged']. See #14
|
||
|
}
|
||
|
if ( ! empty( $args['s'] ) ) {
|
||
|
$args['s'] = urlencode( $args['s'] ); // Searched string needs to be encoded as it comes from $_POST
|
||
|
}
|
||
|
PLL_Settings::redirect( $args );
|
||
|
}
|
||
|
}
|