Введение
В этом лабораторном занятии мы научимся создавать интерактивную гистограмму с использованием Matplotlib. Взаимодействие кодируется на ECMAScript (JavaScript) и вставляется в SVG-код на стадии постобработки. Результат будет гистограммой, у которой столбцы можно скрывать или показывать, нажимая на маркеры легенды.
Советы по работе с ВМ
После запуска ВМ нажмите в левом верхнем углу, чтобы переключиться на вкладку Notebook и получить доступ к Jupyter Notebook для практики.
Иногда вам может потребоваться подождать несколько секунд, пока Jupyter Notebook загрузится. Валидация операций не может быть автоматизирована из-за ограничений Jupyter Notebook.
Если вы сталкиваетесь с проблемами во время обучения, не стесняйтесь обращаться к Labby. Оставьте отзыв после занятия, и мы оперативно решим проблему для вас.
Создание гистограммы, легенды и заголовка
Сначала мы создадим гистограмму, легенду и заголовок с использованием Matplotlib. Также мы присвоим идентификаторы каждому объекту с помощью метода set_gid(). Это поможет связать объекты Matplotlib, созданные в Python, и соответствующие SVG-конструкции, которые будут разобраны на втором этапе.
import matplotlib.pyplot as plt
import numpy as np
## Fixing random state for reproducibility
np.random.seed(19680801)
## Create histogram, legend, and title
plt.figure()
r = np.random.randn(100)
r1 = r + 1
labels = ['Rabbits', 'Frogs']
H = plt.hist([r, r1], label=labels)
containers = H[-1]
leg = plt.legend(frameon=False)
plt.title("From a web browser, click on the legend\n"
"marker to toggle the corresponding histogram.")
## Assign IDs to the SVG objects we'll modify
hist_patches = {}
for ic, c in enumerate(containers):
hist_patches[f'hist_{ic}'] = []
for il, element in enumerate(c):
element.set_gid(f'hist_{ic}_patch_{il}')
hist_patches[f'hist_{ic}'].append(f'hist_{ic}_patch_{il}')
## Set IDs for the legend patches
for i, t in enumerate(leg.get_patches()):
t.set_gid(f'leg_patch_{i}')
## Set IDs for the text patches
for i, t in enumerate(leg.get_texts()):
t.set_gid(f'leg_text_{i}')
Добавление интерактивности к гистограмме
Далее мы добавим интерактивность к гистограмме, изменив SVG-код с использованием ECMAScript (JavaScript). Мы добавим атрибуты к объектам патча и текста с использованием метода set(). Также мы создадим глобальную переменную container, которая будет хранить идентификаторы патчей, принадлежащих каждой гистограмме. Наконец, мы создадим функцию toggle_hist(), которая будет настраивать атрибут видимости всех патчей каждой гистограммы и непрозрачность маркера самого.
from io import BytesIO
import json
import xml.etree.ElementTree as ET
plt.rcParams['svg.fonttype'] = 'none'
## Save SVG in a fake file object
f = BytesIO()
plt.savefig(f, format="svg")
## Create XML tree from the SVG file
tree, xmlid = ET.XMLID(f.getvalue())
## Add attributes to the patch objects
for i, t in enumerate(leg.get_patches()):
el = xmlid[f'leg_patch_{i}']
el.set('cursor', 'pointer')
el.set('onclick', "toggle_hist(this)")
## Add attributes to the text objects
for i, t in enumerate(leg.get_texts()):
el = xmlid[f'leg_text_{i}']
el.set('cursor', 'pointer')
el.set('onclick', "toggle_hist(this)")
## Create script defining the function `toggle_hist`
script = """
<script type="text/ecmascript">
<![CDATA[
var container = %s
function toggle(oid, attribute, values) {
/* Toggle the style attribute of an object between two values.
Parameters
----------
oid : str
Object identifier.
attribute : str
Name of style attribute.
values : [on state, off state]
The two values that are switched between.
*/
var obj = document.getElementById(oid);
var a = obj.style[attribute];
a = (a == values[0] || a == "")? values[1] : values[0];
obj.style[attribute] = a;
}
function toggle_hist(obj) {
var num = obj.id.slice(-1);
toggle('leg_patch_' + num, 'opacity', [1, 0.3]);
toggle('leg_text_' + num, 'opacity', [1, 0.5]);
var names = container['hist_'+num]
for (var i=0; i < names.length; i++) {
toggle(names[i], 'opacity', [1, 0])
};
}
]]>
</script>
""" % json.dumps(hist_patches)
## Add a transition effect
css = tree.find('.//{http://www.w3.org/2000/svg}style')
css.text = css.text + "g {-webkit-transition:opacity 0.4s ease-out;" + \
"-moz-transition:opacity 0.4s ease-out;}"
## Insert the script and save to file
tree.insert(0, ET.XML(script))
ET.ElementTree(tree).write("svg_histogram.svg")
Просмотр интерактивной гистограммы
Наконец, мы можем просмотреть интерактивную гистограмму, открыв SVG-файл в веб-браузере. Чтобы скрыть или показать столбцы, просто кликните по маркерам легенды.
Резюме
В этом лабораторном занятии мы научились создавать интерактивную гистограмму с использованием Matplotlib. Мы использовали ECMAScript (JavaScript), чтобы добавить интерактивность к гистограмме, изменив SVG-код. Результат — гистограмма, у которой столбцы можно скрывать или показывать, нажимая на маркеры легенды.