import torch


def matones(n: int):
    return torch.ones((n, n))


def gridones(n: int):
    eye = torch.eye(n).flatten()
    return torch.outer(eye, eye)


def is_projector(mat: torch.Tensor):
    frob_norm = torch.sum((mat @ mat - mat) ** 2)
    eye = torch.eye(mat.shape[0], device=mat.device, dtype=mat.dtype)
    frob_skew_norm = torch.sum((mat @ eye - eye @ mat) ** 2)
    evals = torch.linalg.eigvalsh(mat)
    return frob_norm, frob_skew_norm, evals


def positive_definite(
    gen: torch.Generator, n: int, radius: float = 100.0
) -> torch.Tensor:
    v = orthogonal_matrix(gen, n)
    d = torch.rand(n, generator=gen) * radius
    return v @ (d[:, None] * v.T)


def commutation_matrix(
    m: int,
    n: int,
    device: torch.device | None = None,
    dtype: torch.dtype | None = None,
) -> torch.Tensor:
    """Create a commutation matrix K such that K @ vec(A) = vec(A.T)."""

    i = torch.arange(m, device=device)
    j = torch.arange(n, device=device)
    i_grid, j_grid = torch.meshgrid(i, j, indexing="ij")

    row_indices = (j_grid + i_grid * n).flatten()
    col_indices = (i_grid + j_grid * m).flatten()

    K = torch.zeros((m * n, m * n), device=device, dtype=dtype)
    K[row_indices, col_indices] = 1.0

    return K


def commutation_matrix_sparse(m: int, n: int, device=None) -> torch.Tensor:
    """Create a commutation matrix as a sparse tensor for memory efficiency."""
    i = torch.arange(m, device=device)
    j = torch.arange(n, device=device)

    i_grid, j_grid = torch.meshgrid(i, j, indexing="ij")

    row_indices = (j_grid + i_grid * n).flatten()
    col_indices = (i_grid + j_grid * m).flatten()

    indices = torch.stack([row_indices, col_indices])
    values = torch.ones(m * n, device=device)

    K = torch.sparse_coo_tensor(indices, values, (m * n, m * n))

    return K


def circulant_matrix(row: torch.Tensor) -> torch.Tensor:
    """Create a circulant matrix from the given row vector"""
    n = row.shape[-1]
    indices = torch.arange(n, device=row.device)[None, :].repeat(n, 1)
    row_indices = torch.arange(n, device=row.device)[:, None]

    circular_indices = (indices - row_indices) % n
    matrix = row[circular_indices]
    return matrix


def orthogonal_matrix(gen: torch.Generator, size: int) -> torch.Tensor:
    """Generate random orthogonal matrices using QR decomposition"""
    mat = torch.randn(size, size, generator=gen)
    q, r = torch.linalg.qr(mat)
    d = torch.sign(torch.diagonal(r))
    q = q * d[None, :]
    return q


def permutation_matrix(gen: torch.Generator, size: int) -> torch.Tensor:
    """Generate random permutation matrices"""
    perm = torch.randperm(size, generator=gen)
    mat = torch.eye(size)[perm]
    return mat


def signed_permutation_matrix(gen: torch.Generator, size: int) -> torch.Tensor:
    """Generate random signed permutation matrices"""
    mat = permutation_matrix(gen, size)
    signs = torch.randint(0, 2, (size, 1), generator=gen) * 2 - 1
    return mat * signs
