Clone source with SSH url:
git clone git@github.com:anhchangvt1994/webpack-project--template-react-ts.git
Install:
cd webpack-project--template-react-ts
If use npm
npm install
If use yarn 1.x
yarn install
This project is an advanced structure and configuration of scaffolded Webpack + React + Typescript project.
- env - emvironment directory
- src - include assets and coding of project
- tailwind.config.cjs
- webpack.config.js - unplugin-auto-import configuration
- webpack.production.config.js - optimization splitchunks configuration, External configuration
├── env/
│ ├── env.[prefix].mjs
│ ├── env-register.mjs
│ └──
└──
env directory contains environment variable files used to manage environment variable by using .mjs file. You will define environment variables in .mjs file instead of .env file.
- I think defining environment variables in javascript file will similar with JS developer and better for managing than .env file.
Compare them
// env.router.mjs
export default {
prefix: 'router',
data: {
home: {
path: '/',
id: 'HomePage',
},
},
}
# .env
ROUTER_HOME_PATH=/
ROUTER_HOME_ID=HomePage
- Think that you can define any type (not only string like .env) when define env in javascript file.
Imagine that you need to define a payment code validation array
// env.router.mjs
export default {
prefix: 'payment',
data: {
valid_code: [0, 1, 2, 3],
},
}
# .env
PAYMENT_VALID_CODE=[0,1,2,3] #wrong
PAYMENT_VALID_CODE="[0,1,2,3]" #right (you must stringify it)
- The hot benefit of this advanced structure bring out for Environment Variables is the ImportMeta.d.ts generating automation. With this ability, the code editor will auto suggestion available env for you.
You have a large env difination and have to open file, cop and paste variable key when want to use it. Forget it !!!
Imagine that you need create a new env for an api title (prefix) to store all of api endpoint string
- Create an env.api.mjs file and finish that
// env.api.mjs
export default {
prefix: 'api',
data: {
user: {
info: '/api/user/info',
edit: '/api/user/edit',
},
product_list: '/api/product',
},
}
- Open env-register.mjs and regist it
// env-register.mjs
import ENV_API from './env.api.mjs'
export default [ENV_API]
Tada! Done! you're so cool
├── src/
│ ├── App.ts
│ ├── App.vue
│ ├── assets/...
│ ├── pages/...
│ ├── components/...
│ ├── config/...
│ └── utils/...
└──
The src directory contains the resource's assets and logic of your codes like:
file / directory | Description |
---|---|
App.ts | contains initialization code of vue |
App.vue | contains default layout code of vue |
assets/ | contains asset files styles: assets > styles > main.scss images: assets > static > images > logo.svg |
pages/ | contains files of pages layout (ex: HomePage.vue) |
components/ | contains files of component global: components > [GlobalComponentName].tsx page: components > HomePage > ProductSection.tsx |
config/ | contains files of libs or plugins configuration react-router-dom: config > router > index.ts redux: config > store > index.ts |
utils/ | contains files of your customization like React Hooks, Libs, Plugins |
- If your code editor has TSServer, the paths options of tsconfig.json will provide a list of alias for your code editor. That will make you happy with auto alias suggestion when you're typing.
// Normal way you must
import './assets/styles/...'
// tsconfig with paths options
import 'assets/styles/...'
In this case, that looks like the same, except " ./ ". But when you move index.ts to another location (ex: move it into pages/), the path with " ./ " will wrong and the path with alias still right.
- You will see that the images/ directory placed in static/ directory. Because in this project the static/ is set for copying instead of asset url handling.
Normal case in React project, you can handle asset files with some solutions
// 1. import and use it
import Logo from 'assets/images/logo.svg'
return <img src={Logo} />
// 2. require
return <img src={require('assets/images/logo.svg')} />
In case use publicDir, you just easy set static files like a string
<!-- Very similar and you finish! -->
<img src="/images/logo.svg" />
All of your tailwind config for your project tailwind config docs
In the normal, think that if you want to use useState in react you have to:
import { useState } from 'react'
const [something, setSomething] = ref(value)
But if you configed auto-import before, you just do like code below. And don't worry about suggestion or wrong linting.
const [something, setSomething] = ref(value)
I configed the auto-import into webpack.config.js, the syntax of configuration is like this
{
plugins: [
require('unplugin-auto-import/webpack')({
// targets to transform
include: [
/\.[tj]sx?$/, // .ts, .tsx, .js, .jsx
/\.md$/, // .md
],
imports: [
// presets
'react',
{
react: [
['*', 'React'],
'Suspense',
'componentDidCatch',
'StrictMode',
'createContext',
],
},
{
'react-dom/client': ['createRoot'],
},
{
'styled-components': [
['default', 'styled'],
'createGlobalStyle',
'keyframes',
],
},
{
polished: ['rgba'],
},
],
dts: './config/auto-imports.d.ts',
eslintrc: {
enabled: true,
filepath: './config/.eslintrc-auto-import.json',
},
})
],
}
Options | Description |
---|---|
include | Is list of file extension that will be applied the auto-import |
imports | Set what dependencies and values will be valid to auto-import, unplugin-auto-import available some popular libs like: vue, react, solid ... see more Another hand, you can custom some of new your auto-import by using advanced configuration |
dts | Where you want to place auto-imports.d.ts file, this file will support for auto suggestion ability |
eslintrc | Where you want to place .eslintrc-auto-import.json file, this file will support for linting validation |
This configuration used to support for splitting chunks of files. This solution help to reduce the file's sizes in loading processes by loading multiple files with smaller sizes.
Reduce file's sizes, can enhance loading resource performance.
Note: Don't use this solution for all cases, because the large amount of file be loaded in the bad internet connection will make the loading resource process be slower and can be broken.
I configed the NormalSplitChunks into webpack.production.config.js, the syntax of configuration is like this
cacheGroups: {
styles: {
type: 'css/mini-extract',
priority: 100,
minSize: 1000,
maxSize: 50000,
minSizeReduction: 50000,
},
vendor: {
chunks: 'all',
test: /[\\/]node_modules[\\/]/,
filename: '[chunkhash:8].js',
enforce: true,
reuseExistingChunk: true,
},
utils: {
chunks: 'all',
test: /[\\/]utils[\\/]/,
filename: '[chunkhash:8].js',
reuseExistingChunk: true,
minSize: 10000,
maxSize: 100000,
},
config: {
chunks: 'all',
test: /[\\/]config[\\/]/,
filename: '[chunkhash:8].js',
reuseExistingChunk: true,
minSize: 10000,
maxSize: 100000,
},
},
In this case, I configed to split any files in node_modules, utils and config directories. About pages and components directories you can use dynamic import to manual handle split-chunks.
You can search more about splitchunks of webpack in here
NOTE: The webpack.development.config.js is the same, but instead config minSize and maxSize, I configed enforce:true
for all of them.
This configuration is an optional, you can read or ignore it. The ESM External CDN is a solution used to replace some node_module dependencies by the corresponding ESM module in a CDN hosting.
Imagine that you have a code like this
import { useState } from 'react'
The 'react' is a node_module dependency, you can see that infor in package.json.
After the build tool compiles the file include the import dependencies syntax, the system will print dependency's logic code into owner file or split it into another chunk if you have configed split-chunks for that dependency. But when you define external for that dependency, the system will replace it with the synchronize insert script syntax like split-chunks case, but different about where src="from". In ESM External CDN case the src will from CDN hosting.
The configuration syntax like this
// [key is dependency name]: value is esm cdn url (in this case I use https://esm.sh/)
externals: {
react: 'module https://esm.sh/react@18.2.0',
'react-dom': 'module https://esm.sh/react-dom@18.2.0',
'styled-components': 'module https://esm.sh/styled-components@5.3.6',
polished: 'module https://esm.sh/polished@4.2.2',
},
You can use below command line to try
npm run build:esm