local str = {}
function str.extract( frame )
local new_args = str._getParameters( frame.args, {'source', 'target', 'num', 'plain', 'start1', 'start2', 'start3', 'start4', 'start5', 'start6', 'comp', 'comp1', 'comp2', 'comp3', 'comp4', 'comp5', 'comp6' } );
local source_str = new_args['source'] or '';
local pattern = new_args['target'] or '';
local start_pos = tonumber(new_args['start']) or 1;
local start_pos2 = tonumber(new_args['start2']) or 1;
local start_pos3 = tonumber(new_args['start3']) or 1;
local start_pos4 = tonumber(new_args['start4']) or 1;
local start_pos5 = tonumber(new_args['start5']) or 1;
local start_pos6 = tonumber(new_args['start6']) or 1;
local comp = new_args['comp'] or '';
local comp1 = new_args['comp1'] or '';
local comp2 = new_args['comp2'] or '';
local comp3 = new_args['comp3'] or '';
local comp4 = new_args['comp4'] or '';
local comp5 = new_args['comp5'] or '';
local comp6 = new_args['comp6'] or '';
local num = tonumber(new_args['num']) or 1;
local plain = new_args['plain'] or true;
if source_str == '' or pattern == '' then
return 0;
end
source_str = mw.ustring.sub(source_str,1,mw.ustring.len(source_str))
plain = str._getBoolean( plain );
local start1 = mw.ustring.find( source_str, pattern, start_pos, plain )
if start1 == nil then
start1 = 0
end
if start1 > 0 then
start_pos2 = start1 + 1;
end
local start2 = mw.ustring.find( source_str, pattern, start_pos2, plain )
if start2 == nil then
start2 = 0
end
if start2 == 0 then
start_pos3 = start2
else start_pos3 = start2 + 1;
end
local start3 = mw.ustring.find( source_str, pattern, start_pos3, plain )
if start3 == nil then
start3 = 0
end
if start_pos3 == 0 then start3 = 0 end
if start3 == 0 then
start_pos4 = start3
else start_pos4 = start3 + 1;
end
local start4 = mw.ustring.find( source_str, pattern, start_pos4, plain )
if start4 == nil then
start4 = 0
end
if start_pos4 == 0 then start4 = 0 end
if start4 == 0 then
start_pos5 = start4
else start_pos5 = start4 + 1;
end
local start5 = mw.ustring.find( source_str, pattern, start_pos5, plain )
if start5 == nil then
start5 = 0
end
if start_pos5 == 0 then start5 = 0 end
if start5 == 0 then
start_pos6 = start5
else start_pos6 = start5 + 1;
end
local start6 = mw.ustring.find( source_str, pattern, start_pos6, plain )
if start6 == nil then
start6 = 0
end
if start_pos6 == 0 then start6 = 0 end
if start1 > 0 then
comp1 = mw.ustring.sub( source_str, 1, start1-1)
end
if start2 > 0 then
comp2 = mw.ustring.sub( source_str, start1+1, start2-1)
end
if start3 > 0 then
comp3 = mw.ustring.sub( source_str, start2+1, start3-1)
end
if start4 > 0 then
comp4 = mw.ustring.sub( source_str, start3+1, start4-1)
end
if start5 > 0 then
comp5 = mw.ustring.sub( source_str, start4+1, start5-1)
end
if start6 > 0 then
comp6 = mw.ustring.sub( source_str, start5+1, start6-1)
end
if num == 1 then
comp = comp1
end
if num == 2 then
comp = comp2
end
if num == 3 then
comp = comp3
end
if num == 4 then
comp = comp4
end
if num == 5 then
comp = comp5
end
if num == 6 then
comp = comp6
end
return comp
end
function str.len( frame )
local new_args = str._getParameters( frame.args, {'s'} );
local s = new_args['s'] or '';
return mw.ustring.len( s )
end
function str.sub( frame )
local new_args = str._getParameters( frame.args, { 's', 'i', 'j' } );
local s = new_args['s'] or '';
local i = tonumber( new_args['i'] ) or 1;
local j = tonumber( new_args['j'] ) or -1;
local len = mw.ustring.len( s );
-- Convert negatives for range checking
if i < 0 then
i = len + i + 1;
end
if j < 0 then
j = len + j + 1;
end
if i > len or j > len or i < 1 or j < 1 then
return str._error( 'String subset index out of range' );
end
if j < i then
return str._error( 'String subset indices out of order' );
end
return mw.ustring.sub( s, i, j )
end
function str.sublength( frame )
local i = tonumber( frame.args.i ) or 0
local len = tonumber( frame.args.len )
return mw.ustring.sub( frame.args.s, i + 1, len and ( i + len ) )
end
function str.match( frame )
local new_args = str._getParameters( frame.args, {'s', 'pattern', 'start', 'match', 'plain', 'nomatch'} );
local s = new_args['s'] or '';
local start = tonumber( new_args['start'] ) or 1;
local plain_flag = str._getBoolean( new_args['plain'] or false );
local pattern = new_args['pattern'] or '';
local match_index = math.floor( tonumber(new_args['match']) or 1 );
local nomatch = new_args['nomatch'];
if s == '' then
return str._error( 'Target string is empty' );
end
if pattern == '' then
return str._error( 'Pattern string is empty' );
end
if math.abs(start) < 1 or math.abs(start) > mw.ustring.len( s ) then
return str._error( 'Requested start is out of range' );
end
if match_index == 0 then
return str._error( 'Match index is out of range' );
end
if plain_flag then
pattern = str._escapePattern( pattern );
end
local result
if match_index == 1 then
-- Find first match is simple case
result = mw.ustring.match( s, pattern, start )
else
if start > 1 then
s = mw.ustring.sub( s, start );
end
local iterator = mw.ustring.gmatch(s, pattern);
if match_index > 0 then
-- Forward search
for w in iterator do
match_index = match_index - 1;
if match_index == 0 then
result = w;
break;
end
end
else
-- Reverse search
local result_table = {};
local count = 1;
for w in iterator do
result_table[count] = w;
count = count + 1;
end
result = result_table[ count + match_index ];
end
end
if result == nil then
if nomatch == nil then
return str._error( 'Match not found' );
else
return nomatch;
end
else
return result;
end
end
function str.pos( frame )
local new_args = str._getParameters( frame.args, {'target', 'pos'} );
local target_str = new_args['target'] or '';
local pos = tonumber( new_args['pos'] ) or 0;
return mw.ustring.sub( target_str, pos, pos );
end
function str.str_find( frame )
local new_args = str._getParameters( frame.args, {'source', 'target'} );
local source_str = new_args['source'] or '';
local target_str = new_args['target'] or '';
if target_str == '' then
return 1;
end
local start = mw.ustring.find( source_str, target_str, 1, true )
if start == nil then
start = -1
end
return start
end
function str.find( frame )
local new_args = str._getParameters( frame.args, {'source', 'target', 'start', 'plain' } );
local source_str = new_args['source'] or '';
local pattern = new_args['target'] or '';
local start_pos = tonumber(new_args['start']) or 1;
local plain = new_args['plain'] or true;
if source_str == '' or pattern == '' then
return 0;
end
plain = str._getBoolean( plain );
local start = mw.ustring.find( source_str, pattern, start_pos, plain )
if start == nil then
start = 0
end
return start
end
function str.replace( frame )
local new_args = str._getParameters( frame.args, {'source', 'pattern', 'replace', 'count', 'plain' } );
local source_str = new_args['source'] or '';
local pattern = new_args['pattern'] or '';
local replace = new_args['replace'] or '';
local count = tonumber( new_args['count'] );
local plain = new_args['plain'] or true;
if source_str == '' or pattern == '' then
return source_str;
end
plain = str._getBoolean( plain );
if plain then
pattern = str._escapePattern( pattern );
replace = mw.ustring.gsub( replace, "%%", "%%%%" ); --Only need to escape replacement sequences.
end
local result;
if count ~= nil then
result = mw.ustring.gsub( source_str, pattern, replace, count );
else
result = mw.ustring.gsub( source_str, pattern, replace );
end
return result;
end
function str._getParameters( frame_args, arg_list )
local new_args = {};
local index = 1;
local value;
for i,arg in ipairs( arg_list ) do
value = frame_args[arg]
if value == nil then
value = frame_args[index];
index = index + 1;
end
new_args[arg] = value;
end
return new_args;
end
function str._error( error_str )
local frame = mw.getCurrentFrame();
local error_category = frame.args.error_category or 'Errors reported by Module String';
local ignore_errors = frame.args.ignore_errors or false;
local no_category = frame.args.no_category or false;
if str._getBoolean(ignore_errors) then
return '';
end
local error_str = '<strong class="error">String Module Error: ' .. error_str .. '</strong>';
if error_category ~= '' and not str._getBoolean( no_category ) then
error_str = '[[Category:' .. error_category .. ']]' .. error_str;
end
return error_str;
end
function str._getBoolean( boolean_str )
local boolean_value;
if type( boolean_str ) == 'string' then
boolean_str = boolean_str:lower();
if boolean_str == 'false' or boolean_str == 'no' or boolean_str == '0'
or boolean_str == '' then
boolean_value = false;
else
boolean_value = true;
end
elseif type( boolean_str ) == 'boolean' then
boolean_value = boolean_str;
else
error( 'No boolean value found' );
end
return boolean_value
end
function str._escapePattern( pattern_str )
return mw.ustring.gsub( pattern_str, "([%(%)%.%%%+%-%*%?%[%^%$%]])", "%%%1" );
end
return str