1.2.9 • Published 1 year ago

vanilla-to-react v1.2.9

Weekly downloads
-
License
ISC
Repository
github
Last release
1 year ago

vanilla-to-react

바닐라로 React(함수형 컴포넌트) 구현하기

해당 프로젝트는 React의 핵심 기능 5가지

  1. Virtual DOM
  2. JSX
  3. RealDOM 렌더링
  4. Diffing update
  5. useState hook(메모리 최적화는 없음)
  6. useEffect hook(dependency는 null을 제외한 primitive type 1개만 넣을 수 있음. clean up 제외)

를 구현했다.

다만 모든 html element에 props를 전달할 수 있게 코드를 짜면 작성해야 할 코드 양이 너무 방대해져서 생략했다.
state에 대한 변화나 이벤트를 달기 위해서는 아래처럼 timer API나 addEventListener를 사용해야 한다.
이외에도 일부 react에서 사용하는 것과 JSX 기능이 다른 게 존재할 수 있다.

// 이벤트 리스너 다는 방법

const Component = () => {
  // 렌더링될 때까지 기다림
  setTimeout(() => {
    document.querySelector('button').addEventListener('click', () => {
      alert('경고!');
    });
  }, 1000);

  return <button>click!</button>;
};

프로젝트 세팅 및 시작

(프로젝트를 js로 진행한다고 가정)

  1. npm install vanilla-to-react && npm install -D @babel/core @babel/preset-env @babel/preset-react babel-loader html-webpack-plugin webpack webpack-cli webpack-dev-server

  2. package.json 설정

{
  // ...
  "scripts": {
    "dev": "webpack serve --mode development",
    "build": "webpack --mode production"
  }
  // ...
}
  1. webpack.config.js 설정.
const HtmlWebpackPlugin = require('html-webpack-plugin');
const path = require('path');

module.exports = {
  entry: './src/index.js',
  output: {
    path: __dirname + '/dist',
    filename: 'bundle.js',
  },
  devServer: {
    contentBase: path.join(__dirname, 'dist'),
  },
  module: {
    rules: [
      {
        test: /\.js$/,
        exclude: /node_modules/,
        use: {
          loader: 'babel-loader',
          options: {
            presets: ['@babel/preset-env', '@babel/preset-react'],
          },
        },
      },
    ],
  },
  resolve: {
    extensions: ['.js'],
  },
  devServer: {
    port: 5000,
    hot: true,
  },
  plugins: [
    new HtmlWebpackPlugin({
      template: './src/index.html',
    }),
  ],
};
  1. src/index.html 선언.
<!DOCTYPE html>
<html lang="ko">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
    <script type="module" src="./index.js"></script>
  </head>
  <body>
    <div id="root"></div>
  </body>
</html>
  1. 컴포넌트 파일 구현. 파일 최상단에 바벨이 사용할 createElement 명시.
// src/App.js
/* @jsx createElement */

import { createElement, useState, useEffect } from 'vanilla-to-react';

const CoffeeList = () => {
  const [coffees, setCoffees] = useState([
    { title: '에스프레소', size: 'lg' },
    { title: '아메리카노', size: 'lg' },
  ]);

  useEffect(() => {
    setTimeout(() => {
      setCoffees([
        { title: '아메리카노', size: 'sm' },
        { title: '에스프레스', size: 'sm' },
        { title: '추가사항', size: 'sm' },
      ]);
    }, 2000);
  }, []);

  return (
    <ul>
      {coffees.map(({ title, size }, index) =>
        index === 2 ? (
          <div>
            <li class={size === 'lg' ? 'lg' : 'sm'}>{title}</li>
            <button>버튼!</button>
          </div>
        ) : (
          <li class={size === 'lg' ? 'lg' : 'sm'}>{title}</li>
        )
      )}
    </ul>
  );
};

export default CoffeeList;
  1. src/index.html에서 참조하고 있는 src/index.js 구현.
/* @jsx createElement */

import {
  createReactRoot,
  createElement,
  renderRealDOM,
} from 'vanilla-to-react';
import CoffeeList from './app.js';

const root = createReactRoot(document.querySelector('#root')); // live-dom으로 쿼리해도 상관 없음
root.render(CoffeeList);
1.2.9

1 year ago

1.2.8

1 year ago

1.2.7

1 year ago

1.2.6

1 year ago

1.1.6

1 year ago

1.0.6

1 year ago

1.0.5

1 year ago

1.0.4

1 year ago

1.0.3

1 year ago

1.0.2

1 year ago

1.0.1

1 year ago

1.0.0

1 year ago