2012-01-21 20:38:13 +00:00
|
|
|
/**
|
2012-01-22 01:27:22 +00:00
|
|
|
* Some parser functions, and quite a bunch of stubs of parser functions.
|
2012-02-17 10:23:14 +00:00
|
|
|
* There are still quite a few missing, see
|
|
|
|
* http://www.mediawiki.org/wiki/Help:Magic_words and
|
|
|
|
* http://www.mediawiki.org/wiki/Help:Extension:ParserFunctions.
|
|
|
|
* Instantiated and called by the TemplateHandler extension. Any pf_<prefix>
|
|
|
|
* matching a lower-cased template name prefix up to the first colon will
|
|
|
|
* override that template.
|
2012-01-22 01:27:22 +00:00
|
|
|
*
|
2012-02-22 15:59:11 +00:00
|
|
|
* TODO: Implement these more thoroughly, and test against
|
|
|
|
* extensions/ParserFunction/
|
|
|
|
* convertTests.txt
|
|
|
|
* exprTests.txt
|
|
|
|
* funcsParserTests.txt
|
|
|
|
* stringFunctionTests.txt
|
|
|
|
*
|
2012-01-22 01:27:22 +00:00
|
|
|
* @author Gabriel Wicke <gwicke@wikimedia.org>
|
2012-01-21 20:38:13 +00:00
|
|
|
*/
|
|
|
|
|
2012-05-10 08:04:24 +00:00
|
|
|
var async = require('async');
|
2012-01-22 07:07:16 +00:00
|
|
|
|
2012-01-21 20:38:13 +00:00
|
|
|
function ParserFunctions ( manager ) {
|
|
|
|
this.manager = manager;
|
2012-04-12 13:42:09 +00:00
|
|
|
this.env = manager.env;
|
2012-01-21 20:38:13 +00:00
|
|
|
}
|
|
|
|
|
2012-04-04 09:21:23 +00:00
|
|
|
// Temporary helper.
|
2012-05-10 08:04:24 +00:00
|
|
|
ParserFunctions.prototype._rejoinKV = function ( k, v ) {
|
|
|
|
if ( k.length ) {
|
|
|
|
return k.concat( ['='], v );
|
2012-04-03 16:07:05 +00:00
|
|
|
} else {
|
2012-05-10 08:04:24 +00:00
|
|
|
return v;
|
2012-04-03 16:07:05 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-05-10 08:04:24 +00:00
|
|
|
// XXX: move to frame?
|
|
|
|
ParserFunctions.prototype.expandKV = function ( kv, cb, defaultValue, type ) {
|
|
|
|
if ( type === undefined ) {
|
|
|
|
type = 'tokens/x-mediawiki/expanded';
|
|
|
|
}
|
|
|
|
if ( kv === undefined ) {
|
|
|
|
cb( { tokens: [ defaultValue || '' ] } );
|
|
|
|
} else if ( kv.constructor === String ) {
|
|
|
|
return cb( kv );
|
|
|
|
} else if ( kv.k.constructor === String && kv.v.constructor === String ) {
|
|
|
|
if ( kv.k ) {
|
|
|
|
cb( { tokens: [kv.k + '=' + kv.v] } );
|
|
|
|
} else {
|
|
|
|
cb( { tokens: [kv.v] } );
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
var self = this,
|
|
|
|
getCB = function ( v ) {
|
|
|
|
cb ( { tokens:
|
|
|
|
self._rejoinKV( kv.k, v ) } );
|
|
|
|
};
|
|
|
|
kv.v.get({
|
|
|
|
type: type,
|
|
|
|
cb: getCB,
|
|
|
|
asyncCB: cb
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#if'] = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
2012-01-21 20:38:13 +00:00
|
|
|
if ( target.trim() !== '' ) {
|
2012-04-25 14:35:59 +00:00
|
|
|
//this.env.dp('#if, first branch', target.trim(), argDict[1] );
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[1], cb );
|
2012-01-21 20:38:13 +00:00
|
|
|
} else {
|
2012-04-25 14:35:59 +00:00
|
|
|
//this.env.dp('#if, second branch', target.trim(), argDict[2] );
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[2], cb );
|
2012-01-21 20:38:13 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-04-27 11:57:23 +00:00
|
|
|
ParserFunctions.prototype._switchLookupFallback = function ( frame, kvs, key, dict, cb, v ) {
|
2012-04-25 14:35:59 +00:00
|
|
|
var kv,
|
|
|
|
l = kvs.length;
|
2012-04-26 16:18:08 +00:00
|
|
|
this.manager.env.tp('swl');
|
2012-04-25 14:35:59 +00:00
|
|
|
this.manager.env.dp('_switchLookupFallback', kvs.length, key, v );
|
2012-04-27 11:57:23 +00:00
|
|
|
if ( v && key === v.trim() ) {
|
2012-04-25 14:35:59 +00:00
|
|
|
// found. now look for the next entry with a non-empty key.
|
|
|
|
this.manager.env.dp( 'switch found' );
|
2012-05-10 08:04:24 +00:00
|
|
|
cb = function( res ) { cb ( { tokens: res } ); };
|
2012-04-25 14:35:59 +00:00
|
|
|
for ( var j = 0; j < l; j++) {
|
|
|
|
kv = kvs[j];
|
|
|
|
// XXX: make sure the key is always one of these!
|
|
|
|
if ( kv.k.length ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
return kv.v.get({
|
|
|
|
type: 'tokens/x-mediawiki/expanded',
|
|
|
|
cb: cb,
|
|
|
|
asyncCB: cb
|
|
|
|
});
|
2012-04-25 14:35:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// No value found, return empty string? XXX: check this
|
|
|
|
return cb( { } );
|
|
|
|
} else if ( kvs.length ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
// search for value-only entry which matches
|
2012-04-25 14:35:59 +00:00
|
|
|
var i = 0;
|
|
|
|
if ( v ) {
|
|
|
|
i = 1;
|
|
|
|
}
|
|
|
|
for ( ; i < l; i++ ) {
|
|
|
|
kv = kvs[i];
|
|
|
|
if ( kv.k.length || !kv.v.length ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
// skip entries with keys or empty values
|
2012-04-25 14:35:59 +00:00
|
|
|
continue;
|
|
|
|
} else {
|
2012-05-10 08:04:24 +00:00
|
|
|
if ( ! kv.v.get ) {
|
2012-04-27 11:57:23 +00:00
|
|
|
this.manager.env.ap( kv.v );
|
|
|
|
console.trace();
|
|
|
|
}
|
2012-05-10 08:04:24 +00:00
|
|
|
var self = this;
|
|
|
|
cb({ async: true });
|
|
|
|
return kv.v.get( {
|
|
|
|
cb: process.nextTick.bind( process,
|
|
|
|
self._switchLookupFallback.bind( this, frame,
|
|
|
|
kvs.slice(i+1), key, dict, cb ) ),
|
|
|
|
asyncCB: cb
|
|
|
|
});
|
2012-02-22 14:57:50 +00:00
|
|
|
}
|
2012-04-25 14:35:59 +00:00
|
|
|
}
|
|
|
|
// value not found!
|
|
|
|
if ( '#default' in dict ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
dict['#default'].get({
|
|
|
|
type: 'tokens/x-mediawiki/expanded',
|
|
|
|
cb: function( res ) { cb ( { tokens: res } ); },
|
|
|
|
asyncCB: cb
|
|
|
|
});
|
|
|
|
/*} else if ( kvs.length ) {
|
2012-04-25 14:35:59 +00:00
|
|
|
var lastKV = kvs[kvs.length - 1];
|
|
|
|
if ( lastKV && ! lastKV.k.length ) {
|
|
|
|
cb ( { tokens: lastKV.v } );
|
|
|
|
} else {
|
|
|
|
cb ( {} );
|
2012-05-10 08:04:24 +00:00
|
|
|
}*/
|
2012-04-25 14:35:59 +00:00
|
|
|
} else {
|
2012-05-10 08:04:24 +00:00
|
|
|
// nothing found at all.
|
2012-04-25 14:35:59 +00:00
|
|
|
cb ( {} );
|
2012-02-22 14:57:50 +00:00
|
|
|
}
|
2012-05-10 08:04:24 +00:00
|
|
|
} else {
|
|
|
|
// nothing found at all.
|
|
|
|
cb ( {} );
|
2012-02-22 14:57:50 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-02-17 10:23:14 +00:00
|
|
|
// TODO: Implement
|
2012-02-11 16:43:25 +00:00
|
|
|
// http://www.mediawiki.org/wiki/Help:Extension:ParserFunctions#Grouping_results
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#switch'] = function ( token, frame, cb, args ) {
|
|
|
|
target = args[0].k.trim();
|
|
|
|
this.env.dp( 'switch enter', target, token );
|
|
|
|
// create a dict from the remaining args
|
|
|
|
args.shift();
|
|
|
|
var dict = args.dict();
|
2012-04-27 13:59:01 +00:00
|
|
|
if ( target && dict[target] !== undefined ) {
|
2012-04-25 14:35:59 +00:00
|
|
|
this.env.dp( 'switch found: ', target, dict, ' res=', dict[target] );
|
2012-05-10 08:04:24 +00:00
|
|
|
dict[target].get({
|
|
|
|
type: 'tokens/x-mediawiki/expanded',
|
|
|
|
cb: function( res ) { cb ( { tokens: res } ); },
|
|
|
|
asyncCB: cb
|
|
|
|
});
|
2012-02-22 14:57:50 +00:00
|
|
|
} else {
|
2012-04-27 11:57:23 +00:00
|
|
|
this._switchLookupFallback( frame, args, target, dict, cb );
|
2012-01-21 20:38:13 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// #ifeq
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#ifeq'] = function ( token, frame, cb, args ) {
|
|
|
|
if ( args.length < 3 ) {
|
|
|
|
cb( {} );
|
2012-01-21 20:38:13 +00:00
|
|
|
} else {
|
2012-04-27 10:18:30 +00:00
|
|
|
var b = args[1].v;
|
2012-05-10 08:04:24 +00:00
|
|
|
b.get( { cb: this._ifeq_worker.bind( this, cb, args ), asyncCB: cb } );
|
2012-01-21 20:38:13 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-04-27 10:18:30 +00:00
|
|
|
ParserFunctions.prototype._ifeq_worker = function ( cb, args, b ) {
|
|
|
|
if ( args[0].k.trim() === b.trim() ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[2], cb );
|
2012-04-27 10:18:30 +00:00
|
|
|
} else {
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[3], cb );
|
2012-04-27 10:18:30 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#expr'] = function ( token, frame, cb, args ) {
|
|
|
|
var res,
|
|
|
|
target = args[0].k;
|
|
|
|
if ( target ) {
|
|
|
|
try {
|
|
|
|
// FIXME: make this safe and implement MW expressions!
|
|
|
|
var f = new Function ( 'return (' + target + ')' );
|
|
|
|
res = f();
|
|
|
|
} catch ( e ) {
|
|
|
|
return cb( { tokens: [ 'class="error" in expression ' + target ] } );
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
res = '';
|
|
|
|
}
|
|
|
|
cb( { tokens: [ res.toString() ] } );
|
|
|
|
};
|
|
|
|
|
|
|
|
ParserFunctions.prototype['pf_#ifexpr'] = function ( token, frame, cb, args ) {
|
2012-04-27 11:57:23 +00:00
|
|
|
this.env.dp( '#ifexp: ', args );
|
2012-04-25 14:35:59 +00:00
|
|
|
var res = null,
|
|
|
|
target = args[0].k;
|
|
|
|
if ( target ) {
|
|
|
|
try {
|
|
|
|
// FIXME: make this safe, and fully implement MW expressions!
|
|
|
|
var f = new Function ( 'return (' + target + ')' );
|
|
|
|
res = f();
|
|
|
|
} catch ( e ) {
|
|
|
|
cb( { tokens: [ 'class="error" in expression ' + target ] } );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( res ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[1], cb );
|
2012-04-25 14:35:59 +00:00
|
|
|
} else {
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[2], cb );
|
2012-04-25 14:35:59 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
ParserFunctions.prototype['pf_#iferror'] = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
if ( target.indexOf( 'class="error"' ) >= 0 ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[1], cb );
|
2012-04-25 14:35:59 +00:00
|
|
|
} else {
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[1], cb, target );
|
2012-04-25 14:35:59 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
ParserFunctions.prototype.pf_lc = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [ args[0].k.toLowerCase() ] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-01-22 01:27:22 +00:00
|
|
|
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_uc = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [ args[0].k.toUpperCase() ] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-01-22 01:27:22 +00:00
|
|
|
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_ucfirst = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
2012-01-21 20:38:13 +00:00
|
|
|
if ( target ) {
|
2012-04-25 14:35:59 +00:00
|
|
|
cb( { tokens: [ target[0].toUpperCase() + target.substr(1) ] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
} else {
|
2012-04-25 14:35:59 +00:00
|
|
|
cb( {} );
|
2012-01-21 20:38:13 +00:00
|
|
|
}
|
|
|
|
};
|
2012-01-22 01:27:22 +00:00
|
|
|
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_lcfirst = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
2012-01-21 20:38:13 +00:00
|
|
|
if ( target ) {
|
2012-04-25 14:35:59 +00:00
|
|
|
cb( { tokens: [ target[0].toLowerCase() + target.substr(1) ] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
} else {
|
2012-04-25 14:35:59 +00:00
|
|
|
cb( {} );
|
2012-01-21 20:38:13 +00:00
|
|
|
}
|
|
|
|
};
|
2012-05-10 08:04:24 +00:00
|
|
|
ParserFunctions.prototype.pf_padleft = function ( token, frame, cb, params ) {
|
|
|
|
var target = params[0].k;
|
|
|
|
if ( ! params[1] ) {
|
|
|
|
return cb( {} );
|
2012-04-25 14:35:59 +00:00
|
|
|
}
|
2012-05-10 08:04:24 +00:00
|
|
|
// expand parameters 1 and 2
|
|
|
|
params.getSlice( {
|
|
|
|
type: 'text/x-mediawiki/expanded',
|
|
|
|
cb: function ( args ) {
|
|
|
|
if ( args[0].v > 0) {
|
|
|
|
var pad = '0';
|
|
|
|
if ( args[1] && args[1].v !== '' ) {
|
|
|
|
pad = args[1].v;
|
|
|
|
}
|
|
|
|
var n = args[0].v;
|
|
|
|
while ( target.length < n ) {
|
|
|
|
target = pad + target;
|
|
|
|
}
|
|
|
|
cb( { tokens: [target] } );
|
|
|
|
} else {
|
|
|
|
self.env.dp( 'padleft no pad width', args );
|
|
|
|
cb( {} );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
1, 3);
|
2012-04-25 14:35:59 +00:00
|
|
|
};
|
2012-05-10 08:04:24 +00:00
|
|
|
|
|
|
|
ParserFunctions.prototype.pf_padright = function ( token, frame, cb, params ) {
|
|
|
|
var target = params[0].k;
|
|
|
|
if ( ! params[1] ) {
|
|
|
|
return cb( {} );
|
2012-02-09 13:44:20 +00:00
|
|
|
}
|
2012-05-10 08:04:24 +00:00
|
|
|
// expand parameters 1 and 2
|
|
|
|
params.getSlice( {
|
|
|
|
type: 'text/x-mediawiki/expanded',
|
|
|
|
cb: function ( args ) {
|
|
|
|
if ( args[0].v > 0) {
|
|
|
|
if ( args[1] && args[1].v !== '' ) {
|
|
|
|
pad = args[1].v;
|
|
|
|
} else {
|
|
|
|
pad = '0';
|
|
|
|
}
|
|
|
|
var n = args[0].v;
|
|
|
|
while ( target.length < n ) {
|
|
|
|
target = target + pad;
|
|
|
|
}
|
|
|
|
cb( { tokens: [target] } );
|
|
|
|
} else {
|
|
|
|
cb( {} );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
1, 3 );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-01-21 20:38:13 +00:00
|
|
|
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#tag'] = function ( token, frame, cb, args ) {
|
2012-02-17 10:23:14 +00:00
|
|
|
// TODO: handle things like {{#tag:nowiki|{{{input1|[[shouldnotbelink]]}}}}}
|
2012-02-08 15:10:30 +00:00
|
|
|
// https://www.mediawiki.org/wiki/Future/Parser_development#Token_stream_transforms
|
2012-04-25 14:35:59 +00:00
|
|
|
var target = args[0].k;
|
2012-05-10 08:04:24 +00:00
|
|
|
args[1].v.get({
|
|
|
|
type: 'tokens/x-mediawiki/expanded',
|
|
|
|
cb: this.tag_worker.bind( this, target, cb ),
|
|
|
|
asyncCB: cb
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
ParserFunctions.prototype.tag_worker = function( target, cb, content ) {
|
|
|
|
cb({
|
|
|
|
tokens: [ new TagTk( target ) ]
|
|
|
|
.concat( content,
|
|
|
|
[ new EndTagTk( target ) ] )
|
|
|
|
});
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
|
|
|
|
2012-05-10 08:04:24 +00:00
|
|
|
|
2012-02-17 10:23:14 +00:00
|
|
|
// TODO: These are just quick wrappers for now, optimize!
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentyear = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'Y', [], {} ) );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentmonth = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'm', [], {} ) );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentmonthname = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'F', [], {} ) );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-02-22 16:41:01 +00:00
|
|
|
// XXX Actually use genitive form!
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentmonthnamegen = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'F', [], {} ) );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentmonthabbrev = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'M', [], {} ) );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentweek = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'W', [], {} ) );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentdow = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'w', [], {} ) );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentday = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'j', [], {} ) );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentday2 = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'd', [], {} ) );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentdayname = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'l', [], {} ) );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currenttime = function ( token, frame, cb, args ) {
|
|
|
|
cb( this._pf_time_tokens( 'H:i', [], {} ) );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-02-09 13:44:20 +00:00
|
|
|
|
2012-02-17 10:23:14 +00:00
|
|
|
// A first approximation of time stuff.
|
|
|
|
// TODO: Implement time spec (+ 1 day etc), check if formats are complete etc.
|
2012-02-21 11:24:20 +00:00
|
|
|
// See http://www.mediawiki.org/wiki/Help:Extension:ParserFunctions#.23time
|
|
|
|
// for the full list of requirements!
|
|
|
|
//
|
|
|
|
// First (very rough) approximation below based on
|
|
|
|
// http://jacwright.com/projects/javascript/date_format/, MIT licensed.
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#time'] = function ( token, frame, cb, args ) {
|
|
|
|
cb ( { tokens: this._pf_time( args[0].k, args.slice(1) ) } );
|
|
|
|
};
|
|
|
|
|
|
|
|
ParserFunctions.prototype._pf_time_tokens = function ( target, args ) {
|
|
|
|
return { tokens: this._pf_time( target, args ) };
|
|
|
|
};
|
|
|
|
|
|
|
|
ParserFunctions.prototype._pf_time = function ( target, args ) {
|
2012-01-22 07:07:16 +00:00
|
|
|
var res,
|
|
|
|
tpl = target.trim();
|
|
|
|
//try {
|
2012-04-25 14:35:59 +00:00
|
|
|
// var date = new Date( this.env.tokensToString( args[1].v ) );
|
2012-02-01 16:30:43 +00:00
|
|
|
// res = [ date.format( target ) ];
|
2012-01-22 07:07:16 +00:00
|
|
|
//} catch ( e ) {
|
2012-04-12 13:42:09 +00:00
|
|
|
// this.env.dp( 'ERROR: #time ' + e );
|
2012-01-22 07:07:16 +00:00
|
|
|
|
|
|
|
try {
|
2012-02-01 16:30:43 +00:00
|
|
|
res = [ new Date().format ( tpl ) ];
|
2012-01-22 07:07:16 +00:00
|
|
|
} catch ( e2 ) {
|
2012-04-12 13:42:09 +00:00
|
|
|
this.env.dp( 'ERROR: #time ' + e2 );
|
2012-02-01 16:30:43 +00:00
|
|
|
res = [ new Date().toString() ];
|
2012-01-22 07:07:16 +00:00
|
|
|
}
|
2012-04-25 14:35:59 +00:00
|
|
|
return res;
|
2012-01-22 07:07:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Simulates PHP's date function
|
2012-04-25 14:35:59 +00:00
|
|
|
// FIXME: don't patch Date.prototype!
|
2012-01-22 07:07:16 +00:00
|
|
|
Date.prototype.format = function(format) {
|
|
|
|
var returnStr = '';
|
|
|
|
var replace = Date.replaceChars;
|
|
|
|
for (var i = 0; i < format.length; i++) {
|
|
|
|
var curChar = format.charAt(i);
|
|
|
|
if (i - 1 >= 0 && format.charAt(i - 1) == "\\") {
|
|
|
|
returnStr += curChar;
|
|
|
|
}
|
|
|
|
else if (replace[curChar]) {
|
|
|
|
returnStr += replace[curChar].call(this);
|
|
|
|
} else if (curChar != "\\"){
|
|
|
|
returnStr += curChar;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return returnStr;
|
|
|
|
};
|
|
|
|
|
2012-02-01 16:30:43 +00:00
|
|
|
// XXX: support localization
|
2012-01-22 07:07:16 +00:00
|
|
|
Date.replaceChars = {
|
2012-02-13 14:23:48 +00:00
|
|
|
shortMonths: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug',
|
|
|
|
'Sep', 'Oct', 'Nov', 'Dec'],
|
|
|
|
longMonths: ['January', 'February', 'March', 'April', 'May', 'June',
|
|
|
|
'July', 'August', 'September', 'October', 'November', 'December'],
|
2012-01-22 07:07:16 +00:00
|
|
|
shortDays: ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'],
|
2012-02-13 14:23:48 +00:00
|
|
|
longDays: ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday',
|
|
|
|
'Friday', 'Saturday'],
|
2012-01-22 07:07:16 +00:00
|
|
|
|
|
|
|
// Day
|
|
|
|
d: function() { return (this.getDate() < 10 ? '0' : '') + this.getDate(); },
|
|
|
|
D: function() { return Date.replaceChars.shortDays[this.getDay()]; },
|
|
|
|
j: function() { return this.getDate(); },
|
|
|
|
l: function() { return Date.replaceChars.longDays[this.getDay()]; },
|
|
|
|
N: function() { return this.getDay() + 1; },
|
2012-02-13 14:23:48 +00:00
|
|
|
S: function() {
|
|
|
|
return (this.getDate() % 10 == 1 &&
|
|
|
|
this.getDate() != 11 ? 'st' : (this.getDate() % 10 == 2 &&
|
|
|
|
this.getDate() != 12 ? 'nd' : (this.getDate() % 10 == 3 &&
|
|
|
|
this.getDate() != 13 ? 'rd' : 'th')));
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
w: function() { return this.getDay(); },
|
2012-02-13 14:23:48 +00:00
|
|
|
z: function() {
|
|
|
|
var d = new Date(this.getFullYear(),0,1);
|
|
|
|
return Math.ceil((this - d) / 86400000);
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
// Week
|
2012-02-13 14:23:48 +00:00
|
|
|
W: function() {
|
|
|
|
var d = new Date(this.getFullYear(), 0, 1);
|
|
|
|
return Math.ceil((((this - d) / 86400000) + d.getDay() + 1) / 7);
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
// Month
|
|
|
|
F: function() { return Date.replaceChars.longMonths[this.getMonth()]; },
|
|
|
|
m: function() { return (this.getMonth() < 9 ? '0' : '') + (this.getMonth() + 1); },
|
|
|
|
M: function() { return Date.replaceChars.shortMonths[this.getMonth()]; },
|
|
|
|
n: function() { return this.getMonth() + 1; },
|
2012-02-13 14:23:48 +00:00
|
|
|
t: function() {
|
|
|
|
var d = new Date();
|
|
|
|
return new Date(d.getFullYear(), d.getMonth(), 0).getDate();
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
// Year
|
2012-02-13 14:23:48 +00:00
|
|
|
L: function() {
|
|
|
|
var year = this.getFullYear();
|
|
|
|
return (year % 400 === 0 || (year % 100 !== 0 && year % 4 === 0));
|
|
|
|
},
|
|
|
|
o: function() {
|
|
|
|
var d = new Date(this.valueOf());
|
|
|
|
d.setDate(d.getDate() - ((this.getDay() + 6) % 7) + 3);
|
|
|
|
return d.getFullYear();
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
Y: function() { return this.getFullYear(); },
|
|
|
|
y: function() { return ('' + this.getFullYear()).substr(2); },
|
|
|
|
// Time
|
|
|
|
a: function() { return this.getHours() < 12 ? 'am' : 'pm'; },
|
|
|
|
A: function() { return this.getHours() < 12 ? 'AM' : 'PM'; },
|
2012-02-13 14:23:48 +00:00
|
|
|
B: function() {
|
|
|
|
return Math.floor((((this.getUTCHours() + 1) % 24) +
|
|
|
|
this.getUTCMinutes() / 60 +
|
|
|
|
this.getUTCSeconds() / 3600) * 1000 / 24);
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
g: function() { return this.getHours() % 12 || 12; },
|
|
|
|
G: function() { return this.getHours(); },
|
2012-02-13 14:23:48 +00:00
|
|
|
h: function() {
|
|
|
|
return ((this.getHours() % 12 || 12) < 10 ? '0' : '') +
|
|
|
|
(this.getHours() % 12 || 12);
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
H: function() { return (this.getHours() < 10 ? '0' : '') + this.getHours(); },
|
|
|
|
i: function() { return (this.getMinutes() < 10 ? '0' : '') + this.getMinutes(); },
|
|
|
|
s: function() { return (this.getSeconds() < 10 ? '0' : '') + this.getSeconds(); },
|
2012-02-13 14:23:48 +00:00
|
|
|
u: function() {
|
|
|
|
var m = this.getMilliseconds();
|
|
|
|
return (m < 10 ? '00' : (m < 100 ? '0' : '')) + m;
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
// Timezone
|
|
|
|
e: function() { return "Not Yet Supported"; },
|
|
|
|
I: function() { return "Not Yet Supported"; },
|
2012-02-13 14:23:48 +00:00
|
|
|
O: function() {
|
|
|
|
return (-this.getTimezoneOffset() < 0 ? '-' : '+') +
|
|
|
|
(Math.abs(this.getTimezoneOffset() / 60) < 10 ? '0' : '') +
|
|
|
|
(Math.abs(this.getTimezoneOffset() / 60)) + '00';
|
|
|
|
},
|
|
|
|
P: function() {
|
|
|
|
return (-this.getTimezoneOffset() < 0 ? '-' : '+') +
|
|
|
|
(Math.abs(this.getTimezoneOffset() / 60) < 10 ? '0' : '') +
|
|
|
|
(Math.abs(this.getTimezoneOffset() / 60)) + ':00';
|
|
|
|
},
|
|
|
|
T: function() {
|
|
|
|
var m = this.getMonth();
|
|
|
|
this.setMonth(0);
|
|
|
|
var result = this.toTimeString().replace(/^.+ \(?([^\)]+)\)?$/, '$1');
|
|
|
|
this.setMonth(m);
|
|
|
|
return result;
|
|
|
|
},
|
2012-01-22 07:07:16 +00:00
|
|
|
Z: function() { return -this.getTimezoneOffset() * 60; },
|
|
|
|
// Full Date/Time
|
2012-02-13 14:23:48 +00:00
|
|
|
c: function() { return this.format("Y-m-d\\TH:i:sP"); },
|
2012-01-22 07:07:16 +00:00
|
|
|
r: function() { return this.toString(); },
|
|
|
|
U: function() { return this.getTime() / 1000; }
|
|
|
|
};
|
|
|
|
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_localurl = function ( token, frame, cb, args ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
var target = args[0].k,
|
|
|
|
env = this.env,
|
|
|
|
self = this;
|
2012-04-25 14:35:59 +00:00
|
|
|
args = args.slice(1);
|
2012-05-10 08:04:24 +00:00
|
|
|
async.map(
|
|
|
|
args,
|
|
|
|
function ( item, cb ) {
|
|
|
|
self.expandKV( item, function ( res ) { cb( null, res.tokens ); }, '', 'text/x-mediawiki/expanded' );
|
|
|
|
},
|
|
|
|
function ( err, expandedArgs ) {
|
|
|
|
if ( err ) {
|
|
|
|
console.trace();
|
|
|
|
throw( err );
|
|
|
|
}
|
|
|
|
cb({ tokens: [ '/' +
|
|
|
|
// FIXME! Figure out correct prefix to use
|
|
|
|
//this.env.wgScriptPath +
|
|
|
|
'index' +
|
|
|
|
env.wgScriptExtension + '?title=' +
|
|
|
|
env.normalizeTitle( target ) + '&' +
|
|
|
|
expandedArgs.join('&') ]
|
|
|
|
});
|
|
|
|
}
|
|
|
|
);
|
2012-02-07 10:28:23 +00:00
|
|
|
};
|
2012-01-31 16:50:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stub section: Pick any of these and actually implement them!
|
|
|
|
*/
|
|
|
|
|
2012-02-17 10:23:14 +00:00
|
|
|
// The page name and similar information should be carried around in
|
2012-04-12 13:42:09 +00:00
|
|
|
// this.env
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_formatnum = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [ target ] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_currentpage = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [ target ] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_pagenamee = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [ target.split(':', 2)[1] || '' ] } );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_fullpagename = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: target && [target] || ["http://example.com/fixme/"] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_fullpagenamee = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: target && [target] || ["http://example.com/fixme/"] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-02-17 10:23:14 +00:00
|
|
|
// This should be doable with the information in the envirionment
|
2012-04-12 13:42:09 +00:00
|
|
|
// (this.env) already.
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_fullurl = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: target && [target] || ["http://example.com/fixme/"] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_urlencode = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
2012-04-12 13:42:09 +00:00
|
|
|
this.env.tp( 'urlencode: ' + target );
|
2012-04-25 14:35:59 +00:00
|
|
|
cb( { tokens: [encodeURIComponent(target.trim())] } );
|
2012-01-31 16:50:16 +00:00
|
|
|
};
|
2012-02-17 10:23:14 +00:00
|
|
|
|
|
|
|
// The following items all depends on information from the Wiki, so are hard
|
|
|
|
// to implement independently. Some might require using action=parse in the
|
|
|
|
// API to get the value. See
|
|
|
|
// http://www.mediawiki.org/wiki/Parsoid#Token_stream_transforms,
|
|
|
|
// http://etherpad.wikimedia.org/ParserNotesExtensions and
|
|
|
|
// http://www.mediawiki.org/wiki/Wikitext_parser/Environment.
|
|
|
|
// There might be better solutions for some of these.
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#ifexist'] = function ( token, frame, cb, args ) {
|
2012-05-10 08:04:24 +00:00
|
|
|
this.expandKV( args[1], cb );
|
2012-02-17 10:23:14 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_pagesize = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [ '100' ] } );
|
2012-02-17 10:23:14 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_sitename = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [ "MediaWiki" ] } );
|
2012-02-17 10:23:14 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_anchorencode = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [ target.trim() ] } );
|
2012-01-21 20:38:13 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_protectionlevel = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [''] } );
|
2012-01-31 16:50:16 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_ns = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [target] } );
|
2012-01-31 16:50:16 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_subjectspace = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: ['Main'] } );
|
2012-01-31 16:50:16 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_talkspace = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: ['Talk'] } );
|
2012-01-31 16:50:16 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_numberofarticles = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: ["1"] } );
|
2012-02-09 13:44:20 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype['pf_#language'] = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [target] } );
|
2012-02-11 16:43:25 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_contentlang = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: ['en'] } );
|
2012-02-13 17:02:23 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_numberoffiles = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: ['2'] } );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_namespace = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [target.split(':').pop() || 'Main'] } );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_namespacee = function ( token, frame, cb, args ) {
|
|
|
|
var target = args[0].k;
|
|
|
|
cb( { tokens: [target.split(':').pop() || 'Main'] } );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_pagename = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [this.env.pageName] } );
|
2012-04-11 14:34:27 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_pagenamebase = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [this.env.pageName] } );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
2012-04-25 14:35:59 +00:00
|
|
|
ParserFunctions.prototype.pf_scriptpath = function ( token, frame, cb, args ) {
|
|
|
|
cb( { tokens: [this.env.wgScriptPath] } );
|
2012-02-22 16:41:01 +00:00
|
|
|
};
|
|
|
|
|
2012-01-21 20:38:13 +00:00
|
|
|
|
|
|
|
if (typeof module == "object") {
|
|
|
|
module.exports.ParserFunctions = ParserFunctions;
|
|
|
|
}
|