Pixelpage Architecture

← Back to home

pixelpage website

In this post we will investigate the server-side mechanics of how Pixelpage:

Fresh Structure

Pixelpage takes its file structure from Deno Fresh:

├── components
│   ├── ColorPicker.tsx
│   └── PixelGrid.tsx
├── islands
│   └── Pixels.tsx
├── routes
│   ├── api
│   │   ├── listen.ts
│   │   └── update.ts
│   └── index.tsx
├── shared
│   ├── constants.ts
│   ├── db.ts
│   └── types.ts
├── static
│   ├── favicon.ico
│   ├── logo.jpg
│   └── screenshot.png
├── LICENSE
├── README.md
├── deno.json
├── deno.lock
├── dev.ts
├── fresh.gen.ts
├── main.ts
└── twind.config.ts

The components folder contains what Fresh calls server components - inert, javascriptless components rendered by the server and sent to the client, which exist in the /componenets.

Conversely, any interactive / client-side javascript you want to run must exist as an interactive island, in the /islands folder.

The /routes folder contains code defining how requests are handled, and as such /routes/index.tsx represents the public facing entrypoint, and the /routes/api folder represents interfaces which allow data into and out of from the servers.

The /shared folder holds modules which can be imported to elsewhere in the web-app, and the /static folder holds assets which do not require any sort of client-side or server-side rendering.

index.tsx

import { Head } from "$fresh/runtime.ts";
import { Handlers, PageProps } from "$fresh/server.ts";
import Pixels from "../islands/Pixels.tsx";
import { getGrid } from "../shared/db.ts";
import { Grid } from "../shared/types.ts";

export const handler: Handlers<Grid> = {
  async GET(_req, ctx) {
    const grid = await getGrid();
    return ctx.render(grid);
  },
};

export default function Home(props: PageProps<Grid>) {
  return (
    <>
      <Head>
        <title>pixelpage</title>
        <link rel="icon" type="image/jpg" href="/logo.jpg" />
      </Head>
      <div class="p-4 mx-auto max-w-screen-md flex justify-center">
        <Pixels grid={props.data} />
      </div>
      <div class="p-4 flex justify-evenly gap-8 text-center">
        <a
          href="https://github.com/denoland/pixelpage"
          class="text-blue-500 hover:underline"
        >
          View Source on GitHub
        </a>
        <a
          href="https://deno.com/deploy"
          class="text-blue-500 hover:underline"
        >
          Powered by Deno Deploy
        </a>
      </div>
    </>
  );
}

The typescript file, index.tsx exposes a handler object called handler, and a default export function called Home which returns jsx component.

The handler object uses the getGrid() function from /shared/db.ts to retrieve the current state of the pixel grid, and passes that into the component, which Deno has invisibly packaged on the .render() method of the second argument (ctx) passed in to the GET handler method.

For those who are curious, Marvin Hagemeister generously provided me with this psuedocode to help explain what exactly Deno is doing under the hood to achieve this:

import { render as PreactRender } from "preact-render-to-string";
const route = await import("routes/my-route.tsx");

Deno.serve((req) => {
  const url = new URL(req.url);
  if (url.pathname === "/my-route") {
    const component = route.default;
    const handler =
      typeof route.handler === "object"
        ? route.handler
        : { GET: (ctx) => ctx.render() };

    const ctx = {
      render() {
        const html = PreactRender(component);
        return new Response(html, { headers: { "Content-Type": "text/html" } });
      },
    };

    return handler.GET(ctx);
  }
});

Once the Home component has the grid data, it passes it in to the Pixel component it has imported from /islands/Pixels.tsx.

Pixels.tsx

import { Signal } from "@preact/signals";

import { COLORS, WIDTH } from "../shared/constants.ts";
import { Grid } from "../shared/types.ts";

const PIXEL_SIZE = 24;
const DESKTOP_PIXEL_SIZE = 32;

export default function PixelGrid(
  props: {
    grid: Signal<Grid>;
    selected: Signal<number>;
    updateGrid(index: number, color: string): void;
  },
) {
  const { selected, updateGrid } = props;

  return (
    <div
      class={`grid grid-cols-${WIDTH} w-[${WIDTH * PIXEL_SIZE}px] sm:w-[${
        WIDTH * DESKTOP_PIXEL_SIZE
      }px] border`}
    >
      {props.grid.value.tiles.map((color, i) => (
        <div
          class={`w-[${PIXEL_SIZE}px] h-[${PIXEL_SIZE}px] sm:w-[${DESKTOP_PIXEL_SIZE}px] sm:h-[${DESKTOP_PIXEL_SIZE}px] bg-[${color}]`}
          onClick={() => {
            updateGrid(i, COLORS[selected.value]);
          }}
        >
        </div>
      ))}
    </div>
  );
}