Module:Citation/CS1: Difference between revisions
Jump to navigation
Jump to search
m (1 revision imported) |
(sync from sandbox;) |
||
Line 1: | Line 1: | ||
--[[--------------------------< F O R W A R D D E C L A R A T I O N S >-------------------------------------- | --[[--------------------------< F O R W A R D D E C L A R A T I O N S >-------------------------------------- | ||
Line 133: | Line 131: | ||
the first character of the whole domain name including subdomains must be a letter or a digit | the first character of the whole domain name including subdomains must be a letter or a digit | ||
internationalized domain name (ascii characters with .xn-- ASCII Compatible Encoding (ACE) prefix xn-- in the tld) see https://tools.ietf.org/html/rfc3490 | internationalized domain name (ascii characters with .xn-- ASCII Compatible Encoding (ACE) prefix xn-- in the tld) see https://tools.ietf.org/html/rfc3490 | ||
single-letter/digit second-level domains in the .org and . | single-letter/digit second-level domains in the .org, .cash, and .today TLDs | ||
q, x, and z SL domains in the .com TLD | q, x, and z SL domains in the .com TLD | ||
i and q SL domains in the .net TLD | i and q SL domains in the .net TLD | ||
Line 153: | Line 151: | ||
if not domain:match ('^[%a%d]') then -- first character must be letter or digit | if not domain:match ('^[%a%d]') then -- first character must be letter or digit | ||
return false; | |||
end | |||
if domain:match ('^%a+:') then -- hack to detect things that look like s:Page:Title where Page: is namespace at wikisource | |||
return false; | return false; | ||
end | end | ||
Line 161: | Line 163: | ||
return true; | return true; | ||
elseif domain:match ('%f[%a%d][%a%d]%.cash$') then -- one character/digit .cash hostname | elseif domain:match ('%f[%a%d][%a%d]%.cash$') then -- one character/digit .cash hostname | ||
return true; | |||
elseif domain:match ('%f[%a%d][%a%d]%.today') then -- one character/digit .today hostname | |||
return true; | return true; | ||
elseif domain:match ('%f[%a%d][%a%d]%.org$') then -- one character/digit .org hostname | elseif domain:match ('%f[%a%d][%a%d]%.org$') then -- one character/digit .org hostname | ||
Line 399: | Line 403: | ||
local path; | local path; | ||
local base_url; | local base_url; | ||
if not is_set( label ) then | if not is_set( label ) then | ||
label = URL; | label = URL; | ||
Line 419: | Line 423: | ||
base_url = table.concat({ "[", URL, " ", safe_for_url (label), "]" }); -- assemble a wikimarkup url | base_url = table.concat({ "[", URL, " ", safe_for_url (label), "]" }); -- assemble a wikimarkup url | ||
if is_set (access) then -- access level (subscription, registration, limited) | if is_set (access) then -- access level (subscription, registration, limited) | ||
base_url = substitute (cfg.presentation['ext-link-access-signal'], {cfg.presentation[access].class, cfg.presentation[access].title, base_url}); -- add the appropriate icon | base_url = substitute (cfg.presentation['ext-link-access-signal'], {cfg.presentation[access].class, cfg.presentation[access].title, base_url}); -- add the appropriate icon | ||
Line 593: | Line 597: | ||
return substitute( cfg.messages[key], str ); | return substitute( cfg.messages[key], str ); | ||
end | end | ||
end | |||
--[[--------------------------< W I K I S O U R C E _ U R L _ M A K E >---------------------------------------- | |||
makes a wikisource url from wikisource interwiki link. returns the url and appropriate label; nil else. | |||
str is the value assigned to |chapter= (or aliases) or |title= or |title-link= | |||
]] | |||
local function wikisource_url_make (str) | |||
local wl_type, D, L; | |||
local ws_url, ws_label; | |||
wl_type, D, L = is_wikilink (str); -- wl_type is 0 (not a wikilink), 1 (simple wikilink), 2 (complex wikilink) | |||
if 0 == wl_type then -- not a wikilink; might be from |title-link= | |||
str = D:match ('^[Ww]ikisource:(.+)') or D:match ('^[Ss]:(.+)'); -- article title from interwiki link with long-form or short-form namespace | |||
if is_set (str) then | |||
ws_url = table.concat ({ -- build a wikisource url | |||
'https://en.wikisource.org/wiki/', -- prefix | |||
str, -- article title | |||
}); | |||
ws_label = str; -- label for the url | |||
end | |||
elseif 1 == wl_type then -- simple wikilink: [[Wikisource:ws article]] | |||
str = D:match ('^[Ww]ikisource:(.+)') or D:match ('^[Ss]:(.+)'); -- article title from interwiki link with long-form or short-form namespace | |||
if is_set (str) then | |||
ws_url = table.concat ({ -- build a wikisource url | |||
'https://en.wikisource.org/wiki/', -- prefix | |||
str, -- article title | |||
}); | |||
ws_label = str; -- label for the url | |||
end | |||
elseif 2 == wl_type then -- non-so-simple wikilink: [[Wikisource:ws article|displayed text]] ([[L|D]]) | |||
str = L:match ('^[Ww]ikisource:(.+)') or L:match ('^[Ss]:(.+)'); -- article title from interwiki link with long-form or short-form namespace | |||
if is_set (str) then | |||
ws_label = D; -- get ws article name from display portion of interwiki link | |||
ws_url = table.concat ({ -- build a wikisource url | |||
'https://en.wikisource.org/wiki/', -- prefix | |||
str, -- article title without namespace from link portion of wikilink | |||
}); | |||
end | |||
end | |||
if ws_url then | |||
ws_url = mw.uri.encode (ws_url, 'WIKI'); -- make a usable url | |||
ws_url = ws_url:gsub ('%%23', '#'); -- undo percent encoding of anchor | |||
end | |||
return ws_url, ws_label, L or D; -- return proper url or nil and a label or nil | |||
end | end | ||
Line 605: | Line 661: | ||
local function format_chapter_title (scriptchapter, chapter, transchapter, chapterurl, chapter_url_source, no_quotes, access) | local function format_chapter_title (scriptchapter, chapter, transchapter, chapterurl, chapter_url_source, no_quotes, access) | ||
local chapter_error = ''; | local chapter_error = ''; | ||
local ws_url, ws_label, L = wikisource_url_make (chapter); -- make a wikisource url and label from a wikisource interwiki link | |||
if ws_url then | |||
ws_label = ws_label:gsub ('_', ''); -- replace underscore separaters with space characters | |||
chapter = ws_label; | |||
end | |||
if not is_set (chapter) then | if not is_set (chapter) then | ||
chapter = ''; -- to be safe for concatenation | chapter = ''; -- to be safe for concatenation | ||
Line 619: | Line 681: | ||
if is_set (chapterurl) then | if is_set (chapterurl) then | ||
chapter = external_link (chapterurl, chapter, chapter_url_source, access); -- adds bare_url_missing_title error if appropriate | chapter = external_link (chapterurl, chapter, chapter_url_source, access); -- adds bare_url_missing_title error if appropriate | ||
elseif ws_url then | |||
chapter = external_link (ws_url, chapter .. ' ', 'ws link in chapter'); -- adds bare_url_missing_title error if appropriate; space char to move icon away from chap text; TODO: better way to do this? | |||
chapter = substitute (cfg.presentation['interwiki-icon'], {cfg.presentation['class-wikisource'], L, chapter}); | |||
end | end | ||
Line 648: | Line 713: | ||
Detects but ignores nowiki and math stripmarkers. Also detects other named stripmarkers (gallery, math, pre, ref) | Detects but ignores nowiki and math stripmarkers. Also detects other named stripmarkers (gallery, math, pre, ref) | ||
and identifies them with a slightly different error message. See also coins_cleanup(). | and identifies them with a slightly different error message. See also coins_cleanup(). | ||
Output of this function is an error message that identifies the character or the Unicode group, or the stripmarker | Output of this function is an error message that identifies the character or the Unicode group, or the stripmarker | ||
Line 822: | Line 885: | ||
return str; -- nothing to do, we're done | return str; -- nothing to do, we're done | ||
end | end | ||
str = str:gsub ('&[nm]dash;', {['–'] = '–', ['—'] = '—'}); -- replace — and – entities with their characters; semicolon mucks up the text.split | |||
local out = {}; | local out = {}; | ||
Line 1,154: | Line 1,219: | ||
--[[--------------------------< N A M E _ H A S _ E T A L >---------------------------------------------------- | --[[--------------------------< N A M E _ H A S _ E T A L >---------------------------------------------------- | ||
Evaluates the content of author | Evaluates the content of name parameters (author, editor, etc) for variations on the theme of et al. If found, | ||
the et al. is removed, a flag is set to true and the function returns the modified name and the flag. | the et al. is removed, a flag is set to true and the function returns the modified name and the flag. | ||
This function never sets the flag to false but returns it's previous state because it may have been set by | This function never sets the flag to false but returns it's previous state because it may have been set by | ||
previous passes through this function or by the | previous passes through this function or by the associated |display-<names>=etal parameter | ||
]] | ]] | ||
local function name_has_etal (name, etal, nocat) | local function name_has_etal (name, etal, nocat, param) | ||
if is_set (name) then -- name can be nil in which case just return | if is_set (name) then -- name can be nil in which case just return | ||
local | local patterns = cfg.et_al_patterns; --get patterns from configuration | ||
for _, pattern in ipairs (patterns) do -- loop through all of the patterns | |||
if name:match (pattern) then -- if this 'et al' pattern is found in name | |||
name = name:gsub (pattern, ''); -- remove the offending text | |||
if | etal = true; -- set flag (may have been set previously here or by |display-<names>=etal) | ||
if not nocat then -- no categorization for |vauthors= | |||
table.insert( z.message_tail, {set_error ('etal', {param})}); -- and set an error if not added | |||
end | |||
end | end | ||
end | end | ||
end | end | ||
return name, etal; -- | return name, etal; -- | ||
end | end | ||
Line 1,196: | Line 1,257: | ||
local function name_has_ed_markup (name, list_name) | local function name_has_ed_markup (name, list_name) | ||
local _, pattern; | local _, pattern; | ||
local patterns = | local patterns = cfg.editor_markup_patterns; -- get patterns from configuration | ||
if is_set (name) then | if is_set (name) then | ||
Line 1,295: | Line 1,345: | ||
local etal=false; -- return value set to true when we find some form of et al. in an author parameter | local etal=false; -- return value set to true when we find some form of et al. in an author parameter | ||
local | local last_alias, first_alias; -- selected parameter aliases used in error messaging | ||
while true do | while true do | ||
last = select_one( args, cfg.aliases[list_name .. '-Last'], 'redundant_parameters', i ); -- search through args for name components beginning at 1 | last, last_alias = select_one( args, cfg.aliases[list_name .. '-Last'], 'redundant_parameters', i ); -- search through args for name components beginning at 1 | ||
first = select_one( args, cfg.aliases[list_name .. '-First'], 'redundant_parameters', i ); | first, first_alias = select_one( args, cfg.aliases[list_name .. '-First'], 'redundant_parameters', i ); | ||
link = select_one( args, cfg.aliases[list_name .. '-Link'], 'redundant_parameters', i ); | link = select_one( args, cfg.aliases[list_name .. '-Link'], 'redundant_parameters', i ); | ||
mask = select_one( args, cfg.aliases[list_name .. '-Mask'], 'redundant_parameters', i ); | mask = select_one( args, cfg.aliases[list_name .. '-Mask'], 'redundant_parameters', i ); | ||
last, etal = name_has_etal (last, etal, false); | last, etal = name_has_etal (last, etal, false, last_alias); -- find and remove variations on et al. | ||
first, etal = name_has_etal (first, etal, false); | first, etal = name_has_etal (first, etal, false, first_alias); -- find and remove variations on et al. | ||
last, first= name_checks (last, first, list_name); -- multiple names, extraneous annotation, etc checks | last, first= name_checks (last, first, list_name); -- multiple names, extraneous annotation, etc checks | ||
if first and not last then -- if there is a firstn without a matching lastn | if first and not last then -- if there is a firstn without a matching lastn | ||
table.insert( z.message_tail, { set_error( 'first_missing_last', { | table.insert( z.message_tail, { set_error( 'first_missing_last', {first_alias, first_alias:gsub('first', 'last')}, true ) } ); -- add this error message | ||
elseif not first and not last then -- if both firstn and lastn aren't found, are we done? | elseif not first and not last then -- if both firstn and lastn aren't found, are we done? | ||
count = count + 1; -- number of times we haven't found last and first | count = count + 1; -- number of times we haven't found last and first | ||
Line 1,319: | Line 1,369: | ||
n = n + 1; -- point to next location in the names table | n = n + 1; -- point to next location in the names table | ||
if 1 == count then -- if the previous name was missing | if 1 == count then -- if the previous name was missing | ||
table.insert( z.message_tail, { set_error( 'missing_name', { | table.insert( z.message_tail, { set_error( 'missing_name', {list_name:match ("(%w+)List"):lower(), i-1}, true ) } ); -- add this error message | ||
end | end | ||
count = 0; -- reset the counter, we're looking for two consecutive missing names | count = 0; -- reset the counter, we're looking for two consecutive missing names | ||
Line 1,342: | Line 1,392: | ||
extensions. For example, code 'cbk-zam' and its associated name 'Chavacano de Zamboanga' (MediaWiki does not support | extensions. For example, code 'cbk-zam' and its associated name 'Chavacano de Zamboanga' (MediaWiki does not support | ||
code 'cbk' or name 'Chavacano'. Most (all?) of these languages are not used a 'language' codes per se, rather they | code 'cbk' or name 'Chavacano'. Most (all?) of these languages are not used a 'language' codes per se, rather they | ||
are used as sub-domain names: cbk-zam.wikipedia.org. | are used as sub-domain names: cbk-zam.wikipedia.org. A list of language names and codes supported by fetchLanguageNames() | ||
can be found at Template:Citation Style documentation/language/doc | |||
Names | Names that are included in the list will be found if that name is provided in the |language= parameter. For example, | ||
if |language=Chavacano de Zamboanga, that name will be found with the associated code 'cbk-zam'. When names are found | if |language=Chavacano de Zamboanga, that name will be found with the associated code 'cbk-zam'. When names are found | ||
and the associated code is not two or three characters, this function returns only the | and the associated code is not two or three characters, this function returns only the WikiMedia language name. | ||
Some language names have multiple entries under different codes: | |||
Aromanian has code rup and code roa-rup | |||
When this occurs, this function returns the language name and the 2- or 3-character code | |||
Adapted from code taken from Module:Check ISO 639-1. | Adapted from code taken from Module:Check ISO 639-1. | ||
Line 1,358: | Line 1,412: | ||
end | end | ||
local ietf_code; -- because some languages have both ietf-like codes and iso 639-like codes | |||
local ietf_name; | |||
local languages = mw.language.fetchLanguageNames(this_wiki_code, 'all') -- get a list of language names known to Wikimedia | local languages = mw.language.fetchLanguageNames(this_wiki_code, 'all') -- get a list of language names known to Wikimedia | ||
-- ('all' is required for North Ndebele, South Ndebele, and Ojibwa) | -- ('all' is required for North Ndebele, South Ndebele, and Ojibwa) | ||
local langlc = mw.ustring.lower(lang); -- lower case version for comparisons | local langlc = mw.ustring.lower(lang); -- lower case version for comparisons | ||
for code, name in pairs(languages) do -- scan the list to see if we can find our language | for code, name in pairs(languages) do -- scan the list to see if we can find our language | ||
if langlc == mw.ustring.lower(name) then | if langlc == mw.ustring.lower(name) then | ||
if 2 | if 2 == code:len() or 3 == code:len() then -- two- or three-character codes only; extensions not supported | ||
return name; | return name, code; -- so return the name and the code | ||
end | end | ||
ietf_code = code; -- remember that we found an ietf-like code and save its name | |||
ietf_name = name; -- but keep looking for a 2- or 3-char code | |||
end | end | ||
end | end | ||
return lang; | -- didn't find name with 2- or 3-char code; if ietf-like code found return | ||
return ietf_code and ietf_name or lang; -- associated name; return original language text else | |||
end | end | ||
Line 1,404: | Line 1,463: | ||
for _, lang in ipairs (names_table) do -- reuse lang | for _, lang in ipairs (names_table) do -- reuse lang | ||
name = cfg.lang_code_remap[lang:lower()]; -- first see if this is a code that is not supported by MediaWiki but is in remap | |||
if lang:match ('^%a%a%-') then | if name then -- there was a remapped code so | ||
lang = lang:gsub ('^(%a%a%a?)%-.*', '%1'); -- strip ietf tags from code | |||
else | |||
if lang:match ('^%a%a%-') then -- strip ietf tags from code; TODO: is there a need to support 3-char with tag? | |||
lang = lang:match ('(%a%a)%-') -- keep only 639-1 code portion to lang; TODO: do something with 3166 alpha 2 country code? | |||
end | |||
if 2 == lang:len() or 3 == lang:len() then -- if two-or three-character code | |||
name = mw.language.fetchLanguageName (lang:lower(), this_wiki_code); -- get language name if |language= is a proper code | |||
end | end | ||
end | end | ||
if is_set (name) then -- if |language= specified a valid code | if is_set (name) then -- if |language= specified a valid code | ||
code = lang:lower(); -- save it | code = lang:lower(); -- save it | ||
Line 1,441: | Line 1,502: | ||
code = #language_list -- reuse code as number of languages in the list | code = #language_list -- reuse code as number of languages in the list | ||
if 2 >= code then | if 2 >= code then | ||
name = table.concat (language_list, ' | name = table.concat (language_list, cfg.messages['parameter-pair-separator']) -- insert '<space>and<space>' between two language names | ||
elseif 2 < code then | elseif 2 < code then | ||
language_list | name = table.concat (language_list, ', '); -- and concatenate with '<comma><space>' separators | ||
name = | name = name:gsub (', ([^,]+)$', cfg.messages['parameter-final-separator'] .. '%1'); -- replace last '<comma><space>' separator with '<comma><space>and<space>' separator | ||
end | end | ||
if this_wiki_name == name then | if this_wiki_name == name then | ||
Line 1,546: | Line 1,607: | ||
local function is_pdf (url) | local function is_pdf (url) | ||
return url:match ('%.pdf$') or url:match ('%.PDF$') or url:match ('%.pdf[%?#]') or url:match ('%.PDF[%?#]'); | return url:match ('%.pdf$') or url:match ('%.PDF$') or | ||
url:match ('%.pdf[%?#]') or url:match ('%.PDF[%?#]') or | |||
url:match ('%.PDF#') or url:match ('%.pdf#'); | |||
end | end | ||
Line 1,574: | Line 1,637: | ||
--[[--------------------------< G E T _ D I S P L A Y _ A | --[[--------------------------< G E T _ D I S P L A Y _ N A M E S >-------------------------------------------- | ||
Returns a number that defines the number of names displayed for author and editor name lists and a boolean flag | Returns a number that defines the number of names displayed for author and editor name lists and a boolean flag | ||
Line 1,597: | Line 1,660: | ||
]] | ]] | ||
local function | local function get_display_names (max, count, list_name, etal) | ||
if is_set (max) then | if is_set (max) then | ||
if 'etal' == max:lower():gsub("[ '%.]", '') then -- the :gsub() portion makes 'etal' from a variety of 'et al.' spellings and stylings | if 'etal' == max:lower():gsub("[ '%.]", '') then -- the :gsub() portion makes 'etal' from a variety of 'et al.' spellings and stylings | ||
Line 1,605: | Line 1,668: | ||
max = tonumber (max); -- make it a number | max = tonumber (max); -- make it a number | ||
if max >= count then -- if |display-xxxxors= value greater than or equal to number of authors/editors | if max >= count then -- if |display-xxxxors= value greater than or equal to number of authors/editors | ||
add_maint_cat (' | add_maint_cat ('disp_name', cfg.special_case_translation [list_name]); | ||
end | end | ||
else -- not a valid keyword or number | else -- not a valid keyword or number | ||
Line 1,706: | Line 1,769: | ||
vparam, etal = name_has_etal (vparam, etal, true); -- find and remove variations on et al. do not categorize (do it here because et al. might have a period) | vparam, etal = name_has_etal (vparam, etal, true); -- find and remove variations on et al. do not categorize (do it here because et al. might have a period) | ||
v_name_table = get_v_name_table (vparam, v_name_table, v_link_table); | v_name_table = get_v_name_table (vparam, v_name_table, v_link_table); -- names are separated by commas | ||
for i, v_name in ipairs(v_name_table) do | for i, v_name in ipairs(v_name_table) do | ||
Line 1,869: | Line 1,932: | ||
end | end | ||
local vol = ''; | local vol = ''; -- here for all cites except magazine | ||
if is_set (volume) then | if is_set (volume) then | ||
if (4 < mw.ustring.len(volume)) then | if volume:match ('^[MDCLXVI]+$') or volume:match ('^%d+$')then -- volume value is all digits or all uppercase roman numerals | ||
vol = substitute (cfg.messages['j-vol'], {sepc, volume}); | vol = substitute (cfg.presentation['vol-bold'], {sepc, hyphen_to_dash(volume)}); -- render in bold face | ||
else | elseif (4 < mw.ustring.len(volume)) then -- not all digits or roman numerals and longer than 4 characters | ||
vol = substitute (cfg.presentation['vol-bold'], {sepc, hyphen_to_dash(volume)}); | vol = substitute (cfg.messages['j-vol'], {sepc, volume}); -- not bold | ||
add_prop_cat ('long_vol'); | |||
else -- four or less characters | |||
vol = substitute (cfg.presentation['vol-bold'], {sepc, hyphen_to_dash(volume)}); -- bold | |||
end | end | ||
end | end | ||
Line 1,936: | Line 2,002: | ||
return '', '', '', ''; -- return empty strings | return '', '', '', ''; -- return empty strings | ||
end | end | ||
--[[--------------------------< I N S O U R C E _ L O C _ G E T >---------------------------------------------- | |||
returns one of the in-source locators: page, pages, or at. | |||
If any of these are interwiki links to wikisource, returns the label portion of the interwikilink as plain text | |||
for use in COinS. This COinS thing is done because here we convert an interwiki link to and external link and | |||
add an icon span around that; get_coins_pages() doesn't know about the span. TODO: should it? | |||
TODO: add support for sheet and sheets?; streamline; | |||
TODO: make it so that this function returns only one of the three as the single in-source (the return value assigned | |||
to a new name)? | |||
]] | |||
local function insource_loc_get (page, pages, at) | |||
local ws_url, ws_label, coins_pages, L; -- for wikisource interwikilinks; TODO: this corrupts page metadata (span remains in place after cleanup; fix there?) | |||
if is_set (page) then | |||
if is_set (pages) or is_set(at) then | |||
pages = ''; -- unset the others | |||
at = ''; | |||
end | |||
extra_text_in_page_check (page); -- add this page to maint cat if |page= value begins with what looks like p. or pp. | |||
ws_url, ws_label, L = wikisource_url_make (page); -- make ws url from |page= interwiki link; link portion L becomes tool tip label | |||
if ws_url then | |||
page = external_link (ws_url, ws_label .. ' ', 'ws link in page'); -- space char after label to move icon away from in-source text; TODO: a better way to do this? | |||
page = substitute (cfg.presentation['interwiki-icon'], {cfg.presentation['class-wikisource'], L, page}); | |||
coins_pages = ws_label; | |||
end | |||
elseif is_set (pages) then | |||
if is_set (at) then | |||
at = ''; -- unset | |||
end | |||
extra_text_in_page_check (pages); -- add this page to maint cat if |pages= value begins with what looks like p. or pp. | |||
ws_url, ws_label, L = wikisource_url_make (pages); -- make ws url from |pages= interwiki link; link portion L becomes tool tip label | |||
if ws_url then | |||
pages = external_link (ws_url, ws_label .. ' ', 'ws link in pages'); -- space char after label to move icon away from in-source text; TODO: a better way to do this? | |||
pages = substitute (cfg.presentation['interwiki-icon'], {cfg.presentation['class-wikisource'], L, pages}); | |||
coins_pages = ws_label; | |||
end | |||
elseif is_set (at) then | |||
ws_url, ws_label, L = wikisource_url_make (at); -- make ws url from |at= interwiki link; link portion L becomes tool tip label | |||
if ws_url then | |||
at = external_link (ws_url, ws_label .. ' ', 'ws link in at'); -- space char after label to move icon away from in-source text; TODO: a better way to do this? | |||
at = substitute (cfg.presentation['interwiki-icon'], {cfg.presentation['class-wikisource'], L, at}); | |||
coins_pages = ws_label; | |||
end | |||
end | |||
return page, pages, at, coins_pages; | |||
end | |||
Line 2,078: | Line 2,201: | ||
end | end | ||
local translator_etal; | |||
local t = {}; -- translators list from |translator-lastn= / translator-firstn= pairs | local t = {}; -- translators list from |translator-lastn= / translator-firstn= pairs | ||
local Translators; -- assembled translators name list | local Translators; -- assembled translators name list | ||
t = extract_names (args, 'TranslatorList'); -- fetch translator list from |translatorn= / |translator-lastn=, -firstn=, -linkn=, -maskn= | t = extract_names (args, 'TranslatorList'); -- fetch translator list from |translatorn= / |translator-lastn=, -firstn=, -linkn=, -maskn= | ||
local interviewers_list = {}; | local interviewer_etal; | ||
local Interviewers | local interviewers_list = {}; | ||
local Interviewers; -- used later | |||
interviewers_list = extract_names (args, 'InterviewerList'); -- process preferred interviewers parameters | |||
local contributor_etal; | |||
local c = {}; -- contributors list from |contributor-lastn= / contributor-firstn= pairs | local c = {}; -- contributors list from |contributor-lastn= / contributor-firstn= pairs | ||
local Contributors; -- assembled contributors name list | local Contributors; -- assembled contributors name list | ||
Line 2,116: | Line 2,238: | ||
NameListFormat = ''; -- anything else, set to empty string | NameListFormat = ''; -- anything else, set to empty string | ||
end | end | ||
if is_set (Others) then | |||
if 0 == #a and 0 == #e then -- add maint cat when |others= has value and used without |author=, |editor= | |||
add_maint_cat ('others'); | |||
end | |||
end | |||
local Year = A['Year']; | local Year = A['Year']; | ||
Line 2,274: | Line 2,402: | ||
if not is_valid_parameter_value (DF, 'df', cfg.keywords['date-format']) then -- validate reformatting keyword | if not is_valid_parameter_value (DF, 'df', cfg.keywords['date-format']) then -- validate reformatting keyword | ||
DF = ''; -- not valid, set to empty string | DF = ''; -- not valid, set to empty string | ||
end | |||
if not is_set (DF) then | |||
DF = cfg.global_df; | |||
end | end | ||
Line 2,297: | Line 2,428: | ||
-- check for extra |page=, |pages= or |at= parameters. (also sheet and sheets while we're at it) | -- check for extra |page=, |pages= or |at= parameters. (also sheet and sheets while we're at it) | ||
select_one( args, {'page', 'p', 'pp', 'pages', 'at', 'sheet', 'sheets'}, 'redundant_parameters' ); -- this is a dummy call simply to get the error message and category | select_one( args, {'page', 'p', 'pp', 'pages', 'at', 'sheet', 'sheets'}, 'redundant_parameters' ); -- this is a dummy call simply to get the error message and category | ||
local coins_pages; | |||
Page, Pages, At, coins_pages = insource_loc_get (Page, Pages, At); | |||
local NoPP = A['NoPP'] | local NoPP = A['NoPP'] | ||
Line 2,304: | Line 2,439: | ||
NoPP = nil; -- unset, used as a flag later | NoPP = nil; -- unset, used as a flag later | ||
end | end | ||
-- both |publication-place= and |place= (|location=) allowed if different | -- both |publication-place= and |place= (|location=) allowed if different | ||
Line 2,693: | Line 2,815: | ||
['Volume'] = Volume, | ['Volume'] = Volume, | ||
['Issue'] = Issue, | ['Issue'] = Issue, | ||
['Pages'] = get_coins_pages (first_set ({Sheet, Sheets, Page, Pages, At}, 5)), -- pages stripped of external links | ['Pages'] = coins_pages or get_coins_pages (first_set ({Sheet, Sheets, Page, Pages, At}, 5)), -- pages stripped of external links | ||
['Edition'] = Edition, | ['Edition'] = Edition, | ||
['PublisherName'] = PublisherName, | ['PublisherName'] = PublisherName, | ||
Line 2,731: | Line 2,853: | ||
do -- do editor name list first because the now unsupported coauthors used to modify control table | do -- do editor name list first because the now unsupported coauthors used to modify control table | ||
control.maximum , editor_etal = | control.maximum , editor_etal = get_display_names (A['DisplayEditors'], #e, 'editors', editor_etal); | ||
last_first_list, EditorCount = list_people(control, e, editor_etal); | last_first_list, EditorCount = list_people(control, e, editor_etal); | ||
if is_set (Editors) then | if is_set (Editors) then | ||
Editors, editor_etal = name_has_etal (Editors, editor_etal, false, 'editors'); -- find and remove variations on et al. | |||
if editor_etal then | if editor_etal then | ||
Editors = Editors .. ' ' .. cfg.messages['et al']; -- add et al. to editors parameter beause |display-editors=etal | Editors = Editors .. ' ' .. cfg.messages['et al']; -- add et al. to editors parameter beause |display-editors=etal | ||
Line 2,750: | Line 2,873: | ||
end | end | ||
do -- now do interviewers | do -- now do interviewers | ||
control.maximum = #interviewers_list; | control.maximum , interviewer_etal = get_display_names (A['DisplayInterviewers'], #interviewers_list, 'interviewers', interviewer_etal); | ||
Interviewers = list_people(control, interviewers_list, | Interviewers = list_people (control, interviewers_list, interviewer_etal); | ||
end | end | ||
do -- now do translators | do -- now do translators | ||
control.maximum = #t; | control.maximum , translator_etal = get_display_names (A['DisplayTranslators'], #t, 'translators', translator_etal); | ||
Translators = list_people(control, t, | Translators = list_people (control, t, translator_etal); | ||
end | end | ||
do -- now do contributors | do -- now do contributors | ||
control.maximum = #c; -- number of contributors | control.maximum , contributor_etal = get_display_names (A['DisplayContributors'], #c, 'contributors', contributor_etal); | ||
Contributors = list_people (control, c, contributor_etal); | |||
-- control.maximum = #c; -- number of contributors | |||
-- Contributors = list_people(control, c, false); -- et al not currently supported | |||
end | end | ||
do -- now do authors | do -- now do authors | ||
control.maximum , author_etal = | control.maximum , author_etal = get_display_names (A['DisplayAuthors'], #a, 'authors', author_etal); | ||
last_first_list = list_people(control, a, author_etal); | last_first_list = list_people(control, a, author_etal); | ||
if is_set (Authors) then | if is_set (Authors) then | ||
Authors, author_etal = name_has_etal (Authors, author_etal, false); -- find and remove variations on et al. | Authors, author_etal = name_has_etal (Authors, author_etal, false, 'authors'); -- find and remove variations on et al. | ||
if author_etal then | if author_etal then | ||
Authors = Authors .. ' ' .. cfg.messages['et al']; -- add et al. to authors parameter | Authors = Authors .. ' ' .. cfg.messages['et al']; -- add et al. to authors parameter | ||
Line 2,894: | Line 3,019: | ||
end | end | ||
if in_array(config.CitationClass, {'web', 'news', 'journal', 'magazine', 'pressrelease', 'podcast', 'newsgroup', 'mailinglist', 'interview', 'arxiv', 'biorxiv', 'citeseerx'}) or | if in_array(config.CitationClass, {'web', 'news', 'journal', 'magazine', 'pressrelease', 'podcast', 'newsgroup', 'mailinglist', 'interview', 'arxiv', 'biorxiv', 'citeseerx'}) or | ||
('citation' == config.CitationClass and is_set (Periodical) and not is_set (Encyclopedia)) or | ('citation' == config.CitationClass and is_set (Periodical) and not is_set (Encyclopedia)) or | ||
Line 2,922: | Line 3,043: | ||
end | end | ||
end | end | ||
if is_set(Title) then | if is_set(Title) then | ||
if not is_set(TitleLink) and is_set(URL) then | if not is_set (TitleLink) and is_set (URL) then | ||
Title = external_link (URL, Title, URLorigin, UrlAccess) .. TransTitle .. TransError .. Format; | |||
Title = external_link( URL, Title, URLorigin, UrlAccess ) .. TransTitle .. TransError .. Format; | |||
URL = ''; -- unset these because no longer needed | URL = ''; -- unset these because no longer needed | ||
Format = ""; | Format = ""; | ||
elseif is_set (TitleLink) and not is_set (URL) then | |||
local ws_url; | |||
ws_url = wikisource_url_make (TitleLink); -- ignore ws_label return; not used here | |||
if ws_url then | |||
Title = external_link (ws_url, Title .. ' ', 'ws link in title-link'); -- space char after Title to move icon away from italic text; TODO: a better way to do this? | |||
Title = substitute (cfg.presentation['interwiki-icon'], {cfg.presentation['class-wikisource'], TitleLink, Title}); | |||
Title = Title .. TransTitle .. TransError; | |||
else | |||
Title = make_wikilink (TitleLink, Title) .. TransTitle .. TransError; | |||
end | |||
else | else | ||
Title = Title .. TransTitle .. TransError; | local ws_url, ws_label; | ||
ws_url, ws_label, L = wikisource_url_make (Title); -- make ws url from |title= interwiki link; link portion L becomes tool tip label | |||
if ws_url then | |||
Title = Title:gsub ('%b[]', ws_label); -- replace interwiki link with ws_label to retain markup | |||
Title = external_link (ws_url, Title .. ' ', 'ws link in title'); -- space char after Title to move icon away from italic text; TODO: a better way to do this? | |||
Title = substitute (cfg.presentation['interwiki-icon'], {cfg.presentation['class-wikisource'], L, Title}); | |||
Title = Title .. TransTitle .. TransError; | |||
else | |||
Title = Title .. TransTitle .. TransError; | |||
end | |||
end | end | ||
else | else | ||
Line 3,212: | Line 3,351: | ||
elseif 'episode' == config.CitationClass then -- special case for cite episode | elseif 'episode' == config.CitationClass then -- special case for cite episode | ||
tcommon = safe_join( {Title, TitleNote, TitleType, Series | tcommon = safe_join( {Title, TitleNote, TitleType, Series, Language, Edition, Publisher}, sepc ); | ||
else -- all other CS1 templates | else -- all other CS1 templates | ||
Line 3,225: | Line 3,364: | ||
end | end | ||
local idcommon = safe_join( { ID_list, URL, Archived, AccessDate, Via, SubscriptionRequired, Lay, Quote }, sepc ); | local idcommon; | ||
if 'audio-visual' == config.CitationClass or 'episode' == config.CitationClass then -- special case for cite AV media & cite episode position transcript | |||
idcommon = safe_join( { ID_list, URL, Archived, Transcript, AccessDate, Via, SubscriptionRequired, Lay, Quote }, sepc ); | |||
else | |||
idcommon = safe_join( { ID_list, URL, Archived, AccessDate, Via, SubscriptionRequired, Lay, Quote }, sepc ); | |||
end | |||
local text; | local text; | ||
local pgtext = Position .. Sheet .. Sheets .. Page .. Pages .. At; | local pgtext = Position .. Sheet .. Sheets .. Page .. Pages .. At; | ||
Line 3,251: | Line 3,396: | ||
if (sepc ~= '.') then | if (sepc ~= '.') then | ||
in_text = in_text:lower() -- lowercase for cs2 | in_text = in_text:lower() -- lowercase for cs2 | ||
end | |||
end | |||
if EditorCount <= 1 then | |||
post_text = " (" .. cfg.messages['editor'] .. ")"; -- be consistent with no-author, no-date case | |||
else | else | ||
post_text = " (" .. cfg.messages['editors'] .. ")"; | |||
end | |||
end | |||
Editors = terminate_name_list (in_text .. Editors .. post_text, sepc); -- terminate with 0 or 1 sepc and a space | Editors = terminate_name_list (in_text .. Editors .. post_text, sepc); -- terminate with 0 or 1 sepc and a space | ||
end | end | ||
Line 3,353: | Line 3,497: | ||
table.insert (render, substitute (cfg.presentation['ocins'], {OCinSoutput})); -- append metadata to the citation | table.insert (render, substitute (cfg.presentation['ocins'], {OCinSoutput})); -- append metadata to the citation | ||
if #z.message_tail | if 0 ~= #z.message_tail then | ||
table.insert (render, ' '); | table.insert (render, ' '); | ||
for i,v in ipairs( z.message_tail ) do | for i,v in ipairs( z.message_tail ) do | ||
Line 3,366: | Line 3,510: | ||
end | end | ||
if #z.maintenance_cats | if 0 ~= #z.maintenance_cats then | ||
local maint_msgs = {}; -- here we collect all of the maint messages | |||
for _, v in ipairs( z.maintenance_cats ) do -- append maintenance categories | for _, v in ipairs( z.maintenance_cats ) do -- append maintenance categories | ||
table.insert ( | local maint = {}; -- here we assemble a maintenence message | ||
table.insert ( | table.insert (maint, v); -- maint msg is the category name | ||
table.insert ( | table.insert (maint, ' ('); -- open the link text | ||
table.insert ( | table.insert (maint, make_wikilink (':Category:' .. v, 'link')); -- add the link | ||
table.insert (maint, ')'); -- and close it | |||
table.insert (maint_msgs, table.concat (maint)); -- assemble new maint message and add it to the maint_msgs table | |||
end | end | ||
table.insert (render, ' | table.insert (render, substitute (cfg.presentation['hidden-maint'], table.concat (maint_msgs, ' '))); -- wrap the group of maint message with proper presentation and save | ||
end | end | ||
Line 3,471: | Line 3,617: | ||
]] | ]] | ||
local function missing_pipe_check (value) | local function missing_pipe_check (parameter, value) | ||
local capture; | local capture; | ||
value = value:gsub ('%b<>', ''); -- remove xml/html tags because attributes: class=, title=, etc | value = value:gsub ('%b<>', ''); -- remove xml/html tags because attributes: class=, title=, etc | ||
Line 3,477: | Line 3,623: | ||
capture = value:match ('%s+(%a[%a%d]+)%s*=') or value:match ('^(%a[%a%d]+)%s*='); -- find and categorize parameters with possible missing pipes | capture = value:match ('%s+(%a[%a%d]+)%s*=') or value:match ('^(%a[%a%d]+)%s*='); -- find and categorize parameters with possible missing pipes | ||
if capture and validate (capture) then -- if the capture is a valid parameter name | if capture and validate (capture) then -- if the capture is a valid parameter name | ||
table.insert( z.message_tail, {set_error ('missing_pipe',parameter)}); | |||
end | end | ||
end | end | ||
Line 3,488: | Line 3,634: | ||
]] | ]] | ||
function | local function citation(frame) | ||
Frame = frame; -- save a copy incase we need to display an error message in preview mode | Frame = frame; -- save a copy incase we need to display an error message in preview mode | ||
local pframe = frame:getParent() | local pframe = frame:getParent() | ||
Line 3,604: | Line 3,750: | ||
end | end | ||
end | end | ||
missing_pipe_check (v); -- do we think that there is a parameter that is missing a pipe? | missing_pipe_check (k, v); -- do we think that there is a parameter that is missing a pipe? | ||
-- TODO: is this the best place for this translation? | -- TODO: is this the best place for this translation? | ||
args[k] = v; | args[k] = v; | ||
Line 3,620: | Line 3,766: | ||
end | end | ||
return | --[[--------------------------< E X P O R T E D F U N C T I O N S >------------------------------------------ | ||
]] | |||
return {citation = citation}; |