Module:EUPP seats/sandbox
![]() | This is the module sandbox page for Module:EUPP seats (diff). See also the companion subpage for test cases (run). |
![]() | Module rating is invalid or not specified. |
Description
[edit]This module provides:
- the number or share of seats occupied by European political parties in various European institutions (function
main
), as well as across lower and upper houses in EU member states, used by Template:EUPP seats; and - composition bars displaying the number of seats of a European political party in a given institution (function
composition_bar
), as well as across lower and upper houses in EU member states, used by Template:EUPP composition bar.
Data is obtained from Wikidata.
EU-level institutions covered are the European Parliament, the European Commission, the European Council, and the European Committee of the Regions.
Parties covered are all European political parties, meaning those duly registered with the Authority for European Political Parties and European Political Foundations.
The module seeks to replace and improve upon:
- Template:EUCouncilcountEuropeanParty, which focuses only on the European Council;
- Template:SeatCountInstitutionsInEurope, which only gives out the number of seats of EU institutions (not of parties therein); and
- Template:SeatsEUPPs (old version), which did not disaggregate seats by European party.
Usage
[edit]Acronyms
[edit]The following acronyms are used for institutions:
- European Parliament: EP
- European Commission: EC
- European Council: EUCO
- European Committee of the Regions: COR
- Lower houses in EU member states: ms-lower-house (alternatively, "lower-house" and "lower" also work)
- Upper houses in EU member states: ms-upper-house (alternatively, "upper-house" and "upper" also work)
The following acronyms are used for European parties:
- European People's Party: EPP
- Party of European Socialists: PES
- European Conservatives and Reformists: ECR
- Patriots.eu: Patriots
- Alliance of Liberals and Democrats for Europe Party: ALDE
- European Green Party: EGP
- Europe of Sovereign Nations: ESN
- European Left Alliance for the People and the Planet: ELA
- Party of the European Left: EL
- European Democratic Party: EDP
- European Free Alliance: EFA
- European Christian Political Movement: ECPM
In addition, the following European political alliances are covered:
- Animal Politics EU: Animal
- European Communist Action: ECA
- European Pirate Party: PPEU
- Democracy in Europe Movement 2025: DiEM25
- Volt Europa: Volt
In addition, four special parameters can be used instead of parties:
- all: for the seats of all European parties combined;
- none: for the seats not occupied by members of European parties (not for lower and upper houses);
- ind: for independent politicians on the European Council (only); and
- thisparty: to be used instead of a specific party name when the module is called from the page of a European political party.
In order to return the share of seats instead of their number, "%" or "share" must be added as a third parameter.
Notes:
- the capitalisation of parameters does not matter;
- given the cost of calling Wikidata when using a specific eID (i.e., not calling Wikidata from a linked Wikipedia page), the special parameter "thisparty" should be used whenever this module is called from the page of a European party; and
- since the listed European political alliances are not European political parties, they do not count towards to sum of seats occupied by European political parties, nor the seats not occupied by European political parties.
Function main
[edit]Structure
[edit]The structure of the function is as follows:
{{#invoke:EUPP seats|main|<institution>|<party or special parameter>|<share>}}
- the first argument is mandatory is must be a recognised institution from the list above;
- the second argument is optional and can be a recognised European party or a special parameter from the list above; and
- the third argument is option and returns the share, instead of the absolute number.
Calls are structured as follows:
Code | Output |
---|---|
{{#invoke:EUPP seats|main|institution}}
|
total number of seats of the institution given in argument |
{{#invoke:EUPP seats|main|institution|party}}
|
number of seats of the European party in the institution given in argument |
{{#invoke:EUPP seats|main|institution|all}}
|
number of seats of all European parties in the institution given in argument |
{{#invoke:EUPP seats|main|institution|none}}
|
number of seats not occupied by European parties in the institution given in argument |
{{#invoke:EUPP seats|main|EUCO|ind}}
|
number of independent politicians in the European Council (only applicable to the European Council) |
{{#invoke:EUPP seats|main|institution|thisparty<}}
|
number of seats of the European party from which page the module is called in the institution given in argument |
{{#invoke:EUPP seats|main|institution|party|%}}
|
share of seats of the European party in the institution given in argument -- this works mutatis mutandis with the four special parameters |
Note: "%" and "share" can be used interchangeably.
Examples
[edit]Code | Result | Output |
---|---|---|
{{#invoke:EUPP seats|main|EP}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | total number of seats of the European Parliament |
{{#invoke:EUPP seats|main|EC|EPP}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | number of seats of the EPP in the European Commission |
{{#invoke:EUPP seats|main|EUCO|PES|%}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | share of seats of the PES in the European Council |
{{#invoke:EUPP seats|main|EP|all}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | total number of seats occupied by European parties in the European Parliament |
{{#invoke:EUPP seats|main|EC|all|share}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | total share of seats occupied by European parties in the European Commission |
{{#invoke:EUPP seats|main|COR|none}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | number of seats not occupied by European parties in the European Committee of the Regions |
{{#invoke:EUPP seats|main|COR|none|%}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | share of seats not occupied by European parties in the European Committee of the Regions |
{{#invoke:EUPP seats|main|EUCO|IND}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | number of seats occupied by independent Heads of State or Government in the European Council |
{{#invoke:EUPP seats|main|ms-lower-house|ALDE}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | number of seats occupied by national member parties of ALDE in lower houses of EU member states |
{{#invoke:EUPP seats|main|EP|thisparty}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | number of seats occupied by in the European Parliament by the European party from which page the module is called (here, the module adequately gives out an error message) |
Function composition_bar
[edit]Additional parameters
[edit]For this function, the module allows for additional parameters:
- width: the size of the composition bar;
- percent: option to display the percentage;
- reference: option to display a trailing reference from Wikidata;
- bar-color: option to override the default colour (either the official colour of the European party, or grey for
all
,none
, andind
); - background-color: colour of the background of the composition bar; and
- border: colour of the border of the composition bar.
Structure
[edit]The structure of the call is as follows:
{{#invoke:EUPP seats|composition_bar|<institution>|<party>|width=<width>|percent=yes|reference=yes|bar-color=<color>|background-color=<color>|border=<color>}}
- the first argument is mandatory and must always be an institution, as listed above;
- the second argument is mandatory and always be a European party or special parameter, as listed above; and
- the remaining arguments are optional, have no specific order, and are referred to using their name:
width
can have the following formats: "80" (default unit is pixel), "80px", "80em", or "80%" (without spaces);percent
will display the percentage for "yes" (regardless of capitalisation), and ignore all other values;reference
will display the reference for "yes" (regardless of capitalisation), and ignore all other values; andbar-color
,background-color
, andborder
will override default colours when provided an hexadecimal value (e.g."#123ABC", including the pound sign) or a recognised colour (e.g. "red").
Code | Output |
---|---|
{{#invoke:EUPP seats|composition_bar|institution}}
|
composition bar of total number of seats of the institution given in argument |
{{#invoke:EUPP seats|composition_bar|institution|party}}
|
composition bar of the seats of a European party in the institution given in argument |
{{#invoke:EUPP seats|composition_bar|institution|all}}
|
composition bar of the seats of all European parties in the institution given in argument |
{{#invoke:EUPP seats|composition_bar|institution|none}}
|
composition bar of the seats not occupied by European parties in the institution given in argument |
{{#invoke:EUPP seats|composition_bar|EUCO|ind}}
|
composition bar of the seats of independent politicians in the European Council (only applicable to the European Council) |
{{#invoke:EUPP seats|composition_bar|institution|thisparty<}}
|
composition bar of the seats of the European party from which page the module is called in the institution given in argument |
Examples
[edit]Code | Result | Output |
---|---|---|
{{#invoke:EUPP seats|composition_bar|EC|EPP}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar for the number of seats of the EPP in the European Commission |
{{#invoke:EUPP seats|composition_bar|EP|all}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar for the total number of seats occupied by European parties in the European Parliament |
{{#invoke:EUPP seats|composition_bar|COR|none}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar for the number of seats not occupied by European parties in the European Committee of the Regions |
{{#invoke:EUPP seats|composition_bar|EUCO|IND}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar for the number of seats occupied by independent Heads of State or Government in the European Council |
{{#invoke:EUPP seats|composition_bar|ms-lower-house|ALDE}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar for the number of seats occupied by national member parties of ALDE in lower houses of EU member states |
{{#invoke:EUPP seats|composition_bar|EC|EPP|width=50%}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar with a size of 50% |
{{#invoke:EUPP seats|composition_bar|EP|PES|width=50%|percent=yes}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar with display of the share |
{{#invoke:EUPP seats|composition_bar|EP|EPP|width=4em|percent=yes|reference=yes}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar with display of the reference |
{{#invoke:EUPP seats|composition_bar|EP|EPP|bar-colour=red}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar with overriding bar colour |
{{#invoke:EUPP seats|composition_bar|EP|EPP|background-color=#334477}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar with background colour |
{{#invoke:EUPP seats|composition_bar|EP|EPP|background-color=#334477|border=green}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar with background colour and border |
{{#invoke:EUPP seats|composition_bar|EC|thisparty}}
|
Lua error in Module:EUPP_seats/config at line 151: table index is nil. | Composition bar for the number of seats occupied by in the European Parliament by the European party from which page the module is called (here, the module adequately gives out an error message) |
require ('strict');
local get_args = require ('Module:Arguments').getArgs; -- function to fetch frame and parent frame arguments
local cfg = mw.loadData ('Module:EUPP seats/config'); -- defines, configuration data, and i18n support
local namespace = mw.title.getCurrentTitle().namespace; -- used for categorization
--[[--------------------------< S U B S T I T U T E >----------------------------------------------------------
Substitutes $1, $2, etc in <message> with data from <data_t>. Returns plain-text substituted string when
<data_t> not nil; returns <message> else.
]]
local function substitute (message, data_t)
return data_t and mw.message.newRawMessage (message, data_t):plain() or message;
end
--[[--------------------------< M A K E _ E R R O R _ M S G >--------------------------------------------------
Assembles an error message from template name, message text, help link, and error category.
]]
local function make_error_msg (msg, template_name, nocat)
local category;
local category_link = ((0 == namespace) and not nocat) and substitute ('[[Category:$1]]', {cfg.settings_t.err_category}) or '';
return substitute ('<span style="color:#d33">Error: {{$1}}: $2 ([[:Template:$1|$3]])</span>$4',
{
template_name or cfg.settings_t.template_name, -- the template name without namespace
msg, -- the error message
cfg.settings_t.help, -- help wikilink display text
category_link -- link to error category (main namespace only)
})
end
--[[--------------------------< R O U N D >--------------------------------------------------------------------
return the rounded value of the arguments with two digits
]]
local function round (n)
return math.floor(100 * n + 0.5) / 100 -- round argument to two decimals
end
--[[--------------------------< S U M >------------------------------------------------------------------------
return the sum of seats for all parties of <institution> listed in <cfg.parties_t> (so not including lower and upper houses). <body_prop> is the wikidata
property:
P194: legislative body
P208: executive body
<frame> required to expand {{wikidata}} template
Note: P1410 is the property "number of seats in assembly", used to record an entity's seats in legislative or executive bodies
]]
local function sum (frame, institution, body_prop)
local sum = 0; -- init
local args_t = {[1]='property', [3]='P1410'}; -- init some of the {{wikidata}} parameters
args_t[body_prop] = cfg.institutions_t[institution];
for _, qid in pairs (cfg.parties_t) do -- loop through all parties in <cfg.parties_t>
args_t[2] = qid; -- set the last {{wikidata}} parameter
sum = sum + frame:expandTemplate ({title='wikidata', args = args_t}); -- expand and tally
end
return sum;
end
--[[--------------------------< G E T _ C O L O U R >----------------------------------------------------------
return the hex colour of a European party with '#' prefix
<frame> required to expand {{wikidata}} template
]]
local function get_colour (frame, party)
local args_t = {'property'}; -- build an arguments table for expandTemplate()
if party ~= "THISPARTY" then -- when not on a party page
if cfg.parties_t[party] then
table.insert (args_t, cfg.parties_t[party]); -- must name the party
elseif cfg.alliances_t[party] then
table.insert (args_t, cfg.alliances_t[party]); -- must name the alliance
end
end
table.insert (args_t, 'P465'); -- last argument is P465; sRGB color hex triplet property
local color = frame:expandTemplate ({title='wikidata', args = args_t}); -- get the color
if '' == color then -- if no color
color = 'BBB'; -- use a default color; some sort of gray
end
return '#' .. color; -- add the '#' prefix
end
--[[--------------------------< G E T _ R E F >---------------------------------------------------------------
return the reference for a seat claim relating to <institution> listed in <cfg.parties_t> or in <cfg.alliances_t> (so not including lower and upper houses)
<frame> required to expand {{wikidata}} template
]]
local function get_ref (frame, institution, party)
if institution == 'EP' or institution == 'COR' then
if party == "THISPARTY" then
return frame:expandTemplate ({title='wikidata', args = {'references', 'P1410', P194 = cfg.institutions_t[institution]}});
elseif cfg.parties_t[party] then
return frame:expandTemplate ({title='wikidata', args = {'references', cfg.parties_t[party], 'P1410', P194 = cfg.institutions_t[institution]}});
elseif cfg.alliances_t[party] then
return frame:expandTemplate ({title='wikidata', args = {'references', cfg.alliances_t[party], 'P1410', P194 = cfg.institutions_t[institution]}});
end
elseif institution == 'EP' or institution == 'COR' then
if party == "THISPARTY" then
return frame:expandTemplate ({title='wikidata', args = {'references', 'P1410', P208 = cfg.institutions_t[institution]}});
elseif parties_t[party] then
return frame:expandTemplate ({title='wikidata', args = {'references', cfg.parties_t[party], 'P1410', P208 = cfg.institutions_t[institution]}});
elseif alliances_t[party] then
return frame:expandTemplate ({title='wikidata', args = {'references', cfg.alliances_t[party], 'P1410', P208 = cfg.institutions_t[institution]}});
end
end
end
--[[--------------------------< S I N G L E >------------------------------------------------------------------
return the number of seats occupied by one party in an <institution> listed in <cfg.parties_t> or in <cfg.alliances_t> (so not including lower and upper houses). <body_prop> is the wikidata property:
P194: legislative body
P208: executive body
<frame> required to expand {{wikidata}} template
Note: P1410 is the property "number of seats in assembly", used to record an entity's seats in legislative or executive bodies
]]
local function single (frame, institution, party, body_prop)
local args_t = {};
if party == "THISPARTY" then -- flag used when module is called from the page of a European party; less expensive
args_t = {'property', 'P1410'}; -- init some of the {{wikidata}} parameters with THISPARTY (only when called from the page of a European party)
elseif cfg.alliances_t[party] then
args_t = {'property', cfg.alliances_t[party], 'P1410'}; -- init some of the {{wikidata}} parameters
else
args_t = {'property', cfg.parties_t[party], 'P1410'}; -- init some of the {{wikidata}} parameters
end
args_t[body_prop] = cfg.institutions_t[institution];
local retval = frame:expandTemplate ({title='wikidata', args = args_t})
if '' == retval then -- {{wikidata}} returns empty string when <party> not known to <institution>
if party == "THISPARTY" then -- specific error message if the module was called with THISPARTY from the wrong page
return make_error_msg (cfg.error_messages_t.thisparty);
elseif not party then
return make_error_msg (substitute (cfg.error_messages_t.party_req_share));
else
return make_error_msg (substitute (cfg.error_messages_t.inst_unknown_party, {institution, party}));
end
end
return retval;
end
--[[--------------------------< S H A R E _ F C >--------------------------------------------------------------
return the share of a party's seats relative to the total size of a given institution listed in <cfg.parties_t> or <cfg.alliances_t> (so not including lower and upper houses)
<frame> required to expand {{wikidata}} template
Note: P1342 is the property "number of seats", used to record an institution's number of seats
]]
local function share_fc (frame, party_seats, institution)
return tonumber (party_seats) and round (100 * party_seats / frame:expandTemplate ({title='wikidata', args = {'property', cfg.institutions_t[institution], 'P1342'}})) or party_seats;
end
--[[--------------------------< S E A T S >--------------------------------------------------------------------
return a number of seats either for an institution or occupied by one or more parties, or by none of them, in an <institution> listed in <cfg.parties_t> or <cfg.alliances_t> (so not including lower and upper houses).
<frame> required to expand {{wikidata}} template
Note:
* P1410 is the property "number of seats in assembly", used to record an entity's seats in legislative or executive bodies
* P1342 is the property "number of seats", used to record an institution's number of seats
* P208 is the property "executive body"
* P194 is the property "legislative body"
]]
local function seats (frame, institution, party)
if not party then -- party not specified, returns seats of the institution
return frame:expandTemplate ({title='wikidata', args = {'property', cfg.institutions_t[institution], 'P1342'}});
elseif party == "IND" and institution == "EUCO" then -- special case of independent politicians on European Council
return frame:expandTemplate ({title='wikidata', args = {'property', cfg.misc_parties_t['IND'], 'P1410', P208 = cfg.institutions_t[institution]}});
elseif party == "NONE" then -- returns seats not occupied by European parties
local retval = frame:expandTemplate ({title='wikidata', args = {'property', cfg.institutions_t[institution], 'P1342'}}); -- get number of seats in the institution
if institution == "EUCO" then -- if EUCO, use P208 and separate case to account for independent politicians
local ind = frame:expandTemplate ({title='wikidata', args = {'property', cfg.misc_parties_t['IND'], 'P1410', P208 = cfg.institutions_t[institution]}});
return retval - (sum (frame, institution, cfg.body_prop_t[institution]) + ind);
else -- COR, EC, EP
return retval - sum (frame, institution, cfg.body_prop_t[institution]);
end
elseif party == "ALL" then -- returns seats occupied by all European parties combined
return sum (frame, institution, cfg.body_prop_t[institution]);
else -- returns the number of seats occupied by one party in one institution
return single (frame, institution, party, cfg.body_prop_t[institution]);
end
end
--[[--------------------------< S E A T S _ S H A R E >--------------------------------------------------------
return a share of seats occupied by one or more parties, or by none of them, in an <institution> listed in <cfg.parties_t> or <cfg.alliances_t> (so not including lower and upper houses).
<frame> required to expand {{wikidata}} template
Note:
* P1410 is the property "number of seats in assembly", used to record an entity's seats in legislative or executive bodies
* P1342 is the property "number of seats", used to record an institution's number of seats
* P208 is the property "executive body"
* P194 is the property "legislative body"
]]
local function seats_share (frame, institution, party)
if party == "IND" and institution == "EUCO" then -- special case of independent politicians on European Council
return share_fc (frame, frame:expandTemplate ({title='wikidata', args = {'property', cfg.misc_parties_t['IND'], 'P1410', P208 = cfg.institutions_t[institution]}}), institution);
elseif party == "NONE" then -- returns seats not occupied by European parties
local retval = frame:expandTemplate ({title='wikidata', args = {'property', cfg.institutions_t[institution], 'P1342'}});
if institution == "EUCO" then -- if EUCO, use P208 and separate case to account for independent politicians
local ind = frame:expandTemplate ({title='wikidata', args = {'property', cfg.misc_parties_t['IND'], 'P1410', P208 = cfg.institutions_t[institution]}})
return share_fc (frame, retval - (sum (frame, institution, cfg.body_prop_t[institution]) + ind), institution);
else -- for COR, EC, EP
return share_fc (frame, retval - sum (frame, institution, cfg.body_prop_t[institution]), institution);
end
elseif party == "ALL" then -- returns seats occupied by all European parties combined
return share_fc (frame, sum (frame, institution, cfg.body_prop_t[institution]), institution);
else -- returns the number of seats occupied by one party in one institution
return share_fc (frame, single (frame, institution, party, cfg.body_prop_t[institution]), institution);
end
end
--[[--------------------------< V A L I D A T E _ W I D T H >--------------------------------------------------
validates data format for width parameter (for composition bar()); returns boolean true when valid; nil else
]]
local function validate_width (width)
local patterns_t = {'^%d+$', '^%d+px$', '^%d+%%$', '^%d+em$'}; -- valid <width> patterns
for i, pattern in ipairs (patterns_t) do -- loop through the patterns in <patterns_t>
if width:match (pattern) then -- is there a match?
return true; -- yes, done
end
end
end
--[[--------------------------< V A L I D A T E _ I N S T I T U T I O N _ P A R T Y >--------------------------
validates data format for institution and party parameters, for main() and compositionbar() for institutions listed in <cfg.parties_t> or <cfg.alliances_t> (so not including lower and upper houses)
returns boolean true when valid; error message else
]]
local function validate_institution_party (institution, party, template_name)
if not institution then -- institution is required
return make_error_msg (cfg.error_messages_t.missing_inst, template_name);
elseif not cfg.institutions_t[institution] then
return make_error_msg (substitute (cfg.error_messages_t.unknown_inst, {institution}), template_name); -- if institution is present, it must be known
end
if party == "%" or party == "SHARE" then -- if party is missing and %/share is entered instead
return make_error_msg (cfg.error_messages_t.party_req_share, template_name);
end
if party and not cfg.parties_t[party] and not cfg.alliances_t[party] and not cfg.misc_parties_t[party] and not cfg.keywords_t[party] then -- party is optional; but if party is present, it must be known
return make_error_msg (substitute (cfg.error_messages_t.unknown_party, {party}), template_name);
end
if 'THISPARTY' == party then -- 'THISPARTY' parameter only to be used by templates on European party articles
local frame = mw.getCurrentFrame(); -- we need a copy of the frame for this test
if '' == frame:expandTemplate ({title='wikidata', args = {'property', 'P1410', cfg.institutions_t[institution]}}) then -- empty string when this article not an EU party article
return make_error_msg (cfg.error_messages_t.thisparty , template_name);
end
end
return true;
end
--[[--------------------------< S T R I P _ H O U S E _ T Y P E >----------------------------------------------
strips down parts of the house type (for calls relating to lower and upper houses)
]]
local function strip_house_type (house_type)
if house_type == "LOWER-HOUSE" or house_type == "LOWER" or house_type == "MS-LOWER-HOUSE"then -- three accepted formats for input
return "LOWER-HOUSE"; -- format actually used in the code
elseif house_type == "UPPER-HOUSE" or house_type == "UPPER" or house_type == "MS-UPPER-HOUSE" then -- three accepted formats for input
return "UPPER-HOUSE"; -- format actually used in the code
else
return house_type;
end
end
--[[--------------------------< V A L I D A T E _ P A R T Y _ H O U S E _ T Y P E >----------------------------
validates data format for party name (for main() and compositionbar() for calls relating to lower and upper houses).
returns boolean true when valid; error message else
]]
local function validate_party_house_type (house_type, european_party, template_name)
if european_party and not cfg.parties_t[european_party] and not cfg.alliances_t[european_party] and not cfg.keywords_lower_upper_t[european_party] then -- party is optional; but if party is present, it must be known
return make_error_msg (substitute (cfg.error_messages_t.unknown_party, {european_party}), template_name);
end
if 'THISPARTY' == european_party then -- 'THISPARTY' parameter only to be used by templates on European party articles
local this_page_qid = mw.wikibase.getEntityIdForCurrentPage(); -- looking up qID of the page from which the call is made
if cfg.rev_parties_t[this_page_qid] then
return true; -- true if the <this_page_qid> matches a European party in the config file
end
return make_error_msg (cfg.error_messages_t.thisparty , template_name);
end
return true;
end
--[[--------------------------< G E T _ H O U S E _ S E A T S >------------------------------------------------
returns the number of seats in a given lower or upper house (identified by a row in the master table) from
wikidata. When <house_type> not recognized or when <ms_data_t> does not have 'that' house, returns 0
]]
local function get_house_seats (frame, row, house_type)
local house_qid;
if house_type == "LOWER-HOUSE" then
house_qid = cfg.ms_data_t[row].lower_house_qid; -- get the lower house qid
elseif house_type == "UPPER-HOUSE" then
house_qid = cfg.ms_data_t[row].upper_house_qid; -- get the upper house qid
end
return house_qid and frame:expandTemplate ({title='wikidata', args = {'property', house_qid, 'P1342'}}) or 0; -- house_qid is nil when no upper or lower house
end
--[[--------------------------< S U M _ H O U S E _ S E A T S >------------------------------------------------
returns the total number of seats of all lower or upper houses in all member states.
]]
local function sum_house_seats (frame, house_type)
local sum_seats = 0; -- init sum of seats of European party's member parties in house_type
for row, _ in ipairs (cfg.ms_data_t) do -- for all member states
sum_seats = sum_seats + get_house_seats (frame, row, house_type); -- increase sum_seats
end
return sum_seats;
end
--[[--------------------------< G E T _ N A T I O N A L _ P A R T Y _ S E A T S >------------------------------
returns the number of seats occupied by a given party (identified by a given row in the master table) in the
lower or upper house of its member state from wikidata. When <house_type> not recognized or when <tab_data_t>
does not have 'that' house, returns 0
]]
local function get_national_party_seats (frame, row, house_type)
local house_qid = "";
local national_party_qid = cfg.tab_data_t[row].national_party_qid;
if house_type == "LOWER-HOUSE" then
house_qid = cfg.tab_data_t[row].lower_house_qid;
elseif house_type == "UPPER-HOUSE" then
house_qid = cfg.tab_data_t[row].upper_house_qid;
end
return house_qid and frame:expandTemplate ({title='wikidata', args = {'property', national_party_qid, 'P1410', P194 = house_qid}}) or 0;
end
--[[--------------------------< S U M _ N A T I O N A L _ P A R T Y _ S E A T S >------------------------------
returns the sum of seats occupied by all national parties members of a given European party in the lower or upper house of its member state.
]]
local function sum_national_party_seats (frame, european_party, house_type)
local sum_seats = 0; -- init sum of seats of European party's member parties in house_type
if not european_party then -- if no European party is listed, then returns the total number of seats of all lower or upper houses
sum_seats = sum_house_seats(frame, house_type);
else
for row, _ in ipairs (cfg.tab_data_t) do
if european_party == "ALL" then -- sum for all rows
sum_seats = sum_seats + get_national_party_seats (frame, row, house_type); -- increase sum_seats
elseif european_party == "THISPARTY" then -- if called from the page of a European party
local thisparty_qid = mw.wikibase.getEntityIdForCurrentPage(); -- get party qiD
local thisparty_name = cfg.rev_parties_t[thisparty_qid];
if thisparty_name == cfg.tab_data_t[row]['european_party'] then
sum_seats = sum_seats + get_national_party_seats (frame, row, house_type); -- increase sum_seats
end
else
if european_party == cfg.tab_data_t[row]['european_party'] then
sum_seats = sum_seats + get_national_party_seats (frame, row, house_type); -- increase sum_seats
end
end
end
end
return sum_seats;
end
--[[--------------------------< M A I N >----------------------------------------------------------------------
implements {{EUPP seats}}
carries out input error detection, reporting, and function dispatching
]]
local function main (frame)
local args_t = get_args (frame); -- get arguments; empty string or whitespace positional parameters set to nil
local institution = args_t[1] and args_t[1]:upper(); -- force to upper case
local party = args_t[2] and args_t[2]:upper();
local share = args_t[3] and args_t[3]:upper();
--[=[ harmonise institution name (in case of lower/upper house) ]=]
institution = strip_house_type (institution); -- "ms-lower-house" and "lower" are turned to "lower-house" (same for upper house)
--[=[ data validation for institution and party ]=]
local is_valid = false;
if institution == "LOWER-HOUSE" or institution == "UPPER-HOUSE" then
is_valid = validate_party_house_type (institution, party, 'EU party seats lower upper houses');
else
is_valid = validate_institution_party (institution, party, 'EUPP seats');
end
if true ~= is_valid then -- boolean true when valid; error message else
return is_valid; -- yep, abandon with error message
end
--[=[ function dispatching ]=]
if institution == "LOWER-HOUSE" or institution == "UPPER-HOUSE" then
if not share then
return sum_national_party_seats (frame, party, institution) -- return number of seats by calling sum_national_party_seats()
elseif share == "%" or share == "SHARE" then
return round (100 * sum_national_party_seats (frame, party, institution) / sum_house_seats (frame, institution)); -- return share of seats by calling seats_share()
else
return make_error_msg (substitute (cfg.error_messages_t.unknown_param, {share}));
end
else
if not share then
return seats (frame, institution, party); -- return number of seats by calling seats()
elseif share == "%" or share == "SHARE" then
return seats_share (frame, institution, party); -- return share of seats by calling seats_share()
else
return make_error_msg (substitute (cfg.error_messages_t.unknown_param, {share}));
end
end
end
--[[--------------------------< C O M P O S I T I O N _ B A R >------------------------------------------------
this function does whatever it is that {{composition bar}} does
implements {{EUPP composition bar}}
{{EUPP composition bar|<institution>|<party>|width=<width>|percent=yes|reference=yes|bar-color=<color>|background-color=<color>|border=<color>}}
]]
local function composition_bar (frame)
local args_t = get_args (frame); -- get arguments; empty string or whitespace positional parameters set to nil
local institution = args_t[1] and args_t[1]:upper(); -- force to upper case
local party = args_t[2] and args_t[2]:upper();
local width = args_t.width; -- must be a number, or number with unit suffix: 'px', '%', 'em'; whitespace not allowed
local percentage = args_t.percent and args_t.percent:lower(); --
percentage = 'yes' == percentage; -- make a boolean
local reference = args_t.reference and args_t.reference:lower();
reference = 'yes' == reference; -- make a boolean
local background_color = args_t['background-color'];
local border = args_t.border;
--[=[ harmonise institution name (in case of lower/upper house) ]=]
institution = strip_house_type (institution); -- "ms-lower-house" and "lower" are turned to "lower-house" (same for upper house)
--[=[ data validation for institution, party and width ]=]
local is_valid = false;
if institution == "LOWER-HOUSE" or institution == "UPPER-HOUSE" then
is_valid = validate_party_house_type (institution, party, 'EU party seats lower upper houses');
else
is_valid = validate_institution_party (institution, party, 'EUPP composition bar');
end
if true ~= is_valid then -- boolean true when valid; error message else
return is_valid; -- yep, abandon with error message
end
if width and not validate_width (width) then
return make_error_msg (substitute (cfg.error_messages_t.parameter_invalid, {width}), 'EUPP composition bar'); -- yep, abandon with error message
end
--[=[ prepare arguments for composition bar ]=]
if institution == "LOWER-HOUSE" or institution == "UPPER-HOUSE" then
local sum_house_seats = sum_house_seats(frame, institution); -- get total seats of lower or upper houses
local sum_national_party_seats = sum_national_party_seats(frame, party, institution) -- get sum of seats occupied by members of a European party
local color = args_t['bar-color'] or get_colour (frame, party); -- get color associated with <party>; |bar-color= overrides wikidata
local comp_bar_args_t = {
sum_national_party_seats,
sum_house_seats,
color,
width=width,
per=percentage,
['background-color'] = background_color,
border = border,
}
return frame:expandTemplate ({title='Composition bar', args = comp_bar_args_t});
else
local inst_seats = seats (frame, institution); -- get total seats in <institution>
local party_seats = seats (frame, institution, party); -- get total seats in <institution> occupied by <party>
local color = args_t['bar-color'] or get_colour (frame, party); -- get color associated with <party>; |bar-color= overrides wikidata
local comp_bar_args_t = {
party_seats,
inst_seats,
color,
width=width,
per=percentage,
['background-color'] = background_color,
border = border,
}
return frame:expandTemplate ({title='Composition bar', args = comp_bar_args_t}) .. ((reference and get_ref (frame, institution, party)) or '');
end
end
--[[--------------------------< T E S T >----------------------------------------------------------------------
]]
local function test (frame) -- to test calls and functions, for verification purposes
local args_t = get_args (frame); -- get arguments; empty string or whitespace positional parameters set to nil
local institution = args_t[1] and args_t[1]:upper(); -- force to upper case
local party = args_t[2] and args_t[2]:upper();
local share = args_t[3] and args_t[3]:upper();
institution = strip_house_type (institution); -- here, testing strip_house_type
return institution;
end
--[[--------------------------< E X P O R T S >----------------------------------------------------------------
]]
return {
main = main,
composition_bar = composition_bar,
test = test,
}