# -*- coding: utf-8 -*-
from typing import Union
from honeybees.artists import Artists as honeybeesArtists
import numpy as np
import re
from operator import attrgetter
from geb.store import DynamicArray
[docs]
class Artists(honeybeesArtists):
"""This class is used to configure how the display environment works.
Args:
model: The GEB model.
"""
def __init__(self, model) -> None:
honeybeesArtists.__init__(self, model)
self.color = "#1386FF"
self.min_colorbar_alpha = 0.4
self.background_variable = (
"data.HRU.land_use_type" # set initial background iamge.
)
self.custom_plot = self.get_custom_plot()
self.set_variables()
[docs]
def draw_crop_farmers(
self, model, agents, idx: int, color: str = "#ff0000"
) -> dict:
"""This function is used to draw farmers. First it is determined what crop is grown by the farmer, then the we get the color used to display that crop from the model configuration.
Args:
model: The GEB model.
agents: The farmer class to plot.
idx: The farmer index.
Returns:
portrayal: Portrayal of farmer.
"""
# if self.model.agents.farmers.flooded[idx] == True:
# color = '#ff0000'
# else:
# color = '#0000ff'
# if idx == self.model.agents.farmers.sample[0]:
# color = '#ff0000'
# r = 3
# elif idx == self.model.agents.farmers.sample[1]:
# color = '#00ff00'
# r = 3
# elif idx == self.model.agents.farmers.sample[2]:
# color = '#0000ff'
# r = 3
# else:
r = 0.5
return {
"type": "shape",
"shape": "circle",
"r": r,
"filled": True,
"color": "#ff0000",
}
def draw_tehsil(self, properties):
return {
"type": "shape",
"shape": "polygon",
"filled": False,
"color": properties["color"],
"edge": True,
"linewidth": 2,
}
[docs]
def get_custom_plot(self) -> dict[dict]:
"""Here you can specify custom options for plotting the background.
Returns:
custom_dict: Dictionary of dictionaries. The first level is the name of each of the variables, the second level the options for those variables.
Example:
.. code-block:: python
{
'HRU.crop_map': {
'type': 'discrete',
'nanvalue': -1,
'names': ['crop name 1', 'crop name 2'],
'colors': ['#00FF00', '#FF0000']
}
}
"""
return {
"data.HRU.land_use_type": {
"type": "categorical",
"nanvalue": -1,
"names": [
"forest",
"grassland/non-irrigated",
"paddy-irrigated",
"non-paddy irrigated",
"sealed",
"water",
],
"colors": [
"#274e2e",
"#adffbc",
"#8555aa",
"#007d13",
"#7e8180",
"#2636d9",
],
},
}
[docs]
def set_variables(self) -> None:
"""This function is used to get a dictionary of variables that can be shown as background variable. The dictionary :code:`self.variables_dict` contains the name of each variable to display as key, and the actual variable as value.
Checks are performed to see whether the data is the right size. Only compressed data can be shown. If a dataset has multiple dimensions, the dimensions can be shown seperately as `variable[0], variable[1], ...`.
"""
self.variables_dict = {}
def add_vars(name, compressed_size, dtypes, variant_dim, invariant_dim):
assert np.intersect1d(variant_dim, invariant_dim).size == 0
container = attrgetter(name)(self.model)
for varname, variable in vars(container).items():
if isinstance(variable, dtypes):
if variable.ndim == 1 and variable.size == compressed_size:
self.variables_dict[f"{name}.{varname}"] = variable
if (
variable.ndim == 2
and variable.shape[invariant_dim] == compressed_size
):
for i in range(variable.shape[variant_dim]):
self.variables_dict[f"{name}.{varname}[{i}]"] = variable[
:, i
]
else:
continue
add_vars(
"data.grid",
compressed_size=self.model.data.grid.compressed_size,
dtypes=np.ndarray,
variant_dim=0,
invariant_dim=1,
)
add_vars(
"data.HRU",
compressed_size=self.model.data.HRU.compressed_size,
dtypes=np.ndarray,
variant_dim=0,
invariant_dim=1,
)
add_vars(
"agents.crop_farmers",
compressed_size=self.model.agents.crop_farmers.n,
dtypes=DynamicArray,
variant_dim=1,
invariant_dim=0,
)
[docs]
def get_background_variables(self) -> list:
"""This function gets a list of variables that can be used to show in the background.
Returns:
options: List of names for options to show in background.
"""
self.set_variables()
return list(self.variables_dict.keys())
[docs]
def set_background_variable(self, option_name: str) -> None:
"""This function is used to update the name of the variable to use for drawing the background of the map."""
self.background_variable = option_name
[docs]
def get_background(
self,
minvalue: Union[float, int, None] = None,
maxvalue: Union[float, int, None] = None,
color: str = "#1386FF",
) -> tuple[np.ndarray, dict]:
"""This function is called from the canvas class to draw the canvas background. The name of the variable to draw is stored in `self.background_variable`.
Args:
minvalue: The minimum value for the display scale.
maxvalue: The maximum value for the display scale.
color: The color to use to display the variable.
Returns:
background: RGBA-array to display as background.
legend: Dictionary with data and formatting rules for background legend.
"""
if self.background_variable.startswith("agents.farmers"):
slicer = re.search(r"\[([0-9]+)\]$", self.background_variable)
if slicer:
array = attrgetter(self.background_variable[: slicer.span(0)[0]])(
self.model
)
array = DynamicArray(array[:, int(slicer.group(1))], n=array.shape[0])
else:
array = attrgetter(self.background_variable)(self.model)
mask = self.model.data.HRU.mask
else:
compressed_array, array = self.model.reporter.hydrology_reporter.get_array(
self.background_variable, decompress=True
)
mask = attrgetter(".".join(self.background_variable.split(".")[:-1]))(
self.model
).mask
if self.background_variable in self.custom_plot:
options = self.custom_plot[self.background_variable]
else:
options = {}
if "type" not in options:
if np.issubdtype(array.dtype, np.floating):
options["type"] = "continuous"
options["nanvalue"] = np.nan
elif np.issubdtype(array.dtype, np.integer):
if np.unique(array).size < 30:
options["type"] = "categorical"
options["nanvalue"] = -1
else:
print(
"Type for array might be categorical, but more than 30 categories were found, so rendering as continous."
)
options["type"] = "continuous"
array = array.astype(np.float64)
elif np.issubdtype(array.dtype, bool):
options["type"] = "bool"
options["nanvalue"] = -1
else:
raise ValueError
if self.background_variable.startswith("agents.farmers"):
compressed_array = array.copy()
array = array.by_field(self.model.data.HRU.land_owners, options["nanvalue"])
array = self.model.data.HRU.decompress(array)
if options["type"] == "bool":
minvalue, maxvalue = 0, 1
else:
if not maxvalue:
maxvalue = np.nanmax(array[~mask]).item()
if not minvalue:
minvalue = np.nanmin(array[~mask]).item()
if np.isnan(maxvalue): # minvalue must be nan as well
minvalue, maxvalue = 0, 0
background = np.zeros((*array.shape, 4), dtype=np.uint8)
if options["type"] == "continuous":
array -= minvalue
if maxvalue - minvalue != 0:
array *= 255 / (maxvalue - minvalue)
else:
array *= 0
array[array < 0] = 0
array[array > 255] = 255
rgb = self.hex_to_rgb(color)
for channel in (0, 1, 2):
background[:, :, channel][~np.isnan(array)] = rgb[channel] * 255
background[:, :, 3] = array
background[:, :, 0][np.isnan(array)] = 200
background[:, :, 1][np.isnan(array)] = 200
background[:, :, 2][np.isnan(array)] = 200
background[:, :, 3][np.isnan(array)] = 255
legend = {
"type": "colorbar",
"color": color,
"min": self.round_to_n_significant_digits(minvalue, 3),
"max": self.round_to_n_significant_digits(maxvalue, 3),
"min_colorbar_alpha": 0,
"unit": "",
}
else:
if "nanvalue" in options:
nanvalue = options["nanvalue"]
else:
nanvalue = None
if options["type"] == "categorical":
unique_values = np.unique(compressed_array)
if nanvalue is not None:
unique_values = unique_values[unique_values != nanvalue]
unique_values = unique_values.tolist()
if unique_values: # no data to be shown on map
if "colors" in options:
colors = np.array(options["colors"])[
np.array(unique_values)
].tolist()
colors = [self.hex_to_rgb(color) for color in colors]
else:
colors = self.generate_distinct_colors(
len(unique_values), mode="rgb"
)
if "names" in options:
names = np.array(options["names"])[
np.array(unique_values)
].tolist()
else:
names = unique_values
else:
colors = []
names = []
channels = (0, 1, 2)
background[:, :, 3][array != nanvalue] = 255
elif options["type"] == "discrete":
unique_values = np.arange(
compressed_array[compressed_array != nanvalue].min(),
compressed_array[compressed_array != nanvalue].max() + 1,
1,
).tolist()
if "names" in options:
names = options["names"]
else:
names = unique_values
colors = self.generate_discrete_colors(
len(unique_values),
self.hex_to_rgb(color),
mode="rgb",
min_alpha=0.4,
)
channels = (0, 1, 2, 3)
elif options["type"] == "bool":
unique_values = [False, True]
names = ["False", "True"]
channels = (0, 1, 2, 3)
colors = [(1, 0, 0, 1), (0, 1, 0, 1)]
else:
raise ValueError
if unique_values:
assert np.all(np.diff(unique_values) > 0) # check if array is sorted
for channel in channels:
channel_colors = np.array(
[color[channel] * 255 for color in colors]
)
color_array_size = unique_values[-1] + 1
if unique_values[0] < 0:
color_array_size += abs(unique_values[0])
color_array = np.zeros(color_array_size, dtype=np.float32)
color_array[np.array(unique_values).astype(np.int32)] = (
channel_colors
)
background[:, :, channel] = color_array[array.astype(np.int32)]
legend = {
"type": "legend",
"labels": {
name: self.rgb_to_hex(colors[i]) for i, name in enumerate(names)
},
}
background[mask] = 200
return background, legend