require 'Modul:No globals'
local p = {}
local lang = mw.language.getContentLanguage()
local getArgs = require('Modul:Arguments').getArgs
local TableTools = require('Modul:TableTools')
local Wikidata = require('Modul:Wikidata')
local Formatters = require('Modul:Wikidata/Formatters')
local function getIndexedData(id, property)
local statements = Wikidata.getStatements{
id = id,
property = property,
rank = 'valid',
withqualifier = 'P585'
}
local numbers_raw = {}
for _, statement in ipairs(statements) do
local n = Formatters.getRawValue(statement.mainsnak, {})
for _, qualifier in ipairs(statement.qualifiers.P585) do
local d = Formatters.getRawValue(qualifier, {})
if d and d.year then
numbers_raw[d.year] = n
end
end
end
return numbers_raw
end
local function getKeys(rows)
local keys = {}
for _, v in pairs(rows) do
for key in pairs(v) do
keys[key] = true
end
end
return keys
end
local function formatNumOrNothing(num)
num = tonumber(num)
if num then
return lang:formatNum(num)
else
return ''
end
end
local function createTable(data,sep)
local popTables={}
local divided={}
local i,j=0,sep
for _,v in pairs(data) do
if j == sep then
i=i+1
divided[i]={}
j=0
end
j=j+1
divided[i][j] = v
end
for _, t in pairs(divided) do
local keys = getKeys(t)
local popTable = mw.html.create( 'table' )
local thead = popTable
:css( 'display', 'inline-block' )
:css( 'vertical-align', 'top' )
:addClass( 'wikitable' )
:tag( 'tr' )
thead:tag( 'th' ):wikitext( 'Rok' )
if keys.population then
thead:tag( 'th' ):wikitext( 'Obyvatelé' )
end
if keys.houses then
thead:tag( 'th' ):wikitext( 'Domy' )
end
for _, v in pairs(t) do
local tr = popTable:tag( 'tr' )
:tag( 'th' )
:wikitext( v.year )
:done()
if keys.population then
tr:tag( 'td' )
:css( 'text-align', 'right' )
:wikitext( formatNumOrNothing(v.population) )
end
if keys.houses then
tr:tag( 'td' )
:css( 'text-align', 'right' )
:wikitext( formatNumOrNothing(v.houses) )
end
end
table.insert(popTables,tostring(popTable))
end
return table.concat(popTables)
end
local function drawGraph(indexed_data, opts)
local data = {}
local x_fields = {}
local y_fields = {}
for key, values in pairs(indexed_data) do
table.insert(data, { name = key, values = values })
table.insert(x_fields, { data = key, field = 'x' })
table.insert(y_fields, { data = key, field = 'y' })
end
local json = {
version = 2,
width = opts.width, height = opts.height,
data = data,
scales = {
{
name = "x",
type = "linear",
domain = { fields = x_fields },
range = "width",
zero = false,
nice = true
},
{
name = "y",
range = "height",
nice = true,
domain = { fields = y_fields }
}
},
axes = {
{ type = "x", scale = "x", tickcount = 20, format = "d", grid = true },
{ type = "y", scale = "y", format = "2d", grid = true }
},
marks = {
{
type = "line",
from = { data = "population" },
properties = {
enter = {
x = { scale = "x", field = "x" },
y = { scale = "y", field = "y" },
strokeWidth = { value = 5 },
stroke = { value = "steelblue" }
}
}
},
{
type = "line",
from = { data = "houses" },
properties = {
enter = {
x = { scale = "x", field = "x" },
y = { scale = "y", field = "y" },
strokeWidth = { value = 5 },
stroke = { value = "orange" }
}
}
}
}
}
if opts.labels == "ano" then
table.insert(json["marks"],
{
type = "text",
from = { data = "table" },
properties = {
enter = {
x = { scale = "x", field = "x" },
y = { scale = "y", field = "y" },
fill = { value = "#000" },
align = { value = "left" },
baseline = { value = "top" },
dy = { value = -13 },
text = { field = "y" }
}
}
}
)
end
return tostring(mw.html.create("div")
:css("display","inline-block")
:css("vertical-align","top")
:wikitext(mw.getCurrentFrame():extensionTag("graph", mw.text.jsonEncode(json))))
end
function p.main(frame)
local args
if type(frame) ~= 'table' then
args = { id = frame }
else
args = getArgs(frame)
end
local opts = {
width = tonumber(args["width"]) or 430,
height = tonumber(args["height"]) or 200,
labels = args["labels"],
}
local property = args["property"] or 'P1082'
local from = tonumber(args["from"])
local to = tonumber(args["to"])
local skip = tonumber(args["skip"]) or 1
local zalamovat = tonumber(args["zalamovat"])
local keys = {}
keys.population = property
if args.domy == 'ano' then
keys.houses = 'P4080'
end
local indexed_data = {}
local dates = {}
for key, property in pairs(keys) do
indexed_data[key] = getIndexedData(args.id, property)
for d in pairs(indexed_data[key]) do
dates[d] = true
end
end
dates = TableTools.keysToList(dates, true)
if #dates < 1 then
return ''
end
local data = {}
local tabular = {}
local last
for _, d in ipairs(dates) do
if (not(from) or (d >= from))
and (not(to) or (d <= to))
and (not(last) or ((d-last)>=skip)) then
local t = { year = d }
for key in pairs(keys) do
local val = indexed_data[key][d]
if val then
if not data[key] then
data[key] = {}
end
table.insert(data[key], { x = d, y = val })
t[key] = val
end
end
table.insert(tabular, t)
last = d
end
end
local output = {}
output[1] = args["nadpis"] or '<div style="font-weight:bold">Vývoj počtu obyvatel</div><div><small>Data pocházejí z datové položky Wikidat</small></div>'
local graphPos, tabPos
if args["tabulka"] == "nad" then
tabPos = 2
output[3] = "<br/>"
graphPos = 4
elseif (args["tabulka"] == "pod") then
graphPos = 2
output[3] = "<br/>"
tabPos = 4
elseif (args["tabulka"] == "před") then
tabPos = 2
graphPos = 3
elseif (args["tabulka"] == "za") then
graphPos = 2
tabPos = 3
else
graphPos = 2
end
output[graphPos] = drawGraph(data, opts)
if tabPos then output[tabPos] = createTable(tabular, zalamovat) end
return table.concat(output)
end
return p