%PDF- %PDF-
Mini Shell

Mini Shell

Direktori : /proc/985914/root/www/varak.net/wiki.varak.net/extensions/Babel/
Upload File :
Create Path :
Current File : //proc/985914/root/www/varak.net/wiki.varak.net/extensions/Babel/Babel.class.php

<?php
/**
 * Contains main code.
 *
 * @file
 * @author Robert Leverington
 * @author Robin Pepermans
 * @author Niklas Laxström
 * @author Brian Wolff
 * @author Purodha Blissenbach
 * @author Sam Reed
 * @author Siebrand Mazeland
 * @license http://www.gnu.org/copyleft/gpl.html GNU General Public License 2.0 or later
 */

/**
 * Main class for the Babel extension.
 */
class Babel {
	/**
	 * @var Title
	 */
	protected static $title;

	/**
	 * Render the Babel tower.
	 *
	 * @param $parser Parser.
	 * @return string: Babel tower.
	 */
	public static function Render( $parser ) {
		global $wgBabelUseUserLanguage;
		$parameters = func_get_args();
		array_shift( $parameters );
		self::$title = $parser->getTitle();

		self::mTemplateLinkBatch( $parameters );

		$parser->getOutput()->addModuleStyles( 'ext.babel' );

		$content = '';
		$templateParameters = array(); // collects name=value parameters to be passed to wiki templates.
		$createCategories = !$parser->getOptions()->getIsPreview();
		foreach ( $parameters as $name ) {
			if ( strpos( $name, '=' ) !== false ) {
				$templateParameters[] = $name;
				continue;
			}
			$components = self::mParseParameter( $name );
			$template = wfMessage( 'babel-template', $name )->inContentLanguage()->text();
			if ( $name === '' ) {
				continue;
			} elseif ( $components !== false ) {
				// Valid parameter syntax (with lowercase language code), babel box
				$content .= self::mGenerateBox( $components['code'], $components['level'] );
				$content .= self::mGenerateCategories(
					$components['code'],
					$components['level'],
					$createCategories
				);
			} elseif ( self::mPageExists( $template ) ) {
				// Check for an existing template
				$templateParameters[0] = $template;
				$template = implode( '|', $templateParameters );
				$content .= self::mGenerateNotaBox( $parser->replaceVariables( "{{{$template}}}" ) );
			} elseif ( self::mValidTitle( $template ) ) {
				// Non-existing page, so try again as a babel box,
				// with converting the code to lowercase
				$components2 = self::mParseParameter( $name, /* code to lowercase */
					true );
				if ( $components2 !== false ) {
					$content .= self::mGenerateBox( $components2['code'], $components2['level'] );
					$content .= self::mGenerateCategories(
						$components2['code'],
						$components2['level'],
						$createCategories
					);
				} else {
					// Non-existent page and invalid parameter syntax, red link.
					$content .= self::mGenerateNotaBox( '[[' . $template . ']]' );
				}
			} else {
				// Invalid title, output raw.
				$content .= self::mGenerateNotaBox( $template );
			}
		}

		if ( $wgBabelUseUserLanguage ) {
			$uiLang = $parser->getOptions()->getUserLangObj();
		} else {
			$uiLang = self::$title->getPageLanguage();
		}

		$top = wfMessage( 'babel', self::$title->getDBkey() )->inLanguage( $uiLang );

		if ( $top->isDisabled() ) {
			$top = '';
		} else {
			$top = $top->text();
			$url = wfMessage( 'babel-url' )->inContentLanguage();
			if ( !$url->isDisabled() ) {
				$top = '[[' . $url->text() . '|' . $top . ']]';
			}
			$top = '! class="mw-babel-header" | ' . $top;
		}
		$footer = wfMessage( 'babel-footer', self::$title->getDBkey() )->inLanguage( $uiLang );

		$url = wfMessage( 'babel-footer-url' )->inContentLanguage();
		$showfooter = '';
		if ( !$footer->isDisabled() && !$url->isDisabled() ) {
			$showfooter = '! class="mw-babel-footer" | [[' .
				$url->text() . '|' . $footer->text() . ']]';
		}
		$spacing = Babel::mCssAttrib( 'border-spacing', 'babel-box-cellspacing', true );
		$padding = Babel::mCssAttrib( 'padding', 'babel-box-cellpadding', true );

		if ( $spacing === '' ) {
			$style = ( $padding === '' ) ? '' : ( 'style="' . $padding . '"' );
		} else {
			$style = ( $padding === '' ) ?
				'style="' . $spacing . '"' :
				'style="' . $padding . ' ' . $spacing . '"';
		}

		$tower = <<<EOT
{|$style class="mw-babel-wrapper"
$top
|-
| $content
|-
$showfooter
|}
EOT;

		return $tower;
	}

	/**
	 * Performs a link batch on a series of templates.
	 *
	 * @param $parameters Array: Templates to perform the link batch on.
	 */
	protected static function mTemplateLinkBatch( $parameters ) {
		$titles = array();
		foreach ( $parameters as $name ) {
			$title = Title::newFromText( wfMessage( 'babel-template', $name )->inContentLanguage()->text() );
			if ( is_object( $title ) ) {
				$titles[] = $title;
			}
		}

		$batch = new LinkBatch( $titles );
		$batch->setCaller( __METHOD__ );
		$batch->execute();
	}

	/**
	 * Identify whether or not a page exists.
	 *
	 * @param $name String: Name of the page to check.
	 * @return Boolean: Indication of whether the page exists.
	 */
	protected static function mPageExists( $name ) {
		$titleObj = Title::newFromText( $name );

		return ( is_object( $titleObj ) && $titleObj->exists() );
	}

	/**
	 * Identify whether or not the passed string would make a valid page name.
	 *
	 * @param $name string: Name of page to check.
	 * @return Boolean: Indication of whether or not the title is valid.
	 */
	protected static function mValidTitle( $name ) {
		$titleObj = Title::newFromText( $name );

		return is_object( $titleObj );
	}

	/**
	 * Parse a parameter, getting a language code and level.
	 *
	 * @param $parameter String: Parameter.
	 * @param $strtolower Boolean: Whether to convert the language code to lowercase
	 * @return Array: { 'code' => xx, 'level' => xx }
	 */
	protected static function mParseParameter( $parameter, $strtolower = false ) {
		global $wgBabelDefaultLevel, $wgBabelCategoryNames;
		$return = array();

		$babelcode = $strtolower ? strtolower( $parameter ) : $parameter;
		// Try treating the paramter as a language code (for default level).
		$code = BabelLanguageCodes::getCode( $babelcode );
		if ( $code !== false ) {
			$return['code'] = $code;
			$return['level'] = $wgBabelDefaultLevel;
			return $return;
		}
		// Try splitting the paramter in to language and level, split on last hyphen.
		$lastSplit = strrpos( $parameter, '-' );
		if ( $lastSplit === false ) {
			return false;
		}
		$code = substr( $parameter, 0, $lastSplit );
		$level = substr( $parameter, $lastSplit + 1 );

		$babelcode = $strtolower ? strtolower( $code ) : $code;
		// Validate code.
		$return['code'] = BabelLanguageCodes::getCode( $babelcode );
		if ( $return['code'] === false ) {
			return false;
		}
		// Validate level.
		$level = strtoupper( $level );
		if ( !isset( $wgBabelCategoryNames[$level] ) ) {
			return false;
		}
		$return['level'] = $level;

		return $return;
	}

	/**
	 * Generate an inner item which is not a babel box.
	 *
	 * @param $content String: what's inside the box, in wikitext format.
	 * @return String: A single non-babel box, in wikitext format.
	 */
	protected static function mGenerateNotaBox( $content ) {
		$dir_head = self::$title->getPageLanguage()->getDir();
		$notabox = <<<EOT
<div class="mw-babel-notabox" dir="$dir_head">$content</div>
EOT;

		return $notabox;
	}

	/**
	 * Generate a babel box for the given language and level.
	 *
	 * @param $code String: Language code to use.
	 * @param $level String or Integer: Level of ability to use.
	 * @return String: A single babel box, in wikitext format.
	 */
	protected static function mGenerateBox( $code, $level ) {
		$lang = wfBCP47( $code );
		$portal = wfMessage( 'babel-portal', $code )->inContentLanguage()->plain();
		if ( $portal !== '' ) {
			$portal = "[[$portal|$lang]]";
		} else {
			$portal = $lang;
		}
		$header = "$portal<span class=\"mw-babel-box-level-$level\">-$level</span>";

		$code = strtolower( $code );
		$name = BabelLanguageCodes::getName( $code );
		$code = BabelLanguageCodes::getCode( $code );
		$text = self::mGetText( $name, $code, $level );

		$dir_current = Language::factory( $code )->getDir();

		$spacing = Babel::mCssAttrib( 'border-spacing', 'babel-cellspacing', true );
		$padding = Babel::mCssAttrib( 'padding', 'babel-cellpadding', true );

		if ( $spacing === '' ) {
			$style = ( $padding === '' ) ? '' : ( 'style="' . $padding . '"' );
		} else {
			$style = ( $padding === '' ) ?
				'style="' . $spacing . '"' :
				'style="' . $padding . ' ' . $spacing . '"';
		}

		$dir_head = self::$title->getPageLanguage()->getDir();

		$box = <<<EOT
<div class="mw-babel-box mw-babel-box-$level" dir="$dir_head">
{|$style
! dir="$dir_head" | $header
| dir="$dir_current" lang="$lang" | $text
|}
</div>
EOT;

		return $box;
	}

	/**
	 * Get the text to display in the language box for specific language and
	 * level.
	 *
	 * @param $name string
	 * @param $language String: Language code of language to use.
	 * @param $level String: Level to use.
	 * @return String: Text for display, in wikitext format.
	 */
	protected static function mGetText( $name, $language, $level ) {
		global $wgBabelMainCategory, $wgBabelCategoryNames;

		if ( $wgBabelCategoryNames[$level] === false ) {
			$categoryLevel = self::$title->getFullText();
		} else {
			$categoryLevel = ':Category:' .
				self::mReplaceCategoryVariables( $wgBabelCategoryNames[$level], $language );
		}

		if ( $wgBabelMainCategory === false ) {
			$categoryMain = self::$title->getFullText();
		} else {
			$categoryMain = ':Category:' .
				self::mReplaceCategoryVariables( $wgBabelMainCategory, $language );
		}

		// Give grep a chance to find the usages:
		// babel-0-n, babel-1-n, babel-2-n, babel-3-n, babel-4-n, babel-5-n, babel-N-n
		$text = wfMessage( "babel-$level-n",
			$categoryLevel, $categoryMain, '', self::$title->getDBkey()
		)->inLanguage( $language )->text();

		$fallbackLanguage = Language::getFallbackfor( $language );
		$fallback = wfMessage( "babel-$level-n",
			$categoryLevel, $categoryMain, '', self::$title->getDBkey()
		)->inLanguage( $fallbackLanguage ? $fallbackLanguage : $language )->text();

		// Give grep a chance to find the usages:
		// babel-0, babel-1, babel-2, babel-3, babel-4, babel-5, babel-N
		if ( $text == $fallback ) {
			$text = wfMessage( "babel-$level",
				$categoryLevel, $categoryMain, $name, self::$title->getDBkey()
			)->inLanguage( $language )->text();
		}

		return $text;
	}

	/**
	 * Generate categories for the given language and level.
	 *
	 * @param $code String: Language code to use.
	 * @param $level String or Integer: Level of ability to use.
	 * @param $createCategories Boolean: If true, creates non existing categories;
	 * otherwise, doesn't create them.
	 * @return String: Wikitext to add categories.
	 */
	protected static function mGenerateCategories( $code, $level, $createCategories = true ) {
		global $wgBabelMainCategory, $wgBabelCategoryNames;

		$r = '';

		# Add main category
		if ( $wgBabelMainCategory !== false ) {
			$category = self::mReplaceCategoryVariables( $wgBabelMainCategory, $code );
			$r .= "[[Category:$category|$level]]";
			if ( $createCategories ) {
				BabelAutoCreate::create( $category, $code );
			}
		}

		# Add level category
		if ( $wgBabelCategoryNames[$level] !== false ) {
			$category = self::mReplaceCategoryVariables( $wgBabelCategoryNames[$level], $code );
			$r .= "[[Category:$category]]";
			if ( $createCategories ) {
				BabelAutoCreate::create( $category, $code, $level );
			}
		}

		return $r;
	}

	/**
	 * Replace the placeholder variables from the category names configurtion
	 * array with actual values.
	 *
	 * @param $category String: Category name (containing variables).
	 * @param $code String: Language code of category.
	 * @return String: Category name with variables replaced.
	 */
	protected static function mReplaceCategoryVariables( $category, $code ) {
		global $wgLanguageCode;
		$category = strtr( $category, array(
			'%code%' => $code,
			'%wikiname%' => BabelLanguageCodes::getName( $code, $wgLanguageCode ),
			'%nativename%' => BabelLanguageCodes::getName( $code )
		) );

		return $category;
	}

	/**
	 * Determine a CSS attribute, such as "border-spacing", from a localizeable message.
	 *
	 * @param $name String: name of CSS attribute.
	 * @param $key String: Message key of attribute value.
	 * @param $assumeNumbersArePixels Boolean: if true, treat numbers values as pixels;
	 * otherwise, keep values as is (default: false).
	 * @todo Move this function to a more appropriate place, likely outside the class.
	 * @return Message|string
	 */
	protected static function mCssAttrib( $name, $key, $assumeNumbersArePixels = false ) {
		$value = wfMessage( $key )->inContentLanguage();
		if ( $value->isDisabled() ) {
			$value = '';
		} else {
			$value = htmlentities( $value->text(), ENT_COMPAT, 'UTF-8' );
			if ( $assumeNumbersArePixels && is_numeric( $value ) && $value !== "0" ) {
				//Compatibility: previous babel-box-cellpadding and
				//babel-box-cellspacing entries were in HTML, not CSS
				//and so used numbers without unity as pixels.
				$value .= 'px';
			}
			$value = ' ' . $name . ': ' . $value . ';';
		}

		return $value;
	}

	/**
	 * Determine an HTML attribute, such as "cellspacing" or "title", from a localizeable message.
	 *
	 * @param $name String: name of HTML attribute.
	 * @param $key String: Message key of attribute value.
	 * TODO: move this function to a more appropriate place, likely outside the class.
	 *       or consider to deprecate it as it's not used anymore.
	 * @return Message|string
	 */
	protected static function mHtmlAttrib( $name, $key ) {
		$value = wfMessage( $key )->inContentLanguage();
		if ( $value->isDisabled() ) {
			$value = '';
		} else {
			$value = ' ' . $name . '="' . htmlentities( $value->text(), ENT_COMPAT, 'UTF-8' ) .
				'"'; // must get rid of > and " inside value
		}

		return $value;
	}

	/**
	 * Gets the list of languages a user has set up with Babel
	 *
	 * TODO Can be done much smarter, e.g. by saving the languages in the DB and getting them there
	 * TODO There could be an API module that returns the result of this function
	 *
	 * @param User $user
	 * @param string $level minimal level as given in $wgBabelCategoryNames
	 * @return string[] List of language codes
	 *
	 * @since Version 1.9.0
	 */
	public static function getUserLanguages( User $user, $level = null ) {
		// Right now the function only returns something if the user is categorized appropriately
		// (as defined by the $wgBabelMainCategory setting). If categorization is off, this function
		// will return an empty array.
		// If Babel would save the languages of the user in a Database table, this workaround using
		// the categories would not be needed.
		global $wgBabelMainCategory;
		// If Babel is not configured as required, return nothing.
		// Note also that "Set to false to disable main category".
		if ( $wgBabelMainCategory === false ) {
			return array();
		}

		// The string we construct here will be a pony, it will not be a valid category
		$babelCategoryTitle = Title::makeTitle( NS_CATEGORY, $wgBabelMainCategory );
		// Quote everything to avoid unexpected matches due to parenthesis form
		// It is not necessary to quote any additional chars except the special chars for the regex
		// and perhaps the limiting char, but that should not be respected as anything other than
		// edge delimiter.
		$babelCategoryString = preg_quote( $babelCategoryTitle->getPrefixedDBkey(), '/' );
		// Look for the %code% inside the string and put a group match in the same place
		// This will only work if the previous works so the string isn't misinterpreted as a regular
		// expression itself
		$codeRegex = '/^' . preg_replace( '/%code%/', '(.+?)(-([0-5N]))?', $babelCategoryString ) . '$/';

		$categories = array_keys( $user->getUserPage()->getParentCategories() );

		// We sort on proficiency level
		$result = array();
		foreach ( $categories as $category ) {
			// Only process categories that matches, $match will be created if necessary
			$res = preg_match( $codeRegex, $category, $match );
			if ( $res ) {
				// lowercase the first char, but stay away from the others in case of region codes
				$code = BabelLanguageCodes::getCode( lcfirst( $match[1] ) );
				if ( $code !== false ) {
					$result[$code] = isset( $match[3] ) ? $match[3] : 'N';
				}
			}
		}

		if ( isset( $level ) ) {
			$level = (string)$level;
			// filter down the set, note that this uses a text sort!
			$result = array_filter(
				$result,
				function ( $value ) use ( $level ) {
					return ( strcmp( $value, $level ) >= 0 );
				}
			);
			// sort and retain keys
			uasort(
				$result,
				function ( $a, $b ) {
					return -strcmp( $a, $b );
				}
			);
		}

		return array_keys( $result );
	}
}

Zerion Mini Shell 1.0