React crossword component built to work with Guardian crossword data.
- Displays crossword grid and list of clues
- Displays separators in the grid for hyphenated and multi-word solutions
- Responsive to different screen sizes
- Clue selection highlights relevant cells
- Grouped clues span multiple columns or rows
- Tab key cycles between clues
- Arrow keys navigate between cells
- Displays attempted clues as greyed out
- Autosaves progress to local storage
- Smart clearing only removes cells not part of other completed solutions
- Check and reveal solution functions (provided
solutionsAvailable: true
) - Anagram helper
npm install mycrossword --save
yarn add mycrossword
import MyCrossword from 'mycrossword';
import 'mycrossword/dist/index.css';
import React from 'react';
const data = {
/* ... crossword data (see below) ... */
};
export default function MyPage() {
return <MyCrossword id="crossword-1" data={data} />;
}
Property | Description |
---|---|
allowedHtmlTags |
string[] = ['b', 'strong', 'i', 'em', 'sub', 'sup']Optional list of HTML tags allowed within the clues. Use [] to prevent all HTML tags. Defaults to ['b', 'strong', 'i', 'em', 'sub', 'sup'] . |
cellMatcher |
RegExp = '/[A-Z]/'Optional regular expression to match against entered cell characters. Defaults to /[A-Z]/ . |
className |
string Optional string to apply a space-delimited list of class names. |
data |
GuardianCrossword Required object that contains crossword clues, solutions and other information needed to draw the grid. See crossword data below for more information. |
id |
string Required string to uniquely identify the crossword. |
loadGrid |
GuessGrid | undefined Optional object to override storage mechanism. Called when the component is initialized with the ID of the crossword. Should return an array-based representation of the crossword grid. See guess grid below for more information. |
onCellChange |
(cellChange: CellChange) => void | undefined Optional function. Called after a grid cell has changed its guess. The object contains the properties pos , guess and previousGuess . |
onCellFocus |
(cellFocus: CellFocus) => void | undefined Optional function. Called after the focus switches to a new cell. The object returned contains the properties pos and clueId . |
saveGrid |
(value: GuessGrid | ((val: GuessGrid) => GuessGrid)) => void | undefined Optional function to override storage mechanism. Called after the grid has changed with the ID of the crossword and array-based representation of the grid. See guess grid below for more information. |
stickyClue |
'always' | 'never' | 'auto' = 'auto'Optional value to define when to show the sticky clue above the grid. Defaults to 'auto' (shown on xs and sm breakpoints). |
theme |
'red' | 'pink' | 'purple' | 'deepPurple' | 'indigo' | 'blue' | 'lightBlue' | 'cyan' | 'teal' | 'green' | 'deepOrange' | 'blueGrey' = 'blue'Optional value to override the main colour applied to the highlighted cells and clues. Defaults to 'blue' . |
For more type information, see interfaces.
This is an example of the JSON data required to create the crossword shown above. Its structure is defined by the GuardianCrossword interface.
{
id: 'simple/1',
number: 1,
name: 'Simple Crossword #1',
date: 1542326400000,
entries: [
{
id: '1-across',
number: 1,
humanNumber: '1',
clue: 'Toy on a string (2-2)',
direction: 'across',
length: 4,
group: ['1-across'],
position: { x: 0, y: 0 },
separatorLocations: {
'-': [2],
},
solution: 'YOYO',
},
{
id: '4-across',
number: 4,
humanNumber: '4',
clue: 'Have a rest (3,4)',
direction: 'across',
length: 7,
group: ['4-across'],
position: { x: 0, y: 2 },
separatorLocations: {
',': [3],
},
solution: 'LIEDOWN',
},
{
id: '1-down',
number: 1,
humanNumber: '1',
clue: 'Colour (6)',
direction: 'down',
length: 6,
group: ['1-down'],
position: { x: 0, y: 0 },
separatorLocations: {},
solution: 'YELLOW',
},
{
id: '2-down',
number: 2,
humanNumber: '2',
clue: 'Bits and bobs (4,3,4)',
direction: 'down',
length: 7,
group: ['2-down', '3-down'],
position: { x: 3, y: 0 },
separatorLocations: {
',': [4, 7],
},
solution: 'ODDSAND',
},
{
id: '3-down',
number: 3,
humanNumber: '3',
clue: 'See 2',
direction: 'down',
length: 4,
group: ['2-down', '3-down'],
position: {
x: 6,
y: 1,
},
separatorLocations: {},
solution: 'ENDS',
},
],
solutionAvailable: true,
dateSolutionAvailable: 1542326400000,
dimensions: {
cols: 13,
rows: 13,
},
crosswordType: 'quick',
};
Some functions require or return the state of the crossword grid. This is a 2-dimensional array holding the user's guess for each cell. Incomplete cells or cells that are not part of any answer are represented as the empty string (""
). Note that it follows the convention of indexing by column first (x) and then row (y) so the printed array is transposed compared to how the crossword grid appears. For example, the following crossword...
■ | D | ■ |
■ | O | ■ |
A | G | E |
...would be represented as...
{
"value": [
["", "", "A"],
["D", "O", "G"],
["", "", "E"]
]
}
While the project has been created from scratch, it should be considered as a continuation of @guardian/react-crossword. With that in mind, here's a list of improvements when compared to the original:
- Use of TypeScript
- Switched to functional components
- Better performance (memoization, useCallback etc.)
- Simplified global state management using RTK
- More modular component hierarchy
- Scoped CSS
- Full unit test coverage
- CI using GitHub actions
- Theme styles
- Combined buttons to use dropdown menus
- Check/Reveal letter
- Arrow keys can skip over dark cells
- Backspace key moves backwards without the cell having to be empty
- Anagram helper
- Click clue words to add to anagram
- Added letter to centre of word wheel
- Character counter
- Escape key to clear/close
- Show missing letters from grid
- Removed hidden input and implicitly fixed the window resize bug
- Fixed HTML display in StickyClue and AnagramHelper
- Better loading and error state display
yarn
to install dependencies
yarn test
to run unit tests
yarn build
to package for publishing
yarn build:watch
to package and watch for changes
cd example && yarn start
to run the example application
MIT, © 2021 Tom Blackwell