mirror of https://github.com/Kozea/pygal.git
Guillaume Ayoub
11 years ago
27 changed files with 1262 additions and 250 deletions
After Width: | Height: | Size: 584 KiB |
After Width: | Height: | Size: 370 KiB |
@ -0,0 +1,387 @@
|
||||
# -*- coding: utf-8 -*- |
||||
# This file is part of pygal |
||||
# |
||||
# A python svg graph plotting library |
||||
# Copyright © 2012-2014 Kozea |
||||
# |
||||
# This library is free software: you can redistribute it and/or modify it under |
||||
# the terms of the GNU Lesser General Public License as published by the Free |
||||
# Software Foundation, either version 3 of the License, or (at your option) any |
||||
# later version. |
||||
# |
||||
# This library is distributed in the hope that it will be useful, but WITHOUT |
||||
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS |
||||
# FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more |
||||
# details. |
||||
# |
||||
# You should have received a copy of the GNU Lesser General Public License |
||||
# along with pygal. If not, see <http://www.gnu.org/licenses/>. |
||||
""" |
||||
Worldmap chart |
||||
|
||||
""" |
||||
|
||||
from __future__ import division |
||||
from collections import defaultdict |
||||
from pygal.ghost import ChartCollection |
||||
from pygal.util import cut, cached_property, decorate |
||||
from pygal.graph.graph import Graph |
||||
from pygal._compat import u |
||||
from numbers import Number |
||||
from lxml import etree |
||||
import os |
||||
|
||||
|
||||
DEPARTMENTS = { |
||||
'01': u("Ain"), |
||||
'02': u("Aisne"), |
||||
'03': u("Allier"), |
||||
'04': u("Alpes-de-Haute-Provence"), |
||||
'05': u("Hautes-Alpes"), |
||||
'06': u("Alpes-Maritimes"), |
||||
'07': u("Ardèche"), |
||||
'08': u("Ardennes"), |
||||
'09': u("Ariège"), |
||||
'10': u("Aube"), |
||||
'11': u("Aude"), |
||||
'12': u("Aveyron"), |
||||
'13': u("Bouches-du-Rhône"), |
||||
'14': u("Calvados"), |
||||
'15': u("Cantal"), |
||||
'16': u("Charente"), |
||||
'17': u("Charente-Maritime"), |
||||
'18': u("Cher"), |
||||
'19': u("Corrèze"), |
||||
'2A': u("Corse-du-Sud"), |
||||
'2B': u("Haute-Corse"), |
||||
'21': u("Côte-d'Or"), |
||||
'22': u("Côtes-d'Armor"), |
||||
'23': u("Creuse"), |
||||
'24': u("Dordogne"), |
||||
'25': u("Doubs"), |
||||
'26': u("Drôme"), |
||||
'27': u("Eure"), |
||||
'28': u("Eure-et-Loir"), |
||||
'29': u("Finistère"), |
||||
'30': u("Gard"), |
||||
'31': u("Haute-Garonne"), |
||||
'32': u("Gers"), |
||||
'33': u("Gironde"), |
||||
'34': u("Hérault"), |
||||
'35': u("Ille-et-Vilaine"), |
||||
'36': u("Indre"), |
||||
'37': u("Indre-et-Loire"), |
||||
'38': u("Isère"), |
||||
'39': u("Jura"), |
||||
'40': u("Landes"), |
||||
'41': u("Loir-et-Cher"), |
||||
'42': u("Loire"), |
||||
'43': u("Haute-Loire"), |
||||
'44': u("Loire-Atlantique"), |
||||
'45': u("Loiret"), |
||||
'46': u("Lot"), |
||||
'47': u("Lot-et-Garonne"), |
||||
'48': u("Lozère"), |
||||
'49': u("Maine-et-Loire"), |
||||
'50': u("Manche"), |
||||
'51': u("Marne"), |
||||
'52': u("Haute-Marne"), |
||||
'53': u("Mayenne"), |
||||
'54': u("Meurthe-et-Moselle"), |
||||
'55': u("Meuse"), |
||||
'56': u("Morbihan"), |
||||
'57': u("Moselle"), |
||||
'58': u("Nièvre"), |
||||
'59': u("Nord"), |
||||
'60': u("Oise"), |
||||
'61': u("Orne"), |
||||
'62': u("Pas-de-Calais"), |
||||
'63': u("Puy-de-Dôme"), |
||||
'64': u("Pyrénées-Atlantiques"), |
||||
'65': u("Hautes-Pyrénées"), |
||||
'66': u("Pyrénées-Orientales"), |
||||
'67': u("Bas-Rhin"), |
||||
'68': u("Haut-Rhin"), |
||||
'69': u("Rhône"), |
||||
'70': u("Haute-Saône"), |
||||
'71': u("Saône-et-Loire"), |
||||
'72': u("Sarthe"), |
||||
'73': u("Savoie"), |
||||
'74': u("Haute-Savoie"), |
||||
'75': u("Paris"), |
||||
'76': u("Seine-Maritime"), |
||||
'77': u("Seine-et-Marne"), |
||||
'78': u("Yvelines"), |
||||
'79': u("Deux-Sèvres"), |
||||
'80': u("Somme"), |
||||
'81': u("Tarn"), |
||||
'82': u("Tarn-et-Garonne"), |
||||
'83': u("Var"), |
||||
'84': u("Vaucluse"), |
||||
'85': u("Vendée"), |
||||
'86': u("Vienne"), |
||||
'87': u("Haute-Vienne"), |
||||
'88': u("Vosges"), |
||||
'89': u("Yonne"), |
||||
'90': u("Territoire de Belfort"), |
||||
'91': u("Essonne"), |
||||
'92': u("Hauts-de-Seine"), |
||||
'93': u("Seine-Saint-Denis"), |
||||
'94': u("Val-de-Marne"), |
||||
'95': u("Val-d'Oise"), |
||||
'971': u("Guadeloupe"), |
||||
'972': u("Martinique"), |
||||
'973': u("Guyane"), |
||||
'974': u("Réunion"), |
||||
# Not a area anymore but in case of... |
||||
'975': u("Saint Pierre et Miquelon"), |
||||
'976': u("Mayotte") |
||||
} |
||||
|
||||
|
||||
REGIONS = { |
||||
'11': u("Île-de-France"), |
||||
'21': u("Champagne-Ardenne"), |
||||
'22': u("Picardie"), |
||||
'23': u("Haute-Normandie"), |
||||
'24': u("Centre"), |
||||
'25': u("Basse-Normandie"), |
||||
'26': u("Bourgogne"), |
||||
'31': u("Nord-Pas-de-Calais"), |
||||
'41': u("Lorraine"), |
||||
'42': u("Alsace"), |
||||
'43': u("Franche-Comté"), |
||||
'52': u("Pays-de-la-Loire"), |
||||
'53': u("Bretagne"), |
||||
'54': u("Poitou-Charentes"), |
||||
'72': u("Aquitaine"), |
||||
'73': u("Midi-Pyrénées"), |
||||
'74': u("Limousin"), |
||||
'82': u("Rhône-Alpes"), |
||||
'83': u("Auvergne"), |
||||
'91': u("Languedoc-Roussillon"), |
||||
'93': u("Provence-Alpes-Côte d'Azur"), |
||||
'94': u("Corse"), |
||||
'01': u("Guadeloupe"), |
||||
'02': u("Martinique"), |
||||
'03': u("Guyane"), |
||||
'04': u("Réunion"), |
||||
# Not a region anymore but in case of... |
||||
'05': u("Saint Pierre et Miquelon"), |
||||
'06': u("Mayotte") |
||||
} |
||||
|
||||
|
||||
with open(os.path.join( |
||||
os.path.dirname(__file__), |
||||
'fr.departments.svg')) as file: |
||||
DPT_MAP = file.read() |
||||
|
||||
|
||||
with open(os.path.join( |
||||
os.path.dirname(__file__), |
||||
'fr.regions.svg')) as file: |
||||
REG_MAP = file.read() |
||||
|
||||
|
||||
class FrenchMapDepartments(Graph): |
||||
"""French department map""" |
||||
_dual = True |
||||
x_labels = list(DEPARTMENTS.keys()) |
||||
area_names = DEPARTMENTS |
||||
area_prefix = 'z' |
||||
svg_map = DPT_MAP |
||||
|
||||
|
||||
@cached_property |
||||
def _values(self): |
||||
"""Getter for series values (flattened)""" |
||||
return [val[1] |
||||
for serie in self.series |
||||
for val in serie.values |
||||
if val[1] is not None] |
||||
|
||||
def _plot(self): |
||||
map = etree.fromstring(self.svg_map) |
||||
map.set('width', str(self.view.width)) |
||||
map.set('height', str(self.view.height)) |
||||
|
||||
for i, serie in enumerate(self.series): |
||||
safe_vals = list(filter( |
||||
lambda x: x is not None, cut(serie.values, 1))) |
||||
if not safe_vals: |
||||
continue |
||||
min_ = min(safe_vals) |
||||
max_ = max(safe_vals) |
||||
for j, (area_code, value) in enumerate(serie.values): |
||||
if isinstance(area_code, Number): |
||||
area_code = '%2d' % area_code |
||||
if value is None: |
||||
continue |
||||
if max_ == min_: |
||||
ratio = 1 |
||||
else: |
||||
ratio = .3 + .7 * (value - min_) / (max_ - min_) |
||||
areae = map.xpath( |
||||
"//*[contains(concat(' ', normalize-space(@class), ' ')," |
||||
" ' %s%s ')]" % (self.area_prefix, area_code)) |
||||
|
||||
if not areae: |
||||
continue |
||||
for area in areae: |
||||
cls = area.get('class', '').split(' ') |
||||
cls.append('color-%d' % i) |
||||
area.set('class', ' '.join(cls)) |
||||
area.set('style', 'fill-opacity: %f' % (ratio)) |
||||
|
||||
metadata = serie.metadata.get(j) |
||||
if metadata: |
||||
parent = area.getparent() |
||||
node = decorate(self.svg, area, metadata) |
||||
if node != area: |
||||
area.remove(node) |
||||
index = parent.index(area) |
||||
parent.remove(area) |
||||
node.append(area) |
||||
parent.insert(index, node) |
||||
|
||||
last_node = len(area) > 0 and area[-1] |
||||
if last_node is not None and last_node.tag == 'title': |
||||
title_node = last_node |
||||
text = title_node.text + '\n' |
||||
else: |
||||
title_node = self.svg.node(area, 'title') |
||||
text = '' |
||||
title_node.text = text + '[%s] %s: %s' % ( |
||||
serie.title, |
||||
self.area_names[area_code], self._format(value)) |
||||
|
||||
self.nodes['plot'].append(map) |
||||
|
||||
|
||||
class FrenchMapRegions(FrenchMapDepartments): |
||||
"""French regions map""" |
||||
x_labels = list(REGIONS.keys()) |
||||
area_names = REGIONS |
||||
area_prefix = 'a' |
||||
svg_map = REG_MAP |
||||
|
||||
|
||||
class FrenchMap(ChartCollection): |
||||
Regions = FrenchMapRegions |
||||
Departments = FrenchMapDepartments |
||||
|
||||
|
||||
DEPARTMENTS_REGIONS = { |
||||
"01": "82", |
||||
"02": "22", |
||||
"03": "83", |
||||
"04": "93", |
||||
"05": "93", |
||||
"06": "93", |
||||
"07": "82", |
||||
"08": "21", |
||||
"09": "73", |
||||
"10": "21", |
||||
"11": "91", |
||||
"12": "73", |
||||
"13": "93", |
||||
"14": "25", |
||||
"15": "83", |
||||
"16": "54", |
||||
"17": "54", |
||||
"18": "24", |
||||
"19": "74", |
||||
"21": "26", |
||||
"22": "53", |
||||
"23": "74", |
||||
"24": "72", |
||||
"25": "43", |
||||
"26": "82", |
||||
"27": "23", |
||||
"28": "24", |
||||
"29": "53", |
||||
"2A": "94", |
||||
"2B": "94", |
||||
"30": "91", |
||||
"31": "73", |
||||
"32": "73", |
||||
"33": "72", |
||||
"34": "91", |
||||
"35": "53", |
||||
"36": "24", |
||||
"37": "24", |
||||
"38": "82", |
||||
"39": "43", |
||||
"40": "72", |
||||
"41": "24", |
||||
"42": "82", |
||||
"43": "83", |
||||
"44": "52", |
||||
"45": "24", |
||||
"46": "73", |
||||
"47": "72", |
||||
"48": "91", |
||||
"49": "52", |
||||
"50": "25", |
||||
"51": "21", |
||||
"52": "21", |
||||
"53": "52", |
||||
"54": "41", |
||||
"55": "41", |
||||
"56": "53", |
||||
"57": "41", |
||||
"58": "26", |
||||
"59": "31", |
||||
"60": "22", |
||||
"61": "25", |
||||
"62": "31", |
||||
"63": "83", |
||||
"64": "72", |
||||
"65": "73", |
||||
"66": "91", |
||||
"67": "42", |
||||
"68": "42", |
||||
"69": "82", |
||||
"70": "43", |
||||
"71": "26", |
||||
"72": "52", |
||||
"73": "82", |
||||
"74": "82", |
||||
"75": "11", |
||||
"76": "23", |
||||
"77": "11", |
||||
"78": "11", |
||||
"79": "54", |
||||
"80": "22", |
||||
"81": "73", |
||||
"82": "73", |
||||
"83": "93", |
||||
"84": "93", |
||||
"85": "52", |
||||
"86": "54", |
||||
"87": "74", |
||||
"88": "41", |
||||
"89": "26", |
||||
"90": "43", |
||||
"91": "11", |
||||
"92": "11", |
||||
"93": "11", |
||||
"94": "11", |
||||
"95": "11", |
||||
"971": "01", |
||||
"972": "02", |
||||
"973": "03", |
||||
"974": "04", |
||||
"975": "05", |
||||
"976": "06" |
||||
} |
||||
|
||||
|
||||
def aggregate_regions(values): |
||||
if isinstance(values, dict): |
||||
values = values.items() |
||||
regions = defaultdict(int) |
||||
for department, value in values: |
||||
regions[DEPARTMENTS_REGIONS[department]] += value |
||||
return list(regions.items()) |
Before Width: | Height: | Size: 752 KiB After Width: | Height: | Size: 754 KiB |
Loading…
Reference in new issue