2007-06-29 04:33:14 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
class SyntaxHighlight_GeSHi {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Has GeSHi been initialised this session?
|
|
|
|
*/
|
|
|
|
private static $initialised = false;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* List of languages available to GeSHi
|
|
|
|
*/
|
|
|
|
private static $languages = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parser hook
|
|
|
|
*
|
|
|
|
* @param string $text
|
|
|
|
* @param array $args
|
|
|
|
* @param Parser $parser
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public static function parserHook( $text, $args = array(), $parser ) {
|
|
|
|
self::initialise();
|
|
|
|
$text = trim( $text );
|
|
|
|
// Validate language
|
|
|
|
if( isset( $args['lang'] ) ) {
|
|
|
|
$lang = $args['lang'];
|
|
|
|
} else {
|
|
|
|
return self::formatError( htmlspecialchars( wfMsgForContent( 'syntaxhighlight-err-language' ) ) );
|
|
|
|
}
|
|
|
|
if( !preg_match( '/^[A-Za-z_0-9-]*$/', $lang ) )
|
|
|
|
return self::formatError( htmlspecialchars( wfMsgForContent( 'syntaxhighlight-err-language' ) ) );
|
|
|
|
$geshi = self::prepare( $text, $lang );
|
|
|
|
if( !$geshi instanceof GeSHi )
|
|
|
|
return self::formatError( htmlspecialchars( wfMsgForContent( 'syntaxhighlight-err-language' ) ) );
|
|
|
|
// "Enclose" parameter
|
|
|
|
$enclose = isset( $args['enclose'] ) && $args['enclose'] == 'div'
|
|
|
|
? GESHI_HEADER_DIV
|
|
|
|
: GESHI_HEADER_PRE;
|
|
|
|
// Line numbers?
|
|
|
|
if( isset( $args['line'] ) ) {
|
|
|
|
// Force <div> mode to maintain valid XHTML, see
|
|
|
|
// http://sourceforge.net/tracker/index.php?func=detail&aid=1201963&group_id=114997&atid=670231
|
|
|
|
$enclose = GESHI_HEADER_DIV;
|
|
|
|
$geshi->enable_line_numbers( GESHI_FANCY_LINE_NUMBERS );
|
|
|
|
}
|
|
|
|
// Starting line number
|
|
|
|
if( isset( $args['start'] ) )
|
|
|
|
$geshi->start_line_numbers_at( $args['start'] );
|
|
|
|
$geshi->set_header_type( $enclose );
|
|
|
|
// Strict mode
|
|
|
|
if( isset( $args['strict'] ) )
|
|
|
|
$geshi->enable_strict_mode();
|
|
|
|
// Format
|
|
|
|
$out = $geshi->parse_code();
|
|
|
|
$err = $geshi->error();
|
|
|
|
if( $err ) {
|
|
|
|
// Error!
|
|
|
|
return self::formatError( $err );
|
|
|
|
} else {
|
|
|
|
// Armour for Parser::doBlockLevels()
|
|
|
|
if( $enclose == GESHI_HEADER_DIV )
|
|
|
|
$out = str_replace( "\n", '', $out );
|
|
|
|
// Register CSS
|
|
|
|
$parser->mOutput->addHeadItem( self::buildHeadItem( $geshi ), "source-{$lang}" );
|
|
|
|
return $out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-02 23:24:19 +00:00
|
|
|
/**
|
|
|
|
* Hook into Article::view() to provide syntax highlighting for
|
|
|
|
* custom CSS and JavaScript pages
|
|
|
|
*
|
|
|
|
* @param string $text
|
|
|
|
* @param Title $title
|
|
|
|
* @param OutputPage $output
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public static function viewHook( $text, $title, $output ) {
|
|
|
|
// Determine the language
|
|
|
|
preg_match( '!\.(css|js)$!u', $title->getText(), $matches );
|
|
|
|
$lang = $matches[1] == 'css' ? 'css' : 'javascript';
|
|
|
|
// Attempt to format
|
|
|
|
$geshi = self::prepare( $text, $lang );
|
|
|
|
if( $geshi instanceof GeSHi ) {
|
|
|
|
$out = $geshi->parse_code();
|
|
|
|
if( !$geshi->error() ) {
|
|
|
|
// Done
|
|
|
|
$output->addHeadItem( "source-$lang", self::buildHeadItem( $geshi ) );
|
|
|
|
$output->addHtml( "<div dir=\"ltr\">{$out}</div>" );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Bottle out
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2007-06-29 04:33:14 +00:00
|
|
|
/**
|
|
|
|
* Initialise a GeSHi object to format some code, performing
|
|
|
|
* common setup for all our uses of it
|
|
|
|
*
|
|
|
|
* @param string $text
|
|
|
|
* @param string $lang
|
|
|
|
* @return GeSHi
|
|
|
|
*/
|
|
|
|
private static function prepare( $text, $lang ) {
|
|
|
|
self::initialise();
|
|
|
|
$geshi = new GeSHi( $text, $lang );
|
|
|
|
if( $geshi->error() == GESHI_ERROR_NO_SUCH_LANG )
|
|
|
|
return null;
|
|
|
|
$geshi->set_encoding( 'UTF-8' );
|
|
|
|
$geshi->enable_classes();
|
|
|
|
$geshi->set_overall_class( "source-$lang" );
|
|
|
|
$geshi->enable_keyword_links( false );
|
|
|
|
return $geshi;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Prepare a CSS snippet suitable for use as a ParserOutput/OutputPage
|
|
|
|
* head item
|
|
|
|
*
|
|
|
|
* @param GeSHi $geshi
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private static function buildHeadItem( $geshi ) {
|
|
|
|
global $wgUseSiteCss, $wgSquidMaxage;
|
|
|
|
$lang = $geshi->language;
|
|
|
|
$css[] = '<style type="text/css">/*<![CDATA[*/';
|
|
|
|
$css[] = ".source-$lang {line-height: normal;}";
|
|
|
|
$css[] = ".source-$lang li {line-height: normal;}";
|
|
|
|
$css[] = $geshi->get_stylesheet( false );
|
|
|
|
if( $wgUseSiteCss ) {
|
|
|
|
$title = Title::makeTitle( NS_MEDIAWIKI, 'Geshi.css' );
|
|
|
|
$q = "usemsgcache=yes&action=raw&ctype=text/css&smaxage={$wgSquidMaxage}";
|
|
|
|
$css[] = '@import "' . $title->getLocalUrl( $q ) . '";';
|
|
|
|
}
|
|
|
|
$css[] = '/*]]>*/';
|
|
|
|
$css[] = '</style>';
|
|
|
|
return implode( "\n", $css );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format an error message
|
|
|
|
*
|
|
|
|
* @param string $error
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private static function formatError( $error = '' ) {
|
|
|
|
$html = '';
|
|
|
|
if( $error )
|
|
|
|
$html .= "<p>{$error}</p>";
|
|
|
|
$html .= '<p>' . htmlspecialchars( wfMsgForContent( 'syntaxhighlight-specify' ) )
|
|
|
|
. ' <samp><source lang="html">...</source></samp></p>'
|
|
|
|
. '<p>' . htmlspecialchars( wfMsgForContent( 'syntaxhighlight-supported' ) ) . '</p>'
|
|
|
|
. self::formatLanguages();
|
|
|
|
return "<div style=\"border: solid red 1px; padding: .5em;\">{$html}</div>";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Format the list of supported languages
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
private static function formatLanguages() {
|
|
|
|
$langs = self::getSupportedLanguages();
|
|
|
|
$list = array();
|
|
|
|
if( count( $langs ) > 0 ) {
|
|
|
|
for( $i = 0; $i < count( $langs ); $i++ ) {
|
|
|
|
if( preg_match( '!/([^\.]+)\.php$!u', $langs[$i], $m ) )
|
|
|
|
$list[] = '<samp>' . $m[1] . '</samp>';
|
|
|
|
}
|
|
|
|
return '<p style="padding: 0em 1em;">' . implode( ', ', $list ) . '</p>';
|
|
|
|
} else {
|
|
|
|
return '<p>' . htmlspecialchars( wfMsgForContent( 'syntaxhighlight-err-loading' ) ) . '</p>';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the list of supported languages
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
private static function getSupportedLanguages() {
|
|
|
|
if( !is_array( self::$languages ) ) {
|
|
|
|
self::initialise();
|
|
|
|
self::$languages = glob( GESHI_LANG_ROOT . "/*.php" );
|
|
|
|
sort( self::$languages );
|
|
|
|
}
|
|
|
|
return self::$languages;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialise messages and ensure the GeSHi class is loaded
|
|
|
|
*/
|
|
|
|
private static function initialise() {
|
|
|
|
global $wgMessageCache;
|
|
|
|
if( !self::$initialised ) {
|
|
|
|
require_once( dirname( __FILE__ ) . '/SyntaxHighlight_GeSHi.i18n.php' );
|
|
|
|
foreach( efSyntaxHighlight_GeSHiMessages() as $lang => $messages )
|
|
|
|
$wgMessageCache->addMessages( $messages, $lang );
|
|
|
|
if( !class_exists( 'GeSHi' ) )
|
|
|
|
require( 'geshi/geshi.php' );
|
|
|
|
self::$initialised = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-02 23:24:19 +00:00
|
|
|
}
|