Remix. Run (Remix for short) is ready for open source release 1.0 in JavaScript Weekly. Within two days, Remix topped Github Trending. It feels like it’s already catching on. I wonder how it will compare to Next. But it doesn’t look too complicated, a lot of stuff is wrapped up, and the syntax is light. So get ready to study and take notes.

Create a project

Download the latest version of Remix from NPM and build the project.

npx create-remix@latest

? Where would you like to create your app? my-cv

I plan to make a Web application to display my resume and dynamic display of project experience, which is somewhat necessary. Otherwise, it is just a Demo. After all, there is no need for in-depth study. And SSR(Server Side Render) features is convenient SEO, do a show dynamic data class application is the most suitable. After my CV.

? Where do you want to deploy? Choose Remix if you're unsure, it's easy to change deployment targets.

  • Remix App Server
  • Express Server
  • Architect (AWS Lambda)
  • Fly.io
  • Netlify
  • Vercel
  • Cloudflare Workers

Of course, I will choose the original Remix App Server

? TypeScript or JavaScript? TypeScript

  • JavaScript

I chose TypeScript. I don’t usually choose TypeScript for my own projects, but this time I’ll try to emulate it more formally. TypeScript does more harm than good to individual developers. While it does reduce errors and the time cost of finding them, it takes more time to prepare for a startup project such as defining types, and the risk of finding unpopular libraries that don’t support TS.

? Do you want me to run npm install? Y

Just install it last.

Start the project

Enter the project directory

cd my-cv

Turn it on. Let’s see.

npm run dev

Node version 12 has an error, 16 has no problem.

“Could not locate @remix-run/serve. Please verify you have it installed to use the dev command.”

You can go to http://localhost:3000. There is a Demo that shows the various states of the route, 404,401, etc., and the route with parameters. You can keep it for reference, you can delete it.

Create the page

This Demo style is still ok, I will leave it, anyway, write your own style for learning Remix does not have much significance. So I changed the navigation to Chinese, and then changed the second page to a new path that I could create later to display my resume. Then the third Github connection changed to its own.

<ul>
  <li>
    <Link to="/">Home page</Link>
  </li>
  <li>
    <Link to="/resume">resume</Link>
  </li>
  <li>
    <a href="https://github.com/tychio">GitHub</a>
  </li>
</ul>
Copy the code

Then, create the corresponding page.

mkdir app/routes/resume
touch app/routes/resume/index.tsx
Copy the code

Then fill in some static text, a name and an introduction. There are skills, and this can be done by loading dynamic data, doing the front end first, returning directly from the literal. Returns data using useLoaderData.

import type { LoaderFunction } from "remix";
import { useLoaderData, json } from "remix";
type ResumeData = {
  skills: ArrayThe < {name: string} >. };export const loader: LoaderFunction = () = > {
  const data: ResumeData = {
    skills: [
      'JavaScript'.'CSS/HTML'.'React'.'Remix']};return json(data);
}

export default function ResumeIndex() {
  const resume = useLoaderData<ResumeData>();
  return (
    <div>
      <h1>Zhang Zhengzheng</h1>
      <p>
        A full-stack developer, Senior consultant, Freelancer.
      </p>
      <p>
        {resume.skills.map((skill, index) => (
          <span>{index ! = = 0? ', ' : ''}{skill.name}</span>
        ))}
      </p>
    </div>
  );
}
Copy the code

Note: The loader is called by the backend API hook useLoaderData, so it is not used

I also defined the ResumeData type for dynamic data on the page, which contains Skills.

Use database ORM

Next, find an ORM and put the data into the database completely. I chose Prisma,

npm install --save-dev prisma

npm install @prisma/client

Initialize the ORM

npx prisma init --datasource-provider mysql

SQL Lite NPX PRISma Init — Datasource-provider SQLite

Then set DATABASE_URL in the added. Env file

mysql://<username>:<password>@<host | localhost>:<port>/<database_name>

Then execute NPX PRISma DB pull to read the database and automatically generate the schema.

Then run NPX prisma generate to generate the client. This allows you to use ORM with the following code.

import { PrismaClient } from '@prisma/client'
const prisma = new PrismaClient()
Copy the code

Create a table

I created the skills table ahead of time, so when I pulled, I had the model in the prisma/schema.prisma file.

model skills {
  id   Int     @id @default(autoincrement())
  name String? @db.VarChar(30)
}
Copy the code

If the table does not exist in the database, you can create the corresponding table in the database by writing model schema here and then executing NPX PRISma DB push.

Make sure you add.env to.gitignore. If using SQLite, there is also /prisma/xxx.db.

Insert data

Create the prisma/seed.ts file for the initial data.

import { PrismaClient } from "@prisma/client";
let db = new PrismaClient();

async function seed() {
  await Promise.all(
    getSkills().map(joke= > {
      return db.skills.create({ data: joke }); })); } seed();function getSkills() {
  return[{name: 'JavaScript'},... ] }Copy the code

Install the TS-Node package to execute the seed.

npm install --save-dev ts-node

For convenience, add it to package.json

"prisma": {
  "seed": "ts-node prisma/seed.ts"
},
Copy the code

And then execute the seed

npx prisma db seed

Using the data

Create the utils directory under the app directory and the file utils/db.server.ts

import { PrismaClient } from "@prisma/client";

let db: PrismaClient;

declare global {
  var __db: PrismaClient | undefined;
}

if (process.env.NODE_ENV === "production") {
  db = new PrismaClient();
  db.$connect();
} else {
  if (!global.__db) {
    global.__db = new PrismaClient();
    global.__db.$connect();
  }
  db = global.__db;
}

export { db };
Copy the code

The difference in the Development environment is that the connection instances are cached so there is no restart every time

Use it on the previous Resume /index.tsx page.

import { db } from "~/utils/db.server";

~ is configured in the Remix template tsconfig.json by default and represents the app directory.

Update loader method

export const loader: LoaderFunction = async() = > {const data: ResumeData = {
    skills: await db.skills.findMany()
  };
  return data;
}
Copy the code

This completes the basic Remix process. From database to page.

I also replaced the original logo. Google Graphics makes SVG quite useful.

The code is put on Github, and will continue later, and the text may be different.