'."\n";
}
/**
* Translates term parent if exists when using "Add new" ( translation )
*
* @since 0.7
*
* @param string html markup for dropdown list of categories
* @return string modified html
*/
public function wp_dropdown_cats( $output ) {
if ( isset( $_GET['taxonomy'], $_GET['from_tag'], $_GET['new_lang'] ) && taxonomy_exists( $_GET['taxonomy'] ) && $id = get_term( (int) $_GET['from_tag'], $_GET['taxonomy'] )->parent ) {
$lang = $this->model->get_language( $_GET['new_lang'] );
if ( $parent = $this->model->term->get_translation( $id, $lang ) ) {
return str_replace( '"'.$parent.'"', '"'.$parent.'" selected="selected"', $output );
}
}
return $output;
}
/**
* stores the current post_id when bulk editing posts for use in save_language and pre_term_slug
*
* @since 1.7
*
* @param int $post_id
*/
public function pre_post_update( $post_id ) {
if ( isset( $_GET['bulk_edit'] ) ) {
$this->post_id = $post_id;
}
}
/**
* Allows to set a language by default for terms if it has no language yet
*
* @since 1.5.4
*
* @param int $term_id
* @param string $taxonomy
*/
protected function set_default_language( $term_id, $taxonomy ) {
if ( ! $this->model->term->get_language( $term_id ) ) {
// Sets language from term parent if exists thanks to Scott Kingsley Clark
if ( ( $term = get_term( $term_id, $taxonomy ) ) && ! empty( $term->parent ) && $parent_lang = $this->model->term->get_language( $term->parent ) ) {
$this->model->term->set_language( $term_id, $parent_lang );
}
else {
$this->model->term->set_language( $term_id, $this->pref_lang );
}
}
}
/**
* Saves language
*
* @since 1.5
*
* @param int $term_id
* @param string $taxonomy
*/
protected function save_language( $term_id, $taxonomy ) {
global $wpdb;
// Security checks are necessary to accept language modifications
// as 'wp_update_term' can be called from outside WP admin
// Edit tags
if ( isset( $_POST['term_lang_choice'] ) ) {
if ( 'add-' . $taxonomy == $_POST['action'] ) {
check_ajax_referer( $_POST['action'], '_ajax_nonce-add-' . $taxonomy ); // category metabox
}
else {
check_admin_referer( 'pll_language', '_pll_nonce' ); // edit tags or tags metabox
}
$this->model->term->set_language( $term_id, $this->model->get_language( $_POST['term_lang_choice'] ) );
}
// *Post* bulk edit, in case a new term is created
elseif ( isset( $_GET['bulk_edit'], $_GET['inline_lang_choice'] ) ) {
check_admin_referer( 'bulk-posts' );
// Bulk edit does not modify the language
// So we possibly create a tag in several languages
if ( -1 == $_GET['inline_lang_choice'] ) {
// The language of the current term is set a according to the language of the current post
$this->model->term->set_language( $term_id, $this->model->post->get_language( $this->post_id ) );
$term = get_term( $term_id, $taxonomy );
// Get all terms with the same name
// FIXME backward compatibility WP < 4.2
// No WP function to get all terms with the exact same name so let's use a custom query
// $terms = get_terms( $taxonomy, array( 'name' => $term->name, 'hide_empty' => false, 'fields' => 'ids' ) ); should be OK in 4.2
// I may need to rework the loop below
$terms = $wpdb->get_results( $wpdb->prepare( "
SELECT t.term_id FROM $wpdb->terms AS t
INNER JOIN $wpdb->term_taxonomy AS tt ON t.term_id = tt.term_id
WHERE tt.taxonomy = %s AND t.name = %s",
$taxonomy, $term->name
) );
// If we have several terms with the same name, they are translations of each other
if ( count( $terms ) > 1 ) {
foreach ( $terms as $term ) {
$translations[ $this->model->term->get_language( $term->term_id )->slug ] = $term->term_id;
}
$this->model->term->save_translations( $term_id, $translations );
}
}
else {
$this->model->term->set_language( $term_id, $this->model->get_language( $_GET['inline_lang_choice'] ) );
}
}
// Quick edit
elseif ( isset( $_POST['inline_lang_choice'] ) ) {
check_ajax_referer(
isset( $_POST['action'] ) && 'inline-save' == $_POST['action'] ? 'inlineeditnonce' : 'taxinlineeditnonce', // Post quick edit or tag quick edit ?
'_inline_edit'
);
$old_lang = $this->model->term->get_language( $term_id ); // Stores the old language
$lang = $this->model->get_language( $_POST['inline_lang_choice'] ); // New language
$translations = $this->model->term->get_translations( $term_id );
// Checks if the new language already exists in the translation group
if ( $old_lang && $old_lang->slug != $lang->slug ) {
if ( array_key_exists( $lang->slug, $translations ) ) {
$this->model->term->delete_translation( $term_id );
}
elseif ( array_key_exists( $old_lang->slug, $translations ) ) {
unset( $translations[ $old_lang->slug ] );
$this->model->term->save_translations( $term_id, $translations );
}
}
$this->model->term->set_language( $term_id, $lang ); // Set new language
}
// Edit post
elseif ( isset( $_POST['post_lang_choice'] ) ) { // FIXME should be useless now
check_admin_referer( 'pll_language', '_pll_nonce' );
$this->model->term->set_language( $term_id, $this->model->get_language( $_POST['post_lang_choice'] ) );
}
else {
$this->set_default_language( $term_id, $taxonomy );
}
}
/**
* Save translations from our form
*
* @since 1.5
*
* @param int $term_id
* @return array
*/
protected function save_translations( $term_id ) {
// Security check as 'wp_update_term' can be called from outside WP admin
check_admin_referer( 'pll_language', '_pll_nonce' );
// Save translations after checking the translated term is in the right language ( as well as cast id to int )
foreach ( $_POST['term_tr_lang'] as $lang => $tr_id ) {
$tr_lang = $this->model->term->get_language( (int) $tr_id );
$translations[ $lang ] = $tr_lang && $tr_lang->slug == $lang ? (int) $tr_id : 0;
}
$this->model->term->save_translations( $term_id, $translations );
return $translations;
}
/**
* Called when a category or post tag is created or edited
* Saves language and translations
*
* @since 0.1
*
* @param int $term_id
* @param int $tt_id term taxononomy id
* @param string $taxonomy
*/
public function save_term( $term_id, $tt_id, $taxonomy ) {
// Does nothing except on taxonomies which are filterable
if ( ! $this->model->is_translated_taxonomy( $taxonomy ) ) {
return;
}
// Capability check
// As 'wp_update_term' can be called from outside WP admin
// 2nd test for creating tags when creating / editing a post
$tax = get_taxonomy( $taxonomy );
if ( current_user_can( $tax->cap->edit_terms ) || ( isset( $_POST['tax_input'][ $taxonomy ] ) && current_user_can( $tax->cap->assign_terms ) ) ) {
$this->save_language( $term_id, $taxonomy );
if ( isset( $_POST['term_tr_lang'] ) ) {
$translations = $this->save_translations( $term_id );
}
/**
* Fires after the term language and translations are saved
*
* @since 1.2
*
* @param int $term_id term id
* @param string $taxonomy taxonomy name
* @param array $translations the list of translations term ids
*/
do_action( 'pll_save_term', $term_id, $taxonomy, empty( $translations ) ? $this->model->term->get_translations( $term_id ) : $translations );
}
// Attempts to set a default language even if no capability
else {
$this->set_default_language( $term_id, $taxonomy );
}
}
/**
* Stores the term name for use in pre_term_slug
*
* @since 0.9.5
*
* @param string $name term name
* @return string unmodified term name
*/
public function pre_term_name( $name ) {
return $this->pre_term_name = $name;
}
/**
* Creates the term slug in case the term already exists in another language
*
* @since 0.9.5
*
* @param string $slug
* @param string $taxonomy
* @return string
*/
public function pre_term_slug( $slug, $taxonomy ) {
$name = sanitize_title( $this->pre_term_name );
// If the new term has the same name as a language, we *need* to differentiate the term
// See http://core.trac.wordpress.org/ticket/23199
// Backward compatibility with WP < 4.1
if ( version_compare( $GLOBALS['wp_version'], '4.1', '<' ) && term_exists( $name, 'language' ) && ! term_exists( $name, $taxonomy ) && ( ! $slug || $slug == $name ) ) {
$slug = $name . '-' . $taxonomy; // A convenient slug which may be modified later by the user
}
// If the term already exists in another language
if ( ! $slug && $this->model->is_translated_taxonomy( $taxonomy ) && term_exists( $name, $taxonomy ) ) {
if ( isset( $_POST['term_lang_choice'] ) ) {
$slug = $name . '-' . $this->model->get_language( $_POST['term_lang_choice'] )->slug;
}
elseif ( isset( $_POST['inline_lang_choice'] ) ) {
$slug = $name . '-' . $this->model->get_language( $_POST['inline_lang_choice'] )->slug;
}
// *Post* bulk edit, in case a new term is created
elseif ( isset( $_GET['bulk_edit'], $_GET['inline_lang_choice'] ) ) {
// Bulk edit does not modify the language
if ( -1 == $_GET['inline_lang_choice'] ) {
$slug = $name . '-' . $this->model->post->get_language( $this->post_id )->slug;
}
else {
$slug = $name . '-' . $this->model->get_language( $_GET['inline_lang_choice'] )->slug;
}
}
}
return $slug;
}
/**
* Called when a category or post tag is deleted
* Deletes language and translations
*
* @since 0.1
*
* @param int $term_id
*/
public function delete_term( $term_id ) {
$this->model->term->delete_translation( $term_id );
$this->model->term->delete_language( $term_id );
}
/**
* Ajax response for edit term form
*
* @since 0.2
*/
public function term_lang_choice() {
check_ajax_referer( 'pll_language', '_pll_nonce' );
$lang = $this->model->get_language( $_POST['lang'] );
$term_id = isset( $_POST['term_id'] ) ? (int) $_POST['term_id'] : null;
$taxonomy = $_POST['taxonomy'];
$post_type = $_POST['post_type'];
if ( ! post_type_exists( $post_type ) || ! taxonomy_exists( $taxonomy ) ) {
die( 0 );
}
ob_start();
if ( $lang ) {
include( PLL_ADMIN_INC.'/view-translations-term.php' );
}
$x = new WP_Ajax_Response( array( 'what' => 'translations', 'data' => ob_get_contents() ) );
ob_end_clean();
// Parent dropdown list ( only for hierarchical taxonomies )
// $args copied from edit_tags.php except echo
if ( is_taxonomy_hierarchical( $taxonomy ) ) {
$args = array(
'hide_empty' => 0,
'hide_if_empty' => false,
'taxonomy' => $taxonomy,
'name' => 'parent',
'orderby' => 'name',
'hierarchical' => true,
'show_option_none' => __( 'None' ),
'echo' => 0,
);
$x->Add( array( 'what' => 'parent', 'data' => wp_dropdown_categories( $args ) ) );
}
// Tag cloud
// Tests copied from edit_tags.php
else {
$tax = get_taxonomy( $taxonomy );
if ( ! is_null( $tax->labels->popular_items ) ) {
$args = array( 'taxonomy' => $taxonomy, 'echo' => false );
if ( current_user_can( $tax->cap->edit_terms ) ) {
$args = array_merge( $args, array( 'link' => 'edit' ) );
}
if ( $tag_cloud = wp_tag_cloud( $args ) ) {
$html = sprintf( '
%1$s
%2$s
', esc_html( $tax->labels->popular_items ), $tag_cloud );
$x->Add( array( 'what' => 'tag_cloud', 'data' => $html ) );
}
}
}
// Flag
$x->Add( array( 'what' => 'flag', 'data' => empty( $lang->flag ) ? esc_html( $lang->slug ) : $lang->flag ) );
$x->send();
}
/**
* Ajax response for input in translation autocomplete input box
*
* @since 1.5
*/
public function ajax_terms_not_translated() {
check_ajax_referer( 'pll_language', '_pll_nonce' );
$s = wp_unslash( $_GET['term'] );
$post_type = $_GET['post_type'];
$taxonomy = $_GET['taxonomy'];
if ( ! post_type_exists( $post_type ) || ! taxonomy_exists( $taxonomy ) ) {
die( 0 );
}
$term_language = $this->model->get_language( $_GET['term_language'] );
$translation_language = $this->model->get_language( $_GET['translation_language'] );
$return = array();
// It is more efficient to use one common query for all languages as soon as there are more than 2
foreach ( get_terms( $taxonomy, 'hide_empty=0&lang=0&name__like=' . $s ) as $term ) {
$lang = $this->model->term->get_language( $term->term_id );
if ( $lang && $lang->slug == $translation_language->slug && ! $this->model->term->get_translation( $term->term_id, $term_language ) ) {
$return[] = array(
'id' => $term->term_id,
'value' => $term->name,
'link' => $this->links->edit_term_translation_link( $term->term_id, $taxonomy, $post_type ),
);
}
}
// Add current translation in list
// Not in add term for as term_id is not set
if ( 'undefined' !== $_GET['term_id'] && $term_id = $this->model->term->get_translation( (int) $_GET['term_id'], $translation_language ) ) {
$term = get_term( $term_id, $taxonomy );
array_unshift( $return, array(
'id' => $term_id,
'value' => $term->name,
'link' => $this->links->edit_term_translation_link( $term->term_id, $taxonomy, $post_type ),
) );
}
wp_die( json_encode( $return ) );
}
/**
* Get the language(s) to filter get_terms
*
* @since 1.7.6
*
* @param array $taxonomies queried taxonomies
* @param array $args get_terms arguments
* @return object|string|bool the language(s) to use in the filter, false otherwise
*/
protected function get_queried_language( $taxonomies, $args ) {
// Does nothing except on taxonomies which are filterable
if ( ! $this->model->is_translated_taxonomy( $taxonomies ) ) {
return false;
}
// If get_terms is queried with a 'lang' parameter
if ( isset( $args['lang'] ) ) {
return $args['lang'];
}
// On tags page, everything should be filtered according to the admin language filter except the parent dropdown
if ( 'edit-tags.php' === $GLOBALS['pagenow'] && empty( $args['class'] ) ) {
return $this->filter_lang;
}
return $this->curlang;
}
/**
* Adds language dependent cache domain when querying terms
* Useful as the 'lang' parameter is not included in cache key by WordPress
*
* @since 1.3
*
* @param array $args
* @param array $taxonomies
* @return array modified arguments
*/
public function get_terms_args( $args, $taxonomies ) {
// don't break _get_term_hierarchy()
if ( 'all' === $args['get'] && 'id' === $args['orderby'] && 'id=>parent' === $args['fields'] ) {
$args['lang'] = '';
}
if ( $lang = $this->get_queried_language( $taxonomies, $args ) ) {
$key = '_' . ( is_array( $lang ) ? implode( ',', $lang ) : $this->model->get_language( $lang )->slug );
$args['cache_domain'] = empty( $args['cache_domain'] ) ? 'pll' . $key : $args['cache_domain'] . $key;
}
return $args;
}
/**
* Filters categories and post tags by language(s) when needed on admin side
*
* @since 0.5
*
* @param array $clauses list of sql clauses
* @param array $taxonomies list of taxonomies
* @param array $args get_terms arguments
* @return array modified sql clauses
*/
public function terms_clauses( $clauses, $taxonomies, $args ) {
$lang = $this->get_queried_language( $taxonomies, $args );
return ! empty( $lang ) ? $this->model->terms_clauses( $clauses, $lang ) : $clauses; // adds our clauses to filter by current language
}
/**
* Hack to avoid displaying delete link for the default category in all languages
* Also returns the default category in the right language when called from wp_delete_term
*
* @since 1.2
*
* @param int $value
* @return int
*/
public function option_default_category( $value ) {
$traces = debug_backtrace();
$n = version_compare( PHP_VERSION, '7', '>=' ) ? 3 : 4; // PHP 7 does not include call_user_func_array
if ( isset( $traces[ $n ] ) ) {
// FIXME 'column_name' for backward compatibility with WP < 4.3
if ( in_array( $traces[ $n ]['function'], array( 'column_cb', 'column_name', 'handle_row_actions' ) ) && in_array( $traces[ $n ]['args'][0]->term_id, $this->model->term->get_translations( $value ) ) ) {
return $traces[ $n ]['args'][0]->term_id;
}
if ( 'wp_delete_term' == $traces[ $n ]['function'] ) {
return $this->model->term->get( $value, $this->model->term->get_language( $traces[ $n ]['args'][0] ) );
}
}
// Filters the default category in note below the category list table and in settings->writing dropdown
elseif ( isset( $traces[ $n - 1 ]['file'] ) && ( false !== stripos( $traces[ $n - 1 ]['file'], 'edit-tags.php' ) || false !== stripos( $traces[ $n - 1 ]['file'], 'options-writing.php' ) ) ) {
return $this->model->term->get( $value, $this->pref_lang );
}
return $value;
}
/**
* Checks if the new default category is translated in all languages
* If not, create the translations
*
* @since 1.7
*
* @param int $old_value
* @param int $value
*/
public function update_option_default_category( $old_value, $value ) {
$default_cat_lang = $this->model->term->get_language( $value );
// Assign a default language to default category
if ( ! $default_cat_lang ) {
$default_cat_lang = $this->model->get_language( $this->options['default_lang'] );
$this->model->term->set_language( (int) $value, $default_cat_lang );
}
foreach ( $this->model->get_languages_list() as $language ) {
if ( $language->slug != $default_cat_lang->slug && ! $this->model->term->get_translation( $value, $language ) ) {
$this->model->create_default_category( $language );
}
}
}
/**
* Updates the translations term ids when splitting a shared term
* Splits translations if these are shared terms too
*
* @since 1.7
*
* @param int $term_id shared term_id
* @param int $new_term_id
* @param int $term_taxonomy_id
* @param string $taxonomy
*/
public function split_shared_term( $term_id, $new_term_id, $term_taxonomy_id, $taxonomy ) {
if ( ! $this->model->is_translated_taxonomy( $taxonomy ) ) {
return;
}
// Avoid recursion
static $avoid_recursion = false;
if ( $avoid_recursion ) {
return;
}
$avoid_recursion = true;
$lang = $this->model->term->get_language( $term_id );
foreach ( $this->model->term->get_translations( $term_id ) as $key => $tr_id ) {
if ( $lang->slug == $key ) {
$translations[ $key ] = $new_term_id;
}
else {
$tr_term = get_term( $tr_id, $taxonomy );
$translations[ $key ] = _split_shared_term( $tr_id, $tr_term->term_taxonomy_id );
// Hack translation ids sent by the form to avoid overwrite in PLL_Admin_Filters_Term::save_translations
if ( isset( $_POST['term_tr_lang'][ $key ] ) && $_POST['term_tr_lang'][ $key ] == $tr_id ) {
$_POST['term_tr_lang'][ $key ] = $translations[ $key ];
}
}
$this->model->term->set_language( $translations[ $key ], $key );
}
$this->model->term->save_translations( $new_term_id, $translations );
$avoid_recursion = false;
}
}