A minimal example of integrating TypeScript and Vue.js using Webpack 4
LAST UPDATE of NPM packages and configuration to latest versions + successful test of the setup: 2019-01-22.
Please note that parts of the following documentation are outdated (how else could it be in crazy JS land...). However, the code and configuration was found compile and run fine the last time on the date mentioned above.
If you just want to get a Typescript+Vue.js project up and running as quickly as possible and don't feel the need to understand the individual steps, simply do the following (assuming that you already have git and npm installed on your system):
git clone https://github.com/sebastian-bechtold/vue-typescript-starterpack.git
npm install
npm run dev
Your Typescript+Vue.js sample web app is now ready for viewing at http://localhost:8080
If you want to understand a little more about what's going on here, read on. The following sections describe the individual project set-up steps in some more detail (work in progress!):
When I decided to start using Vue.js, it was clear for me from the first moment on that I wanted to use it together with TypeScript. I had used TypeScript for a while and was more than convinced by the amazing increase in productivity that TypeScript provides compared to plain JavaScript. So, I started to look for a tutorial about how to set up the two to work together. I googled for hours, and while I found many articles and sample projects on the topic, none of the ones that I tried was fully functional in the way that I wanted. Sometimes, instructions were written for outdated versions of the involved software packages, sometimes a small but critical step was missing.
After many hours of googleing, reading, trial and error, I finally had a working project up and running. I'm sharing this project here with you in the hope to save you these hours if you're in a similar situation.
The instructions and sample code provided here worked on my system as of 2019-01-22 with the latest versions of the involved packages available on npm at this time. Since this might not mean much in the crazily fast-moving world of front-end web development - good luck!
npm init -y
This command will create a file named 'package.json' in your project folder. package.json is the central configuration and "housekeeping" file of npm, the Node Package Manager. We'll use npm to download and manage all software packages required by this project.
Open your newly created package.json with your text editor and add the following properties to the "scripts" section:
"build": "webpack",
"dev" : "webpack-dev-server"
These lines will later enable you to start the development server from your command line by typing npm run dev
in your project directory, and to create a permanent build of your project by typing npm run build
.
Now, your package.json should look like this:
{
"name": "vue-ts-sample",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"build": "webpack",
"dev" : "webpack-dev-server",
"test": "echo \"Error: no test specified\" && exit 1"
},
"keywords": [],
"author": "",
"license": "ISC"
}
npm install css-loader node-sass sass-loader style-loader ts-loader typescript vue vue-class-component vue-loader vue-template-compiler webpack webpack-dev-server
Let's take a closer look at what we're downloading here:
- typescript: The TypeScript compiler.
- vue: The Vue.js library.
- vue-class-component: A TypeScript extension that allows you to write Vue.js components as TypeScript classes. Not absolutely necessary to combine TypeScript with Vue, but highly recommended for much cleaner and TypeScript-idiomatic code structure.
- vue-template-compiler: A module to compile Vue.js templates to JavaScript.
- node-sass: A module to compile SASS code (enhanced CSS) to browser-readable "classic" CSS.
- webpack: A build tool that compiles different types of source files into something that can be run by the browser.
- css-loader, sass-loader, style-loader, ts-loader and vue-loader: These are plug-ins for Webpack that enable Webpack to compile CSS, SCSS/SASS, TypeScript and Vue component source files into a project.
- webpack-dev-server: A small web server that can be launched from the command line and makes your project accessible via http for development purposes. What makes it extremely helpful is that it automatically rebuilds and reloads your app when a source file has changed.
In your project root directory, create a file named webpack.config.js
with the following content. This will be your Webpack configuration file:
var path = require('path')
var webpack = require('webpack')
module.exports = {
entry: './src/index.ts',
output: {
path: path.resolve(__dirname, './dist'),
publicPath: '/dist/',
filename: 'build.js'
},
/*
sass-loader - Transforms Sass to CSS. (Input: Sass, Output: CSS)
css-loader - Transforms CSS to a JavaScript module. (Input: CSS, Output: JavaScript)
style-loader - Injects the CSS, that is exported by the JavaScript module, into a <style> tag at runtime. (Input: JavaScript, Output: JavaScript).
// Source: https://stackoverflow.com/questions/43913420/css-loader-vs-style-loader-vs-sass-loader
*/
module: {
rules: [
// TODO: Is this what we want?
/*
{
test: /\.css$/,
loader: "style-loader!css-loader"
},
*/
{
//test: /\.scss$/,
test: /\.(css|scss)$/,
//loaders: ["style-loader", "css-loader", "sass-loader"]
loaders: ["style-loader", "css-loader", "sass-loader"]
},
{
test: /\.vue$/,
loader: 'vue-loader',
options: {
esModule: true
}
},
{
test: /\.ts$/,
loader: 'ts-loader',
exclude: /node_modules/,
options: {
appendTsSuffixTo: [/\.vue$/],
}
},
{
test: /\.(png|jpg|gif|svg)$/,
loader: 'file-loader',
options: {
name: '[name].[ext]?[hash]'
}
}
]
},
resolve: {
extensions: ['.ts', '.js', '.vue'],
alias: {
'vue$': 'vue/dist/vue.esm.js'
}
},
devServer: {
historyApiFallback: true,
noInfo: true
},
performance: {
hints: false
},
devtool: '#eval-source-map'
}
if (process.env.NODE_ENV === 'production') {
module.exports.devtool = '#source-map'
// http://vue-loader.vuejs.org/en/workflow/production.html
module.exports.plugins = (module.exports.plugins || []).concat([
new webpack.DefinePlugin({
'process.env': {
NODE_ENV: '"production"'
}
}),
new webpack.optimize.UglifyJsPlugin({
sourceMap: true,
compress: {
warnings: false
}
}),
new webpack.LoaderOptionsPlugin({
minimize: true
})
])
}
In your project root directory, create a file named tsconfig.json
with the following content. This will be the configuration file for the TypeScript compiler:
{
"compilerOptions": {
"allowSyntheticDefaultImports": true,
"declaration": true,
"experimentalDecorators": true,
"lib": [
"dom",
"es5",
"es2015.promise"
],
"isolatedModules": false,
"module": "es2015",
"moduleResolution": "node",
"noImplicitAny": true,
"noImplicitThis": true,
"outDir": "lib",
"removeComments": true,
"strictNullChecks": true,
"suppressImplicitAnyIndexErrors": true,
"target": "es5"
},
"include": [
"src/**/*.ts"
]
}
mkdir src
mkdir src/components
<!doctype html>
<html>
<body>
<div id="app"></div>
</body>
<script src="./dist/build.js"></script>
</html>
import Vue from 'vue'
import App from './components/App.vue'
new Vue({
el: '#app',
render: h => h(App, {
props: { msg: 'Hello World!' }
})
})
This little file is critical! Without it, you won't be able to import
Vue component files (.vue) in your TypeScript code.
declare module "*.vue" {
import Vue from "vue";
export default Vue;
}
<template>
<div class="app">
<p>The message: <input v-model="message"/></p>
<p>TypeScript and Vue say:</p>
<h1>{{message}}</h1>
</div>
</template>
<script lang="ts">
import Vue from 'vue'
import Component from 'vue-class-component'
@Component({
props: {
// Define 'msg' as a prop (read below for details):
msg: String
}
})
export default class App extends Vue {
// 'msg' holds the initial message that was passed as a component tag attribute, a so-called 'prop'.
// In the '@Component' decorator above, 'msg' is defined as a prop.
msg : string;
// Members defined as a prop should not be modified by component code,
// so we need to create another member to store the modified message:
message: string = "";
// mounted() is automatically called when the component is added to the DOM. When this happens,
// we copy the initial message from the prop to a "normal" class member so that we can modify it
// through the input field:
mounted() {
this.message = this.msg;
}
}
</script>
<style>
div.app h1 {
font-family:sans-serif;
}
</style>
npm run dev
Now open the web browser of your choice and navigate to http://localhost:8080
. If everything is set up correctly, you should see something like this:
Note that the dev server works with a temporary build that is not stored on your disk. In order to produce permanent build files for deployment or use with another web server, use the following comand:
npm run build
A great (perhaps the best?) editor to work on Vue+TypeScript projects is Visual Studio Code with the Vetur plug-in. VS Code is famous for its great TypeScript support, and the Vetur plug-in provides capabilities like auto-complete and refactoring for .vue files (Vue.js single file component definitions). It's not yet as good as it could be (e.g. no auto-complete for TypeScript code in the template section), but definitely a lot better than plain text editing.