Modul:Sort: Unterschied zwischen den Versionen
w>PerfektesChaos (2018-03-16) |
K (14 Versionen von wikivoyage:Modul:Sort importiert) |
||
(4 dazwischenliegende Versionen von 3 Benutzern werden nicht angezeigt) | |||
Zeile 1: | Zeile 1: | ||
local Sort = { suite = "Sort", | local Sort = { suite = "Sort", | ||
serial = " | serial = "2019-10-29", | ||
item = 24205172 } | item = 24205172 } | ||
--[=[ | --[=[ | ||
Sort | 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() | |||
Zeile 13: | Zeile 67: | ||
-- Precondition: | -- Precondition: | ||
-- adjust -- string to be aligned | -- adjust -- string to be aligned | ||
-- apply -- string with base | -- apply -- string or table, with base | ||
-- "latin" | -- "latin" | ||
-- adapt -- variation, or false | -- adapt -- string or table, with variation, or false | ||
-- "DIN5007m2" -- DIN 5007 mode "2" | -- "DIN5007m2" -- DIN 5007 mode "2" | ||
local r = adjust | local r = adjust | ||
if adapt or not r:match( "^[ -~]*$" ) then | if adapt or not r:match( "^[ -~]*$" ) then | ||
local collate, post, pre | |||
local collate | |||
if apply then | if apply then | ||
collate = apply | collate = apply | ||
Zeile 26: | Zeile 79: | ||
collate = "uni" | collate = "uni" | ||
end | end | ||
if type( collate ) == "string" then | |||
collate = foreignModule( Sort.suite, | |||
false, | |||
collate, | |||
Sort.item ) | |||
end | |||
if adapt and type( collate ) == "table" then | if adapt and type( collate ) == "table" then | ||
local variants = type( adapt ) | local variants = type( adapt ) | ||
Zeile 49: | Zeile 107: | ||
collate = tmp | collate = tmp | ||
for i = 1, n do | for i = 1, n do | ||
tmp = foreignModule( Sort.suite, | |||
false, | |||
variants[ i ], | |||
Sort.item ) | |||
if type( tmp ) == "table" then | if type( tmp ) == "table" then | ||
var = tmp.single | var = tmp.single | ||
Zeile 80: | Zeile 140: | ||
end -- for k, v | end -- for k, v | ||
end | end | ||
elseif | elseif type( tmp ) == "string" then | ||
collate = | collate = tmp | ||
break -- for i | break -- for i | ||
else | else | ||
collate = | collate = "Invalid table " .. variants[ i ] | ||
break -- for i | break -- for i | ||
end | end | ||
Zeile 105: | Zeile 165: | ||
s = " " | s = " " | ||
end | end | ||
elseif (k >= | elseif ( k >= 0x0300 and k <= 0x0362 ) or | ||
(k >= | ( k >= 0x1AB0 and k <= 0x1AFF ) or | ||
(k >= | ( k >= 0x1DC0 and k <= 0x1DFF ) or | ||
(k >= | ( k >= 0xFE20 and k <= 0xFE2F ) then | ||
-- COMBINING ... | -- COMBINING ... | ||
s = "" | s = "" | ||
Zeile 128: | Zeile 188: | ||
else | else | ||
r = "**ERROR** Sort.lex ** Submodule unavailable " .. collate | r = "**ERROR** Sort.lex ** Submodule unavailable " .. collate | ||
end | end | ||
end | end | ||
Zeile 274: | Zeile 332: | ||
Failsafe.failsafe = function ( atleast ) | |||
-- Retrieve versioning and check for compliance | -- Retrieve versioning and check for compliance | ||
-- Precondition: | -- Precondition: | ||
-- | -- atleast -- string, with required version or "wikidata" or "~" | ||
-- | -- or false | ||
-- Postcondition: | -- Postcondition: | ||
-- Returns string with | -- Returns string -- with queried version, also if problem | ||
local since = | -- false -- if appropriate | ||
-- 2019-10-15 | |||
local last = ( atleast == "~" ) | |||
local since = atleast | |||
local r | local r | ||
if since == "wikidata" then | if last or since == "wikidata" then | ||
local item = | local item = Failsafe.item | ||
since = false | since = false | ||
if type( item ) == "number" and item > 0 then | if type( item ) == "number" and item > 0 then | ||
Zeile 290: | Zeile 351: | ||
item ) ) | item ) ) | ||
if type( entity ) == "table" then | if type( entity ) == "table" then | ||
local vsn = entity:formatPropertyValues( | local seek = Failsafe.serialProperty or "P348" | ||
local vsn = entity:formatPropertyValues( seek ) | |||
if type( vsn ) == "table" and | if type( vsn ) == "table" and | ||
type( vsn.value) == "string" and | type( vsn.value ) == "string" and | ||
vsn.value ~= "" then | vsn.value ~= "" then | ||
r = vsn.value | if last and vsn.value == Failsafe.serial then | ||
r = false | |||
else | |||
r = vsn.value | |||
end | |||
end | end | ||
end | end | ||
end | end | ||
end | end | ||
if | if type( r ) == "nil" then | ||
if not since or since <= | if not since or since <= Failsafe.serial then | ||
r = | r = Failsafe.serial | ||
else | else | ||
r = false | r = false | ||
Zeile 307: | Zeile 373: | ||
end | end | ||
return r | return r | ||
end -- | end -- Failsafe.failsafe() | ||
Zeile 366: | Zeile 432: | ||
end | end | ||
end | end | ||
return | return Failsafe.failsafe( since ) or "" | ||
end -- p.failsafe() | 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