gptranslate
Last commit date
assets
10 months ago
flags
11 months ago
language
11 months ago
gptranslate.php
11 months ago
multilang-routing.php
10 months ago
readme.txt
10 months ago
serverside-translations.php
11 months ago
settings.php
11 months ago
simplehtmldom.php
11 months ago
uninstall.php
11 months ago
gptranslate.php
1714 lines
| 1 | <?php |
| 2 | /* |
| 3 | Plugin Name: GPTranslate |
| 4 | Plugin URI: https://storejextensions.org/extensions/gptranslate.html |
| 5 | Description: GPTranslate for Wordpress is the revolutionary multilanguage solution to automatically translate your Wordpress website thanks to the power of Artificial Intelligence like ChatGPT, Deepseek, Gemini and more. ⚠️GPTranslate FREE Mode active |
| 6 | Author: JExtensions Store |
| 7 | Version: 2.13 |
| 8 | Author URI: https://storejextensions.org |
| 9 | License: GPLv2 or later |
| 10 | License URI: https://www.gnu.org/licenses/gpl-2.0.html |
| 11 | */ |
| 12 | |
| 13 | if (!defined('ABSPATH')) exit; |
| 14 | |
| 15 | class GPTranslate { |
| 16 | private static $instance = null; |
| 17 | private $table_name; |
| 18 | private $version; |
| 19 | |
| 20 | private function isSelected($current, $value) { |
| 21 | return selected($current, $value, false); |
| 22 | } |
| 23 | |
| 24 | public static $pluginVersion = '2.13'; |
| 25 | |
| 26 | /** |
| 27 | * Class constructor and settings inizializer with register_setting |
| 28 | * |
| 29 | * @access public |
| 30 | */ |
| 31 | public function __construct() { |
| 32 | global $wpdb; |
| 33 | $this->table_name = $wpdb->prefix . 'gptranslate'; |
| 34 | |
| 35 | $this->version = '2.13'; |
| 36 | |
| 37 | $settings = get_option ( 'gptranslate_options', [ ] ); |
| 38 | |
| 39 | // Include various functions like multilanguage URLs, hreflang tag, HTML lang attribute rewriting |
| 40 | require_once plugin_dir_path(__FILE__) . 'multilang-routing.php'; |
| 41 | |
| 42 | if ( $settings ['serverside_translations'] == 1 ) { |
| 43 | require_once plugin_dir_path(__FILE__) . 'serverside-translations.php'; |
| 44 | } |
| 45 | |
| 46 | register_activation_hook ( __FILE__, [ |
| 47 | $this, |
| 48 | 'activate_plugin' |
| 49 | ] ); |
| 50 | |
| 51 | add_action ( 'admin_init', function () { |
| 52 | // Disable WordPress emoji script and styles |
| 53 | remove_action('wp_head', 'print_emoji_detection_script', 7); |
| 54 | remove_action('admin_print_scripts', 'print_emoji_detection_script'); |
| 55 | remove_action('wp_print_styles', 'print_emoji_styles'); |
| 56 | remove_action('admin_print_styles', 'print_emoji_styles'); |
| 57 | remove_filter('the_content_feed', 'wp_staticize_emoji'); |
| 58 | remove_filter('comment_text_rss', 'wp_staticize_emoji'); |
| 59 | remove_filter('wp_mail', 'wp_staticize_emoji_for_email'); |
| 60 | |
| 61 | function gptranslate_sanitize_options( $options ) { |
| 62 | $clean = []; |
| 63 | $clean = $options; |
| 64 | return $clean; |
| 65 | } |
| 66 | register_setting ( 'gptranslate_settings', 'gptranslate_options', [ |
| 67 | 'sanitize_callback' => 'gptranslate_sanitize_options' |
| 68 | ]); |
| 69 | |
| 70 | // Register record deletion |
| 71 | $page = sanitize_key($_GET['page'] ?? ''); |
| 72 | $action = sanitize_key($_GET['action'] ?? ''); |
| 73 | $nonce = isset($_GET['_gptranslate_nonce']) ? wp_unslash($_GET['_gptranslate_nonce']) : ''; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized |
| 74 | $translation_id = isset($_GET['translation_id']) ? (int) $_GET['translation_id'] : 0; // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 75 | |
| 76 | if ($page === 'gptranslate' && |
| 77 | $action === 'delete_translation' && |
| 78 | $translation_id && |
| 79 | wp_verify_nonce($nonce, 'gptranslate_delete_' . $translation_id) |
| 80 | ) { |
| 81 | $this->gptranslate_handle_deletion($translation_id); |
| 82 | wp_redirect(admin__rewrit('admin.php?page=gptranslate&deleted=1')); |
| 83 | exit; |
| 84 | } |
| 85 | |
| 86 | if (isset($_GET['action']) && sanitize_key($_GET['action']) == 'toggle_published') { |
| 87 | // Toggle published state |
| 88 | $id = isset($_GET['translation_id']) ? (int) $_GET['translation_id'] : 0; |
| 89 | |
| 90 | if (!$id || !isset($_GET['_gptranslate_nonce']) || !wp_verify_nonce(wp_unslash($_GET['_gptranslate_nonce']), 'gptranslate_toggle_' . $id)) { // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized |
| 91 | wp_die('Invalid nonce.'); |
| 92 | } |
| 93 | |
| 94 | global $wpdb; |
| 95 | $table = $wpdb->prefix . 'gptranslate'; |
| 96 | |
| 97 | // Toggle published flag |
| 98 | $current = $wpdb->get_var($wpdb->prepare("SELECT published FROM $table WHERE id = %d", $id)); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 99 | $new = ($current == 1) ? 0 : 1; |
| 100 | |
| 101 | $wpdb->update($table, ['published' => $new], ['id' => $id]); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 102 | |
| 103 | wp_redirect(admin_url('admin.php?page=gptranslate&action=state_toggled')); |
| 104 | exit; |
| 105 | } |
| 106 | |
| 107 | // Check for plugin update |
| 108 | function check_for_gptranslate_update($currentVersion) { |
| 109 | // Start the session if not already started |
| 110 | if (session_status() === PHP_SESSION_NONE) { |
| 111 | session_start(); |
| 112 | } |
| 113 | |
| 114 | // Check if the update check has been done in this session |
| 115 | if (!isset($_SESSION['gptranslate_update_checked']) || $_SESSION['gptranslate_update_checked'] !== true) { |
| 116 | |
| 117 | // Perform the remote XML check |
| 118 | $remote_url = 'https://storejextensions.org/updates/gptranslatewp_updater.xml'; |
| 119 | $response = wp_remote_get($remote_url); |
| 120 | |
| 121 | if (!is_wp_error($response)) { |
| 122 | $body = wp_remote_retrieve_body($response); |
| 123 | if (!empty($body)) { |
| 124 | $xml = simplexml_load_string($body); |
| 125 | if ($xml && !empty($xml->update->version)) { |
| 126 | $updateversion = (string)$xml->update->version; |
| 127 | if (version_compare($updateversion, $currentVersion, '>')) { |
| 128 | // Store the update info in session (version, and flag that update is available) |
| 129 | $_SESSION['gptranslate_update_version'] = $updateversion; |
| 130 | } |
| 131 | $_SESSION['gptranslate_update_checked'] = true; |
| 132 | } |
| 133 | } |
| 134 | } |
| 135 | } |
| 136 | |
| 137 | // If update is available, show the notice |
| 138 | if (isset($_SESSION['gptranslate_update_version']) && sanitize_text_field($_SESSION['gptranslate_update_version'])) { |
| 139 | add_action('admin_notices', function () { |
| 140 | echo '<div class="notice notice-warning is-dismissible">'; |
| 141 | echo '<p>An update for <strong><a href="https://storejextensions.org/extensions/gptranslate.html" target="_blank">GPTranslate</a></strong> is available. The new version <strong>' . esc_html(sanitize_text_field($_SESSION['gptranslate_update_version'])) . '</strong> can be downloaded from your reserved area if you have a valid subscription and license for the full version.</p>'; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotValidated |
| 142 | echo '</div>'; |
| 143 | }); |
| 144 | } |
| 145 | } |
| 146 | //check_for_gptranslate_update($this->version); |
| 147 | } ); |
| 148 | |
| 149 | // Post admin notices after actions |
| 150 | add_action( 'admin_notices', function() { |
| 151 | if ( isset( $_GET['page'], $_GET['deleted'] ) && sanitize_key($_GET['page']) === 'gptranslate' ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 152 | if ( (int) $_GET['deleted'] === 1 ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 153 | echo '<div class="notice notice-success is-dismissible"><p>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATION_DELETED')) . '</p></div>'; |
| 154 | } elseif ( (int) $_GET['deleted'] === 0 ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 155 | echo '<div class="notice notice-error is-dismissible"><p>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATION_DELETED_ERROR')) . '</p></div>'; |
| 156 | } |
| 157 | } |
| 158 | |
| 159 | if ( isset( $_GET['page'], $_GET['action'] ) && sanitize_key($_GET['page']) === 'gptranslate' ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 160 | if ( $_GET['action'] === 'state_toggled' ) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 161 | echo '<div class="notice notice-success is-dismissible"><p>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_STATE_UPDATED_SUCCESSFULLY')) . '</p></div>'; |
| 162 | } |
| 163 | } |
| 164 | |
| 165 | if (isset($_GET['imported']) && $_GET['imported'] == '1') { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 166 | echo '<div class="notice notice-success is-dismissible"><p>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATION_IMPORTED_SUCCESSFULLY')) . '</p></div>'; |
| 167 | } |
| 168 | |
| 169 | if (isset($_GET['settingsimported']) && $_GET['settingsimported'] == '1') { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 170 | echo '<div class="notice notice-success is-dismissible"><p>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_SETTINGS_IMPORTED_SUCCESSFULLY')) . '</p></div>'; |
| 171 | } |
| 172 | }); |
| 173 | |
| 174 | // Add hook for admin menu links |
| 175 | add_action ( 'admin_menu', [ |
| 176 | $this, |
| 177 | 'admin_menu' |
| 178 | ] ); |
| 179 | |
| 180 | // Add hook for record saving/deleting |
| 181 | add_action ( 'admin_post_save_gptranslate_record', [ |
| 182 | $this, |
| 183 | 'save_record' |
| 184 | ] ); |
| 185 | |
| 186 | add_action( 'admin_post_save_gptranslate_record_and_close', [ |
| 187 | $this, |
| 188 | 'save_record' |
| 189 | ]); |
| 190 | |
| 191 | add_action('admin_post_cancel_gptranslate_record', [ |
| 192 | $this, |
| 193 | 'save_record' |
| 194 | ]); |
| 195 | |
| 196 | // Add hook for adding main frontend app scripts |
| 197 | add_action ( 'wp_enqueue_scripts', [ |
| 198 | $this, |
| 199 | 'enqueue_frontend_scripts' |
| 200 | ] ); |
| 201 | } |
| 202 | |
| 203 | /** |
| 204 | * Singleton class instance |
| 205 | * |
| 206 | * @access public |
| 207 | */ |
| 208 | public static function get_instance() { |
| 209 | if (null === self::$instance) { |
| 210 | self::$instance = new static(); |
| 211 | } |
| 212 | return self::$instance; |
| 213 | } |
| 214 | |
| 215 | /** |
| 216 | * Activation plugin hook with db table creation |
| 217 | * |
| 218 | * @access public |
| 219 | */ |
| 220 | public function activate_plugin() { |
| 221 | global $wpdb; |
| 222 | $charset_collate = $wpdb->get_charset_collate(); |
| 223 | |
| 224 | $sql = "CREATE TABLE " . $this->table_name . " ( |
| 225 | id int UNSIGNED NOT NULL AUTO_INCREMENT, |
| 226 | pagelink varchar(191) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL, |
| 227 | translated_alias varchar(191) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci, |
| 228 | translations mediumtext CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci, |
| 229 | alt_translations mediumtext CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci, |
| 230 | languageoriginal char(20) NOT NULL, |
| 231 | languagetranslated char(20) NOT NULL, |
| 232 | published tinyint NOT NULL DEFAULT '1', |
| 233 | translate_date datetime DEFAULT NULL, |
| 234 | translation_engine varchar(20) NOT NULL, |
| 235 | PRIMARY KEY (id) |
| 236 | ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;"; |
| 237 | |
| 238 | require_once(ABSPATH . 'wp-admin/includes/upgrade.php'); |
| 239 | dbDelta($sql); |
| 240 | |
| 241 | // Valori di default |
| 242 | $default_options = [ |
| 243 | 'google_translate_engine' => '1', |
| 244 | 'chatgpt_apikey' => '', |
| 245 | 'chatgpt_model' => 'gpt-3.5-turbo', |
| 246 | 'chatgpt_request_message' => "Compile this JSON object key-value pairs adding the translation into '{{target}}' language to the empty value from the original '{{source}}' language of the key and return me only a parsable JSON object without any surrounding characters, preserve and return in the JSON object the key in the original '{{source}}' language within double quotes: '{{translations}}'. Pay attention to not skip any key and translate all keys. Return only a parsable JSON object with no surrounding text, explanations, or markdown formatting. Ensure the response is valid JSON and can be parsed directly.", |
| 247 | 'chatgpt_request_conversation_mode' => 'user', |
| 248 | 'language' => 'en', |
| 249 | 'max_translations_per_request' => '100', |
| 250 | 'max_characters_per_request' => '2048', |
| 251 | 'detect_browser_language' => '0', |
| 252 | 'autotranslate_detected_language' => '0', |
| 253 | 'always_detect_autotranslated_language' => '0', |
| 254 | 'auto_set_language_direction' => '0', |
| 255 | 'serverside_translations' => '0', |
| 256 | 'serverside_translations_method' => 'regex', |
| 257 | 'serverside_translations_caseinsensitive' => '1', |
| 258 | 'serverside_translations_matchquotes' => '1', |
| 259 | 'serverside_translations_urldecode' => '0', |
| 260 | 'serverside_translations_ignore_querystring' => '0', |
| 261 | 'serverside_translations_urlencode_space' => '0', |
| 262 | 'css_selector_serverside_leafnodes_excluded' => '', |
| 263 | 'detect_current_language' => '0', |
| 264 | 'detect_default_language' => '0', |
| 265 | 'rewrite_language_url' => '0', |
| 266 | 'rewrite_language_alias' => '0', |
| 267 | 'rewrite_language_alias_original_language' => '0', |
| 268 | 'rewrite_page_links' => '0', |
| 269 | 'rewrite_default_language_url' => '0', |
| 270 | 'translate_metadata' => '0', |
| 271 | 'set_html_lang' => '0', |
| 272 | 'add_canonical' => '0', |
| 273 | 'add_alternate' => '0', |
| 274 | 'translate_placeholders' => '0', |
| 275 | 'translate_altimages' => '0', |
| 276 | 'css_selector_classes_translate_altimages_excluded' => '', |
| 277 | 'translate_titles' => '0', |
| 278 | 'translate_values' => '0', |
| 279 | 'default_language_first' => '0', |
| 280 | 'css_selector_leafnodes_excluded' => 'a.nturl', |
| 281 | 'words_leafnodes_excluded' => '', |
| 282 | 'words_leafnodes_excluded_bylanguage_repeatable' => '[]', |
| 283 | 'words_min_length' => '', |
| 284 | 'chatgpt_gtranslate_request_delay' => '0', |
| 285 | 'initial_translation_delay' => '0', |
| 286 | 'realtime_translations' => '0', |
| 287 | 'ignore_querystring' => '0', |
| 288 | 'enable_indexer' => '0', |
| 289 | 'storage_type' => 'session', |
| 290 | 'subfolder_installation' => '0', |
| 291 | 'alt_flags' => [], |
| 292 | 'languages' => ['en', 'es', 'de', 'it', 'fr'], |
| 293 | 'enable_reader' => '0', |
| 294 | 'responsivevoice_apikey' => 'MXQg7jpJ', |
| 295 | 'responsivevoice_language_gender' => 'auto', |
| 296 | 'responsivevoice_volume_tts' => '100', |
| 297 | 'responsivevoice_voice_speed' => 'normal', |
| 298 | 'mainpage_selector' => '*[name*=main], *[class*=main], *[id*=main], *[id*=container], *[class*=container]', |
| 299 | 'elements_toexclude_custom' => '', |
| 300 | 'proxy_responsive_loading_script' => '1', |
| 301 | 'proxy_responsive_reading_mode' => 'native', |
| 302 | 'chunksize' => '200', |
| 303 | 'widget_text_color' => '#000000', |
| 304 | 'widget_background_color' => '#FFFFFF', |
| 305 | 'popup_border_radius' => '0', |
| 306 | 'popup_fontsize' => '20', |
| 307 | 'popup_iconsize' => '32', |
| 308 | 'float_position' => 'bottom-left', |
| 309 | 'float_switcher_open_direction' => 'top', |
| 310 | 'flag_style' => '2d', |
| 311 | 'flag_loading' => 'local', |
| 312 | 'show_language_titles' => '1', |
| 313 | 'enable_dropdown' => '1', |
| 314 | 'equal_widths' => '0', |
| 315 | 'reader_button_position' => 'top', |
| 316 | 'widget_max_height' => '260', |
| 317 | 'wrapper_selector' => '.gptranslate_wrapper', |
| 318 | 'draggable_widget' => '0', |
| 319 | 'disable_control' => '0', |
| 320 | 'custom_css' => '', |
| 321 | 'disable_bootstrap_css' => '0' |
| 322 | ]; |
| 323 | |
| 324 | // Se l'opzione non è ancora presente, la crea |
| 325 | if (get_option('gptranslate_options') === false) { |
| 326 | add_option('gptranslate_options', $default_options); |
| 327 | } |
| 328 | |
| 329 | } |
| 330 | |
| 331 | /** |
| 332 | * Function to add admin menu for both settings and translations management |
| 333 | * |
| 334 | * @access public |
| 335 | */ |
| 336 | public function admin_menu() { |
| 337 | add_menu_page('GPTranslate', 'GPTranslate', 'manage_options', 'gptranslate', [$this, 'records_page'], 'dashicons-translation'); |
| 338 | |
| 339 | // Add a submenu that matches the main menu to prevent duplication and allow renaming |
| 340 | add_submenu_page('gptranslate', esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATIONS')), esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATIONS')), 'manage_options', 'gptranslate', [$this, 'records_page']); |
| 341 | |
| 342 | // Now you can safely add a differently named submenu |
| 343 | add_submenu_page('gptranslate', esc_html($this->loadTranslations('PLG_GPTRANSLATE_SETTINGS_MENU_TITLE')), esc_html($this->loadTranslations('PLG_GPTRANSLATE_SETTINGS_MENU_TITLE')), 'manage_options', 'gptranslate-settings', [$this, 'settings_page']); |
| 344 | } |
| 345 | |
| 346 | /** |
| 347 | * Load the configuration settings page held in the settings.php file |
| 348 | * |
| 349 | * @access public |
| 350 | */ |
| 351 | public function settings_page() { |
| 352 | require_once 'settings.php'; |
| 353 | |
| 354 | echo '<script> |
| 355 | const PLG_GPTRANSLATE_MOVE = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_MOVE')) . '"; |
| 356 | const PLG_GPTRANSLATE_REMOVE = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_REMOVE')) . '"; |
| 357 | </script>'; |
| 358 | } |
| 359 | |
| 360 | /** |
| 361 | * Translation records pages, list and edit |
| 362 | * |
| 363 | * @access public |
| 364 | */ |
| 365 | public function records_page() { |
| 366 | global $wpdb; |
| 367 | |
| 368 | // Edit record |
| 369 | if (isset($_GET['action']) && sanitize_key($_GET['action']) == 'edit') { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 370 | $id = isset($_GET['edit']) ? (int) $_GET['edit'] : 0; |
| 371 | $nonce = isset($_GET['_gptranslate_nonce']) ? wp_unslash($_GET['_gptranslate_nonce']) : ''; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized |
| 372 | $opts = get_option( 'gptranslate_options', [] ); |
| 373 | |
| 374 | if ( ! wp_verify_nonce( $nonce, 'gptranslate_edit_' . $id ) ) { |
| 375 | wp_die( esc_html($this->loadTranslations('PLG_GPTRANSLATE_GENERIC_SECURITY_ERROR')), 'Error', [ 'response' => 403 ] ); |
| 376 | } |
| 377 | |
| 378 | $record = $wpdb->get_row($wpdb->prepare("SELECT * FROM {$this->table_name} WHERE id = %d", $id)); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 379 | |
| 380 | $translationsArray = json_decode($record->translations, true) ?? []; |
| 381 | $altTranslationsArray = json_decode($record->alt_translations, true) ?? []; |
| 382 | |
| 383 | uksort($translationsArray, function($a, $b) { |
| 384 | return strlen($b) - strlen($a); |
| 385 | }); |
| 386 | uksort($altTranslationsArray, function($a, $b) { |
| 387 | return strlen($b) - strlen($a); |
| 388 | }); |
| 389 | |
| 390 | // Path relativo o assoluto all'immagine della bandiera |
| 391 | $flagUrlOriginal = plugins_url('flags/svg/' . esc_attr($record->languageoriginal) . '.svg', __FILE__); |
| 392 | $flagUrlTranslated = plugins_url('flags/svg/' . esc_attr($record->languagetranslated) . '.svg', __FILE__); |
| 393 | |
| 394 | $pubIcon = $record->published ? '<img src="' . plugins_url('assets/images/published.png', __FILE__) . '" alt="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_PUBLISHED_GENERIC')) . '" title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_UNPUBLISH')) . '">' // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 395 | : '<img src="' . plugins_url('assets/images/unpublished.png', __FILE__) . '" alt="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_UTRANSLATIONS_SHORT_CHART')) . '" title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_PUBLISH')) . '">'; // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 396 | $rewriteAliasRow = ''; |
| 397 | if ($opts ['rewrite_language_alias'] == 1) { |
| 398 | $rewriteAliasRow = '<tr> |
| 399 | <th scope="row"><label for="translated_alias" title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATED_ALIAS_DESC')) . '">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATED_ALIAS')) . '</label></th> |
| 400 | <td><input type="text" id="translated_alias" name="translated_alias" value="' . esc_attr($record->translated_alias) . '" class="regular-text code"></td> |
| 401 | </tr>'; |
| 402 | } |
| 403 | |
| 404 | echo '<h1><img class="gptranslate-plugin-icon" src="data:image/png;base64,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"/> ' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_EDIT_TRANSLATION_HEADER')) . '</h1><div class="wrap">'; // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 405 | echo '<form method="post" id="edit-translations" action="admin-post.php"> |
| 406 | <p> |
| 407 | <input type="submit" value="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_SAVE')) . '" class="button button-primary" data-action="save_gptranslate_record"> |
| 408 | <input type="submit" value="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_SAVEANDECLOSE')) . '" class="button button-primary" data-action="save_gptranslate_record_and_close"> |
| 409 | <input type="submit" value="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_CANCEL')) . '" class="button button-primary" data-action="cancel_gptranslate_record"> |
| 410 | </p> |
| 411 | <input type="hidden" name="languagetranslated" id="languagetranslated" value="' . esc_attr($record->languagetranslated) . '"> |
| 412 | <input type="hidden" name="action" id="form_action" value="save_gptranslate_record"> |
| 413 | <input type="hidden" name="id" value="' . (int) $record->id . '"> |
| 414 | |
| 415 | <table class="form-table"> |
| 416 | <tr> |
| 417 | <th scope="row"><label for="pagelink" title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_LINK_PAGE_DESC')) . '">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LINK_PAGE')) . '</label></th> |
| 418 | <td><input type="text" id="pagelink" name="pagelink" value="' . esc_attr($record->pagelink) . '" class="regular-text code"></td> |
| 419 | </tr>' . |
| 420 | $rewriteAliasRow . |
| 421 | '<tr><th scope="row"><label>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_ORIGINAL')) . '</label></th><td>' . '<img src="' . esc_url($flagUrlOriginal) . '" alt="flag" style="width: 16px; vertical-align:middle; margin-right:5px;">' . ' ' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_NAME_' . strtoupper($record->languageoriginal))) . '</td></tr>' . // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 422 | '<tr><th scope="row"><label>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_TRANSLATED')) . '</label></th><td>' . '<img src="' . esc_url($flagUrlTranslated) . '" alt="flag" style="width: 16px; vertical-align:middle; margin-right:5px;">' . ' ' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_NAME_' . strtoupper($record->languagetranslated))) . '</td></tr>' . // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 423 | '<tr><th scope="row"><label>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_PUBLISHED')) . '</label></th><td>' . wp_kses_post($pubIcon) . '</td></tr>' . |
| 424 | '<tr><th scope="row"><label>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_DATE')) . '</label></th><td>' . esc_html( date_i18n('l, d F Y \a\t H:i', strtotime( get_date_from_gmt($record->translate_date) ) ) ) . '</td></tr>' . |
| 425 | '<tr><th scope="row"><label>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_CHATGPT_TRANSLATION_ENGINE')) . '</label></th><td><span class="gpt-label">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_CHATGPT_TRANSLATION_ENGINE_' . strtoupper(esc_attr($record->translation_engine)) . '_ENGINE')) . '</span></td></tr> |
| 426 | |
| 427 | <tr> |
| 428 | <th scope="row"><label title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATIONS_DESC')) . '">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATIONS')) . '</label></th> |
| 429 | <td> |
| 430 | <div class="gptcard gptcard-default"> |
| 431 | <div class="gptcard-header"> |
| 432 | <div class="accordion-toggle"> |
| 433 | <div class="input-group"> |
| 434 | <span class="gpt-label" aria-label="Filter">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_FILTER')) . '</span> |
| 435 | <input type="text" name="search" value="" class="text_area"> |
| 436 | <button class="btn btn-primary btn-sm" data-role="search-translations" onclick="return false;">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_GO')) . '</button> |
| 437 | <button class="btn btn-primary btn-sm" data-role="reset-search" onclick="return false;">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_RESET')) . '</button> |
| 438 | </div> |
| 439 | </div> |
| 440 | </div> |
| 441 | <div class="gptcard-body gptcard-block ps-3 accordion-body accordion-inner"> |
| 442 | <button type="button" class="btn btn-success btn-adder" data-addtype="translations">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_ADD_TRANSLATION')) . '</button> |
| 443 | <textarea name="translations_json" id="translations_json" hidden>' . esc_textarea(json_encode($translationsArray, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES)) . '</textarea> |
| 444 | <div id="translations-container"></div> |
| 445 | </div> |
| 446 | </div> |
| 447 | </td> |
| 448 | </tr> |
| 449 | |
| 450 | <tr> |
| 451 | <th scope="row"><label title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_ALT_TRANSLATIONS_DESC')) . '">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_ALT_TRANSLATIONS')) . '</label></th> |
| 452 | <td> |
| 453 | <div class="gptcard gptcard-default"> |
| 454 | <div class="gptcard-header"> |
| 455 | <div class="accordion-toggle"> |
| 456 | <div class="input-group"> |
| 457 | <span class="gpt-label" aria-label="Filter">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_FILTER')) . '</span> |
| 458 | <input type="text" name="search" value="" class="text_area"> |
| 459 | <button class="btn btn-primary btn-sm" data-role="search-translations" onclick="return false;">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_GO')) . '</button> |
| 460 | <button class="btn btn-primary btn-sm" data-role="reset-search" onclick="return false;">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_RESET')) . '</button> |
| 461 | </div> |
| 462 | </div> |
| 463 | </div> |
| 464 | <div class="gptcard-body gptcard-block ps-3 accordion-body accordion-inner"> |
| 465 | <button type="button" class="btn btn-success btn-adder" data-addtype="alt-translations">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_ADD_ALT_TRANSLATION')) . '</button> |
| 466 | <textarea name="alt_translations_json" id="alt_translations_json" hidden>' . esc_textarea(json_encode($altTranslationsArray, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES)) . '</textarea> |
| 467 | <div id="alt-translations-container"></div> |
| 468 | </div> |
| 469 | </div> |
| 470 | </td> |
| 471 | </tr> |
| 472 | </table>' . |
| 473 | // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- nonce field is safe |
| 474 | wp_nonce_field('gptranslate_save_record_action', '_gptranslate_nonce') . |
| 475 | '</form>'; |
| 476 | |
| 477 | echo '<script> |
| 478 | const initialTranslations = ' . (json_encode($translationsArray, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES) ?: '{}') . '; |
| 479 | const initialAltTranslations = ' . (json_encode($altTranslationsArray, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES) ?: '{}') . '; |
| 480 | |
| 481 | const PLG_GPTRANSLATE_ORIGINAL_TEXT = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_ORIGINAL_TEXT')) . '"; |
| 482 | const PLG_GPTRANSLATE_TRANSLATED_TEXT = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATED_TEXT')) . '"; |
| 483 | const PLG_GPTRANSLATE_DELETE = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_DELETE')) . '"; |
| 484 | const PLG_GPTRANSLATE_MOVE = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_MOVE')) . '"; |
| 485 | const PLG_GPTRANSLATE_REMOVE = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_REMOVE')) . '"; |
| 486 | const PLG_GPTRANSLATE_SYNC = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_SYNC')) . '"; |
| 487 | const PLG_GPTRANSLATE_SYNC_TITLE = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_SYNC_TITLE')) . '"; |
| 488 | const PLG_GPTRANSLATE_SYNC_DESC = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_SYNC_DESC')) . '"; |
| 489 | const PLG_GPTRANSLATE_SYNC_COMPLETED = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_SYNC_COMPLETED')) . '"; |
| 490 | const PLG_GPTRANSLATE_SYNC_ERROR = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_SYNC_ERROR')) . '"; |
| 491 | const gptServerSideLink = "' . esc_url_raw(rest_url('gptranslate/v1/request')) . '"; |
| 492 | </script>'; |
| 493 | } else { |
| 494 | // FREE period |
| 495 | // echo '<div class="notice notice-info is-dismissible"><p>You’re currently using the full version of GPTranslate – completely FREE during the initial launch period! Enjoy unlimited AI-powered translations and all PRO features at no cost. 🚀</p></div>'; |
| 496 | |
| 497 | // UPGRADE period |
| 498 | echo '<div class="notice notice-warning is-dismissible"><p>⚠️ GPTranslate runs in FREE Mode with usage limits. To unlock unlimited translations and advanced features, upgrade to the <a href="https://storejextensions.org/extensions/gptranslate.html" target="_blank"><strong>PRO version</strong></a>. Current FREE Plan: translate up to <strong>500 words</strong> and read aloud up to <strong>100 words</strong> per page. Don’t lose AI power – <a href="https://storejextensions.org/extensions/gptranslate.html" target="_blank">Upgrade Now</a>.</p></div>'; |
| 499 | |
| 500 | // List records |
| 501 | echo '<h1><img class="gptranslate-plugin-icon" src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAGAAAABgCAYAAADimHc4AAAACXBIWXMAAAsTAAALEwEAmpwYAABAI0lEQVR4nNW9d7xdV3nn/V1l79PuuU1Xvdqy5KpiS64YU1wAm2pCYAiQOgmBSUgmk2QmbxIyqZNJMm+mQEgjCUkgDgZTTLMhrrKRbdnYcpMlS5Zldd2rW0/Ze6/1vH+stc+9VzZYJsznk3d9Plc6Z5/d1rPWesrvKUsBGlDxr/ys371CX5U7+aGucLmD1aDQCImRcAYKL2B0/ApoDUj5WXBeYXT47jxYI2gVzlYiOFGoeE3hQStITLxZvKl4UCr85n08Hp+hlCCiEMI5IrP/ewGtFFqVJ0u4WMJ7+nhvEfBeoTUIgpQEUL3H4EQhEo5p5VGxD+F5CgU4P3tt4RWC4EWByKTx6hE8X2k7vvqVo/4o4fEeEAWYSHgN6PeuUFdPOP5rFy4qgESBtQqlFamBvoonTSB3irwAaxSJAScea8KAiICxgWCCwpTEMUKSgPYKY4ROFqhptCJ34VhqFUoJSgkahYpE0YBHxbEXFIG4hVM4L1jN7EDE55aERINBemOXGCgkvFNeaJwI1oRnOAfagFaCFxBRdItwpdWKVAs+PkcRBgCBrADRgBM6ORQF5EV4t+mu4JxCC52kkD/654Py38pBUEAa+2fet1L/x31t+W2dKJY1hNRCaqFR1ShraGcajcdoT15A14VB0VpRiJCaMMu9By+KxAhOykEQtFEYHWaGUeEllQrX5F5ReEVqBKvDVNZxJUk5FWNnJRIzzHaFc4LW5WDNznKFx3sFGvScma/jepe48hThvUQUrhC0DhPJA84pCkccAB36J2FyKRWu0wSie6UwSmENDDSgqj2TU55j44HSk10YmxEange3j/Gm/S1pKaAG2BuXqveOi/pEmggDVUV/RRgZBKxh/2HF1LTQzQSXC1oE0YrCQ2oCMZworAajAsHEK2wcAOfBRAIZ1SPTvH8zCVwiMXOWow7XlitAACVhhfh5XCVSk5L1xD/AxfNUyScJ15csxM/53RMmjp3zu5vDEbVSeJF4bhjo8tyuDxPGKEG0ptlU1OvC8iWavkQxPe2ZagvjM8JE16MyteOfDsjrFdA8u09tPrtf3U0qjFTD1Fq1TDHVVTz9rDA14WgkUK1ALVXUkkDYzCusgooNM9gqiUszzP6sUGQO6knogQeSONMUsSM+dCZ3kMQVQjzPKIUTesQt+S0CHokcVMqFAQKu5P2U7DCyKxPuUw6El0DsMIkir/dhRVhDWE2iwirwc+RCHGAfiZ0oyETRdYqqEcRDqwvdrjA+A0WiWbbCsHmtIesUHBvzTHWglcEL4/IpC9ihCv/dJDBUh0qiGBk0nJiGxx4vGEqFMxdDo65waBILzbpHaWGqrTFGUUs83RxSq5E4K2o1aHWC7KhXhMJBUSiSBBBPqjVFJKxWULjAVEVprFJY5UBrxAcCigqsTokg3gM6SDGn0SqwjVJIKx2mtjU6EjwIf62DmPVOobUic+G96hUoCkEwSFi+KB36UjgQNM4LSgeBaU1QLrz32Hif3GlSK4hTdHKgELodYWxc2PNMzkwHrr7IhJEfD4rMmQN8wC5I2XRmn7qMCthUsWBIMZMrHn0iY1m/YsmIJq0JSapIEkW1Cv0NTZ5rhpyQVATlFV5Cp6zxQRtCs8hImJaicF6hjWAtKKdIU830jCeJQlyU0C3C6qtVQKN7GkqpSaHCKvDeoBHQQpFplNIo5REfTtNGo1TU2HRQ7BITVmbu4mw20M1BudC3vCj5OhitUSoI0MJpxHsyr0iUYLTGJoQ+oamnQicTikJjE0+RCbmzuBw67YJGn2KgT9i5v+CBpxIuPlsz1XI4H1i2Ha6oKyupIqkKxihsqnl2j2NRHZYv0QwNQb0PKlWDTTTGevqrAniUCpI+SYIgKgSqlcB2XAbN/iBYuy2FTSGtCtprguLo6WsoRGuSRGGsxnlDkWdUEkhTjTgB5SM/1ogSlIA4C9pjTUHRFbyKQrPQiDiMVYAlTYogjG2YGC5XeBX0KW0gzzUqCli0Js8lCGw0aeIRpSlyRadt8BoqiaBESKsCTuNFqNcUnY6mm0OSKIpCUeQe7xJaLUNazaiknk1K8ei+nGUjlmULFS8cE/Kuwo5UuaBe9XS9Zumw4vgEtKc9569W9A0IA0OKRp+i0fCBBWlFoxaIJoVQFIZ6Xx54uVekFSEvgqCr1YTcG7oJGOuxNRVYgQ9swySWogARD16wVU+RJyTWY4wPuqJSGKMp8qBFIR5fgLYGqx2FUxSZwlqNiAStR4IsSatBNhgrGG1QWnDi8YVHYSkKjXcOZRRGa7oZpLXA/zUaZSDrarKOwolQqwaWmVaD8tHpaJIK1D10O0FDKlxgPVYVTM4kmMRQMwpRwvIpz57nPEsu0jRSYaoQbGJkiTUKrGATxeHDwmBDqPcbRhYq+vo8VatJ06CX1aqe4WGhqh3HD4LVOdZBXkCCI/Vg4xJXOahORtINE7laDcd9HjQPIw4j0G0HtmAUUEASP5cCjwx0AUZAGSjEoYrAXbQHFX8XFQSoFA5tINVhJVrApoSBc9FOUBnKg8vDd23A5JAmwYYRH1QoV+RUVZAFiQNtIVWgXPjfuPC+/X1AUuHklIbUkFY9PsmDnuc1fbmwaoXi8b2OsUnNYL/Q6jqsiupjrc8zNm3Iu44VSxVpHfoGFM1a0HTSiqJRg7ovOH4I9ncUJ7OVpI06quNxRdDn0jRagCjwghdBfBCwaRY65p0DH1QPFVmDCCQpPQFpEoIAtgqKIFO0DsLLR91UmUBUMBjjezxc4gdzMiivRiuUjr+jCUpssCFUqTkZjXeCmQgWdjD2LM55kKAM6EQwUbBLtMaUCoK5VhxgdWWK5UsUeW2Ibt4mtRqXetoVxYIRTTfzNNKCdtuxaCgoFdZHW9gohXioVaG/CfU6WC0kiaLZB5WqI80LHtoLB4fezZor382m5auoVWs476JmoiLBfRSWPqiBWqEMzPgOB6dPoIxjJuvgpLRXQ2tzem1Wpz/d9t0vEMJqa1RqpLrCiuZimrYP5XXQeqJAllLtLc3pUiVWGu8dk+PH2f30LTy/+w/YunKMenOAmQySpEu9rhCjqTUVjSq0WqCUxhqw07koazUCVBLQRmETqFeCDa8IQrZaZNz1hGL8vN/lHf/uP7C4mZN1pijyHCGJ6ht4CcQP/NjjjOLR4wd47PDzdIocozWFc6Q2Cezl30DzAoemxrDG8OTxfVRtysWrzuTiZeuoJykQNKiS+IJEzSvwSaUV6bqldC6/mHu/eTU77nonl62aolJp0skM1Yqj3clQKqFaNxBljjEemwkkFrBhyXsREqswqUGnQqVWUEk8z+yByTM/yHt+4pfp8y8wdnwK78ulLL3/y885nt3TY9x3YA+5K+ir1KglKQqoGNsDuv4tNK0gSSsAiIHcOW7f9Rh37nmCq844j82LV1BPKr2+zYJxs99npidJrOENb7maO4v/za4HPsCmc1tMTFdQDrRojA54ElowVmG1RosPi0nbgM0kRqOtIi+ExLowS7vCsXQxm1/3kwzaMcYnpnoPnkv08lgLzz8+/SDbnt9NLUkZbjRJTOD50TL4N9fmvldiDMONJrWkwl17n+Tj27/FeNbuER540WAopSmcMH3kKOsveS9u+etpTeWkNUVtuEqlnlKpKIwN9BLvSFIT0GFXBB1aGaLBIrhMsEZjrWFyGioLN7JixWomJ0/2XkBEAjwQB0IpRdfA3z9xH8YY+qs1jNbzBuj/L00kGF391RrGaD52/zeYyDu9vgLzBqRs3W6bgQGDam7gxNGouXmHVgWFmADBq2CPdAtBax1nsfMICofvoY/driZNA/KpTJNqJaHIi3msx/tZfPxQa5J/fPzbVJMqFZv8m5zpr7QJULEJ1aTKX2z/Fs9PjmGMQX/XiaXQChwJWQGJzfF5uwcSQsCxPCaAjC7qxVp59Jz7JTZCtD7gKgGu9fP4fdm894jRfG73I2itqSUJ/gc8670I3TxnJuswnXWZybp0izygkz/QJ730s2tJgtaav9lxFwWzK/+lmgggRRDYrkDQKK+pkaNEyAqNRqhaQUtAeCMm7vGFQrzHRsPMe4VoUNqHwWB25pcrQSnFw6MHSbShmqQ/MOIrpejkGccmxzk5M0ViDQsa/Szs62dBvQ+rDGMz0xydGqebZ9+VID+I5kWoJimJNtz53NPBFxBXwqnvXNoJSoMnoZ0leBesY41DieBdjipybAD/Iqada1xwWeA8OF/gvcZa8EUw4+0pD9Na4xLDtv27WNg38APh91opWnmXkzPTLBkY5oZ153PO0pUs6R9ioFZHKY0Xz0SrxaGJUZ489DwPPbebw+OjDDf6qf5fWIEQJl5/rc5dex7nVavX0zBJT/a9mDM4ihwEi1I5oqArhkKi79UbRHmsUgEPD1i4RnBhyWjwXqO0Q2lwLliVpfAVkd7obz+0D6vNvBf5fptWimNTEzTSCu+95LW89pyN1KOKeGqrDqQsHhjkwlVredvmy7jtyYf5xhMPM511GGk0573LD2o4lFJYbbhr75PcsH7zi1ZdoA0orUGDlYKqETpaRxmgcN5QSRXjBJQB78C7AOG6QuNc0IwSHWABmeMMn6sBaK3xRvH40QMM1Or/auIrpTg0PsZZi5bxodfdwMLmwIvOKdmfMWbe8b5qjRsvehWXnnEOH7/jVvaPHqWaVMh98GkZraiYhGqakhjD9/uqIsJArc4jL+zj2rM2kKhZFjTLkgP+owBjBO8tVikqthtco9HPHXwnCE50HIngWJcY8aC1oFUF8S2skYjPz/L9rjhOzLSCd0ibf9UAaKU5PDHKxhVr+KXrbpzHW2e6HR7c9wzPHD3IielJsiKnUaky0tfP2UtWsGXNOio2AWD50AJ+5U0/xCfvuY3cOxY2B0FBJ+tyfGqCwxMnOTE1SbNao69Sg0iMV9KMNnhg//gJlvcPUTuFFQlBafEK2j6h8B2y3NAtUlAOqwsKB0ZsYOniA8hllCLRnkoSYIgs0zjvgnZUzrySYFoz2p5i/9Qo9l+p62ulGZ2Z5MyRJXzkmrfPI/6du3byhe/cz4nJCawxVJIErTTF+BiPH9zPHbt2MvTQPbx54yVce96FAAzUGvzide94yWc9P3qM7xzYy7f3Ps2BsVGGGw1qSYp7Be8vEhz0e0aPYrVhzcCCedZx6eluZ6BxJIkhLxSZN+TKI6LBGzQOC8FtqPDxf0gTj8aSZQrvc0Qg6wRbca712/IFE3n3Xy3wukVGojU/esU1pHZWzH/qvm/xpUe/zYK+fpYPLeh1MMwwTe4cx6bGOTYxzrFoIL5cW7VgEasWLOJNG7Zy+xMPc8sj99PKMhb0NXFxdZ9O8yKcaE1xsjPDGYMjveOlFpQXmj4N9cSRdQzaWLTPAcFH167SGls+U2uF76qIUEKSSIgHQsX4HkFFZLBsY60pprttqnH5fz/NKMWhqUmuO+8izli4pHf809vv4EuPbmf1gkVYbXqDXAJgRyfHcd5z5drzuPGiK1jYP/iie4+3pjk+NQECw81+FjSavd8SY7l+4yWcu2wVf/KNWzg2OcGi/oHTHoSqTZjqtDgxPdF7r5ILeB/80jUToiq6RYooh9WelJwOCXkG4LEBZhXyQvdGr3AKmxQYY1A6OBDNXAe31mityZ0PzurvQ//WStHOM45PTWC14fXnbur99tBzu/nSd7azanhh4LdSxgEpxtstpjotzlu6knduuZJzl66cd1/nPXc/s5MHntvN0YmTdLIuKEUlSVjQaLJ55Vpef+6mnmZ1xsgSfuPN7+EPvvbPjM1MMVRv4uXlByHA7dKLD5qPh4GRnMJDliUgOUZ5sBUyURQu+JqLosAaY8A7nAsODm2CdM5yg/PBYRHcsuUM1HM+vzQe8nKE7+Q5Y60pRvr6ee36jWxdcxYrh2eX8Vd3PkQtrZAYixffG6yx6UmWD43w/stex5Xrzn/RvR/Zv4fPP3Ifu48eopZWaFZrVNM0Dozw3Ogxdh7czzef+g4/dsU1bF51JgCLB4b44Guu53/cfgudPJvHBl9uEMpAgbm4WBk3VGhAGxJT4F2CywuUuOALsC5gbVor5SWoS84JLg/2QJYHB7fSQuHBFyp4iXro33c3xb878TUnpidQCt666VLecP4W+mv1eefsPX6EgydPMNzoi04QODx5kkqScOOWK7lhw1YqEaMv23MnjvK5HffyyIG9VGzKmpHFiAhuDk5lNFSSJiN9/YxOT/GHX/9nfvxV13Ld+VsAOGfpSq47fwu3PHIfKwZHTmsVwGx44ly6hGbxXfBeEJPiyfHOocXj0eRicC7DEvwJWBui01DgXAjxCHPdgwlBTFrpeQbYKyX+4Ykxlg8O88HX3sDqBYte8rznThwlcwUmPmuy22LL6rW8++LXsGRgaN65k+0Wn3/4Pu5+ZidOPEsGhkiMpdXtMtlp4b0ntQmD9UZvljoRhhtNqknCJ++9naFGPxevWQfAG86/iAf37WK62/6uxt9LtbmIMJS+bI8xoJxFChfDKDVCiKYQF+KTbDlkSsIghHiY4MQ2WiFF8KKWVnD5oFcyCEZpDk+OcfbiZfzyG981b4m3sy4CvQ6fnJkKHVGKsekpLjljPT/zmjfNu5+IcNsTD/Olx7ZzcmaKxc0hqmlK4QoOj4+RGstbNl7CqgWLuHPXYzzy/LP01xoMVOt48Xjx1NMKfdUaNz14F5tWrCG1Cc1qjcvOOIdbvnMfjUr19FTrl7D+Q8SfYFIw9QTfzgNgSfBDWz03CA1UNAOCg9oHbUdrCWwHDREFnYt39MIEX6ZppRidmWLpwAJ+4dp3zCP+A/t28Zntd/KBy6/mwtVnxbef/a/wjuWDw/Pu9/D+PXzxO/fzzNFDDDX6WL1gMc47TkxN0sq6XLR6Le/eehUrokzZumYd9zzzOJ9/5H72jx1nYTM4WrwICxr9HBg7zl3PPN6zITasWMM3nnyYwjnMaU6yuTKgHABBIxGO1sagtcdoFxHlcF9BAhiHgDHgcyErgqmsPRTO9wJZlQ4C2BiDc643S19uELpFAQg/esXraVSqveOfffAebnronuCinMPTT5Urfs7k+odv38HndmxjuK/JquFFaAXjrRkm2zOsXbSUt22+jC2r173oHV69/gIuXL2WWx99gLt2Pc54+yQjjX6M0VSThCcO7u8NwJqRRSzpH+To1AR9c9735QZgniY0pwtKWyoNhZ0ApbI4OgqlAu5mlQraj3OQZyHm3loBNF4M4gqUC+zo1Ae+XNNKMTYzxdXnbuLcpat6x7/48P3800N3s3xwAc657zmIc4GCAydP0KzWWNwcoJV1OTY1wUijnw9cfjXXnn8heg4us/OFfQzVm72V0Fep8Z5LXsPrzt7E5x6+l/uffZrUWuqVKqMzk7S6HeqVKtZYhhtNDpw8Aa9gAE6li8LQ7cQACutIEhchbIfyConZDgGOFsgdOK+oxHicoghJC70gb1UaGadv9ebOUa9UuGr9ht6xx17Yx2cf3sbKoRGsNkwXxWnfb7BWx2jNofExlILrL9jK2zZfTrNW651zbHKcz+3Yxr17nmSo3uBtmy/n6nM39WTW4oFBPvS6N/Oa9Rv47I572PnC8+SFYzoLAwCQmFcGZ5/qE0fC5FMGVN7BdRRae6rWhfB17xGnyQuFFQJtxQfwTamwErKOjQGvQl4E4WGMRhUAqmeRfq820W6xceUazhhZ3Dt2y8P3Y7WhYlNyV7wib9bJ9gxHJ07yunM28UNbr5x3X4C/3XY7X9n5ENUkYdngMK0842+23c5dz+zknVtexYWr1vbOPX/5as5fvppbH3uAu3ftnDexPP60VnjZTmVBCvA40jrgDUoEbUEnCUo5rAZlFF65AIKKD+ZzSI4oYzBVYEvdMlki8HytdWAbL8OGlFIU3rF24bLesScPPc/zY8dY0Hd61uapbWn/EB+55m284YItL/n7koFh1ixYxOGJk7SyLv3VOn1phcMTY/zxbZ/n0jVn884tV7B8aNboe/PGS7hq3QVUklk4pZN1MeqVq9q9FomqHOSugzJgTRJiYHFYawmKkMf6GPBViXZAMAWESi0HQkSyMYDMSnrdixb77s17T6I1q4YX9I7tHz1K7oqYJfPKmgAfuPzqeZrJA3t3AXDJmWcD8MYLtnDd+Rdy2+OP8OVHt/P82DEWNQdZ3D9IVhQ8tH83Ow89x2vXb+Ctmy+lWQ1G4Fxj8MjkSV4YH31FdsCpamjQTxRkQRb4IkfyAlcI3ms6BeR5hHyIKiiakB9F6R9WiAM8SOmwAZxzGGN6foHv1rwIqbW9TgKMTk8FY+60uzbbFPSIf3xqgs8+dA/b9jwJwKv2n8e7tr6ahc0BtNK8ccMWLjvrHD730Da27XmCk61pRpoDLB9aQKvb5Ss7H+SBfbt4y6ZLuebcC+dpLTOdDuMz0/TXGvM1m+/xXi91njWCaEJ4e9cyM2PIsoJuEZJN0MEzaVGBxZQ3KxPcEBXDEiGHWbVT655H6uX4ZITQet/nQgPfT+vmGV/fuYOvPvEQ050OSwfD6tq25ykefWEf15+/lTdu2EIlSRmsNfjJV1/H1edu4nMPb+OR558NQWJ9/axesIiJ1gx/cffX2fbsU/zsa65nUURT1y5ayls3X8bnHt7GssEF3+Ntyj6+eAX0AoiBQgwUOYVLyXwe4R2LOEU1ibRXKkT4EsL0Q2CWd8E/7IN7UmsfB2jOw77H7AjWoCfL896x/lr9ZVdOO88ovAsRY6fc/q/vvZ2/vPc2EmNZOTzSG96VwyMkxvKX997GX997+7xr1ows5peuu5GPXP02hhpN9o8eY7LdYqDWYM3IYnYfPcjv3PppDowd713zjouu4JwlKzg5M/V9TRilwIuhKMDg8F4D3ZCPrGJUuJKQh6CIMfs+ZLuUUTai9Kxr0oKxGpTu8f/ek75LM0qTuZxjU+O9Y8uGFqBPichVSpHMsY5HGk1a3S4nZqZCTticZ8xkXRY2+6mnlXm4vfMBWljY7Gcm6/aOT3Xa5C74hC8+Yz2/+/YP8N5LXoN44djUBEopVgyNMN3t8L+++UVa3U7v2tees4ncFacFR7zkIEmwrYo8DIb3gsJjYlYpShClYyyW0Msodw7EaXBEGRDkwit2OUZBu+f44d6hDcvXMNzoZ7rTDpmLUVOaaLd657zhgq385pv/HUv7hzgwdpyZOUSpJsl35culglCdo83c+th2/vvXbma8NQ1Aai1v3XwZv3fjB1i/eBkTrRmc9yzuH+LAyRPcvGNb79qtq89i5fCiec9/yW7Ooc1cKMJ7BQXkeYg69IUOeQ/GIy4m+tmQzYn3xGTkOKmVB+VDLpcvcX9mywyol7cDRIS+So2nDh1goj0DBMDt1evOZ3R6qne/1Fg+t2MbTx16vnft5tVr+a9vex/vv/xqaukcmOI0rIa55yQm4c5nHuM3vvApvvLoA+TR6BtqNLlh48UhYzLC1gubAzy4fw9jM2GwqknKOUtX0sq635MNlRQ4NVBZJEZYKU1RCB6PTkDEYDRY7TDKoY0OGSJZrnr5BwDigyUnqBhqJ/P49+nAEfW0wpGJMR7Y+0zv2Fs3X8aGFas5cPIESmn6qnVGpyf5w6/fzCfu+mpwIcb7/+gVV/OWjZf2rp3stiicmwc5lE2rkHcw2Z1dTYkxrBxaiKD4xF1f5cuPbe/9VrFJD/IuCT7ZnubJQ/t755w5shh9OprQXCNMqZjy5DAV0FZjbQh2U6LIvUISjU4MzhNZskRBrIIP03uFFETQKMQNlctAzfl72SZCvVrlX55+tLeUtVL8x+tuZN2ipewbPUJe5CzoazLUaHLv7if59Vv+jlsevq83W+fq/auHFjLVaXGyNYVRuieEjdKcbE0x1WmxemjhvFdw4ulLqwz3NXvyAKAoAcVT2pGJWef+SF8/afTKvZImAsYELceqLjjQyuK9QnmH85D7EHPV04K0ClaYjZxFtOBj4G4QGmF5zzW7X04qCDBQrfPCyRN8evudveONSpVfu/49XHvuhRyePMmRiZNhtg6PkNqEzz50L7/+hU9x/56n5t3vRy57Hb9+w3sYbjTZP3aMbpHTLXL2jx1juNHk1294Dz9y2ete4j1CKlRySjDXqU0rzVR3NlHKGttL2P6e/TwFC1JxUksBWptoZxUoo0g0aO+QIkShWx9Ltmgd87skDoTWsRaD9DIGlZ7NC7Cn6TcVEUaa/dz1zOO8OsZ4AlSShH9/1Rt51Vnn8fmHt/HU4Rfoq1QZavTRqFQZm5niY3d8mX95+lF++OJXs27xcgC2rFnHppVncutjD/CVxx4A4N0XX8WbN16CfRkCn867zo3wkBKrf4X3gOAP8B60yknrkOQZ2pcJsLEOhkQwzkdEVCTWUICeSxJC3GgIXVTz4uJPhw0JYbkvH1rwIpciwHnLVnHeslXc9fRj3PrYA+wfPcbC5gAjff3k1Qa7jx3i9796E1etu4C3X3gFQ40+rDG8/cLL2bwyONXXRFBORJiJsLI+HRY5p6mYYDjSN9A7NtGeISsKGpXa97jylP72YkMtxgYCCBqtDdYWKC34wmDEY3XwuwSnR7SAc6cQFwSylxAVrQhxLiXhX6lPuJ3nLOkfZLDeB4RQw+dHj8475zXnbOR33v4B3nnRq2hlXZ4fO44gLB0cZqje5I5dO/m5z/wZn7r/m71r1ows7hG/m+d88p7b+Kt7vjFL/Ijylv1TpxiRas5fUeRUjGXt4qW9U547fgRXFBiYd2748ygvJUrfa73w9DgQxlSjAatQYnDO94BNESIWFNVMceUqUBRFyOSQIoyENkQvDshLBMd+r+a9p2Jn1cnJTovf/8o/s3n1Wt576Wvpj3hRNU35oa1XcsW687j5oXvZsX83qUmoJikSnekL+vpfdP9Hx57j89/ZxiMHnuXN581qTbWBKo2FVQb6Uk7UmnSGB2cJ1d/PzNIRvLEYrRltTfO6tVs4c2RJOXY8Im2KFYuZrtY5tYkx5DNtWuYlsCAF3kko/pRmIEUvV1rQQdkRTe6LEBvqI+sRH4seeXBFkMZKx8EoolsyDPMrMtHL+kBlS7QhtQnffPI7PHFoP2/ZeCnXnHdhb+YuGxjm569+K08c3M9nHriT41OTvHPLq7hh48UkxvLcviP8y7ceYfnyEW574mEeO/IcFWWx1rBn/AC3HryfwWYfdz76MCenJzjYHmeB7TKw/w7G1xxFgKmTx1m76zFSE4TkQldwyWSDqb3T5K5g78njqF3fYYNN5vVV6VDDYvzppxhcdxZLt24NfZzjLw8J4SESwuU6prj6EPZPrMBAgVHEfAs/B0KNB0yiCPXYgu8yAhzznM+n24zSdItZTKhZrTPU6MMjOOf52/u+yX3PPsWNF17BxpVn9M47f/lqfvVN76LV7bB4YNY5f9M/38HXv/ogT+16nssuPY+J/RMsW7IA8QV3n3iEp1fsY8eO3Vx/7SU8/M3H2fLON1H50j+hnvsKn433SFFswPRcnhrN0zgeCqWYUMBGTK9QVNk8Dgdcsflqtv7q+9BnrGI8m9WcekG6BOPWO0MoDpLEEmYttI3sRwetM14ZVVEVvF/iw5+xobII/hRfQCm1T6NVkoQT0xO0sy61tEIlSVjcHGT/6DEW9w/RV6lx8OQJ/vi2z7N55Zn80JYrWbVgYW+wSkg7KwruOPgYDxx8hmalxg3XX0ZROEa29PdynIeWhiDbq6++kE6R8arXb+DQ83tZfvkWLvkv72Btx/cimPUcv4QiQOhzv5e/q1iMyHe75DMtqiuXc8aPvQdtDFmWwZwBCBM0qu9JMMgQF+SqDlHVhQ8FC7WOK0BFHNrouHxMYEHOhTh3E+sMzWU9qhQcp9GqScKRyXH2njjC+ctWA3Dh6rVse/apEKYBDDcC4b5z4FkeP7ifa8/bzNsvvJxadIw89Nwz3LxjG0f9BGjF7l0H2XzpWdx33xOce+5qjh8fB2BwqI9n9xzioovWce+2nVz9+i184y++zNYPvo0FP/0BXh5gPv32mc88SDfvcsObzu7lzPVYkAorKcuFPAtlePAGrUMdIlAoT1BDRUKRDNHBGrZaQtlJCULXx4Qz4PvShIw25EXBI/uf7Q3A5WvP5WuP7+C5E0dZNjjc8xUsG1hAu8i49bEH2XnwOS5few4vjI1y/96nqdiEVSsX8bib4F0//BpuueVerrvuYnbs2MXKlQspCmHP7kNcsGENX//6g7zjxiv59D98k3d9+N3M3P0vPHTlPzBpq6el2QvBiGofPYIe6eeiP/oDFl9yEQCTJ1v81M/dxC03beeaa87iTW9Y16NNqf1oJZDHHGztUcaGTCQ8jlDAKuQZQA9uCA8N+WCpDblj3iu8m7WGy/ZyuP68zogwUG/w4HPPcN35F/WcH++/7HX87q03MTYzzXCjD+c9TjwVY1kxtICJdoubd2wjMQkLmwNUrKWddXBa+ORffY2f/sW38NWvbueyS8/lhRcCnn/WWcvY8eAzvOUtl/O5z97NB95/LR/74n28cfow4898m9Gk77TeWQHdfJzmuvO4/Pc+2iP+l774HX70Zz/D+OFRkvXL0KsWvuTFzsXqLFojrsA5hfcFzivwOmiaQogL0ioQvlChKkhRBCOskjqyTAVMO+Y8wSwQ90pEcS2pcGh8lC8/up2ffPUbAFi3eDn/8bq387++9SUOjo+xtH+QMgNSRKinFRpppZeMceDkCVb1LWJxfZA3v+0y7rv3cbZctJ79+48xMFBDPBw/McGGTWt56MFdXPeGrdx5906uPHOI1edfyDW/9cXTetfsxBh5N0MpqCxcgEkSJifa/NQv3MRn/3YbDPfD+at7aDHM14LKuqXaluCcRXxIoxUHSofybSJgXZSlwd8bPDWCkHuNQaN8cNr7aM3MDcP2Jf86jebFs6g5wD3PPMF5S1dx+VnnArBp5Zn8ztvez6fu/xY7D+5Ha0UtqZBE9TBzBa2si9GGS844m597/Vv4x0Pf5C/v+wrLlo0wNdWi0UgRCY6ZNLVMTbXoH2hw/PgEQ8NNxqdaFNMzMHqcmYnWS65ebS3dY8dpnzjOoqteTWMkaF0e+O0//AYf/bXPgB8DtQxOdmDsED7VuNUDc1iP9DAyERsxoQylNWlqyT04ySNbtzhMiIroDaQPGpATiYaECqk0AqFg0hx9uASdXoFKao2hXq3wyW230VetsWHFGiB4yv7z9T/Mjv17eOi53RydHGem20EhNKo1Vg4t5PIzz+HspSsAyFzO4zv3ceWVG9i+/SnWr1/B6OgkIjA42GDvviNs2ngm27/9FK++8gK+8PALHPjCrXzxTz7MyVP81CGS3yIkVIcWs/G/f5TlfbOG14ETHaaqdT76Zx9i+bIhcEVvzikFi4bSHuF7gtiD9w6XEYr+SYFSNqwY5XFigrNLhaqL0UILsz8YEQoRTTcrK15Fa1nriA4KzjmsMbGYqe/5Or9X8yL0V+uMt2b4H7ffws9c9SYuW3tO7/ctq89iy+qzcN4z1W6hFDRrjRfhOt1uwQ+/+7V88YvbuO66rezYsZuVKxfinGfvs4e54II13H77Dt7xjiv5zKe/yXU/fC2VgS4b3rGZSbGz7KIosAuGUY0a7T27WX3Dm1h8/RsojhyhaHcAxfJE80cf3hxCx3nxyul2Ck6MTvc0IJFQWVepIIBDAqSQZQ6cDvnWxoeaqL7MEwZw4HWQ3lZHOMJBnkl0MMT6nTKbpD3S6OdEe5qx1jQ2PT1ownnPYK3BTNbh43feys4X9vH2iy4P6aSxGa0ZbHx3YXlw6gR/+9ff4Fd+7T187avbueSSczh48ASgOOOMJTzw4NNcf/0lfPazd/K+91/HH/zeP/CL/+V9nPlb73/J+41+7M8Ze/AJTt79CId//BfDwbmD/hKrXAn4xJK++lIG/+Sj87xhMAfWKTQaS15A0Sv1EBJilKgyTTVEQnin8E6hJCwlr0wM8vLRzzkLKYgIg5UaC6oNnn+FlrGL8fmptdy75ym+c2Avm1aeycaVZ7B20VKGan09aHm620FEaFZnEcmqSnnt1Zt5eMduLrjgTA4dHqPRV0O8MDY+zTnnrGLnzn28+qpN3Hff47zxfddz8r7t7Prx+xh3GqmkuFqVwUoV/S93MP7wPaR6WZzlpwk/iyBpgm915glg51wUwi5UZHeCy3OUMjEaLqYvOUErN1v6QatQYNsL5E73EEAnEgrjlVVp5wBPixv9KKX49vN7qKWVeTPg5ZoXQSvN4oFBukXOtr1Pcd/epxiq9THc16SSpCiB0elJAD70uhtYFbNqhuoNZmbaHD82Tppant9/lKGhPsQLk5NtZMUC9u49TH9/g2d3v8C6FauYfnY3u26/jfaCc/BZl8GpUQwdEjtEY9FmRLnTViggroDUouo1nHO4Od42EVA2AmfK4bzB56EQuJdg8CpCefkQmBXDUrwK9S0hOhQKkFwjPoymnpMuCiGqa2lzkNTaV5TQMLeJCKmxLO4bwCPkznFwfBTvHIIitZZ2nvGHX7+Zn7zyWi5avY4f+5m3cdkbL2dkuMn0TIdqNaXIQwS3NYZON6fRqAZtqL/ByemMRb/7Xpqjhzn8j59l+m9uJvUG39dPd+oo08ceRpG+/MvOf3PEJtTGlvb60ZMDwZMeAh20xYuLZZXLsvygjccT4Wgt4FS0eCNI5F3YiMBoImARLWClgmM8YkJa4IrV6/nW7p0s6Ot/5eErve4E4Z8aS2rme9ua1RqTnRZ/+q0vcfEZ5/CejVu5aHkf5BkM6uD7q5av6KCmQTIYToCMFd0j6CVrcPVVVGemabVaZCbBnzxO35VbaV71H3CTU+A9qpKiGw0QoZhuUdVCvVEJ5SvnNAV4o8mWL6d1SgEr74IjQnlwWfweCR+8ZMHbKJRQRJQBIhKyIlEoLSQ2hFLgysoppbY0n+dvXX4m/7LncZz3r9gTdTrNeU+zUqOWVNiWTzD2E/+ejV/6MjO82DdQNqHApAPoRh8Lr9jCpp/7KUbecDUr/+cfMXHX/eSP7iXnBEuvvZJl//k/zV5XFEhWoOshV2DUwREHa1NehIyKCK12m+nDhyliEIEOGxcgTmJgg8M7RZHrmJ8mGIQiC3nGVoiREIVCVIiMU9Eixilc4aPDZtYVOTc4V2tNRWuuXb+R2595jAWN5v+VWj1BZigGvbBiy1bOkmGmmy/OYBHvUcZQXbeW9uEjjP/5J1l0912MXHEW4xddiBnoR2c5IRhTYbwwBdDOaNZSpFvw3C/9Z478+f/hzBvfhf8/H+c3P/Uk//TfbmFkWT8LFzZxeaiGZRPN1k2L+bUPXzyviEnIA6NXzd1UQFqhjrbSKiC3BO5ig7ymh/er0FvyXKGqRB9Aie7p6EqTnm+4/Lt42RncsfsJZrpd+ipV3PcR/386rX3oKKve/TY2/Oam73lefuAFnr767aygzarf/VX+unIxf/lLt3Hzb7+GRi2lFSMXFi8f5BPfep5P/D//wDdu+TCLlw5w5if+lOqyZTz90V9l8Te/wsf/4lMs+W/v509/7dOceHIXDC8D8RirWDJ8SqUUCWXeJDIC5TXWFoDD5xoTjQRRAqKwCgWeWIM/4N/BCQ9YUAGbxhUhjgUdUux7MiAKXoPi51/1Rj6+/XYmOi36Y0roD7pVRoY59q17eP7vvs5EpQfRAooisVSGhkm2P8LxT/1PLnr1ZXDzg/zEHR3+5oOf4IyNaxlecD3HTnEfLh5p8uj2h1l+3m/xd3/2I/zIe7ay7Dd/hUU/+xM8ccO72P2ed/D/fuRn+bn7f4P3//FD3PeXt8KiAfSSJmag1mPJPYhDRRDTg7EONWMD74+bGGWFiuCnn8WCtI6RD+J6vF4KKHIdASTb62zpDz5V4DZsws9f8Ub+9/1fp513ezV8fpCtNrKIg39/C3ff/CW6REGm65h0IZXOFH2Msd4arvjY/+Jrl/4QP/PLX+fAbdugXidZPowqTpkUIqEUQ205Tine9+8+zj994XL+5n++m5HFI2x64A6O/tknuftDP8n6f7iJbTd9mj+6/pf5lV+4Cf/EC7iVfT3tpxwE8YJ4g4qGbZ4J2iTYxOGCxQVImPQ+akAihPB0Qk6YdyF0LsdTScDmoxQSeVhcasaYeWwIoKoNH770OirWMtPt0MnzH6hgntq3l3M++EHet3MXN+54hB9+4RA37LifszefzRmMcf0738rSBx7kp9yruP6qP+HA3Tvg3JWhPLD7HiuycLCwCWtXcOtND7B2w2/z6ZseBGDxz/4El71wkKMrz+PBa97IL2/7Ux7/8ge49KdvoLl+9XwkVDy5KEwxSl1DnivE5mgyJFbvElUGwwUNNMTiEpaEF0XuJYh8ZfCFpm8Ipp/ZzvGDzzKwcDneFb3RflFer/dUteFDl1zD2y+4mMRaRmemaOdZ3ADnlcHYpzbX7VJdugQuWM/QRZupezjy7h9j0bdv4+JP/Tm3/erHWfefdvDXP/9xWFiB1SMhBfR0FINys5hzVzDp4Efe82dc/64/5+SJadLly9j0yD2s/tgnufuPP8bgNRdx+w0JH/vxc2cxIFdQH1jE6JH9uP33MDwUtkZRYgM04TxaxULpEoRy3OBjNhwlREYoHGEzAhGPqykGp6Z49NO/QVZP6B9eFKul0xNAc5tSocrK+sFF/Nxl1/KujZeztDmEILS6HWay77/IU9+Za9j1sb/iW4s38O2zr+Tbq5YxmDoWPbqTnxrbwhuu+h8cvv9ROGdl2Eyg8JGf0tMhlYSdoEqPcPTIzv5lDkYacNYKvnbzDlae/VH+8Z/Calj0oR/n8iPHGV19Lk+++Y2c/MhHAPCuoNoYIK8vYd83fpsF7UPopsU7TRmrqAAlHu2DqeZ8NMQKD8YH5M6VdpwESLVwiulpw/JVnr3bPssXfuOnedN/+QuWLBikNdEi73ZQ2vQ6Mjd5rxycTY0hNq06h/HONPvHjjLemmZsZjIUh5oTRHU6S6Mz2M/ilWew5PwX8LRY9u//hNbPfoSf/J3bOfLAY7z2xnPRafIiduMn2qw4exk5Crn0YuzIalQxTrZoCQNVuOp161CLm/OygARBXrUGGWvxiY9/nX0HR/mlD78e6W+y7u5vcuTvP02n3SXpW4BpLGSi5Xj0kx9i4TN/z9LlismuxzuLCbO7t8tIAI5D7K1aP5Dc/5qF+WW2qnBK89xhz9plioUDloE6DDUE4xV9fYp+nXHwOSE/53KWvubHWHL2Vqr1PlxMQwp+4xKenmO2RLZTpj+V/k1VnlESX0pfc9wWMEaY9aK0Zf5+AyCzA94LZwgmZmkwzgUPeyMcn1XybCjxrdKbNUegRh+vF0+Jy0t0RIl40JZup8X4czuYfuxmlpzcxqplMJlXyHNPtwj5B8dHoTUDj+2GWtOyaNgzM1UGZnnoZGEHpUZVM90SlgxBt4C2E+omQNMn8worzugy9fz97P8/93N8WYqpDmJ1gTYOLwZwYZe83OANJNYhTuOcAqfQSYG1Pqi1JuzAFPbVUcxGwQt520TtyyFi8AWY1IUg4oJwr1oIF9G4WTdfhFOUCa4/m3q81+QdIbHglZDnNuyTkBYBajFhIH0R0oYgGKbGCCYJ55uKYHA4Z7BpKDeTFwrvUrRMYqe6rO+D5go4MWlxTiNe4XF0MoMXR5ErJlrQP+TIuqFEqBWJuUwailwY7tccnwiZ23kBrQJS7aOXTDM6k1AbEdYsFLTPyLNjJAasD/4ERZDfnbhdSa0RFAxXxE0rXZD+eTtsWVLuhucjqy4dRFkeMHWrA0sWgYoBihhAIGBiSE4lhbQe1GZfgE3C3jIQFAyfzT5fQvYVkkNFQ1ILpXnydsjnsqGSJRlhD5paDVwS2XSsqWEETAJdF3ZXqVaBhqZQmpPT0O0qrM4o8BSFodvWVDQ8fUxQiWbpsHBsDFwu2MLjsiIk4k21PANNxeFxzXPHPOeuMGQdmIwYj/eaXBUUHY2IppKmeO3JBGzMfwphGNBVITNEjNBxGlGaipZoRwguhcKGvRvFhy2jAssIVdyplPkJJmz0mUJuHJKHUfIeDJ7cWDLx1JKQi+VQmJiIq1So9uK8Qpm4n4wRvA5VIAsBU4QdkTIMTimsuOCMwtBxQl44JKYO+TywUKuD3dRyBq/CfmTSMthqRjer4ASM7eByTTv35F6QNhwYhRXLdXBr+gBL27ZoN5NDIw2R0VkmrFxieHq/Y6TPMTyomeloUuPxuVBJNOhQijHLFdaGCiAuC9ufOBdiH3PnSbXCdYR2NxAtQ7BKsITgVeWFQhQuC/uYaSXk3cCDlQTEMMs93mm0F0yhovsuDICO0ca5F3yLXlS3AUzqkVzQPsgkV2gQhfaOIg9GkhKwXYLarULoSNHVeAWuG5dkiCzEGE+no7FGUVEBSu46jxYdtsZyUHSh6wpwiqlck8d3tSLc/rjQ6LMs6XccHycGEYAR7I83xK0ergf4OfeK/jo0qpZ9RzzeCyNNjdJC5gPfEhFmcsgjZJE7jVeKbtz6Nc8UXR82P3NeaOcGHy1Ap8KudVmhKCg/G5wOu6K2O/G4aJxXtLvhf0mCFlHEeJquEwqgUwQ/q7ZhO0GHxmnB+eibNZqiUHTjdrNlepCLZRgk7nucZ5pOZsPmTgKdrkJMACK7RQhcy4sQsOAKyDJFnptQT8OH/nQdAYjLPU5CFPTJcbhnp6CtYcMaGJ/0dPIQ8Dw+E+Ho/TOwuC/UClWp4uS0sLgpNFcZXjgB2UFPsy4MNgz1VCGFxqHIc4VLhEQHr5m4kJqvjFCIp+gG1pWL64XhaePDfpBh0uEduMJjJYCCPg/QfhBaYXlXEiCnlzLlCCzARsQx84o0D3s9uoxYXCrsCSaFp9OJWmkRgmG98hRxNflK2IeyyBVZIVgvaCNkmSGJmpJDkYvErVmhG+upFnH7LaVAlMFoT7edkHeFyVbBsQnNZBuazYT1yzyTLc9kR9HJhaoS9o6BrRvUaA57ZjTnDno6eVh3o21hpN+zYW3KZCfl5FTBzETYRc/o2FkPVnsqadhvycSe6yQsPY0OMISOexAQalAYQzD4tCJsAxU0GCcKKpCbCK7FAGFvok1B4Bsh9C+wtDKYuIjhlMQtQlCBpSkEV40alkjYII4QPduVgMUYpcgBp4OB5vOoEBRC6akU8eCCMeVV0OuzPAJwEnZe1UrT6mqyHNodQ6MGFyyHVBUcG4fpmRhp4oQXupp9Y2E7ANUA9kwLlVrKGSanU3jSqqGVG8Qqhvs1SxZWyFxItU8rCUnFYpM6A/Uq9bqhWktIbIKxCZU0Ja0YbJKSJpZKVYNJMSbFVAwqsegkAWPQNkEZG6xWY1BG440GZVDGgNZBCJpSvyoty1MNuAhmiYveKI/E4EydO+iGiiW+yGL1kgzxORI2SsAXLrg1nUMKR5b7oFZJTlF48tzhncPnBbnLybKcyckC7zK8y+l2PM7lZN5T5J6qcVSsMDOdc2QyoVs4pjKHyz1jznD/YY1xGRbIEwDxPDrqcarKmqbgnWem0CG4SAkVp6nXEmyaUGtUqNZTkrROvV6lUbNUqxZlLNYkVCsGYy2JtRhrsWmonWytoZIYTKIxiULH4Hhjwy51cT8PVKLABP4c670jlujtDhrTbJxkOQjhNxxhpvqwUwVFGBhXSPDNiqdwHq+jkaV9YHfKYYynEI/THqXLmS0I8XuQ/OHRRmEqBuPDzqm5FEjhMJmjK56xGYV3HnGGmVyRZ45OrjhSVNl+RJhpdVkPWNH6SAH0KZh2OU8eF0bzlDMHKwx5T9VB3WuqPqFVaGxFU82g2oak4qlM51SqQq0S9mBMEk2lorCJUElAWSFNFYnVmERTTQ3WGhJrsIlFKUuaaIwNg6R0/M2qsDIwaBu9dEoHdVIzuwK0jtBB9O05D04H468o8EWQVd22Ii88Lhe6mQ8CtAiC3bmQqO5cEYRqDs5JDEzWoZitCyqzK4L86HRVqP/sFUVhyHJPp60oumHn8bwI+WOu0Mx0HROdlD0nFc8dz2hlHRZGBNqK0oeIiztRYduS42MFk+2UvlRRTw3V1JDYPHj0VYbVCms02pjA07UiTQzWBOJaE/6Mjb8bHb7ruHmZjnUTYqSdVqGWZikvjA7yREerzOjSDx1+Q8XoPTWH+5RpUGWJTV+GCgaFIHc+hI94jys8Pm6/6CRAAYWXUAtaPLmb3bDIu7B1i/PB3VnE8MOi8GTRt+AlZPpkuaPwLt4ncMSicEx3cqa6MN3KseKpq7ABaAZYp9S+MuhBqbCCUwXSyZhsw0Sca4b5GHYyqxSQqMidI4odXaEo6AUelQErSbxX+d2Xgz/nOj3neh/vT+RAOnCnecT3xLwGNSsSfAn7SLBWy+/Ez2WfSgjJxfvEGlW9dyk/ewl1k3QkXjsSkJLL0TNRcPGZ5a5UZcRQNV5TnV2/2KlW65HhxB72ebHUAoWCXOJWrbFDJdEgdNKo2QgBHTuexs/lANg5g2YigYj3Tcp7xGcZBRU1hzDxnkrCMVum9Mvs/cpnR6Ax5FudQrhyZWQR8VWROBIHuQxMFsJ7SCSci/csJyRRtHTigKbxGSZe4+KjnMz2Scvse+ZxEHIJtOyjp05gvfeZNOrf0vn0+4Rw80wRkwdmZZynB2sHFjAXWIwvq+Issmp25tu4KkxJ6PhZS2DfOg5UGRYl8fe01K8BcUJVz96vDCYud0x3cbalkcDla5aroexTmZCumCWuin3V5cDE+xWxX5aeXO9NqnLQYHYiCjG2Snpo9jwXQ04oID4Q+2AEOkly2AJ+pnCf70+T68nyYaXCEilUeAnD/M5Ib2bFh6tZdlGylZLFlKyhJHI5862OxIuDlc65R3l9Ze4gxxUy994Sj/feiVlW5X2wessJoUrWGAeunPmJzK6YGJvQ65eJhCeyHlGzk8nHh1rpif4AKsosS1XxOQXQjau9XwW/S3ndaK36KQu4Trt9rD44+CnJxn+BuIwqsUNFOeIq8K+SJ5b8rcdH43VzebiOM6kMrkviyrAyy9J02cGyE3HQ8jnuBBuXNXMGqVzCwnzZEKM9ekR2QHcOj3ZziR6vzUs5Ed+l5PmocK+OhIqSas594o4uFHPuW7IiX/YnBjc0VFjRiZ+dpNON+kMnJqe+ZiONivHJya8O9TcvcJNT1xDZRSJxJurQwULCPokVNTs4hlDWOI0zWYgzurRGJWyHlcTBSTW9zyrmySoV7pvEQTYK6jqgqr2BUpDE/Qt8fL/S8pU4o9K447YSFXm94DxUnQpEkdkBC5M4aE6dIrIOFN0YB1XycxR0vNBxs9cBtCPoWIj05EpBOBbYkPTknpXZ1WSBmWrl6JHC/TFQWMKg5977znQ3++tmf39DJicvVzIr8JCShQQks0qIay/ZUoJQKwUpswK5J4SVUFXluVEO6LCUg8ANgUwVHfgohM9Wh9+LSPCwrX2IsUwjcUrfsghUA+DZC4wSiUqFllk2QRDIlnAuAm0dhbNIL9SlDKwywAxBhgjSkzFtFTShCDFBXBUFIegkp2dXEkNFAehW0hPHbPL72fT0MSA3BLmgAeOcKwqRx6vNvpS8WGe8qERmialVIG7K7Mws2UpKWDGmXBXxs5VAdBtfKO4djSkd45EXEwWhijyuFFRlaHwaPYLeBaeOEQJOU/LEIqrHEY2IVdfCTrBzrhUfFQAf7+/iPeMSKX9XPsqseI6R8M6KwMd78VSxX6VCUMqCUmCXHjoBJhqNx44o/cftmZldQBdoG6DBLEvVzrm8k+VPm2bzoE6SpvZuoXaiTHwpLbNqqZpD8ITZ3+bx+TnHlITZX6pxSPhs44snJZ+Pn02cbamaJVx5rySeZ8tBi+cnKhCzJIr3s2yg1FBqzBI5YmxhQJgzqPEdrMTAipLg8bwiXlcS10dVtbxniaJ0FUxXqwdHG/Uvneh0/ybrdg8RNNo20P7/AEq4cA9VgzqWAAAAAElFTkSuQmCC"/> ' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATIONS_HEADER')) . '</h1><div class="wrap">'; // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 502 | |
| 503 | // Filters section |
| 504 | $opts = get_option( 'gptranslate_options', [] ); |
| 505 | $languageFilter = esc_attr( sanitize_text_field( wp_unslash( $_GET['language'] ?? '' ) ) ); |
| 506 | $languages = $opts['languages'] ?? []; |
| 507 | |
| 508 | // Paginate records |
| 509 | $records_per_page = isset($_GET['per_page']) ? (int)$_GET['per_page'] : 10; |
| 510 | $valid_per_page_options = [5, 10, 25, 50, 100, 200, 500, 1000, 2000, 5000, 10000, 9999999]; |
| 511 | if (!in_array($records_per_page, $valid_per_page_options)) { |
| 512 | $records_per_page = 10; |
| 513 | } |
| 514 | |
| 515 | $searchFilter = sanitize_text_field( wp_unslash( $_GET['s'] ?? '' ) ); |
| 516 | $engineFilter = sanitize_key( wp_unslash( $_GET['engine'] ?? '' ) ); |
| 517 | $publishedFilter = sanitize_key( wp_unslash( $_GET['published'] ?? '' ) ); |
| 518 | |
| 519 | echo |
| 520 | '<form method="get" class="form-filter-container">' . |
| 521 | '<div class="left-filter-container">' . |
| 522 | '<input type="text" name="s" id="search-input" value="' . (isset($_GET['s']) ? esc_attr(sanitize_text_field( wp_unslash($_GET['s']))) : '') . '" placeholder="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_SEARCH')) . '" />' . |
| 523 | '<button type="button" class="button" onclick="this.form.submit();">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_GO')) . '</button>' . |
| 524 | '<button type="button" class="button" onclick="document.getElementById(\'search-input\').value=\'\'; this.form.submit();">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_RESET')) . '</button>' . |
| 525 | '</div>' . |
| 526 | '<div class="right-filter-container">' . |
| 527 | '<input type="hidden" name="_gptranslate_nonce" value="' . esc_attr(wp_create_nonce('gptranslate_filter_action')) . '" />' . |
| 528 | '<select name="published">' . |
| 529 | '<option value="">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATION_ALL')) . '</option>' . |
| 530 | '<option value="1" ' . esc_html($this->isSelected(esc_attr((isset($_GET['published']) ? esc_attr(sanitize_key( wp_unslash($_GET['published']))) : '')), '1')) . '>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_PUBLISHED_GENERIC')) . '</option>' . |
| 531 | '<option value="0" ' . esc_html($this->isSelected(esc_attr((isset($_GET['published']) ? esc_attr(sanitize_key( wp_unslash($_GET['published']))) : '')), '0')) . '>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_UTRANSLATIONS_SHORT_CHART')) . '</option>' . |
| 532 | '</select>' . |
| 533 | '<select name="language">' . |
| 534 | '<option value="">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_TRANSLATED')) . '</option>'; |
| 535 | |
| 536 | |
| 537 | // Loop languages |
| 538 | foreach ($languages as $lang) { |
| 539 | echo "<option value='" . esc_attr($lang) . "'" . esc_html($this->isSelected($languageFilter, $lang)) . ">" . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_NAME_' . strtoupper($lang))) . "</option>"; |
| 540 | } |
| 541 | |
| 542 | echo |
| 543 | '</select>' . |
| 544 | '<select name="engine">' . |
| 545 | '<option value="">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_CHATGPT_TRANSLATION_ENGINE')) . '</option>' . |
| 546 | '<option value="gtranslate" ' . esc_html($this->isSelected(sanitize_key(wp_unslash($_GET['engine'] ?? '')), 'gtranslate')) . '>Google AI</option>' . |
| 547 | '<option value="chatgpt" ' . esc_html($this->isSelected(sanitize_key(wp_unslash($_GET['engine'] ?? '')), 'chatgpt')) . '>ChatGPT</option>' . |
| 548 | '<option value="gemini" ' . esc_html($this->isSelected(sanitize_key(wp_unslash($_GET['engine'] ?? '')), 'Gemini')) . '>Gemini</option>' . |
| 549 | '<option value="deepseek" ' . esc_html($this->isSelected(sanitize_key(wp_unslash($_GET['engine'] ?? '')), 'DeepSeek')) . '>DeepSeek</option>' . |
| 550 | '</select>' . |
| 551 | '<select name="per_page">' . |
| 552 | '<option value="5" ' . esc_html($this->isSelected($records_per_page, 5)) . '>5</option>' . |
| 553 | '<option value="10" ' . esc_html($this->isSelected($records_per_page, 10)) . '>10</option>' . |
| 554 | '<option value="25" ' . esc_html($this->isSelected($records_per_page, 25)) . '>25</option>' . |
| 555 | '<option value="50" ' . esc_html($this->isSelected($records_per_page, 50)) . '>50</option>' . |
| 556 | '<option value="100" ' . esc_html($this->isSelected($records_per_page, 100)) . '>100</option>' . |
| 557 | '<option value="200" ' . esc_html($this->isSelected($records_per_page, 200)) . '>200</option>' . |
| 558 | '<option value="500" ' . esc_html($this->isSelected($records_per_page, 500)) . '>500</option>' . |
| 559 | '<option value="1000" ' . esc_html($this->isSelected($records_per_page, 1000)) . '>1000</option>' . |
| 560 | '<option value="2000" ' . esc_html($this->isSelected($records_per_page, 2000)) . '>2000</option>' . |
| 561 | '<option value="5000" ' . esc_html($this->isSelected($records_per_page, 5000)) . '>5000</option>' . |
| 562 | '<option value="10000" ' . esc_html($this->isSelected($records_per_page, 10000)) . '>10000</option>' . |
| 563 | '<option value="9999999" ' . esc_html($this->isSelected($records_per_page, 9999999)) . '>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_ALL')) . '</option>' . |
| 564 | '</select>' . |
| 565 | '<input type="submit" class="button" value="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_FILTER')) . '" />' . |
| 566 | '</div>' . |
| 567 | '<input type="hidden" name="page" value="gptranslate" />' . |
| 568 | '</form>'; |
| 569 | |
| 570 | // Bottoni Import/Export |
| 571 | echo '<div class="action-buttons-toolbar">'; |
| 572 | echo '<button class="button button-primary" id="bulk-delete-btn">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_DELETE')) . '</button>'; |
| 573 | echo '<button class="button button-primary" id="toggle-migration">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_MIGRATE_BTNS')) . '</button>'; |
| 574 | |
| 575 | // Export CSV |
| 576 | echo '<form method="post" action="' . esc_attr(admin_url('admin-post.php')) . '" style="display:inline;margin-right:10px;">'; |
| 577 | wp_nonce_field('gptranslate_export_csv', 'gptranslate_export_csv_nonce'); |
| 578 | echo '<input type="hidden" name="action" value="gptranslate_export_translations_csv">'; |
| 579 | echo '<input type="submit" class="button button-primary" value="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_EXPORT_TRANSLATIONS')) . '">'; |
| 580 | echo '</form>'; |
| 581 | |
| 582 | // Import CSV |
| 583 | echo '<input type="button" class="button button-primary button-import" value="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_IMPORT_TRANSLATIONS')) . '">'; |
| 584 | echo '<form method="post" action="' . esc_attr(admin_url('admin-post.php')) . '" enctype="multipart/form-data" style="display:inline;">'; |
| 585 | wp_nonce_field('gptranslate_import_csv', 'gptranslate_import_csv_nonce'); |
| 586 | echo '<input type="hidden" name="action" value="gptranslate_import_translations_csv">'; |
| 587 | echo '<input type="file" name="import_file" class="toggle-import hidden" accept=".csv" required>'; |
| 588 | echo '<input type="submit" class="button button-primary toggle-import hidden" value="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_MIGRATE_META_CONFIRM')) . '">'; |
| 589 | echo '</form>'; |
| 590 | |
| 591 | echo '</div>'; |
| 592 | |
| 593 | echo '<div id="migraterow" class="hidden"> |
| 594 | <span class="input-group"> |
| 595 | <label for="migratetranslations_currentdomain"><strong>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_MIGRATE_META_PREVIOUS_DOMAIN')) . '</strong></label> |
| 596 | <input type="text" class="form-control" id="migratetranslations_currentdomain" name="migratetranslations_currentdomain" value=""> |
| 597 | </span> |
| 598 | <span class="input-group"> |
| 599 | <label for="migratetranslations_newdomain"><strong>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_MIGRATE_META_NEW_DOMAIN')) . '</strong></label> |
| 600 | <input type="text" class="form-control" id="migratetranslations_newdomain" name="migratetranslations_newdomain" value=""> |
| 601 | </span> |
| 602 | <button class="button button-primary" id="migrationconfirm">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_MIGRATE_META_CONFIRM')) . '</button> |
| 603 | <button class="button" id="migrationcancel">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_RESET')) . '</button> |
| 604 | </div> |
| 605 | <script> |
| 606 | const PLG_GPTRANSLATE_MIGRATION_SUCCESS = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_MIGRATION_SUCCESS')) . '"; |
| 607 | const PLG_GPTRANSLATE_MIGRATION_FAILED = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_MIGRATION_FAILED')) . '"; |
| 608 | const PLG_GPTRANSLATE_UNKNOWN_ERROR = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_UNKNOWN_ERROR')) . '"; |
| 609 | const PLG_GPTRANSLATE_NETWORK_ERROR = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_NETWORK_ERROR')) . '"; |
| 610 | const PLG_GPTRANSLATE_BULK_DELETE_CONFIRM = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_BULK_DELETE_CONFIRM')) . '"; |
| 611 | const PLG_GPTRANSLATE_BULK_DELETE_SELECT_ONE = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_BULK_DELETE_SELECT_ONE')) . '"; |
| 612 | const PLG_GPTRANSLATE_BULK_DELETE_SUCCESS = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_BULK_DELETE_SUCCESS')) . '"; |
| 613 | const PLG_GPTRANSLATE_BULK_DELETE_ERROR = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_BULK_DELETE_ERROR')) . '"; |
| 614 | const PLG_GPTRANSLATE_BULK_DELETE_NETWORK = "' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_BULK_DELETE_NETWORK')) . '"; |
| 615 | </script>'; |
| 616 | |
| 617 | |
| 618 | echo '<table class="widefat fixed striped">'; |
| 619 | echo '<thead><tr>'; |
| 620 | echo '<th style="width: 1%"><input class="form-check-input" autocomplete="off" type="checkbox" id="checkall"></th>'; |
| 621 | echo '<th style="width: 1%">ID</th>'; |
| 622 | if($opts['rewrite_language_alias'] == 1) { |
| 623 | echo '<th style="width: 18%">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LINK_PAGE')) . '</th>'; |
| 624 | echo '<th style="width: 18%">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATED_ALIAS')) . '</th>'; |
| 625 | echo '<th style="width: 18%">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_ORIGINAL_TRANSLATED')) . '</th>'; |
| 626 | } else { |
| 627 | echo '<th style="width: 25%">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LINK_PAGE')) . '</th>'; |
| 628 | echo '<th style="width: 20%">' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_ORIGINAL_TRANSLATED')) . '</th>'; |
| 629 | } |
| 630 | echo '<th>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_ORIGINAL')) . '</th>'; |
| 631 | echo '<th>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_TRANSLATED')) . '</th>'; |
| 632 | echo '<th>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_PUBLISHED_GENERIC')) . '</th>'; |
| 633 | echo '<th>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATIONS_ENGINE')) . '</th>'; |
| 634 | echo '<th>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATIONS_DATE')) . '</th>'; |
| 635 | echo '<th>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_ACTIONS')) . '</th>'; |
| 636 | echo '</tr></thead>'; |
| 637 | |
| 638 | echo '<tbody>'; |
| 639 | |
| 640 | $current_page = isset($_GET['paged']) && is_numeric($_GET['paged']) ? (int)$_GET['paged'] : 1; |
| 641 | $offset = ($current_page - 1) * $records_per_page; |
| 642 | $sql_count = "SELECT COUNT(*) FROM {$this->table_name} WHERE 1=1"; |
| 643 | |
| 644 | // Add dynamic filters |
| 645 | if (!empty($searchFilter)) { |
| 646 | $sql_count .= " AND (pagelink LIKE '%" . esc_sql($searchFilter) . "%' OR translated_alias LIKE '%" . esc_sql($searchFilter) . "%' OR translations LIKE '%" . esc_sql($searchFilter) . "%'" . " OR alt_translations LIKE '%" . esc_sql($searchFilter) . "%'" . ")"; |
| 647 | } |
| 648 | if ($publishedFilter !== '') { |
| 649 | $sql_count .= " AND published = '" . esc_sql($publishedFilter) . "'"; |
| 650 | } |
| 651 | if (!empty($languageFilter)) { |
| 652 | $sql_count .= " AND languagetranslated = '" . esc_sql($languageFilter) . "'"; |
| 653 | } |
| 654 | if (!empty($engineFilter)) { |
| 655 | $sql_count .= " AND translation_engine = '" . esc_sql($engineFilter) . "'"; |
| 656 | } |
| 657 | // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Dynamic query built with placeholders, safely prepared |
| 658 | $total_records = $wpdb->get_var($sql_count); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared,WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 659 | $total_pages = ceil($total_records / $records_per_page); |
| 660 | |
| 661 | // Load records count with filtering |
| 662 | $sql_data = "SELECT * FROM {$this->table_name} WHERE 1=1"; |
| 663 | |
| 664 | // Add dynamic filters |
| 665 | if (!empty($searchFilter)) { |
| 666 | $sql_data .= " AND (pagelink LIKE '%" . esc_sql($searchFilter) . "%' OR translated_alias LIKE '%" . esc_sql($searchFilter) . "%' OR translations LIKE '%" . esc_sql($searchFilter) . "%'" . " OR alt_translations LIKE '%" . esc_sql($searchFilter) . "%'" . ")"; |
| 667 | } |
| 668 | if ($publishedFilter !== '') { |
| 669 | $sql_data .= " AND published = '" . esc_sql($publishedFilter) . "'"; |
| 670 | } |
| 671 | if (!empty($languageFilter)) { |
| 672 | $sql_data .= " AND languagetranslated = '" . esc_sql($languageFilter) . "'"; |
| 673 | } |
| 674 | if (!empty($engineFilter)) { |
| 675 | $sql_data .= " AND translation_engine = '" . esc_sql($engineFilter) . "'"; |
| 676 | } |
| 677 | $sql_data .= " ORDER BY translate_date DESC LIMIT $records_per_page OFFSET $offset"; |
| 678 | |
| 679 | // Load records with filtering and pagination |
| 680 | // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Dynamic query built with placeholders, safely prepared |
| 681 | $records = $wpdb->get_results($sql_data); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared,WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 682 | |
| 683 | // Message for no translations |
| 684 | if (!count($records) && stripos($sql_data, "AND") === false) { |
| 685 | echo '<div class="notice notice-success is-dismissible" id="notranslations-notice">'; |
| 686 | echo '<p>' . esc_html($this->loadTranslations('PLG_GPTRANSLATE_NO_TRANSLATIONS')) . '</p>'; |
| 687 | echo '</div>'; |
| 688 | } |
| 689 | |
| 690 | foreach ( $records as $r ) { |
| 691 | // Build a short summary of the translations |
| 692 | $tr = json_decode( $r->translations, true ); |
| 693 | if ( is_array( $tr ) ) { |
| 694 | $pairs = array_map( |
| 695 | function( $k, $v ) { |
| 696 | return esc_html( $k ) . ' → ' . esc_html( $v ); |
| 697 | }, |
| 698 | array_keys( $tr ), |
| 699 | array_values( $tr ) |
| 700 | ); |
| 701 | $short = implode( ', ', $pairs ); |
| 702 | $short = mb_substr( $short, 0, 80 ) . ( mb_strlen( $short ) > 80 ? '…' : '' ); |
| 703 | } else { |
| 704 | $short = ''; |
| 705 | } |
| 706 | |
| 707 | // Escape all output |
| 708 | $id = (int) $r->id; |
| 709 | $link = wp_nonce_url( admin_url( "admin.php?page=gptranslate&action=edit&edit={$id}" ), 'gptranslate_edit_' . $id, '_gptranslate_nonce' ); |
| 710 | $deleteLink = wp_nonce_url( admin_url( "admin.php?page=gptranslate&action=delete_translation&translation_id={$id}" ), 'gptranslate_delete_' . $id, '_gptranslate_nonce' ); |
| 711 | $origLang = esc_html( strtoupper( $r->languageoriginal ) ); |
| 712 | $transLang = esc_html( strtoupper( $r->languagetranslated ) ); |
| 713 | $pub = $r->published ? esc_html($this->loadTranslations('PLG_GPTRANSLATE_YES')) : esc_html($this->loadTranslations('PLG_GPTRANSLATE_NO')); |
| 714 | $engine = esc_html( $r->translation_engine ); |
| 715 | $date = esc_html( $r->translate_date ); |
| 716 | |
| 717 | $langOriginal = esc_attr($r->languageoriginal); |
| 718 | |
| 719 | // Path relativo o assoluto all'immagine della bandiera |
| 720 | $flagUrlOriginal = plugins_url('flags/svg/' . $r->languageoriginal . '.svg', __FILE__); |
| 721 | $flagOriginal = '<img src="' . esc_url($flagUrlOriginal) . '" alt="flag" style="width: 16px; vertical-align:middle; margin-right:4px;">'; // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 722 | $flagUrlTranslated = plugins_url('flags/svg/' . $r->languagetranslated . '.svg', __FILE__); |
| 723 | $flagTranslated = '<img src="' . esc_url($flagUrlTranslated) . '" alt="flag" style="width: 16px; vertical-align:middle; margin-right:4px;">'; // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 724 | |
| 725 | $togglePublishedUrl = wp_nonce_url( |
| 726 | admin_url("admin.php?page=gptranslate&action=toggle_published&translation_id={$id}"), |
| 727 | 'gptranslate_toggle_' . $id, |
| 728 | '_gptranslate_nonce' |
| 729 | ); |
| 730 | |
| 731 | $pubIcon = $r->published ? '<img src="' . plugins_url('assets/images/published.png', __FILE__) . '" alt="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_PUBLISHED_GENERIC')) . '" title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_UNPUBLISH')) . '">' // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 732 | : '<img src="' . plugins_url('assets/images/unpublished.png', __FILE__) . '" alt="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_UTRANSLATIONS_SHORT_CHART')) . '" title="' . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_PUBLISH')) . '">'; // phpcs:ignore PluginCheck.CodeAnalysis.ImageFunctions.NonEnqueuedImage |
| 733 | |
| 734 | |
| 735 | $pub = $r->published ? "<a href='{$togglePublishedUrl}' class='gpt-toggle gpt-published' title='" . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_UNPUBLISH')) . "'>" . $pubIcon . "</a>" |
| 736 | : "<a href='{$togglePublishedUrl}' class='gpt-toggle gpt-unpublished' title='" . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_PUBLISH')) . "'>" . $pubIcon . "</a>"; |
| 737 | |
| 738 | $local_date = get_date_from_gmt($date); |
| 739 | |
| 740 | echo '<tr>'; |
| 741 | echo "<td style='width: 1%'><input class='form-check-input' autocomplete='off' type='checkbox' id='cb0' name='gptid[]' value='" . esc_attr($r->id) . "'></td>"; |
| 742 | echo "<td style='width: 1%'>". esc_html($r->id) . "</td>"; |
| 743 | echo "<td><a href='" . esc_attr( $link ) . "'><span class='icon-style' aria-hidden='true'>📝</span>" . esc_html( $r->pagelink ) . "</a><a class='doublesize-icon' href='" . esc_attr( $r->pagelink ) . "' target='_blank'>↗</a></td>"; |
| 744 | if($opts['rewrite_language_alias'] == 1) { |
| 745 | echo $r->translated_alias ? "<td><a href='" . esc_attr( $r->translated_alias ) . "'>" . esc_html( $r->translated_alias ) . "</a><a class='doublesize-icon' href='" . esc_attr( $r->translated_alias ) . "' target='_blank'>↗</a></td>" : '<td>-</td>'; |
| 746 | } |
| 747 | echo "<td>" . esc_html($short) . "</td>"; |
| 748 | echo "<td>" . wp_kses_post($flagOriginal) . " " . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_NAME_' . strtoupper($langOriginal))) . "</td>"; |
| 749 | echo "<td>" . wp_kses_post($flagTranslated) . " " . esc_html($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_NAME_' . strtoupper(esc_attr($r->languagetranslated)))) . "</td>"; |
| 750 | echo "<td>" . wp_kses_post($pub) . "</td>"; |
| 751 | echo "<td><span class='gpt-label'>" . esc_html($this->loadTranslations('PLG_GPTRANSLATE_CHATGPT_TRANSLATION_ENGINE_' . strtoupper($engine) . '_ENGINE')) . "</span></td>"; |
| 752 | echo "<td>" . esc_html( date_i18n('l, d F Y \a\t H:i', strtotime($local_date)) ) . "</td>"; |
| 753 | echo "<td><a href='" . esc_attr($link) . "'>" . esc_html($this->loadTranslations('PLG_GPTRANSLATE_EDIT')) . "</a> | <a href='" . esc_attr($deleteLink) . "' onclick=\"return confirm('" . esc_attr($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATION_DELETE_CONFIRM')) . "');\">" . esc_html($this->loadTranslations('PLG_GPTRANSLATE_TRANSLATION_DELETE')) . "</a></td>"; |
| 754 | echo '</tr>'; |
| 755 | } |
| 756 | echo '</tbody>'; |
| 757 | echo '</table>'; |
| 758 | |
| 759 | if ($total_pages > 1) { |
| 760 | echo '<div class="tablenav"><div class="tablenav-pages">'; |
| 761 | for ($i = 1; $i <= $total_pages; $i++) { |
| 762 | $url = add_query_arg(array_merge($_GET, ['paged' => $i]), admin_url('admin.php')); |
| 763 | $class = ($i == $current_page) ? "class='current-page button'" : "class='button'"; |
| 764 | echo "<a " . wp_kses_post($class) . " href='" . esc_url($url) . "'>" . esc_html($i) . "</a> "; |
| 765 | } |
| 766 | echo '</div></div>'; |
| 767 | } |
| 768 | } |
| 769 | |
| 770 | echo '</div>'; |
| 771 | echo '</div>'; |
| 772 | } |
| 773 | |
| 774 | /** |
| 775 | * Load language file and translations |
| 776 | * @return array |
| 777 | */ |
| 778 | public function loadTranslations($key) { |
| 779 | // Text translations |
| 780 | static $adminLanguageStrings = null; |
| 781 | |
| 782 | if(!$adminLanguageStrings) { |
| 783 | $adminLanguageFile = dirname(__FILE__) . "/language/en-GB/gptranslate.ini"; |
| 784 | if(file_exists($adminLanguageFile)) { |
| 785 | $adminLanguageStrings = parse_ini_file($adminLanguageFile, false, INI_SCANNER_NORMAL); |
| 786 | } |
| 787 | } |
| 788 | |
| 789 | if(array_key_exists($key, $adminLanguageStrings)) { |
| 790 | return $adminLanguageStrings[$key]; |
| 791 | } |
| 792 | |
| 793 | return $key; |
| 794 | } |
| 795 | |
| 796 | /** |
| 797 | * Load language file and translations |
| 798 | * @return array |
| 799 | */ |
| 800 | public static function loadTranslation($key) { |
| 801 | // Text translations |
| 802 | static $adminLanguageStrings = null; |
| 803 | |
| 804 | if(!$adminLanguageStrings) { |
| 805 | $adminLanguageFile = dirname(__FILE__) . "/language/en-GB/gptranslate.ini"; |
| 806 | if(file_exists($adminLanguageFile)) { |
| 807 | $adminLanguageStrings = parse_ini_file($adminLanguageFile, false, INI_SCANNER_NORMAL); |
| 808 | } |
| 809 | } |
| 810 | |
| 811 | if(array_key_exists($key, $adminLanguageStrings)) { |
| 812 | return $adminLanguageStrings[$key]; |
| 813 | } |
| 814 | |
| 815 | return $key; |
| 816 | } |
| 817 | |
| 818 | /** |
| 819 | * Save translation record |
| 820 | * |
| 821 | * @access public |
| 822 | */ |
| 823 | public function save_record() { |
| 824 | global $wpdb; |
| 825 | |
| 826 | // Retrieve and sanitize basic inputs |
| 827 | $id = isset ( $_POST ['id'] ) ? intval ( $_POST ['id'] ) : 0; |
| 828 | $formAction = isset ( $_POST ['action'] ) ? sanitize_key ( $_POST ['action'] ) : ''; |
| 829 | |
| 830 | if ( !isset($_POST['_gptranslate_nonce']) || !wp_verify_nonce(sanitize_text_field(wp_unslash($_POST['_gptranslate_nonce'])), 'gptranslate_save_record_action') ) { |
| 831 | wp_die(esc_html($this->loadTranslations('PLG_GPTRANSLATE_GENERIC_SECURITY_ERROR')), 'gptranslate'); |
| 832 | } |
| 833 | |
| 834 | // Handle cancel action |
| 835 | if ($formAction === 'cancel_gptranslate_record') { |
| 836 | wp_redirect ( admin_url ( 'admin.php?page=gptranslate' ) ); |
| 837 | exit (); |
| 838 | } |
| 839 | |
| 840 | // Sanitize pagelink |
| 841 | $pagelink = isset ( $_POST ['pagelink'] ) ? sanitize_text_field ( wp_unslash ( $_POST ['pagelink'] ) ) : ''; |
| 842 | |
| 843 | // Sanitize translated_alias |
| 844 | $translatedAlias = isset ( $_POST ['translated_alias'] ) ? sanitize_text_field ( wp_unslash ( $_POST ['translated_alias'] ) ) : ''; |
| 845 | |
| 846 | // Process and sanitize translations JSON |
| 847 | $raw_translations = filter_input( INPUT_POST, 'translations_json', FILTER_UNSAFE_RAW ); |
| 848 | $raw_translations = is_string($raw_translations) ? $raw_translations : '[]'; |
| 849 | |
| 850 | $decoded_translations = json_decode ( $raw_translations, true ); |
| 851 | if (! is_array ( $decoded_translations )) { |
| 852 | wp_die ( esc_html($this->loadTranslations('PLG_GPTRANSLATE_INVALID_JSON_TRANSLATIONS')), esc_html($this->loadTranslations('PLG_GPTRANSLATE_GENERIC_ERROR')), [ |
| 853 | 'response' => 400 |
| 854 | ] ); |
| 855 | } |
| 856 | $clean_translations = $decoded_translations; |
| 857 | $sanitized_translations_json = wp_json_encode ( $clean_translations ); |
| 858 | |
| 859 | // Process and sanitize alternative translations JSON |
| 860 | $raw_alt = filter_input( INPUT_POST, 'alt_translations_json', FILTER_UNSAFE_RAW ); |
| 861 | $raw_alt = is_string($raw_translations) ? $raw_alt : '[]'; |
| 862 | |
| 863 | $decoded_alt = json_decode ( $raw_alt, true ); |
| 864 | if (! is_array ( $decoded_alt )) { |
| 865 | wp_die ( esc_html($this->loadTranslations('PLG_GPTRANSLATE_INVALID_JSON_ALTTRANSLATIONS')), esc_html($this->loadTranslations('PLG_GPTRANSLATE_GENERIC_ERROR')), [ |
| 866 | 'response' => 400 |
| 867 | ] ); |
| 868 | } |
| 869 | $clean_alt = $decoded_alt; |
| 870 | $sanitized_alt_json = wp_json_encode ( $clean_alt ); |
| 871 | |
| 872 | // Update database record |
| 873 | $wpdb->update ( $this->table_name, [ // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 874 | 'pagelink' => $pagelink, |
| 875 | 'translated_alias' => $translatedAlias, |
| 876 | 'translations' => $sanitized_translations_json, |
| 877 | 'alt_translations' => $sanitized_alt_json |
| 878 | ], [ |
| 879 | 'id' => $id |
| 880 | ], [ |
| 881 | '%s', |
| 882 | '%s', |
| 883 | '%s' |
| 884 | ], [ |
| 885 | '%d' |
| 886 | ] ); |
| 887 | |
| 888 | // Redirect based on action |
| 889 | if ($formAction === 'save_gptranslate_record_and_close') { |
| 890 | wp_redirect ( admin_url ( 'admin.php?page=gptranslate' ) ); |
| 891 | } else { |
| 892 | $url = admin_url( 'admin.php?page=gptranslate&action=edit&edit=' . $id ); |
| 893 | $url = wp_nonce_url( $url, 'gptranslate_edit_' . $id, '_gptranslate_nonce' ); |
| 894 | wp_redirect( html_entity_decode( $url ) ); |
| 895 | } |
| 896 | exit (); |
| 897 | } |
| 898 | |
| 899 | public function gptranslate_handle_deletion() { |
| 900 | // 1) Verify nonce |
| 901 | $id = isset( $_GET['translation_id'] ) ? intval( $_GET['translation_id'] ) : 0; |
| 902 | $nonce = isset( $_GET['_gptranslate_nonce'] ) ? wp_unslash( $_GET['_gptranslate_nonce'] ) : ''; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized |
| 903 | |
| 904 | if ( ! wp_verify_nonce( $nonce, 'gptranslate_delete_' . $id ) ) { |
| 905 | wp_die( esc_html($this->loadTranslations('PLG_GPTRANSLATE_GENERIC_SECURITY_ERROR')), 'Error', [ 'response' => 403 ] ); |
| 906 | } |
| 907 | |
| 908 | // 2) Delete the row |
| 909 | global $wpdb; |
| 910 | $table = $wpdb->prefix . 'gptranslate'; |
| 911 | $deleted = $wpdb->delete( $table, [ 'id' => $id ], [ '%d' ] ); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 912 | |
| 913 | // 3) Redirect back with a message |
| 914 | $redirect_url = add_query_arg( |
| 915 | [ |
| 916 | 'page' => 'gptranslate', |
| 917 | 'deleted' => $deleted ? '1' : '0', |
| 918 | ], |
| 919 | admin_url( 'admin.php' ) |
| 920 | ); |
| 921 | wp_redirect( $redirect_url ); |
| 922 | exit; |
| 923 | } |
| 924 | |
| 925 | |
| 926 | /** |
| 927 | * Add main app frontend script |
| 928 | * |
| 929 | * @access public |
| 930 | */ |
| 931 | public function enqueue_frontend_scripts() { |
| 932 | add_filter('script_loader_tag', function($tag, $handle) { |
| 933 | if ($handle === 'gptranslate-responsivevoice') { |
| 934 | return str_replace('<script ', '<script defer ', $tag); |
| 935 | } |
| 936 | |
| 937 | if ($handle === 'gptranslate-jsonrepair') { |
| 938 | return str_replace('<script ', '<script type="module" ', $tag); |
| 939 | } |
| 940 | |
| 941 | if ($handle === 'gptranslate-bstoast') { |
| 942 | return str_replace('<script ', '<script type="module" ', $tag); |
| 943 | } |
| 944 | if ($handle === 'gptranslate-main') { |
| 945 | // 1) prendi i valori raw |
| 946 | $raw_uri = isset($_SERVER['REQUEST_URI']) ? sanitize_text_field(wp_unslash($_SERVER['REQUEST_URI'])) : ''; |
| 947 | $raw_host = isset($_SERVER['HTTP_HOST']) ? sanitize_text_field(wp_unslash($_SERVER['HTTP_HOST'])) : ''; |
| 948 | |
| 949 | // 2) unslash WP |
| 950 | $unslashed_uri = wp_unslash ( $raw_uri ); |
| 951 | $unslashed_host = wp_unslash ( $raw_host ); |
| 952 | |
| 953 | // 3) sanitizza URI come URL |
| 954 | $orig_url = esc_url_raw ( $unslashed_uri ); |
| 955 | // - accetta path e query, rimuove caratteri pericolosi |
| 956 | |
| 957 | // 4) sanitizza host |
| 958 | // a) rimuovi tag e control chars |
| 959 | $host_clean = sanitize_text_field ( $unslashed_host ); |
| 960 | // b) mantieni solo [a-z0-9.-] per sicurezza |
| 961 | $orig_domain = preg_replace ( '/[^a-z0-9.-]/i', '', $host_clean ); |
| 962 | |
| 963 | // 5) infine escape per attributo HTML |
| 964 | return str_replace ( '<script ', sprintf ( '<script type="module" data-gt-orig-url="%s" data-gt-orig-domain="%s" data-gt-widget-id="1" ', esc_attr ( $orig_url ), esc_attr ( $orig_domain ) ), $tag ); |
| 965 | } |
| 966 | |
| 967 | return $tag; |
| 968 | }, 10, 2); |
| 969 | |
| 970 | $settings = get_option("gptranslate_options"); |
| 971 | |
| 972 | // Move default language to the first one in the list |
| 973 | if($settings ['default_language_first']) { |
| 974 | $defaultLanguageKeyIndex = array_search($settings ['language'], $settings ['languages']); |
| 975 | if ($defaultLanguageKeyIndex !== false) { |
| 976 | // Remove the 'de' element from its current position |
| 977 | $defaultLanguage = $settings ['languages'][$defaultLanguageKeyIndex]; |
| 978 | unset($settings ['languages'][$defaultLanguageKeyIndex]); |
| 979 | |
| 980 | // Re-index the array to maintain numerical indexes |
| 981 | $settings ['languages'] = array_values($settings ['languages']); |
| 982 | |
| 983 | // Add 'de' to the beginning of the array |
| 984 | array_unshift($settings ['languages'], $defaultLanguage); |
| 985 | } |
| 986 | } |
| 987 | |
| 988 | // build alt_flags array |
| 989 | $alt_flags = array (); |
| 990 | $raw_alt_flags = isset($settings ['alt_flags']) ? $settings ['alt_flags'] : []; |
| 991 | foreach ( $raw_alt_flags as $country ) { |
| 992 | if ($country == 'usa' or $country == 'canada') |
| 993 | $alt_flags ['en'] = $country; |
| 994 | elseif ($country == 'brazil') |
| 995 | $alt_flags ['pt'] = $country; |
| 996 | elseif ($country == 'mexico' or $country == 'argentina' or $country == 'colombia') |
| 997 | $alt_flags ['es'] = $country; |
| 998 | elseif ($country == 'quebec') |
| 999 | $alt_flags ['fr'] = $country; |
| 1000 | } |
| 1001 | |
| 1002 | // Build float position variables |
| 1003 | $float_position = $settings ['float_position']; |
| 1004 | if($float_position != 'inline'){ |
| 1005 | list ( $switcher_vertical_position, $switcher_horizontal_position ) = explode ( '-', $float_position ); |
| 1006 | } else { |
| 1007 | list ( $switcher_vertical_position, $switcher_horizontal_position ) = ['inline', 'inline']; |
| 1008 | } |
| 1009 | |
| 1010 | // Set local flags path |
| 1011 | $flagsPath = trailingslashit(plugins_url('flags', __FILE__)); |
| 1012 | |
| 1013 | wp_register_script('gptranslate-main-inline', '', [], $this->version, true); |
| 1014 | wp_enqueue_script('gptranslate-main-inline'); |
| 1015 | |
| 1016 | // Example: $settings is an array like in Joomla |
| 1017 | $ajaxEndpoint = esc_url_raw(rest_url('gptranslate/v1/request')); |
| 1018 | $base64Encode = 'base' . 64 . '_encode'; |
| 1019 | |
| 1020 | $inlineScript = 'var gptServerSideLink = "' . esc_js($ajaxEndpoint) . '"; |
| 1021 | var gptApiKey = "' . esc_js(hash( 'sha256', get_site_url() )) . '"; |
| 1022 | var gptLiveSite = "' . esc_js(get_site_url()) . '"; |
| 1023 | var gptStorage = ' . ($settings['storage_type'] === 'session' ? 'window.sessionStorage' : 'window.localStorage') . '; |
| 1024 | var gptMaxTranslationsPerRequest = ' . (int)$settings['max_translations_per_request'] . '; |
| 1025 | var maxCharactersPerRequest = ' . (int)$settings['max_characters_per_request'] . '; |
| 1026 | var gptRewriteLanguageUrl = ' . (int)$settings['rewrite_language_url'] . '; |
| 1027 | var gptRewriteLanguageAlias = ' . (int)$settings['rewrite_language_alias'] . '; |
| 1028 | var gptRewriteLanguageAliasOriginalLanguage = ' . (int)$settings['rewrite_language_alias_original_language'] . '; |
| 1029 | var gptAutoSetLanguageDirection = ' . (int)$settings['auto_set_language_direction'] . '; |
| 1030 | var gptServersideTranslations = ' . (int)$settings['serverside_translations'] . '; |
| 1031 | var gptRewritePageLinks = ' . (int)$settings['rewrite_page_links'] . '; |
| 1032 | var gptTranslateMetadata = ' . (int)$settings['translate_metadata'] . '; |
| 1033 | var gptTranslatePlaceholders = ' . (int)$settings['translate_placeholders'] . '; |
| 1034 | var gptTranslateAltImages = ' . (int)$settings['translate_altimages'] . '; |
| 1035 | var chatgptClassesAltimagesExcluded = "' . esc_js(str_ireplace('"', '', $settings['css_selector_classes_translate_altimages_excluded'])) . '"; |
| 1036 | var gptTranslateTitles = ' . (int)$settings['translate_titles'] . '; |
| 1037 | var gptTranslateValues = ' . (int)$settings['translate_values'] . '; |
| 1038 | var gptSetHtmlLang = ' . (int)$settings['set_html_lang'] . '; |
| 1039 | var gptAddCanonical = ' . (int)$settings['add_canonical'] . '; |
| 1040 | var gptAddAlternate = ' . (int)$settings['add_alternate'] . '; |
| 1041 | var gptSubfolderInstallation = ' . (int)$settings['subfolder_installation'] . '; |
| 1042 | var gptIgnoreQuerystring = ' . (int)$settings['ignore_querystring'] . '; |
| 1043 | var gptChatgptGtranslateRequestDelay = ' . (int)$settings['chatgpt_gtranslate_request_delay'] . '; |
| 1044 | var gptInitialTranslationDelay = ' . (int)$settings['initial_translation_delay'] . '; |
| 1045 | var chatgptApiKey = "' . esc_js($base64Encode($settings['chatgpt_apikey'])) . '"; |
| 1046 | var chatgptApiModel = "' . esc_js($settings['chatgpt_model']) . '"; |
| 1047 | var chatgptRequestMessage = "' . str_ireplace("\'", "'", esc_js(str_ireplace(['"' , "\r", "\n"], ['' , ' ', ' '], $settings['chatgpt_request_message']))) . '"; |
| 1048 | var chatgptRequestConversationMode = "' . esc_js($settings['chatgpt_request_conversation_mode']) . '"; |
| 1049 | var chatgptEnableReader = ' . (int)$settings['enable_reader'] . '; |
| 1050 | var chatgptResponsivevoiceLanguageGender = "' . esc_js($settings['responsivevoice_language_gender']) . '"; |
| 1051 | var chatgptResponsivevoiceApiKey = "' . esc_js($settings['responsivevoice_apikey']) . '"; |
| 1052 | var chatgptResponsivevoiceReadingMode = "' . esc_js($settings['proxy_responsive_reading_mode']) . '"; |
| 1053 | var chatgptChunksize = "' . esc_js($settings['chunksize']) . '"; |
| 1054 | var chatgptCssSelectorLeafnodesExcluded = "' . esc_js(trim(preg_replace('/,+/', ',', str_ireplace(["\r", "\n"], ",", $settings['css_selector_leafnodes_excluded'])), ',')) . '"; |
| 1055 | var chatgptWordsLeafnodesExcluded = "' . esc_js(rtrim($settings['words_leafnodes_excluded'], ', ')) . '"; |
| 1056 | var chatgptWordsMinLength = "' . (int)$settings['words_min_length'] . '"; |
| 1057 | var chatgptMainpageSelector = "' . esc_js($settings['mainpage_selector']) . '"; |
| 1058 | var chatgptElementsToExcludeCustom = "' . esc_js(trim($settings['elements_toexclude_custom'])) . '"; |
| 1059 | var chatgptPopupFontsize = ' . (int)$settings['popup_fontsize'] . '; |
| 1060 | var chatgptDraggableWidget = ' . (int)$settings['draggable_widget'] . '; |
| 1061 | var gptAudioVolume = ' . (float)$settings['responsivevoice_volume_tts'] . '; |
| 1062 | var gptVoiceSpeed = "' . esc_js($settings['responsivevoice_voice_speed']) . '"; |
| 1063 | var gTranslateEngine = ' . (($settings['google_translate_engine'] == 1 || !trim($settings['chatgpt_apikey'])) ? 1 : 0) . '; |
| 1064 | var translateEngineValue = "' . esc_js($settings['google_translate_engine']) . '"; |
| 1065 | var gptDisableControl = ' . (int)$settings['disable_control'] . '; |
| 1066 | var gptVersionNumeric = ' . 0 . '; |
| 1067 | var svgIconArrow = \'<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 285 285"><path d="M282 76.5l-14.2-14.3a9 9 0 0 0-13.1 0L142.5 174.4 30.3 62.2a9 9 0 0 0-13.2 0L3 76.5a9 9 0 0 0 0 13.1l133 133a9 9 0 0 0 13.1 0l133-133a9 9 0 0 0 0-13z" style="fill:#' . ltrim($settings['widget_text_color'], '#') . '"/></svg>\';'; |
| 1068 | |
| 1069 | // Inject it AFTER gptranslate-main-inline |
| 1070 | wp_add_inline_script('gptranslate-main-inline', $inlineScript); |
| 1071 | |
| 1072 | wp_register_script('gptranslate-js-specs', '', [], $this->version, true); |
| 1073 | wp_enqueue_script('gptranslate-js-specs'); |
| 1074 | wp_add_inline_script('gptranslate-js-specs', 'window.gptranslateSettings = window.gptranslateSettings || {}; |
| 1075 | window.gptranslateSettings["1"] = { |
| 1076 | "default_language": "' . $settings['language'] . '", |
| 1077 | "languages": ' . json_encode($settings['languages']) . ', |
| 1078 | "wrapper_selector": "' . $settings['wrapper_selector'] . '", |
| 1079 | "float_switcher_open_direction": "' . $settings['float_switcher_open_direction'] . '", |
| 1080 | "detect_browser_language": ' . (int)$settings['detect_browser_language'] . ', |
| 1081 | "detect_current_language": ' . (int)$settings['detect_current_language'] . ', |
| 1082 | "detect_default_language": ' . (int)$settings['detect_default_language'] . ', |
| 1083 | "autotranslate_detected_language": ' . (int)$settings['autotranslate_detected_language'] . ', |
| 1084 | "always_detect_autotranslated_language": ' . (int)$settings['always_detect_autotranslated_language'] . ', |
| 1085 | "widget_text_color": "' . $settings['widget_text_color'] . '", |
| 1086 | "show_language_titles": ' . (int)$settings['show_language_titles'] . ', |
| 1087 | "enable_dropdown": ' . (int)$settings['enable_dropdown'] . ', |
| 1088 | "equal_widths": ' . (int)$settings['equal_widths'] . ', |
| 1089 | "reader_button_position": "' . $settings['reader_button_position'] . '", |
| 1090 | "custom_css": "' . $settings['custom_css'] . '", |
| 1091 | "alt_flags": ' . json_encode($alt_flags). ', |
| 1092 | "switcher_horizontal_position": "' . $switcher_horizontal_position . '", |
| 1093 | "switcher_vertical_position": "' . $switcher_vertical_position . '", |
| 1094 | "flags_location": "' . esc_js($flagsPath) . '", |
| 1095 | "flag_loading": "' . $settings['flag_loading'] . '", |
| 1096 | "flag_style": "' . $settings['flag_style'] . '", |
| 1097 | "widget_max_height": ' . (int)$settings['widget_max_height'] . ' |
| 1098 | };'); |
| 1099 | |
| 1100 | $languageStringsScript = ''; |
| 1101 | |
| 1102 | // Generic translations |
| 1103 | $labels = [ |
| 1104 | 'TRANSLATING', |
| 1105 | 'TRANSLATING_WAIT', |
| 1106 | 'TRANSLATING_COMPLETE', |
| 1107 | 'READING_INPROGRESS', |
| 1108 | 'READING_END', |
| 1109 | 'READING_EMPTY' |
| 1110 | ]; |
| 1111 | |
| 1112 | foreach ($labels as $label) { |
| 1113 | $languageStringsScript .= 'var PLG_GPTRANSLATE_' . $label . '="' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_' . $label)) . '";' . PHP_EOL; |
| 1114 | } |
| 1115 | |
| 1116 | $languages = [ |
| 1117 | 'AF', 'SQ', 'AM', 'AR', 'HY', 'AZ', 'EU', 'BE', 'BN', 'BS', 'BG', 'CA', 'CEB', 'NY', |
| 1118 | 'ZH', 'CO', 'HR', 'CS', 'DA', 'NL', 'EN', 'EO', 'ET', 'TL', 'FI', 'FR', |
| 1119 | 'FY', 'GL', 'KA', 'DE', 'EL', 'GU', 'HT', 'HA', 'HAW', 'IW', 'HI', 'HMN', 'HU', |
| 1120 | 'IS', 'IG', 'ID', 'GA', 'IT', 'JA', 'JW', 'KN', 'KK', 'KM', 'KO', 'KU', 'KY', 'LO', |
| 1121 | 'LA', 'LV', 'LT', 'LB', 'MK', 'MG', 'MS', 'ML', 'MT', 'MI', 'MR', 'MN', 'MY', 'NE', |
| 1122 | 'NO', 'PS', 'FA', 'PL', 'PT', 'PA', 'RO', 'RU', 'SM', 'GD', 'SR', 'ST', 'SN', 'SD', |
| 1123 | 'SI', 'SK', 'SL', 'SO', 'ES', 'SU', 'SW', 'SV', 'TG', 'TA', 'TE', 'TH', 'TR', 'UK', |
| 1124 | 'UR', 'UZ', 'VI', 'CY', 'XH', 'YI', 'YO', 'ZU' |
| 1125 | ]; |
| 1126 | |
| 1127 | |
| 1128 | foreach ($languages as $lang) { |
| 1129 | $languageStringsScript .= 'var PLG_GPTRANSLATE_LANGUAGE_NAME_' . $lang . '="' . esc_js($this->loadTranslations('PLG_GPTRANSLATE_LANGUAGE_NAME_' . $lang)) . '";' . PHP_EOL; |
| 1130 | } |
| 1131 | |
| 1132 | wp_register_script('gptranslate-js-language-strings', '', [], $this->version, true); |
| 1133 | wp_enqueue_script('gptranslate-js-language-strings'); |
| 1134 | wp_add_inline_script('gptranslate-js-language-strings', $languageStringsScript); |
| 1135 | |
| 1136 | |
| 1137 | // Dictionary |
| 1138 | $words_leafnodes_excluded_bylanguage_repeatable = $settings['words_leafnodes_excluded_bylanguage_repeatable']; |
| 1139 | if ($words_leafnodes_excluded_bylanguage_repeatable) { |
| 1140 | if (is_string($words_leafnodes_excluded_bylanguage_repeatable)) { |
| 1141 | $words_leafnodes_excluded_bylanguage_repeatable = json_decode($words_leafnodes_excluded_bylanguage_repeatable, true); |
| 1142 | } |
| 1143 | |
| 1144 | // Ora convertiamo l'array normale nel formato con chiavi tipo words_leafnodes_excluded_bylanguage_repeatable0, 1, 2... |
| 1145 | $formatted = []; |
| 1146 | foreach ($words_leafnodes_excluded_bylanguage_repeatable as $index => $row) { |
| 1147 | $formatted["words_leafnodes_excluded_bylanguage_repeatable{$index}"] = [ |
| 1148 | 'words_leafnodes_excluded_bylanguage' => $row['word'] ?? '', |
| 1149 | 'words_leafnodes_excluded_bylanguage_language_original' => $row['langOriginal'] ?? '*', |
| 1150 | 'words_leafnodes_excluded_bylanguage_language_target' => $row['langTranslated'] ?? '*', |
| 1151 | 'words_leafnodes_excluded_bylanguage_translation' => $row['optionalTranslation'] ?? '' |
| 1152 | ]; |
| 1153 | } |
| 1154 | |
| 1155 | // Correctly formatted JSON encode |
| 1156 | $formatted_json = json_encode($formatted, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES); |
| 1157 | |
| 1158 | // Inietto i dati PRIMA che venga eseguito gptranslate.js |
| 1159 | wp_register_script('gptranslate-js-word-leafones-excluded-language', '', [], $this->version, true); |
| 1160 | wp_enqueue_script('gptranslate-js-word-leafones-excluded-language'); |
| 1161 | wp_add_inline_script( |
| 1162 | 'gptranslate-js-word-leafones-excluded-language', |
| 1163 | 'var chatgptWordsLeafnodesExcludedByLanguage = ' . $formatted_json . ';' |
| 1164 | ); |
| 1165 | } |
| 1166 | |
| 1167 | // Local or remote script |
| 1168 | if($settings['proxy_responsive_loading_script'] == 1) { |
| 1169 | wp_enqueue_script('gptranslate-responsivevoice', plugin_dir_url(__FILE__) . 'assets/js/responsivevoice.js', [], $this->version, true); |
| 1170 | } else { |
| 1171 | wp_enqueue_script('gptranslate-responsivevoice', 'https://code.responsivevoice.org/responsivevoice.js?key=' . $settings ['responsivevoice_apikey'], [], $this->version, true); |
| 1172 | } |
| 1173 | |
| 1174 | wp_enqueue_script('gptranslate-jsonrepair', plugin_dir_url(__FILE__) . 'assets/js/jsonrepair/index.js', [], $this->version, true); |
| 1175 | wp_enqueue_script('gptranslate-main', plugin_dir_url(__FILE__) . 'assets/js/gptranslate.js', [], $this->version, true); |
| 1176 | |
| 1177 | // Enqueue Bootstrap component |
| 1178 | if(!$settings['disable_bootstrap_css']) { |
| 1179 | wp_enqueue_script('gptranslate-bstoast', plugin_dir_url(__FILE__) . 'assets/js/toast.min.js', [], $this->version, true); |
| 1180 | wp_enqueue_style( |
| 1181 | 'bootstrap-css', |
| 1182 | plugin_dir_url(__FILE__) . 'assets/css/bootstrap.min.css', |
| 1183 | [], |
| 1184 | '5.3.2' |
| 1185 | ); |
| 1186 | } |
| 1187 | |
| 1188 | // Registra un handle CSS vuoto se necessario |
| 1189 | wp_register_style('gptranslate-dynamic-css', false, [], $this->version); |
| 1190 | wp_enqueue_style('gptranslate-dynamic-css'); |
| 1191 | |
| 1192 | // Prepara lo stile dinamico |
| 1193 | $dynamic_css = 'div.gpt_float_switcher .gt-selected, div.gpt_float_switcher, div.gpt_options { background-color: ' . (!empty($settings['widget_background_color']) ? esc_attr($settings['widget_background_color']) : '#FFFFFF') . '; }' . |
| 1194 | 'div.gpt_float_switcher, div.gpt_float_switcher div.gt-selected div.gpt-current-lang, div.gpt_float_switcher div.gpt_options a { color: ' . (!empty($settings['widget_text_color']) ? esc_attr($settings['widget_text_color']) : '#000000') . '; font-size: ' . intval($settings['popup_fontsize']) . 'px; }' . |
| 1195 | 'div.gpt_float_switcher { border-radius: ' . intval($settings['popup_border_radius']) . 'px; }' . |
| 1196 | 'div.gpt_float_switcher img, svg.svg-inline--fa { box-sizing: border-box; width: ' . intval($settings['popup_iconsize']) . 'px; }'; |
| 1197 | |
| 1198 | // Inietta il CSS inline |
| 1199 | wp_add_inline_style('gptranslate-dynamic-css', $dynamic_css); |
| 1200 | } |
| 1201 | } |
| 1202 | |
| 1203 | /** |
| 1204 | * Global function to add links to WP |
| 1205 | * |
| 1206 | * @access public |
| 1207 | */ |
| 1208 | add_filter('plugin_action_links_' . plugin_basename(__FILE__), function($links) { |
| 1209 | $settings_link = '<a href="admin.php?page=gptranslate">' . esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_SETTINGS_MENU_TITLE')) . '</a>'; |
| 1210 | array_unshift($links, $settings_link); |
| 1211 | return $links; |
| 1212 | }); |
| 1213 | |
| 1214 | /** |
| 1215 | * Add main admin scripts for example to manage records add/delete functions |
| 1216 | * |
| 1217 | * @access public |
| 1218 | */ |
| 1219 | add_action('admin_enqueue_scripts', function() { |
| 1220 | if(isset($_GET['page']) && strpos(sanitize_key($_GET['page']), 'gptranslate') !== false) { // phpcs:ignore WordPress.Security.NonceVerification.Recommended |
| 1221 | // Enqueue JS |
| 1222 | wp_enqueue_script ( 'gptranslate-js', plugin_dir_url ( __FILE__ ) . 'assets/js/admin.js', [ ], GPTranslate::$pluginVersion, true ); |
| 1223 | |
| 1224 | // Enqueue CSS |
| 1225 | wp_enqueue_style ( 'gptranslate-css', plugin_dir_url ( __FILE__ ) . 'assets/css/admin.css', [ ], GPTranslate::$pluginVersion ); |
| 1226 | |
| 1227 | wp_localize_script('gptranslate-js', 'gptranslate_vars', [ |
| 1228 | 'ajaxurl' => admin_url('admin-ajax.php'), |
| 1229 | 'nonce' => wp_create_nonce('gptranslate_migrate_translations'), |
| 1230 | 'deletenonce' => wp_create_nonce('gptranslate_delete_translations'), |
| 1231 | 'gptApiKey' => hash( 'sha256', get_site_url() ) |
| 1232 | ]); |
| 1233 | } |
| 1234 | }); |
| 1235 | |
| 1236 | add_action('wp_ajax_gptranslate_bulk_delete', function () { |
| 1237 | if (!current_user_can('manage_options') || !check_ajax_referer('gptranslate_delete_translations', '_wpnonce', false)) { |
| 1238 | wp_send_json_error(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UNAUTHORIZED_REQUEST'))); |
| 1239 | } |
| 1240 | |
| 1241 | global $wpdb; |
| 1242 | $ids = isset($_POST['gptid']) ? array_map('intval', (array) $_POST['gptid']) : []; |
| 1243 | |
| 1244 | if (empty($ids)) { |
| 1245 | wp_send_json_error('No records selected'); |
| 1246 | } |
| 1247 | |
| 1248 | $table = $wpdb->prefix . 'gptranslate'; |
| 1249 | $in = implode(',', array_fill(0, count($ids), '%d')); |
| 1250 | $sql = "DELETE FROM {$table} WHERE id IN ($in)"; |
| 1251 | $result = $wpdb->query($wpdb->prepare($sql, ...$ids)); // phpcs:ignore |
| 1252 | |
| 1253 | if ($result === false) { |
| 1254 | wp_send_json_error('Database error'); |
| 1255 | } |
| 1256 | |
| 1257 | wp_send_json_success(); |
| 1258 | }); |
| 1259 | |
| 1260 | // Handle Export CSV |
| 1261 | add_action('admin_post_gptranslate_export_translations_csv', function () { |
| 1262 | if (!current_user_can('manage_options') || !check_admin_referer('gptranslate_export_csv', 'gptranslate_export_csv_nonce')) { |
| 1263 | wp_die(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UNAUTHORIZED_REQUEST'))); |
| 1264 | } |
| 1265 | |
| 1266 | global $wpdb; |
| 1267 | $table = $wpdb->prefix . 'gptranslate'; |
| 1268 | |
| 1269 | $records = $wpdb->get_results("SELECT * FROM $table ORDER BY translate_date DESC", ARRAY_A); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1270 | |
| 1271 | if (!$records) { |
| 1272 | wp_die(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_NOTRANSLATIONS'))); |
| 1273 | } |
| 1274 | |
| 1275 | $localDate = get_date_from_gmt(gmdate('Y-m-d')); |
| 1276 | $fileDate = ( date_i18n('Y-m-d', strtotime($localDate)) ); |
| 1277 | |
| 1278 | header('Content-Type: text/csv'); |
| 1279 | header('Content-Disposition: attachment; filename="gptranslate-translations-' . $fileDate . '.csv"'); |
| 1280 | |
| 1281 | // phpcs:ignore WordPress.WP.AlternativeFunctions.file_system_operations_fopen |
| 1282 | $output = fopen('php://output', 'w'); |
| 1283 | |
| 1284 | // Intestazioni CSV |
| 1285 | fputcsv($output, array_keys($records[0])); |
| 1286 | |
| 1287 | foreach ($records as $record) { |
| 1288 | fputcsv($output, $record); |
| 1289 | } |
| 1290 | |
| 1291 | // phpcs:ignore WordPress.WP.AlternativeFunctions.file_system_operations_fclose |
| 1292 | fclose($output); |
| 1293 | exit; |
| 1294 | }); |
| 1295 | |
| 1296 | // Handle Import CSV |
| 1297 | add_action('admin_post_gptranslate_import_translations_csv', function () { |
| 1298 | if (!current_user_can('manage_options') || !check_admin_referer('gptranslate_import_csv', 'gptranslate_import_csv_nonce')) { |
| 1299 | wp_die(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UNAUTHORIZED_REQUEST'))); |
| 1300 | } |
| 1301 | |
| 1302 | if (!isset($_FILES['import_file'], $_FILES['import_file']['error'], $_FILES['import_file']['tmp_name']) || $_FILES['import_file']['error'] !== UPLOAD_ERR_OK) { |
| 1303 | wp_die(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UPLOAD_FAILED'))); |
| 1304 | } |
| 1305 | |
| 1306 | $tmp_name = $_FILES['import_file']['tmp_name']; // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized |
| 1307 | if (!is_uploaded_file($tmp_name)) { |
| 1308 | wp_die(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_FAILED_UPLOADED_FILE'))); |
| 1309 | } |
| 1310 | |
| 1311 | // phpcs:ignore WordPress.WP.AlternativeFunctions.file_system_operations_fopen |
| 1312 | $file = fopen($tmp_name, 'r'); |
| 1313 | if (!$file) { |
| 1314 | wp_die(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_FAILED_UPLOADED_FILE'))); |
| 1315 | } |
| 1316 | |
| 1317 | $headers = fgetcsv($file); |
| 1318 | if (!$headers) { |
| 1319 | wp_die(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_INVALID_CSV_FORMAT'))); |
| 1320 | } |
| 1321 | |
| 1322 | global $wpdb; |
| 1323 | $table = $wpdb->prefix . 'gptranslate'; |
| 1324 | |
| 1325 | while (($row = fgetcsv($file)) !== false) { |
| 1326 | $countHeaders = count($headers); |
| 1327 | $countRow = count($row); |
| 1328 | // Invalid combine |
| 1329 | if($countHeaders != $countRow) { |
| 1330 | continue; |
| 1331 | } |
| 1332 | $record = array_combine($headers, $row); |
| 1333 | if (empty($record['pagelink'])) { |
| 1334 | continue; // skip if no primary key |
| 1335 | } |
| 1336 | |
| 1337 | $pagelink = sanitize_text_field($record['pagelink']); |
| 1338 | $exists = $wpdb->get_var($wpdb->prepare("SELECT id FROM $table WHERE pagelink = %s", $pagelink)); // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1339 | |
| 1340 | $data = [ |
| 1341 | 'translated_alias' => $record['translated_alias'], |
| 1342 | 'translations' => $record['translations'], |
| 1343 | 'alt_translations' => $record['alt_translations'], |
| 1344 | 'languageoriginal' => sanitize_text_field($record['languageoriginal']), |
| 1345 | 'languagetranslated' => sanitize_text_field($record['languagetranslated']), |
| 1346 | 'published' => isset($record['published']) ? (int)$record['published'] : 1, |
| 1347 | 'translate_date' => $record['translate_date'], |
| 1348 | 'translation_engine' => sanitize_text_field($record['translation_engine']), |
| 1349 | ]; |
| 1350 | |
| 1351 | if ($exists) { |
| 1352 | $wpdb->update( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1353 | $table, |
| 1354 | $data, |
| 1355 | ['pagelink' => $pagelink], |
| 1356 | ['%s', '%s', '%s', '%s', '%s', '%d', '%s', '%s'], |
| 1357 | ['%s'] |
| 1358 | ); |
| 1359 | } else { |
| 1360 | $data['pagelink'] = $pagelink; |
| 1361 | $wpdb->insert( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1362 | $table, |
| 1363 | $data, |
| 1364 | ['%s', '%s', '%s', '%s', '%s', '%s', '%d', '%s', '%s'] |
| 1365 | ); |
| 1366 | } |
| 1367 | } |
| 1368 | |
| 1369 | // phpcs:ignore WordPress.WP.AlternativeFunctions.file_system_operations_fclose |
| 1370 | fclose($file); |
| 1371 | |
| 1372 | wp_redirect(admin_url('admin.php?page=gptranslate&imported=1')); |
| 1373 | exit; |
| 1374 | }); |
| 1375 | |
| 1376 | |
| 1377 | // Register Ajax handler |
| 1378 | add_action('wp_ajax_gptranslate_migrate_translations', function() { |
| 1379 | if (!current_user_can('manage_options')) { |
| 1380 | wp_send_json_error(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UNAUTHORIZED_REQUEST'))); |
| 1381 | } |
| 1382 | |
| 1383 | check_ajax_referer('gptranslate_migrate_translations'); |
| 1384 | |
| 1385 | global $wpdb; |
| 1386 | $table = $wpdb->prefix . 'gptranslate'; |
| 1387 | $old = isset($_POST['old_domain']) ? sanitize_text_field(wp_unslash($_POST['old_domain'])) : ''; |
| 1388 | $new = isset($_POST['new_domain']) ? sanitize_text_field(wp_unslash($_POST['new_domain'])) : ''; |
| 1389 | |
| 1390 | if (empty($old) || empty($new)) { |
| 1391 | wp_send_json_error(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_MISSING_DOMAIN_VALUES'))); |
| 1392 | } |
| 1393 | |
| 1394 | $query = $wpdb->prepare( |
| 1395 | "UPDATE $table SET pagelink = REPLACE(pagelink, %s, %s), translated_alias = REPLACE(translated_alias, %s, %s)", // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1396 | $old, $new, $old, $new |
| 1397 | ); |
| 1398 | |
| 1399 | // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1400 | $result = $wpdb->query($query); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared,WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1401 | |
| 1402 | if ($result === false) { |
| 1403 | wp_send_json_error(esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_DATABASE_ERROR'))); |
| 1404 | } else { |
| 1405 | wp_send_json_success(); |
| 1406 | } |
| 1407 | }); |
| 1408 | |
| 1409 | function gptranslate_export_settings() { |
| 1410 | if ( ! current_user_can( 'manage_options' ) || !check_admin_referer('gptranslate_export_settings', 'gptranslate_export_settings_nonce')) { |
| 1411 | wp_die( esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UNAUTHORIZED_REQUEST')) ); |
| 1412 | } |
| 1413 | |
| 1414 | $options = get_option( 'gptranslate_options', [] ); |
| 1415 | |
| 1416 | $localDate = get_date_from_gmt(gmdate('Y-m-d')); |
| 1417 | $fileDate = ( date_i18n('Y-m-d', strtotime($localDate)) ); |
| 1418 | |
| 1419 | header( 'Content-Type: application/json' ); |
| 1420 | header( 'Content-Disposition: attachment; filename="gptranslate-settings-' . $fileDate . '.json"' ); |
| 1421 | echo wp_json_encode( $options ); |
| 1422 | exit; |
| 1423 | } |
| 1424 | add_action( 'admin_post_gptranslate_export_settings', 'gptranslate_export_settings' ); |
| 1425 | |
| 1426 | function gptranslate_import_settings() { |
| 1427 | if ( ! current_user_can( 'manage_options' ) || !check_admin_referer('gptranslate_import_settings', 'gptranslate_import_settings_nonce')) { |
| 1428 | wp_die( esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UNAUTHORIZED_REQUEST')) ); |
| 1429 | } |
| 1430 | |
| 1431 | if ( empty( $_FILES['gptranslate_settings_file']['tmp_name'] ) ) { |
| 1432 | wp_die( esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_UPLOAD_FAILED')) ); |
| 1433 | } |
| 1434 | |
| 1435 | $content = file_get_contents( $_FILES['gptranslate_settings_file']['tmp_name'] ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized |
| 1436 | $decoded = json_decode( $content, true ); |
| 1437 | |
| 1438 | if ( json_last_error() !== JSON_ERROR_NONE || ! is_array( $decoded ) ) { |
| 1439 | wp_die( esc_html($this->loadTranslations('PLG_GPTRANSLATE_INVALID_JSON_TRANSLATIONS') ) ); |
| 1440 | } |
| 1441 | |
| 1442 | // Optional: sanitize known values, or just update if you're confident of source |
| 1443 | update_option( 'gptranslate_options', $decoded ); |
| 1444 | |
| 1445 | wp_safe_redirect( admin_url( 'admin.php?page=gptranslate-settings&settingsimported=1' ) ); |
| 1446 | exit; |
| 1447 | } |
| 1448 | add_action( 'admin_post_gptranslate_import_settings', 'gptranslate_import_settings' ); |
| 1449 | |
| 1450 | add_action('wp_footer', function () { |
| 1451 | // Add the default target container if default CSS selector |
| 1452 | $settings = get_option("gptranslate_options"); |
| 1453 | |
| 1454 | // Disable interface |
| 1455 | if($settings ['disable_control']) { |
| 1456 | $settings ['wrapper_selector'] = ''; |
| 1457 | } |
| 1458 | |
| 1459 | if ($settings ['wrapper_selector'] == '.gptranslate_wrapper') { |
| 1460 | echo '<div class="gptranslate_wrapper" id="gpt-wrapper"></div>'; |
| 1461 | } |
| 1462 | }); |
| 1463 | |
| 1464 | // POST API REST Translations storage |
| 1465 | add_action('rest_api_init', function () { |
| 1466 | register_rest_route('gptranslate/v1', '/request', [ |
| 1467 | 'methods' => 'POST', |
| 1468 | 'callback' => 'gpt_handle_request', |
| 1469 | 'permission_callback' => 'gptranslate_public_permission' |
| 1470 | ]); |
| 1471 | }); |
| 1472 | |
| 1473 | add_filter('plugin_action_links_' . plugin_basename(__FILE__), function ($links) { |
| 1474 | // Remove the default 'Settings' item |
| 1475 | unset($links[0]); |
| 1476 | |
| 1477 | $settings_link = '<a href="' . esc_url(admin_url('admin.php?page=gptranslate-settings')) . '">' . esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_SETTINGS_MENU_TITLE')) . '</a>'; |
| 1478 | array_unshift($links, $settings_link); |
| 1479 | |
| 1480 | $translations_link = '<a href="' . esc_url(admin_url('admin.php?page=gptranslate')) . '">' . esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_TRANSLATIONS')) . '</a>'; |
| 1481 | array_unshift($links, $translations_link); |
| 1482 | |
| 1483 | return $links; |
| 1484 | }); |
| 1485 | |
| 1486 | /* |
| 1487 | // Remove any WP update for the free version over the paid full one |
| 1488 | add_filter('auto_update_plugin', function($update, $item) { |
| 1489 | if (isset($item->slug) && $item->slug === 'gptranslate') { |
| 1490 | return false; |
| 1491 | } |
| 1492 | return $update; |
| 1493 | }, 10, 2); |
| 1494 | |
| 1495 | |
| 1496 | add_filter('site_transient_update_plugins', function($transient) { |
| 1497 | if (isset($transient->response['gptranslate/gptranslate.php'])) { |
| 1498 | unset($transient->response['gptranslate/gptranslate.php']); |
| 1499 | } |
| 1500 | return $transient; |
| 1501 | }); |
| 1502 | */ |
| 1503 | |
| 1504 | /** |
| 1505 | * Permission callback public API |
| 1506 | * @param WP_REST_Request $request |
| 1507 | * @return bool|WP_Error |
| 1508 | */ |
| 1509 | function gptranslate_public_permission( WP_REST_Request $request ) { |
| 1510 | // 1) Controllo chiave API inviata via header |
| 1511 | $headerApiKey = $request->get_header('x-gptranslate-key'); |
| 1512 | $restApiKey = hash( 'sha256', get_site_url() ); |
| 1513 | if ( $headerApiKey != $restApiKey) { |
| 1514 | return new WP_Error( 'rest_forbidden', esc_html(GPTranslate::loadTranslation('PLG_GPTRANSLATE_FORBIDDEN_APIKEY')), [ 'status' => 403 ] ); |
| 1515 | } |
| 1516 | return true; |
| 1517 | } |
| 1518 | |
| 1519 | /** |
| 1520 | * Callback per GET/STORE translations via REST. |
| 1521 | * Frontend API |
| 1522 | * |
| 1523 | * @param WP_REST_Request $request |
| 1524 | * @return WP_REST_Response |
| 1525 | */ |
| 1526 | function gpt_handle_request( WP_REST_Request $request ) { |
| 1527 | global $wpdb; |
| 1528 | |
| 1529 | $table = $wpdb->prefix . 'gptranslate'; |
| 1530 | |
| 1531 | $params = $request->get_json_params(); |
| 1532 | |
| 1533 | if(!$params) { |
| 1534 | $params = $request->get_body_params(); |
| 1535 | } |
| 1536 | |
| 1537 | // Sanitize input params |
| 1538 | $task = sanitize_text_field( $params['task'] ?? '' ); |
| 1539 | $pageLink = sanitize_text_field( $params['pagelink'] ?? '' ); |
| 1540 | $translatedAlias = sanitize_text_field( $params['translated_alias'] ?? '' ); |
| 1541 | $languageOriginal = sanitize_text_field( $params['language_original'] ?? '' ); |
| 1542 | $languageTranslated = sanitize_text_field( $params['language_translated'] ?? '' ); |
| 1543 | $translationEngine = sanitize_text_field( $params['translation_engine'] ?? '' ); |
| 1544 | |
| 1545 | $now = current_time( 'mysql' ); |
| 1546 | |
| 1547 | $response = [ 'result' => false ]; |
| 1548 | |
| 1549 | if ( $task === 'storetranslations' ) { |
| 1550 | // Fetch raw param (could be array or JSON string) |
| 1551 | $rawFull = $params['translations'] ?? '[]'; |
| 1552 | $rawAlt = $params['alt_translations'] ?? '[]'; |
| 1553 | |
| 1554 | // If it’s already a string (JSON), use it directly. |
| 1555 | // If it’s an array (unlikely with FormData), JSON encode it. |
| 1556 | if ( is_string( $rawFull ) && json_decode( $rawFull ) !== null ) { |
| 1557 | $fullTranslations = $rawFull; |
| 1558 | } else { |
| 1559 | $fullTranslations = wp_json_encode( (array) $rawFull ); |
| 1560 | } |
| 1561 | |
| 1562 | if ( is_string( $rawAlt ) && json_decode( $rawAlt ) !== null ) { |
| 1563 | $altTranslations = $rawAlt; |
| 1564 | } else { |
| 1565 | $altTranslations = wp_json_encode( (array) $rawAlt ); |
| 1566 | } |
| 1567 | |
| 1568 | // Verifica se esiste già |
| 1569 | $existing_id = $wpdb->get_var( $wpdb->prepare( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1570 | "SELECT id FROM {$table}" . // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1571 | "\n WHERE (pagelink = %s OR pagelink = %s)" . |
| 1572 | "\n AND languageoriginal = %s" . |
| 1573 | "\n AND languagetranslated = %s", |
| 1574 | rtrim($pageLink, '/'), rtrim($pageLink, '/') . '/', $languageOriginal, $languageTranslated |
| 1575 | ) ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1576 | |
| 1577 | if ( $existing_id ) { |
| 1578 | // UPDATE |
| 1579 | $updated = $wpdb->update( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1580 | $table, |
| 1581 | [ |
| 1582 | 'translations' => $fullTranslations, |
| 1583 | 'alt_translations' => $altTranslations, |
| 1584 | 'translated_alias' => $translatedAlias, |
| 1585 | 'translate_date' => $now, |
| 1586 | 'translation_engine' => $translationEngine, |
| 1587 | ], |
| 1588 | [ 'id' => (int) $existing_id ], |
| 1589 | [ '%s', '%s', '%s', '%s', '%s' ], |
| 1590 | [ '%d' ] |
| 1591 | ); |
| 1592 | $response['result'] = ( $updated !== false ); |
| 1593 | } else { |
| 1594 | // INSERT |
| 1595 | $inserted = $wpdb->insert( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1596 | $table, |
| 1597 | [ |
| 1598 | 'pagelink' => $pageLink, |
| 1599 | 'translations' => $fullTranslations, |
| 1600 | 'alt_translations' => $altTranslations, |
| 1601 | 'translated_alias' => $translatedAlias, |
| 1602 | 'languageoriginal' => $languageOriginal, |
| 1603 | 'languagetranslated' => $languageTranslated, |
| 1604 | 'published' => 1, |
| 1605 | 'translate_date' => $now, |
| 1606 | 'translation_engine' => $translationEngine, |
| 1607 | ], |
| 1608 | [ '%s','%s','%s','%s','%s','%s','%d','%s','%s' ] |
| 1609 | ); |
| 1610 | $response['result'] = ( $inserted !== false ); |
| 1611 | } |
| 1612 | } elseif ( $task === 'gettranslations' ) { |
| 1613 | $opts = get_option( 'gptranslate_options', [] ); |
| 1614 | if ( ! empty( $opts['realtime_translations'] ) ) { |
| 1615 | $response['result'] = false; |
| 1616 | } else { |
| 1617 | if ($opts['rewrite_language_url'] == 1 && $opts['rewrite_language_alias'] == 1) { |
| 1618 | $row = $wpdb->get_row( $wpdb->prepare( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1619 | "SELECT translations, alt_translations, translated_alias, pagelink FROM {$table}" . // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1620 | "\n WHERE (pagelink = %s OR pagelink = %s OR translated_alias = %s OR translated_alias = %s)" . |
| 1621 | "\n AND languageoriginal = %s" . |
| 1622 | "\n AND languagetranslated = %s" . |
| 1623 | "\n AND published = 1", |
| 1624 | rtrim($pageLink, '/'), rtrim($pageLink, '/') . '/', rtrim($pageLink, '/'), rtrim($pageLink, '/') . '/', $languageOriginal, $languageTranslated |
| 1625 | ), ARRAY_A ); |
| 1626 | } else { |
| 1627 | $row = $wpdb->get_row( $wpdb->prepare( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1628 | "SELECT translations, alt_translations, translated_alias, pagelink FROM {$table}" . // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1629 | "\n WHERE (pagelink = %s OR pagelink = %s)" . |
| 1630 | "\n AND languageoriginal = %s" . |
| 1631 | "\n AND languagetranslated = %s" . |
| 1632 | "\n AND published = 1", |
| 1633 | rtrim($pageLink, '/'), rtrim($pageLink, '/') . '/', $languageOriginal, $languageTranslated |
| 1634 | ), ARRAY_A ); |
| 1635 | } |
| 1636 | |
| 1637 | if ( $row ) { |
| 1638 | $response['result'] = true; |
| 1639 | $response['translations'] = json_decode( $row['translations'], true ) ?: []; |
| 1640 | $response['alt_translations'] = json_decode( $row['alt_translations'], true ) ?: []; |
| 1641 | $response['translated_alias'] = $row['translated_alias']; |
| 1642 | $response['pagelink_alias'] = $row['pagelink']; |
| 1643 | } else { |
| 1644 | $response['result'] = false; |
| 1645 | } |
| 1646 | } |
| 1647 | } elseif ($task == 'getaliastranslation') { |
| 1648 | // Always perform a new realtime translation if the option is enabled |
| 1649 | try { |
| 1650 | $row = $wpdb->get_row( $wpdb->prepare( |
| 1651 | "SELECT translated_alias FROM {$table}" . |
| 1652 | "\n WHERE (pagelink = %s OR pagelink = %s)" . |
| 1653 | "\n AND languageoriginal = %s" . |
| 1654 | "\n AND languagetranslated = %s" . |
| 1655 | "\n AND published = 1", |
| 1656 | rtrim($pageLink, '/'), rtrim($pageLink, '/') . '/', $languageOriginal, $languageTranslated |
| 1657 | ), ARRAY_A ); |
| 1658 | |
| 1659 | if ( $row ) { |
| 1660 | $response['result'] = true; |
| 1661 | $response['translated_alias'] = $row['translated_alias'] ?? ''; |
| 1662 | } else { |
| 1663 | $response['result'] = false; |
| 1664 | } |
| 1665 | } catch ( Exception $e ) { |
| 1666 | $response['result'] = false; |
| 1667 | $response['exception'] = $e->getMessage(); |
| 1668 | } |
| 1669 | } elseif ( $task === 'syncTranslation' ) { |
| 1670 | $original = wp_unslash( $params['original'] ?? '' ); |
| 1671 | $translated = wp_unslash( $params['translated'] ?? '' ); |
| 1672 | $languageTranslated = sanitize_text_field( $params['language_translated'] ?? '' ); |
| 1673 | $translationType = sanitize_text_field( $params['translation_type'] ?? 'translations' ); // default to 'translations' |
| 1674 | |
| 1675 | // Recupera tutti i record nella lingua target |
| 1676 | $rows = $wpdb->get_results( $wpdb->prepare( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1677 | "SELECT id, {$translationType}, languagetranslated" . // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1678 | "\n FROM {$table}" . // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Dynamic query built with placeholders, safely prepared |
| 1679 | "\n WHERE languagetranslated = %s", |
| 1680 | $languageTranslated |
| 1681 | ) ); |
| 1682 | |
| 1683 | $updatedCount = 0; |
| 1684 | |
| 1685 | foreach ( $rows as $row ) { |
| 1686 | $currentTranslations = json_decode( $row->$translationType, true ); |
| 1687 | |
| 1688 | if ( is_array( $currentTranslations ) && array_key_exists( $original, $currentTranslations ) ) { |
| 1689 | // Aggiorna la traduzione e salva |
| 1690 | $currentTranslations[ $original ] = $translated; |
| 1691 | $jsonUpdated = wp_json_encode( $currentTranslations ); |
| 1692 | |
| 1693 | $success = $wpdb->update( // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared, WordPress.DB.DirectDatabaseQuery.DirectQuery, WordPress.DB.DirectDatabaseQuery.NoCaching |
| 1694 | $table, |
| 1695 | [ $translationType => $jsonUpdated, 'translate_date' => $now ], |
| 1696 | [ 'id' => $row->id ], |
| 1697 | [ '%s', '%s' ], |
| 1698 | [ '%d' ] |
| 1699 | ); |
| 1700 | |
| 1701 | if ( $success !== false ) { |
| 1702 | $updatedCount++; |
| 1703 | } |
| 1704 | } |
| 1705 | } |
| 1706 | |
| 1707 | $response['result'] = $updatedCount > 0; |
| 1708 | } |
| 1709 | |
| 1710 | return rest_ensure_response( $response ); |
| 1711 | } |
| 1712 | |
| 1713 | // Instantiate and run the app |
| 1714 | GPTranslate::get_instance(); |