From f9a1557bad0f5b6654432d014c88be8bc4f031dd Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Mon, 16 Oct 2023 16:08:58 +0200 Subject: [PATCH 01/19] Update colormap.py --- branca/colormap.py | 164 +++++++++++++++++++++++++++------------------ 1 file changed, 98 insertions(+), 66 deletions(-) diff --git a/branca/colormap.py b/branca/colormap.py index e5dd839..6f53410 100644 --- a/branca/colormap.py +++ b/branca/colormap.py @@ -9,51 +9,69 @@ import json import math import os +from typing import Dict, List, Optional, Sequence, Tuple, Union from jinja2 import Template from branca.element import ENV, Figure, JavascriptLink, MacroElement from branca.utilities import legend_scaler -rootpath = os.path.abspath(os.path.dirname(__file__)) +rootpath: str = os.path.abspath(os.path.dirname(__file__)) with open(os.path.join(rootpath, "_cnames.json")) as f: - _cnames = json.loads(f.read()) + _cnames: Dict[str, str] = json.loads(f.read()) with open(os.path.join(rootpath, "_schemes.json")) as f: - _schemes = json.loads(f.read()) + _schemes: Dict[str, List[str]] = json.loads(f.read()) -def _is_hex(x): +TypeRGBInts = Tuple[int, int, int] +TypeRGBFloats = Tuple[float, float, float] +TypeRGBAInts = Tuple[int, int, int, int] +TypeRGBAFloats = Tuple[float, float, float, float] +TypeAnyColorType = Union[TypeRGBInts, TypeRGBFloats, TypeRGBAInts, TypeRGBAFloats, str] + + +def _is_hex(x: str) -> bool: return x.startswith("#") and len(x) == 7 -def _parse_hex(color_code): +def _parse_hex(color_code: str) -> TypeRGBAFloats: return ( - int(color_code[1:3], 16), - int(color_code[3:5], 16), - int(color_code[5:7], 16), + _color_int_to_float(int(color_code[1:3], 16)), + _color_int_to_float(int(color_code[3:5], 16)), + _color_int_to_float(int(color_code[5:7], 16)), + 1.0, ) -def _parse_color(x): +def _color_int_to_float(x: int) -> float: + """Convert an integer between 0 and 255 to a float between 0. and 1.0""" + return x / 255.0 + + +def _color_float_to_int(x: float) -> int: + """Convert a float between 0. and 1.0 to an integer between 0 and 255""" + return int(x * 255.9999) + + +def _parse_color(x: Union[tuple, list, str]) -> TypeRGBAFloats: + color_tuple: TypeRGBAFloats if isinstance(x, (tuple, list)): - color_tuple = tuple(x)[:4] - elif isinstance(x, (str, bytes)) and _is_hex(x): + color_tuple = tuple(tuple(x) + (1.0,))[:4] # type: ignore + elif isinstance(x, str) and _is_hex(x): color_tuple = _parse_hex(x) - elif isinstance(x, (str, bytes)): + elif isinstance(x, str): cname = _cnames.get(x.lower(), None) if cname is None: raise ValueError(f"Unknown color {cname!r}.") color_tuple = _parse_hex(cname) else: raise ValueError(f"Unrecognized color code {x!r}") - if max(color_tuple) > 1.0: - color_tuple = tuple(u / 255.0 for u in color_tuple) - return tuple(map(float, (color_tuple + (1.0,))[:4])) + return color_tuple -def _base(x): +def _base(x: float) -> float: if x > 0: base = pow(10, math.floor(math.log10(x))) return round(x / base) * base @@ -78,15 +96,15 @@ class ColorMap(MacroElement): Maximum number of legend tick labels """ - _template = ENV.get_template("color_scale.js") + _template: Template = ENV.get_template("color_scale.js") def __init__( self, - vmin=0.0, - vmax=1.0, - caption="", - text_color="black", - max_labels=10, + vmin: float = 0.0, + vmax: float = 1.0, + caption: str = "", + text_color: str = "black", + max_labels: int = 10, ): super().__init__() self._name = "ColorMap" @@ -97,7 +115,7 @@ def __init__( self.text_color = text_color self.index = [vmin, vmax] self.max_labels = max_labels - self.tick_labels = None + self.tick_labels: Optional[Sequence[float]] = None self.width = 450 self.height = 40 @@ -127,7 +145,7 @@ def render(self, **kwargs): name="d3", ) # noqa - def rgba_floats_tuple(self, x): + def rgba_floats_tuple(self, x: float) -> TypeRGBAFloats: """ This class has to be implemented for each class inheriting from Colormap. This has to be a function of the form float -> @@ -137,37 +155,37 @@ def rgba_floats_tuple(self, x): """ raise NotImplementedError - def rgba_bytes_tuple(self, x): + def rgba_bytes_tuple(self, x: float) -> TypeRGBAInts: """Provides the color corresponding to value `x` in the form of a tuple (R,G,B,A) with int values between 0 and 255. """ - return tuple(int(u * 255.9999) for u in self.rgba_floats_tuple(x)) + return tuple(_color_float_to_int(u) for u in self.rgba_floats_tuple(x)) # type: ignore - def rgb_bytes_tuple(self, x): + def rgb_bytes_tuple(self, x: float) -> TypeRGBInts: """Provides the color corresponding to value `x` in the form of a tuple (R,G,B) with int values between 0 and 255. """ return self.rgba_bytes_tuple(x)[:3] - def rgb_hex_str(self, x): + def rgb_hex_str(self, x: float) -> str: """Provides the color corresponding to value `x` in the form of a string of hexadecimal values "#RRGGBB". """ return "#%02x%02x%02x" % self.rgb_bytes_tuple(x) - def rgba_hex_str(self, x): + def rgba_hex_str(self, x: float) -> str: """Provides the color corresponding to value `x` in the form of a string of hexadecimal values "#RRGGBBAA". """ return "#%02x%02x%02x%02x" % self.rgba_bytes_tuple(x) - def __call__(self, x): + def __call__(self, x: float) -> str: """Provides the color corresponding to value `x` in the form of a string of hexadecimal values "#RRGGBBAA". """ return self.rgba_hex_str(x) - def _repr_html_(self): + def _repr_html_(self) -> str: """Display the colormap in a Jupyter Notebook. Does not support all the class arguments. @@ -264,14 +282,14 @@ class LinearColormap(ColorMap): def __init__( self, - colors, - index=None, - vmin=0.0, - vmax=1.0, - caption="", - text_color="black", - max_labels=10, - tick_labels=None, + colors: Sequence[TypeAnyColorType], + index: Optional[Sequence[float]] = None, + vmin: float = 0.0, + vmax: float = 1.0, + caption: str = "", + text_color: str = "black", + max_labels: int = 10, + tick_labels: Optional[Sequence[float]] = None, ): super().__init__( vmin=vmin, @@ -289,9 +307,9 @@ def __init__( self.index = [vmin + (vmax - vmin) * i * 1.0 / (n - 1) for i in range(n)] else: self.index = list(index) - self.colors = [_parse_color(x) for x in colors] + self.colors: List[TypeRGBAFloats] = [_parse_color(x) for x in colors] - def rgba_floats_tuple(self, x): + def rgba_floats_tuple(self, x: float) -> TypeRGBAFloats: """Provides the color corresponding to value `x` in the form of a tuple (R,G,B,A) with float values between 0. and 1. """ @@ -308,20 +326,20 @@ def rgba_floats_tuple(self, x): else: raise ValueError("Thresholds are not sorted.") - return tuple( + return tuple( # type: ignore (1.0 - p) * self.colors[i - 1][j] + p * self.colors[i][j] for j in range(4) ) def to_step( self, - n=None, - index=None, - data=None, - method=None, - quantiles=None, - round_method=None, - max_labels=10, - ): + n: Optional[int] = None, + index: Optional[Sequence[float]] = None, + data: Optional[Sequence[float]] = None, + method: Optional[str] = None, + quantiles: Optional[Sequence[float]] = None, + round_method: Optional[str] = None, + max_labels: int = 10, + ) -> "StepColormap": """Splits the LinearColormap into a StepColormap. Parameters @@ -454,7 +472,12 @@ def to_step( tick_labels=self.tick_labels, ) - def scale(self, vmin=0.0, vmax=1.0, max_labels=10): + def scale( + self, + vmin: float = 0.0, + vmax: float = 1.0, + max_labels: int = 10, + ) -> "LinearColormap": """Transforms the colorscale so that the minimal and maximal values fit the given parameters. """ @@ -510,14 +533,14 @@ class StepColormap(ColorMap): def __init__( self, - colors, - index=None, - vmin=0.0, - vmax=1.0, - caption="", - text_color="black", - max_labels=10, - tick_labels=None, + colors: Sequence[TypeAnyColorType], + index: Optional[Sequence[float]] = None, + vmin: float = 0.0, + vmax: float = 1.0, + caption: str = "", + text_color: str = "black", + max_labels: int = 10, + tick_labels: Optional[Sequence[float]] = None, ): super().__init__( vmin=vmin, @@ -535,9 +558,9 @@ def __init__( self.index = [vmin + (vmax - vmin) * i * 1.0 / n for i in range(n + 1)] else: self.index = list(index) - self.colors = [_parse_color(x) for x in colors] + self.colors: List[TypeRGBAFloats] = [_parse_color(x) for x in colors] - def rgba_floats_tuple(self, x): + def rgba_floats_tuple(self, x: float) -> TypeRGBAFloats: """ Provides the color corresponding to value `x` in the form of a tuple (R,G,B,A) with float values between 0. and 1. @@ -549,9 +572,13 @@ def rgba_floats_tuple(self, x): return self.colors[-1] i = len([u for u in self.index if u <= x]) # 0 < i < n. - return tuple(self.colors[i - 1]) + return self.colors[i - 1] - def to_linear(self, index=None, max_labels=10): + def to_linear( + self, + index: Optional[Sequence[float]] = None, + max_labels: int = 10, + ) -> LinearColormap: """ Transforms the StepColormap into a LinearColormap. @@ -584,7 +611,12 @@ def to_linear(self, index=None, max_labels=10): max_labels=max_labels, ) - def scale(self, vmin=0.0, vmax=1.0, max_labels=10): + def scale( + self, + vmin: float = 0.0, + vmax: float = 1.0, + max_labels: int = 10, + ) -> "StepColormap": """Transforms the colorscale so that the minimal and maximal values fit the given parameters. """ @@ -611,7 +643,7 @@ def __init__(self): for key, val in _schemes.items(): setattr(self, key, LinearColormap(val)) - def _repr_html_(self): + def _repr_html_(self) -> str: return Template( """ @@ -634,7 +666,7 @@ def __init__(self): for key, val in _schemes.items(): setattr(self, key, StepColormap(val)) - def _repr_html_(self): + def _repr_html_(self) -> str: return Template( """
From 891642f02613b4ebb8288737dc536d2a0ac390de Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Mon, 16 Oct 2023 16:11:06 +0200 Subject: [PATCH 02/19] Create test_mypy.yml --- .github/workflows/test_mypy.yml | 29 +++++++++++++++++++++++++++++ 1 file changed, 29 insertions(+) create mode 100644 .github/workflows/test_mypy.yml diff --git a/.github/workflows/test_mypy.yml b/.github/workflows/test_mypy.yml new file mode 100644 index 0000000..6a68406 --- /dev/null +++ b/.github/workflows/test_mypy.yml @@ -0,0 +1,29 @@ +name: Mypy type hint checks + +on: [push, pull_request] + +jobs: + run: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + + - name: Setup Micromamba env + uses: mamba-org/setup-micromamba@v1 + with: + environment-name: TEST + create-args: >- + python=3 + --file requirements.txt + --file requirements-dev.txt + + - name: Install branca from source + shell: bash -l {0} + run: | + python -m pip install -e . --no-deps --force-reinstall + + - name: Mypy test + shell: bash -l {0} + run: | + mypy branca From 8f5aa00639f27943e4eff570229023728aa3cc79 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Mon, 16 Oct 2023 16:16:04 +0200 Subject: [PATCH 03/19] add Mypy requirement --- requirements-dev.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/requirements-dev.txt b/requirements-dev.txt index 549938d..3c3b55f 100644 --- a/requirements-dev.txt +++ b/requirements-dev.txt @@ -7,6 +7,7 @@ flake8-mutable flake8-print isort jupyter +mypy nbsphinx nbval numpy From 7525a9f9a184d8546629af4dadaa5cfb63a77de7 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Mon, 16 Oct 2023 16:25:36 +0200 Subject: [PATCH 04/19] Update pyproject.toml --- pyproject.toml | 3 +++ 1 file changed, 3 insertions(+) diff --git a/pyproject.toml b/pyproject.toml index 97c7d5c..26a8a96 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,3 +1,6 @@ [build-system] requires = ["setuptools>=41.2", "setuptools_scm"] build-backend = "setuptools.build_meta" + +[tool.mypy] +ignore_missing_imports = true From 3aa41f6c411cd59f5e39847a01447fcf3537ddf9 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Tue, 17 Oct 2023 18:26:05 +0200 Subject: [PATCH 05/19] Update element.py --- branca/element.py | 191 ++++++++++++++++++++++++++++---------------- branca/utilities.py | 9 ++- 2 files changed, 129 insertions(+), 71 deletions(-) diff --git a/branca/element.py b/branca/element.py index 1912aca..e8520f7 100644 --- a/branca/element.py +++ b/branca/element.py @@ -14,11 +14,12 @@ from html import escape from os import urandom from pathlib import Path +from typing import BinaryIO, List, Optional, Tuple, Type, Union from urllib.request import urlopen from jinja2 import Environment, PackageLoader, Template -from .utilities import _camelify, _parse_size, none_max, none_min +from .utilities import TypeParseSize, _camelify, _parse_size, none_max, none_min ENV = Environment(loader=PackageLoader("branca", "templates")) @@ -45,26 +46,30 @@ class Element: """ - _template = Template( + _template: Template = Template( "{% for name, element in this._children.items() %}\n" " {{element.render(**kwargs)}}" "{% endfor %}", ) - def __init__(self, template=None, template_name=None): - self._name = "Element" - self._id = hexlify(urandom(16)).decode() - self._children = OrderedDict() - self._parent = None - self._template_str = template - self._template_name = template_name + def __init__( + self, + template: Optional[str] = None, + template_name: Optional[str] = None, + ): + self._name: str = "Element" + self._id: str = hexlify(urandom(16)).decode() + self._children: OrderedDict[str, Element] = OrderedDict() + self._parent: Optional[Element] = None + self._template_str: Optional[str] = template + self._template_name: Optional[str] = template_name if template is not None: self._template = Template(template) elif template_name is not None: self._template = ENV.get_template(template_name) - def __getstate__(self): + def __getstate__(self) -> dict: """Modify object state when pickling the object. jinja2 Templates cannot be pickled, so remove the instance attribute @@ -83,7 +88,7 @@ def __setstate__(self, state: dict): self.__dict__.update(state) - def get_name(self): + def get_name(self) -> str: """Returns a string representation of the object. This string has to be unique and to be a python and javascript-compatible @@ -91,13 +96,13 @@ def get_name(self): """ return _camelify(self._name) + "_" + self._id - def _get_self_bounds(self): + def _get_self_bounds(self) -> List[List[Optional[float]]]: """Computes the bounds of the object itself (not including it's children) in the form [[lat_min, lon_min], [lat_max, lon_max]] """ return [[None, None], [None, None]] - def get_bounds(self): + def get_bounds(self) -> List[List[Optional[float]]]: """Computes the bounds of the object and all it's children in the form [[lat_min, lon_min], [lat_max, lon_max]]. """ @@ -117,7 +122,12 @@ def get_bounds(self): ] return bounds - def add_children(self, child, name=None, index=None): + def add_children( + self, + child: "Element", + name: Optional[str] = None, + index: Optional[int] = None, + ) -> "Element": """Add a child.""" warnings.warn( "Method `add_children` is deprecated. Please use `add_child` instead.", @@ -126,7 +136,12 @@ def add_children(self, child, name=None, index=None): ) return self.add_child(child, name=name, index=index) - def add_child(self, child, name=None, index=None): + def add_child( + self, + child: "Element", + name: Optional[str] = None, + index: Optional[int] = None, + ) -> "Element": """Add a child.""" if name is None: name = child.get_name() @@ -139,13 +154,24 @@ def add_child(self, child, name=None, index=None): child._parent = self return self - def add_to(self, parent, name=None, index=None): + def add_to( + self, + parent: "Element", + name: Optional[str] = None, + index: Optional[int] = None, + ) -> "Element": """Add element to a parent.""" parent.add_child(self, name=name, index=index) return self - def to_dict(self, depth=-1, ordered=True, **kwargs): + def to_dict( + self, + depth: int = -1, + ordered: bool = True, + **kwargs, + ) -> Union[dict, OrderedDict]: """Returns a dict representation of the object.""" + dict_fun: Type[Union[dict, OrderedDict]] if ordered: dict_fun = OrderedDict else: @@ -162,22 +188,27 @@ def to_dict(self, depth=-1, ordered=True, **kwargs): ) # noqa return out - def to_json(self, depth=-1, **kwargs): + def to_json(self, depth: int = -1, **kwargs) -> str: """Returns a JSON representation of the object.""" return json.dumps(self.to_dict(depth=depth, ordered=True), **kwargs) - def get_root(self): + def get_root(self) -> "Element": """Returns the root of the elements tree.""" if self._parent is None: return self else: return self._parent.get_root() - def render(self, **kwargs): + def render(self, **kwargs) -> str: """Renders the HTML representation of the element.""" return self._template.render(this=self, kwargs=kwargs) - def save(self, outfile, close_file=True, **kwargs): + def save( + self, + outfile: Union[str, bytes, Path, BinaryIO], + close_file: bool = True, + **kwargs, + ): """Saves an Element into a file. Parameters @@ -187,6 +218,7 @@ def save(self, outfile, close_file=True, **kwargs): close_file : bool, default True Whether the file has to be closed after write. """ + fid: BinaryIO if isinstance(outfile, (str, bytes, Path)): fid = open(outfile, "wb") else: @@ -202,15 +234,27 @@ def save(self, outfile, close_file=True, **kwargs): class Link(Element): """An abstract class for embedding a link in the HTML.""" - def get_code(self): + def __init__(self, url: str, download: bool = False): + super().__init__() + self.url = url + self.code: Optional[bytes] = None + if download: + self.get_code() + + def get_code(self) -> bytes: """Opens the link and returns the response's content.""" if self.code is None: self.code = urlopen(self.url).read() return self.code - def to_dict(self, depth=-1, **kwargs): + def to_dict( + self, + depth: int = -1, + ordered: bool = True, + **kwargs, + ) -> Union[dict, OrderedDict]: """Returns a dict representation of the object.""" - out = super().to_dict(depth=-1, **kwargs) + out = super().to_dict(depth=depth, ordered=ordered, **kwargs) out["url"] = self.url return out @@ -235,13 +279,9 @@ class JavascriptLink(Link): "{% endif %}", ) - def __init__(self, url, download=False): - super().__init__() + def __init__(self, url: str, download: bool = False): + super().__init__(url=url, download=download) self._name = "JavascriptLink" - self.url = url - self.code = None - if download: - self.get_code() class CssLink(Link): @@ -264,13 +304,9 @@ class CssLink(Link): "{% endif %}", ) - def __init__(self, url, download=False): - super().__init__() + def __init__(self, url: str, download: bool = False): + super().__init__(url=url, download=download) self._name = "CssLink" - self.url = url - self.code = None - if download: - self.get_code() class Figure(Element): @@ -314,11 +350,11 @@ class Figure(Element): def __init__( self, - width="100%", - height=None, - ratio="60%", - title=None, - figsize=None, + width: str = "100%", + height: Optional[str] = None, + ratio: str = "60%", + title: Optional[str] = None, + figsize: Optional[Tuple[int, int]] = None, ): super().__init__() self._name = "Figure" @@ -346,7 +382,12 @@ def __init__( name="meta_http", ) - def to_dict(self, depth=-1, **kwargs): + def to_dict( + self, + depth: int = -1, + ordered: bool = True, + **kwargs, + ) -> Union[dict, OrderedDict]: """Returns a dict representation of the object.""" out = super().to_dict(depth=depth, **kwargs) out["header"] = self.header.to_dict(depth=depth - 1, **kwargs) @@ -354,17 +395,17 @@ def to_dict(self, depth=-1, **kwargs): out["script"] = self.script.to_dict(depth=depth - 1, **kwargs) return out - def get_root(self): + def get_root(self) -> "Figure": """Returns the root of the elements tree.""" return self - def render(self, **kwargs): + def render(self, **kwargs) -> str: """Renders the HTML representation of the element.""" for name, child in self._children.items(): child.render(**kwargs) return self._template.render(this=self, kwargs=kwargs) - def _repr_html_(self, **kwargs): + def _repr_html_(self, **kwargs) -> str: """Displays the Figure in a Jupyter notebook.""" html = escape(self.render(**kwargs)) if self.height is None: @@ -387,7 +428,7 @@ def _repr_html_(self, **kwargs): ).format(html=html, width=self.width, height=self.height) return iframe - def add_subplot(self, x, y, n, margin=0.05): + def add_subplot(self, x: int, y: int, n: int, margin: float = 0.05) -> "Div": """Creates a div child subplot in a matplotlib.figure.add_subplot style. Parameters @@ -398,8 +439,11 @@ def add_subplot(self, x, y, n, margin=0.05): The number of columns in the grid. n : int The cell number in the grid, counted from 1 to x*y. + margin : float, default 0.05 + Factor to add to the left, top, width and height parameters. - Example: + Example + ------- >>> fig.add_subplot(3, 2, 5) # Create a div in the 5th cell of a 3rows x 2columns grid(bottom-left corner). @@ -447,9 +491,15 @@ class Html(Element): '
' # noqa "{% if this.script %}{{this.data}}{% else %}{{this.data|e}}{% endif %}
", - ) # noqa + ) - def __init__(self, data, script=False, width="100%", height="100%"): + def __init__( + self, + data: str, + script: bool = False, + width: TypeParseSize = "100%", + height: TypeParseSize = "100%", + ): super().__init__() self._name = "Html" self.script = script @@ -494,11 +544,11 @@ class Div(Figure): def __init__( self, - width="100%", - height="100%", - left="0%", - top="0%", - position="relative", + width: TypeParseSize = "100%", + height: TypeParseSize = "100%", + left: TypeParseSize = "0%", + top: TypeParseSize = "0%", + position: str = "relative", ): super(Figure, self).__init__() self._name = "Div" @@ -522,20 +572,17 @@ def __init__( self.html._parent = self self.script._parent = self - def get_root(self): + def get_root(self) -> "Div": """Returns the root of the elements tree.""" return self - def render(self, **kwargs): + def render(self, **kwargs) -> str: """Renders the HTML representation of the element.""" figure = self._parent assert isinstance(figure, Figure), ( "You cannot render this Element " "if it is not in a Figure." ) - for name, element in self._children.items(): - element.render(**kwargs) - for name, element in self.header._children.items(): figure.header.add_child(element, name=name) @@ -554,14 +601,16 @@ def render(self, **kwargs): if script is not None: figure.script.add_child(Element(script(self, kwargs)), name=self.get_name()) - def _repr_html_(self, **kwargs): + return figure.render() + + def _repr_html_(self, **kwargs) -> str: """Displays the Div in a Jupyter notebook.""" if self._parent is None: self.add_to(Figure()) - out = self._parent._repr_html_(**kwargs) + out = self._parent._repr_html_(**kwargs) # type: ignore self._parent = None else: - out = self._parent._repr_html_(**kwargs) + out = self._parent._repr_html_(**kwargs) # type: ignore return out @@ -588,7 +637,14 @@ class IFrame(Element): width="600px", height="300px". """ - def __init__(self, html=None, width="100%", height=None, ratio="60%", figsize=None): + def __init__( + self, + html: Optional[Union[str, Element]] = None, + width: str = "100%", + height: Optional[str] = None, + ratio: str = "60%", + figsize: Optional[Tuple[int, int]] = None, + ): super().__init__() self._name = "IFrame" @@ -599,12 +655,12 @@ def __init__(self, html=None, width="100%", height=None, ratio="60%", figsize=No self.width = str(60 * figsize[0]) + "px" self.height = str(60 * figsize[1]) + "px" - if isinstance(html, str) or isinstance(html, bytes): + if isinstance(html, str): self.add_child(Element(html)) elif html is not None: self.add_child(html) - def render(self, **kwargs): + def render(self, **kwargs) -> str: """Renders the HTML representation of the element.""" html = super().render(**kwargs) html = "data:text/html;charset=utf-8;base64," + base64.b64encode( @@ -658,7 +714,7 @@ def __init__(self): super().__init__() self._name = "MacroElement" - def render(self, **kwargs): + def render(self, **kwargs) -> str: """Renders the HTML representation of the element.""" figure = self.get_root() assert isinstance(figure, Figure), ( @@ -677,5 +733,4 @@ def render(self, **kwargs): if script is not None: figure.script.add_child(Element(script(self, kwargs)), name=self.get_name()) - for name, element in self._children.items(): - element.render(**kwargs) + return figure.render() diff --git a/branca/utilities.py b/branca/utilities.py index 4bb3540..445b852 100644 --- a/branca/utilities.py +++ b/branca/utilities.py @@ -14,14 +14,14 @@ import struct import typing import zlib -from typing import Any, Callable, Union +from typing import Any, Callable, Tuple, Union from jinja2 import Environment, PackageLoader try: import numpy as np except ImportError: - np = None + np = None # type: ignore if typing.TYPE_CHECKING: from branca.colormap import ColorMap @@ -30,6 +30,9 @@ rootpath = os.path.abspath(os.path.dirname(__file__)) +TypeParseSize = Union[int, float, str, Tuple[float, str]] + + def get_templates(): """Get Jinja templates.""" return Environment(loader=PackageLoader("branca", "templates")) @@ -371,7 +374,7 @@ def _camelify(out): ) # noqa -def _parse_size(value): +def _parse_size(value: TypeParseSize) -> Tuple[float, str]: if isinstance(value, (int, float)): return float(value), "px" elif isinstance(value, str): From 8f02e42882a19257966f099292f46d5ce1670109 Mon Sep 17 00:00:00 2001 From: Conengmo <33519926+Conengmo@users.noreply.github.com> Date: Tue, 24 Oct 2023 17:22:26 +0200 Subject: [PATCH 06/19] utilities.py wip --- branca/colormap.py | 6 +++++- branca/utilities.py | 24 +++++++++++------------- 2 files changed, 16 insertions(+), 14 deletions(-) diff --git a/branca/colormap.py b/branca/colormap.py index 6f53410..1ddfa87 100644 --- a/branca/colormap.py +++ b/branca/colormap.py @@ -112,10 +112,14 @@ def __init__( self.vmin = vmin self.vmax = vmax self.caption = caption +<<<<<<< HEAD self.text_color = text_color self.index = [vmin, vmax] +======= + self.index: List[float] = [vmin, vmax] +>>>>>>> ec6251d... utilities.py wip self.max_labels = max_labels - self.tick_labels: Optional[Sequence[float]] = None + self.tick_labels: Optional[Sequence[Union[float, str]]] = None self.width = 450 self.height = 40 diff --git a/branca/utilities.py b/branca/utilities.py index 445b852..030ac97 100644 --- a/branca/utilities.py +++ b/branca/utilities.py @@ -14,7 +14,7 @@ import struct import typing import zlib -from typing import Any, Callable, Tuple, Union +from typing import Any, Callable, Tuple, Union, List, Sequence from jinja2 import Environment, PackageLoader @@ -27,26 +27,29 @@ from branca.colormap import ColorMap -rootpath = os.path.abspath(os.path.dirname(__file__)) +rootpath: str = os.path.abspath(os.path.dirname(__file__)) TypeParseSize = Union[int, float, str, Tuple[float, str]] -def get_templates(): +def get_templates() -> Environment: """Get Jinja templates.""" return Environment(loader=PackageLoader("branca", "templates")) -def legend_scaler(legend_values, max_labels=10.0): +def legend_scaler( + legend_values: Sequence[float], max_labels: int = 10 +) -> list[Union[float, str]]: """ Downsamples the number of legend values so that there isn't a collision of text on the legend colorbar (within reason). The colorbar seems to support ~10 entries as a maximum. """ + legend_ticks: List[Union[float, str]] if len(legend_values) < max_labels: - legend_ticks = legend_values + legend_ticks = list(legend_values) else: spacer = int(math.ceil(len(legend_values) / max_labels)) legend_ticks = [] @@ -56,16 +59,11 @@ def legend_scaler(legend_values, max_labels=10.0): return legend_ticks -def linear_gradient(hexList, nColors): +def linear_gradient(hexList: List[str], nColors: int) -> List[str]: """ Given a list of hexcode values, will return a list of length nColors where the colors are linearly interpolated between the (r, g, b) tuples that are given. - - Examples - -------- - >>> linear_gradient([(0, 0, 0), (255, 0, 0), (255, 255, 0)], 100) - """ def _scale(start, finish, length, i): @@ -83,7 +81,7 @@ def _scale(start, finish, length, i): thex = "0" + thex return thex - allColors = [] + allColors: List[str] = [] # Separate (R, G, B) pairs. for start, end in zip(hexList[:-1], hexList[1:]): # Linearly interpolate between pair of hex ###### values and @@ -96,7 +94,7 @@ def _scale(start, finish, length, i): allColors.append("".join(["#", r, g, b])) # Pick only nColors colors from the total list. - result = [] + result: List[str] = [] for counter in range(nColors): fraction = float(counter) / (nColors - 1) index = int(fraction * (len(allColors) - 1)) From 5b009bd5d57b8d49fb2681f67e670b208dbf5030 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Tue, 24 Oct 2023 15:23:25 +0000 Subject: [PATCH 07/19] [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci --- branca/utilities.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/branca/utilities.py b/branca/utilities.py index 030ac97..7937200 100644 --- a/branca/utilities.py +++ b/branca/utilities.py @@ -14,7 +14,7 @@ import struct import typing import zlib -from typing import Any, Callable, Tuple, Union, List, Sequence +from typing import Any, Callable, List, Sequence, Tuple, Union from jinja2 import Environment, PackageLoader @@ -39,7 +39,7 @@ def get_templates() -> Environment: def legend_scaler( - legend_values: Sequence[float], max_labels: int = 10 + legend_values: Sequence[float], max_labels: int = 10, ) -> list[Union[float, str]]: """ Downsamples the number of legend values so that there isn't a collision From b3cc3ee9c27de7668a8e206fdde385e8a75ed644 Mon Sep 17 00:00:00 2001 From: Frank Anema <33519926+Conengmo@users.noreply.github.com> Date: Tue, 24 Oct 2023 17:35:33 +0200 Subject: [PATCH 08/19] Update branca/utilities.py --- branca/utilities.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/branca/utilities.py b/branca/utilities.py index 7937200..82d6ae0 100644 --- a/branca/utilities.py +++ b/branca/utilities.py @@ -40,7 +40,7 @@ def get_templates() -> Environment: def legend_scaler( legend_values: Sequence[float], max_labels: int = 10, -) -> list[Union[float, str]]: +) -> List[Union[float, str]]: """ Downsamples the number of legend values so that there isn't a collision of text on the legend colorbar (within reason). The colorbar seems to From 58346650e331bb5febbe6780e6bfc956907e0957 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Tue, 24 Oct 2023 15:35:42 +0000 Subject: [PATCH 09/19] [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci --- branca/utilities.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/branca/utilities.py b/branca/utilities.py index 82d6ae0..ffc1f25 100644 --- a/branca/utilities.py +++ b/branca/utilities.py @@ -39,7 +39,8 @@ def get_templates() -> Environment: def legend_scaler( - legend_values: Sequence[float], max_labels: int = 10, + legend_values: Sequence[float], + max_labels: int = 10, ) -> List[Union[float, str]]: """ Downsamples the number of legend values so that there isn't a collision From a7e7045d599e1d9c480664ef7ded510b7eef618d Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Wed, 3 Apr 2024 18:00:11 +0200 Subject: [PATCH 10/19] Restore Div and MacroElement --- branca/element.py | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/branca/element.py b/branca/element.py index e8520f7..9c87d2b 100644 --- a/branca/element.py +++ b/branca/element.py @@ -583,6 +583,9 @@ def render(self, **kwargs) -> str: "You cannot render this Element " "if it is not in a Figure." ) + for name, element in self._children.items(): + element.render(**kwargs) + for name, element in self.header._children.items(): figure.header.add_child(element, name=name) @@ -601,8 +604,6 @@ def render(self, **kwargs) -> str: if script is not None: figure.script.add_child(Element(script(self, kwargs)), name=self.get_name()) - return figure.render() - def _repr_html_(self, **kwargs) -> str: """Displays the Div in a Jupyter notebook.""" if self._parent is None: @@ -733,4 +734,5 @@ def render(self, **kwargs) -> str: if script is not None: figure.script.add_child(Element(script(self, kwargs)), name=self.get_name()) - return figure.render() + for name, element in self._children.items(): + element.render(**kwargs) From 6faee4240e1a2629e2472ecafcce6eee1290abb3 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Thu, 4 Apr 2024 20:30:51 +0200 Subject: [PATCH 11/19] Update test_mypy.yml --- .github/workflows/test_mypy.yml | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/.github/workflows/test_mypy.yml b/.github/workflows/test_mypy.yml index 6a68406..96715e0 100644 --- a/.github/workflows/test_mypy.yml +++ b/.github/workflows/test_mypy.yml @@ -1,6 +1,10 @@ name: Mypy type hint checks -on: [push, pull_request] +on: + pull_request: + push: + branches: + - main jobs: run: From 59500f84c1b6349d464cda1e9d0392b5791e4385 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Thu, 4 Apr 2024 20:48:23 +0200 Subject: [PATCH 12/19] Update colormap.py --- branca/colormap.py | 22 ++++++++++------------ 1 file changed, 10 insertions(+), 12 deletions(-) diff --git a/branca/colormap.py b/branca/colormap.py index 1ddfa87..fc13b19 100644 --- a/branca/colormap.py +++ b/branca/colormap.py @@ -56,19 +56,17 @@ def _color_float_to_int(x: float) -> int: def _parse_color(x: Union[tuple, list, str]) -> TypeRGBAFloats: - color_tuple: TypeRGBAFloats if isinstance(x, (tuple, list)): - color_tuple = tuple(tuple(x) + (1.0,))[:4] # type: ignore + return tuple(tuple(x) + (1.0,))[:4] # type: ignore elif isinstance(x, str) and _is_hex(x): - color_tuple = _parse_hex(x) + return _parse_hex(x) elif isinstance(x, str): cname = _cnames.get(x.lower(), None) if cname is None: raise ValueError(f"Unknown color {cname!r}.") - color_tuple = _parse_hex(cname) + return _parse_hex(cname) else: raise ValueError(f"Unrecognized color code {x!r}") - return color_tuple def _base(x: float) -> float: @@ -275,10 +273,14 @@ class LinearColormap(ColorMap): vmax : float, default 1. The maximal value for the colormap. Values higher than `vmax` will be bound directly to `colors[-1]`. +<<<<<<< HEAD caption: str A caption to draw with the colormap. text_color: str, default "black" The color for the text. +======= + caption : str, default "" +>>>>>>> a7239e8... Update colormap.py max_labels : int, default 10 Maximum number of legend tick labels tick_labels: list of floats, default None @@ -302,7 +304,7 @@ def __init__( text_color=text_color, max_labels=max_labels, ) - self.tick_labels = tick_labels + self.tick_labels: Optional[Sequence[float]] = tick_labels n = len(colors) if n < 2: @@ -339,7 +341,7 @@ def to_step( n: Optional[int] = None, index: Optional[Sequence[float]] = None, data: Optional[Sequence[float]] = None, - method: Optional[str] = None, + method: str = "linear", quantiles: Optional[Sequence[float]] = None, round_method: Optional[str] = None, max_labels: int = 10, @@ -404,11 +406,7 @@ def to_step( max_ = max(data) min_ = min(data) scaled_cm = self.scale(vmin=min_, vmax=max_) - method = ( - "quantiles" - if quantiles is not None - else method if method is not None else "linear" - ) + method = "quantiles" if quantiles is not None else method if method.lower().startswith("lin"): if n is None: raise ValueError(msg) From 9323d12ba734b8e48e19fd5245da8b296a83e831 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Thu, 4 Apr 2024 20:57:57 +0200 Subject: [PATCH 13/19] Update element.py --- branca/element.py | 16 +++++++--------- 1 file changed, 7 insertions(+), 9 deletions(-) diff --git a/branca/element.py b/branca/element.py index 9c87d2b..cda7e96 100644 --- a/branca/element.py +++ b/branca/element.py @@ -184,8 +184,8 @@ def to_dict( [ (name, child.to_dict(depth=depth - 1)) for name, child in self._children.items() - ], - ) # noqa + ] + ) return out def to_json(self, depth: int = -1, **kwargs) -> str: @@ -554,8 +554,8 @@ def __init__( self._name = "Div" # Size Parameters. - self.width = _parse_size(width) - self.height = _parse_size(height) + self.width = _parse_size(width) # type: ignore + self.height = _parse_size(height) # type: ignore self.left = _parse_size(left) self.top = _parse_size(top) self.position = position @@ -576,7 +576,7 @@ def get_root(self) -> "Div": """Returns the root of the elements tree.""" return self - def render(self, **kwargs) -> str: + def render(self, **kwargs): """Renders the HTML representation of the element.""" figure = self._parent assert isinstance(figure, Figure), ( @@ -666,9 +666,7 @@ def render(self, **kwargs) -> str: html = super().render(**kwargs) html = "data:text/html;charset=utf-8;base64," + base64.b64encode( html.encode("utf8"), - ).decode( - "utf8", - ) # noqa + ).decode("utf8") if self.height is None: iframe = ( @@ -715,7 +713,7 @@ def __init__(self): super().__init__() self._name = "MacroElement" - def render(self, **kwargs) -> str: + def render(self, **kwargs): """Renders the HTML representation of the element.""" figure = self.get_root() assert isinstance(figure, Figure), ( From ef91cd9e26b07b99820396147bd957f5ea72481c Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Thu, 4 Apr 2024 21:10:05 +0200 Subject: [PATCH 14/19] Update utilities.py --- branca/utilities.py | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/branca/utilities.py b/branca/utilities.py index ffc1f25..2d9afaa 100644 --- a/branca/utilities.py +++ b/branca/utilities.py @@ -14,7 +14,7 @@ import struct import typing import zlib -from typing import Any, Callable, List, Sequence, Tuple, Union +from typing import Any, Callable, List, Sequence, Tuple, Union, Optional from jinja2 import Environment, PackageLoader @@ -103,7 +103,7 @@ def _scale(start, finish, length, i): return result -def color_brewer(color_code, n=6): +def color_brewer(color_code: str, n: int = 6) -> List[str]: """ Generate a colorbrewer color scheme of length 'len', type 'scheme. Live examples can be seen at http://colorbrewer2.org/ @@ -200,7 +200,7 @@ def color_brewer(color_code, n=6): return color_scheme -def image_to_url(image, colormap=None, origin="upper"): +def image_to_url(image: Any, colormap: Union["ColorMap", Callable, None] = None, origin: str = "upper") -> str: """Infers the type of an image argument and transforms it into a URL. Parameters @@ -214,7 +214,7 @@ def image_to_url(image, colormap=None, origin="upper"): origin : ['upper' | 'lower'], optional, default 'upper' Place the [0, 0] index of the array in the upper left or lower left corner of the axes. - colormap : callable, used only for `mono` image. + colormap : ColorMap or callable, used only for `mono` image. Function of the form [x -> (r,g,b)] or [x -> (r,g,b,a)] for transforming a mono image into RGB. It must output iterables of length 3 or 4, with values between @@ -346,7 +346,7 @@ def png_pack(png_tag, data): ) -def _camelify(out): +def _camelify(out: str) -> str: return ( ( "".join( @@ -355,12 +355,12 @@ def _camelify(out): "_" + x.lower() if i < len(out) - 1 and x.isupper() - and out[i + 1].islower() # noqa + and out[i + 1].islower() else ( x.lower() + "_" if i < len(out) - 1 and x.islower() - and out[i + 1].isupper() # noqa + and out[i + 1].isupper() else x.lower() ) ) @@ -370,7 +370,7 @@ def _camelify(out): ) .lstrip("_") .replace("__", "_") - ) # noqa + ) def _parse_size(value: TypeParseSize) -> Tuple[float, str]: @@ -423,7 +423,7 @@ def _locations_tolist(x): return x -def none_min(x, y): +def none_min(x: Optional[float], y: Optional[float]) -> Optional[float]: if x is None: return y elif y is None: @@ -432,7 +432,7 @@ def none_min(x, y): return min(x, y) -def none_max(x, y): +def none_max(x: Optional[float], y: Optional[float]) -> Optional[float]: if x is None: return y elif y is None: @@ -441,7 +441,7 @@ def none_max(x, y): return max(x, y) -def iter_points(x): +def iter_points(x: Union[List, Tuple]) -> list: """Iterates over a list representing a feature, and returns a list of points, whatever the shape of the array (Point, MultiPolyline, etc). """ From 00f9a0e7c53b469c5a69ef4b14dd94d7e52025bd Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Thu, 4 Apr 2024 19:10:17 +0000 Subject: [PATCH 15/19] [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci --- branca/element.py | 2 +- branca/utilities.py | 16 ++++++++-------- 2 files changed, 9 insertions(+), 9 deletions(-) diff --git a/branca/element.py b/branca/element.py index cda7e96..a1e005a 100644 --- a/branca/element.py +++ b/branca/element.py @@ -184,7 +184,7 @@ def to_dict( [ (name, child.to_dict(depth=depth - 1)) for name, child in self._children.items() - ] + ], ) return out diff --git a/branca/utilities.py b/branca/utilities.py index 2d9afaa..14a1d46 100644 --- a/branca/utilities.py +++ b/branca/utilities.py @@ -14,7 +14,7 @@ import struct import typing import zlib -from typing import Any, Callable, List, Sequence, Tuple, Union, Optional +from typing import Any, Callable, List, Optional, Sequence, Tuple, Union from jinja2 import Environment, PackageLoader @@ -200,7 +200,11 @@ def color_brewer(color_code: str, n: int = 6) -> List[str]: return color_scheme -def image_to_url(image: Any, colormap: Union["ColorMap", Callable, None] = None, origin: str = "upper") -> str: +def image_to_url( + image: Any, + colormap: Union["ColorMap", Callable, None] = None, + origin: str = "upper", +) -> str: """Infers the type of an image argument and transforms it into a URL. Parameters @@ -353,14 +357,10 @@ def _camelify(out: str) -> str: [ ( "_" + x.lower() - if i < len(out) - 1 - and x.isupper() - and out[i + 1].islower() + if i < len(out) - 1 and x.isupper() and out[i + 1].islower() else ( x.lower() + "_" - if i < len(out) - 1 - and x.islower() - and out[i + 1].isupper() + if i < len(out) - 1 and x.islower() and out[i + 1].isupper() else x.lower() ) ) From d59cee901bd1768385e8c79900cf9a0c04c7b925 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Sat, 18 May 2024 10:30:29 +0200 Subject: [PATCH 16/19] error after rebase --- branca/colormap.py | 4 ---- 1 file changed, 4 deletions(-) diff --git a/branca/colormap.py b/branca/colormap.py index fc13b19..e3b8861 100644 --- a/branca/colormap.py +++ b/branca/colormap.py @@ -110,12 +110,8 @@ def __init__( self.vmin = vmin self.vmax = vmax self.caption = caption -<<<<<<< HEAD self.text_color = text_color - self.index = [vmin, vmax] -======= self.index: List[float] = [vmin, vmax] ->>>>>>> ec6251d... utilities.py wip self.max_labels = max_labels self.tick_labels: Optional[Sequence[Union[float, str]]] = None From e21b6d4dc5c5510da8c20767d871d07f27253b70 Mon Sep 17 00:00:00 2001 From: "pre-commit-ci[bot]" <66853113+pre-commit-ci[bot]@users.noreply.github.com> Date: Sat, 18 May 2024 08:30:56 +0000 Subject: [PATCH 17/19] [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci --- branca/colormap.py | 68 +++++++++++++++++++++++----------------------- 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/branca/colormap.py b/branca/colormap.py index e3b8861..83d5a1f 100644 --- a/branca/colormap.py +++ b/branca/colormap.py @@ -245,42 +245,42 @@ def _repr_html_(self) -> str: class LinearColormap(ColorMap): """Creates a ColorMap based on linear interpolation of a set of colors - over a given index. + over a given index. - Parameters - ---------- + Parameters + ---------- - colors : list-like object with at least two colors. - The set of colors to be used for interpolation. - Colors can be provided in the form: - * tuples of RGBA ints between 0 and 255 (e.g: `(255, 255, 0)` or - `(255, 255, 0, 255)`) - * tuples of RGBA floats between 0. and 1. (e.g: `(1.,1.,0.)` or - `(1., 1., 0., 1.)`) - * HTML-like string (e.g: `"#ffff00`) - * a color name or shortcut (e.g: `"y"` or `"yellow"`) - index : list of floats, default None - The values corresponding to each color. - It has to be sorted, and have the same length as `colors`. - If None, a regular grid between `vmin` and `vmax` is created. - vmin : float, default 0. - The minimal value for the colormap. - Values lower than `vmin` will be bound directly to `colors[0]`. - vmax : float, default 1. - The maximal value for the colormap. - Values higher than `vmax` will be bound directly to `colors[-1]`. -<<<<<<< HEAD - caption: str - A caption to draw with the colormap. - text_color: str, default "black" - The color for the text. -======= - caption : str, default "" ->>>>>>> a7239e8... Update colormap.py - max_labels : int, default 10 - Maximum number of legend tick labels - tick_labels: list of floats, default None - If given, used as the positions of ticks.""" + colors : list-like object with at least two colors. + The set of colors to be used for interpolation. + Colors can be provided in the form: + * tuples of RGBA ints between 0 and 255 (e.g: `(255, 255, 0)` or + `(255, 255, 0, 255)`) + * tuples of RGBA floats between 0. and 1. (e.g: `(1.,1.,0.)` or + `(1., 1., 0., 1.)`) + * HTML-like string (e.g: `"#ffff00`) + * a color name or shortcut (e.g: `"y"` or `"yellow"`) + index : list of floats, default None + The values corresponding to each color. + It has to be sorted, and have the same length as `colors`. + If None, a regular grid between `vmin` and `vmax` is created. + vmin : float, default 0. + The minimal value for the colormap. + Values lower than `vmin` will be bound directly to `colors[0]`. + vmax : float, default 1. + The maximal value for the colormap. + Values higher than `vmax` will be bound directly to `colors[-1]`. + <<<<<<< HEAD + caption: str + A caption to draw with the colormap. + text_color: str, default "black" + The color for the text. + ======= + caption : str, default "" + >>>>>>> a7239e8... Update colormap.py + max_labels : int, default 10 + Maximum number of legend tick labels + tick_labels: list of floats, default None + If given, used as the positions of ticks.""" def __init__( self, From 810a860f11fceba839ac95465296adb65616093e Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Sat, 18 May 2024 10:46:28 +0200 Subject: [PATCH 18/19] Revert "[pre-commit.ci] auto fixes from pre-commit.com hooks" This reverts commit e21b6d4dc5c5510da8c20767d871d07f27253b70. --- branca/colormap.py | 68 +++++++++++++++++++++++----------------------- 1 file changed, 34 insertions(+), 34 deletions(-) diff --git a/branca/colormap.py b/branca/colormap.py index 83d5a1f..e3b8861 100644 --- a/branca/colormap.py +++ b/branca/colormap.py @@ -245,42 +245,42 @@ def _repr_html_(self) -> str: class LinearColormap(ColorMap): """Creates a ColorMap based on linear interpolation of a set of colors - over a given index. + over a given index. - Parameters - ---------- + Parameters + ---------- - colors : list-like object with at least two colors. - The set of colors to be used for interpolation. - Colors can be provided in the form: - * tuples of RGBA ints between 0 and 255 (e.g: `(255, 255, 0)` or - `(255, 255, 0, 255)`) - * tuples of RGBA floats between 0. and 1. (e.g: `(1.,1.,0.)` or - `(1., 1., 0., 1.)`) - * HTML-like string (e.g: `"#ffff00`) - * a color name or shortcut (e.g: `"y"` or `"yellow"`) - index : list of floats, default None - The values corresponding to each color. - It has to be sorted, and have the same length as `colors`. - If None, a regular grid between `vmin` and `vmax` is created. - vmin : float, default 0. - The minimal value for the colormap. - Values lower than `vmin` will be bound directly to `colors[0]`. - vmax : float, default 1. - The maximal value for the colormap. - Values higher than `vmax` will be bound directly to `colors[-1]`. - <<<<<<< HEAD - caption: str - A caption to draw with the colormap. - text_color: str, default "black" - The color for the text. - ======= - caption : str, default "" - >>>>>>> a7239e8... Update colormap.py - max_labels : int, default 10 - Maximum number of legend tick labels - tick_labels: list of floats, default None - If given, used as the positions of ticks.""" + colors : list-like object with at least two colors. + The set of colors to be used for interpolation. + Colors can be provided in the form: + * tuples of RGBA ints between 0 and 255 (e.g: `(255, 255, 0)` or + `(255, 255, 0, 255)`) + * tuples of RGBA floats between 0. and 1. (e.g: `(1.,1.,0.)` or + `(1., 1., 0., 1.)`) + * HTML-like string (e.g: `"#ffff00`) + * a color name or shortcut (e.g: `"y"` or `"yellow"`) + index : list of floats, default None + The values corresponding to each color. + It has to be sorted, and have the same length as `colors`. + If None, a regular grid between `vmin` and `vmax` is created. + vmin : float, default 0. + The minimal value for the colormap. + Values lower than `vmin` will be bound directly to `colors[0]`. + vmax : float, default 1. + The maximal value for the colormap. + Values higher than `vmax` will be bound directly to `colors[-1]`. +<<<<<<< HEAD + caption: str + A caption to draw with the colormap. + text_color: str, default "black" + The color for the text. +======= + caption : str, default "" +>>>>>>> a7239e8... Update colormap.py + max_labels : int, default 10 + Maximum number of legend tick labels + tick_labels: list of floats, default None + If given, used as the positions of ticks.""" def __init__( self, From 26f7a37a69202722c74a4772a7b1c196a14f0478 Mon Sep 17 00:00:00 2001 From: Frank <33519926+Conengmo@users.noreply.github.com> Date: Sat, 18 May 2024 10:46:37 +0200 Subject: [PATCH 19/19] another rebase artefact :/ --- branca/colormap.py | 4 ---- 1 file changed, 4 deletions(-) diff --git a/branca/colormap.py b/branca/colormap.py index e3b8861..2e76ba5 100644 --- a/branca/colormap.py +++ b/branca/colormap.py @@ -269,14 +269,10 @@ class LinearColormap(ColorMap): vmax : float, default 1. The maximal value for the colormap. Values higher than `vmax` will be bound directly to `colors[-1]`. -<<<<<<< HEAD caption: str A caption to draw with the colormap. text_color: str, default "black" The color for the text. -======= - caption : str, default "" ->>>>>>> a7239e8... Update colormap.py max_labels : int, default 10 Maximum number of legend tick labels tick_labels: list of floats, default None