Module:Chart: Difference between revisions

Content deleted Content added
replaced by Ita140188's version
Copyedit. Add number got to error message.
 
(14 intermediate revisions by 8 users not shown)
Line 1:
--<source lang=lua>
--[[
keywords are used for languages: they are the names of the actual
parameters of the template
]]
 
local keywords = {
barChart = 'bar chart',
pieChart = 'pie chart',
width = 'width',
height = 'height',
stack = 'stack',
colors = 'colors',
group = 'group',
xlegend = 'x legends',
yticks = 'y tick marks',
tooltip = 'tooltip',
tooltip = 'tooltip',
accumulateTooltip = 'tooltip value accumulation',
links = 'links',
defcolor = 'default color',
scalePerGroup = 'scale per group',
unitsPrefix = 'units prefix',
unitsSuffix = 'units suffix',
groupNames = 'group names',
hideGroupLegends = 'hide group legends',
slices = 'slices',
optint = 'only integer values',
slices slice = 'slicesslice',
slice radius = 'sliceradius',
radius percent = 'radiuspercent',
percent = 'percent',
 
} -- here is what you want to translate
 
local defColors = mw.loadData("Module:Chart/Default colors")
 
 
local defColors = require "Module:Plotter/DefaultColors"
local hideGroupLegends
 
local function nulOrWhitespace( s )
return not s or mw.text.trim( s ) == ''
end
 
local function createGroupList( tab, legends, cols )
if #legends > 1 and not hideGroupLegends then
table.insert( tab, mw.text.tag( 'div' ) )
local list = {}
local spanStyle = "padding:0 1em;background-color:%s;border:1px solid %s;margin-right:1em;-webkit-print-color-adjust:exact;"
for gi = 1, #legends do
local span = mw.text.tag( 'span', { style = string.format( spanStyle, cols[gi], cols[gi] ) }, '&nbsp;' ) .. ' '.. legends[gi]
table.insert( list, mw.text.tag( 'li', {}, span ) )
end
table.insert( tab,
mw.text.tag( 'ul',
{style="width:100%;list-style:none;-webkit-column-width:12em;-moz-column-width:12em;column-width:12em;"},
table.concat( list, '\n' )
)
)
)
)
table.insert( tab, '</div>' )
end
end
 
local function pieChart( frame )
local res, imslices, args = {}, {}, frame.args
local radius
local values, colors, names, legends, links = {}, {}, {}, {}, {}
local delimiter = args.delimiter or ':'
local lang = mw.getContentLanguage()
 
local function pieChartgetArg( frames, def, subst, with )
local res,result imslices,= args[keywords[s]] =or {},def {},or frame.args''
if subst and with then result = string.gsub( result, subst, with ) end
local radius
return result
local values, colors, names, legends, links = {}, {}, {}, {}, {}
end
local delimiter = args.delimiter or ':'
local lang = mw.getContentLanguage()
 
local function getArganalyzeParams( s, def, subst, with )
local function addSlice( i, slice )
local result = args[keywords[s]] or def or ''
local value, name, color, link = unpack( mw.text.split( slice, '%s*' .. delimiter .. '%s*' ) )
if subst and with then result = mw.ustring.gsub( result, subst, with ) end
values[i] = tonumber( lang:parseFormattedNumber( value ) )
return result
or error( string.format( 'Slice %d: "%s", first item("%s") could not be parsed as a number', i, value or '', slice ) )
end
colors[i] = not nulOrWhitespace( color ) and color or defColors[i * 2]
names[i] = name or ''
links[i] = link
end
 
radius = getArg( 'radius', 150 )
function analyzeParams()
hideGroupLegends = not nulOrWhitespace( args[keywords.hideGroupLegends] )
function addSlice( i, slice )
local slicesStr = getArg( 'slices' )
local value, name, color, link = unpack( mw.text.split( slice, '%s*' .. delimiter .. '%s*' ) )
local prefix = getArg( 'unitsPrefix', '', '_', ' ' )
values[i] = tonumber( lang:parseFormattedNumber( value ) )
local suffix = getArg( 'unitsSuffix', '', '_', ' ' )
or error( string.format( 'Slice %d: "%s", first item("%s") could not be parsed as a number', i, value or '', sliceStr ) )
local percent = args[keywords.percent]
colors[i] = not nulOrWhitespace( color ) and color or defColors[i * 2]
local sum = 0
names[i] = name or ''
local i = 0
links[i] = link
for slice in string.gmatch( slicesStr or '', "%b()" ) do
end
i = i + 1
addSlice( i, string.match( slice, '^%(%s*(.-)%s*%)$' ) )
radius = getArg( 'radius', 150 )
end
hideGroupLegends = not nulOrWhitespace( args[keywords.hideGroupLegends] )
local slicesStr = getArg( 'slices' )
local prefix = getArg( 'unitsPrefix', '', '_', ' ' )
local suffix = getArg( 'unitsSuffix', '', '_', ' ' )
local percent = args[keywords.percent]
local sum = 0
local i, value = 0
for slice in mw.ustring.gmatch( slicesStr or '', "%b()" ) do
i = i + 1
addSlice( i, mw.ustring.match( slice, '^%(%s*(.-)%s*%)$' ) )
end
for k, v in pairs(args) do
local ind = mw.ustring.match( k, '^' .. keywords.slice .. '%s+(%d+)$' )
if ind then addSlice( tonumber( ind ), v ) end
end
for _, val in ipairs( values ) do sum = sum + val end
for i, value in ipairs( values ) do
local addprec = percent and string.format( ' (%0.1f%%)', value / sum * 100 ) or ''
legends[i] = mw.ustring.format( '%s: %s%s%s%s', names[i], prefix, lang:formatNum( value ), suffix, addprec )
links[i] = mw.text.trim( links[i] or mw.ustring.format( '[[#noSuchAnchor|%s]]', legends[i] ) )
end
end
 
for k, v in pairs(args) do
function addRes( ... )
local ind = string.match( k, '^' .. keywords.slice .. '%s+(%d+)$' )
for _, v in pairs( { ... } ) do
if ind then addSlice( tonumber( ind ), v ) end
table.insert( res, v )
end
end
 
for _, val in ipairs( values ) do sum = sum + val end
function createImageMap()
for i, value in ipairs( values ) do
addRes( '{{#tag:imagemap|', 'Image:Circle frame.svg{{!}}' .. ( radius * 2 ) .. 'px' )
local addprec = percent and string.format( ' (%0.1f%%)', value / sum * 100 ) or ''
addRes( unpack( imslices ) )
legends[i] = string.format( '%s: %s%s%s%s', names[i], prefix, lang:formatNum( value ), suffix, addprec )
addRes( 'desc none', '}}' )
links[i] = mw.text.trim( links[i] or string.format( '[[#noSuchAnchor|%s]]', legends[i] ) )
end
end
end
 
local function drawSliceaddRes( i, q, start... )
for _, v in pairs( { ... } ) do
local color = colors[i]
table.insert( res, v )
local angle = start * 2 * math.pi
end
local sin, cos = math.abs( math.sin( angle ) ), math.abs( math.cos( angle ) )
end
local wsin, wcos = sin * radius, cos * radius
local s1, s2, w1, w2, w3, w4, width, border
local style
if q == 1 then
border = 'left'
w1, w2, w3, w4 = 0, 0, wsin, wcos
s1, s2 = 'bottom', 'left'
elseif q == 2 then
border = 'bottom'
w1, w2, w3, w4 = 0, wcos, wsin, 0
s1, s2 = 'bottom', 'right'
elseif q == 3 then
border = 'right'
w1, w2, w3, w4 = wsin, wcos, 0, 0
s1, s2 = 'top', 'right'
else
border = 'top'
w1, w2, w3, w4 = wsin, 0, 0, wcos
s1, s2 = 'top', 'left'
end
 
local function createImageMap()
local style = string.format( 'border:solid transparent;position:absolute;%s:%spx;%s:%spx;width:%spx;height:%spx', s1, radius, s2, radius, radius, radius )
addRes( '{{#tag:imagemap|', 'File:Circle frame.svg{{!}}' .. ( radius * 2 ) .. 'px' )
if start <= ( q - 1 ) * 0.25 then
addRes( unpack( imslices ) )
style = string.format( '%s;border:0;background-color:%s', style, color )
addRes( 'desc none', '}}' )
else
end
style = string.format( '%s;border-width:%spx %spx %spx %spx;border-%s-color:%s', style, w1, w2, w3, w4, border, color )
end
addRes( mw.text.tag( 'div', { style = style }, '' ) )
end
 
local function drawSlice( i, q, start )
function createSlices()
local color = colors[i]
function coordsOfAngle( angle )
local angle = start * 2 * math.pi
return ( 100 + math.floor( 100 * math.cos( angle ) ) ) .. ' ' .. ( 100 - math.floor( 100 * math.sin( angle ) ) )
local sin, cos = math.abs( math.sin( angle ) ), math.abs( math.cos( angle ) )
end
local wsin, wcos = sin * radius, cos * radius
local s1, s2, w1, w2, w3, w4, border
if q == 1 then
border = 'left'
w1, w2, w3, w4 = 0, 0, wsin, wcos
s1, s2 = 'bottom', 'left'
elseif q == 2 then
border = 'bottom'
w1, w2, w3, w4 = 0, wcos, wsin, 0
s1, s2 = 'bottom', 'right'
elseif q == 3 then
border = 'right'
w1, w2, w3, w4 = wsin, wcos, 0, 0
s1, s2 = 'top', 'right'
else
border = 'top'
w1, w2, w3, w4 = wsin, 0, 0, wcos
s1, s2 = 'top', 'left'
end
 
local style = string.format( 'border:solid transparent;position:absolute;%s:%spx;%s:%spx;width:%spx;height:%spx', s1, radius, s2, radius, radius, radius )
local sum, start = 0, 0
if start <= ( q - 1 ) * 0.25 then
for _, value in ipairs( values ) do sum = sum + value end
style = string.format( '%s;border:0;background-color:%s', style, color )
for i, value in ipairs(values) do
else
local poly = { 'poly 100 100' }
style = string.format( '%s;border-width:%spx %spx %spx %spx;border-%s-color:%s', style, w1, w2, w3, w4, border, color )
local startC, endC = start / sum, ( start + value ) / sum
end
local startQ, endQ = math.floor( startC * 4 + 1 ), math.floor( endC * 4 + 1 )
addRes( mw.text.tag( 'div', { style = style }, '' ) )
for q = startQ, math.min( endQ, 4 ) do drawSlice( i, q, startC ) end
end
for angle = startC * 2 * math.pi, endC * 2 * math.pi, 0.02 do
table.insert( poly, coordsOfAngle( angle ) )
end
table.insert( poly, coordsOfAngle( endC * 2 * math.pi ) .. ' 100 100 ' .. links[i] )
table.insert( imslices, table.concat( poly, ' ' ) )
start = start + values[i]
end
end
 
local function createSlices()
analyzeParams()
local function coordsOfAngle( angle )
if #values == 0 then error( "no slices found - can't draw pie chart" ) end
return ( 100 + math.floor( 100 * math.cos( angle ) ) ) .. ' ' .. ( 100 - math.floor( 100 * math.sin( angle ) ) )
addRes( mw.text.tag( 'div', { class = 'chart noresize', style = string.format( 'margin-top:0.5em;max-width:%spx;', radius * 2 ) } ) )
end
addRes( mw.text.tag( 'div', { style = string.format( 'position:relative;min-width:%spx;min-height:%spx;max-width:%spx;overflow:hidden;', radius * 2, radius * 2, radius * 2 ) } ) )
createSlices()
addRes( mw.text.tag( 'div', { style = string.format( 'position:absolute;min-width:%spx;min-height:%spx;overflow:hidden;', radius * 2, radius * 2 ) } ) )
createImageMap()
addRes( '</div>' ) -- close "position:relative" div that contains slices and imagemap.
addRes( '</div>' ) -- close "position:relative" div that contains slices and imagemap.
createGroupList( res, legends, colors ) -- legends
addRes( '</div>' ) -- close containing div
return frame:preprocess( table.concat( res, '\n' ) )
end
 
local sum, start = 0, 0
for _, value in ipairs( values ) do sum = sum + value end
for i, value in ipairs(values) do
local poly = { 'poly 100 100' }
local startC, endC = start / sum, ( start + value ) / sum
local startQ, endQ = math.floor( startC * 4 + 1 ), math.floor( endC * 4 + 1 )
for q = startQ, math.min( endQ, 4 ) do drawSlice( i, q, startC ) end
for angle = startC * 2 * math.pi, endC * 2 * math.pi, 0.02 do
table.insert( poly, coordsOfAngle( angle ) )
end
table.insert( poly, coordsOfAngle( endC * 2 * math.pi ) .. ' 100 100 ' .. links[i] )
table.insert( imslices, table.concat( poly, ' ' ) )
start = start + values[i]
end
end
 
analyzeParams()
function barChart( frame )
if #values == 0 then error( "no slices found - can't draw pie chart" ) end
addRes( mw.text.tag( 'div', { class = 'chart noresize', style = string.format( 'margin-top:0.5em;max-width:%spx;', radius * 2 ) } ) )
local res = {}
addRes( mw.text.tag( 'div', { style = string.format( 'position:relative;min-width:%spx;min-height:%spx;max-width:%spx;overflow:hidden;', radius * 2, radius * 2, radius * 2 ) } ) )
local args = frame.args -- can be changed to frame:getParent().args
createSlices()
local values, xlegends, colors, tooltips, yscales, yscalesneg = {}, {}, {}, {} ,{}, {}, {}, {}
addRes( mw.text.tag( 'div', { style = string.format( 'position:absolute;min-width:%spx;min-height:%spx;overflow:hidden;', radius * 2, radius * 2 ) } ) )
local groupNames, unitsSuffix, unitsPrefix, links = {}, {}, {}, {}
createImageMap()
local width, height, stack, delimiter = 500, 350, false, args.delimiter or ':'
addRes( '</div>' ) -- close "position:relative" div that contains slices and imagemap.
local chartWidth, chartHeight, defcolor, scalePerGroup, accumulateTooltip
addRes( '</div>' ) -- close "position:relative" div that contains slices and imagemap.
 
createGroupList( res, legends, colors ) -- legends
local numGroups, numValues
addRes( '</div>' ) -- close containing div
local scaleWidth
return frame:preprocess( table.concat( res, '\n' ) )
 
end
local defColors = require "Module:Plotter/DefaultColors"
local hideGroupLegends
local optint
local function nulOrWhitespace( s )
return not s or mw.text.trim( s ) == ''
end
table.insert( res, frame:extensionTag{ name = 'templatestyles', args = { src = 'TemplateStyles sandbox/Ita140188/styles.css'} })
 
function createGroupList( tab, legends, cols )
if #legends > 1 and not hideGroupLegends then
table.insert( tab, mw.text.tag( 'div', { style = string.format( "width:%spx;", chartWidth ) } ) )
local list = {}
local spanStyle = "padding:0 1em;background-color:%s;border:1px solid %s;margin-right:1em;-webkit-print-color-adjust:exact;"
for gi = 1, #legends do
local span = mw.text.tag( 'span', { style = string.format( spanStyle, cols[gi], cols[gi] ) }, '&nbsp;' ) .. ' '.. legends[gi]
table.insert( list, mw.text.tag( 'li', {}, span ) )
end
table.insert( tab,
mw.text.tag( 'ul',
{style="width:100%;list-style:none;-webkit-column-width:12em;-moz-column-width:12em;column-width:12em;"},
table.concat( list, '\n' )
)
)
table.insert( tab, '</div>' )
end
end
 
 
 
function validate()
function asGroups( name, tab, toDuplicate, emptyOK )
if #tab == 0 and not emptyOK then
error( "must supply values for " .. keywords[name] )
end
if #tab == 1 and toDuplicate then
for i = 2, numGroups do tab[i] = tab[1] end
end
if #tab > 0 and #tab ~= numGroups then
error ( keywords[name] .. ' must contain the same number of items as the number of groups, but it contains ' .. #tab .. ' items and there are ' .. numGroups .. ' groups')
end
end
 
-- do all sorts of validation here, so we can assume all params are good from now on.
-- among other things, replace numerical values with mw.language:parseFormattedNumber() result
 
 
local function barChart( frame )
chartHeight = height - 80
local res = {}
numGroups = #values
local args = frame.args -- can be changed to frame:getParent().args
numValues = #values[1]
local values, xlegends, colors, tooltips, yscales = {}, {}, {}, {}, {}
defcolor = defcolor or 'blue'
local groupNames, unitsSuffix, unitsPrefix, links = {}, {}, {}, {}
colors[1] = colors[1] or defcolor
local width, height, yticks, stack, delimiter = 500, 350, -1, false, args.delimiter or ':'
scaleWidth = scalePerGroup and 80 * numGroups or 30
local chartWidth, chartHeight, defcolor, scalePerGroup, accumulateTooltip
chartWidth = width -scaleWidth
asGroups( 'unitsPrefix', unitsPrefix, true, true )
asGroups( 'unitsSuffix', unitsSuffix, true, true )
asGroups( 'colors', colors, true, true )
asGroups( 'groupNames', groupNames, false, false )
if stack and scalePerGroup then
error( string.format( 'Illegal settings: %s and %s are incompatible.', keyword.stack, keyword.scalePerGroup ) )
end
for gi = 2, numGroups do
if #values[gi] ~= numValues then error( keywords.group .. " " .. gi .. " does not have same number of values as " .. keywords.group .. " 1" ) end
end
if #xlegends ~= numValues then error( 'Illegal number of ' .. keywords.xlegend .. '. Should be exactly ' .. numValues ) end
end
 
function extractParams()
function testone( keyword, key, val, tab )
i = keyword == key and 0 or key:match( keyword .. "%s+(%d+)" )
if not i then return end
i = tonumber( i ) or error("Expect numerical index for key " .. keyword .. " instead of '" .. key .. "'")
if i > 0 then tab[i] = {} end
for s in mw.text.gsplit( val, '%s*' .. delimiter .. '%s*' ) do
table.insert( i == 0 and tab or tab[i], s )
end
return true
end
 
local numGroups, numValues
for k, v in pairs( args ) do
local scaleWidth
if k == keywords.width then
width = tonumber( v )
if not width or width < 200 then
error( 'Illegal width value (must be a number, and at least 200): ' .. v )
end
elseif k == keywords.height then
height = tonumber( v )
if not height or height < 200 then
error( 'Illegal height value (must be a number, and at least 200): ' .. v )
end
elseif k == keywords.stack then stack = true
elseif k == keywords.optint then optint = true
elseif k == keywords.scalePerGroup then scalePerGroup = true
elseif k == keywords.defcolor then defcolor = v
elseif k == keywords.accumulateTooltip then accumulateTooltip = not nulOrWhitespace( v )
elseif k == keywords.hideGroupLegends then hideGroupLegends = not nulOrWhitespace( v )
else
for keyword, tab in pairs( {
group = values,
xlegend = xlegends,
colors = colors,
tooltip = tooltips,
unitsPrefix = unitsPrefix,
unitsSuffix = unitsSuffix,
groupNames = groupNames,
links = links,
} ) do
if testone( keywords[keyword], k, v, tab )
then break
end
end
end
end
end
 
local function validate()
function roundup( x ) -- returns the next round number: eg., for 30 to 39.999 will return 40, for 3000 to 3999.99 wil return 4000. for 10 - 14.999 will return 15.
local function asGroups( name, tab, toDuplicate, emptyOK )
local ordermag = 10 ^ math.floor( math.log10( x ) )
if #tab == 0 and not emptyOK then
local normalized = x / ordermag
error( "must supply values for " .. keywords[name] )
local top = normalized >= 1.5 and ( math.floor( normalized + 1 ) ) or 1.5
end
return ordermag * top, top, ordermag
if #tab == 1 and toDuplicate then
end
for i = 2, numGroups do tab[i] = tab[1] end
end
if #tab > 0 and #tab ~= numGroups then
error ( keywords[name] .. ' must contain the same number of items as the number of groups, but it contains ' .. #tab .. ' items and there are ' .. numGroups .. ' groups')
end
end
 
-- do all sorts of validation here, so we can assume all params are good from now on.
function calcHeightLimits() -- if limits were passed by user, use them, otherwise calculate. for "stack" there's only one limet.
-- among other things, replace numerical values with mw.language:parseFormattedNumber() result
if stack then
local sums = {}
for _, group in pairs( values ) do
for i, val in ipairs( group ) do sums[i] = ( sums[i] or 0 ) + val end
end
local sum = math.max( unpack( sums ) )
local sumneg = math.min( unpack( sums ) )
for i = 1, #values do
yscales[i] = sum
yscalesneg[i] = 0 -- -sumneg
end
else
for i, group in ipairs( values ) do
yscales[i] = math.max( unpack( group ) )
yscalesneg[i] = -math.min( unpack( group ) )
end
end
for i, scale in ipairs( yscales ) do yscales[i] = roundup( scale * 0.9999 ) end
for i, scale in ipairs( yscalesneg ) do yscalesneg[i] = roundup( scale * 0.9999 ) end
if not scalePerGroup then for i = 1, #values do
yscales[i] = math.max( unpack( yscales ) )
yscalesneg[i] = math.max ( 0 , math.max( unpack( yscalesneg ) ) )
end end
end
 
function tooltip( gi, i, val )
if tooltips and tooltips[gi] and not nulOrWhitespace( tooltips[gi][i] ) then return tooltips[gi][i], true end
local groupName = not nulOrWhitespace( groupNames[gi] ) and groupNames[gi] .. ': ' or ''
local prefix = unitsPrefix[gi] or unitsPrefix[1] or ''
local suffix = unitsSuffix[gi] or unitsSuffix[1] or ''
return mw.ustring.gsub(groupName .. prefix .. mw.getContentLanguage():formatNum( tonumber( val ) or 0 ) .. suffix, '_', ' '), false
end
 
chartHeight = height - 80
function calcHeights( gi, i, val )
numGroups = #values
local barHeight = 0
numValues = #values[1]
local top, base = 0, 0
defcolor = defcolor or 'blue'
if tonumber( val ) >= 0 then
colors[1] = colors[1] or defcolor
barHeight = math.floor( val / ( yscales[gi] + yscalesneg[gi] ) * chartHeight + 0.5 ) -- add half to make it "round" instead of "trunc"
scaleWidth = scalePerGroup and 80 * numGroups or 100
top, base = chartHeight - barHeight - yscalesneg[gi] / ( yscales[gi] + yscalesneg[gi] ) * chartHeight , 0
chartWidth = width - scaleWidth
-- barHeight = math.floor( val / yscales[gi] * chartHeight + 0.5 ) -- add half to make it "round" instead of "trunc"
asGroups( 'unitsPrefix', unitsPrefix, true, true )
-- top, base = chartHeight - barHeight, 0
asGroups( 'unitsSuffix', unitsSuffix, true, true )
else
asGroups( 'colors', colors, true, true )
barHeight = math.floor( -val / ( yscales[gi] + yscalesneg[gi] ) * chartHeight + 0.5 ) -- add half to make it "round" instead of "trunc"
asGroups( 'groupNames', groupNames, false, false )
top, base = chartHeight - yscales[gi] / ( yscales[gi] + yscalesneg[gi] ) * chartHeight, 0
if stack and scalePerGroup then
end
error( string.format( 'Illegal settings: %s and %s are incompatible.', keywords.stack, keywords.scalePerGroup ) )
if stack then
end
local rawbase = 0
for gi = 2, numGroups do
for j = 1, gi - 1 do rawbase = rawbase + values[j][i] end -- sum the "i" value of all the groups below our group, gi.
if #values[gi] ~= numValues then error( keywords.group .. " " .. gi .. " does not have same number of values as " .. keywords.group .. " 1" ) end
base = math.floor( chartHeight * rawbase / yscales[gi] ) -- normally, and especially if it's "stack", all the yscales must be equal.
end
if #xlegends ~= numValues then error( 'Illegal number of ' .. keywords.xlegend .. '. Should be exactly ' .. numValues .. ' not ' .. #xlegends) end
return barHeight, top - base
end
function calcHeightsOld( gi, i, val )
local barHeight = math.floor( val / yscales[gi] * chartHeight + 0.5 ) -- add half to make it "round" instead of "trunc"
local top, base = chartHeight - barHeight, 0
if stack then
local rawbase = 0
for j = 1, gi - 1 do rawbase = rawbase + values[j][i] end -- sum the "i" value of all the groups below our group, gi.
base = math.floor( chartHeight * rawbase / yscales[gi] ) -- normally, and especially if it's "stack", all the yscales must be equal.
end
return barHeight, top - base
end
 
local function groupBoundsextractParams( i )
local function testone( keyword, key, val, tab )
local setWidth = math.floor( chartWidth / numValues )
local i = keyword == key and 0 or key:match( keyword .. "%s+(%d+)" )
-- local setOffset = ( i - 1 ) * setWidth
if not i then return end
local setOffset = ( i - 1 ) * setWidth
i = tonumber( i ) or error("Expect numerical index for key " .. keyword .. " instead of '" .. key .. "'")
return setOffset, setWidth
if i > 0 then tab[i] = {} end
for s in mw.text.gsplit( val, '%s*' .. delimiter .. '%s*' ) do
table.insert( i == 0 and tab or tab[i], s )
end
return true
end
 
for k, v in function calcxpairs( gi,args i) )do
if k == keywords.width then
local setOffset, setWidth = groupBounds( i )
width = tonumber( v )
if stack or numGroups == 1 then
if not width or width < 200 then
local barWidth = math.min( 38, math.floor( 0.8 * setWidth ) )
error( 'Illegal width value (must be a number, and at least 200): ' .. v )
return setOffset + (setWidth - barWidth) / 2, barWidth
end
elseif k == keywords.height then
setWidth = 0.85 * setWidth
height = tonumber( v )
local barWidth = math.floor( 0.75 * setWidth / numGroups )
if not height or height < 200 then
local left = setOffset + math.floor( ( gi - 1 ) / numGroups * setWidth )
error( 'Illegal height value (must be a number, and at least 200): ' .. v )
return left, barWidth
end
elseif k == keywords.stack then stack = true
elseif k == keywords.yticks then yticks = tonumber(v) or -1
elseif k == keywords.scalePerGroup then scalePerGroup = true
elseif k == keywords.defcolor then defcolor = v
elseif k == keywords.accumulateTooltip then accumulateTooltip = not nulOrWhitespace( v )
elseif k == keywords.hideGroupLegends then hideGroupLegends = not nulOrWhitespace( v )
else
for keyword, tab in pairs( {
group = values,
xlegend = xlegends,
colors = colors,
tooltip = tooltips,
unitsPrefix = unitsPrefix,
unitsSuffix = unitsSuffix,
groupNames = groupNames,
links = links,
} ) do
if testone( keywords[keyword], k, v, tab )
then break
end
end
end
end
end
 
local function roundup( x ) -- returns the next round number: eg., for 30 to 39.999 will return 40, for 3000 to 3999.99 wil return 4000. for 10 - 14.999 will return 15.
function drawbar( gi, i, val, ttval )
local ordermag = 10 ^ math.floor( math.log10( x ) )
if val == '0' then return end -- do not show single line (borders....) if value is 0, or rather, '0'. see talkpage
local normalized = x / ordermag
local top = normalized >= 1.5 and ( math.floor( normalized + 1 ) ) or 1.5
return ordermag * top, top, ordermag
end
 
local function calcHeightLimits() -- if limits were passed by user, use them, otherwise calculate. for "stack" there's only one limet.
local color, tooltip, custom = colors[gi] or defcolor or 'blue', tooltip( gi, i, ttval or val )
if stack then
local left, barWidth = calcx( gi, i )
local sums = {}
local barHeight, top = calcHeights( gi, i, val )
for _, group in pairs( values ) do
for i, val in ipairs( group ) do sums[i] = ( sums[i] or 0 ) + val end
-- borders so it shows up when printing
end
local style = string.format("position:absolute;left:%spx;top:%spx;height:%spx;min-width:%spx;max-width:%spx;background-color:%s;-webkit-print-color-adjust:exact;border:1px solid %s;border-bottom:none;overflow:hidden;",
local sum = math.max( unpack( sums ) )
left, top, barHeight-1, barWidth-2, barWidth-2, color, color)
for i = 1, #values do yscales[i] = sum end
local link = links[gi] and links[gi][i] or ''
else
local img = not nulOrWhitespace( link ) and mw.ustring.format( '[[File:Transparent.png|1000px|link=%s|%s]]', link, custom and tooltip or '' ) or ''
for i, group in ipairs( values ) do yscales[i] = math.max( unpack( group ) ) end
table.insert( res, mw.text.tag( 'div', { style = style, title = tooltip, class = "chart2-bar" }, img ) )
end
for i, scale in ipairs( yscales ) do yscales[i] = roundup( scale * 0.9999 ) end
if not scalePerGroup then for i = 1, #values do yscales[i] = math.max( unpack( yscales ) ) end end
end
 
local function tooltip( gi, i, val )
if tooltips and tooltips[gi] and not nulOrWhitespace( tooltips[gi][i] ) then return tooltips[gi][i], true end
local groupName = mw.text.killMarkers(not nulOrWhitespace( groupNames[gi] ) and groupNames[gi] .. ': ' or '')
local prefix = unitsPrefix[gi] or unitsPrefix[1] or ''
local suffix = unitsSuffix[gi] or unitsSuffix[1] or ''
return string.gsub(groupName .. prefix .. mw.getContentLanguage():formatNum( tonumber( val ) or 0 ) .. suffix, '_', ' '), false
end
 
local function calcHeights( gi, i, val )
local barHeight = math.max( 2, math.floor( val / yscales[gi] * chartHeight + 0.5 ) ) -- add half to make it "round" instead of "trunc", min height to 2 to avoid negative bar sizes
local top, base = chartHeight - barHeight, 0
if stack then
for j = 1, gi - 1 do
if tonumber(values[j][i]) > 0 then
base = base + math.max( 2, math.floor( values[j][i] / yscales[gi] * chartHeight + 0.5 ) ) -- sum the "i" value of all the groups below our group, gi, and keep the same calculation for each bar
end
end
end
return barHeight, top - base
end
 
local function groupBounds( i )
local setWidth = math.floor( chartWidth / numValues )
local setOffset = ( i - 1 ) * setWidth
return setOffset, setWidth
end
 
local function calcx( gi, i )
local setOffset, setWidth = groupBounds( i )
if stack or numGroups == 1 then
local barWidth = math.min( 38, math.floor( 0.8 * setWidth ) )
return setOffset + (setWidth - barWidth) / 2, barWidth
end
setWidth = 0.85 * setWidth
local barWidth = math.floor( 0.75 * setWidth / numGroups )
local left = setOffset + math.floor( ( gi - 1 ) / numGroups * setWidth )
return left, barWidth
end
 
local function drawbar( gi, i, val, ttval )
function drawYScale()
if val == '0' then return end -- do not show single line (borders....) if value is 0, or rather, '0'. see talkpage
function drawSingle( gi, color, single )
local yscale = yscales[gi]
local _, top, ordermag = roundup( yscale * 0.999 )
local numnotches = top <= 1.5 and top * 4
or top < 4 and top * 2
or top
local valStyleStrCntnr = 'display:block;position:relative;height:%spx;text-align:right;margin:0px;' -- SINGLE ELEMENT OF Y AXIS
local valStyleStrValue = 'display:block;position:relative;float:right;height:%spx;text-align:right;margin:%spx 0px 0px 0px;vertical-align:middle;line-height:%spx;' -- value
local valStyleStrNotch = 'display:block;position:relative;float:right;height:%spx;text-align:right;width:5px; border-top:1px solid black' -- notch
-- or 'position:relative;height=20px;text-align:right;vertical-align:middle;max-width:%spx;top:%spx;left:3px;background-color:%s;color:white;font-weight:bold;text-shadow:-1px -1px 0 #000,1px -1px 0 #000,-1px 1px 0 #000,1px 1px 0 #000;padding:0 2px'
-- local notchStyleStr = 'position:absolute;height=1px;min-width:5px;top:%spx;left:%spx;border:1px solid %s;'
for i = 1, numnotches do
local val = (numnotches - i + 1) / numnotches * yscale -- value of this notch
local y = ( 1 / numnotches * chartHeight ) --chartHeight - calcHeights( gi, 1, val ) -- height of a single notch
local divCntnr = mw.text.tag( 'div', { style = string.format( valStyleStrCntnr, y, color ) } )
local divValue = mw.text.tag( 'div', { style = string.format( valStyleStrValue, y, -y/2, y, color ) }, mw.getContentLanguage():formatNum( tonumber( val ) or 0 ) )
local divNotch = mw.text.tag( 'div', { style = string.format( valStyleStrNotch, y, color ) }, '&nbsp;' )
table.insert( res, divCntnr )
if val~=math.floor(val) and optint then
else
table.insert( res, divNotch )
table.insert( res, divValue )
end
table.insert( res, '</div>' )
table.insert( res, '<div style="clear:right;display:block"></div>' )
-- div = mw.text.tag( 'div', { style = string.format( notchStyleStr, y, width - 4, color ) }, '' )
-- table.insert( res, div )
end
end
 
local color, tooltip, custom = colors[gi] or defcolor or 'blue', tooltip( gi, i, ttval or val )
if scalePerGroup then
local left, barWidth = calcx( gi, i )
-- local colWidth = 80
local barHeight, top = calcHeights( gi, i, val )
-- local colStyle = "position:absolute;height:%spx;min-width:%spx;left:%spx;border-right:1px solid %s;color:%s"
-- local colStyle = "height:%spx;border-right:1px solid %s;color:%s;display:inline-block;text-align:right"
-- for gi = 1, numGroups do
-- -- local left = ( gi - 1 ) * colWidth
-- local color = colors[gi] or defcolor
-- -- table.insert( res, mw.text.tag( 'div', { style = string.format( colStyle, chartHeight, colWidth, left, color, color ) } ) )
-- table.insert( res, mw.text.tag( 'div', { style = string.format( colStyle, chartHeight, color, color ) } ) )
-- drawSingle( gi, color )
-- table.insert( res, '</div>' )
-- end
else
drawSingle( 1, 'black', true ) -- gi is the id of y axis when more than 1
end
end
 
-- borders so it shows up when printing
local style = string.format("position:absolute;left:%spx;top:%spx;height:%spx;min-width:%spx;max-width:%spx;background-color:%s;-webkit-print-color-adjust:exact;border:1px solid %s;border-bottom:none;overflow:hidden;",
left, top, barHeight-1, barWidth-2, barWidth-2, color, color)
local link = links[gi] and links[gi][i] or ''
local img = not nulOrWhitespace( link ) and string.format( '[[File:Transparent.png|1000px|link=%s|%s]]', link, custom and tooltip or '' ) or ''
table.insert( res, mw.text.tag( 'div', { style = style, title = tooltip, }, img ) )
end
 
 
local function drawYScale()
local function drawSingle( gi, color, width, yticks, single )
local yscale = yscales[gi]
local _, top, ordermag = roundup( yscale * 0.999 )
local numnotches = yticks >= 0 and yticks or
(top <= 1.5 and top * 4
or top < 4 and top * 2
or top)
local valStyleStr =
single and 'position:absolute;height=20px;text-align:right;vertical-align:middle;width:%spx;top:%spx;padding:0 2px'
or 'position:absolute;height=20px;text-align:right;vertical-align:middle;width:%spx;top:%spx;left:3px;background-color:%s;color:white;font-weight:bold;text-shadow:-1px -1px 0 #000,1px -1px 0 #000,-1px 1px 0 #000,1px 1px 0 #000;padding:0 2px'
local notchStyleStr = 'position:absolute;height=1px;min-width:5px;top:%spx;left:%spx;border:1px solid %s;'
for i = 1, numnotches do
local val = i / numnotches * yscale
local y = chartHeight - calcHeights( gi, 1, val )
local div = mw.text.tag( 'div', { style = string.format( valStyleStr, width - 10, y - 10, color ) }, mw.getContentLanguage():formatNum( tonumber( val ) or 0 ) )
table.insert( res, div )
div = mw.text.tag( 'div', { style = string.format( notchStyleStr, y, width - 4, color ) }, '' )
table.insert( res, div )
end
end
 
if scalePerGroup then
function drawXlegends()
local colWidth = 80
local setOffset, setWidth
local colStyle = "position:absolute;height:%spx;min-width:%spx;left:%spx;border-right:1px solid %s;color:%s"
local legendDivStyleFormat = "display:block;float:left;position:relative;vertical-align:top;width:%spx;text-align:center;margin:0px 0px 0px %spx;"
for gi = 1, numGroups do
--local tickDivstyleFormat = "display:block;float:left;position:relative;vertical-align:top;border-left:1px solid black;width:%spx;text-align:center;margin:0px 0px 0px %spx;"
local left = ( gi - 1 ) * colWidth
local offsetleft = 0;
local color = colors[gi] or defcolor
setOffset, setWidth = groupBounds( 1 )
table.insert( res, mw.text.tag( 'div', { style = string.format( colStyle, chartHeight, colWidth, left, color, color ) } ) )
for i = 1, numValues do
drawSingle( gi, color, colWidth, yticks )
if not nulOrWhitespace( xlegends[i] ) then
table.insert( res, '</div>' )
--table.insert( res, mw.text.tag( 'div', { style = string.format( tickDivStyleFormat, setWidth/2, offsetleft+setWidth/2 ) }, xlegends[i] or '' ) )
end
table.insert( res, mw.text.tag( 'div', { style = string.format( legendDivStyleFormat, setWidth, offsetleft ) }, xlegends[i] or '' ) )
else
offsetleft=0;
drawSingle( 1, 'black', scaleWidth, yticks, true )
else
end
offsetleft=offsetleft+setWidth;
end
end
-- setOffset, setWidth = groupBounds( i )
-- table.insert( res, mw.text.tag( 'div', { style = string.format( legendDivStyleFormat, setWidth ) }, xlegends[i] or '' ) )
end
end
 
local function drawXticksdrawXlegends()
local setOffset, setWidth
local tickDivStyleFormatlegendDivStyleFormat = "displayposition:blockabsolute;float:left;position:relative%spx;heighttop:5px10px;verticalmin-alignwidth:top%spx;bordermax-left:1px solid black;width:%spx;text-align:center;marginvertical-align:0px 0px 0px %spxtop;"
local tickDivstyleFormat = "position:absolute;left:%spx;height:10px;width:1px;border-left:1px solid black;"
local offsetleft = 0;
for i = 1, numValues do
setOffset, setWidth = groupBounds( 1 )
if not nulOrWhitespace( xlegends[i] ) then
for i = 1, numValues do
setOffset, setWidth = groupBounds( i )
if not nulOrWhitespace( xlegends[i] ) then
-- setWidth = 0.85 * setWidth
table.insert( res, mw.text.tag( 'div', { style = string.format( tickDivStyleFormat, math.floor(setWidth/2), math.floor(offsetleft+setWidth/2) ) }, '' ) )
table.insert( res, mw.text.tag( 'div', { style = string.format( legendDivStyleFormat, setOffset + 1, setWidth - 2, setWidth - 2 ) }, xlegends[i] or '' ) )
offsetleft=0;
table.insert( res, mw.text.tag( 'div', { style = string.format( tickDivstyleFormat, setOffset + setWidth / 2 ) }, '' ) )
else
end
offsetleft=offsetleft+setWidth;
end
end
end
end
 
local function drawChart()
table.insert( res, mw.text.tag( 'div', { class = 'chart noresize', style = string.format( 'padding-top:10px;margin-top:1em;max-width:%spx;', width ) } ) )
table.insert( res, mw.text.tag( 'div', { style = string.format("position:relative;min-height:%spx;min-width:%spx;max-width:%spx;", height, width, width ) } ) )
 
table.insert( res, mw.text.tag( 'div', { style = string.format("float:right;position:relative;min-height:%spx;min-width:%spx;max-width:%spx;border-left:1px black solid;border-bottom:1px black solid;", chartHeight, chartWidth, chartWidth ) } ) )
function drawChart()
local acum = stack and accumulateTooltip and {}
table.insert( res, mw.text.tag( 'div', { style = string.format("position:relative;padding:1em 0em 1em 0em;") } ) ) -- container div
for gi, group in pairs( values ) do
table.insert( res, mw.text.tag( 'div', { style = string.format("position:relative;" ) } ) ) -- container div
for i, val in ipairs( group ) do
table.insert( res, mw.text.tag( 'div', { style = string.format("position:relative;height:%spx;display:inline-block;text-align:right;vertical-align:top;", chartHeight ) } ) )
if acum then acum[i] = ( acum[i] or 0 ) + val end
drawYScale()
drawbar( gi, i, val, acum and acum[i] )
table.insert( res, '</div><div style="position:relative;display:inline-block">' )
end
-- table.insert( res, mw.text.tag( 'div', { style = string.format("position:relative;display:inline-block" ) } ) ) -- right side container div (should not have space)
end
table.insert( res, mw.text.tag( 'div', { style = string.format("height:%spx;width:%spx;border-left:1px black solid;border-bottom:1px black solid;display:block;margin:0px;padding:0px;", chartHeight, chartWidth ) } ) ) -- the actual chart
table.insert( res, '</div>' )
local acum = stack and accumulateTooltip and {}
table.insert( res, mw.text.tag( 'div', { style = string.format("position:absolute;height:%spx;min-width:%spx;max-width:%spx;", chartHeight, scaleWidth, scaleWidth, scaleWidth ) } ) )
for gi, group in pairs( values ) do
drawYScale()
for i, val in ipairs( group ) do
table.insert( res, '</div>' )
if acum then acum[i] = ( acum[i] or 0 ) + val end
table.insert( res, mw.text.tag( 'div', { style = string.format( "position:absolute;top:%spx;left:%spx;width:%spx;", chartHeight, scaleWidth, chartWidth ) } ) )
drawbar( gi, i, val, acum and acum[i] )
drawXlegends()
end
table.insert( res, '</div>' )
end
table.insert( res, '</div>' )
createGroupList( res, groupNames, colors )
table.insert( res, mw.text.tag( 'div', { style = string.format( "position:relative;width:%spx;float:left;", chartWidth ) } ) ) -- X ticks
table.insert( res, '</div>' )
drawXticks()
end
table.insert( res, '</div>' )
table.insert( res, mw.text.tag( 'div', { style = string.format( "position:relative;width:%spx;", chartWidth ) } ) ) -- X legends
drawXlegends()
table.insert( res, '</div>' )
table.insert( res, '</div>' )
table.insert( res, '</div>' )
createGroupList( res, groupNames, colors )
table.insert( res, '</div>' )
end
 
extractParams()
validate()
calcHeightLimits()
drawChart()
return table.concat( res, "\n" )
end
 
return {
['bar-chart'] = barChart,
[keywords.barChart] = barChart,
[keywords.pieChart] = pieChart,
}
--</source>