Shape2Force / utils /display.py
kaveh's picture
optimised, removed dead codes
fd5af45
"""Display utilities for heatmaps and colormaps."""
import numpy as np
import cv2
from config.constants import COLORMAPS, COLORMAP_N_SAMPLES
def cv_colormap_to_plotly_colorscale(colormap_name, n_samples=None):
"""Build a Plotly colorscale from OpenCV colormap so UI matches download/PDF exactly."""
n = n_samples or COLORMAP_N_SAMPLES
cv2_cmap = COLORMAPS.get(colormap_name, cv2.COLORMAP_JET)
gradient = np.linspace(0, 255, n, dtype=np.uint8).reshape(1, -1)
rgb = cv2.applyColorMap(gradient, cv2_cmap)
rgb = cv2.cvtColor(rgb, cv2.COLOR_BGR2RGB)
scale = []
for i in range(n):
r, g, b = rgb[0, i]
scale.append([i / (n - 1), f"rgb({r},{g},{b})"])
return scale
def is_display_range_remapped(display_mode, clip_min, clip_max):
"""
True when the UI uses Range mode with a clip window other than the full [0, 1]
(colorbar tick labels map normalized 0–1 to that interval).
"""
if display_mode != "Range":
return False
lo, hi = float(clip_min), float(clip_max)
return hi > lo and not (lo == 0.0 and hi == 1.0)
def apply_display_scale(heatmap, mode, clip_min=0, clip_max=1, clamp_only=False):
"""
Apply display scaling. Display only—does not change underlying values.
Supports modes used by the app: ``Default`` (clip to [0, 1]) and ``Range``
(window to [clip_min, clip_max]). Unknown ``mode`` is treated like ``Default``.
Parameters
----------
clamp_only : bool
For ``mode == "Range"`` when ``clip_max > clip_min``:
- **False** (default in the app for Range): pixels outside ``[clip_min, clip_max]`` are set
to 0; values inside are linearly mapped to ``[0, 1]`` for the colormap.
- **True**: values are clamped to ``[clip_min, clip_max]`` without rescaling to ``[0, 1]``.
"""
if mode != "Range":
return np.clip(heatmap, 0, 1).astype(np.float32)
# Range
vmin, vmax = float(clip_min), float(clip_max)
if vmax > vmin:
h = heatmap.astype(np.float32)
if clamp_only:
return np.clip(h, vmin, vmax).astype(np.float32)
mask = (h >= vmin) & (h <= vmax)
out = np.zeros_like(h)
out[mask] = (h[mask] - vmin) / (vmax - vmin)
return np.clip(out, 0, 1).astype(np.float32)
return np.clip(heatmap, 0, 1).astype(np.float32)