Flipping Off CSS and JavaScript

HTML/CSS, Javascript, programming

I asked CoPilot in Microsoft Edge to help me get started with something in a Web page to allow me flip between cards:

You are an expert in Javascript and CSS.

Generate the code in HTML, Javascript, and CSS to animate flipping a picture along the X-axis and then revealing a different picture.

Honestly I didn’t know how well that was going to turn out. Surprisingly, very well. It generated these:

HTML:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Image Flip Animation</title>
    <link rel="stylesheet" href="styles.css">
</head>
<body>
    <div class="flip-container" onclick="flipImage()">
        <div class="flipper">
            <img src="image1.jpg" alt="Image 1" class="front">
            <img src="image2.jpg" alt="Image 2" class="back">
        </div>
    </div>
    <script src="script.js"></script>
</body>
</html>

CSS

body {
    display: flex;
    justify-content: center;
    align-items: center;
    height: 100vh;
    background-color: #f0f0f0;
    margin: 0;
}

.flip-container {
    perspective: 1000px;
}

.flipper {
    position: relative;
    width: 300px;
    height: 200px;
    transform-style: preserve-3d;
    transition: transform 0.6s;
}

.flip-container.flipped .flipper {
    transform: rotateX(180deg);
}

.front, .back {
    position: absolute;
    width: 100%;
    height: 100%;
    backface-visibility: hidden;
}

.front {
    z-index: 2;
    transform: rotateX(0deg);
}

.back {
    transform: rotateX(180deg);
}

JavaScript:

function flipImage() {
    const container = document.querySelector('.flip-container');
    container.classList.toggle('flipped');
}

I plugged in two JPGs I had lying around, and it did work quite well.

With that, I tweaked it to fit more what I had in mind.

HTML:

<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Image Flip Animation</title>
    <link rel="stylesheet" href="styles.css">
</head>
<body>
    <div class="flip-container">
        <div class="flip-element">
            <div class="textcard-container front">
                <div class="title">FRONT</div>
                <div class="body">This is the front card</div>
            </div>
            <div class="textcard-container back">
                <div class="title">BACK</div>
                <div class="body">
                    Lorem ipsum dolor sit amet, consectetur adipiscing elit. Pellentesque imperdiet libero eu neque facilisis, ac pretium nisi dignissim. Integer nec odio. Praesent libero. Sed cursus ante dapibus diam. Sed nisi. Nulla quis sem at nibh elementum imperdiet. Duis sagittis ipsum. Praesent mauris. Fusce nec tellus sed augue semper porta. Mauris massa. Vestibulum lacinia arcu eget nulla.
                </div>
            </div>
        </div>
    </div>
    <div id="flipper">
        <button onclick="flipImage()">Flip</button>
    </div>
    <script src="script.js"></script>
</body>
</html>

CSS

.flip-container {
    perspective: 1000px;
    display: flex;
    justify-content: center;
    align-items: center;
    height: 80vh;
    background-color: #f0f0f0;
    margin: 0;    
}

.flip-element {
    position: relative;
    width: 300px;
    height: 200px;
    transform-style: preserve-3d;
    transition: transform 0.3s;
}

.flip-container.flipped .flip-element {
    transform: rotateX(180deg);
}

.front, .back {
    position: absolute;
    width: 100%;
    height: 100%;
    backface-visibility: hidden;
}

.front {
    z-index: 2;
    transform: rotateX(0deg);
    background-color: cornsilk;
}

.back {
    transform: rotateX(180deg);
    background-color: white;
}

.textcard-container {
    display: flex;
    flex-direction: column;
    justify-content: space-around;
    align-items: center;
    text-align: center;
}

.textcard-container .title {
    justify-self: flex-start;
    width: 100%;
    box-sizing: border-box;
}

.textcard-container .body {
    justify-self: flex-end;
    box-sizing: border-box;
    padding: 15px;
    overflow: scroll;
}

.textcard-container .body:focus {
    outline-color: cornsilk;
}

#flipper {
    width: 100%;
    text-align: center;
    margin-top: 15px;
}

The JavaScript remains unchanged for now.

Looking at the various pieces, here are the notes:

Stacking Two Elements

The first thing is stacking the two cards with the “front” card on top of the “back” card. This is done by:

  1. Use the same dimensions for two DIVs (cards) inside a parent DIV (container).
  2. Add transform-style: preserve-3d for the container so that whatever animation we apply will stay.
  3. Add a transition: transform 0.3s (or however long desired for the animation sequence) to the container as well.
  4. Use absolute positioning for the cards so that they overlap.
  5. Give the front card a z-index of 2 (anything higher than that of the back card).
  6. Set the backface-visibility property to hidden. This ensures that cards flipped will be hidden.
  7. Rotate the back card by 180 degrees. This makes the back card invisible for now. It’s strictly not necessary since it’s obscured by the front card. However, this will be important when we do the flip.

Flip Animation

Set a CSS rule “flipped” that, when applied to the container, will rotate both child cards 180 degrees. This will:

  1. Flip the front card 180 degrees, and because we have backface-visibility: hidden, the front card will now be invisible (even though its z-index is still higher than that of the back card).
  2. Flip the back card 180 degrees. Since the back card was initially flipped 180 degrees, another 180 degrees now flips it to be front-facing. And since it’s now no longer flipped, it will be visible!
  3. In summary, when we flip both cards, we make the front card invisible and the back card visible.
    • If we didn’t need to animate, then theoretically we could leave the back card alone (visible) and just flip the front card.
    • However, because we ARE animating the flip, leaving the back card stationary will have it show through as the front card rotates in and out of visibility. Therefore, flipping both cards will give the smooth illusion that we are flipping a two-faced card with the front and back cards’ contents. Sneaky, isn’t it?

Now it just relies on toggling the CSS class “flipped” on the container. And that’s where the JavaScript comes in.

I didn’t know that JavaScript now can toggle a class for an element easily using the {element}.classList.toggle({class name}) function. I used to have to split the class list into an array and then iterate through the array to either add or skip and then setting the class list to the resulting list. This was a nice surprise.

Perspective

Setting perspective: 1000px on the container adds a nice touch that, when flipping, there’s a small 3-D effect. Though it’s something I’d miss if I blinked.

Celery/RabbitMQ Notes on Task Timings

celery, django, rabbitmq

The following notes apply when using Celery with the RabbitMQ broker.

Celery Settings

task_acks_late

The Celery setting task_acks_late (by default disabled), if set, will defer message ACK with RabbitMQ until the task completes successfully.

  • If it is enabled, and the Celery task takes too long (see consumer_timeout below), the message will be requeued and redelivered to the next worker. This will cause the message to be processed multiple times.
  • If it is disabled, then Celery will ACK the message to RabbitMQ as soon as the task starts. This tells RabbitMQ that the message was “delivered and processed,” and RabbitMQ will delete this message from the queue. This will cause the message to be lost if the Celery task was interrupted before it finishes.

task_acks_on_failure_or_timeout

Then there is task_acks_on_failure_or_timeout (by default enabled). According to the doc, this will ACK the message even if the task failed or timed out. This may or may not be the correct choice depending on the task.

RabbitMQ Settings

consumer_timeout

The RabbitMQ consumer_timeout configuration (by default 30 minutes) gives a task a maximum timeout before requeuing the message. If a client (a Celery task in this case) does not ACK the message before this timeout expires, the message will be requeued and redelivered.

TTL / message TTL

Furthermore, the TTL / message TTL configuration (by default ??) determines how long a message will stay in RabbitMQ before being discarded. If a message stays in a queue for longer than this time, then it is removed from the queue.

Implications

To increase the chance that tasks are not aborted/lost due to restarts (e.g. deployments):

  • Design each task to finish as soon as possible. Spawning additional smaller tasks if necessary.
  • The entire task is designed to survive a partial completion state and is able to completely restart without ending up in a bad state (e.g. duplicate records created, abandoned/orphaned/partial updates).
  • Enable task_acks_late so that RabbitMQ ACKs are not sent until tasks are finished.
  • Extend the TTL / message TTL for RabbitMQ so that we have enough time to consume messages (in case the Celery task needs to be paused for some time to fix bugs).

There is no support to retry failed tasks. In fact, unless there is a Result Backend set up, there won’t even be a good way to audit which tasks failed (other than application logs).

PyTorch in Docker

docker, programming, Python

I’m getting into the fray that is Generative AI since, according to some, my job as a programmer will soon be taken over by some literal code cranking machine.

There are a few things to set up:

Like most tools and libraries, the instructions assume that I have nothing else happening on my machine, so install just Anaconda globally according to their simplistic assumptions. All the other dependencies like even the Python version on my machine can go to hell.

Dockerizing the environment

Well. Until I have enough $$$ to buy a new machine for each new tool I want to try out, I’ll be using Docker (think goodness I don’t need an actual VM) to isolate an environment to play with.

After some trial and error, this is a template Dockerfile and docker-compose.yml I’m using:

Dockerfile

FROM python:3.11-bookworm
ENV PYTHONUNBUFFERED 1

WORKDIR /code

RUN apt update && apt install -y \
    vim

RUN curl -O https://repo.anaconda.com/miniconda/Miniconda3-py311_24.1.2-0-Linux-x86_64.sh
RUN sh Miniconda3-py311_24.1.2-0-Linux-x86_64.sh -b
ENV PATH="$PATH:/root/miniconda3/bin"

RUN conda install -y pytorch torchvision torchaudio pytorch-cuda=12.1 -c pytorch -c nvidia
RUN conda init

The above environment has:

  • Vim
  • Python 3.11
  • Miniconda3
  • PyTorch with CUDA 12.1

docker-compose.yml

services:
  app:
    build: .
    volumes:
      - ./:/code
    tty: true
    stdin_open: true

The docker-compose.yml is a template one I use for other things. Nothing special here.

Redux Toolkit w/ Vite in Docker

docker, Node.js, programming, react

Notes on setting up a Docker/Vite/React/Redux Toolkit service in Docker. That’s a mouthful.

Docker Files

Start with a name for the project (e.g. “myproject”). Then create these Docker files:

Dockerfile

FROM node:20.10

RUN apt update && apt install -y \
  xdg-utils

EXPOSE 5173

docker-compose.yml

version: '3'
services:
  app:
    build: .
    command: >
      sh -c "npm run dev"
    ports:
      - "5173:5173"
    expose:
      - "5173"
    volumes:
      - .:/app
    tty: true
    stdin_open: true

Now build the working image and shell into the container:

docker-compose build

Redux Toolkit Setup

Shell into a container of the working image and initialize the project:

docker-compose run --rm app /bin/bash
...
npx degit reduxjs/redux-templates/packages/vite-template-redux myproject

Of course, use the name of the project instead of myproject.

This will create the starter files for the project under myproject/.

Exit the shell.

File Updates

Dockerfile

Modify Dockerfile to build the image with the project:

FROM node:20.10

RUN apt update && apt install -y \
  xdg-utils

EXPOSE 5173

WORKDIR /app/myproject

RUN npm install -g npm 

vite.config.ts

Modify the Vite config to allow it to host from a Docker container:

import { defineConfig } from "vitest/config"
import react from "@vitejs/plugin-react"

// https://vitejs.dev/config/
export default defineConfig({
  plugins: [react()],
  server: {
    host: true,  // <-- ADD THIS
    open: true,
  },
  ....

Start

Finally, to start the server, first build the image again:

docker-compose build

Then bring up the app:

docker-compose up

This should now bring up the app at http://localhost:5173/

To work on the project’s settings (e.g. installing packages, running tests, etc.), shell in with:

docker-compose run --rm --service-ports app /bin/bash

The --service-ports option will allow commands like npm run dev to start the app correctly (i.e. so that http://localhost:5173/ works). Without it, the port 5173 will not be mapped to the host, and docker-compose up will be the only way to run the app.