├── .gitignore
├── dist
├── .gitignore
└── ng-staticize.js
├── webpack.config.js
├── HOW.MD
├── scripts
└── pack-demo.js
├── .eslintrc
├── package.json
├── LICENSE
├── src
├── index.js
├── parser.js
└── builder.js
├── README.MD
└── examples
└── index.html
/.gitignore:
--------------------------------------------------------------------------------
1 | .idea
2 | node_modules
--------------------------------------------------------------------------------
/dist/.gitignore:
--------------------------------------------------------------------------------
1 | *
2 | !.gitignore
3 | !ng-staticize.js
--------------------------------------------------------------------------------
/webpack.config.js:
--------------------------------------------------------------------------------
1 | var plugins = [];
2 |
3 | if (process.env.COMPRESS) {
4 | plugins.push(
5 | new webpack.optimize.UglifyJsPlugin({})
6 | );
7 | }
8 |
9 | module.exports = {
10 | entry: './src/index.js',
11 | output: {
12 | path: './dist',
13 | filename: 'ng-staticize.js'
14 | },
15 | module: {
16 | preLoaders: [
17 | { test: /\.js$/, exclude: /node_modules|bower_components/, loader: 'eslint-loader' }
18 | ]
19 | },
20 | plugins: plugins
21 | };
--------------------------------------------------------------------------------
/HOW.MD:
--------------------------------------------------------------------------------
1 | ## 起因
2 | Angular的模板在数据量较大(比如渲染一个2000条数据表格)可能会遇到渲染速度比较慢的问题,[bindonce](https://github.com/Pasvaz/bindonce)提供了一个解决方案,用来解决页面中因为watcher数量较多造成的性能问题。
3 |
4 | 在IE8下,DOM操作是一个非常昂贵的操作,你可以阅读这个stack overflow上这个[问答](http://stackoverflow.com/questions/9639703/ie8-javascript-very-slow-to-load-large-list-of-options-in-select-element)来了解其中的细节。
5 |
6 | 简单来讲,在IE8下创建6000个option元素,需要花费16s。这个问题的解决办法就是使用innerHTML来创建option,不使用createElement来创建option。
7 |
8 | ## 解决办法
9 |
10 | ng-staticize做的事情非常简单,自己解析了Angular的模板,并通过解析后的模板构建出要输出的HTML。
--------------------------------------------------------------------------------
/scripts/pack-demo.js:
--------------------------------------------------------------------------------
1 | var fs = require('fs');
2 | var path = require('path');
3 | var useMin = require('usemin-lib');
4 | const EXAMPLES_PATH = './examples';
5 | const DIST_PATH = './dist/';
6 |
7 | var entries = ['index.html'];
8 |
9 | entries.map(function(fileName) {
10 | var entry = path.join(EXAMPLES_PATH, fileName);
11 | var content = fs.readFileSync(entry).toString();
12 | var blocks = useMin.getBlocks(entry, content, true);
13 | useMin.processBlocks(blocks, DIST_PATH);
14 | fs.writeFile(DIST_PATH + fileName, useMin.getHTML(content, blocks, false));
15 | });
16 |
--------------------------------------------------------------------------------
/.eslintrc:
--------------------------------------------------------------------------------
1 | {
2 | "rules": {
3 | "indent": [
4 | 2,
5 | 2,
6 | {
7 | "SwitchCase": 1
8 | }
9 | ],
10 | "quotes": [
11 | 2,
12 | "single"
13 | ],
14 | "linebreak-style": [
15 | 2,
16 | "unix"
17 | ],
18 | "semi": [
19 | 2,
20 | "always"
21 | ]
22 | },
23 | "env": {
24 | "es6": true,
25 | "browser": true,
26 | "commonjs": true
27 | },
28 | "extends": "eslint:recommended",
29 | "ecmaFeatures": {
30 | "jsx": true
31 | },
32 | "globals": {
33 | "angular": true,
34 | "process": true,
35 | "webpack": true
36 | }
37 | }
38 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "ng-staticize",
3 | "version": "0.1.0",
4 | "description": "Staticize your angular template. Zero watcher and fast as template engine.",
5 | "main": "src/index.js",
6 | "scripts": {
7 | "dev": "node_modules/.bin/webpack --watch --devtool inline-source-map",
8 | "dist": "COMPRESS=1 node_modules/.bin/webpack"
9 | },
10 | "repository": {
11 | "type": "git",
12 | "url": "git+https://github.com/ElemeFE/ng-staticize.git"
13 | },
14 | "dependencies": {
15 | "domify": "^1.4.0"
16 | },
17 | "devDependencies": {
18 | "angular": "1.2.28",
19 | "bootstrap": "^3.3.5",
20 | "eslint": "^1.5.1",
21 | "eslint-loader": "^1.0.0",
22 | "jquery": "1.11.3",
23 | "mocha": "~2.2.5",
24 | "should": "~7.0.2",
25 | "sinon": "~1.10.2",
26 | "usemin-lib": "0.0.5",
27 | "webpack": "^1.9.10"
28 | },
29 | "author": "long.zhang",
30 | "license": "MIT"
31 | }
32 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | The MIT License (MIT)
2 |
3 | Copyright (c) 2015 饿了么前端
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
23 |
--------------------------------------------------------------------------------
/src/index.js:
--------------------------------------------------------------------------------
1 | var parser = require('./parser');
2 | var builder = require('./builder');
3 |
4 | angular.module('ngStaticize', [])
5 | .factory('templateParser', parser)
6 | .factory('templateBuilder', builder)
7 | .directive('ngStaticize', ['templateParser', 'templateBuilder', '$parse', function(templateParser, templateBuilder) {
8 | return {
9 | restrict: 'EA',
10 | replace: true,
11 | terminal: true,
12 | priority: 1001,
13 | compile: function(element) {
14 | var html = element[0].innerHTML;
15 | var template = templateParser.parse(html);
16 | return {
17 | post: function(scope, element, attrs) {
18 | var reRender = function () {
19 | var result = templateBuilder.build(template, scope);
20 | element[0].innerHTML = result;
21 | };
22 |
23 | reRender();
24 |
25 | var watchExpr = attrs.ngStaticize;
26 |
27 | if (watchExpr) {
28 | scope.$watch(watchExpr, function() {
29 | reRender();
30 | }, true);
31 | }
32 | }
33 | };
34 | }
35 | };
36 | }]);
37 |
38 | module.exports = {};
--------------------------------------------------------------------------------
/README.MD:
--------------------------------------------------------------------------------
1 | # Introduce
2 |
3 | ng-staticize 是一个把 Angular 模板静态化的 directive,使用之后 watcher 数量会显著降低,渲染速度(在数据量较大的情况下)在 IE8 上会有10-20x的性能提升。
4 |
5 | 你可以花几分钟查看 [这个例子](http://elemefe.github.io/ng-staticize/) 来体验一下 ng-staticize 带来的性能提升,在 IE8 或者 Firefox 下会看到更明显的性能提升。
6 |
7 | 从降低 Angular watcher 的角度来看,ng-staticize 是一个与 [bindonce](https://github.com/Pasvaz/bindonce) 类似的项目,区别在于 ng-staticize 无需改变模板中 directive 的定义。
8 |
9 | 从设计之初,ng-staticize 就存在以下缺陷,请知悉:
10 |
11 | 1. 应用 ng-staticize 的区域中不能使用 directive 兼容列表以外的 directive。
12 | 2. 渲染出的 DOM 不再动态,即在 Controller 中的数据变更后,应用了 ng-staticize 的区域不会重新渲染。不过你可以为 ng-staticize 指定一个表达式,在表达式变更后这个区域会重新渲染。
13 |
14 | 以下是 ng-staticize 适用场景:
15 |
16 | 1. 页面动态渲染比较少,类似于静态页面,只是使用了 Angular 的模板来描述数据绑定。
17 | 2. 页面中需要渲染的数据较多,需要做性能优化。
18 |
19 | 如果你对 ng-staticize 的实现细节感兴趣,你可以阅读 [这篇文档](HOW.MD)。
20 |
21 | # Usage
22 |
23 | ## Install
24 |
25 | 如果你使用 Browerify 或者 Webpack,可以使用 npm 来安装 ng-staticize:
26 |
27 | ```Bash
28 | npm install ng-staticize —save
29 | ```
30 |
31 | 如果没有使用 npm,则可以下载项目后,在项目中引用 dist 下的 ng-staticize.js。
32 |
33 | ## Include ngStaticize
34 |
35 | ```JavaScript
36 | angular.module('demo', [ 'ngStaticize' ]);
37 | ```
38 |
39 | ## Use
40 |
41 | 如果只是想把页面中的某一个页面中的某个区域进行静态化操作,只需要为这个区域的元素添加一个属性:ng-staticize。
42 |
43 | ```HTML
44 |
...
45 | ```
46 |
47 | 如果需要在某些数据变更之后重新渲染这块区域,为 ng-staticize 属性设置一个表达式,在该表达式变更之后这个区域会重新渲染。比如在 scope 中的 todos 属性发生了变化之后重新渲染,则这么定义:
48 |
49 | ```HTML
50 | ...
51 | ```
52 |
53 | > 如果需要兼容低版本浏 IE 览器(IE8、IE9),请不要在 table、tbody、tfoot、thead、title、tr 等元素上使用 ng-staticize,原因见 [此文章](http://w3help.org/zh-cn/causes/BX9046)。
54 |
55 | # Compatible Directive
56 | ng-staticize 只兼容了一些常见的 directive,列表如下:
57 |
58 | - ng-if
59 | - ng-repeat
60 | - ng-style
61 | - ng-class
62 | - ng-show
63 | - ng-hide
64 | - ng-html
65 | - ng-bind
66 | - ng-text
67 | - ng-src
68 | - ng-href
69 | - ng-alt
70 | - ng-title
71 | - ng-id
72 | - ng-disabled
73 | - ng-value
74 |
75 | # Fork
76 | 如果你想修改 ng-staticize 的代码,在项目文件夹下执行这两个命令:
77 |
78 | ```Bash
79 | npm install
80 | npm run dev
81 | ```
82 |
83 | # License
84 | MIT
85 |
--------------------------------------------------------------------------------
/src/parser.js:
--------------------------------------------------------------------------------
1 | var templateParser = ['$parse', '$interpolate', function($parse, $interpolate) {
2 | var needInterpolate = { 'ng-src': true, 'ng-href': true };
3 | var TEXT_NODE = 3;
4 | var ELEMENT_NODE = 1;
5 |
6 | function walk(el) {
7 | if (el.nodeType !== ELEMENT_NODE) return;
8 |
9 | var children = [];
10 | var node = { tagName: el.tagName.toLowerCase() };
11 | var attributes = el.attributes;
12 | var attrs, dirs, i, j;
13 |
14 | for (i = 0, j = attributes.length; i < j; i++) {
15 | var attribute = attributes[i];
16 | var name = attribute.name;
17 | var value = attribute.nodeValue;
18 | if (name && name.substr(0, 3) === 'ng-') {
19 | if (!dirs) {
20 | dirs = {};
21 | }
22 | if (name.length > 8 && name.substr(0, 8) === 'ng-attr-') {
23 | if (!attrs) {
24 | attrs = {};
25 | }
26 | attrs[name.substr(9)] = $parse(value);
27 | } else if (name === 'ng-repeat') {
28 | var matches = /(\w+)\s+in\s+(.*?)$/.exec(value);
29 | if (matches) {
30 | dirs[name] = {
31 | itemName: matches[1],
32 | getArray: $parse(matches[2])
33 | };
34 | }
35 | } else {
36 | dirs[name] = needInterpolate[name] ? $interpolate(value) : $parse(value);
37 | }
38 | } else {
39 | if (!attrs) {
40 | attrs = {};
41 | }
42 | if (/\s*({{\s*(.+?)\s*}})\s*/gi.test(value)) {
43 | attrs[name] = $interpolate(value);
44 | } else {
45 | attrs[name] = value;
46 | }
47 | }
48 | }
49 |
50 | var childNodes = el.childNodes;
51 | for (i = 0, j = childNodes.length; i < j; i++) {
52 | var child = childNodes[i];
53 | if (child.nodeType === TEXT_NODE) {
54 | var text = child.nodeValue;
55 | if (text) {
56 | if (/\s*({{\s*(.+?)\s*}})\s*/gi.test(text)) {
57 | children.push($interpolate(text));
58 | } else {
59 | text = text.replace(/(\r\n|\n|\r|\s)/gm, '');
60 | if (text.length) {
61 | children.push(text);
62 | }
63 | }
64 | }
65 | continue;
66 | }
67 | var parseResult = walk(child);
68 | if (parseResult) {
69 | children.push(parseResult);
70 | }
71 | }
72 |
73 | if (children.length > 0) {
74 | node.children = children;
75 | }
76 |
77 | if (attrs) {
78 | node.attrs = attrs;
79 | }
80 |
81 | if (dirs) {
82 | node.dirs = dirs;
83 | }
84 |
85 | return node;
86 | }
87 |
88 | return {
89 | parse: function(template) {
90 | return walk(require('domify')(template));
91 | }
92 | };
93 | }];
94 |
95 | module.exports = templateParser;
--------------------------------------------------------------------------------
/src/builder.js:
--------------------------------------------------------------------------------
1 | var trim = function (string) {
2 | return string.replace(/^[\s\uFEFF]+|[\s\uFEFF]+$/g, '');
3 | };
4 |
5 | var templateBuilder = [function() {
6 | function shallowClone(object) {
7 | var result = {};
8 | for (var prop in object) {
9 | if (object.hasOwnProperty(prop)) result[prop] = object[prop];
10 | }
11 | return result;
12 | }
13 |
14 | function newContext(context) {
15 | var empty = function() {};
16 | empty.prototype = context;
17 |
18 | return new empty();
19 | }
20 |
21 | function getStyle(object) {
22 | var cssText = '';
23 | for (var prop in object) {
24 | if (object.hasOwnProperty(prop)) {
25 | cssText += prop + ':' + object[prop] + ';';
26 | }
27 | }
28 | return cssText;
29 | }
30 |
31 | function getClasses(object) {
32 | var classes = [];
33 | for (var prop in object) {
34 | if (object.hasOwnProperty(prop) && !!object[prop]) {
35 | classes.push(prop);
36 | }
37 | }
38 | return classes.join(' ');
39 | }
40 |
41 | function cloneRepeatNode(node) {
42 | var cloneNode = shallowClone(node);
43 | cloneNode.dirs = shallowClone(node.dirs);
44 | cloneNode.dirs['ng-repeat'] = null;
45 | delete cloneNode.dirs['ng-repeat'];
46 |
47 | return cloneNode;
48 | }
49 |
50 | var ATTR_DIR_MAP = {
51 | 'ng-src': true,
52 | 'ng-href': true,
53 | 'ng-alt': true,
54 | 'ng-title': true,
55 | 'ng-id': true,
56 | 'ng-disabled': true,
57 | 'ng-value': true
58 | };
59 |
60 | var HTML_DIR_MAP = {
61 | 'ng-html': true,
62 | 'ng-bind': true,
63 | 'ng-text': true
64 | };
65 |
66 | var emptyArray = [];
67 |
68 | function toHTML(node, context) {
69 | var html, i, j;
70 | if (node instanceof Array) {
71 | html = '';
72 |
73 | for (i = 0, j = node.length; i < j; i++) {
74 | html += toHTML(node[i], context);
75 | }
76 |
77 | return html;
78 | }
79 |
80 | if (typeof node === 'string') return node;
81 | if (typeof node === 'function') return node(context);
82 |
83 | var tag = node.tagName;
84 | var children = node.children;
85 | var attrs = node.attrs;
86 | var dirs = node.dirs;
87 | var content = node.textContent;
88 | var classes = '';
89 | var cssText = '';
90 |
91 | if (dirs && dirs['ng-repeat']) {
92 | var cloneNode = cloneRepeatNode(node);
93 | var repeatDir = dirs['ng-repeat'];
94 | var array = repeatDir.getArray(context) || emptyArray;
95 | var itemName = repeatDir.itemName;
96 |
97 | html = '';
98 |
99 | for (i = 0, j = array.length; i < j; i++) {
100 | var subContext = newContext(context);
101 | subContext.$index = i;
102 | subContext.$first = i === 0;
103 | subContext.$last = i === j - 1;
104 | subContext.$middle = !(subContext.$first || subContext.$last);
105 | subContext[itemName] = array[i];
106 |
107 | html += toHTML(cloneNode, subContext);
108 | }
109 |
110 | return html;
111 | }
112 |
113 | html = '<' + tag;
114 |
115 | for (var dir in dirs) {
116 | if (dirs.hasOwnProperty(dir)) {
117 | var fn = dirs[dir];
118 | var value;
119 | if (typeof fn === 'function') {
120 | value = fn(context);
121 | }
122 |
123 | if (dir === 'ng-if') {
124 | if (!value) {
125 | return '';
126 | }
127 | } else if (ATTR_DIR_MAP[dir]) {
128 | html += ' ' + dir.substr(3) + '="' + value + '"';
129 | } else if (HTML_DIR_MAP[dir]) {
130 | content = value;
131 | } else if (dir === 'ng-style') {
132 | cssText = getStyle(value) + cssText;
133 | } else if (dir === 'ng-class') {
134 | classes += getClasses(value);
135 | } else if (dir === 'ng-show' || dir === 'ng-hide') {
136 | if (value !== null && value !== undefined) {
137 | classes += ' ' + dir;
138 | }
139 | }
140 | }
141 | }
142 |
143 | for (var attr in attrs) {
144 | if (attrs.hasOwnProperty(attr)) {
145 | if (attr === 'style') continue;
146 |
147 | var attrValue = attrs[attr];
148 | if (typeof attrValue === 'function') {
149 | attrValue = attrValue(context);
150 | }
151 | if (attr === 'class') {
152 | classes += ' ' + attrValue;
153 | } else {
154 | html += ' ' + attr + '="' + attrValue + '"';
155 | }
156 | }
157 | }
158 |
159 | if (classes) {
160 | html += ' class="' + trim(classes) + '"';
161 | }
162 |
163 | if (cssText) {
164 | html += ' style="' + cssText + '"';
165 | }
166 |
167 | html += '>';
168 |
169 | if (content) {
170 | html += content;
171 | }
172 |
173 | if (children) {
174 | for (i = 0, j = children.length; i < j; i++) {
175 | var child = children[i];
176 | html += toHTML(child, context);
177 | }
178 | }
179 |
180 | html += '' + tag + '>';
181 |
182 | return html;
183 | }
184 |
185 | return {
186 | build: toHTML
187 | };
188 | }];
189 |
190 | module.exports = templateBuilder;
--------------------------------------------------------------------------------
/examples/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | Ng Staticize Demo
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
72 |
73 |
74 |
75 |
76 |
77 |
172 |
173 |
174 |
175 |
ng-staticize demo
176 |
177 |
This demo demonstrates the render speed of ng-staticize. Switch tabs to view the render speed of using ng-staticize or not.
178 |
If you are using Chrome, you can use Angular watchers to view the watcher's change.
179 |
180 |
181 |
185 |
186 |
187 |
188 |
189 |
190 |
191 |
192 | Load 500
193 | Load 1000
194 | Load 2000
195 | Load 5000
196 | Load 10000
197 |
198 |
199 |
200 | This tab is using ng-staticize, you can click buttons above to get the time of rendering.
201 |
...
202 |
203 |
204 |
205 |
206 |
207 | #
208 | First Name
209 | Last Name
210 | Age
211 | Gender
212 | Profile
213 |
214 |
215 |
216 |
217 |
218 |
219 |
220 |
221 |
222 |
223 |
224 |
225 |
226 |
227 |
228 |
229 |
230 |
231 |
232 |
233 |
234 | Load 500
235 | Load 1000
236 | Load 2000
237 | Load 5000
238 | Load 10000
239 |
240 |
241 |
242 | This tab is
not using ng-staticze, you can click buttons above to get the time of rendering.
243 |
...
244 |
245 |
246 |
247 |
248 |
249 | #
250 | First Name
251 | Last Name
252 | Age
253 | Gender
254 | Profile
255 |
256 |
257 |
258 |
259 |
260 |
261 |
262 |
263 |
264 |
265 |
266 |
267 |
268 |
269 |
270 |
271 |
272 |
273 |
274 |
275 |
276 |
277 |
--------------------------------------------------------------------------------
/dist/ng-staticize.js:
--------------------------------------------------------------------------------
1 | /******/ (function(modules) { // webpackBootstrap
2 | /******/ // The module cache
3 | /******/ var installedModules = {};
4 | /******/
5 | /******/ // The require function
6 | /******/ function __webpack_require__(moduleId) {
7 | /******/
8 | /******/ // Check if module is in cache
9 | /******/ if(installedModules[moduleId])
10 | /******/ return installedModules[moduleId].exports;
11 | /******/
12 | /******/ // Create a new module (and put it into the cache)
13 | /******/ var module = installedModules[moduleId] = {
14 | /******/ exports: {},
15 | /******/ id: moduleId,
16 | /******/ loaded: false
17 | /******/ };
18 | /******/
19 | /******/ // Execute the module function
20 | /******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
21 | /******/
22 | /******/ // Flag the module as loaded
23 | /******/ module.loaded = true;
24 | /******/
25 | /******/ // Return the exports of the module
26 | /******/ return module.exports;
27 | /******/ }
28 | /******/
29 | /******/
30 | /******/ // expose the modules object (__webpack_modules__)
31 | /******/ __webpack_require__.m = modules;
32 | /******/
33 | /******/ // expose the module cache
34 | /******/ __webpack_require__.c = installedModules;
35 | /******/
36 | /******/ // __webpack_public_path__
37 | /******/ __webpack_require__.p = "";
38 | /******/
39 | /******/ // Load entry module and return exports
40 | /******/ return __webpack_require__(0);
41 | /******/ })
42 | /************************************************************************/
43 | /******/ ([
44 | /* 0 */
45 | /***/ function(module, exports, __webpack_require__) {
46 |
47 | var parser = __webpack_require__(1);
48 | var builder = __webpack_require__(3);
49 |
50 | angular.module('ngStaticize', [])
51 | .factory('templateParser', parser)
52 | .factory('templateBuilder', builder)
53 | .directive('ngStaticize', ['templateParser', 'templateBuilder', '$parse', function(templateParser, templateBuilder) {
54 | return {
55 | restrict: 'EA',
56 | replace: true,
57 | terminal: true,
58 | priority: 1001,
59 | compile: function(element) {
60 | var html = element[0].innerHTML;
61 | var template = templateParser.parse(html);
62 | return {
63 | post: function(scope, element, attrs) {
64 | var reRender = function () {
65 | var result = templateBuilder.build(template, scope);
66 | element[0].innerHTML = result;
67 | };
68 |
69 | reRender();
70 |
71 | var watchExpr = attrs.ngStaticize;
72 |
73 | if (watchExpr) {
74 | scope.$watch(watchExpr, function() {
75 | reRender();
76 | }, true);
77 | }
78 | }
79 | };
80 | }
81 | };
82 | }]);
83 |
84 | module.exports = {};
85 |
86 | /***/ },
87 | /* 1 */
88 | /***/ function(module, exports, __webpack_require__) {
89 |
90 | var templateParser = ['$parse', '$interpolate', function($parse, $interpolate) {
91 | var needInterpolate = { 'ng-src': true, 'ng-href': true };
92 | var TEXT_NODE = 3;
93 | var ELEMENT_NODE = 1;
94 |
95 | function walk(el) {
96 | if (el.nodeType !== ELEMENT_NODE) return;
97 |
98 | var children = [];
99 | var node = { tagName: el.tagName.toLowerCase() };
100 | var attributes = el.attributes;
101 | var attrs, dirs, i, j;
102 |
103 | for (i = 0, j = attributes.length; i < j; i++) {
104 | var attribute = attributes[i];
105 | var name = attribute.name;
106 | var value = attribute.nodeValue;
107 | if (name && name.substr(0, 3) === 'ng-') {
108 | if (!dirs) {
109 | dirs = {};
110 | }
111 | if (name.length > 8 && name.substr(0, 8) === 'ng-attr-') {
112 | if (!attrs) {
113 | attrs = {};
114 | }
115 | attrs[name.substr(9)] = $parse(value);
116 | } else if (name === 'ng-repeat') {
117 | var matches = /(\w+)\s+in\s+(.*?)$/.exec(value);
118 | if (matches) {
119 | dirs[name] = {
120 | itemName: matches[1],
121 | getArray: $parse(matches[2])
122 | };
123 | }
124 | } else {
125 | dirs[name] = needInterpolate[name] ? $interpolate(value) : $parse(value);
126 | }
127 | } else {
128 | if (!attrs) {
129 | attrs = {};
130 | }
131 | if (/\s*({{\s*(.+?)\s*}})\s*/gi.test(value)) {
132 | attrs[name] = $interpolate(value);
133 | } else {
134 | attrs[name] = value;
135 | }
136 | }
137 | }
138 |
139 | var childNodes = el.childNodes;
140 | for (i = 0, j = childNodes.length; i < j; i++) {
141 | var child = childNodes[i];
142 | if (child.nodeType === TEXT_NODE) {
143 | var text = child.nodeValue;
144 | if (text) {
145 | if (/\s*({{\s*(.+?)\s*}})\s*/gi.test(text)) {
146 | children.push($interpolate(text));
147 | } else {
148 | text = text.replace(/(\r\n|\n|\r|\s)/gm, '');
149 | if (text.length) {
150 | children.push(text);
151 | }
152 | }
153 | }
154 | continue;
155 | }
156 | var parseResult = walk(child);
157 | if (parseResult) {
158 | children.push(parseResult);
159 | }
160 | }
161 |
162 | if (children.length > 0) {
163 | node.children = children;
164 | }
165 |
166 | if (attrs) {
167 | node.attrs = attrs;
168 | }
169 |
170 | if (dirs) {
171 | node.dirs = dirs;
172 | }
173 |
174 | return node;
175 | }
176 |
177 | return {
178 | parse: function(template) {
179 | return walk(__webpack_require__(2)(template));
180 | }
181 | };
182 | }];
183 |
184 | module.exports = templateParser;
185 |
186 | /***/ },
187 | /* 2 */
188 | /***/ function(module, exports) {
189 |
190 |
191 | /**
192 | * Expose `parse`.
193 | */
194 |
195 | module.exports = parse;
196 |
197 | /**
198 | * Tests for browser support.
199 | */
200 |
201 | var innerHTMLBug = false;
202 | var bugTestDiv;
203 | if (typeof document !== 'undefined') {
204 | bugTestDiv = document.createElement('div');
205 | // Setup
206 | bugTestDiv.innerHTML = ' a ';
207 | // Make sure that link elements get serialized correctly by innerHTML
208 | // This requires a wrapper element in IE
209 | innerHTMLBug = !bugTestDiv.getElementsByTagName('link').length;
210 | bugTestDiv = undefined;
211 | }
212 |
213 | /**
214 | * Wrap map from jquery.
215 | */
216 |
217 | var map = {
218 | legend: [1, '', ' '],
219 | tr: [2, ''],
220 | col: [2, ''],
221 | // for script/link/style tags to work in IE6-8, you have to wrap
222 | // in a div with a non-whitespace character in front, ha!
223 | _default: innerHTMLBug ? [1, 'X', '
'] : [0, '', '']
224 | };
225 |
226 | map.td =
227 | map.th = [3, ''];
228 |
229 | map.option =
230 | map.optgroup = [1, '', ' '];
231 |
232 | map.thead =
233 | map.tbody =
234 | map.colgroup =
235 | map.caption =
236 | map.tfoot = [1, ''];
237 |
238 | map.polyline =
239 | map.ellipse =
240 | map.polygon =
241 | map.circle =
242 | map.text =
243 | map.line =
244 | map.path =
245 | map.rect =
246 | map.g = [1, '',' '];
247 |
248 | /**
249 | * Parse `html` and return a DOM Node instance, which could be a TextNode,
250 | * HTML DOM Node of some kind ( for example), or a DocumentFragment
251 | * instance, depending on the contents of the `html` string.
252 | *
253 | * @param {String} html - HTML string to "domify"
254 | * @param {Document} doc - The `document` instance to create the Node for
255 | * @return {DOMNode} the TextNode, DOM Node, or DocumentFragment instance
256 | * @api private
257 | */
258 |
259 | function parse(html, doc) {
260 | if ('string' != typeof html) throw new TypeError('String expected');
261 |
262 | // default to the global `document` object
263 | if (!doc) doc = document;
264 |
265 | // tag name
266 | var m = /<([\w:]+)/.exec(html);
267 | if (!m) return doc.createTextNode(html);
268 |
269 | html = html.replace(/^\s+|\s+$/g, ''); // Remove leading/trailing whitespace
270 |
271 | var tag = m[1];
272 |
273 | // body support
274 | if (tag == 'body') {
275 | var el = doc.createElement('html');
276 | el.innerHTML = html;
277 | return el.removeChild(el.lastChild);
278 | }
279 |
280 | // wrap map
281 | var wrap = map[tag] || map._default;
282 | var depth = wrap[0];
283 | var prefix = wrap[1];
284 | var suffix = wrap[2];
285 | var el = doc.createElement('div');
286 | el.innerHTML = prefix + html + suffix;
287 | while (depth--) el = el.lastChild;
288 |
289 | // one element
290 | if (el.firstChild == el.lastChild) {
291 | return el.removeChild(el.firstChild);
292 | }
293 |
294 | // several elements
295 | var fragment = doc.createDocumentFragment();
296 | while (el.firstChild) {
297 | fragment.appendChild(el.removeChild(el.firstChild));
298 | }
299 |
300 | return fragment;
301 | }
302 |
303 |
304 | /***/ },
305 | /* 3 */
306 | /***/ function(module, exports) {
307 |
308 | var trim = function (string) {
309 | return string.replace(/^[\s\uFEFF]+|[\s\uFEFF]+$/g, '');
310 | };
311 |
312 | var templateBuilder = [function() {
313 | function shallowClone(object) {
314 | var result = {};
315 | for (var prop in object) {
316 | if (object.hasOwnProperty(prop)) result[prop] = object[prop];
317 | }
318 | return result;
319 | }
320 |
321 | function newContext(context) {
322 | var empty = function() {};
323 | empty.prototype = context;
324 |
325 | return new empty();
326 | }
327 |
328 | function getStyle(object) {
329 | var cssText = '';
330 | for (var prop in object) {
331 | if (object.hasOwnProperty(prop)) {
332 | cssText += prop + ':' + object[prop] + ';';
333 | }
334 | }
335 | return cssText;
336 | }
337 |
338 | function getClasses(object) {
339 | var classes = [];
340 | for (var prop in object) {
341 | if (object.hasOwnProperty(prop) && !!object[prop]) {
342 | classes.push(prop);
343 | }
344 | }
345 | return classes.join(' ');
346 | }
347 |
348 | function cloneRepeatNode(node) {
349 | var cloneNode = shallowClone(node);
350 | cloneNode.dirs = shallowClone(node.dirs);
351 | cloneNode.dirs['ng-repeat'] = null;
352 | delete cloneNode.dirs['ng-repeat'];
353 |
354 | return cloneNode;
355 | }
356 |
357 | var ATTR_DIR_MAP = {
358 | 'ng-src': true,
359 | 'ng-href': true,
360 | 'ng-alt': true,
361 | 'ng-title': true,
362 | 'ng-id': true,
363 | 'ng-disabled': true,
364 | 'ng-value': true
365 | };
366 |
367 | var HTML_DIR_MAP = {
368 | 'ng-html': true,
369 | 'ng-bind': true,
370 | 'ng-text': true
371 | };
372 |
373 | var emptyArray = [];
374 |
375 | function toHTML(node, context) {
376 | var html, i, j;
377 | if (node instanceof Array) {
378 | html = '';
379 |
380 | for (i = 0, j = node.length; i < j; i++) {
381 | html += toHTML(node[i], context);
382 | }
383 |
384 | return html;
385 | }
386 |
387 | if (typeof node === 'string') return node;
388 | if (typeof node === 'function') return node(context);
389 |
390 | var tag = node.tagName;
391 | var children = node.children;
392 | var attrs = node.attrs;
393 | var dirs = node.dirs;
394 | var content = node.textContent;
395 | var classes = '';
396 | var cssText = '';
397 |
398 | if (dirs && dirs['ng-repeat']) {
399 | var cloneNode = cloneRepeatNode(node);
400 | var repeatDir = dirs['ng-repeat'];
401 | var array = repeatDir.getArray(context) || emptyArray;
402 | var itemName = repeatDir.itemName;
403 |
404 | html = '';
405 |
406 | for (i = 0, j = array.length; i < j; i++) {
407 | var subContext = newContext(context);
408 | subContext.$index = i;
409 | subContext.$first = i === 0;
410 | subContext.$last = i === j - 1;
411 | subContext.$middle = !(subContext.$first || subContext.$last);
412 | subContext[itemName] = array[i];
413 |
414 | html += toHTML(cloneNode, subContext);
415 | }
416 |
417 | return html;
418 | }
419 |
420 | html = '<' + tag;
421 |
422 | for (var dir in dirs) {
423 | if (dirs.hasOwnProperty(dir)) {
424 | var fn = dirs[dir];
425 | var value;
426 | if (typeof fn === 'function') {
427 | value = fn(context);
428 | }
429 |
430 | if (dir === 'ng-if') {
431 | if (!value) {
432 | return '';
433 | }
434 | } else if (ATTR_DIR_MAP[dir]) {
435 | html += ' ' + dir.substr(3) + '="' + value + '"';
436 | } else if (HTML_DIR_MAP[dir]) {
437 | content = value;
438 | } else if (dir === 'ng-style') {
439 | cssText = getStyle(value) + cssText;
440 | } else if (dir === 'ng-class') {
441 | classes += getClasses(value);
442 | } else if (dir === 'ng-show' || dir === 'ng-hide') {
443 | if (value !== null && value !== undefined) {
444 | classes += ' ' + dir;
445 | }
446 | }
447 | }
448 | }
449 |
450 | for (var attr in attrs) {
451 | if (attrs.hasOwnProperty(attr)) {
452 | if (attr === 'style') continue;
453 |
454 | var attrValue = attrs[attr];
455 | if (typeof attrValue === 'function') {
456 | attrValue = attrValue(context);
457 | }
458 | if (attr === 'class') {
459 | classes += ' ' + attrValue;
460 | } else {
461 | html += ' ' + attr + '="' + attrValue + '"';
462 | }
463 | }
464 | }
465 |
466 | if (classes) {
467 | html += ' class="' + trim(classes) + '"';
468 | }
469 |
470 | if (cssText) {
471 | html += ' style="' + cssText + '"';
472 | }
473 |
474 | html += '>';
475 |
476 | if (content) {
477 | html += content;
478 | }
479 |
480 | if (children) {
481 | for (i = 0, j = children.length; i < j; i++) {
482 | var child = children[i];
483 | html += toHTML(child, context);
484 | }
485 | }
486 |
487 | html += '' + tag + '>';
488 |
489 | return html;
490 | }
491 |
492 | return {
493 | build: toHTML
494 | };
495 | }];
496 |
497 | module.exports = templateBuilder;
498 |
499 | /***/ }
500 | /******/ ]);
501 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["webpack:///webpack/bootstrap f7d53c9967aa1071d670","webpack:///./src/index.js","webpack:///./src/parser.js","webpack:///./~/domify/index.js","webpack:///./src/builder.js"],"names":[],"mappings":";AAAA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,uBAAe;AACf;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;;AAGA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;;;;;;ACtCA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA,gBAAe;AACf;AACA;AACA;AACA;AACA;AACA,IAAG;;AAEH,qB;;;;;;ACrCA;AACA,0BAAyB;AACzB;AACA;;AAEA;AACA;;AAEA;AACA,iBAAgB;AAChB;AACA;;AAEA,uCAAsC,OAAO;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAS;AACT;AACA;AACA,QAAO;AACP;AACA;AACA;AACA,oBAAmB,aAAa;AAChC;AACA,UAAS;AACT;AACA;AACA;AACA;;AAEA;AACA,uCAAsC,OAAO;AAC7C;AACA;AACA;AACA;AACA,sBAAqB,aAAa;AAClC;AACA,YAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,EAAC;;AAED,iC;;;;;;;AC7FA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,YAAW,OAAO;AAClB,YAAW,SAAS;AACpB,aAAY,QAAQ;AACpB;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA,yCAAwC;;AAExC;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;;;;;;AC/GA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,kDAAiD;AACjD;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA,mCAAkC,OAAO;AACzC;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA,oCAAmC,OAAO;AAC1C;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAS;AACT;AACA,UAAS;AACT;AACA,UAAS;AACT;AACA,UAAS;AACT;AACA,UAAS;AACT;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,UAAS;AACT;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA,uCAAsC,OAAO;AAC7C;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,EAAC;;AAED,kC","file":"ng-staticize.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n/** WEBPACK FOOTER **\n ** webpack/bootstrap f7d53c9967aa1071d670\n **/","var parser = require('./parser');\nvar builder = require('./builder');\n\nangular.module('ngStaticize', [])\n  .factory('templateParser', parser)\n  .factory('templateBuilder', builder)\n  .directive('ngStaticize', ['templateParser', 'templateBuilder', '$parse', function(templateParser, templateBuilder) {\n    return {\n      restrict: 'EA',\n      replace: true,\n      terminal: true,\n      priority: 1001,\n      compile: function(element) {\n        var html = element[0].innerHTML;\n        var template = templateParser.parse(html);\n        return {\n          post: function(scope, element, attrs) {\n            var reRender = function () {\n              var result = templateBuilder.build(template, scope);\n              element[0].innerHTML = result;\n            };\n\n            reRender();\n\n            var watchExpr = attrs.ngStaticize;\n\n            if (watchExpr) {\n              scope.$watch(watchExpr, function() {\n                reRender();\n              }, true);\n            }\n          }\n        };\n      }\n    };\n  }]);\n\nmodule.exports = {};\n\n\n/*****************\n ** WEBPACK FOOTER\n ** ./src/index.js\n ** module id = 0\n ** module chunks = 0\n **/","var templateParser = ['$parse', '$interpolate', function($parse, $interpolate) {\n  var needInterpolate = { 'ng-src': true, 'ng-href': true };\n  var TEXT_NODE = 3;\n  var ELEMENT_NODE = 1;\n\n  function walk(el) {\n    if (el.nodeType !== ELEMENT_NODE) return;\n\n    var children = [];\n    var node = { tagName: el.tagName.toLowerCase() };\n    var attributes = el.attributes;\n    var attrs, dirs, i, j;\n\n    for (i = 0, j = attributes.length; i < j; i++) {\n      var attribute = attributes[i];\n      var name = attribute.name;\n      var value = attribute.nodeValue;\n      if (name && name.substr(0, 3) === 'ng-') {\n        if (!dirs) {\n          dirs = {};\n        }\n        if (name.length > 8 && name.substr(0, 8) === 'ng-attr-') {\n          if (!attrs) {\n            attrs = {};\n          }\n          attrs[name.substr(9)] = $parse(value);\n        } else if (name === 'ng-repeat') {\n          var matches = /(\\w+)\\s+in\\s+(.*?)$/.exec(value);\n          if (matches) {\n            dirs[name] = {\n              itemName: matches[1],\n              getArray: $parse(matches[2])\n            };\n          }\n        } else {\n          dirs[name] = needInterpolate[name] ? $interpolate(value) : $parse(value);\n        }\n      } else {\n        if (!attrs) {\n          attrs = {};\n        }\n        if (/\\s*({{\\s*(.+?)\\s*}})\\s*/gi.test(value)) {\n          attrs[name] = $interpolate(value);\n        } else {\n          attrs[name] = value;\n        }\n      }\n    }\n\n    var childNodes = el.childNodes;\n    for (i = 0, j = childNodes.length; i < j; i++) {\n      var child = childNodes[i];\n      if (child.nodeType === TEXT_NODE) {\n        var text = child.nodeValue;\n        if (text) {\n          if (/\\s*({{\\s*(.+?)\\s*}})\\s*/gi.test(text)) {\n            children.push($interpolate(text));\n          } else {\n            text = text.replace(/(\\r\\n|\\n|\\r|\\s)/gm, '');\n            if (text.length) {\n              children.push(text);\n            }\n          }\n        }\n        continue;\n      }\n      var parseResult = walk(child);\n      if (parseResult) {\n        children.push(parseResult);\n      }\n    }\n\n    if (children.length > 0) {\n      node.children = children;\n    }\n\n    if (attrs) {\n      node.attrs = attrs;\n    }\n\n    if (dirs) {\n      node.dirs = dirs;\n    }\n\n    return node;\n  }\n\n  return {\n    parse: function(template) {\n      return walk(require('domify')(template));\n    }\n  };\n}];\n\nmodule.exports = templateParser;\n\n\n/*****************\n ** WEBPACK FOOTER\n ** ./src/parser.js\n ** module id = 1\n ** module chunks = 0\n **/","\n/**\n * Expose `parse`.\n */\n\nmodule.exports = parse;\n\n/**\n * Tests for browser support.\n */\n\nvar innerHTMLBug = false;\nvar bugTestDiv;\nif (typeof document !== 'undefined') {\n  bugTestDiv = document.createElement('div');\n  // Setup\n  bugTestDiv.innerHTML = '  <link/><table></table><a href=\"/a\">a</a><input type=\"checkbox\"/>';\n  // Make sure that link elements get serialized correctly by innerHTML\n  // This requires a wrapper element in IE\n  innerHTMLBug = !bugTestDiv.getElementsByTagName('link').length;\n  bugTestDiv = undefined;\n}\n\n/**\n * Wrap map from jquery.\n */\n\nvar map = {\n  legend: [1, '<fieldset>', '</fieldset>'],\n  tr: [2, '<table><tbody>', '</tbody></table>'],\n  col: [2, '<table><tbody></tbody><colgroup>', '</colgroup></table>'],\n  // for script/link/style tags to work in IE6-8, you have to wrap\n  // in a div with a non-whitespace character in front, ha!\n  _default: innerHTMLBug ? [1, 'X<div>', '</div>'] : [0, '', '']\n};\n\nmap.td =\nmap.th = [3, '<table><tbody><tr>', '</tr></tbody></table>'];\n\nmap.option =\nmap.optgroup = [1, '<select multiple=\"multiple\">', '</select>'];\n\nmap.thead =\nmap.tbody =\nmap.colgroup =\nmap.caption =\nmap.tfoot = [1, '<table>', '</table>'];\n\nmap.polyline =\nmap.ellipse =\nmap.polygon =\nmap.circle =\nmap.text =\nmap.line =\nmap.path =\nmap.rect =\nmap.g = [1, '<svg xmlns=\"http://www.w3.org/2000/svg\" version=\"1.1\">','</svg>'];\n\n/**\n * Parse `html` and return a DOM Node instance, which could be a TextNode,\n * HTML DOM Node of some kind (<div> for example), or a DocumentFragment\n * instance, depending on the contents of the `html` string.\n *\n * @param {String} html - HTML string to \"domify\"\n * @param {Document} doc - The `document` instance to create the Node for\n * @return {DOMNode} the TextNode, DOM Node, or DocumentFragment instance\n * @api private\n */\n\nfunction parse(html, doc) {\n  if ('string' != typeof html) throw new TypeError('String expected');\n\n  // default to the global `document` object\n  if (!doc) doc = document;\n\n  // tag name\n  var m = /<([\\w:]+)/.exec(html);\n  if (!m) return doc.createTextNode(html);\n\n  html = html.replace(/^\\s+|\\s+$/g, ''); // Remove leading/trailing whitespace\n\n  var tag = m[1];\n\n  // body support\n  if (tag == 'body') {\n    var el = doc.createElement('html');\n    el.innerHTML = html;\n    return el.removeChild(el.lastChild);\n  }\n\n  // wrap map\n  var wrap = map[tag] || map._default;\n  var depth = wrap[0];\n  var prefix = wrap[1];\n  var suffix = wrap[2];\n  var el = doc.createElement('div');\n  el.innerHTML = prefix + html + suffix;\n  while (depth--) el = el.lastChild;\n\n  // one element\n  if (el.firstChild == el.lastChild) {\n    return el.removeChild(el.firstChild);\n  }\n\n  // several elements\n  var fragment = doc.createDocumentFragment();\n  while (el.firstChild) {\n    fragment.appendChild(el.removeChild(el.firstChild));\n  }\n\n  return fragment;\n}\n\n\n\n/*****************\n ** WEBPACK FOOTER\n ** ./~/domify/index.js\n ** module id = 2\n ** module chunks = 0\n **/","var trim = function (string) {\n  return string.replace(/^[\\s\\uFEFF]+|[\\s\\uFEFF]+$/g, '');\n};\n\nvar templateBuilder = [function() {\n  function shallowClone(object) {\n    var result = {};\n    for (var prop in object) {\n      if (object.hasOwnProperty(prop)) result[prop] = object[prop];\n    }\n    return result;\n  }\n\n  function newContext(context) {\n    var empty = function() {};\n    empty.prototype = context;\n\n    return new empty();\n  }\n\n  function getStyle(object) {\n    var cssText = '';\n    for (var prop in object) {\n      if (object.hasOwnProperty(prop)) {\n        cssText += prop + ':' + object[prop] + ';';\n      }\n    }\n    return cssText;\n  }\n\n  function getClasses(object) {\n    var classes = [];\n    for (var prop in object) {\n      if (object.hasOwnProperty(prop) && !!object[prop]) {\n        classes.push(prop);\n      }\n    }\n    return classes.join(' ');\n  }\n\n  function cloneRepeatNode(node) {\n    var cloneNode = shallowClone(node);\n    cloneNode.dirs = shallowClone(node.dirs);\n    cloneNode.dirs['ng-repeat'] = null;\n    delete cloneNode.dirs['ng-repeat'];\n\n    return cloneNode;\n  }\n\n  var ATTR_DIR_MAP = {\n    'ng-src': true,\n    'ng-href': true,\n    'ng-alt': true,\n    'ng-title': true,\n    'ng-id': true,\n    'ng-disabled': true,\n    'ng-value': true\n  };\n\n  var HTML_DIR_MAP = {\n    'ng-html': true,\n    'ng-bind': true,\n    'ng-text': true\n  };\n\n  var emptyArray = [];\n\n  function toHTML(node, context) {\n    var html, i, j;\n    if (node instanceof Array) {\n      html = '';\n\n      for (i = 0, j = node.length; i < j; i++) {\n        html += toHTML(node[i], context);\n      }\n\n      return html;\n    }\n\n    if (typeof node === 'string') return node;\n    if (typeof node === 'function') return node(context);\n\n    var tag = node.tagName;\n    var children = node.children;\n    var attrs = node.attrs;\n    var dirs = node.dirs;\n    var content = node.textContent;\n    var classes = '';\n    var cssText = '';\n\n    if (dirs && dirs['ng-repeat']) {\n      var cloneNode = cloneRepeatNode(node);\n      var repeatDir = dirs['ng-repeat'];\n      var array = repeatDir.getArray(context) || emptyArray;\n      var itemName = repeatDir.itemName;\n\n      html = '';\n\n      for (i = 0, j = array.length; i < j; i++) {\n        var subContext = newContext(context);\n        subContext.$index = i;\n        subContext.$first = i === 0;\n        subContext.$last = i === j - 1;\n        subContext.$middle = !(subContext.$first || subContext.$last);\n        subContext[itemName] = array[i];\n\n        html += toHTML(cloneNode, subContext);\n      }\n\n      return html;\n    }\n\n    html = '<' + tag;\n\n    for (var dir in dirs) {\n      if (dirs.hasOwnProperty(dir)) {\n        var fn = dirs[dir];\n        var value;\n        if (typeof fn === 'function') {\n          value = fn(context);\n        }\n\n        if (dir === 'ng-if') {\n          if (!value) {\n            return '';\n          }\n        } else if (ATTR_DIR_MAP[dir]) {\n          html += ' ' + dir.substr(3) + '=\"' + value + '\"';\n        } else if (HTML_DIR_MAP[dir]) {\n          content = value;\n        } else if (dir === 'ng-style') {\n          cssText = getStyle(value) + cssText;\n        } else if (dir === 'ng-class') {\n          classes += getClasses(value);\n        } else if (dir === 'ng-show' || dir === 'ng-hide') {\n          if (value !== null && value !== undefined) {\n            classes += ' ' + dir;\n          }\n        }\n      }\n    }\n\n    for (var attr in attrs) {\n      if (attrs.hasOwnProperty(attr)) {\n        if (attr === 'style') continue;\n\n        var attrValue = attrs[attr];\n        if (typeof attrValue === 'function') {\n          attrValue = attrValue(context);\n        }\n        if (attr === 'class') {\n          classes += ' ' + attrValue;\n        } else {\n          html += ' ' + attr + '=\"' + attrValue + '\"';\n        }\n      }\n    }\n\n    if (classes) {\n      html += ' class=\"' + trim(classes) + '\"';\n    }\n\n    if (cssText) {\n      html += ' style=\"' + cssText + '\"';\n    }\n\n    html += '>';\n\n    if (content) {\n      html += content;\n    }\n\n    if (children) {\n      for (i = 0, j = children.length; i < j; i++) {\n        var child = children[i];\n        html += toHTML(child, context);\n      }\n    }\n\n    html += '</' + tag + '>';\n\n    return html;\n  }\n\n  return {\n    build: toHTML\n  };\n}];\n\nmodule.exports = templateBuilder;\n\n\n/*****************\n ** WEBPACK FOOTER\n ** ./src/builder.js\n ** module id = 3\n ** module chunks = 0\n **/"],"sourceRoot":""}
--------------------------------------------------------------------------------