0.8 $langs = array_combine($lang_parse[1], $lang_parse[4]); // set default to 1 for any without q factor foreach ($langs as $lang => $val) { if ($val === '') $langs[$lang] = 1; } // sort list based on value arsort($langs, SORT_NUMERIC); } } return $langs; } /** * @brief Returns the best language for which also a translation exists. * * This function takes the results from get_browser_language() and compares it * with the available translations and returns the best fitting language for * which there exists a translation. * * If there is no match fall back to config['system']['language'] * * @return Language code in 2-letter ISO 639-1 (en). */ function get_best_language() { $langs = get_browser_language(); if(isset($langs) && count($langs)) { foreach ($langs as $lang => $v) { $lang = strtolower($lang); if(is_dir("view/$lang")) { $preferred = $lang; break; } } } if(! isset($preferred)) { /* * We could find no perfect match for any of the preferred languages. * For cases where the preference is fr-fr and we have fr but *not* fr-fr * run the test again and only look for the language base * which should provide an interface they can sort of understand */ if(isset($langs) && count($langs)) { foreach ($langs as $lang => $v) { if(strlen($lang) === 2) { /* we have already checked this language */ continue; } /* Check the base */ $lang = strtolower(substr($lang,0,2)); if(is_dir("view/$lang")) { $preferred = $lang; break; } } } } if(! isset($preferred)) { $preferred = 'unset'; } $arr = array('langs' => $langs, 'preferred' => $preferred); call_hooks('get_best_language',$arr); if($arr['preferred'] !== 'unset') return $arr['preferred']; return ((isset(App::$config['system']['language'])) ? App::$config['system']['language'] : 'en'); } /* * push_lang and pop_lang let you temporarily override the default language. * Often used to email the administrator during a session created in another language. * The stack is one level deep - so you must pop after every push. */ function push_lang($language) { App::$langsave = App::$language; if($language === App::$language) return; if(isset(App::$strings) && count(App::$strings)) { App::$stringsave = App::$strings; } App::$strings = array(); load_translation_table($language); App::$language = $language; } function pop_lang() { if(App::$language === App::$langsave) return; if(isset(App::$stringsave) && is_array(App::$stringsave)) App::$strings = App::$stringsave; else App::$strings = array(); App::$language = App::$langsave; } /** * @brief Load string translation table for alternate language. * * @param string $lang language code in 2-letter ISO 639-1 (en, de, fr) format * @param boolean $install (optional) default false */ function load_translation_table($lang, $install = false) { App::$strings = array(); if(file_exists("view/$lang/hstrings.php")) { include("view/$lang/hstrings.php"); } elseif(file_exists("view/$lang/strings.php")) { include("view/$lang/strings.php"); } if(! $install) { $plugins = q("SELECT aname FROM addon WHERE installed=1;"); if ($plugins !== false) { foreach($plugins as $p) { $name = $p['aname']; if(file_exists("addon/$name/lang/$lang/hstrings.php")) { include("addon/$name/lang/$lang/hstrings.php"); } elseif(file_exists("addon/$name/lang/$lang/strings.php")) { include("addon/$name/lang/$lang/strings.php"); } } } } // Allow individual strings to be over-ridden on this site // Either for the default language or for all languages if(file_exists("view/local-$lang/hstrings.php")) { include("view/local-$lang/hstrings.php"); } elseif(file_exists("view/local-$lang/strings.php")) { include("view/local-$lang/strings.php"); } } /** * @brief Translate string if translation exists. * * @param string $s string that should get translated * @param string $ctx (optional) context to appear in po file * @return string translated string if exists, otherwise return $s * */ function t($s, $ctx = ''): string { $cs = $ctx ? '__ctx:' . $ctx . '__ ' . $s : $s; if (x(App::$strings, $cs)) { $t = App::$strings[$cs]; return ((is_array($t)) ? translate_projectname($t[0]) : translate_projectname($t)); } return translate_projectname($s); } /** * @brief translate product name * Merging strings from different project names is problematic so we'll do that with a string replacement */ function translate_projectname($s) { if(strpos($s,'rojectname') !== false) { return str_replace(array('$projectname','$Projectname'),array(Zotlabs\Lib\System::get_platform_name(),ucfirst(Zotlabs\Lib\System::get_platform_name())),$s); } return $s; } /** * @brief * * @param string $singular * @param string $plural * @param int $count * @param string $ctx * @return string */ function tt($singular, $plural, $count, $ctx = ''){ $cs = $ctx ? "__ctx:" . $ctx . "__ " . $singular : $singular; if (x(App::$strings,$cs)) { $t = App::$strings[$cs]; $f = 'string_plural_select_' . str_replace('-', '_', App::$language); if (! function_exists($f)) $f = 'string_plural_select_default'; $k = $f(intval($count)); return is_array($t) ? $t[$k] : $t; } if ($count != 1) { return $plural; } else { return $singular; } } /** * @brief Return slash separated string of plurals translation forms * * @param string $k key in translations array * @return string */ function ta($k){ $t = null; if(isset(App::$strings[$k])) $t = App::$strings[$k]; if (is_array($t)) $t = implode("/", $t); return ($t == "" ? $k : $t); } /** * @brief Return string_plural_select_xx function code * * @return string */ function tf() { $s = "plural_function_code"; return (x(App::$strings, $s) ? App::$strings[$s] : "0"); } /** * @brief Provide a fallback which will not collide with a function defined in * any language file. * * @param int $n * @return boolean */ function string_plural_select_default($n) { return ($n != 1); } /** * @brief Takes a string and tries to identify the language. * * Strings need to have a min length config['system']['language_detect_min_length'] * and you can influence the confidence that must be met before a result will get * returned through config['system']['language_detect_min_confidence']. * * @param string $s A string to examine * @return string Language code in 2-letter ISO 639-1 (en, de, fr) format */ function detect_language($s) { if (!$s) { return EMPTY_STR; } $min_length = Config::Get('system', 'language_detect_min_length', LANGUAGE_DETECT_MIN_LENGTH); $min_confidence = Config::Get('system', 'language_detect_min_confidence', LANGUAGE_DETECT_MIN_CONFIDENCE); // embedded apps have long base64 strings which will trip up the detector. $naked_body = preg_replace('/\[app\](.*?)\[\/app\]/', '', $s); // strip off bbcode $naked_body = preg_replace('/\[(.+?)\]/', '', $naked_body); // strip any links $naked_body = preg_replace('/\b(https?|ftp|file):\/\/[-A-Z0-9+&@#\/%?=~_|$!:,.;]*[A-Z0-9+&@#\/%=~_|$]/i', '', $naked_body); if (mb_strlen($naked_body) < intval($min_length)) { logger('string length less than ' . intval($min_length), LOGGER_DATA); return EMPTY_STR; } $lang = new Language; $lang_arr = $lang->detect($naked_body)->limit(0, 1)->close(); $confidence = reset($lang_arr); if ($confidence >= intval($min_confidence)) { logger('detect language: ' . print_r($lang_arr, true) . $naked_body, LOGGER_DATA); return key($lang_arr); } logger('detect language: confidence less than ' . $min_confidence, LOGGER_DATA); return EMPTY_STR; } /** * @brief Returns the display name of a given language code. * * By default we use the localized language name. You can switch the result * to any language with the optional 2nd parameter $l. * * $s and $l should be in 2-letter ISO 639-1 format. * * If nothing could be looked up it returns $s. * * @param string $s Language code to look up * @param string $l (optional) In which language to return the name * @return string with the language name, or $s if unrecognized */ function get_language_name($s, $l = null) { // get() expects the second part to be in upper case if (strpos($s, '-') !== false) $s = substr($s, 0, 2) . strtoupper(substr($s, 2)); if ($l !== null && strpos($l, '-') !== false) $l = substr($l, 0, 2) . strtoupper(substr($l, 2)); $languageRepository = new LanguageRepository; // Sometimes intl doesn't like the second part at all ... try { $language = $languageRepository->get($s, $l); } catch(CommerceGuys\Intl\Exception\UnknownLanguageException $e) { $s = substr($s, 0, 2); if($l !== null) $l = substr($s, 0, 2); try { $language = $languageRepository->get($s, $l); } catch (CommerceGuys\Intl\Exception\UnknownLanguageException $e) { return $s; // Give up } catch (CommerceGuys\Intl\Exception\UnknownLocaleException $e) { return $s; // Give up } } return $language->getName(); } function language_list() { $langs = glob('view/*/hstrings.php'); $lang_options = array(); $selected = ""; if(is_array($langs) && count($langs)) { if(! in_array('view/en/hstrings.php',$langs)) $langs[] = 'view/en/'; asort($langs); foreach($langs as $l) { $ll = substr($l,5); $ll = substr($ll,0,strrpos($ll,'/')); $lang_options[$ll] = get_language_name($ll, $ll) . " ($ll)"; } } return $lang_options; } function lang_selector() { $langs = glob('view/*/hstrings.php'); $lang_options = array(); $selected = ""; if(is_array($langs) && count($langs)) { $langs[] = ''; if(! in_array('view/en/hstrings.php',$langs)) $langs[] = 'view/en/'; asort($langs); foreach($langs as $l) { if($l == '') { $lang_options[""] = t('default'); continue; } $ll = substr($l,5); $ll = substr($ll,0,strrpos($ll,'/')); $selected = (($ll === App::$language && (x($_SESSION, 'language'))) ? $ll : $selected); $lang_options[$ll] = get_language_name($ll, $ll) . " ($ll)"; } } $tpl = get_markup_template('lang_selector.tpl'); $o = replace_macros($tpl, array( '$title' => t('Select an alternate language'), '$langs' => array($lang_options, $selected), )); return $o; } function rtl_languages() { return [ 'ar', 'arc', 'ckb', 'dv', 'fa', 'ha', 'he', 'khw', 'ks', 'ps', 'sd', 'ur', 'yi' ]; }