Chen Hui Jing
The chronicles of a self-taught designer and developer.
Recreating the Fool’s Mate chess move with CSS Grid
When Firefox 66 was released, one of the features that got myself and a couple other layout enthusiasts really excited was the ability to animate grid rows and columns (in addition to grid gaps) when using CSS Grid. It had always been written in the specification, but it took some time for browsers to implement it.
I wrote an introductory post on the Bits blog earlier but still had more to explore. That article links to as many animation demos as I could find, but the one that I kept thinking about was the DVD logo by Andrew Harvard.
My preferred method of figuring out exactly how the demo worked was to observe it with Firefox DevTools. And it blew my mind. Go ahead, try it for yourself, I’ll wait. As long as you’re using Firefox 66 or later, the logo should be bouncing around the embed.
Okay, for those of you who chose not to try it out yourself for any reason, you can play the video below and see what’s actually powering the bouncing DVD logo.
What is being animated is the grid row or column. That’s why we had that cool effect with the DevTools (it was cool to me, I don’t know about you…) but that also meant that you had to distort your grid to animate stuff within it.
One of my favourite analogies to use when talking about how CSS grid is a game-changer for web layout is by comparing placement of grid items like placing pieces on a chessboard. So my next idea was to create a chessboard where the pieces could be animated across the board.
But I realised the challenge once I figured out how animation of grid items really worked. I couldn’t maintain a regular chessboard AND move chess pieces at the same time.
So you need to move some stuff around
If you’re expecting some really smart solution to this problem, I’m sorry to disappoint but my solution is the most un-smart of all. Each piece that needed to move was in its own grid, and all the grids were stacked on top of each other with positioning.
So Fool’s Mate is also known as the Two-Move Checkmate in chess. According to Wikipedia:
Fool’s Mate received its name because it can only occur if White commits an extraordinary blunder.
The opening move by White is f3, then Black responds with g4, then White goes g4 and Black does Qh4 and it’s all over. There are variations of this but I went with the default one on Wikipedia. And I chose Fool’s Mate to begin with because it only involved a total of 4 moves to animate. ¯\_(ツ)_/¯
Creating the chess board
Let’s talk about the chess board though, because I still kinda like how that turned out. I went with a board configuration where Black is on top and White is at the bottom. I’m guessing that most people would firstly think about creating 64 grid items for each square on the board.
But I wanted to get away with less markup. So instead of doing that, I only created 28 grid items to be my static chess pieces and instead applied a linear-gradient on the grid container to generate the requisite checkerboard pattern.
Also, there is emoji support for every chess piece. Emoji chess for me, thanks.
As for the actual checkerboard, here’s how it was generated. Works pretty well, but the only thing is depending on the size of the viewport, you might see a very faint white diagonal gap across some of the darker coloured squares.
Let’s break down this block of CSS, shall we? Applying that linear gradient on a square on its own gives you something like this. Because we are applying hard stops at a 45 degree angle. I’ve also declared 2 exactly the same linear gradients, but there’s a reason for that. You’ll see.
Next, we have the background-size property. Background properties I find a lot of fun because they usually have many different syntaxes. In this case, you can use keywords, a 1-value syntax, a 2-value syntax, multiple backgrounds (using commas) and global values. Fun, right?
One thing to remember is that ALL CSS properties that a browser supports will have a default value applied if you don’t explicitly declare it. The default value for background-repeat is repeat .
Anyhow, by setting the background size to a quarter of width and height of the main , I end up with a 4 x 4 repeated pattern. You’d get the same effect by setting the value to 25% 25% .
Maybe some of you can see where this is going, but the next step is to position the triangular shapes in a manner that they combine to form squares. background-position also does the 1-value or 2-value syntax thing. Here’s where the second linear gradient comes in.
background-position can be applied to multiple backgrounds by matching them up accordingly with commas. So the first set of values 0, 0 is for the first linear gradient, and the second set of value 12vmin 12vmin is for the second linear gradient.
So what we are doing is pushing the second linear gradient down and right so the triangles meet up and form squares. For my example, I’m using the length value of 12vmin but you’d get the same effect by using 50% as well.
Placing the chess pieces
You might be wondering about my choice of CSS classes for the chess pieces. I had a previous experiment involving lots of grid items with gaps in between them and learned that you don’t necessarily have to explicitly place every grid item.
What I want here is to have the grid match up to the background of my grid container, and that’s why I chose to use viewport units instead of percentages. So my grid rows and columns look like this, where the values match up to the background size:
Out of the 32 chess pieces, 28 are static. But I have to take into account gaps for the pieces that need to be animated.
.board .piece:nth-child(4) < grid-column: 5 >.board .piece:nth-child(12) < grid-column: 6 >.board .piece:nth-child(20) < grid-column: 8 >.row7 < grid-row: 7 >.row8
Because of auto-placement, I can “push” the grid items forward by explicitly placing the King (4th child) on column 5, the pawn on f7 (12th child) on column 6 and the pawn on h2 (20th child) on column 8.
But all of White is at the bottom of the grid, so we have .row7 class to push all the White pawns onto row 7 and the .row8 class to push the White non-pawns onto row 8.
Animating the 4 moves
The moving pieces for Black are the Queen, which is on the first row, and the pawn at e7, which is on the second row. For White, the pieces are on rows 7 and 8 of my chessboard grid. And the moving pieces are the pawns at f2 and g2.
Remember the DVD demo? So each move will be a singular piece in its own grid. For example, this is the code for the Black Queen.
I went with transitions instead of keyframes because my piece was just making 1 move. The DVD logo uses keyframes because it’s a continuous animation. Right tool for the right job, no?
There is Javascript involved here because I wanted to make it kind of sort of interactive where people can click on a button to move the next piece.
It’s only 4 moves, so I figured it wouldn’t be too hard. Needed a counter to track the start and end of the moves, but other than that, it was adding and removing CSS classes.
const prevBtn = document.getElementById('prev') const nextBtn = document.getElementById('next') let moves = 0 prevBtn.addEventListener('click', prevHandler, false) nextBtn.addEventListener('click', nextHandler, false) function nextHandler() < if (moves < 4) < countUp() addActive(moves) console.log(moves) >else < console.log('Already checkmate') >> function prevHandler() < if (moves >0) < removeActive(moves) countDown() console.log(moves) >else < console.log('Back to the start') >> function addActive(moves) < const activeMove = document.querySelector('.move' + moves) activeMove.classList.add('active') >function removeActive(moves) < const activeMove = document.querySelector('.move' + moves) activeMove.classList.remove('active') >function countUp() < return moves++ >function countDown()
Wrapping up
Here’s the whole thing on CodePen if you want to see exactly how everything comes together.
See the Pen CSS Fool’s Mate by Chen Hui Jing (@huijing) on CodePen.
If you found this interesting, maybe give it a try as well? I’m really curious to see what other things can be designed and built with this behaviour, so feel free to ping me with your creation. I also want to shout out Christopher Powroznik for creating the One HTML Page Challenge.
I’ve put up this Fool’s Mate demo on there, but there’s so much more that can be done in 1 single page. It’s pretty fun to build stuff without external dependencies for a change.
Hello!
I’m Hui Jing, a front-end designer and developer who believes HTML and CSS are the foundation of the web.
Draw Chess Board using CSS Grid
CSS grid is pretty great to build layout for the web and is supported in almost any browser at the present. We can take its advantage to draw a chess board in a very simple way.
In a CSS grid, there is a parent, which is a grid, and children, which are elements to arrange inside the grid. So to draw the chess board, we will start with a simple grid container.
To make it become a grid, display: grid; is necessary. The next line grid-template-columns indicates the template of children inside the grid, which is defined by repeat(8, 1fr) . To explain in few words, it means, to layout 8 columns inside grid with equal width. The repeat function is a shortcut for: 1fr 1fr 1fr 1fr 1fr 1fr 1fr 1fr .
Since I don’t want to statically add black and white cells, I add them via Javascript on page initialization, which is more handy.
(function () ', _cellWhite = ' '; var MAX_CELL = 8; var _gridContainer = document.querySelector('#container'); for(var i = 0; i < MAX_CELL; i++) < for(var j = 0; j < MAX_CELL; j++) < if((i + j) % 2 == 0) < _gridContainer.insertAdjacentHTML('beforeend', _cellBlack); >else < _gridContainer.insertAdjacentHTML('beforeend', _cellWhite); >> > >)();
Nothing special about that.
That’s basically done for the chess board. As you can see, by using CSS grid layout, the design will be faster and simpler.
You can check if CSS grid is widely supported in browsers https://caniuse.com/#feat=css-grid
As you can see at the support chart, the feature is ready, so you don’t have to worry about it from now on.
Создание шахматной доски с помощью CSS Grid
Мне нужно создать шахматную доску с html а также css (с помощью grid ). Я создал доску и верх и низ. Теперь я остановился на средней части. Я создал размер каждого поля ( 80px Икс 80px ) Моя проблема в том, что индивиду field классы переполняются внизу моей сетки ( chess-board ). Как я могу решить эту проблему, что все мои divs «поле» останется в созданной мной сетке?
* < margin: 0; padding: 0; box-sizing: border-box; >body < box-sizing: border-box; margin: 5%; background-color: lightgreen; display: flex; justify-content: center; >.chess-board < width: 640px; height: 720px; background-color: white; display: grid; grid-template-columns: repeat(8, 80px); grid-template-rows: 40px repeat(8, 80px) 40px; grid-gap: 0; >.upper-numbers < display: grid; grid-column: 1/9; grid-row: 1/1; grid-auto-flow: column; background-color: moccasin; >.lower-numbers < display: grid; grid-column: 1/9; grid-row: 10/10; grid-auto-flow: column; background-color: moccasin; >.numbers < display: grid; grid-auto-flow: column; text-align: center; align-self: center; >/*.main-field < display: grid; grid-column: 1/9; grid-row: 2/10; background-color: white ; >*/ .field < height: 80px; width: 80px; >.field:nth-child(-2n+8)
1 2 3 4 5 6 7 8 a b c d e f g h i j q r s 1 2 3 4 5 6 7 8
1 ответ
Основная проблема, с которой вы сталкиваетесь, заключается в том, что свойства сетки не наследуются за пределами дочерних элементов контейнера. Вы пытаетесь применить свойства сетки к правнукам, но это не работает.
В вашем коде есть это правило создания контейнера сетки:
С участием .chess-board как контейнер, только .upper-numbers , .main-field а также .lower-numbers являются элементами сетки и могут принимать команды сетки.
Все элементы, вложенные в элементы сетки, не участвуют в макете сетки. Это стандартные элементы блочной компоновки. Поэтому они игнорируют ваш grid-template-columns а также grid-template-rows команды.
Мое предложение, если вы хотите использовать единый контейнер сетки и до тех пор, пока функция «подсетки» не будет поддерживаться браузерами, было бы «сгладить» структуру HTML.
Вот простой пример, который я построил, используя комбинацию макетов сетки и гибкости.
.chess-board < margin: auto; width: 640px; height: 720px; background-color: white; display: grid; grid-template-columns: 1fr; grid-template-rows: 40px 1fr 40px; grid-gap: 0; >.upper-numbers, .lower-numbers < display: flex; background-color: moccasin; >.upper-numbers>div, .lower-numbers>div < flex: 1; display: flex; align-items: center; justify-content: center; >.main-field < display: flex; flex-direction: column; flex-wrap: wrap; height: 640px; >.field < height: 80px; width: 80px; display: flex; align-items: center; justify-content: center; >.field:nth-child(-2n+8) < background-color: black; >* < margin: 0; padding: 0; box-sizing: border-box; >body
1 2 3 4 5 6 7 8 a b c d e f g h i j q r s 1 2 3 4 5 6 7 8