Перейти к содержанию

Знакомство с популярными JS фреймворками - Solid.js

А вот вам и вариант фреймворка, который похож как на React, так и на Svelte. А своим размером он не уступает Preact. Встречайте — Solid!

В этой серии⚓︎

Вступление⚓︎

Можете взять за основу любой из предыдущих проектов, либо начать с нуля. Решать вам.

Подготовка⚓︎

Итак, перейдите в папку projects, откройте консоль и запустите следующую команду:

npm create vite@latest solid-todo -- --template solid

Теперь перейдите в созданную папку solid-todo и установите UnoCSS:

npm i -D unocss @iconify-json/heroicons
pnpm create vite solid-todo --template solid

Теперь перейдите в созданную папку solid-todo и установите UnoCSS:

pnpm add -D unocss @iconify-json/heroicons
yarn create vite solid-todo --template solid

Теперь перейдите в созданную папку solid-todo и установите UnoCSS:

yarn add -D unocss @iconify-json/heroicons
bun create vite solid --template solid

Теперь перейдите в созданную папку solid-todo и установите UnoCSS:

bun add -D unocss @iconify-json/heroicons

и обновите vite.config.js:

import { defineConfig } from 'vite'
import UnoCSS from 'unocss/vite'
import solid from 'vite-plugin-solid'

export default defineConfig({
  plugins: [UnoCSS(), solid()],
})

Добавьте uno.config.js:

import { defineConfig, presetIcons, presetWind4 } from 'unocss'

export default defineConfig({
  presets: [
    presetWind4(),
    presetIcons(),
  ],
})

App.css можно удалить, он нам не нужен.

Обновите src/index.jsx:

/* @refresh reload */
import { render } from 'solid-js/web'

import 'virtual:uno.css'
import App from './App'

const root = document.getElementById('root')

render(() => <App />, root)

Файл src/App.jsx:

import TodoList from './components/TodoList';

function App() {
  return <TodoList title='Список дел' />;
}

export default App;

Сможете найти отличия от React?

Осталось запустить dev-сервер и начать создавать компоненты:

npm run dev
pnpm run dev
yarn dev
bun run dev

Компонент TodoList⚓︎

Как и прежде, сначала создадим компонент списка. Создайте в директории src/components файл TodoList.jsx:

import { createStore } from "solid-js/store";

function TodoList(props) {
  const [store, setStore] = createStore({ todos: [] });

  return (
    <div class='max-w-sm md:max-w-lg mx-auto my-10 bg-white rounded-md shadow-md overflow-hidden'>
      <h1 class='text-2xl font-bold text-center py-4 bg-gray-100'>{props.title}</h1>
      <ul class='list-none p-4'></ul>
    </div>
  );
}

export default TodoList;

Здесь мы с помощью createSignal (аналог useState в React, или useSignal в Preact) определяем две функции — todos и setTodos. Первая будет возвращать список задач, а вторая — обновлять этот список.

Получать список текущих задач мы, как и раньше, будем с сервера https://dummyjson.com/todos, с использованием хука жизненного цикла onMount — не забудьте импортировать его из solid-js:

import { onMount } from "solid-js";
import { createStore } from "solid-js/store";

function TodoList(props) {
  const [store, setStore] = createStore({ todos: [] });

  onMount(() => {
    fetch('https://dummyjson.com/todos')
      .then(response => response.json())
      .then(data => setStore('todos', data.todos.slice(0, 10)))
  })

  // ...
}

Далее, создадим заготовки компонентов TodoItem.jsx и TodoForm.jsx и используем их в разметке:

import TodoItem from './TodoItem';
import TodoForm from './TodoForm';
import { onMount, For } from "solid-js";
import { createStore } from "solid-js/store";

function TodoList(props) {
  const [store, setStore] = createStore({ todos: [] });

  onMount(() => {
    fetch('https://dummyjson.com/todos')
      .then(response => response.json())
      .then(data => setStore('todos', data.todos.slice(0, 10)))
  })

  // Заготовки для методов действий, которые определим позже
  const addTask = (title) => {};

  const toggleTask = (id) => {};

  const deleteTask = (id) => {};

  return (
    <div class='max-w-sm md:max-w-lg mx-auto my-10 bg-white rounded-md shadow-md overflow-hidden'>
      <h1 class='text-2xl font-bold text-center py-4 bg-gray-100'>{props.title}</h1>
      <ul class='list-none p-4'>
        <For each={store.todos}>
          {t => <TodoItem key={t.id} task={t} onToggle={toggleTask} onRemove={deleteTask} />}
        </For>
      </ul>
      <TodoForm onSubmit={addTask} />
    </div>
  );
}

export default TodoList;

Обратите внимание на использование встроенного компонента For в качестве управляющей конструкции для вывода всех задач в цикле.

Осталось реализовать методы управления задачами (добавление, переключение статуса и удаление), которые мы будем передавать в компоненты TodoItem и TodoForm, соответственно:

const addTask = (title) => {
  if (!title) return;

  setStore('todos', (todos) => [...todos, { id: crypto.randomUUID(), title, completed: false }]);
}

const toggleTask = (id) => {
  setStore('todos', (t) => t.id === id, 'completed', (completed) => !completed);
};

const deleteTask = (id) => {
  setStore('todos', (t) => t.filter((t) => t.id !== id))
}

Заметили, как осуществляется доступ к массиву задач, хранящемуся в todos? Через вызов функции todos().

Напоследок добавим условие отображения списка только при ненулевом количестве задач:

// Используем сигнал, доступный только для чтения
const length = createMemo(() => store.todos.length);

return (
  <div class='max-w-sm md:max-w-lg mx-auto my-10 bg-white rounded-md shadow-md overflow-hidden'>
    <h1 class='text-2xl font-bold text-center py-4 bg-gray-100'>{props.title}</h1>
    {length() > 0 && (
      <ul class='list-none p-4'>
        <For each={store.todos}>
          {t => <TodoItem key={t.id} task={t} onToggle={toggleTask} onRemove={deleteTask} />}
        </For>
      </ul>
    )}
    <TodoForm onSubmit={addTask} />
  </div>
);

Не забудьте импортировать createMemo (аналог computed во Vue) из solid-js.

Компонент TodoItem⚓︎

Обратите внимание на разметку. Solid, хоть и похож на React, но не требует написания className вместо class, а также имеет некоторые специальные атрибуты JSX (например, classList, позволяющий обойтись без внешних библиотек типа clx).

function TodoItem({ task, onToggle, onRemove }) {
  const toggleTask = () => onToggle(task.id);
  const deleteTask = () => onRemove(task.id);

  return (
    <li class='flex items-center mb-2 hover:cursor-pointer' onClick={toggleTask}>
      <input type='checkbox' class='mr-2' checked={task.completed} readOnly />
      <span classList={{ 'line-through': task.completed }}>{task.title}</span>
      <div class="ml-auto text-gray-400 hover:text-gray-600">
        <button class="i-heroicons-trash w-6 h-6" onClick={deleteTask} />
      </div>
    </li>
  );
}

export default TodoItem;

Примечание

Обработчики событий в Solid обычно имеют форму onclick или onClick в зависимости от стиля. А вот название самого события всегда в нижнем регистре.

Компонент TodoForm⚓︎

А здесь мы просто определяем переменную inputRef и связываем её с помощью атрибута ref с элементом input:

function TodoForm(props) {
  let inputRef;

  const addTask = () => {
    // Передаем значение текстового ввода
    props.onSubmit(inputRef.value);

    // Очищаем текстовый ввод
    inputRef.value = '';

    // И фокусируемся на нём
    inputRef.focus();
  };

  return (
    <div class='p-4 bg-gray-100'>
      <div class='flex items-center'>
        <input
          ref={inputRef}
          type='text'
          class='flex-1 mr-2 py-2 px-4 rounded-md border border-gray-300'
          placeholder='Новая задача'
          autoFocus
        />
        <button
          class='bg-blue-500 hover:bg-blue-600 text-white py-2 px-4 rounded-md'
          onClick={addTask}
        >
          Добавить
        </button>
      </div>
    </div>
  );
}

export default TodoForm;

Документация⚓︎

Если вы заинтересовались Solid JS, загляните на официальный сайт или в переведённую документацию.

Сравнить Solid с другими фреймворками можно здесь.

Заключение⚓︎

Итак, мы закончили наше простое приложение TODO на Solid JS:

  • успешно адаптировали исходную разметку для использования с Solid
  • познакомились с некоторыми директивами библиотеки
  • нашли много похожих элементов (включая названия хуков и методов) из других фреймворков/библиотек

В следующей части этой серии мы наконец подведём итоги.


Скачать готовый проект

Комментарии