2007-09-16 16:14:52 +00:00
|
|
|
<?php
|
|
|
|
/**
|
2007-10-21 19:53:57 +00:00
|
|
|
* The {Translatable} decorator allows your DataObjects to have versions in different languages,
|
2007-09-16 16:14:52 +00:00
|
|
|
* defining which fields are can be translated.
|
2007-10-21 19:53:57 +00:00
|
|
|
*
|
|
|
|
* Common language names (e.g. 'en') are used in {Translatable} for
|
|
|
|
* database-entities. On the other hand, the file-based i18n-translations
|
|
|
|
* always have a "locale" (e.g. 'en_US').
|
|
|
|
*
|
|
|
|
* You can enable {Translatabe} for any DataObject-subclass:
|
|
|
|
* <example>
|
|
|
|
* static $extensions = array(
|
|
|
|
* "Translatable('MyTranslatableVarchar', 'OtherTranslatableText')"
|
|
|
|
* );
|
|
|
|
* </example>
|
|
|
|
*
|
|
|
|
* Caution: Does not apply any character-set conversion, it is assumed that all content
|
|
|
|
* is stored and represented in UTF-8 (Unicode). Please make sure your database and
|
|
|
|
* HTML-templates adjust to this.
|
|
|
|
*
|
|
|
|
* Caution: Further decorations of DataObject might conflict with this implementation,
|
|
|
|
* e.g. when overriding the get_one()-calls (which are already extended by {Translatable}).
|
|
|
|
*
|
|
|
|
* @author Bernat Foj Capell <bernat@silverstripe.com>
|
2008-02-25 02:10:37 +00:00
|
|
|
* @package sapphire
|
|
|
|
* @subpackage misc
|
2007-09-16 16:14:52 +00:00
|
|
|
*/
|
|
|
|
class Translatable extends DataObjectDecorator {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if the multilingual feature is enabled
|
|
|
|
*
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
protected static $enabled = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The 'default' language.
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected static $default_lang = 'en';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The language in which we are reading dataobjects.
|
2008-02-26 01:22:52 +00:00
|
|
|
* Usually stored in session, specific to the "site mode":
|
|
|
|
* either 'site' or 'cms'.
|
|
|
|
* @see Director::get_site_mode()
|
2007-09-16 16:14:52 +00:00
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected static $reading_lang = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if the start language has been determined using choose_site_lang
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
protected static $language_decided = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates whether the 'Lang' transformation when modifying queries should be bypassed
|
|
|
|
* If it's true
|
|
|
|
*
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
protected static $bypass = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A cached list of existing tables
|
|
|
|
*
|
|
|
|
* @var mixed
|
|
|
|
*/
|
|
|
|
protected static $tableList = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dataobject's original ID when we're creating a new language version of an object
|
|
|
|
*
|
|
|
|
* @var unknown_type
|
|
|
|
*/
|
|
|
|
protected static $creatingFromID;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An array of fields that can be translated.
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $translatableFields;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A map of the field values of the original (untranslated) DataObject record
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $original_values = null;
|
|
|
|
|
2008-11-18 03:07:09 +00:00
|
|
|
function getLang() {
|
|
|
|
$record = $this->owner->toMap();
|
|
|
|
return (isset($record["Lang"])) ? $record["Lang"] : Translatable::default_lang();
|
|
|
|
}
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if a table given table exists in the db
|
|
|
|
*
|
|
|
|
* @param mixed $table Table name
|
|
|
|
* @return boolean Returns true if $table exists.
|
|
|
|
*/
|
|
|
|
static function table_exists($table) {
|
2007-10-21 00:39:14 +00:00
|
|
|
if (!self::$tableList) self::$tableList = DB::tableList();
|
|
|
|
return isset(self::$tableList[strtolower($table)]);
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Choose the language the site is currently on.
|
2007-10-21 19:53:57 +00:00
|
|
|
* If $_GET['lang'] or $_COOKIE['lang'] is set, then it will use that language, and store it in the session.
|
2008-02-26 01:22:52 +00:00
|
|
|
* Otherwise it checks the session for a possible stored language, either from namespace to the site_mode
|
|
|
|
* ('site' or 'cms'), or for a 'global' language setting.
|
|
|
|
* The final option is the member preference.
|
|
|
|
*
|
|
|
|
* @uses Director::get_site_mode()
|
2007-10-21 19:53:57 +00:00
|
|
|
*
|
|
|
|
* @param $langsAvailable array A numerical array of languages which are valid choices (optional)
|
|
|
|
* @return string Selected language (also saved in $reading_lang).
|
2007-09-16 16:14:52 +00:00
|
|
|
*/
|
2007-10-21 19:53:57 +00:00
|
|
|
static function choose_site_lang($langsAvailable = null) {
|
2008-02-26 01:22:52 +00:00
|
|
|
$siteMode = Director::get_site_mode(); // either 'cms' or 'site'
|
|
|
|
|
|
|
|
if(isset($_GET['lang']) && (!isset($langsAvailable) || in_array($_GET['lang'], $langsAvailable))) {
|
|
|
|
// get from GET parameter
|
|
|
|
self::set_reading_lang($_GET['lang']);
|
|
|
|
} elseif(isset($_COOKIE['lang.' . $siteMode]) && $siteMode && (!isset($langsAvailable) || in_array($_COOKIE['lang.' . $siteMode], $langsAvailable))) {
|
|
|
|
// get from namespaced cookie
|
|
|
|
self::set_reading_lang($_COOKIE[$siteMode . '.lang']);
|
|
|
|
} elseif(isset($_COOKIE['lang']) && (!isset($langsAvailable) || in_array($_COOKIE['lang'], $langsAvailable))) {
|
|
|
|
// get from generic cookie
|
|
|
|
self::set_reading_lang($_COOKIE['lang']);
|
|
|
|
} else if(Session::get('lang.' . $siteMode) && (!isset($langsAvailable) || in_array(Session::get('lang.' . $siteMode), $langsAvailable))) {
|
|
|
|
// get from namespaced session ('cms' or 'site')
|
|
|
|
self::set_reading_lang(Session::get('lang.' . $siteMode));
|
|
|
|
} else if(Session::get('lang.global') && (!isset($langsAvailable) || in_array(Session::get('lang.global'), $langsAvailable))) {
|
|
|
|
// get from global session
|
|
|
|
self::set_reading_lang(Session::get('lang.global'));
|
2007-10-21 19:53:57 +00:00
|
|
|
} else {
|
2008-02-26 01:22:52 +00:00
|
|
|
// get default lang stored in class
|
|
|
|
self::set_reading_lang(self::default_lang());
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
2007-10-21 19:53:57 +00:00
|
|
|
|
|
|
|
return self::$reading_lang;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the current reading language.
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
static function default_lang() {
|
2007-10-21 00:39:14 +00:00
|
|
|
return self::$default_lang;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
2007-10-21 00:49:13 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Set default language.
|
|
|
|
*
|
2008-02-25 02:10:37 +00:00
|
|
|
* @param $lang String
|
2007-10-21 00:49:13 +00:00
|
|
|
*/
|
|
|
|
static function set_default_lang($lang) {
|
2007-10-26 01:48:08 +00:00
|
|
|
self::$default_lang = $lang;
|
2007-10-21 00:49:13 +00:00
|
|
|
}
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether the default and current reading language are the same.
|
|
|
|
* @return boolean Return true if both default and reading language are the same.
|
|
|
|
*/
|
|
|
|
static function is_default_lang() {
|
2007-10-21 00:39:14 +00:00
|
|
|
return (!self::current_lang() || self::$default_lang == self::current_lang());
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the current reading language.
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
static function current_lang() {
|
2007-10-21 00:39:14 +00:00
|
|
|
if (!self::$language_decided) self::choose_site_lang();
|
|
|
|
return self::$reading_lang;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2008-02-26 01:22:52 +00:00
|
|
|
* Set the reading language, either namespaced to 'site' (website content)
|
|
|
|
* or 'cms' (management backend).
|
|
|
|
*
|
2007-09-16 16:14:52 +00:00
|
|
|
* @param string $lang New reading language.
|
|
|
|
*/
|
|
|
|
static function set_reading_lang($lang) {
|
2008-02-26 01:22:52 +00:00
|
|
|
$key = (Director::get_site_mode()) ? 'lang.' . Director::get_site_mode() : 'lang.global';
|
|
|
|
Session::set($key, $lang);
|
2007-10-26 01:48:08 +00:00
|
|
|
self::$reading_lang = $lang;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a singleton instance of a class in the given language.
|
|
|
|
* @param string $class The name of the class.
|
|
|
|
* @param string $lang The name of the language.
|
|
|
|
* @param string $filter A filter to be inserted into the WHERE clause.
|
2007-12-02 21:33:24 +00:00
|
|
|
* @param boolean $cache Use caching (default: false)
|
|
|
|
* @param string $orderby A sort expression to be inserted into the ORDER BY clause.
|
2007-09-16 16:14:52 +00:00
|
|
|
* @return DataObject
|
|
|
|
*/
|
2007-12-02 21:33:24 +00:00
|
|
|
static function get_one_by_lang($class, $lang, $filter = '', $cache = false, $orderby = "") {
|
2007-10-21 00:39:14 +00:00
|
|
|
$oldLang = self::current_lang();
|
|
|
|
self::set_reading_lang($lang);
|
2007-12-02 21:33:24 +00:00
|
|
|
$result = DataObject::get_one($class, $filter, $cache, $orderby);
|
2007-10-21 00:39:14 +00:00
|
|
|
self::set_reading_lang($oldLang);
|
2007-09-16 16:14:52 +00:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a singleton instance of a class in the most convenient language (@see choose_site_lang())
|
|
|
|
*
|
|
|
|
* @param string $callerClass The name of the class
|
|
|
|
* @param string $filter A filter to be inserted into the WHERE clause.
|
2007-12-02 21:33:24 +00:00
|
|
|
* @param boolean $cache Use caching (default: false)
|
|
|
|
* @param string $orderby A sort expression to be inserted into the ORDER BY clause.
|
2007-09-16 16:14:52 +00:00
|
|
|
* @return DataObject
|
|
|
|
*/
|
2007-12-02 21:33:24 +00:00
|
|
|
static function get_one($callerClass, $filter = "", $cache = false, $orderby = "") {
|
2007-10-21 19:53:57 +00:00
|
|
|
self::$language_decided = true;
|
|
|
|
self::$reading_lang = self::default_lang();
|
2007-09-16 16:14:52 +00:00
|
|
|
$record = DataObject::get_one($callerClass, $filter);
|
|
|
|
if (!$record) {
|
2007-10-21 00:39:14 +00:00
|
|
|
self::$bypass = true;
|
2007-12-02 21:33:24 +00:00
|
|
|
$record = DataObject::get_one($callerClass, $filter, $cache, $orderby);
|
2007-10-21 00:39:14 +00:00
|
|
|
self::$bypass = false;
|
|
|
|
if ($record) self::set_reading_lang($record->Lang);
|
2007-09-16 16:14:52 +00:00
|
|
|
} else {
|
2007-10-21 00:39:14 +00:00
|
|
|
$langsAvailable = (array)self::get_langs_by_id($callerClass, $record->ID);
|
|
|
|
$langsAvailable[] = self::default_lang();
|
2007-10-21 19:53:57 +00:00
|
|
|
$lang = self::choose_site_lang($langsAvailable);
|
2007-09-16 16:14:52 +00:00
|
|
|
if (isset($lang)) {
|
2007-10-21 00:39:14 +00:00
|
|
|
$transrecord = self::get_one_by_lang($callerClass, $lang, "`$callerClass`.ID = $record->ID");
|
2007-09-16 16:14:52 +00:00
|
|
|
if ($transrecord) {
|
2007-10-21 00:39:14 +00:00
|
|
|
self::set_reading_lang($lang);
|
2007-09-16 16:14:52 +00:00
|
|
|
$record = $transrecord;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $record;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get all the instances of the given class translated to the given language
|
|
|
|
*
|
|
|
|
* @param string $class The name of the class
|
|
|
|
* @param string $lang The name of the language
|
|
|
|
* @param string $filter A filter to be inserted into the WHERE clause.
|
|
|
|
* @param string $sort A sort expression to be inserted into the ORDER BY clause.
|
2007-12-02 21:33:24 +00:00
|
|
|
* @param string $join A single join clause. This can be used for filtering, only 1 instance of each DataObject will be returned.
|
|
|
|
* @param string $limit A limit expression to be inserted into the LIMIT clause.
|
|
|
|
* @param string $containerClass The container class to return the results in.
|
|
|
|
* @param string $having A filter to be inserted into the HAVING clause.
|
2007-09-16 16:14:52 +00:00
|
|
|
* @return mixed The objects matching the conditions.
|
|
|
|
*/
|
2007-12-02 21:33:24 +00:00
|
|
|
static function get_by_lang($class, $lang, $filter = '', $sort = '', $join = "", $limit = "", $containerClass = "DataObjectSet", $having = "") {
|
2007-10-21 00:39:14 +00:00
|
|
|
$oldLang = self::current_lang();
|
|
|
|
self::set_reading_lang($lang);
|
2007-12-02 21:33:24 +00:00
|
|
|
$result = DataObject::get($class, $filter, $sort, $join, $limit, $containerClass, $having);
|
2007-10-21 00:39:14 +00:00
|
|
|
self::set_reading_lang($oldLang);
|
2007-09-16 16:14:52 +00:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a record in his original language version.
|
|
|
|
* @param string $class The name of the class.
|
|
|
|
* @param string $originalLangID The original record id.
|
|
|
|
* @return DataObject
|
|
|
|
*/
|
|
|
|
static function get_original($class, $originalLangID) {
|
|
|
|
$baseClass = $class;
|
|
|
|
while( ($p = get_parent_class($baseClass)) != "DataObject") $baseClass = $p;
|
2007-10-21 00:39:14 +00:00
|
|
|
return self::get_one_by_lang($class,self::default_lang(),"`$baseClass`.ID = $originalLangID");
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
2008-11-18 03:07:09 +00:00
|
|
|
function getTranslatedLangs() {
|
|
|
|
$class = ClassInfo::baseDataClass($this->owner->class); //Base Class
|
|
|
|
if($this->owner->hasExtension("Versioned") && Versioned::current_stage() == "Live") {
|
|
|
|
$class = $class."_Live";
|
|
|
|
}
|
|
|
|
|
|
|
|
$id = $this->owner->ID;
|
|
|
|
if(is_numeric($id)) {
|
|
|
|
$query = new SQLQuery('distinct Lang',"$class","(`$class`.OriginalID =$id)");
|
|
|
|
$langs = $query->execute()->column();
|
|
|
|
}
|
|
|
|
return ($langs) ? array_values($langs) : array();
|
|
|
|
}
|
|
|
|
|
2007-09-16 16:14:52 +00:00
|
|
|
/**
|
|
|
|
* Get a list of languages in which a given element has been translated
|
|
|
|
*
|
|
|
|
* @param string $class Name of the class of the element
|
|
|
|
* @param int $id ID of the element
|
|
|
|
* @return array List of languages
|
|
|
|
*/
|
|
|
|
static function get_langs_by_id($class, $id) {
|
|
|
|
$query = new SQLQuery('Lang',"{$class}_lang","(`{$class}_lang`.OriginalLangID =$id)");
|
2007-10-21 19:53:57 +00:00
|
|
|
$langs = $query->execute()->column();
|
|
|
|
return ($langs) ? array_values($langs) : false;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Writes an object in a certain language. Use this instead of $object->write() if you want to write
|
|
|
|
* an instance in a determinated language independently of the currently set working language
|
|
|
|
*
|
|
|
|
* @param DataObject $object Object to be written
|
|
|
|
* @param string $lang The name of the language
|
|
|
|
*/
|
|
|
|
static function write(DataObject $object, $lang) {
|
2007-10-21 00:39:14 +00:00
|
|
|
$oldLang = self::current_lang();
|
|
|
|
self::set_reading_lang($lang);
|
2007-09-16 16:14:52 +00:00
|
|
|
$result = $object->write();
|
2007-10-21 00:39:14 +00:00
|
|
|
self::set_reading_lang($oldLang);
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enables the multilingual feature
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static function enable() {
|
2007-10-21 00:39:14 +00:00
|
|
|
self::$enabled = true;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
2007-09-16 17:23:59 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Disable the multilingual feature
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static function disable() {
|
2007-10-21 00:39:14 +00:00
|
|
|
self::$enabled = false;
|
2007-09-16 17:23:59 +00:00
|
|
|
}
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether multilingual support has been enabled
|
|
|
|
*
|
|
|
|
* @return boolean True if enabled
|
|
|
|
*/
|
|
|
|
static function is_enabled() {
|
2007-10-21 00:39:14 +00:00
|
|
|
return self::$enabled;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When creating, set the original ID value
|
|
|
|
*
|
|
|
|
* @param int $id
|
|
|
|
*/
|
|
|
|
static function creating_from($id) {
|
2007-10-21 00:39:14 +00:00
|
|
|
self::$creatingFromID = $id;
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------------------------//
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Construct a new Translatable object.
|
|
|
|
* @var array $translatableFields The different fields of the object that can be translated.
|
|
|
|
*/
|
|
|
|
function __construct($translatableFields) {
|
|
|
|
parent::__construct();
|
|
|
|
|
|
|
|
if(!is_array($translatableFields)) {
|
|
|
|
$translatableFields = func_get_args();
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->translatableFields = $translatableFields;
|
|
|
|
}
|
|
|
|
|
|
|
|
function augmentSQL(SQLQuery &$query) {
|
2008-11-18 03:07:09 +00:00
|
|
|
if (! $this->stat('enabled', true)) return false;
|
2007-10-21 00:39:14 +00:00
|
|
|
if((($lang = self::current_lang()) && !self::is_default_lang()) || self::$bypass) {
|
2007-09-16 16:14:52 +00:00
|
|
|
foreach($query->from as $table => $dummy) {
|
|
|
|
if(!isset($baseTable)) {
|
|
|
|
$baseTable = $table;
|
|
|
|
}
|
|
|
|
|
2007-10-21 00:39:14 +00:00
|
|
|
if (self::table_exists("{$table}_lang")) {
|
2007-09-16 16:14:52 +00:00
|
|
|
$query->renameTable($table, $table . '_lang');
|
|
|
|
if (stripos($query->sql(),'.ID')) {
|
|
|
|
// Every reference to ID is now OriginalLangID
|
|
|
|
$query->replaceText(".ID",".OriginalLangID");
|
|
|
|
$query->where = str_replace("`ID`", "`OriginalLangID`",$query->where);
|
|
|
|
$query->select[] = "`{$baseTable}_lang`.OriginalLangID AS ID";
|
|
|
|
}
|
|
|
|
if ($query->where) foreach ($query->where as $i => $wherecl) {
|
|
|
|
if (substr($wherecl,0,4) == 'ID =')
|
|
|
|
// Another reference to ID to be changed
|
|
|
|
$query->where[$i] = str_replace('ID =','OriginalLangID =',$wherecl);
|
|
|
|
else {
|
|
|
|
$parts = explode(' AND ',$wherecl);
|
|
|
|
foreach ($parts as $j => $part) {
|
2007-09-16 16:22:36 +00:00
|
|
|
// Divide this clause between the left ($innerparts[1]) and right($innerparts[2]) part of the condition
|
|
|
|
ereg('(`?[[:alnum:]_-]*`?\.?`?[[:alnum:]_-]*`?)(.*)', $part, $innerparts);
|
|
|
|
if (strpos($innerparts[1],'.') === false)
|
2007-09-16 16:14:52 +00:00
|
|
|
//it may be ambiguous, so sometimes we will need to add the table
|
2007-09-16 16:22:36 +00:00
|
|
|
$parts[$j] = ($this->isInAugmentedTable($innerparts[1], $table) ? "`{$table}_lang`." : "")."$part";
|
2007-09-16 16:14:52 +00:00
|
|
|
else {
|
|
|
|
/* if the table has been specified we have to determine if the original (without _lang) name has to be used
|
|
|
|
* because we don't have the queried field in the augmented table (which usually means
|
|
|
|
* that is not a translatable field)
|
|
|
|
*/
|
2007-09-16 16:22:36 +00:00
|
|
|
$clauseparts = explode('.',$innerparts[1]);
|
2007-09-16 16:14:52 +00:00
|
|
|
$originalTable = str_replace('`','',str_replace('_lang','',$clauseparts[0]));
|
|
|
|
$parts[$j] = ($this->isInAugmentedTable($clauseparts[1], $originalTable) ? "`{$originalTable}_lang`" : "`$originalTable`")
|
2007-09-16 16:22:36 +00:00
|
|
|
. ".{$clauseparts[1]}{$innerparts[2]}";
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
$query->where[$i] = implode(' AND ',$parts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if($table != $baseTable) {
|
|
|
|
$query->from["{$table}_lang"] = $query->from[$table];
|
|
|
|
} else {
|
|
|
|
// _lang is now the base table (the first one)
|
|
|
|
$query->from = array("{$table}_lang" => $query->from[$table]) + $query->from;
|
|
|
|
}
|
|
|
|
|
|
|
|
// unless we are bypassing this query, add the language filter
|
2007-10-21 00:39:14 +00:00
|
|
|
if (!self::$bypass) $query->where[] = "`{$table}_lang`.Lang = '$lang'";
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
// unless this is a deletion, the query is applied to the joined table
|
|
|
|
if (!$query->delete) {
|
|
|
|
$query->from[$table] = "INNER JOIN `$table`".
|
|
|
|
" ON `{$table}_lang`.OriginalLangID = `$table`.ID";
|
|
|
|
/* if we are selecting fields (not doing counts for example) we need to select everything from
|
|
|
|
* the original table (was renamed to _lang) since some fields that we require may be there
|
|
|
|
*/
|
|
|
|
if ($query->select[0][0] == '`') $query->select = array_merge(array("`$table`.*"),$query->select);
|
2007-10-21 19:53:57 +00:00
|
|
|
} else unset($query->from[$table]);
|
2007-09-16 16:14:52 +00:00
|
|
|
} else {
|
|
|
|
$query->from[$table] = str_replace("`{$table}`.OriginalLangID","`{$table}`.ID",$query->from[$table]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether a WHERE clause should be applied to the augmented table
|
|
|
|
*
|
|
|
|
* @param string $clause Where clause that need to know if can be applied to the augmented (suffixed) table
|
|
|
|
* @param string $table Name of the non-augmented table
|
|
|
|
* @return boolean True if the clause can be applied to the augmented table
|
|
|
|
*/
|
|
|
|
function isInAugmentedTable($clause, $table) {
|
|
|
|
$clause = str_replace('`','',$clause);
|
|
|
|
$table = str_replace('_lang','',$table);
|
|
|
|
if (strpos($table,'_') !== false) return false;
|
|
|
|
$field = ereg_replace('[[:blank:]]*([[:alnum:]]*).*','\\1',$clause);
|
|
|
|
$field = trim($field);
|
|
|
|
$allFields = $this->allFieldsInTable($table);
|
|
|
|
return (array_search($field,$allFields) !== false);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if the DataObject has any own translatable field (not inherited).
|
|
|
|
* @return boolean
|
|
|
|
*/
|
2007-09-16 16:58:19 +00:00
|
|
|
function hasOwnTranslatableFields() {
|
2007-09-16 16:14:52 +00:00
|
|
|
$ownFields = $this->owner->stat('db');
|
|
|
|
if ($ownFields == singleton($this->owner->parentClass())->stat('db'))return false;
|
|
|
|
foreach ((array)$this->translatableFields as $translatableField) {
|
|
|
|
if (isset($ownFields[$translatableField])) return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-09-16 16:58:19 +00:00
|
|
|
/**
|
|
|
|
* Determine if a table needs Versioned support
|
|
|
|
* This is called at db/build time
|
|
|
|
*
|
|
|
|
* @param string $table Table name
|
|
|
|
* @return boolean
|
|
|
|
*/
|
|
|
|
function isVersionedTable($table) {
|
|
|
|
// Every _lang table wants Versioned support
|
|
|
|
return ($this->owner->databaseFields() && $this->hasOwnTranslatableFields());
|
|
|
|
}
|
2007-09-16 17:23:59 +00:00
|
|
|
|
2007-09-16 16:14:52 +00:00
|
|
|
function augmentDatabase() {
|
2008-11-18 03:07:09 +00:00
|
|
|
if (! $this->stat('enabled', true)) return false;
|
2007-10-21 00:39:14 +00:00
|
|
|
self::set_reading_lang(self::default_lang());
|
2007-09-16 16:14:52 +00:00
|
|
|
$table = $this->owner->class;
|
|
|
|
|
2007-09-16 16:58:19 +00:00
|
|
|
if(($fields = $this->owner->databaseFields()) && $this->hasOwnTranslatableFields()) {
|
2007-09-16 16:14:52 +00:00
|
|
|
//Calculate the required fields
|
|
|
|
foreach ($fields as $field => $type) {
|
|
|
|
if (array_search($field,$this->translatableFields) === false) unset($fields[$field]);
|
|
|
|
}
|
|
|
|
$metaFields = array_diff((array)$this->owner->databaseFields(), (array)$this->owner->customDatabaseFields());
|
|
|
|
$indexes = $this->owner->databaseIndexes();
|
|
|
|
|
|
|
|
$langFields = array_merge(
|
|
|
|
array(
|
|
|
|
"Lang" => "Varchar(12)",
|
|
|
|
"OriginalLangID" => "Int"
|
|
|
|
),
|
|
|
|
$fields,
|
|
|
|
$metaFields
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ($indexes as $index => $type) {
|
|
|
|
if (true === $type && array_search($index,$langFields) === false) unset($indexes[$index]);
|
|
|
|
}
|
2007-12-02 21:33:48 +00:00
|
|
|
|
|
|
|
$langIndexes = array_merge(
|
|
|
|
array(
|
|
|
|
'OriginalLangID_Lang' => '(OriginalLangID, Lang)',
|
|
|
|
'OriginalLangID' => true,
|
|
|
|
'Lang' => true,
|
|
|
|
),
|
|
|
|
(array)$indexes
|
|
|
|
);
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
// Create table for translated instances
|
2007-12-02 21:33:48 +00:00
|
|
|
DB::requireTable("{$table}_lang", $langFields, $langIndexes);
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
DB::dontRequireTable("{$table}_lang");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Augment a write-record request.
|
|
|
|
* @param SQLQuery $manipulation Query to augment.
|
|
|
|
*/
|
|
|
|
function augmentWrite(&$manipulation) {
|
2008-11-18 03:07:09 +00:00
|
|
|
if (! $this->stat('enabled', true)) return false;
|
2007-10-21 00:39:14 +00:00
|
|
|
if(($lang = self::current_lang()) && !self::is_default_lang()) {
|
2007-09-16 16:14:52 +00:00
|
|
|
$tables = array_keys($manipulation);
|
|
|
|
foreach($tables as $table) {
|
2007-10-21 00:39:14 +00:00
|
|
|
if (self::table_exists("{$table}_lang")) {
|
2007-09-16 16:14:52 +00:00
|
|
|
$manipulation["{$table}_lang"] = $manipulation[$table];
|
|
|
|
if ($manipulation[$table]['command'] == 'insert') {
|
|
|
|
$fakeID = $this->owner->ID;
|
|
|
|
// In an insert we've to populate our fields and generate a new id (since the passed one it's relative to $table)
|
|
|
|
$SessionOrigID = Session::get($this->owner->ID.'_originalLangID');
|
|
|
|
$manipulation["{$table}_lang"]['fields']['OriginalLangID'] = $this->owner->ID =
|
2007-10-21 00:39:14 +00:00
|
|
|
( $SessionOrigID ? $SessionOrigID : self::$creatingFromID);
|
2007-09-16 16:14:52 +00:00
|
|
|
$manipulation["{$table}_lang"]['RecordID'] = $manipulation["{$table}_lang"]['fields']['OriginalLangID'];
|
2007-09-16 16:58:19 +00:00
|
|
|
// populate lang field
|
|
|
|
$manipulation["{$table}_lang"]['fields']['Lang'] = "'$lang'" ;
|
|
|
|
// get a valid id, pre-inserting
|
|
|
|
DB::query("INSERT INTO {$table}_lang SET Created = NOW(), Lang = '$lang'");
|
|
|
|
$manipulation["{$table}_lang"]['id'] = $manipulation["{$table}_lang"]['fields']['ID'] = DB::getGeneratedID("{$table}_lang");
|
|
|
|
$manipulation["{$table}_lang"]['command'] = 'update';
|
2007-09-16 16:14:52 +00:00
|
|
|
// we don't have to insert anything in $table if we are inserting in $table_lang
|
|
|
|
unset($manipulation[$table]);
|
2007-09-16 16:58:19 +00:00
|
|
|
// now dataobjects may create a record before the real write in the base table, so we have to delete it - 20/08/2007
|
|
|
|
if (is_numeric($fakeID)) DB::query("DELETE FROM $table WHERE ID=$fakeID");
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (!isset($manipulation[$table]['fields']['OriginalLangID'])) {
|
|
|
|
// for those updates that may become inserts populate these fields
|
|
|
|
$manipulation["{$table}_lang"]['fields']['OriginalLangID'] = $this->owner->ID;
|
|
|
|
$manipulation["{$table}_lang"]['fields']['Lang'] = "'$lang'";
|
|
|
|
}
|
|
|
|
$id = $manipulation["{$table}_lang"]['id'];
|
|
|
|
if(!$id) user_error("Couldn't find ID in manipulation", E_USER_ERROR);
|
|
|
|
if (isset($manipulation["{$table}_lang"]['where'])) {
|
|
|
|
$manipulation["{$table}_lang"]['where'] .= "AND (Lang = '$lang') AND (OriginalLangID = $id)";
|
|
|
|
} else {
|
|
|
|
$manipulation["{$table}_lang"]['where'] = "(Lang = '$lang') AND (OriginalLangID = $id)";
|
|
|
|
}
|
|
|
|
$realID = DB::query("SELECT ID FROM {$table}_lang WHERE (OriginalLangID = $id) AND (Lang = '$lang') LIMIT 1")->value();
|
|
|
|
$manipulation["{$table}_lang"]['id'] = $realID;
|
|
|
|
$manipulation["{$table}_lang"]['RecordID'] = $manipulation["{$table}_lang"]['fields']['OriginalLangID'];
|
|
|
|
// we could be updating non-translatable fields at the same time, so these will remain
|
|
|
|
foreach ($manipulation[$table]['fields'] as $field => $dummy) {
|
|
|
|
if ($this->isInAugmentedTable($field, $table) ) unset($manipulation[$table]['fields'][$field]);
|
|
|
|
}
|
|
|
|
if (count($manipulation[$table]['fields']) == 0) unset($manipulation[$table]);
|
|
|
|
}
|
|
|
|
foreach ($manipulation["{$table}_lang"]['fields'] as $field => $dummy) {
|
|
|
|
if (! $this->isInAugmentedTable($field, $table) ) unset($manipulation["{$table}_lang"]['fields'][$field]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------------------------//
|
|
|
|
|
|
|
|
function updateCMSFields(FieldSet &$fields) {
|
2008-11-18 03:07:09 +00:00
|
|
|
if(!$this->stat('enabled', true)) return false;
|
|
|
|
|
|
|
|
// add hidden fields for the used language and original record
|
|
|
|
$fields->push(new HiddenField("Lang", "Lang", $this->getLang()) );
|
|
|
|
$fields->push(new HiddenField("OriginalID", "OriginalID", $this->owner->OriginalID) );
|
|
|
|
|
|
|
|
// if a language other than default language is used, we're in "translation mode",
|
|
|
|
// hence have to modify the original fields
|
|
|
|
$isTranslationMode = (Translatable::default_lang() != $this->getLang() && $this->getLang());
|
|
|
|
if($isTranslationMode) {
|
|
|
|
$originalLangID = Session::get($this->owner->ID . '_originalLangID');
|
|
|
|
|
|
|
|
$translatableFieldNames = $this->getTranslatableFields();
|
|
|
|
$allDataFields = $fields->dataFields();
|
|
|
|
$transformation = new Translatable_Transformation(Translatable::get_original($this->owner->class, $this->owner->ID));
|
|
|
|
|
|
|
|
// iterate through sequential list of all datafields in fieldset
|
|
|
|
// (fields are object references, so we can replace them with the translatable CompositeField)
|
|
|
|
foreach($allDataFields as $dataField) {
|
|
|
|
|
|
|
|
if(in_array($dataField->Name(), $translatableFieldNames)) {
|
|
|
|
//var_dump($dataField->Name());
|
|
|
|
// if the field is translatable, perform transformation
|
|
|
|
$fields->replaceField($dataField->Name(), $transformation->transformFormField($dataField));
|
|
|
|
} else {
|
|
|
|
// else field shouldn't be editable in translation-mode, make readonly
|
|
|
|
$fields->replaceField($dataField->Name(), $dataField->performReadonlyTransformation());
|
|
|
|
}
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
} else {
|
2008-11-18 03:07:09 +00:00
|
|
|
// if we're not in "translation mode", show a dropdown to create a new translation.
|
|
|
|
// this action should just be possible when showing the default language,
|
|
|
|
// you can't create new translations from within a "translation mode" form.
|
|
|
|
|
|
|
|
$alreadyTranslatedLangs = array();
|
2007-09-16 16:14:52 +00:00
|
|
|
foreach ($alreadyTranslatedLangs as $i => $langCode) {
|
|
|
|
$alreadyTranslatedLangs[$i] = i18n::get_language_name($langCode);
|
|
|
|
}
|
|
|
|
$fields->addFieldsToTab(
|
|
|
|
'Root',
|
2008-02-25 02:10:37 +00:00
|
|
|
new Tab(_t('Translatable.TRANSLATIONS', 'Translations'),
|
2008-11-18 03:07:09 +00:00
|
|
|
new HeaderField('CreateTransHeader', _t('Translatable.CREATE', 'Create new translation'), 2),
|
2008-02-25 02:10:37 +00:00
|
|
|
$langDropdown = new LanguageDropdownField("NewTransLang", _t('Translatable.NEWLANGUAGE', 'New language'), $alreadyTranslatedLangs),
|
|
|
|
$createButton = new InlineFormAction('createtranslation',_t('Translatable.CREATEBUTTON', 'Create'))
|
2007-09-16 16:14:52 +00:00
|
|
|
)
|
|
|
|
);
|
|
|
|
if (count($alreadyTranslatedLangs)) {
|
|
|
|
$fields->addFieldsToTab(
|
|
|
|
'Root.Translations',
|
|
|
|
new FieldSet(
|
2008-11-18 03:07:09 +00:00
|
|
|
new HeaderField('ExistingTransHeader', _t('Translatable.EXISTING', 'Existing translations:'), 3),
|
2007-09-16 16:14:52 +00:00
|
|
|
new LiteralField('existingtrans',implode(', ',$alreadyTranslatedLangs))
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
$langDropdown->addExtraClass('languageDropdown');
|
|
|
|
$createButton->addExtraClass('createTranslationButton');
|
|
|
|
$createButton->includeDefaultJS(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a list of fields from the tables created by this extension
|
|
|
|
*
|
|
|
|
* @param string $table Name of the table
|
|
|
|
* @return array Map where the keys are db, indexes and the values are the table fields
|
|
|
|
*/
|
|
|
|
function fieldsInExtraTables($table){
|
|
|
|
|
2007-09-16 16:58:19 +00:00
|
|
|
if(($fields = $this->owner->databaseFields()) && $this->hasOwnTranslatableFields()) {
|
2007-09-16 16:14:52 +00:00
|
|
|
//Calculate the required fields
|
|
|
|
foreach ($fields as $field => $type) {
|
|
|
|
if (array_search($field,$this->translatableFields) === false) unset($fields[$field]);
|
|
|
|
}
|
|
|
|
$metaFields = array_diff((array)$this->owner->databaseFields(), (array)$this->owner->customDatabaseFields());
|
|
|
|
$indexes = $this->owner->databaseIndexes();
|
|
|
|
|
|
|
|
$langFields = array_merge(
|
|
|
|
array(
|
|
|
|
"Lang" => "Varchar(12)",
|
|
|
|
"OriginalLangID" => "Int"
|
|
|
|
),
|
|
|
|
$fields,
|
|
|
|
$metaFields
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ($indexes as $index => $type) {
|
|
|
|
if (true === $type && array_search($index,$langFields) === false) unset($indexes[$index]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return array('db' => $langFields, 'indexes' => $indexes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a list of fields in the {$table}_lang table
|
|
|
|
*
|
|
|
|
* @param string $table Table name
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function allFieldsInTable($table){
|
|
|
|
|
|
|
|
$fields = singleton($table)->databaseFields();
|
|
|
|
//Calculate the required fields
|
|
|
|
foreach ($fields as $field => $type) {
|
|
|
|
if (array_search($field,$this->translatableFields) === false) unset($fields[$field]);
|
|
|
|
}
|
|
|
|
$metaFields = array_diff((array)singleton('DataObject')->databaseFields(), (array)$this->owner->customDatabaseFields());
|
|
|
|
|
|
|
|
$langFields = array_merge(
|
|
|
|
array(
|
|
|
|
"ID",
|
|
|
|
"LastEdited",
|
|
|
|
"Created",
|
|
|
|
"ClassName",
|
|
|
|
"Version",
|
|
|
|
"WasPublished",
|
|
|
|
"Lang",
|
|
|
|
"OriginalLangID"
|
|
|
|
),
|
|
|
|
$this->translatableFields,
|
|
|
|
array_keys($fields),
|
|
|
|
array_keys($metaFields)
|
|
|
|
);
|
|
|
|
return $langFields;
|
|
|
|
}
|
2008-11-18 03:07:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the names of all translatable fields on this class
|
|
|
|
* as a numeric array.
|
|
|
|
* @todo Integrate with blacklist once branches/translatable is merged back.
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function getTranslatableFields() {
|
|
|
|
return $this->translatableFields;
|
|
|
|
}
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the base table - the class that directly extends DataObject.
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
function baseTable($stage = null) {
|
|
|
|
$tableClasses = ClassInfo::dataClassesFor($this->owner->class);
|
|
|
|
$baseClass = array_shift($tableClasses);
|
|
|
|
return (!$stage || $stage == $this->defaultStage) ? $baseClass : $baseClass . "_$stage";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extends $table with a suffix if required
|
|
|
|
*
|
|
|
|
* @param string $table Name of the table
|
|
|
|
* @return string Extended table name
|
|
|
|
*/
|
|
|
|
function extendWithSuffix($table) {
|
2007-10-21 00:39:14 +00:00
|
|
|
if((($lang = self::current_lang()) && !self::is_default_lang())) {
|
|
|
|
if (self::table_exists("{$table}_lang")) return $table.'_lang';
|
2007-09-16 16:14:52 +00:00
|
|
|
}
|
|
|
|
return $table;
|
|
|
|
}
|
2009-01-08 03:36:58 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a list of languages with at least one element translated in (including the default language)
|
|
|
|
*
|
|
|
|
* @param string $className Look for languages in elements of this class
|
|
|
|
* @return array Map of languages in the form langCode => langName
|
|
|
|
*/
|
|
|
|
static function get_existing_content_languages($className = 'SiteTree', $where = '') {
|
|
|
|
if(!Translatable::is_enabled()) return false;
|
|
|
|
$baseTable = ClassInfo::baseDataClass($className);
|
|
|
|
$query = new SQLQuery('Distinct Lang',$baseTable,$where,"",'Lang');
|
|
|
|
$dbLangs = $query->execute()->column();
|
|
|
|
$langlist = array_merge((array)Translatable::default_lang(), (array)$dbLangs);
|
|
|
|
$returnMap = array();
|
|
|
|
$allCodes = array_merge(i18n::$all_locales, i18n::$common_languages);
|
|
|
|
foreach ($langlist as $langCode) {
|
|
|
|
if($langCode)
|
|
|
|
$returnMap[$langCode] = (is_array($allCodes[$langCode]) ? $allCodes[$langCode][0] : $allCodes[$langCode]);
|
|
|
|
}
|
|
|
|
return $returnMap;
|
|
|
|
}
|
2007-09-16 16:14:52 +00:00
|
|
|
|
|
|
|
}
|
2008-11-18 03:07:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Transform a formfield to a "translatable" representation,
|
|
|
|
* consisting of the original formfield plus a readonly-version
|
|
|
|
* of the original value, wrapped in a CompositeField.
|
|
|
|
*
|
|
|
|
* @param DataObject $original Needs the original record as we populate the readonly formfield with the original value
|
|
|
|
*
|
|
|
|
* @package sapphire
|
|
|
|
* @subpackage misc
|
|
|
|
*/
|
|
|
|
class Translatable_Transformation extends FormTransformation {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var DataObject
|
|
|
|
*/
|
|
|
|
private $original = null;
|
|
|
|
|
|
|
|
function __construct(DataObject $original) {
|
|
|
|
$this->original = $original;
|
|
|
|
parent::__construct();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the original DataObject attached to the Transformation
|
|
|
|
*
|
|
|
|
* @return DataObject
|
|
|
|
*/
|
|
|
|
function getOriginal() {
|
|
|
|
return $this->original;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @todo transformTextareaField() not used at the moment
|
|
|
|
*/
|
|
|
|
function transformTextareaField(TextareaField $field) {
|
|
|
|
$nonEditableField = new ToggleField($fieldname,$field->Title(),'','+','-');
|
|
|
|
$nonEditableField->labelMore = '+';
|
|
|
|
$nonEditableField->labelLess = '-';
|
|
|
|
return $this->baseTransform($nonEditableField, $field);
|
|
|
|
|
|
|
|
return $nonEditableField;
|
|
|
|
}
|
|
|
|
|
|
|
|
function transformFormField(FormField $field) {
|
|
|
|
$newfield = $field->performReadOnlyTransformation();
|
|
|
|
return $this->baseTransform($newfield, $field);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function baseTransform($nonEditableField, $originalField) {
|
|
|
|
$fieldname = $originalField->Name();
|
|
|
|
|
|
|
|
$nonEditableField_holder = new CompositeField($nonEditableField);
|
|
|
|
$nonEditableField_holder->setName($fieldname.'_holder');
|
|
|
|
$nonEditableField_holder->addExtraClass('originallang_holder');
|
|
|
|
|
|
|
|
$nonEditableField->setValue($this->original->$fieldname);
|
|
|
|
$nonEditableField->setName($fieldname.'_original');
|
|
|
|
$nonEditableField->addExtraClass('originallang');
|
|
|
|
$nonEditableField->setTitle('Original '.$originalField->Title());
|
|
|
|
|
|
|
|
$nonEditableField_holder->insertBefore($originalField, $fieldname.'_original');
|
|
|
|
return $nonEditableField_holder;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
?>
|