Caquinhos / Ladrilhos 05

2026-06-06

"""2026-06-06
Caquinhos / Ladrilhos 05
Homenagem aos pisos de caquinhos de São Paulo.
ericof.com|https://ericof.com/en/sketches/2023-05-09
png
Sketch,py5,CreativeCoding
"""

from opensimplex import OpenSimplex
from random import shuffle
from sketches.utils.draw import canvas
from sketches.utils.draw.grade import cria_grade
from sketches.utils.helpers import sketches as helpers

import py5


sketch = helpers.info_for_sketch(__file__, __doc__)

cor_fundo = py5.color(0)
cor_fundo_interna = py5.color(226, 214, 187)

celula_x = 30
celula_y = 30
lado_min_mult = 1.5
lado_max_mult = 2.5
mult_x = 15
mult_y = 20
borda = 0.30

lado_min = int(max(celula_x, celula_y) * lado_min_mult)
lado_max = int(max(celula_x, celula_y) * lado_max_mult)
tamanho = lado_max

noise_generator = OpenSimplex(seed=py5.random_int(10_000))

formas: list[tuple[float, float, int, int, int, py5.Py5Shape]] = []


def par_bagunca(faixa_bagunca: int) -> tuple[float, float]:
    bx = py5.random(-faixa_bagunca, faixa_bagunca)
    by = py5.random(-faixa_bagunca, faixa_bagunca)
    return bx, by


def caco(
    x: float,
    y: float,
    tamanho: int,
    faixa_bagunca: int,
    cor_interna: int,
    cor_externa: int,
) -> py5.Py5Shape:
    s = py5.create_shape(py5.GROUP)
    bx, by = par_bagunca(faixa_bagunca)
    x0, y0 = x - tamanho / 2 + bx, y - tamanho / 2 + by
    bx, by = par_bagunca(faixa_bagunca)
    x1, y1 = x + tamanho / 2 + bx, y - tamanho / 2 + by
    bx, by = par_bagunca(faixa_bagunca)
    x2, y2 = x + tamanho / 2 + bx, y + tamanho / 2 + by
    bx, by = par_bagunca(faixa_bagunca)
    x3, y3 = x - tamanho / 2 + bx, y + tamanho / 2 + by
    for cor, escala in ((cor_externa, 1 + borda), (cor_interna, 1)):
        forma = py5.create_shape()
        forma.set_fill(cor)
        forma.set_stroke_weight(0)
        with forma.begin_closed_shape():
            forma.vertex(x + (x0 - x) * escala, y + (y0 - y) * escala)
            forma.vertex(x + (x1 - x) * escala, y + (y1 - y) * escala)
            forma.vertex(x + (x2 - x) * escala, y + (y2 - y) * escala)
            forma.vertex(x + (x3 - x) * escala, y + (y3 - y) * escala)
        s.add_child(forma)
    return s


def calcula_pontos(
    largura: int,
    altura: int,
    celula_x: int,
    celula_y: int,
    mult_x: float = 15,
    mult_y: float = 10,
) -> list[tuple[float, float, int]]:
    pontos = []
    for x0, y0 in cria_grade(largura, altura, 0, 0, celula_x, celula_y, True):
        z = py5.random_int(0, 5)
        noise = noise_generator.noise2(x0, y0)
        y = y0 + (mult_y * noise)
        x = x0 + (mult_x * noise)
        pontos.append((x, y, z))
    shuffle(pontos)
    return pontos


def cor_caco(idx: int) -> int:
    cor = py5.color(152, 59, 47)
    if idx % 3 == 0:
        cor = py5.color(0)
    elif idx % 4 == 0:
        cor = py5.color(218, 160, 57)
    return cor


def inicializa():
    global formas, lado_min, lado_max, tamanho
    lado_min = int(max(celula_x, celula_y) * lado_min_mult)
    lado_max = int(max(celula_x, celula_y) * lado_max_mult)
    tamanho = lado_max
    formas = []
    pontos = calcula_pontos(
        *helpers.DIMENSOES.internal, celula_x, celula_y, mult_x, mult_y
    )
    for idx, (x, y, z) in enumerate(pontos):
        largura = py5.random_int(lado_min, lado_max)
        altura = py5.random_int(lado_min, lado_max)
        forma = caco(
            0, 0, tamanho, 40, cor_interna=cor_caco(idx), cor_externa=cor_fundo_interna
        )
        formas.append((x, y, z, largura, altura, forma))


def setup():
    py5.size(*helpers.DIMENSOES.external, py5.P3D)
    inicializa()


def desenha_fundo(cor: int):
    """Desenha o fundo por baixo dos caquinhos."""
    with py5.push():
        py5.fill(cor)
        py5.no_stroke()
        py5.rect(0, 0, *helpers.DIMENSOES.internal)


def draw():
    py5.background(cor_fundo)
    with py5.push():
        py5.translate(*helpers.DIMENSOES.pos_interno, -10)
        desenha_fundo(cor_fundo_interna)
        for x, y, z, largura, altura, forma in formas:
            with py5.push():
                py5.translate(x, y, z)
                py5.shape(forma, celula_x / 2, celula_y / 2, largura, altura)
    msg = (
        f"celula (x - y): {celula_x} - {celula_y} | "
        f"lado (min - max): {lado_min_mult} - {lado_max_mult} | "
        f"borda: {borda:.2f}"
    )
    # Credits and go
    canvas.sketch_frame(
        sketch,
        cor_fundo,
        "large_transparent_white",
        "transparent_white",
        version=2,
        msg=msg,
    )


def key_pressed():
    global lado_min_mult, lado_max_mult, celula_x, celula_y, borda
    key = py5.key
    match key:
        case "r":
            inicializa()
        case ">":
            celula_x += 2
            celula_y += 2
            inicializa()
        case "<":
            celula_x -= 2
            celula_y -= 2
            inicializa()
        case "w":
            borda += 0.1
            inicializa()
        case "s":
            diff = 0.1 if borda > 0.2 else 0.0
            borda -= diff
            inicializa()
        case "+":
            lado_min_mult *= 1.1
            lado_max_mult *= 1.1
            inicializa()
        case "-":
            lado_min_mult *= 0.9
            lado_max_mult *= 0.9
            inicializa()
        case " ":
            save_and_close()


def save_and_close():
    py5.no_loop()
    canvas.save_sketch_image(sketch)
    py5.exit_sketch()


if __name__ == "__main__":
    py5.run_sketch()