Module:WikidataIB: Difference between revisions
From Zoophilia Wiki
Jump to navigationJump to search
meta>RexxS Sanitising labels returned from WIkidata |
meta>Dipsacus fullonum Don'r use before it is tested if the argument is nil (it would give script errors). In p.getLabel return Qid if label doesn't exist as the function description says. |
||
| Line 178: | Line 178: | ||
if (refs > 0) or (onlysrc == false) then -- has valid refs or all values required | if (refs > 0) or (onlysrc == false) then -- has valid refs or all values required | ||
local sitelink = mw.wikibase.sitelink("Q" .. v.mainsnak.datavalue.value["numeric-id"]) | local sitelink = mw.wikibase.sitelink("Q" .. v.mainsnak.datavalue.value["numeric-id"]) | ||
local label = | local label = mw.wikibase.label("Q" .. v.mainsnak.datavalue.value["numeric-id"]) | ||
if label = | if label then | ||
label = mw.text.nowiki(label) | |||
else | |||
label = "Q" .. v.mainsnak.datavalue.value["numeric-id"] | |||
end | |||
if sitelink then | if sitelink then | ||
out[#out + 1] = "[[" .. sitelink .. "|" .. label .. "]]" | out[#out + 1] = "[[" .. sitelink .. "|" .. label .. "]]" | ||
| Line 432: | Line 435: | ||
local valueID = v3.datavalue.value.id | local valueID = v3.datavalue.value.id | ||
local sitelink = mw.wikibase.sitelink(valueID) | local sitelink = mw.wikibase.sitelink(valueID) | ||
local label = | local label = mw.wikibase.label(valueID) | ||
if | if label then | ||
label = mw.text.nowiki(label) | |||
else | |||
label = valueID | |||
end | |||
if sitelink then | if sitelink then | ||
out[#out + 1] = "[[" .. sitelink .. "|" .. label .. "]]" | out[#out + 1] = "[[" .. sitelink .. "|" .. label .. "]]" | ||
| Line 462: | Line 469: | ||
if itemID == "" then return end | if itemID == "" then return end | ||
local sitelink = mw.wikibase.sitelink(itemID) | local sitelink = mw.wikibase.sitelink(itemID) | ||
local label = | local label = mw.wikibase.label(itemID) | ||
if | if label then | ||
label = mw.text.nowiki(label) | |||
else | |||
label = itemID | |||
end | |||
if sitelink then | if sitelink then | ||
return "[[" .. sitelink .. "|" .. label .. "]]" | return "[[" .. sitelink .. "|" .. label .. "]]" | ||
| Line 479: | Line 490: | ||
local itemID = mw.text.trim(frame.args[1] or "") | local itemID = mw.text.trim(frame.args[1] or "") | ||
if itemID == "" then return end | if itemID == "" then return end | ||
return mw.text.nowiki( | local label = mw.wikibase.label(itemID) | ||
if label then return mw.text.nowiki(label) end | |||
return itemID | |||
end | end | ||
return p | return p | ||
Revision as of 11:19, 6 February 2017
Documentation for this module may be created at Module:WikidataIB/doc
-- Module to try out use of a blacklist and whitelist for infobox fields
-- can take a named parameter |qid which is the Wikidata ID for the article. This will not normally be used
-- Fields in blacklist are never to be displayed, i.e. module must return nil in all circumstances
-- Fields in whitelist return local value if it exists or the Wikidata value otherwise
-- The name of the field that this function is called from is passed in named parameter |name
-- The name is compulsory when blacklist or whitelist is used, so the module returns nil if it is not supplied
-- blacklist is passed in named parameter |suppressfields
-- whitelist is passed in named parameter |fetchwikidata
local p = {}
local i18n =
{
["errors"] =
{
["property-not-found"] = "Property not found.",
["entity-not-found"] = "Wikidata entity not found.",
["unknown-claim-type"] = "Unknown claim type.",
["unknown-entity-type"] = "Unknown entity type.",
["qualifier-not-found"] = "Qualifier not found.",
["site-not-found"] = "Wikimedia project not found.",
["unknown-datetime-format"] = "Unknown datetime format.",
["local-article-not-found"] = "Article is available on Wikidata, but not on Wikipedia"
},
}
-------------------------------------------------------------------------------
-- formatDate takes a datetime of the usual format from mw.wikibase.entity:formatPropertyValues
-- like "1 August 30 BCE" as parameter 1 and formats it according to the df (date format) and bc parameters
-- df = ["dmy" / "mdy" / "y"] default will be "dmy"
-- bc = ["BC" / "BCE"] default will be "BCE"
-- first the local version
local format_Date = function(datetime, dateformat, bc)
local datetime = datetime or "1 August 30 BCE" -- in case of nil value
-- chop off multiple vales and/or any hours, mins, etc.
-- keep anything before punctuation - we just want a single date:
local dateval = string.match( datetime, "[%w ]+")
local dateformat = string.lower(dateformat or "dmy") -- default to dmy
local bc = string.upper(bc or "") -- can't use nil for bc
-- we only want to accept two possibilities: BC or default to BCE
if bc=="BC" then
bc = " BC" -- prepend the space. **internationalise later**
else
bc = " BCE"
end
local postchrist = true -- start by assuming no BCE
local dateparts = {}
for word in string.gmatch(dateval, "%w+") do
if word == "BCE" then -- **internationalise later**
postchrist = false
else
-- we'll keep the parts that are not 'BCE' in a table
dateparts[#dateparts + 1] = word
end
end
if postchrist then bc = "" end -- set AD dates to no suffix **internationalise later**
local sep = " " -- separator is nbsp
local fdate = table.concat(dateparts, " ") -- formatted date defaults to same order as input
-- if we have day month year, check dateformat
if #dateparts == 3 then
if dateformat == "y" then
fdate = dateparts[3]
elseif dateformat == "mdy" then
fdate = dateparts[2] .. sep .. dateparts[1] .. "," .. sep .. dateparts[3]
end
elseif #dateparts == 2 and dateformat == "y" then
fdate = dateparts[2]
end
return fdate .. bc
end
-- now wrap it up to export it
p.formatDate = function(frame)
return format_Date(frame.args[1], frame.args.df, frame.args.bc)
end
-------------------------------------------------------------------------------
-- getValue is used to get a value, or a comma separated list of them if multiple values exist
--
p.getValue = function(frame)
local propertyID = mw.text.trim(frame.args[1] or "")
-- There may be a local parameter supplied, if it's blank, set it to nil
local input_parm = mw.text.trim(frame.args[2] or "")
if input_parm and (#input_parm == 0) then input_parm = nil end
-- can take a named parameter |qid which is the Wikidata ID for the article.
-- This will not normally be used because it's an expensive call.
local qid = frame.args.qid
if qid and (#qid == 0) then qid = nil end
-- The blacklist is passed in named parameter |suppressfields
local blacklist = frame.args.suppressfields
-- The whitelist is passed in named parameter |fetchwikidata
local whitelist = frame.args.fetchwikidata
-- The name of the field that this function is called from is passed in named parameter |name
local fieldname = frame.args.name
-- onlysourced is a boolean passed to return only values sourced to other than Wikipedia
-- if nothing or an empty string is passed set it false
-- if "false" or "no" or 0 is passed set it false
local onlysrc = frame.args.onlysourced
if onlysrc and (#onlysrc > 0) then
onlysrc = onlysrc:lower()
if (onlysrc == "false") or (onlysrc == "no") or (onlysrc == 0) then
onlysrc = false
else
onlysrc = true
end
else
onlysrc = true -- this is the default for empty or missing 'onlysourced' parameter
end
-- noicon is a boolean passed to suppress the "edit at Wikidata" icon for use when the value is processed further by the infobox
-- if nothing or an empty string is passed set it false
-- if "false" or "no" or 0 is passed set it false
local noic = frame.args.noicon
if noic and (#noic > 0) then
noic = noic:lower()
if (noic == "false") or (noic == "no") or (noic == 0) then
noic = false
else
noic = true
end
else
noic = false
end
if blacklist then
-- The name is compulsory when blacklist is used, so return nil if it is not supplied
if not fieldname or (#fieldname == 0) then return nil end
-- If this field is on the blacklist, then return nil
if blacklist:find(fieldname) then return nil end
end
-- If we got this far then we're not on the blacklist
-- The blacklist overrides any locally supplied parameter as well
-- If a non-blank input parameter was supplied return it
if input_parm then return input_parm end
-- Otherwise see if this field is on the whitelist:
if whitelist and (whitelist == 'ALL' or whitelist:find(fieldname)) then
local entity = mw.wikibase.getEntityObject(qid)
local props
if entity and entity.claims then
props = entity.claims[propertyID]
end
if props then
local lang = mw.language.getContentLanguage().code
local thisQid
if qid then thisQid = qid else thisQid = entity.id end
local icon = " [[File:Blue pencil.svg |frameless |text-top |10px |alt=Edit this on Wikidata |link=https://www.wikidata.org/wiki/" .. thisQid .. "?uselang=" .. lang .. "#" .. propertyID .. "|Edit this on Wikidata]]"
if props[1] and props[1].mainsnak.snaktype == "value" and props[1].mainsnak.datavalue.type == "wikibase-entityid" then
-- it's wiki-linked value, so output as link if possible
local out = {}
for k, v in pairs(props) do
-- check for references,
-- construct a reference list for debugging
-- and count valid references
local reflist = ""
local refs = 0
if v.references then
for kr, vr in pairs(v.references) do
local ref = mw.wikibase.renderSnaks(vr.snaks)
reflist = reflist .. " <span style='color:#0DD;'>" .. ref .. "</span>"
if not ref:find("Wikipedia") then refs = refs + 1 end
end
end
if (refs > 0) or (onlysrc == false) then -- has valid refs or all values required
local sitelink = mw.wikibase.sitelink("Q" .. v.mainsnak.datavalue.value["numeric-id"])
local label = mw.wikibase.label("Q" .. v.mainsnak.datavalue.value["numeric-id"])
if label then
label = mw.text.nowiki(label)
else
label = "Q" .. v.mainsnak.datavalue.value["numeric-id"]
end
if sitelink then
out[#out + 1] = "[[" .. sitelink .. "|" .. label .. "]]"
else
out[#out + 1] = "[[:d:Q" .. v.mainsnak.datavalue.value["numeric-id"] .. "|" .. label .. "]] <span title='" .. i18n["errors"]["local-article-not-found"] .. "'>[[File:Wikidata-logo.svg|16px|alt=|link=]]</span>"
end
end
end
if #out > 0 then
if noic then
return table.concat(out, ", ")
else
return table.concat(out, ", ") .. icon
end
else
return nil -- no items had valid reference
end
else
-- not a linkable article title
-- this needs to be expanded to cater for multiple values
local reflist = ""
local refs = 0
for k, v in pairs(props) do
-- check for references,
-- construct a reference list for debugging
-- and count valid references
if v.references then
for kr, vr in pairs(v.references) do
local ref = mw.wikibase.renderSnaks(vr.snaks)
reflist = reflist .. " <span style='color:#0DD;'>" .. ref .. "</span>"
if not ref:find("Wikipedia") then refs = refs + 1 end
end
end
end
local propertyValue = entity:formatPropertyValues(propertyID).value
if props[1].mainsnak.datatype == "time" then
propertyValue = format_Date(propertyValue, frame.args.df, frame.args.bc)
end
if (refs > 0) or (onlysrc == false) then -- has valid refs or all values required
if noic then
return propertyValue
else
return propertyValue .. icon
end
else
return nil
end
end
else
-- no property stored for this article
return nil
end
else
-- not on the whitelist so just return what should be a nil input parameter
return input_parm
end
end
-------------------------------------------------------------------------------
-- getSourcedValue is used to get a value, or a comma separated list of them if multiple values exist
-- but only values that are sourced are returned
-- redundant to getValue with onlysourced=true but kept for backwards compatibility
-- now defined via getValue
--
p.getSourcedValue = function(frame)
frame.args.onlysourced = "yes"
return p.getValue(frame)
end
-------------------------------------------------------------------------------
-- getCoords is used to get coordinates for display in an infobox
-- whitelist and blacklist are implemented
-- optional 'display' parameter is allowed, defaults to "inline, title"
--
p.getCoords = function(frame)
local propertyID = "P625"
local input_parm = mw.text.trim(frame.args[1] or "")
if input_parm and (#input_parm == 0) then input_parm = nil end
local qid = frame.args.qid
if qid and (#qid == 0) then qid = nil end
-- if there is a 'display' parameter supplied, use it
-- otherwise default to "inline, title"
local disp = frame.args.display
if (not disp) or (#disp == 0) then
disp = "inline, title"
end
local blacklist = frame.args.suppressfields
local whitelist = frame.args.fetchwikidata
-- The name of the field that this function is called from is passed in named parameter |name
-- it's probably 'coords' but we can't be certain
local fieldname = frame.args.name
if blacklist then
if not fieldname or (#fieldname == 0) then return nil end
if blacklist:find(fieldname) then return nil end
end
if input_parm then return input_parm end
if whitelist and (whitelist == 'ALL' or whitelist:find(fieldname)) then
local entity = mw.wikibase.getEntityObject(qid)
local props
if entity and entity.claims then
props = entity.claims[propertyID]
end
if props then
local lat_long = {}
local coords = entity:formatPropertyValues(propertyID).value
-- the latitude and longitude are returned like this: nn°nn'nn.n"
-- using html entities with hex values really screws up parsing the numbers - thanks devs
local lat = mw.ustring.match(coords, "^[^,]*") -- everything from the start to before the comma
local long = mw.ustring.match(coords, "[^ ]*$") -- everything from after the space to the end
lat = lat:gsub("&#%d%d;", ":") -- clean out the html entities
long = long:gsub("&#%d%d;", ":") -- clean out the html entities
-- read the latitude numbers into a table
for num in mw.ustring.gmatch(lat, "%d+%.?%d*") do
lat_long[#lat_long + 1] = num
end
-- add the N/S
lat_long[#lat_long + 1] = lat:sub(-1)
-- read the longitude numbers into a table
for num in mw.ustring.gmatch(long, "%d+%.?%d*") do
lat_long[#lat_long + 1] = num
end
-- add E/W for long
lat_long[#lat_long + 1] = long:sub(-1)
-- add named parameter for display
lat_long["display"] = disp
-- invoke template Coord with the values stored in the table
return frame:expandTemplate{title = 'coord', args = lat_long}
else
-- no coords in Wikidata for this article
return nil
end
else
-- not on the whitelist so just return what should be a nil input parameter
return input_parm
end
end
-------------------------------------------------------------------------------
-- getQualifierValue is used to get a formatted value of a qualifier
--
-- The call needs: a property (the unnamed parameter or 1=)
-- a target value for that property (pval=)
-- a qualifier for that target value (qual=)
-- The usual whitelisting and blacklisting of the property is implemented
-- The boolean onlysourced= parameter can be set to return nothing
-- when the property is unsourced (or only sourced to Wikipedia)
--
p.getQualifierValue = function(frame)
local propertyID = mw.text.trim(frame.args[1] or "")
-- The PropertyID of the target value of the property
-- whose qualifier is to be returned is passed in named parameter |pval=
local propvalue = frame.args.pval
-- The PropertyID of the qualifier
-- whose value is to be returned is passed in named parameter |qual=
local qualifierID = frame.args.qual
-- Can take a named parameter |qid which is the Wikidata ID for the article.
-- This will not normally be used because it's an expensive call.
local qid = frame.args.qid
if qid and (#qid == 0) then qid = nil end
-- The blacklist is passed in named parameter |suppressfields=
local blacklist = frame.args.suppressfields
-- The whitelist is passed in named parameter |fetchwikidata=
local whitelist = frame.args.fetchwikidata
-- The name of the field to check against the whitelist and blacklist
-- is passed in named parameter |name
local fieldname = frame.args.name
-- onlysourced is a boolean passed to return qualifiers
-- only when property values are sourced to something other than Wikipedia
-- if nothing or an empty string is passed set it false
-- if "false" or "no" or 0 is passed set it false
local onlysrc = frame.args.onlysourced
if onlysrc and (#onlysrc > 0) then
onlysrc = onlysrc:lower()
if (onlysrc == "false") or (onlysrc == "no") or (onlysrc == 0) then
onlysrc = false
else
onlysrc = true
end
else
onlysrc = false
end
if blacklist then
-- The name is compulsory when blacklist is used, so return nil if it is not supplied
if not fieldname or (#fieldname == 0) then return nil end
-- If this field is on the blacklist, then return nil
if blacklist:find(fieldname) then return nil end
end
-- If we got this far then we're not on the blacklist
-- So see if this field is on the whitelist:
if whitelist and (whitelist == 'ALL' or whitelist:find(fieldname)) then
local entity = mw.wikibase.getEntityObject(qid)
local props
if entity and entity.claims then
props = entity.claims[propertyID]
end
if props then
-- Scan through the values of the property
-- we want something like property is P793, significant event (in propertyID)
-- whose value is something like Q385378, construction (in propvalue)
-- then we can return the value(s) of a qualifier such as P580, start time (in qualifierID)
for k1, v1 in pairs(props) do
if v1.mainsnak.snaktype == "value" and v1.mainsnak.datavalue.type == "wikibase-entityid" then
-- It's a wiki-linked value, so check if it's the target (in propvalue)
-- and if it has qualifiers
if v1.mainsnak.datavalue.value.id == propvalue and v1.qualifiers then
if v1.references then
-- count how many refs are sourced
local numrefs = 0
for k2, v2 in pairs(v1.references) do
local ref = mw.wikibase.renderSnaks(v2.snaks)
if not ref:find("Wikipedia") then numrefs = numrefs + 1 end
end
if (numrefs == 0) and (onlysrc == true) then
-- no sourced refs and sourced is required
return nil
end
else
if onlysrc == true then
-- no refs and sourced refs is required
return nil
end
end
-- if we've got this far, we have a (sourced) claim with qualifiers
-- which matches the target, so find the value(s) of the qualifier we want
local quals = v1.qualifiers[qualifierID]
local out = {}
if quals then
if quals[1].datatype == "wikibase-item" then
for k3, v3 in pairs(quals) do
local valueID = v3.datavalue.value.id
local sitelink = mw.wikibase.sitelink(valueID)
local label = mw.wikibase.label(valueID)
if label then
label = mw.text.nowiki(label)
else
label = valueID
end
if sitelink then
out[#out + 1] = "[[" .. sitelink .. "|" .. label .. "]]"
else
out[#out + 1] = "[[:d:" .. valueID .. "|" .. label .. "]] <span title='" .. i18n["errors"]["local-article-not-found"] .. "'>[[File:Wikidata-logo.svg|16px|alt=|link=]]</span>"
end
end
return table.concat(out, ", ")
else
return mw.wikibase.renderSnaks(quals)
end
end
end
end
end -- of loop through values of propertyID
end
end
return nil
end
-------------------------------------------------------------------------------
-- getLink returns the label for a Qid wiki-linked to the local article (if the article exists)
-- if label doesn't exist, it returns the Qid wiki-linked to the local article (if the article exists)
--
p.getLink = function(frame)
local itemID = mw.text.trim(frame.args[1] or "")
if itemID == "" then return end
local sitelink = mw.wikibase.sitelink(itemID)
local label = mw.wikibase.label(itemID)
if label then
label = mw.text.nowiki(label)
else
label = itemID
end
if sitelink then
return "[[" .. sitelink .. "|" .. label .. "]]"
else
return label
end
end
-------------------------------------------------------------------------------
-- getLabel returns the label for a Qid
-- if label doesn't exist, it returns the Qid
--
p.getLabel = function(frame)
local itemID = mw.text.trim(frame.args[1] or "")
if itemID == "" then return end
local label = mw.wikibase.label(itemID)
if label then return mw.text.nowiki(label) end
return itemID
end
return p