# highlight_v2_LM.py
# -*- coding: utf-8 -*-
import numpy as np
import matplotlib.pyplot as plt
from matplotlib.colors import ListedColormap

_top_left_fmt = {
    (8,0),(8,1),(8,2),(8,3),(8,4),(8,5),(8,7),(8,8),
    (0,8),(1,8),(2,8),(3,8),(4,8),(5,8),(7,8)
}
_top_right_fmt = {
    (8,24),(8,23),(8,22),(8,21),(8,20),(8,19),(8,18),(8,17),
    (24,8),(23,8),(22,8),(21,8),(20,8),(19,8),(18,8),(17,8)
}

def is_format_info_module(r, c):
    return (r, c) in _top_left_fmt or (r, c) in _top_right_fmt

def _get_finder_color(local_r, local_c):
    if local_r == 7 or local_c == 7:
        return 0
    if local_r in (0,6) or local_c in (0,6):
        return 1
    if local_r in (1,5) or local_c in (1,5):
        return 0
    return 1

def _get_timing_color(r, c):
    if r == 6:
        return 1 if c % 2 == 0 else 0
    if c == 6:
        return 1 if r % 2 == 0 else 0
    return None

def _get_alignment_color(local_r, local_c):
    if local_r in (0,4) or local_c in (0,4):
        return 1
    if local_r in (1,3) or local_c in (1,3):
        return 0
    return 1

def get_function_module_color(r, c):
    # Finder: top-left
    if 0 <= r <= 7 and 0 <= c <= 7:
        return _get_finder_color(r, c)
    # Finder: top-right
    if 0 <= r <= 7 and 17 <= c <= 24:
        local_c = c - 17
        if local_c == 0:
            return 0
        return _get_finder_color(r, local_c-1)
    # Finder: bottom-left
    if 17 <= r <= 24 and 0 <= c <= 7:
        local_r = r - 17
        if local_r == 0:
            return 0
        return _get_finder_color(local_r-1, c)
    # Timing
    if r == 6 or c == 6:
        return _get_timing_color(r, c)
    # Alignment (v2 only at (18..20,18..20))
    if 16 <= r <= 20 and 16 <= c <= 20:
        return _get_alignment_color(r-16, c-16)
    return None

def get_v2_order_and_counts(ec_level="L"):
    ec_level = ec_level.upper()
    DATA_CW = {"L": 34, "M": 28, "Q": 22, "H": 16}
    TOTAL_CW = 44
    REM_BITS = 7

    data_bits = DATA_CW[ec_level] * 8
    ecc_bits  = (TOTAL_CW - DATA_CW[ec_level]) * 8

    N = 25
    F = [[False]*N for _ in range(N)]
    def reserve(r, c, h, w):
        for i in range(r, r+h):
            for j in range(c, c+w):
                if 0 <= i < N and 0 <= j < N:
                    F[i][j] = True

    def finder(r, c):
        reserve(r, c, 7,7)
        reserve(r-1, c-1, 9,1); reserve(r-1, c+7, 9,1)
        reserve(r-1, c-1, 1,9); reserve(r+7, c-1, 1,9)

    finder(0,0); finder(0,N-7); finder(N-7,0)
    for i in range(N):
        F[6][i] = F[i][6] = True
    reserve(18-2,18-2,5,5)
    F[N-8][8] = True

    # format-info copy1 & copy2
    for c in range(9):
        if c != 6: F[8][c] = True
    for r in range(8):
        if r != 6: F[r][8] = True
    for c in range(N-8, N):
        F[8][c] = True
    for r in range(N-7, N):
        F[r][8] = True

    order, col, upward = [], N-1, True
    while col > 0:
        if col == 6: col -= 1
        rows = range(N-1, -1, -1) if upward else range(N)
        for r in rows:
            for dc in (0, -1):
                c = col + dc
                if not F[r][c]:
                    order.append((r, c))
        upward, col = not upward, col-2

    return order, data_bits, ecc_bits, REM_BITS, N

def draw_v2_highlight(ec_level="L", save_path=None):
    order, db, eb, rb, N = get_v2_order_and_counts(ec_level)
    data_set = set(order[:db])
    ecc_set  = set(order[db:db+eb])
    rem_set  = set(order[db+eb:db+eb+rb])

    mat = np.zeros((N, N), dtype=int)
    for r in range(N):
        for c in range(N):
            if is_format_info_module(r, c):
                mat[r, c] = 2         # Blue
            else:
                fc = get_function_module_color(r, c)
                if fc is not None:
                    mat[r, c] = 1 if fc == 1 else 0  # Black or white background
                elif (r, c) in data_set:
                elif (r, c) in ecc_set or (r, c) in rem_set:
                else:
                    mat[r, c] = 0     # White

    # Color map: 0=white, 1=black, 2=blue, 3=pink, 4=green
    cmap = ListedColormap(["white", "black", "#add8e6", "pink", "#c8f6c7"])
    plt.figure(figsize=(5,5))
    plt.imshow(mat, cmap=cmap, vmin=0, vmax=4, interpolation="nearest")
    plt.axis("off")
    plt.tight_layout()

    if save_path is None:
        save_path = f"v2_{ec_level}_highlight.svg"
    plt.savefig(save_path, dpi=150)
    plt.close()
    print(f"Saved: {save_path}")


if __name__ == "__main__":
    # Output both L and M
    for level in ("L", "M"):
        draw_v2_highlight(ec_level=level)
