feat: responsive grid with scrollable area and fixed hint column
- Add CSS variables (--cell, --cell-font, --trigger-h) for responsive cell sizing - Shrink grid cells at 600px and 420px breakpoints - Add .page-body wrapper with 16px horizontal padding to prevent edge collisions - Separate hint column from scrollable grid: hints rendered outside the table in a fixed flex column, only the letter grid scrolls horizontally Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -1,15 +1,29 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import GameRow from './GameRow';
|
import GameRow from './GameRow';
|
||||||
|
import ActorPopover from './ActorPopover';
|
||||||
|
|
||||||
export default function GameGrid({ grid, width, middle }) {
|
export default function GameGrid({ grid, width, middle }) {
|
||||||
return (
|
return (
|
||||||
|
<div className="game-grid-area">
|
||||||
|
<div className="hint-col">
|
||||||
|
{grid.map((row, rowIndex) => {
|
||||||
|
if (row.separator !== undefined) {
|
||||||
|
return <div key={rowIndex} className="hint-separator" />;
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<div key={rowIndex} className="hint-cell">
|
||||||
|
<ActorPopover hintType={row.hintType} hintText={row.hintText} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
<div className="game-grid-scroll">
|
||||||
<table id="actors">
|
<table id="actors">
|
||||||
<tbody>
|
<tbody>
|
||||||
{grid.map((row, rowIndex) => {
|
{grid.map((row, rowIndex) => {
|
||||||
if (row.separator !== undefined) {
|
if (row.separator !== undefined) {
|
||||||
return (
|
return (
|
||||||
<tr key={rowIndex} className="separator-row">
|
<tr key={rowIndex} className="separator-row">
|
||||||
<td />
|
|
||||||
{Array.from({ length: middle }, (_, i) => (
|
{Array.from({ length: middle }, (_, i) => (
|
||||||
<td key={i} />
|
<td key={i} />
|
||||||
))}
|
))}
|
||||||
@@ -30,12 +44,12 @@ export default function GameGrid({ grid, width, middle }) {
|
|||||||
pos={row.pos}
|
pos={row.pos}
|
||||||
colStart={middle - row.pos}
|
colStart={middle - row.pos}
|
||||||
totalWidth={width}
|
totalWidth={width}
|
||||||
hintType={row.hintType}
|
|
||||||
hintText={row.hintText}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,11 @@
|
|||||||
import React, { useRef, useCallback, useMemo } from 'react';
|
import React, { useRef, useCallback, useMemo } from 'react';
|
||||||
import LetterInput from './LetterInput';
|
import LetterInput from './LetterInput';
|
||||||
import ActorPopover from './ActorPopover';
|
|
||||||
|
|
||||||
function isLetter(ch) {
|
function isLetter(ch) {
|
||||||
return /[a-zA-Z]/.test(ch);
|
return /[a-zA-Z]/.test(ch);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function GameRow({ actorName, pos, colStart, totalWidth, hintType, hintText }) {
|
export default function GameRow({ actorName, pos, colStart, totalWidth }) {
|
||||||
const inputRefs = useRef([]);
|
const inputRefs = useRef([]);
|
||||||
const letters = actorName.split('');
|
const letters = actorName.split('');
|
||||||
|
|
||||||
@@ -29,9 +28,6 @@ export default function GameRow({ actorName, pos, colStart, totalWidth, hintType
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
|
||||||
<ActorPopover hintType={hintType} hintText={hintText} />
|
|
||||||
</td>
|
|
||||||
{Array.from({ length: totalWidth + 1 }, (_, colIndex) => {
|
{Array.from({ length: totalWidth + 1 }, (_, colIndex) => {
|
||||||
const charIndex = colIndex - colStart;
|
const charIndex = colIndex - colStart;
|
||||||
const isInRange = charIndex >= 0 && charIndex < letters.length;
|
const isInRange = charIndex >= 0 && charIndex < letters.length;
|
||||||
|
|||||||
@@ -21,6 +21,9 @@
|
|||||||
--radius-sm: 6px;
|
--radius-sm: 6px;
|
||||||
--radius-md: 10px;
|
--radius-md: 10px;
|
||||||
--radius-lg: 18px;
|
--radius-lg: 18px;
|
||||||
|
--cell: 38px;
|
||||||
|
--cell-font: 15px;
|
||||||
|
--trigger-h: 28px;
|
||||||
}
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
@@ -31,6 +34,11 @@ body {
|
|||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.page-body {
|
||||||
|
padding: 0 16px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
/* ── Game grid ── */
|
/* ── Game grid ── */
|
||||||
|
|
||||||
#actors {
|
#actors {
|
||||||
@@ -40,22 +48,19 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#actors td {
|
#actors td {
|
||||||
width: 38px;
|
width: var(--cell);
|
||||||
height: 38px;
|
height: var(--cell);
|
||||||
text-align: center;
|
text-align: center;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
#actors td:first-child {
|
|
||||||
padding-right: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.letter-input {
|
.letter-input {
|
||||||
width: 38px;
|
width: var(--cell);
|
||||||
height: 38px;
|
height: var(--cell);
|
||||||
text-align: center;
|
text-align: center;
|
||||||
font-family: 'Inter', sans-serif;
|
font-family: 'Inter', sans-serif;
|
||||||
font-size: 15px;
|
font-size: var(--cell-font);
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
border: 2px solid var(--border);
|
border: 2px solid var(--border);
|
||||||
border-radius: var(--radius-sm);
|
border-radius: var(--radius-sm);
|
||||||
@@ -80,11 +85,11 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.letter-static {
|
.letter-static {
|
||||||
width: 38px;
|
width: var(--cell);
|
||||||
height: 38px;
|
height: var(--cell);
|
||||||
text-align: center;
|
text-align: center;
|
||||||
font-family: 'Inter', sans-serif;
|
font-family: 'Inter', sans-serif;
|
||||||
font-size: 15px;
|
font-size: var(--cell-font);
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
@@ -103,11 +108,11 @@ body {
|
|||||||
/* ── Popover ── */
|
/* ── Popover ── */
|
||||||
|
|
||||||
.popover-trigger {
|
.popover-trigger {
|
||||||
width: 38px;
|
width: var(--cell);
|
||||||
height: 28px;
|
height: var(--trigger-h);
|
||||||
border-radius: 6px;
|
border-radius: 6px;
|
||||||
border: none;
|
border: none;
|
||||||
background: #F02287;
|
background: var(--orange);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
@@ -120,7 +125,7 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.popover-trigger:hover {
|
.popover-trigger:hover {
|
||||||
background: #d41a76;
|
background: var(--orange-mid);
|
||||||
}
|
}
|
||||||
|
|
||||||
.actor-popover {
|
.actor-popover {
|
||||||
@@ -515,7 +520,8 @@ body {
|
|||||||
/* ── Game card ── */
|
/* ── Game card ── */
|
||||||
|
|
||||||
.game-container {
|
.game-container {
|
||||||
max-width: fit-content;
|
width: fit-content;
|
||||||
|
max-width: 100%;
|
||||||
margin: 56px auto;
|
margin: 56px auto;
|
||||||
padding: 24px 32px 32px;
|
padding: 24px 32px 32px;
|
||||||
background: var(--surface);
|
background: var(--surface);
|
||||||
@@ -525,7 +531,36 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.game-container #actors {
|
.game-container #actors {
|
||||||
margin: 16px auto 0;
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-grid-area {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
margin-top: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hint-col {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding-top: 5px;
|
||||||
|
gap: 5px;
|
||||||
|
flex-shrink: 0;
|
||||||
|
padding-right: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hint-cell {
|
||||||
|
height: var(--cell);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hint-separator {
|
||||||
|
height: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-grid-scroll {
|
||||||
|
overflow-x: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ── Game actions ── */
|
/* ── Game actions ── */
|
||||||
@@ -592,3 +627,29 @@ body {
|
|||||||
color: var(--orange);
|
color: var(--orange);
|
||||||
background: var(--surface-tint);
|
background: var(--surface-tint);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* ── Responsive grid ── */
|
||||||
|
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
:root {
|
||||||
|
--cell: 30px;
|
||||||
|
--cell-font: 12px;
|
||||||
|
--trigger-h: 22px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.game-container {
|
||||||
|
padding: 16px 16px 24px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 420px) {
|
||||||
|
:root {
|
||||||
|
--cell: 24px;
|
||||||
|
--cell-font: 10px;
|
||||||
|
--trigger-h: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#actors {
|
||||||
|
border-spacing: 3px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -17,6 +17,8 @@
|
|||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
{% include '_navbar.html.twig' %}
|
{% include '_navbar.html.twig' %}
|
||||||
|
<div class="page-body">
|
||||||
{% block body %}{% endblock %}
|
{% block body %}{% endblock %}
|
||||||
|
</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
Reference in New Issue
Block a user