Modul:Sort: Unterschied zwischen den Versionen
w>PerfektesChaos (update) |
K (14 Versionen von wikivoyage:Modul:Sort importiert) |
||
(12 dazwischenliegende Versionen von 4 Benutzern werden nicht angezeigt) | |||
Zeile 1: | Zeile 1: | ||
--[=[ | local Sort = { suite = "Sort", | ||
serial = "2019-10-29", | |||
item = 24205172 } | |||
--[=[ | |||
Sort | Sort | ||
]=] | ]=] | ||
local Failsafe = Sort | |||
local GlobalMod = Sort | |||
local Sort = { } | local foreignModule = function ( access, advanced, append, alt, alert ) | ||
-- Fetch global module | |||
-- Precondition: | |||
-- access -- string, with name of base module | |||
-- advanced -- true, for require(); else mw.loadData() | |||
-- append -- string, with subpage part, if any; or false | |||
-- alt -- number, of wikidata item of root; or false | |||
-- alert -- true, for throwing error on data problem | |||
-- Postcondition: | |||
-- Returns whatever, probably table | |||
-- 2019-10-29 | |||
local storage = access | |||
local finer = function () | |||
if append then | |||
storage = string.format( "%s/%s", | |||
storage, | |||
append ) | |||
end | |||
end | |||
local fun, lucky, r, suited | |||
if advanced then | |||
fun = require | |||
else | |||
fun = mw.loadData | |||
end | |||
GlobalMod.globalModules = GlobalMod.globalModules or { } | |||
suited = GlobalMod.globalModules[ access ] | |||
if not suited then | |||
finer() | |||
lucky, r = pcall( fun, "Module:" .. storage ) | |||
end | |||
if not lucky then | |||
if not suited and | |||
type( alt ) == "number" and | |||
alt > 0 then | |||
suited = string.format( "Q%d", alt ) | |||
suited = mw.wikibase.getSitelink( suited ) | |||
GlobalMod.globalModules[ access ] = suited or true | |||
end | |||
if type( suited ) == "string" then | |||
storage = suited | |||
finer() | |||
lucky, r = pcall( fun, storage ) | |||
end | |||
if not lucky and alert then | |||
error( "Missing or invalid page: " .. storage, 0 ) | |||
end | |||
end | |||
return r | |||
end -- foreignModule() | |||
Sort.lex = function ( adjust, apply, adapt ) | |||
-- Build ASCII sortkey for text value | |||
-- Precondition: | |||
-- adjust -- string to be aligned | |||
-- apply -- string or table, with base | |||
-- "latin" | |||
-- adapt -- string or table, with variation, or false | |||
-- "DIN5007m2" -- DIN 5007 mode "2" | |||
local r = adjust | |||
if adapt or not r:match( "^[ -~]*$" ) then | |||
local collate, post, pre | |||
if apply then | |||
collate = apply | |||
else | |||
collate = "uni" | |||
end | |||
if type( collate ) == "string" then | |||
collate = foreignModule( Sort.suite, | |||
false, | |||
collate, | |||
Sort.item ) | |||
end | |||
if adapt and type( collate ) == "table" then | |||
local variants = type( adapt ) | |||
local n | |||
if variants == "string" then | |||
variants = mw.text.split( adapt, "%s+" ) | |||
elseif variants == "table" then | |||
variants = adapt | |||
else | |||
variants = { } | |||
end | |||
n = #variants | |||
if n == 1 and variants[ 1 ] == "" then | |||
n = 0 | |||
end | |||
if n > 0 then | |||
local tmp = { } | |||
local var | |||
for k, v in pairs( collate ) do | |||
tmp[ k ] = v | |||
end -- for k, v | |||
collate = tmp | |||
for i = 1, n do | |||
tmp = foreignModule( Sort.suite, | |||
false, | |||
variants[ i ], | |||
Sort.item ) | |||
if type( tmp ) == "table" then | |||
var = tmp.single | |||
if type( var ) ~= "table" then | |||
-- legacy | |||
var = tmp | |||
end | |||
if type( var ) == "table" then | |||
for k, v in pairs( var ) do | |||
collate[ k ] = v | |||
end -- for k, v | |||
end | |||
var = tmp.pre | |||
if type( var ) == "table" then | |||
if type( pre ) ~= "table" then | |||
pre = { } | |||
end | |||
for k, v in pairs( var ) do | |||
pre[ k ] = v | |||
end -- for k, v | |||
end | |||
var = tmp.post | |||
if type( var ) == "table" then | |||
if type( post ) ~= "table" then | |||
post = { } | |||
end | |||
for k, v in pairs( var ) do | |||
post[ k ] = v | |||
end -- for k, v | |||
end | |||
elseif type( tmp ) == "string" then | |||
collate = tmp | |||
break -- for i | |||
else | |||
collate = "Invalid table " .. variants[ i ] | |||
break -- for i | |||
end | |||
end -- for i | |||
end | |||
end | |||
if type( collate ) == "table" then | |||
local k, n, s, start | |||
if type( pre ) == "table" then | |||
for k, v in pairs( pre ) do | |||
r = mw.ustring.gsub( r, k, v ) | |||
end -- for k, v | |||
end | |||
n = mw.ustring.len( r ) | |||
for i = n, 1, -1 do | |||
k = mw.ustring.codepoint( r, i, i ) | |||
if k < 127 then -- ASCII | |||
s = ( k < 32 ) -- htab newline whitespace | |||
if s then | |||
s = " " | |||
end | |||
elseif ( k >= 0x0300 and k <= 0x0362 ) or | |||
( k >= 0x1AB0 and k <= 0x1AFF ) or | |||
( k >= 0x1DC0 and k <= 0x1DFF ) or | |||
( k >= 0xFE20 and k <= 0xFE2F ) then | |||
-- COMBINING ... | |||
s = "" | |||
else | |||
s = collate[ k ] | |||
end | |||
if s then | |||
if i > 1 then | |||
s = mw.ustring.sub( r, 1, i - 1 ) .. s | |||
end | |||
r = s .. mw.ustring.sub( r, i + 1 ) | |||
end | |||
end -- for i-- | |||
if type( post ) == "table" then | |||
for k, v in pairs( post ) do | |||
r = mw.ustring.gsub( r, k, v ) | |||
end -- for k, v | |||
end | |||
else | |||
r = "**ERROR** Sort.lex ** Submodule unavailable " .. collate | |||
end | |||
end | |||
r = r:gsub( " +", " " ) | |||
return r | |||
end -- Sort.lex() | |||
Zeile 35: | Zeile 222: | ||
local c | local c | ||
if ad then | if ad then | ||
mid = mw.ustring.codepoint( ad, 1 ) | mid = mw.ustring.codepoint( ad, 1, 1 ) | ||
end | end | ||
if at then | if at then | ||
Zeile 44: | Zeile 231: | ||
end | end | ||
for i = 1, n do | for i = 1, n do | ||
c = mw.ustring.codepoint( source, i ) | c = mw.ustring.codepoint( source, i, i ) | ||
if c > 32 then -- not whitespace | if c > 32 then -- not whitespace | ||
if c >= 48 and c <= 57 then -- digits | if c >= 48 and c <= 57 then -- digits | ||
Zeile 52: | Zeile 239: | ||
elseif c == mid then -- decimal separator | elseif c == mid then -- decimal separator | ||
for j = i + 1, n do | for j = i + 1, n do | ||
c = mw.ustring.codepoint( source, j ) | c = mw.ustring.codepoint( source, j, j ) | ||
if c >= 48 and c <= 57 then -- digits | if c >= 48 and c <= 57 then -- digits | ||
sub = string.format( "%s%c", sub, c ) | sub = string.format( "%s%c", sub, c ) | ||
Zeile 91: | Zeile 278: | ||
r = r .. sub:sub( 1, j ) | r = r .. sub:sub( 1, j ) | ||
sub = sub:sub( j + 1 ) | sub = sub:sub( j + 1 ) | ||
max = max - j | |||
elseif j < 0 then | elseif j < 0 then | ||
j = - j | j = - j | ||
Zeile 98: | Zeile 286: | ||
sub = r:sub( - j ) .. sub | sub = r:sub( - j ) .. sub | ||
r = r:sub( 1, #r - j ) | r = r:sub( 1, #r - j ) | ||
max = max + j | |||
end | end | ||
sub = "." .. sub | sub = "." .. sub | ||
Zeile 140: | Zeile 329: | ||
return r | return r | ||
end -- Sort.num() | end -- Sort.num() | ||
Failsafe.failsafe = function ( atleast ) | |||
-- Retrieve versioning and check for compliance | |||
-- Precondition: | |||
-- atleast -- string, with required version or "wikidata" or "~" | |||
-- or false | |||
-- Postcondition: | |||
-- Returns string -- with queried version, also if problem | |||
-- false -- if appropriate | |||
-- 2019-10-15 | |||
local last = ( atleast == "~" ) | |||
local since = atleast | |||
local r | |||
if last or since == "wikidata" then | |||
local item = Failsafe.item | |||
since = false | |||
if type( item ) == "number" and item > 0 then | |||
local entity = mw.wikibase.getEntity( string.format( "Q%d", | |||
item ) ) | |||
if type( entity ) == "table" then | |||
local seek = Failsafe.serialProperty or "P348" | |||
local vsn = entity:formatPropertyValues( seek ) | |||
if type( vsn ) == "table" and | |||
type( vsn.value ) == "string" and | |||
vsn.value ~= "" then | |||
if last and vsn.value == Failsafe.serial then | |||
r = false | |||
else | |||
r = vsn.value | |||
end | |||
end | |||
end | |||
end | |||
end | |||
if type( r ) == "nil" then | |||
if not since or since <= Failsafe.serial then | |||
r = Failsafe.serial | |||
else | |||
r = false | |||
end | |||
end | |||
return r | |||
end -- Failsafe.failsafe() | |||
Zeile 149: | Zeile 383: | ||
-- Template::latin | -- Template::latin | ||
-- {{{1}}} | -- {{{1}}} | ||
-- #invoke | |||
-- v -- variant, omitted or "DIN5007m2" | |||
local lucky, r = pcall( Sort.lex, | |||
frame.args[ 1 ] or | |||
frame:getParent().args[ 1 ] or | |||
"", | |||
"latin", | |||
frame.args.v ) | |||
return r; | |||
end -- p.Tlatin | end -- p.Tlatin | ||
Zeile 171: | Zeile 405: | ||
-- m -- negative figures by digits; default: by value | -- m -- negative figures by digits; default: by value | ||
local lucky, r = pcall( Sort.num, | local lucky, r = pcall( Sort.num, | ||
frame:getParent().args[ 1 ] or "", | frame.args[ 1 ] or | ||
frame:getParent().args[ 1 ] or | |||
"", | |||
frame.args.d, | frame.args.d, | ||
frame.args.t, | frame.args.t, | ||
tonumber( frame.args.z ), | tonumber( frame.args.z ), | ||
frame.args.m == "1" ) | frame.args.m == "1" ) | ||
return r; | return r; | ||
end -- p.Tn | end -- p.Tn | ||
p.failsafe = function ( frame ) | |||
-- Versioning interface | |||
local s = type( frame ) | |||
local since | |||
if s == "table" then | |||
since = frame.args[ 1 ] | |||
elseif s == "string" then | |||
since = frame | |||
end | |||
if since then | |||
since = mw.text.trim( since ) | |||
if since == "" then | |||
since = false | |||
end | |||
end | |||
return Failsafe.failsafe( since ) or "" | |||
end -- p.failsafe() | |||
Aktuelle Version vom 9. Februar 2023, 15:34 Uhr
Die Dokumentation für dieses Modul kann unter Modul:Sort/doc erstellt werden
local Sort = { suite = "Sort", serial = "2019-10-29", item = 24205172 } --[=[ Sort ]=] local Failsafe = Sort local GlobalMod = Sort local foreignModule = function ( access, advanced, append, alt, alert ) -- Fetch global module -- Precondition: -- access -- string, with name of base module -- advanced -- true, for require(); else mw.loadData() -- append -- string, with subpage part, if any; or false -- alt -- number, of wikidata item of root; or false -- alert -- true, for throwing error on data problem -- Postcondition: -- Returns whatever, probably table -- 2019-10-29 local storage = access local finer = function () if append then storage = string.format( "%s/%s", storage, append ) end end local fun, lucky, r, suited if advanced then fun = require else fun = mw.loadData end GlobalMod.globalModules = GlobalMod.globalModules or { } suited = GlobalMod.globalModules[ access ] if not suited then finer() lucky, r = pcall( fun, "Module:" .. storage ) end if not lucky then if not suited and type( alt ) == "number" and alt > 0 then suited = string.format( "Q%d", alt ) suited = mw.wikibase.getSitelink( suited ) GlobalMod.globalModules[ access ] = suited or true end if type( suited ) == "string" then storage = suited finer() lucky, r = pcall( fun, storage ) end if not lucky and alert then error( "Missing or invalid page: " .. storage, 0 ) end end return r end -- foreignModule() Sort.lex = function ( adjust, apply, adapt ) -- Build ASCII sortkey for text value -- Precondition: -- adjust -- string to be aligned -- apply -- string or table, with base -- "latin" -- adapt -- string or table, with variation, or false -- "DIN5007m2" -- DIN 5007 mode "2" local r = adjust if adapt or not r:match( "^[ -~]*$" ) then local collate, post, pre if apply then collate = apply else collate = "uni" end if type( collate ) == "string" then collate = foreignModule( Sort.suite, false, collate, Sort.item ) end if adapt and type( collate ) == "table" then local variants = type( adapt ) local n if variants == "string" then variants = mw.text.split( adapt, "%s+" ) elseif variants == "table" then variants = adapt else variants = { } end n = #variants if n == 1 and variants[ 1 ] == "" then n = 0 end if n > 0 then local tmp = { } local var for k, v in pairs( collate ) do tmp[ k ] = v end -- for k, v collate = tmp for i = 1, n do tmp = foreignModule( Sort.suite, false, variants[ i ], Sort.item ) if type( tmp ) == "table" then var = tmp.single if type( var ) ~= "table" then -- legacy var = tmp end if type( var ) == "table" then for k, v in pairs( var ) do collate[ k ] = v end -- for k, v end var = tmp.pre if type( var ) == "table" then if type( pre ) ~= "table" then pre = { } end for k, v in pairs( var ) do pre[ k ] = v end -- for k, v end var = tmp.post if type( var ) == "table" then if type( post ) ~= "table" then post = { } end for k, v in pairs( var ) do post[ k ] = v end -- for k, v end elseif type( tmp ) == "string" then collate = tmp break -- for i else collate = "Invalid table " .. variants[ i ] break -- for i end end -- for i end end if type( collate ) == "table" then local k, n, s, start if type( pre ) == "table" then for k, v in pairs( pre ) do r = mw.ustring.gsub( r, k, v ) end -- for k, v end n = mw.ustring.len( r ) for i = n, 1, -1 do k = mw.ustring.codepoint( r, i, i ) if k < 127 then -- ASCII s = ( k < 32 ) -- htab newline whitespace if s then s = " " end elseif ( k >= 0x0300 and k <= 0x0362 ) or ( k >= 0x1AB0 and k <= 0x1AFF ) or ( k >= 0x1DC0 and k <= 0x1DFF ) or ( k >= 0xFE20 and k <= 0xFE2F ) then -- COMBINING ... s = "" else s = collate[ k ] end if s then if i > 1 then s = mw.ustring.sub( r, 1, i - 1 ) .. s end r = s .. mw.ustring.sub( r, i + 1 ) end end -- for i-- if type( post ) == "table" then for k, v in pairs( post ) do r = mw.ustring.gsub( r, k, v ) end -- for k, v end else r = "**ERROR** Sort.lex ** Submodule unavailable " .. collate end end r = r:gsub( " +", " " ) return r end -- Sort.lex() Sort.num = function ( adjust, ad, at, align, absolute ) -- Build sortkey for heading numerical value -- Precondition: -- adjust -- string to be aligned; leading digits / minus -- ad -- decimal separator; "." or ","; defaults to "." -- at -- thousands group separator; defaults to none -- "," "." "'" -- align -- number of leading zeros / maximum length -- defaults to 15 -- absolute -- negative figures by digits; default: by value -- Postcondition: -- Returns string with sortkey local max = 15 local mid = 46 -- "." local min1 = -1 -- none local min2 = -2 -- none local low = false local last = false local lead = true local source = tostring( adjust ) local sub = "." local suffix = false local n = mw.ustring.len( source ) local r = "" local c if ad then mid = mw.ustring.codepoint( ad, 1, 1 ) end if at then min1, min2 = mw.ustring.codepoint( at, 1, 2 ) end if align then max = align end for i = 1, n do c = mw.ustring.codepoint( source, i, i ) if c > 32 then -- not whitespace if c >= 48 and c <= 57 then -- digits r = string.format( "%s%c", r, c ) max = max - 1 elseif c == min1 or c == min2 then -- group separator elseif c == mid then -- decimal separator for j = i + 1, n do c = mw.ustring.codepoint( source, j, j ) if c >= 48 and c <= 57 then -- digits sub = string.format( "%s%c", sub, c ) elseif c == min1 or c == min2 then -- grouping else i = j break -- for j end i = n end -- for j last = true elseif lead then if c == 45 or c == 8722 then -- minus low = true elseif c ~= 43 then -- plus last = true end else last = true end lead = false elseif not lead then -- whitespace not leading last = true end if last then if i < n then suffix = mw.ustring.sub( source, i ) if c == 69 or c == 101 then -- E e local s = suffix:match( "^[Ee](-?%d+)" ) if s then j = tonumber( s ) sub = sub:sub( 2 ) suffix = suffix:sub( #s + 2 ) if j > 0 then if j > #sub then sub = sub .. string.rep( "0", j - #sub ) end r = r .. sub:sub( 1, j ) sub = sub:sub( j + 1 ) max = max - j elseif j < 0 then j = - j if j > #r then r = string.rep( "0", j - #r ) .. r end sub = r:sub( - j ) .. sub r = r:sub( 1, #r - j ) max = max + j end sub = "." .. sub end end end break -- for i end end -- for i if low then if not absolute then -- complementary value local s = "." local cmpl = function ( str, k ) return 57 - str:byte( k ) end for i = 2, #sub do s = string.format( "%s%d", s, cmpl( sub, i ) ) end -- for i for i = #r, 1, -1 do s = string.format( "%d%s", cmpl( r, i ), s ) end -- for i-- r = s if max > 0 then r = string.rep( "9", max ) .. r end sub = false max = 0 end end if sub then r = r .. sub end if max > 0 then r = string.rep( "0", max ) .. r end if low then r = "-" .. r end if suffix then r = string.format( "%s %s", r, suffix ) end return r end -- Sort.num() Failsafe.failsafe = function ( atleast ) -- Retrieve versioning and check for compliance -- Precondition: -- atleast -- string, with required version or "wikidata" or "~" -- or false -- Postcondition: -- Returns string -- with queried version, also if problem -- false -- if appropriate -- 2019-10-15 local last = ( atleast == "~" ) local since = atleast local r if last or since == "wikidata" then local item = Failsafe.item since = false if type( item ) == "number" and item > 0 then local entity = mw.wikibase.getEntity( string.format( "Q%d", item ) ) if type( entity ) == "table" then local seek = Failsafe.serialProperty or "P348" local vsn = entity:formatPropertyValues( seek ) if type( vsn ) == "table" and type( vsn.value ) == "string" and vsn.value ~= "" then if last and vsn.value == Failsafe.serial then r = false else r = vsn.value end end end end end if type( r ) == "nil" then if not since or since <= Failsafe.serial then r = Failsafe.serial else r = false end end return r end -- Failsafe.failsafe() -- Export local p = { } p.Tlatin = function ( frame ) -- Template::latin -- {{{1}}} -- #invoke -- v -- variant, omitted or "DIN5007m2" local lucky, r = pcall( Sort.lex, frame.args[ 1 ] or frame:getParent().args[ 1 ] or "", "latin", frame.args.v ) return r; end -- p.Tlatin p.Tn = function ( frame ) -- Template::numerical -- {{{1}}} -- #invoke -- d -- decimal separator; defaults to "." -- t -- thousands group separator; defaults to none -- z -- number of leading zeros / maximum length; defaults to 15 -- m -- negative figures by digits; default: by value local lucky, r = pcall( Sort.num, frame.args[ 1 ] or frame:getParent().args[ 1 ] or "", frame.args.d, frame.args.t, tonumber( frame.args.z ), frame.args.m == "1" ) return r; end -- p.Tn p.failsafe = function ( frame ) -- Versioning interface local s = type( frame ) local since if s == "table" then since = frame.args[ 1 ] elseif s == "string" then since = frame end if since then since = mw.text.trim( since ) if since == "" then since = false end end return Failsafe.failsafe( since ) or "" end -- p.failsafe() p.Sort = function () return Sort end -- p.Sort return p