Express에 Webpack & Babel 을 이용한 ES6 환경 설정 ( +generator 사용편)
브라우저 호환성 ( browser compatibility)
웹브라우저에서는 다양한 브라우저가 사용되고 있으며, 그 안에서
↑ 브라우저 따른 사용 가능성을 확인할 수 있음
📌 브라우저 호환성을 높이는 몇가지 방안
🍎 polyfill
특정 기능이 지원되지 않는 브라우저를 위해 사용할 수있는 코드조각 or 플러그인
ex) bind 의 polyfill
bind는 ES5에서 추가되었는데, 만약 브라우저에 bind 기능이 없다면 , 아래 polyfill기능을 추가해 놓고
후에 기능 추가가 되면 간단히 삭제하여 관리할 수 있다.
bind가 있지 않은 function 이라면? => 아래 코드를적용해라
출처:developer.mozilla.org/ko/docs/Web/JavaScript/Reference/Global_Objects/Function/bind
🍎 Transpiling ( 트랜스파일 )
소스코드 -> 소스코드 변환
최신사양으로 작성된 코드를 기존 응용프로그램과 호환되는 코드로 변환함으써 오류 없이 작동할 수 있게 하는 활동
ex) Babel , Google Traceur...
ex) arrow function
arrow function 은 ES5에서 지원하지 않음
// ES 6
() => {}
// ES 5
function() { }
이런식으로 지원되지 않는 속성들을 이전 버전으로 변경해줌
📌 Babel
하위브라우저가 지원가능하도록 transpiling 해줌
CLI를 지원하고 다양한 플러그인을 통해 브라우저 호환성을 관리할 수있음
bundle 파일을 만들어 배포
ES model 방식으로 코딩한 것의 관계를 분석한 후 -> bundel로 만들어줌
Babel 기본 구성
{
"presets": [
["@babel/preset-env", {"useBuiltIns":"entry", "corejs":3}],
["@babel/preset-react"]
],
"plugins": [
["styled-jsx/babel", {"optimizeForSpeed": true, "vendorPrefixes": true, "sourceMaps": false}]
]
}
☻ presets
❖ babel/preset-env | 타겟 브라우저에 필요한 구문변경, Polyfill을 제공하는 프리셋 |
- useBuiltIns :entry (usage) |
전체 import를 특정 모듈들의 import로 변경 |
❖ babel/preset-react | JSX코드를 React.CreateElement 호출 코드로 변환 |
☻ Plugin
styled-jsx/babel | styled-jsx 문법을 변환해주는 플러그인 |
Babel 설정 개선
{
"presets": [
[
"@babel/preset-env",
{
"targets" : {
"ie":11
},
"useBuiltIns" : "usage",
"corejs" : {"version": 3, "proposals": true }
}
]
],
"plugins": []
}
@babel/preset-env
자주 사용한는 것들을 set로 만들어서 설정 가능하게 함
IE 11 버전으로 만들어줘
"usage"
필요한 polyfill을 추가해주기
ex) flat사용 ->
corejs
전역 스코프를 오염시키지 않는 corejs 3 버전
📌 Webpack
JavaScript Application 의 module bundler
한 시스템에는 많은 JavaScript 파일이 존재하는데 이 파일들을 하나의 bundle로 묶어서 관리하게 함
ex) babel 사용시 - babel을 돌리며 module들을 transpiling하며 읽어줘 bundle로 만듦
ex) 비슷한 도구 : Browsify(속도 더 느림) , Grunt, Gulp ( dependency 개념 없음) ...
⭐️ bundle
: SW와 HW가 함꼐 작동하는데 필요한것을 포함하는 package
: 각 모듈들의 의존성 관계를 파악하고 어필
🙂 webpack.config.js와 함께보는 webpack구성
☻ entry
dependency 관계를 만들기 위한 input source
module.exports = {
entry: './src/app.js'
}
☻ output
webpack 한 결과물 위치 지정
const path = require('path');
module.exports = {
entry: './src/app.js',
output: {
path: path.resolve(__dirname, 'dist'),
filename: 'bundle.js',
},
-> dist 디렉토리에 bundle.js의 이름으로 저장
☻ loader
webpack은 원래 javascript 와 json 만 이해함
다른 타입의 파일들을 webpack 이 이해하고 처리가능한 모듈로 변환 시킴
( import 들을 분석해 .js?변환! )
const path = require('path');
module.exports = {
entry: './src/app.js',
output: {
path: path.resolve(__dirname, 'dist'),
filename: 'bundle.js',
},
module: {
rules: [
{
test: /\.js$/,
use: 'babel-loader',
exclude: /node_modules/
},
{
test: /\.css$/i,
use: ["style-loader", "css-loader"],
}
,
]
},
};
.js 파일 : node_modules 제외하고 babel-loader를 이용
.css 파일: style-loader, css-loader를 이용
☻ plugin
bundel 된결과 (최종파일)의 후속처리
주석, 공백, console... 지워주는
모듈의 의존성을 파악해서 모듈방식의 프로그래밍을 더 잘 할 수 잇음
const HtmlWebpackPlugin = require('html-webpack-plugin');
module.exports = {
entry: './src/app.js',
output: {
path: path.resolve(__dirname, 'dist'),
filename: 'bundle.js',
},
...
plugins: [new HtmlWebpackPlugin(
{
template: 'src/app.html' //템플릿 경로 지정
}
)]
};
HtmlWebpackPlugin : HTML 파일로 output (dist/ bundle.js) 에 생성
HTML 파일을 후처리하고 빌드타임의 값을 넣거나 코드를 압축
loaders | plugin |
파일단위로 해석하고 변환하는 과정에 모듈 처리 | bundle된 결과물의 형태를 처리 |
HtmlWebpackPlugin 외에도 자주 사용하는 플러그인들을 확인하고 사용할 수 있다.
다양한 플러그인 참고 링크 : bogyum-uncle.tistory.com/112
개발환경을 직접 구축해보자❗️
* Webpack 구성
* Babel 포함
* express -generator 사용
local 설치 | global 설치 (-g) |
Nodejs 프로젝트 하위 > node_modules 설치 | 계정 하위 APP data에 node_modules 설치 모든 프로젝트에서 공유해 해당 모듈 사용 가능 |
1️⃣ npm 프로젝트 생성 (express-generator 이용)
node_modules , package.json
sudo npm i -g express-generator //mac 유저라 sudo 권한으로 global 프로그램 설치
// express <프젝name> --view=ejs // 템플릿 엔진 ejs 로 생성 ( --veiw=pug .. 가능)
express testenv --view=ejs
cd testenv
npm install
npm run start // 서버 동작 확인 가능
2️⃣ Babel 설치
2-1) Babel 설치
babel 필수 모듈 설치
npm install @babel/core @babel/cli @babel/preset-env --save-dev
2-2) 파일구조 변경 및 app.js 수정
src/index.html : 웹 화면 구성
public/main.js : main javascript
index.htm예시
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<h1 class="hi">hello world</h1>
<div class="wrap">
<div id="log"></div>
</div >
<script src="../dist/bundle.js"></script>
</body>
</html>
main.js 예시
import MainService from './Js/mainService.js';
window.addEventListener("DOMContentLoaded", ()=> {
const targetEl = document.querySelector("#log");
const service = new MainService({targetEl})
const datalist = [1,2,3,4,[5,6,[7]]];
const subHtml = service.init(datalist);
targetEl.innerHTML += `datalist is ${subHtml}`;
})
mainService.js 예시
export default class MainService {
constructor({targetEl}) {
this.targetEl = targetEl;
this.dataList = "";
}
init(dataList) {
this.dataList = dataList.flat(2).join('');
return this.render(this.dataList);
}
render(data) {
return `<span>${data}</span>`
}
}
2-3) babel.config.json 파일 생성
IE 11 에서 지원가능 하도록 transpiling
presets(plugin의 집합) , plugin
// 1) 타깃브라우저에 필요한 플러그인만 삽입하도록 설정하는 옵션
{
"presets": [
["@babel/preset-env", {
"targets": "> 0.25%, not dead"
}],
]
}
//2) 타겟 브라우저 설정하는 버전
{
"presets": [
[
"@babel/preset-env",
{
"targets" : {
"ie":11
},
"useBuiltIns" : "usage",
"corejs" : {"version": 3, "proposals": true }
}
]
],
"plugins": []
}
targets: 원하는 브라우저 범위 설정 가능
npx babel public/javascripts/main.js //babel 변환 여부 확인
Polyfill 설정
npm install --save core-js@3
3️⃣ webpack 설치
--save-dev | --save (-S) |
package.json 의 devDependencies에 포함 개발시에만 포함되고 배포시 publishing에 포함되지 않음 |
package.json의 dependencies에 포함 배포시점에 함께 publishing 됨 |
npm install webpack webpack-cli --save-dev
npx webpack//아직은 오류
webpack 은 서버를 실행시키는 것은 아님
단, 연동해줄 수 있는 방법이 있음
3-1) webpack.config.js 파일 생성
const path = require('path');
module.exports = {
mode: "development",//개발모드
entry: ['./public/main.js', './public/style/index.css' ],//entry point(분석할 진입점)
output: {
path: path.resolve(__dirname, "dist"),
filename: "bundle.js",
},
}
//웹팩 첫 실행
npx webpack -w
dist/디렉토리에 bundle.js 파일 생성 확인
app.js에 추가
app.use(express.static(path.join(__dirname, 'dist')));
3-2) Webpack에 Babel연동
webpack 에 바벨을 load 해서 사용해야 함
npm install babel-loader --save-dev
webpack.config.js 에 module 항목과 rule 추가.
const path = require('path');
module.exports = {
mode: "development",
entry: ['./public/main.js', './public/style/index.css' ],
output: {
path: path.resolve(__dirname, "dist"),
filename: "bundle.js",
},
module: {
rules: [
{
test: /\.js$/,
use: 'babel-loader',
exclude: /node_modules/
},
{
test: /\.css$/i,
use: ["style-loader", "css-loader"],
}
,
]
},
};
test 가 .js로 긑나면 바벨 로더가 처리
node_modules 제외 htmlwebpackplugin
css 변환을 위한 loader 추가
npm i --save-dev css-loader
npm i --save-dev stylep-loader
3-3) webpack 에 plugin 추가
npm install --save-dev html-webpack-plugin
const path = require('path');
const HtmlWebpackPlugin = require('html-webpack-plugin');
module.exports = {
mode: "development",
entry: ['./public/main.js', './public/style/index.css' ],
output: {
path: path.resolve(__dirname, "dist"),
filename: "bundle.js",
},
module: {
rules: [
{
test: /\.js$/,
use: 'babel-loader',
exclude: /node_modules/
},
{
test: /\.css$/i,
use: ["style-loader", "css-loader"],
}
,
]
},
plugins: [new HtmlWebpackPlugin(
{
template: 'src/index.html'
}
)]
}
+ devtool추가를 통한 크롬 디버거 사용 가능
module.exports = {
mode: "development",
entry: ['./public/main.js', './public/style/index.css' ],
output: {
path: path.resolve(__dirname, "dist"),
filename: "bundle.js",
},
devtool: 'inline-source-map',
module: { ...
위와 같이 devtool을 추가하게 되면
크롬개발자도구 - Sources - . 디렉토리에서 bundle 되기전 코드를 이용하여 디버깅이 가능하다.
4️⃣ Package.json 수정
이처럼 scripts 를 설정한 뒤
실행 시 두 명령어를 실행시키면 web이 나타나게 된다!
npm run start // express 서버 실행
npm run build // wepback 실행