直接上步驟css
1. 使用VSCode創建項目目錄結構以下:html
文檔結構前端
wabg
├── http
│ ├── app.js
│ ├── controllers
│ │ └── index.js
│ ├── directives
│ │ └── index.js
│ ├── services
│ │ └── index.js
│ ├── templates
│ │ └── index.tmpl.html
│ └── views
│ └── index.js
├── package.json
└── webpack.config.js
2、配置webpack和angularjsnode
1. 安裝webpack和angularjs所需的依賴項jquery
npm install webpack webpack-dev-server angular angular-ui-router jquery html-webpack-plugin ngtemplate-loader html-loader --save-dev
2. 建立一個簡單的頁面,經過這個很是簡單的頁面,驗證webpack和angular各要素是否可以正常使用,該頁面要素包括:webpack
紅色部分是點擊「getAuthor」按鈕後由AngularJS的Controller賦值而出現的;git
藍色部分是經過一個directive渲染的angularjs
綠色部分是經過jQuery動態添加的github
文檔結構以下:web
wabg
├── dist
│ ├── index.html
│ └── js
│ ├── angular.bundle.js
│ ├── angular.bundle.js.map
│ ├── app.bundle.js
│ ├── app.bundle.js.map
│ ├── vendor.bundle.js
│ └── vendor.bundle.js.map
├── http
│ ├── app.js
│ ├── controllers
│ │ ├── home
│ │ │ └── home.controller.js
│ │ └── index.js
│ ├── directives
│ │ ├── demo
│ │ │ └── demo.directive.js
│ │ └── index.js
│ ├── services
│ │ └── index.js
│ ├── templates
│ │ └── index.tmpl.html
│ └── views
│ ├── demo
│ │ └── demo.view.html
│ └── index.js
├── package.json
└── webpack.config.js
下面爲本頁面用到的全部代碼
2. /webpack.config.js 文件
'use strict'; var path = require('path'); var webpack = require('webpack'); var HtmlWebpackPlugin = require('html-webpack-plugin'); module.exports = { devtool: 'evel-source-map', context: path.resolve(__dirname + '/http'), entry: { app: './app.js', vendor: ['jquery'], angular: ['angular', 'angular-ui-router'] }, output: { path: path.resolve(__dirname + '/dist'), filename: 'js/[name].bundle.js', publicPath: '', }, devServer: { historyApiFallback: true, inline: true }, module: { rules: [ { test: /\.view.html$/, use: [ { loader: 'ngtemplate-loader' }, { loader: 'html-loader' } ], } ] }, plugins: [ new HtmlWebpackPlugin({ template: __dirname + '/http/templates/index.tmpl.html', chunksSortMode: function (chunk1, chunk2) { var order = ['angular', 'vendor', 'app']; var order1 = order.indexOf(chunk1.names[0]); var order2 = order.indexOf(chunk2.names[0]); return order1 - order2; } }), new webpack.optimize.CommonsChunkPlugin({ names: ['vendor', 'angular'], filename: 'js/[name].bundle.js' }), new webpack.ProvidePlugin({ $: 'jquery', jQuery: 'jquery', 'window.jQuery': 'jquery' }) ] };
3. /http/app.js 文件
'use strict'; var angular = require('angular'); var uirouter = require('angular-ui-router'); var services = require('./services'); var views = require('./views'); var directives = require('./directives'); var controllers = require('./controllers'); angular.module('wabgApp', [uirouter.default, services.default, views.default, directives.default, controllers.default]);
4. /http/services/index.js 文件
'use strict'; var angular = require('angular'); var servicesModule = angular.module('wabgApp.services', []); exports = module.exports = { default: servicesModule.name };
5. /http/controllers/index.js 文件
'use strict'; var angular = require('angular'); var homeController = require('./home/home.controller'); var controllersModule = angular.module('wabgApp.controllers', []) .controller(homeController.name, ['$scope', homeController]); exports = module.exports = { default: controllersModule.name, homeController: homeController.name };
6. /http/controllers/home/home.controller.js 文件
'use strict'; function homeController($scope) { $scope.name = 'think8848'; $scope.getAuthor = function (name) { $scope.author = name; }; } exports = module.exports = homeController;
7. /http/directives/index.js 文件
'use strict'; var angular = require('angular'); var demoDirective = require('./demo/demo.directive'); var directivesModule = angular.module('wabgApp.directives', []) .directive(demoDirective.name, ['views', demoDirective]); exports = module.exports = { default: directivesModule.name, demoDirective: demoDirective.name };
8. /http/directives/demo/demo.directive.js 文件
'use strict'; function demoDirective(views) { return { restrict: 'E', templateUrl: views.demo, link: function (scope, elem) { $(elem).after('<div style="color: green">The content generated by jQuery</div>'); } }; } exports = module.exports = demoDirective;
9. /http/views/index.js 文件
'use strict'; var angular = require('angular'); function views() { return { demo: require('./demo/demo.view.html') }; } exports = module.exports = views(); var viewsModule = angular.module('wabgApp.views', []) .constant(views.name, exports); exports.default = viewsModule.name;
10. /http/views/demo/demo.view.html 文件
<div style="color:blue">This is a demo directive.</div> <br/>
11. /http/templates/index.tmpl.html 文件
<!DOCTYPE html> <html ng-app="wabgApp"> <head> <meta charset="UTF-8"> <title>Webpack App</title> <style> .ng-cloak { display: none; } </style> </head> <body> <div id="root" ng-controller="homeController"> The Author is: <lable ng-ng-cloak class="ng-cloak" style="color:red">{{author}}</lable><br/> <input type="button" ng-click="getAuthor(name)" value="getAuthor" /> <br/> <br/> <demo-directive></demo-directive> </div> </body> </html>
12. 修改 package.json 文件中的 scripts 節點以下:
"scripts": { "test": "echo \"Error: no test specified\" && exit 1", "start": "webpack", "server": "webpack-dev-server --open" },
13. 運行 npm run server 命令
運行界面效果
1. 當前Bootstrap的最新版本是v4.0 Beta,在使用webpack時有兩點須要注意:一是默認的樣式文件爲 scss 樣式;二是當前v4.0 Beta版中默認不提供圖標了。首先依然是安裝依賴項
npm install bootstrap@4.0.0-beta expose-loader style-loader css-loader postcss-loader precss autoprefixer sass-loader url-loader file-loader popper.js exports-loader node-sass --save-dev
2. 爲Bootstrap加入圖標支持,圖標使用FontAwesome(官網)
2.1 先下載Bootstrap 4.0.0 beta版(本人雖不是前端,可是對前端框架和庫的版本也無力吐槽了)的源代碼,友情提示:經測試 npm 上下載的發佈包是沒法順利集成圖標的
git clone https://github.com/twbs/bootstrap.git
2.2 下載FontAwesome字體包(本文下載到的版本是v4.7.0)
2.3 將下載好的FontAwesome解壓後,相關的子目錄放到指定位置
font-awesome 和 scss 放到 bootstrap 源代碼根目錄下, fonts 放到 bootstrap 源代碼的 dist 目錄下
2.4 使用VSCode打開 bootstrap 源代碼根目錄,修改 /scss/bootstrap.scss 文件,在最下面添加兩行代碼,注意:必定要在文件最後保留一行空行
$fa-font-path: "../fonts"; @import "../font-awesome/scss/font-awesome.scss";
2.5 使用 npm install 下載依賴項,而後再使用 npm run dist 命令發佈
在發佈的過程當中,跳出來不少綠的,藍的說明沒問題,過會兒就行了
2.6 使用源代碼包中的內容替換 wabg 項目中 node_modules/bootstrap 中的相應文件夾
2.7 編輯 /wabg/node_modules/bootstrap/scss/bootstrap.scss 文件,將以前代碼上級目錄的 ../ 替換爲 ~bootstrap/ ,在 fonts 目錄前面加上 dist 目錄
至此,Bootstrap的準備工做完成,下面開始添加演示代碼,項目文檔結構以下:
wabg ├── dist │ ├── index.html │ └── js │ ├── angular.bundle.js │ ├── angular.bundle.js.map │ ├── app.bundle.js │ ├── app.bundle.js.map │ ├── vendor.bundle.js │ └── vendor.bundle.js.map ├── http │ ├── app.js │ ├── controllers │ │ ├── home │ │ │ └── home.controller.js │ │ └── index.js │ ├── directives │ │ ├── demo │ │ │ └── demo.directive.js │ │ ├── index.js │ │ └── menu │ │ └── menu.directive.js │ ├── scss │ │ └── index.scss │ ├── services │ │ └── index.js │ ├── templates │ │ └── index.tmpl.html │ └── views │ ├── demo │ │ └── demo.view.html │ ├── index.js │ └── menu │ └── menu.view.html ├── package.json └── webpack.config.js
3. 編輯 webpack.config.js 文件
'use strict'; var path = require('path'); var webpack = require('webpack'); var HtmlWebpackPlugin = require('html-webpack-plugin'); module.exports = { devtool: 'evel-source-map', context: path.resolve(__dirname + '/http'), entry: { app: './app.js', vendor: ['jquery', 'popper.js', 'bootstrap', 'bootstrap/js/dist/util.js', 'bootstrap/js/dist/dropdown.js'], angular: ['angular', 'angular-ui-router'] }, output: { path: path.resolve(__dirname + '/dist'), filename: 'js/[name].bundle.js', publicPath: '', }, devServer: { historyApiFallback: true, inline: true }, module: { rules: [ { test: /\.view.html$/, use: [ { loader: 'ngtemplate-loader' }, { loader: 'html-loader' } ], }, { test: /\.css$/, use: [{ loader: 'style-loader' }, { loader: 'css-loader' }] }, { test: /\.(scss)$/, use: [{ loader: 'style-loader', }, { loader: 'css-loader', }, { loader: 'postcss-loader', options: { plugins: function () { return [ require('precss'), require('autoprefixer') ]; } } }, { loader: 'sass-loader' }] }, { test: /\.(jpg|png|gif)(\?.*$|$)/, use: [ { loader: 'url-loader', options: { limit: 10000, name: './images/[name].[ext]' } } ] }, { test: /\.(woff|woff2|eot|ttf|svg)(\?.*$|$)/, use: [{ loader: 'file-loader', options: { name: './fonts/[name].[ext]' } }] } ] }, plugins: [ new HtmlWebpackPlugin({ template: __dirname + '/http/templates/index.tmpl.html', chunksSortMode: function (chunk1, chunk2) { var order = ['angular', 'vendor', 'app']; var order1 = order.indexOf(chunk1.names[0]); var order2 = order.indexOf(chunk2.names[0]); return order1 - order2; } }), new webpack.optimize.CommonsChunkPlugin({ names: ['vendor', 'angular'], filename: 'js/[name].bundle.js' }), new webpack.ProvidePlugin({ $: 'jquery', jQuery: 'jquery', 'window.jQuery': 'jquery', Popper: ['popper.js', 'default'], Util: 'exports-loader?Util!bootstrap/js/dist/util', Dropdown: 'exports-loader?Dropdown!bootstrap/js/dist/dropdown' }) ] };
4. 修改 /http/app.js 文件
'use strict'; var angular = require('angular'); var uirouter = require('angular-ui-router'); var scss = require('./scss/index.scss'); var services = require('./services'); var directives = require('./directives'); var controllers = require('./controllers'); var views = require('./views'); angular.module('wabgApp', [uirouter.default, services.default, views.default, directives.default, controllers.default]);
5. 編輯 /http/directives/index.js 文件
'use strict'; var angular = require('angular'); var demoDirective = require('./demo/demo.directive'); var menuDirective = require('./menu/menu.directive'); var directivesModule = angular.module('wabgApp.directives', []) .directive(demoDirective.name, ['views', demoDirective]) .directive(menuDirective.name, ['views', menuDirective]); exports = module.exports = { default: directivesModule.name, demoDirective: demoDirective.name, menuDirective: menuDirective };
6. 建立 /http/directives/menu/menu.directive.js 文件
'use strict'; function menuDirective(views) { return { restrict: 'E', templateUrl: views.menu }; } exports = module.exports = menuDirective;
7. 編輯 /http/views/index.js 文件
'use strict'; var angular = require('angular'); function views() { return { demo: require('./demo/demo.view.html'), menu: require('./menu/menu.view.html') }; } exports = module.exports = views(); var viewsModule = angular.module('wabgApp.views', []) .constant(views.name, exports); exports.default = viewsModule.name;
8. 建立 /http/views/menu/menu.view.html 文件
<nav class="navbar navbar-light bg-light"> <a class="navbar-brand" href="#"> <span width="30" height="30" class="d-inline-block align-top fa fa-camera-retro fa-5x" /> Bootstrap </a> </nav>
9. 修改 /http/templates/index.tmpl.html 文件
<!DOCTYPE html> <html ng-app="wabgApp"> <head> <meta charset="UTF-8"> <title>Webpack App</title> <style> .ng-cloak { display: none; } </style> </head> <body> <div id="root" ng-controller="homeController"> <menu-directive></menu-directive> The Author is: <lable ng-ng-cloak class="ng-cloak" style="color:red">{{author}}</lable><br/> <input type="button" ng-click="getAuthor(name)" value="getAuthor" /> <br/> <br/> <demo-directive></demo-directive> </div> </body> </html>
10 建立 /http/scss/index.scss 文件
@import "~bootstrap/scss/bootstrap";
11. 運行 npm run server 就能看到效果了
1. 安裝依賴項
npm install mapbox-gl angular-mapboxgl-directive --save-dev
2. 下面演示的是AngularJS中使用MapBox-gl-js,爲了顯示效果,加入了一個很簡單的About頁面,用於演示地圖和普通VIew的切換。以下圖所示:
項目目錄結構以下
wabg ├── dist │ ├── fonts │ │ ├── fontawesome-webfont.eot │ │ ├── fontawesome-webfont.svg │ │ ├── fontawesome-webfont.ttf │ │ ├── fontawesome-webfont.woff │ │ └── fontawesome-webfont.woff2 │ ├── index.html │ └── js │ ├── angular.bundle.js │ ├── angular.bundle.js.map │ ├── app.bundle.js │ ├── app.bundle.js.map │ ├── vendor.bundle.js │ └── vendor.bundle.js.map ├── http │ ├── app.js │ ├── controllers │ │ ├── about │ │ │ └── about.controller.js │ │ ├── home │ │ │ └── home.controller.js │ │ ├── index.js │ │ └── map │ │ └── map.controller.js │ ├── css │ │ └── index.css │ ├── directives │ │ ├── demo │ │ │ └── demo.directive.js │ │ ├── index.js │ │ └── menu │ │ └── menu.directive.js │ ├── scss │ │ └── index.scss │ ├── services │ │ ├── index.js │ │ └── map-styles │ │ └── default-style.js │ ├── templates │ │ └── index.tmpl.html │ └── views │ ├── about │ │ └── about.view.html │ ├── demo │ │ └── demo.view.html │ ├── index.js │ ├── map │ │ └── map.view.html │ └── menu │ └── menu.view.html ├── package.json └── webpack.config.js
3. 修改 webpack.config.js 文件
'use strict'; var path = require('path'); var webpack = require('webpack'); var HtmlWebpackPlugin = require('html-webpack-plugin'); module.exports = { devtool: 'evel-source-map', context: path.resolve(__dirname + '/http'), entry: { app: './app.js', vendor: ['jquery', 'popper.js', 'bootstrap', 'bootstrap/js/dist/util.js', 'bootstrap/js/dist/dropdown.js', 'mapbox-gl', 'angular-mapboxgl-directive'], angular: ['angular', 'angular-ui-router'] }, output: { path: path.resolve(__dirname + '/dist'), filename: 'js/[name].bundle.js', publicPath: '', }, devServer: { historyApiFallback: true, inline: true }, module: { rules: [ { test: /\.view.html$/, use: [ { loader: 'ngtemplate-loader' }, { loader: 'html-loader' } ], }, { test: /\.css$/, use: [{ loader: 'style-loader' }, { loader: 'css-loader' }] }, { test: /\.(scss)$/, use: [{ loader: 'style-loader', }, { loader: 'css-loader', }, { loader: 'postcss-loader', options: { plugins: function () { return [ require('precss'), require('autoprefixer') ]; } } }, { loader: 'sass-loader' }] }, { test: /\.(jpg|png|gif)(\?.*$|$)/, use: [ { loader: 'url-loader', options: { limit: 10000, name: './images/[name].[ext]' } } ] }, { test: /\.(woff|woff2|eot|ttf|svg)(\?.*$|$)/, use: [{ loader: 'file-loader', options: { name: './fonts/[name].[ext]' } }] } ] }, plugins: [ new HtmlWebpackPlugin({ template: __dirname + '/http/templates/index.tmpl.html', chunksSortMode: function (chunk1, chunk2) { var order = ['angular', 'vendor', 'app']; var order1 = order.indexOf(chunk1.names[0]); var order2 = order.indexOf(chunk2.names[0]); return order1 - order2; } }), new webpack.optimize.CommonsChunkPlugin({ names: ['vendor', 'angular'], filename: 'js/[name].bundle.js' }), new webpack.ProvidePlugin({ $: 'jquery', jQuery: 'jquery', 'window.jQuery': 'jquery', Popper: ['popper.js', 'default'], Util: 'exports-loader?Util!bootstrap/js/dist/util', Dropdown: 'exports-loader?Dropdown!bootstrap/js/dist/dropdown', mapboxgl: 'mapbox-gl' }) ] };
4. 修改 /http/app.js 文件
'use strict'; var angular = require('angular'); var uirouter = require('angular-ui-router'); var css = require('./css/index.css'); var scss = require('./scss/index.scss'); var services = require('./services'); var directives = require('./directives'); var controllers = require('./controllers'); var views = require('./views'); var mapboxglDirective = 'mapboxgl-directive'; angular.module('wabgApp', [uirouter.default, services.default, views.default, directives.default, controllers.default, mapboxglDirective]) .config(function ($stateProvider) { $stateProvider .state('home', { }) .state('about', { url: '/about', controller: controllers.aboutController, templateUrl: views.about }) .state('map', { url: '/map', controller: controllers.mapController, templateUrl: views.map }); }) .run([function () { mapboxgl.accessToken = true; }]);
5. 修改 /http/services/index.js 文件
'use strict'; var angular = require('angular'); var defaultMapStyle = require('./map-styles/default-style'); var servicesModule = angular.module('wabgApp.services', []) .constant(defaultMapStyle.name, defaultMapStyle()); exports = module.exports = { default: servicesModule.name };
6. 建立 /http/services/map-styles/default-style.js 文件
'use strict'; function defaultMapStyle() { return { 'version': 8, 'sources': { 'raster-tiles': { type: 'raster', 'tiles': ['http://mt3.google.cn/vt/lyrs=s&hl=zh-CN&gl=cn&x={x}&y={y}&z={z}'], 'tileSize': 256 } }, 'layers': [{ 'id': 'simple-tiles', 'type': 'raster', 'source': 'raster-tiles', 'minzoom': 0, 'maxzoom': 22 }] }; } exports = module.exports = defaultMapStyle;
7. 修改 /http/controllers/index.js 文件
'use strict'; var angular = require('angular'); var homeController = require('./home/home.controller'); var mapController = require('./map/map.controller'); var aboutController = require('./about/about.controller'); var controllersModule = angular.module('wabgApp.controllers', []) .controller(homeController.name, ['$scope', homeController]) .controller(mapController.name, ['$scope', 'defaultMapStyle', mapController]) .controller(aboutController.name, ['$scope', aboutController]); exports = module.exports = { default: controllersModule.name, homeController: homeController.name, mapController: mapController.name, aboutController: aboutController.name };
8. 建立 /http/controllers/map/map.controller.js 文件
'use strict'; function mapController($scope, defaultMapStyle) { $scope.glStyle = defaultMapStyle; } exports = module.exports = mapController;
9. 建立 /http/controllers/about/about.controller.js 文件
'use strict'; function aboutController($scope){ $scope.text = 'webpack 3 + angularjs 1.* + bootstrap 4 + mapbox-gl demo'; } exports = module.exports = aboutController;
10. 修改 /http/views/index.js 文件
'use strict'; var angular = require('angular'); function views() { return { demo: require('./demo/demo.view.html'), menu: require('./menu/menu.view.html'), map: require('./map/map.view.html'), about: require('./about/about.view.html') }; } exports = module.exports = views(); var viewsModule = angular.module('wabgApp.views', []) .constant(views.name, exports); exports.default = viewsModule.name;
11. 修改 /http/views/menu/menu.view.html 文件
<nav class="navbar navbar-light bg-light justify-content-between"> <a class="navbar-brand" href="" ui-sref="map">Map</a> <a class="navbar-brand" href="" ui-sref="about">About</a> <form class="form-inline"> <input class="form-control mr-sm-2" type="text" placeholder="Search" aria-label="Search"> <button class="btn btn-outline-success my-2 my-sm-0" type="submit">Search</button> </form> </nav>
12. 建立 /http/views/map/map.view.html 文件
<div> <div mapboxgl gl-controls="glControls" gl-style="glStyle"></div> </div>
13. 建立 /http/views/about/about.view.html 文件
<br /> <div style="font-weight:bold;font-size:18px">About: {{text}}</div>
14. 修改 /http/templates/index.tmpl.html 文件
<!DOCTYPE html> <html ng-app="wabgApp"> <head> <meta charset="UTF-8"> <title>Webpack App</title> <style> .ng-cloak { display: none; } </style> </head> <body> <div id="root" ng-controller="homeController"> <menu-directive></menu-directive> The Author is: <lable ng-ng-cloak class="ng-cloak" style="color:red">{{author}}</lable><br/> <input type="button" ng-click="getAuthor(name)" value="getAuthor" /> <br/> <br/> <demo-directive></demo-directive> <ui-view></ui-view> </div> </body> </html>
15. 建立 /http/css/index.css 文件
@import '~mapbox-gl/dist/mapbox-gl.css';
@import '~angular-mapboxgl-directive/dist/angular-mapboxgl-directive.css';
16. 運行 npm run server 查看效果