Source code for election_day.utils.parties.parties

from decimal import Decimal
from onegov.election_day import _
from operator import itemgetter


from typing import cast
from typing import Any
from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Sequence
    from onegov.core.types import JSONObject_ro
    from onegov.election_day.models import Election
    from onegov.election_day.models import ElectionCompound
    from onegov.election_day.models import ElectionCompoundPart
    from onegov.election_day.models import ProporzElection
    from onegov.election_day.request import ElectionDayRequest


[docs] def get_party_results( item: 'Election | ElectionCompound | ElectionCompoundPart', json_serializable: bool = False ) -> tuple[list[str], dict[str, Any]]: """ Returns the aggregated party results as list. """ if not getattr(item, 'has_party_results', False): return [], {} item = cast( 'ProporzElection | ElectionCompound | ElectionCompoundPart', item ) party_results = ( item.historical_party_results if item.use_historical_party_results else item.party_results ) results = [r for r in party_results if r.domain == item.domain] domain_segment = getattr(item, 'segment', None) if domain_segment: results = [ r for r in party_results if r.domain_segment == domain_segment ] # Get the totals votes per year totals_votes = {r.year: r.total_votes for r in results} years = sorted(str(key) for key in totals_votes.keys()) # Get the results parties: dict[str, Any] = {} colors = item.historical_colors for result in results: party = parties.setdefault(result.party_id, {}) year = party.setdefault(str(result.year), {}) year['color'] = colors.get(result.name or '') year['mandates'] = result.number_of_mandates year['name'] = result.name votes = result.votes or 0 total_votes = totals_votes.get(result.year) or 0 votes_permille = 0 if total_votes: votes_permille = int(round(1000 * (votes / total_votes))) year['votes'] = { 'total': votes, 'permille': votes_permille } voters_count: Decimal | float = result.voters_count or Decimal(0) if not item.exact_voters_counts: voters_count = int(round(voters_count)) elif json_serializable: voters_count = float(voters_count) voters_count_permille: Decimal | float voters_count_permille = result.voters_count_percentage or Decimal(0) voters_count_permille = 10 * voters_count_permille if json_serializable: voters_count_permille = float(voters_count_permille) year['voters_count'] = { 'total': voters_count, 'permille': voters_count_permille } return years, parties
[docs] def get_party_results_deltas( item: 'Election | ElectionCompound | ElectionCompoundPart', years: 'Sequence[str]', parties: dict[str, Any] ) -> tuple[bool, dict[str, list[list[str]]]]: """ Returns the aggregated party results with the differences to the last elections. """ attribute = 'voters_count' if item.voters_counts else 'votes' deltas = len(years) > 1 results: dict[str, list[list[str]]] = {} results_this_year: list[list[str]] for index, year in enumerate(years): results[year] = results_this_year = [] for key in sorted(parties.keys()): result = ['', '', '', ''] party = parties[key] values = party.get(year) if not values: continue permille = values.get(attribute, {}).get('permille') result = [ values.get('name', ''), values.get('mandates', ''), values.get(attribute, {}).get('total', ''), f'{permille / 10}%' if permille else '' ] if deltas: delta = '' if index: last = party.get(years[index - 1]) if values and last: diff = ( (values.get(attribute, {}).get('permille', 0) or 0) - (last.get(attribute, {}).get('permille', 0) or 0) ) / 10 delta = f'{diff}%' result.append(delta) results_this_year.append(result) return deltas, results
[docs] def get_party_results_data( item: 'Election | ElectionCompound | ElectionCompoundPart', horizontal: bool ) -> 'JSONObject_ro': """ Retuns the data used for the diagrams showing the party results. """ if horizontal: return get_party_results_horizontal_data(item) return get_party_results_vertical_data(item)
[docs] def get_party_results_horizontal_data( item: 'Election | ElectionCompound | ElectionCompoundPart' ) -> 'JSONObject_ro': """ Retuns the data used for the horitzonal bar diagram showing the party results. """ if not getattr(item, 'has_party_results', False): return { 'results': [], } attribute = 'voters_count' if item.voters_counts else 'votes' years, parties = get_party_results(item) allocated_mandates = item.allocated_mandates party_names: dict[str, str | None] = {} for party_id, party in parties.items(): party_names[party_id] = None for year in party.values(): party_names[party_id] = party_names[party_id] or year.get('name') results = [] if years: year = years[-1] party_ids = sorted( (values.get(year, {}).get(attribute, {}).get('total', 0), party_id) for party_id, values in parties.items() ) for __, party_id in reversed(party_ids): for year in reversed(years): active = year == years[-1] party = parties.get(party_id, {}).get(year, {}) name = party_names.get(party_id) if len(years) == 1: text = name value = round( party.get(attribute, {}).get('total', 0) or 0 ) percentage = False else: text = f'{name} {year}' if active else year value = float( (party.get(attribute, {}).get('permille', 0) or 0) / 10 ) percentage = True results.append({ 'text': text, 'value': value, 'value2': party.get('mandates'), 'class': ( 'active' if active and ( party.get('mandates') or not allocated_mandates ) else 'inactive' ), 'color': party.get('color'), 'percentage': percentage }) return {'results': results}
[docs] def get_party_results_vertical_data( item: 'Election | ElectionCompound | ElectionCompoundPart' ) -> 'JSONObject_ro': """ Retuns the data used for the grouped bar diagram showing the party results. """ if not getattr(item, 'has_party_results', False): return { 'results': [], 'title': item.title } active_year = str(item.date.year) attribute = 'voters_count' if item.voters_counts else 'votes' years, parties = get_party_results(item) groups: dict[str, str | None] = {} results = [] for party_id, results_per_year in parties.items(): for year, party in sorted( results_per_year.items(), key=itemgetter(0), reverse=True ): group = groups.setdefault( party_id, party.get('name', party_id) ) front = party.get('mandates', 0) back = float(party.get(attribute, {}).get('permille', 0) / 10) color = party.get('color', '#999999') results.append({ 'group': group, 'item': year, 'value': { 'front': front, 'back': back, }, 'active': year == active_year, 'color': color }) return { 'groups': [name for _, name in sorted(groups.items())], 'labels': years, 'maximum': { 'front': item.number_of_mandates, 'back': 100, }, 'axis_units': { 'front': '', 'back': '%' }, 'results': results, 'title': item.title }
[docs] def get_party_results_seat_allocation( years: 'Sequence[str]', parties: dict[str, Any] ) -> list[list[Any]]: """ Returns the aggregated party results for the seat allocation table. """ if not years: return [] party_names: dict[str, str | None] = {} for party_id, party in parties.items(): party_names[party_id] = None for year in party.values(): party_names[party_id] = party_names[party_id] or year.get('name') result = [] for party_id, party in parties.items(): row = [ parties.get(party_id, {}).get(year, {}).get('mandates', 0) for year in years ] row.insert(0, party_names.get(party_id, '')) result.append(row) return result
[docs] def get_parties_panachage_data( item: 'Election | ElectionCompound | ElectionCompoundPart', request: 'ElectionDayRequest | None' = None ) -> 'JSONObject_ro': """" Get the panachage data as JSON. Used to for the panachage sankey chart. """ if not getattr(item, 'has_party_panachage_results', False): return {} item = cast( 'ProporzElection | ElectionCompound | ElectionCompoundPart', item ) results = item.party_panachage_results party_results = [ r for r in item.party_results if r.year == item.date.year ] if not results: return {} parties = sorted( {result.source for result in results} | {result.target for result in results} | {result.party_id for result in party_results} ) def left_node(party: str) -> int: return parties.index(party) def right_node(party: str) -> int: return parties.index(party) + len(parties) active = {r.party_id: r.number_of_mandates > 0 for r in party_results} colors = item.historical_colors colors = { r.party_id: colors[r.name] for r in party_results if r.name in colors } # Create the links links: list[JSONObject_ro] = [] for result in results: if result.source == result.target: continue links.append({ 'source': left_node(result.source), 'target': right_node(result.target), 'value': result.votes, 'color': colors.get(result.source), 'active': active.get(result.source, False) }) # Create the nodes names = {r.party_id: r.name for r in party_results} blank = request.translate(_('Blank list')) if request else '-' nodes: list[JSONObject_ro] = [ { 'name': names.get(party_id, '') or blank, 'id': count + 1, 'color': colors.get(party_id), 'active': active.get(party_id, False) } for count, party_id in enumerate(2 * parties) ] return { 'nodes': nodes, 'links': links, 'title': item.title }