Localization update for he and fixing comment in en.
[lhc/web/wiklou.git] / maintenance / language / languages.inc
index a10cae9..9846429 100644 (file)
@@ -2,19 +2,28 @@
 /**
  * Handle messages in the language files.
  *
- * @addtogroup Maintenance
+ * @file
+ * @ingroup MaintenanceLanguage
  */
 
-require_once( 'messageTypes.inc' );
-
+/**
+ * @ingroup MaintenanceLanguage
+ */
 class languages {
        protected $mLanguages; # List of languages
+
        protected $mRawMessages; # Raw list of the messages in each language
        protected $mMessages; # Messages in each language (except for English), divided to groups
+       protected $mFallback; # Fallback language in each language
        protected $mGeneralMessages; # General messages in English, divided to groups
        protected $mIgnoredMessages; # All the messages which should be exist only in the English file
        protected $mOptionalMessages; # All the messages which may be translated or not, depending on the language
 
+       protected $mNamespaceNames; # Namespace names
+       protected $mNamespaceAliases; # Namespace aliases
+       protected $mMagicWords; # Magic words
+       protected $mSpecialPageAliases; # Special page aliases
+
        /**
         * Load the list of languages: all the Messages*.php
         * files in the languages directory.
@@ -22,7 +31,7 @@ class languages {
         * @param $exif Treat the EXIF messages?
         */
        function __construct( $exif = true ) {
-               global $wgIgnoredMessages, $wgOptionalMessages, $wgEXIFMessages;
+               require( dirname(__FILE__) . '/messageTypes.inc' );
                $this->mIgnoredMessages = $wgIgnoredMessages;
                if ( $exif ) {
                        $this->mOptionalMessages = array_merge( $wgOptionalMessages );
@@ -62,24 +71,46 @@ class languages {
        }
 
        /**
-        * Load the raw messages for a specific langauge from the messages file.
+        * Load the language file.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         */
-       protected function loadRawMessages( $code ) {
-               if ( isset( $this->mRawMessages[$code] ) ) {
+       protected function loadFile( $code ) {
+               if ( isset( $this->mRawMessages[$code] ) &&
+                       isset( $this->mFallback[$code] ) &&
+                       isset( $this->mNamespaceNames[$code] ) &&
+                       isset( $this->mNamespaceAliases[$code] ) &&
+                       isset( $this->mMagicWords[$code] ) &&
+                       isset( $this->mSpecialPageAliases[$code] ) ) {
                        return;
                }
+               $this->mRawMessages[$code] = array();
+               $this->mFallback[$code] = '';
+               $this->mNamespaceNames[$code] = array();
+               $this->mNamespaceAliases[$code] = array();
+               $this->mMagicWords[$code] = array();
+               $this->mSpecialPageAliases[$code] = array();
                $filename = Language::getMessagesFileName( $code );
                if ( file_exists( $filename ) ) {
                        require( $filename );
                        if ( isset( $messages ) ) {
                                $this->mRawMessages[$code] = $messages;
-                       } else {
-                               $this->mRawMessages[$code] = array();
                        }
-               } else {
-                       $this->mRawMessages[$code] = array();
+                       if ( isset( $fallback ) ) {
+                               $this->mFallback[$code] = $fallback;
+                       }
+                       if ( isset( $namespaceNames ) ) {
+                               $this->mNamespaceNames[$code] = $namespaceNames;
+                       }
+                       if ( isset( $namespaceAliases ) ) {
+                               $this->mNamespaceAliases[$code] = $namespaceAliases;
+                       }
+                       if ( isset( $magicWords ) ) {
+                               $this->mMagicWords[$code] = $magicWords;
+                       }
+                       if ( isset( $specialPageAliases ) ) {
+                               $this->mSpecialPageAliases[$code] = $specialPageAliases;
+                       }
                }
        }
 
@@ -88,7 +119,7 @@ class languages {
         * all - all the messages.
         * required - messages which should be translated in order to get a complete translation.
         * optional - messages which can be translated, the fallback translation is used if not translated.
-        * obsolete - messages which should not be translated, either because they are not exist, or they are ignored messages.
+        * obsolete - messages which should not be translated, either because they do not exist, or they are ignored messages.
         * translated - messages which are either required or optional, but translated from English and needed.
         *
         * @param $code The language code.
@@ -97,7 +128,7 @@ class languages {
                if ( isset( $this->mMessages[$code] ) ) {
                        return;
                }
-               $this->loadRawMessages( $code );
+               $this->loadFile( $code );
                $this->loadGeneralMessages();
                $this->mMessages[$code]['all'] = $this->mRawMessages[$code];
                $this->mMessages[$code]['required'] = array();
@@ -129,7 +160,7 @@ class languages {
                if ( isset( $this->mGeneralMessages ) ) {
                        return;
                }
-               $this->loadRawMessages( 'en' );
+               $this->loadFile( 'en' );
                $this->mGeneralMessages['all'] = $this->mRawMessages['en'];
                $this->mGeneralMessages['required'] = array();
                $this->mGeneralMessages['optional'] = array();
@@ -149,15 +180,15 @@ class languages {
        }
 
        /**
-        * Get all the messages for a specific langauge (not English), without the
+        * Get all the messages for a specific language (not English), without the
         * fallback language messages, divided to groups:
         * all - all the messages.
         * required - messages which should be translated in order to get a complete translation.
         * optional - messages which can be translated, the fallback translation is used if not translated.
-        * obsolete - messages which should not be translated, either because they are not exist, or they are ignored messages.
+        * obsolete - messages which should not be translated, either because they do not exist, or they are ignored messages.
         * translated - messages which are either required or optional, but translated from English and needed.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The messages in this language.
         */
@@ -181,29 +212,83 @@ class languages {
                return $this->mGeneralMessages;
        }
 
+       /**
+        * Get fallback language code for a specific language.
+        *
+        * @param $code The language code.
+        *
+        * @return Fallback code.
+        */
+       public function getFallback( $code ) {
+               $this->loadFile( $code );
+               return $this->mFallback[$code];
+       }
+
+       /**
+        * Get namespace names for a specific language.
+        *
+        * @param $code The language code.
+        *
+        * @return Namespace names.
+        */
+       public function getNamespaceNames( $code ) {
+               $this->loadFile( $code );
+               return $this->mNamespaceNames[$code];
+       }
+
+       /**
+        * Get namespace aliases for a specific language.
+        *
+        * @param $code The language code.
+        *
+        * @return Namespace aliases.
+        */
+       public function getNamespaceAliases( $code ) {
+               $this->loadFile( $code );
+               return $this->mNamespaceAliases[$code];
+       }
+
+       /**
+        * Get magic words for a specific language.
+        *
+        * @param $code The language code.
+        *
+        * @return Magic words.
+        */
+       public function getMagicWords( $code ) {
+               $this->loadFile( $code );
+               return $this->mMagicWords[$code];
+       }
+
+       /**
+        * Get special page aliases for a specific language.
+        *
+        * @param $code The language code.
+        *
+        * @return Special page aliases.
+        */
+       public function getSpecialPageAliases( $code ) {
+               $this->loadFile( $code );
+               return $this->mSpecialPageAliases[$code];
+       }
+
        /**
         * Get the untranslated messages for a specific language.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The untranslated messages for this language.
         */
        public function getUntranslatedMessages( $code ) {
                $this->loadGeneralMessages();
                $this->loadMessages( $code );
-               $requiredGeneralMessages = array_keys( $this->mGeneralMessages['required'] );
-               $requiredMessages = array_keys( $this->mMessages[$code]['required'] );
-               $untranslatedMessages = array();
-               foreach ( array_diff( $requiredGeneralMessages, $requiredMessages ) as $key ) {
-                       $untranslatedMessages[$key] = $this->mGeneralMessages['required'][$key];
-               }
-               return $untranslatedMessages;
+               return array_diff_key( $this->mGeneralMessages['required'], $this->mMessages[$code]['required'] );
        }
 
        /**
         * Get the duplicate messages for a specific language.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The duplicate messages for this language.
         */
@@ -220,17 +305,30 @@ class languages {
        }
 
        /**
-        * Get the messages which do not use some variables.
+        * Get the obsolete messages for a specific language.
+        *
+        * @param $code The language code.
+        *
+        * @return The obsolete messages for this language.
+        */
+       public function getObsoleteMessages( $code ) {
+               $this->loadGeneralMessages();
+               $this->loadMessages( $code );
+               return $this->mMessages[$code]['obsolete'];
+       }
+
+       /**
+        * Get the messages whose variables do not match the original ones.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
-        * @return The messages which do not use some variables in this language.
+        * @return The messages whose variables do not match the original ones.
         */
-       public function getMessagesWithoutVariables( $code ) {
+       public function getMessagesWithMismatchVariables( $code ) {
                $this->loadGeneralMessages();
                $this->loadMessages( $code );
                $variables = array( '\$1', '\$2', '\$3', '\$4', '\$5', '\$6', '\$7', '\$8', '\$9' );
-               $messagesWithoutVariables = array();
+               $mismatchMessages = array();
                foreach ( $this->mMessages[$code]['translated'] as $key => $value ) {
                        $missing = false;
                        foreach ( $variables as $var ) {
@@ -238,18 +336,22 @@ class languages {
                                        !preg_match( "/$var/sU", $value ) ) {
                                        $missing = true;
                                }
+                               if ( !preg_match( "/$var/sU", $this->mGeneralMessages['translatable'][$key] ) &&
+                                       preg_match( "/$var/sU", $value ) ) {
+                                       $missing = true;
+                               }
                        }
                        if ( $missing ) {
-                               $messagesWithoutVariables[$key] = $value;
+                               $mismatchMessages[$key] = $value;
                        }
                }
-               return $messagesWithoutVariables;
+               return $mismatchMessages;
        }
 
        /**
         * Get the messages which do not use plural.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The messages which do not use plural in this language.
         */
@@ -268,7 +370,7 @@ class languages {
        /**
         * Get the empty messages.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The empty messages for this language.
         */
@@ -287,7 +389,7 @@ class languages {
        /**
         * Get the messages with trailing whitespace.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The messages with trailing whitespace in this language.
         */
@@ -306,7 +408,7 @@ class languages {
        /**
         * Get the non-XHTML messages.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The non-XHTML messages for this language.
         */
@@ -318,6 +420,8 @@ class languages {
                        '<br *\\?>',
                        '<hr/>',
                        '<br/>',
+                       '<hr>',
+                       '<br>',
                );
                $wrongPhrases = '~(' . implode( '|', $wrongPhrases ) . ')~sDu';
                $nonXHTMLMessages = array();
@@ -332,7 +436,7 @@ class languages {
        /**
         * Get the messages which include wrong characters.
         *
-        * @param $code The langauge code.
+        * @param $code The language code.
         *
         * @return The messages which include wrong characters in this language.
         */
@@ -367,48 +471,278 @@ class languages {
        }
 
        /**
-        * Output a messages list
+        * Get the messages which include dubious links.
+        *
+        * @param $code The language code.
         *
-        * @param $messages The messages list
-        * @param $code The language code
-        * @param $text The text to show before the list (optional)
-        * @param $level The display level (optional)
-        * @param $links Show links (optional)
-        * @param $wikilang The langauge of the wiki to display the list in, for the links (optional)
+        * @return The messages which include dubious links in this language.
         */
-       public function outputMessagesList( $messages, $code, $text = '', $level = 2, $links = false, $wikilang = null ) {
-               if ( count( $messages ) == 0 ) {
-                       return;
-               }
-               if ( $text ) {
-                       echo "$text\n";
+       public function getMessagesWithDubiousLinks( $code ) {
+               $this->loadGeneralMessages();
+               $this->loadMessages( $code );
+               $tc = Title::legalChars() . '#%{}';
+               $messages = array();
+               foreach ( $this->mMessages[$code]['translated'] as $key => $value ) {
+                       $matches = array();
+                       preg_match_all( "/\[\[([{$tc}]+)(?:\\|(.+?))?]]/sDu", $value, $matches );
+                       for ($i = 0; $i < count($matches[0]); $i++ ) {
+                               if ( preg_match( "/.*project.*/isDu",  $matches[1][$i] ) ) {
+                                       $messages[$key][] = $matches[0][$i];
+                               }
+                       }
+
+
+                       if ( isset( $messages[$key] ) ) {
+                               $messages[$key] = implode( $messages[$key],", " );
+                       }
                }
-               if ( $level == 1 ) {
-                       echo "[messages are hidden]\n";
-               } else {
-                       foreach ( $messages as $key => $value ) {
-                               if ( $links ) {
-                                       $displayKey = ucfirst( $key );
-                                       if ( !isset( $wikilang ) ) {
-                                               global $wgContLang;
-                                               $wikilang = $wgContLang->getCode();
-                                       }
-                                       if ( $code == $wikilang ) {
-                                               $displayKey = "[[MediaWiki:$displayKey|$key]]";
-                                       } else {
-                                               $displayKey = "[[MediaWiki:$displayKey/$code|$key]]";
-                                       }
-                               } else {
-                                       $displayKey = $key;
+               return $messages;
+       }
+
+       /**
+        * Get the messages which include unbalanced brackets.
+        *
+        * @param $code The language code.
+        *
+        * @return The messages which include unbalanced brackets in this language.
+        */
+       public function getMessagesWithUnbalanced( $code ) {
+               $this->loadGeneralMessages();
+               $this->loadMessages( $code );
+               $messages = array();
+               foreach ( $this->mMessages[$code]['translated'] as $key => $value ) {
+                       $a = $b = $c = $d = 0;
+                       foreach ( preg_split( '//', $value ) as $char ) {
+                               switch ( $char ) {
+                                       case '[':
+                                               $a++;
+                                               break;
+                                       case ']':
+                                               $b++;
+                                               break;
+                                       case '{':
+                                               $c++;
+                                               break;
+                                       case '}':
+                                               $d++;
+                                               break;
                                }
-                               if ( $level == 2 ) {
-                                       echo "* $displayKey\n";
-                               } else {
-                                       echo "* $displayKey:            '$value'\n";
+                       }
+
+                       if ( $a !== $b || $c !== $d ) {
+                               $messages[$key] = "$a, $b, $c, $d";
+                       }
+                       
+               }
+               return $messages;
+       }
+
+       /**
+        * Get the untranslated namespace names.
+        *
+        * @param $code The language code.
+        *
+        * @return The untranslated namespace names in this language.
+        */
+       public function getUntranslatedNamespaces( $code ) {
+               $this->loadFile( 'en' );
+               $this->loadFile( $code );
+               return array_flip( array_diff_key( $this->mNamespaceNames['en'], $this->mNamespaceNames[$code] ) );
+       }
+
+       /**
+        * Get the project talk namespace names with no $1.
+        *
+        * @param $code The language code.
+        *
+        * @return The problematic project talk namespaces in this language.
+        */
+       public function getProblematicProjectTalks( $code ) {
+               $this->loadFile( $code );
+               $namespaces = array();
+
+               # Check default namespace name
+               if( isset( $this->mNamespaceNames[$code][NS_PROJECT_TALK] ) ) {
+                       $default = $this->mNamespaceNames[$code][NS_PROJECT_TALK];
+                       if ( strpos( $default, '$1' ) === FALSE ) {
+                               $namespaces[$default] = 'default';
+                       }
+               }
+
+               # Check namespace aliases
+               foreach( $this->mNamespaceAliases[$code] as $key => $value ) {
+                       if ( $value == NS_PROJECT_TALK && strpos( $key, '$1' ) === FALSE ) {
+                               $namespaces[$key] = '';
+                       }
+               }
+
+               return $namespaces;
+       }
+
+       /**
+        * Get the untranslated magic words.
+        *
+        * @param $code The language code.
+        *
+        * @return The untranslated magic words in this language.
+        */
+       public function getUntranslatedMagicWords( $code ) {
+               $this->loadFile( 'en' );
+               $this->loadFile( $code );
+               $magicWords = array();
+               foreach ( $this->mMagicWords['en'] as $key => $value ) {
+                       if ( !isset( $this->mMagicWords[$code][$key] ) ) {
+                               $magicWords[$key] = $value[1];
+                       }
+               }
+               return $magicWords;
+       }
+
+       /**
+        * Get the obsolete magic words.
+        *
+        * @param $code The language code.
+        *
+        * @return The obsolete magic words in this language.
+        */
+       public function getObsoleteMagicWords( $code ) {
+               $this->loadFile( 'en' );
+               $this->loadFile( $code );
+               $magicWords = array();
+               foreach ( $this->mMagicWords[$code] as $key => $value ) {
+                       if ( !isset( $this->mMagicWords['en'][$key] ) ) {
+                               $magicWords[$key] = $value[1];
+                       }
+               }
+               return $magicWords;
+       }
+
+       /**
+        * Get the magic words that override the original English magic word.
+        *
+        * @param $code The language code.
+        *
+        * @return The overriding magic words in this language.
+        */
+       public function getOverridingMagicWords( $code ) {
+               $this->loadFile( 'en' );
+               $this->loadFile( $code );
+               $magicWords = array();
+               foreach ( $this->mMagicWords[$code] as $key => $local ) {
+                       if ( !isset( $this->mMagicWords['en'][$key] ) ) {
+                               # Unrecognized magic word
+                               continue;
+                       }
+                       $en = $this->mMagicWords['en'][$key];
+                       array_shift( $local );
+                       array_shift( $en );
+                       foreach ( $en as $word ) {
+                               if ( !in_array( $word, $local ) ) {
+                                       $magicWords[$key] = $word;
+                                       break;
                                }
                        }
                }
+               return $magicWords;
+       }
+
+       /**
+        * Get the magic words which do not match the case-sensitivity of the original words.
+        *
+        * @param $code The language code.
+        *
+        * @return The magic words whose case does not match in this language.
+        */
+       public function getCaseMismatchMagicWords( $code ) {
+               $this->loadFile( 'en' );
+               $this->loadFile( $code );
+               $magicWords = array();
+               foreach ( $this->mMagicWords[$code] as $key => $local ) {
+                       if ( !isset( $this->mMagicWords['en'][$key] ) ) {
+                               # Unrecognized magic word
+                               continue;
+                       }
+                       if ( $local[0] != $this->mMagicWords['en'][$key][0] ) {
+                               $magicWords[$key] = $local[0];
+                       }
+               }
+               return $magicWords;
+       }
+
+       /**
+        * Get the untranslated special page names.
+        *
+        * @param $code The language code.
+        *
+        * @return The untranslated special page names in this language.
+        */
+       public function getUntraslatedSpecialPages( $code ) {
+               $this->loadFile( 'en' );
+               $this->loadFile( $code );
+               $specialPageAliases = array();
+               foreach ( $this->mSpecialPageAliases['en'] as $key => $value ) {
+                       if ( !isset( $this->mSpecialPageAliases[$code][$key] ) ) {
+                               $specialPageAliases[$key] = $value[0];
+                       }
+               }
+               return $specialPageAliases;
+       }
+
+       /**
+        * Get the obsolete special page names.
+        *
+        * @param $code The language code.
+        *
+        * @return The obsolete special page names in this language.
+        */
+       public function getObsoleteSpecialPages( $code ) {
+               $this->loadFile( 'en' );
+               $this->loadFile( $code );
+               $specialPageAliases = array();
+               foreach ( $this->mSpecialPageAliases[$code] as $key => $value ) {
+                       if ( !isset( $this->mSpecialPageAliases['en'][$key] ) ) {
+                               $specialPageAliases[$key] = $value[0];
+                       }
+               }
+               return $specialPageAliases;
        }
 }
 
-?>
+class extensionLanguages extends languages {
+       private $mMessageGroup; # The message group
+
+       /**
+        * Load the messages group.
+        * @param $group The messages group.
+        */
+       function __construct( MessageGroup $group ) {
+               $this->mMessageGroup = $group;
+
+               $bools = $this->mMessageGroup->getBools();
+               $this->mIgnoredMessages = $bools['ignored'];
+               $this->mOptionalMessages = $bools['optional'];
+       }
+
+       /**
+        * Get the extension name.
+        *
+        * @return The extension name.
+        */
+       public function name() {
+               return $this->mMessageGroup->getLabel();
+       }
+
+       /**
+        * Load the language file.
+        *
+        * @param $code The language code.
+        */
+       protected function loadFile( $code ) {
+               if( !isset( $this->mRawMessages[$code] ) ) {
+                       $this->mRawMessages[$code] = $this->mMessageGroup->load( $code );
+                       if( empty( $this->mRawMessages[$code] ) ) {
+                               $this->mRawMessages[$code] = array();
+                       }
+               }
+       }
+}