History of Javascript (until 2017)
Reference. This page is my notes of the article.
Old school way
Javascript first appeared in 1995. Old school way is to use a script tag, just like image in html.
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<title>JavaScript Example</title>
<script src="index.js"></script>
</head>
<body>
<h1>Hello from HTML!</h1>
</body>
</html>
// index.js
console.log("Hello from JavaScript!");
If you wanted to install a library like moment.js, you would download moment.min.js
(min
for minified or compressed) and add following line in header
of html:
<script src="moment.min.js"></script>
and in your index.js
console.log(moment().startOf("day").fromNow());
This is easy to understand, but painful to maintain because you’ve to manually maintain versions.
npm: package manager
There were several competing JS package managers starting 2010. Bower was popular until 2013. npm overtook it in 2015. Yarn picked up steam in 2016, but it’s just an alternate interface to npm. npm is package manager for node.js but that didn’t stop people from using it for front end. See package_managers/npm.md
for detailed notes on npm. Here’s some basic use:
// creates package.json
$ npm init
// add your dependency. updates package.json
$ npm install moment --save
your library is downloaded to node_modules
. Use it in your html:
<script src="node_modules/moment/min/moment.min.js"></script>
Now npm takes care of downloading and updating packages. But it’s still a hassle because you’ve to put path manually.
webpack: module bundler
Most programming languages have import
or include
something like that but js originally didn’t have it. This meant libraries had to share a global variable which is available for all the files loaded later. In 2009, a project named CommonJS was started with a goal to make JS runtime come out of browser closet. Figuring out import and export was big part of this project. Node.js is most well known implementation of the project!
This is how you do it in node. Since node has access to computer’s file system, it knows and loads the path to the exact js file of moment library.
// index.js
var moment = require('moment');
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
But this won’t work on browser because browsers are designed to get no file access. We need a module bundler to replace all the require statements with actual file contents. Result is one single bundled JS file. In 2011, browersify was the most popular bundler. But in 2015, webpack became the more widely used bundler, fueled by popularity of react, a front end framework. Here’s how install webpack:
# --save-dev means development dependencies
$ npm install webpack webpack-cli --save-dev
This create bundle in dist/main.js
./node_modules/.bin/webpack index.js
This bundle is the only file you add in your html header!
<!-- index.html -->
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>JavaScript Example</title>
<script src="dist/main.js"></script>
</head>
<body>
<h1>Hello from HTML!</h1>
</body>
</html>
babel: transpiler
Since browsers are slow to add new features, new and modern language features needed to be transpiled (kinda like compiling) to old school js. CoffeeScript used to be popular but now a days people use babel or TypeScript. babel converts next-gen JS to browser-compatible JS. TypeScript is essentially next-gen JS with optional static typing. Here’s how you install:
$ npm install @babel/core @babel/preset-env babel-loader --save-dev
Add the following in webpack.config.js to make babel work with webpack:
// webpack.config.js
module.exports = {
mode: 'development',
entry: './index.js',
output: {
filename: 'main.js',
publicPath: 'dist'
},
module: {
rules: [
{
test: /\.js$/,
exclude: /node_modules/,
use: {
loader: 'babel-loader',
options: {
presets: ['@babel/preset-env']
}
}
}
]
}
};
Now you can use latest modern js in index.js!
// index.js
import moment from 'moment';
console.log("Hello from JavaScript!");
console.log(moment().startOf('day').fromNow());
var name = "Bob", time = "today";
console.log(`Hello ${name}, how are you ${time}?`);
Run this to get bundled js file which you can use in browser.
$ ./node_modules/.bin/webpack
npm scripts: task runner
You can there are lot of build steps in the modern js. Can we automate these tasks using scripts? Grunt and Gulp used to be popular frontend task runners but npm itself has scripts and it’s the most popular option. You have to edit your package.json to add scripts:
{
"name": "modern-javascript-example",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"build": "webpack --progress --mode=production",
"watch": "webpack --progress --watch"
},
"author": "",
"license": "ISC",
"dependencies": {
"moment": "^2.22.2"
},
"devDependencies": {
"@babel/core": "^7.0.0",
"@babel/preset-env": "^7.0.0",
"babel-loader": "^8.0.2",
"webpack": "^4.17.1",
"webpack-cli": "^3.1.0"
}
We run the scripts using npm run
# build minified package
$ npm run build
# watch changes as we edit files
$ npm run watch
Conclusion
Lot of moving parts, true, but things are stabilizing. With great innovation comes great change! Frameworks like react come these days with tools to make setup easy like create-react-app
.
That’s all folks!