improve parser performance
authoralexlamsl <alexlamsl@gmail.com>
Wed, 2 Mar 2016 05:58:44 +0000 (13:58 +0800)
committeralexlamsl <alexlamsl@gmail.com>
Wed, 2 Mar 2016 21:51:15 +0000 (05:51 +0800)
dist/htmlminifier.js
dist/htmlminifier.min.js
src/htmlparser.js

index c32471d..53ac04c 100644 (file)
@@ -72,7 +72,7 @@
   // Special Elements (can contain anything)
   var special = makeMap('script,style');
 
-  var reCache = {}, stackedTag, reStackedTag, tagMatch;
+  var reCache = {};
 
   function startTagForHandler( handler ) {
     var customStartTagAttrs;
   }
 
   var HTMLParser = global.HTMLParser = function( html, handler ) {
-    var index, match, stack = [], last = html, prevTag, nextTag;
-    stack.last = function() {
-      var last = this[ this.length - 1 ];
-      return last && last.tag;
-    };
-
+    var stack = [], lastTag;
     var startTag = startTagForHandler(handler);
     var attr = attrForHandler(handler);
-
+    var last, prevTag, nextTag;
     while ( html ) {
+      last = html;
       // Make sure we're not in a script or style element
-      if ( !stack.last() || !special[ stack.last() ] ) {
+      if ( !lastTag || !special[ lastTag ] ) {
+        var textEnd = html.indexOf('<');
+        if (textEnd === 0) {
+          // Comment:
+          if ( /^<!--/.test( html ) ) {
+            var commentEnd = html.indexOf('-->');
+
+            if ( commentEnd >= 0 ) {
+              if ( handler.comment ) {
+                handler.comment( html.substring( 4, commentEnd ) );
+              }
+              html = html.substring( commentEnd + 3 );
+              prevTag = '';
+              continue;
+            }
+          }
 
-        // Comment:
-        if ( /^<!--/.test( html ) ) {
-          index = html.indexOf('-->');
+          // http://en.wikipedia.org/wiki/Conditional_comment#Downlevel-revealed_conditional_comment
+          if ( /^<!\[/.test( html ) ) {
+            var conditionalEnd = html.indexOf(']>');
 
-          if ( index >= 0 ) {
-            if ( handler.comment ) {
-              handler.comment( html.substring( 4, index ) );
+            if (conditionalEnd >= 0) {
+              if ( handler.comment ) {
+                handler.comment( html.substring(2, conditionalEnd + 1 ), true /* non-standard */ );
+              }
+              html = html.substring( conditionalEnd + 2 );
+              prevTag = '';
+              continue;
             }
-            html = html.substring( index + 3 );
-            prevTag = '';
-            continue;
           }
-        }
-
-        // http://en.wikipedia.org/wiki/Conditional_comment#Downlevel-revealed_conditional_comment
-        if ( /^<!\[/.test( html ) ) {
-          index = html.indexOf(']>');
 
-          if (index >= 0) {
-            if ( handler.comment ) {
-              handler.comment( html.substring(2, index + 1 ), true /* non-standard */ );
+          // Doctype:
+          var doctypeMatch = html.match( doctype );
+          if ( doctypeMatch ) {
+            if ( handler.doctype ) {
+              handler.doctype( doctypeMatch[0] );
             }
-            html = html.substring( index + 2 );
+            html = html.substring( doctypeMatch[0].length );
             prevTag = '';
             continue;
           }
-        }
-
-        // Doctype:
-        if ( (match = doctype.exec( html )) ) {
-          if ( handler.doctype ) {
-            handler.doctype( match[0] );
-          }
-          html = html.substring( match[0].length );
-          prevTag = '';
-          continue;
-        }
 
-        // End tag:
-        if ( /^<\//.test( html ) ) {
-          match = html.match( endTag );
-          if ( match ) {
-            html = html.substring( match[0].length );
-            match[0].replace( endTag, parseEndTag );
-            prevTag = '/' + match[1].toLowerCase();
+          // End tag:
+          var endTagMatch = html.match( endTag );
+          if ( endTagMatch ) {
+            html = html.substring( endTagMatch[0].length );
+            endTagMatch[0].replace( endTag, parseEndTag );
+            prevTag = '/' + endTagMatch[1].toLowerCase();
             continue;
           }
 
-        }
-        // Start tag:
-        if ( /^</.test( html ) ) {
-          match = html.match( startTag );
-          if ( match ) {
-            html = html.substring( match[0].length );
-            match[0].replace( startTag, parseStartTag );
-            prevTag = match[1].toLowerCase();
+          // Start tag:
+          var startTagMatch = html.match( startTag );
+          if ( startTagMatch ) {
+            html = html.substring( startTagMatch[0].length );
+            startTagMatch[0].replace( startTag, parseStartTag );
+            prevTag = startTagMatch[1].toLowerCase();
             continue;
           }
         }
 
-        index = html.indexOf('<');
-
-        var text = index < 0 ? html : html.substring( 0, index );
-        html = index < 0 ? '' : html.substring( index );
+        var text;
+        if (textEnd >= 0) {
+          text = html.substring( 0, textEnd );
+          html = html.substring( textEnd );
+        }
+        else {
+          text = html;
+          html = '';
+        }
 
         // next tag
-        tagMatch = html.match( startTag );
-        if (tagMatch) {
-          nextTag = tagMatch[1];
+        var nextTagMatch = html.match( startTag );
+        if (nextTagMatch) {
+          nextTag = nextTagMatch[1];
         }
         else {
-          tagMatch = html.match( endTag );
-          if (tagMatch) {
-            nextTag = '/' + tagMatch[1];
+          nextTagMatch = html.match( endTag );
+          if (nextTagMatch) {
+            nextTag = '/' + nextTagMatch[1];
           }
           else {
             nextTag = '';
 
       }
       else {
-
-        stackedTag = stack.last().toLowerCase();
-        reStackedTag = reCache[stackedTag] || (reCache[stackedTag] = new RegExp('([\\s\\S]*?)<\/' + stackedTag + '[^>]*>', 'i'));
+        var stackedTag = lastTag.toLowerCase();
+        var reStackedTag = reCache[stackedTag] || (reCache[stackedTag] = new RegExp('([\\s\\S]*?)<\/' + stackedTag + '[^>]*>', 'i'));
 
         html = html.replace(reStackedTag, function(all, text) {
           if (stackedTag !== 'script' && stackedTag !== 'style' && stackedTag !== 'noscript') {
       if ( html === last ) {
         throw 'Parse Error: ' + html;
       }
-      last = html;
     }
 
     if (!handler.partialMarkup) {
     function parseStartTag( tag, tagName, rest, unary ) {
       var unarySlash = false;
 
-      while ( !handler.html5 && stack.last() && inline[ stack.last() ]) {
-        parseEndTag( '', stack.last() );
+      while ( !handler.html5 && lastTag && inline[ lastTag ]) {
+        parseEndTag( '', lastTag );
       }
 
-      if ( closeSelf[ tagName ] && stack.last() === tagName ) {
+      if ( closeSelf[ tagName ] && lastTag === tagName ) {
         parseEndTag( '', tagName );
       }
 
-      unary = empty[ tagName ] || tagName === 'html' && stack.last() === 'head' || !!unary;
+      unary = empty[ tagName ] || tagName === 'html' && lastTag === 'head' || !!unary;
 
       var attrs = [];
 
 
       if ( !unary ) {
         stack.push( { tag: tagName, attrs: attrs } );
+        lastTag = tagName;
       }
       else {
         unarySlash = tag.match( endingSlash );
 
         // Remove the open elements from the stack
         stack.length = pos;
+        lastTag = pos && stack[ pos - 1 ].tag;
       }
     }
   };
index d5dc2c2..e1c24c7 100644 (file)
@@ -3,4 +3,4 @@
  * Copyright 2010-2016 Juriy "kangax" Zaytsev
  * Licensed under the MIT license
  */
-!function(a){"use strict";function b(a){var b,c=new RegExp("[\\w:\\.-]+(?:\\s*(?:"+d(a)+")\\s*(?:\"[^\"]*\"+?|'[^']*'+?|[^>\\s\"']+?))?");if(a.customAttrSurround){for(var e=[],f=a.customAttrSurround.length-1;f>=0;f--)e[f]="(?:"+a.customAttrSurround[f][0].source+"\\s*"+c.source+"\\s*"+a.customAttrSurround[f][1].source+")";e.unshift("(?:"+c.source+")"),b=new RegExp("((?:\\s*(?:"+e.join("|")+"))*)")}else b=new RegExp("((?:\\s*"+c.source+")*)");return new RegExp(j.source+b.source+k.source)}function c(a){var b=new RegExp(f.source+"(?:\\s*("+d(a)+")\\s*(?:"+i.join("|")+"))?");if(a.customAttrSurround){for(var c=[],e=a.customAttrSurround.length-1;e>=0;e--)c[e]="(?:("+a.customAttrSurround[e][0].source+")\\s*"+b.source+"\\s*("+a.customAttrSurround[e][1].source+"))";return c.unshift("(?:"+b.source+")"),new RegExp(c.join("|"),"g")}return new RegExp(b.source,"g")}function d(a){return h.concat(a.customAttrAssign||[]).map(function(a){return"(?:"+a.source+")"}).join("|")}function e(a){for(var b={},c=a.split(","),d=0;d<c.length;d++)b[c[d]]=!0,b[c[d].toUpperCase()]=!0;return b}var f=/([\w:\.-]+)/,g=/=/,h=[g],i=[/"((?:\\.|[^"])*)"/.source,/'((?:\\.|[^'])*)'/.source,/([^>\s]+)/.source],j=/^<([\w:-]+)/,k=/\s*(\/?)>/,l=/^<\/([\w:-]+)[^>]*>/,m=/\/>$/,n=/^<!DOCTYPE [^>]+>/i,o=!1;"x".replace(/x(.)?/g,function(a,b){o=""===b});var p,q,r,s=e("area,base,basefont,br,col,embed,frame,hr,img,input,isindex,keygen,link,meta,param,source,track,wbr"),t=e("a,abbr,acronym,applet,b,basefont,bdo,big,br,button,cite,code,del,dfn,em,font,i,iframe,img,input,ins,kbd,label,map,noscript,object,q,s,samp,script,select,small,span,strike,strong,sub,sup,svg,textarea,tt,u,var"),u=e("colgroup,dd,dt,li,options,p,td,tfoot,th,thead,tr,source"),v=e("checked,compact,declare,defer,disabled,ismap,multiple,nohref,noresize,noshade,nowrap,readonly,selected"),w=e("script,style"),x={},y=a.HTMLParser=function(a,d){function e(a,b,c,e){for(var g=!1;!d.html5&&k.last()&&t[k.last()];)f("",k.last());u[b]&&k.last()===b&&f("",b),e=s[b]||"html"===b&&"head"===k.last()||!!e;var h=[];c.replace(A,function(){var a,b,c,e,f,g,i,j=7;if(o&&-1===arguments[0].indexOf('""')&&(""===arguments[3]&&(arguments[3]=void 0),""===arguments[4]&&(arguments[4]=void 0),""===arguments[5]&&(arguments[5]=void 0)),a=arguments[1])g=arguments[2],c=arguments[3],b=c||arguments[4]||arguments[5],g&&(i=arguments[0].charAt(a.length+g.length),i="'"===i||'"'===i?i:"");else if(d.customAttrSurround)for(var k=d.customAttrSurround.length-1;k>=0;k--)if(a=arguments[k*j+7],g=arguments[k*j+8],a){c=arguments[k*j+9],b=c||arguments[k*j+10]||arguments[k*j+11],e=arguments[k*j+6],f=arguments[k*j+12];break}void 0===b&&(b=v[a]?a:c),h.push({name:a,value:b,customAssign:g||"=",customOpen:e||"",customClose:f||"",quote:i||""})}),e?g=a.match(m):k.push({tag:b,attrs:h}),d.start&&d.start(b,h,e,g)}function f(a,b){var c;if(b){var e=b.toLowerCase();for(c=k.length-1;c>=0&&k[c].tag.toLowerCase()!==e;c--);}else c=0;if(c>=0){for(var f=k.length-1;f>=c;f--)d.end&&d.end(k[f].tag,k[f].attrs);k.length=c}}var g,h,i,j,k=[],y=a;k.last=function(){var a=this[this.length-1];return a&&a.tag};for(var z=b(d),A=c(d);a;){if(k.last()&&w[k.last()])p=k.last().toLowerCase(),q=x[p]||(x[p]=new RegExp("([\\s\\S]*?)</"+p+"[^>]*>","i")),a=a.replace(q,function(a,b){return"script"!==p&&"style"!==p&&"noscript"!==p&&(b=b.replace(/<!--([\s\S]*?)-->/g,"$1").replace(/<!\[CDATA\[([\s\S]*?)\]\]>/g,"$1")),d.chars&&d.chars(b),""}),f("",p);else{if(/^<!--/.test(a)&&(g=a.indexOf("-->"),g>=0)){d.comment&&d.comment(a.substring(4,g)),a=a.substring(g+3),i="";continue}if(/^<!\[/.test(a)&&(g=a.indexOf("]>"),g>=0)){d.comment&&d.comment(a.substring(2,g+1),!0),a=a.substring(g+2),i="";continue}if(h=n.exec(a)){d.doctype&&d.doctype(h[0]),a=a.substring(h[0].length),i="";continue}if(/^<\//.test(a)&&(h=a.match(l))){a=a.substring(h[0].length),h[0].replace(l,f),i="/"+h[1].toLowerCase();continue}if(/^</.test(a)&&(h=a.match(z))){a=a.substring(h[0].length),h[0].replace(z,e),i=h[1].toLowerCase();continue}g=a.indexOf("<");var B=0>g?a:a.substring(0,g);a=0>g?"":a.substring(g),r=a.match(z),r?j=r[1]:(r=a.match(l),j=r?"/"+r[1]:""),d.chars&&d.chars(B,i,j),i=""}if(a===y)throw"Parse Error: "+a;y=a}d.partialMarkup||f()};a.HTMLtoXML=function(a){var b="";return new y(a,{start:function(a,c,d){b+="<"+a;for(var e=0;e<c.length;e++)b+=" "+c[e].name+'="'+(c[e].value||"").replace(/"/g,"&#34;")+'"';b+=(d?"/":"")+">"},end:function(a){b+="</"+a+">"},chars:function(a){b+=a},comment:function(a){b+="<!--"+a+"-->"},ignore:function(a){b+=a}}),b},a.HTMLtoDOM=function(a,b){var c=e("html,head,body,title"),d={link:"head",base:"head"};b?b=b.ownerDocument||b.getOwnerDocument&&b.getOwnerDocument()||b:"undefined"!=typeof DOMDocument?b=new DOMDocument:"undefined"!=typeof document&&document.implementation&&document.implementation.createDocument?b=document.implementation.createDocument("","",null):"undefined"!=typeof ActiveX&&(b=new ActiveXObject("Msxml.DOMDocument"));var f=[],g=b.documentElement||b.getDocumentElement&&b.getDocumentElement();if(!g&&b.createElement&&!function(){var a=b.createElement("html"),c=b.createElement("head");c.appendChild(b.createElement("title")),a.appendChild(c),a.appendChild(b.createElement("body")),b.appendChild(a)}(),b.getElementsByTagName)for(var h in c)c[h]=b.getElementsByTagName(h)[0];var i=c.body;return new y(a,{start:function(a,e,g){if(c[a])return void(i=c[a]);var h=b.createElement(a);for(var j in e)h.setAttribute(e[j].name,e[j].value);d[a]&&"boolean"!=typeof c[d[a]]?c[d[a]].appendChild(h):i&&i.appendChild&&i.appendChild(h),g||(f.push(h),i=h)},end:function(){f.length-=1,i=f[f.length-1]},chars:function(a){i.appendChild(b.createTextNode(a))},comment:function(){},ignore:function(){}}),b}}("undefined"==typeof exports?this:exports),function(a){"use strict";function b(a){return a?a.replace(/[\t\n\r ]+/g," "):a}function c(a){var b={};return a.forEach(function(a){b[a]=1}),function(a){return 1===b[a]}}function d(a){return c(a.split(/,/))}function e(a,c,d,e,f){var g="",h="";return c.preserveLineBreaks&&(a=a.replace(/^[\t ]*[\n\r][\t\n\r ]*/,function(){return g="\n",""}).replace(/[\t\n\r ]*[\n\r][\t ]*$/,function(){return h="\n",""})),d&&(a=a.replace(/^\s+/,!g&&c.conservativeCollapse?" ":"")),e&&(a=a.replace(/\s+$/,!h&&c.conservativeCollapse?" ":"")),f&&(a=b(a)),g+a+h}function f(a,b,c,d){var f=b&&!V(b);f&&!d.collapseInlineTagWhitespace&&(f="/"===b.charAt(0)?!T(b.slice(1)):!U(b));var g=c&&!V(c);return g&&!d.collapseInlineTagWhitespace&&(g="/"===c.charAt(0)?!U(c.slice(1)):!T(c)),e(a,d,f,g,b&&c)}function g(a){return/^\[if\s[^\]]+\]|\[endif\]$/.test(a)}function h(a,b){if(/^!/.test(a))return!0;if(b.ignoreCustomComments)for(var c=0,d=b.ignoreCustomComments.length;d>c;c++)if(b.ignoreCustomComments[c].test(a))return!0;return!1}function i(a,b){var c=b.customEventAttributes;if(c){for(var d=c.length;d--;)if(c[d].test(a))return!0;return!1}return/^on[a-z]{3,}$/.test(a)}function j(a){return/^[^\x20\t\n\f\r"'`=<>]+$/.test(a)}function k(a,b){for(var c=a.length;c--;)if(a[c].name.toLowerCase()===b)return!0;return!1}function l(a,b,c,d){return c=c?S(c.toLowerCase()):"","script"===a&&"language"===b&&"javascript"===c||"form"===a&&"method"===b&&"get"===c||"input"===a&&"type"===b&&"text"===c||"script"===a&&"charset"===b&&!k(d,"src")||"a"===a&&"name"===b&&k(d,"id")||"area"===a&&"shape"===b&&"rect"===c}function m(a,b,c){return"script"===a&&"type"===b&&"text/javascript"===S(c.toLowerCase())}function n(a,b){if("script"!==a)return!1;for(var c=0,d=b.length;d>c;c++){var e=b[c].name.toLowerCase();if("type"===e){var f=S(b[c].value).split(/;/,2)[0].toLowerCase();return""===f||W(f)}}return!0}function o(a,b,c){return("style"===a||"link"===a)&&"type"===b&&"text/css"===S(c.toLowerCase())}function p(a,b){return X(a)||"draggable"===a&&!Y(b)}function q(a,b){return/^(?:a|area|link|base)$/.test(b)&&"href"===a||"img"===b&&/^(?:src|longdesc|usemap)$/.test(a)||"object"===b&&/^(?:classid|codebase|data|usemap)$/.test(a)||"q"===b&&"cite"===a||"blockquote"===b&&"cite"===a||("ins"===b||"del"===b)&&"cite"===a||"form"===b&&"action"===a||"input"===b&&("src"===a||"usemap"===a)||"head"===b&&"profile"===a||"script"===b&&("src"===a||"for"===a)}function r(a,b){return/^(?:a|area|object|button)$/.test(b)&&"tabindex"===a||"input"===b&&("maxlength"===a||"tabindex"===a)||"select"===b&&("size"===a||"tabindex"===a)||"textarea"===b&&/^(?:rows|cols|tabindex)$/.test(a)||"colgroup"===b&&"span"===a||"col"===b&&"span"===a||("th"===b||"td"===b)&&("rowspan"===a||"colspan"===a)}function s(a,b){if("link"!==a)return!1;for(var c=0,d=b.length;d>c;c++)if("rel"===b[c].name&&"canonical"===b[c].value)return!0}function t(a,c,d,e,f){if(d&&i(c,e)){if(d=S(d).replace(/^javascript:\s*/i,"").replace(/\s*;$/,""),e.minifyJS){var g=L(Z+d+$,e.minifyJS);return g.slice(Z.length,-$.length)}return d}return"class"===c?b(S(d)):q(c,a)?(d=S(d),e.minifyURLs&&!s(a,f)?K(d,e.minifyURLs):d):r(c,a)?S(d):"style"===c?(d=S(d),d&&(d=d.replace(/\s*;\s*$/,"")),e.minifyCSS?M(d,e.minifyCSS,!0):d):(u(a,f)&&"content"===c?d=d.replace(/\s+/g,"").replace(/[0-9]+\.[0-9]+/g,function(a){return(+a).toString()}):d&&e.customAttrCollapse&&e.customAttrCollapse.test(c)&&(d=d.replace(/\n+|\r+|\s{2,}/g,"")),d)}function u(a,b){if("meta"!==a)return!1;for(var c=0,d=b.length;d>c;c++)if("name"===b[c].name&&"viewport"===b[c].value)return!0}function v(a){return"*{"+a+"}"}function w(a){var b=a.match(/^\*\{([\s\S]*)\}$/m);return b&&b[1]?b[1]:a}function x(a,b){return a.replace(/^(\[if\s[^\]]+\]>)([\s\S]*?)(<!\[endif\])$/,function(a,c,d,e){return c+O(d,b,!0)+e})}function y(a){return a.replace(/^(?:\s*\/\*\s*<!\[CDATA\[\s*\*\/|\s*\/\/\s*<!\[CDATA\[.*)/,"").replace(/(?:\/\*\s*\]\]>\s*\*\/|\/\/\s*\]\]>)\s*$/,"")}function z(a,b,c){for(var d=0,e=c.length;e>d;d++)if("type"===c[d].name.toLowerCase()&&b.processScripts.indexOf(c[d].value)>-1)return O(a,b);return a}function A(a,b){return a.replace(_[b],"").replace(aa[b],"")}function B(a,b){switch(a){case"html":case"head":return!0;case"body":return!da(b);case"colgroup":return"col"===b;case"tbody":return"tr"===b}return!1}function C(a,b){switch(b){case"colgroup":return"colgroup"===a;case"tbody":return la(a)}return!1}function D(a,b){switch(a){case"html":case"head":case"body":case"colgroup":case"caption":return!0;case"li":case"optgroup":case"tr":return b===a;case"dt":case"dd":return ea(b);case"p":return fa(b);case"rb":case"rt":case"rp":return ha(b);case"rtc":return ia(b);case"option":return ja(b);case"thead":case"tbody":return ka(b);case"tfoot":return"tbody"===b;case"td":case"th":return ma(b)}return!1}function E(a,b,c){var d=!c||/^\s*$/.test(c);return d?"input"===a&&"value"===b||sa.test(b):!1}function F(a,b){if("textarea"===a)return!1;if("script"===a)for(var c=b.length-1;c>=0;c--)if("src"===b[c].name)return!1;return!0}function G(a){return!/^(?:script|style|pre|textarea)$/.test(a)}function H(a){return!/^(?:pre|textarea)$/.test(a)}function I(a,b,c,d,e,f,g){var h,i,k=f.caseSensitive?a.name:a.name.toLowerCase(),n=a.value,q=a.quote;if(f.removeRedundantAttributes&&l(c,k,n,b)||f.removeScriptTypeAttributes&&m(c,k,n)||f.removeStyleLinkTypeAttributes&&o(c,k,n))return"";if(n=t(c,k,n,f,b),f.removeEmptyAttributes&&E(c,k,n))return"";if(void 0!==n&&!f.removeAttributeQuotes||!j(n)){if(!f.preventAttributesEscaping){if(void 0!==f.quoteCharacter)q="'"===f.quoteCharacter?"'":'"';else{var r=(n.match(/'/g)||[]).length,s=(n.match(/"/g)||[]).length;q=s>r?"'":'"'}n='"'===q?n.replace(/"/g,"&#34;"):n.replace(/'/g,"&#39;")}i=q+n+q,g||f.removeTagWhitespace||(i+=" ")}else i=!g||d||/\/$/.test(n)?n+" ":n;return void 0===n||f.collapseBooleanAttributes&&p(k.toLowerCase(),n.toLowerCase())?(h=k,g||(h+=" ")):h=k+a.customAssign+i,a.customOpen+h+a.customClose}function J(a){for(var b=["canCollapseWhitespace","canTrimWhitespace"],c=0,d=b.length;d>c;c++)a[b[c]]||(a[b[c]]=function(){return!1})}function K(b,c){"object"!=typeof c&&(c={});try{var d=a.RelateUrl;return"undefined"==typeof d&&"function"==typeof require&&(d=require("relateurl")),d&&d.relate?d.relate(b,c):b}catch(e){Q(e)}return b}function L(b,c){"object"!=typeof c&&(c={}),c.fromString=!0;var d=c.output||{};d.inline_script=!0,c.output=d;try{var e=a.UglifyJS;if("undefined"==typeof e&&"function"==typeof require&&(e=require("uglify-js")),!e)return b;if(e.minify)return e.minify(b,c).code;if(e.parse){var f=e.parse(b);f.figure_out_scope();var g=e.Compressor(),h=f.transform(g);h.figure_out_scope(),h.compute_char_frequency(),c.mangle!==!1&&h.mangle_names();var i=e.OutputStream(c.output);return h.print(i),i.toString()}return b}catch(j){Q(j)}return b}function M(b,c,d){"object"!=typeof c&&(c={}),"undefined"==typeof c.advanced&&(c.advanced=!1);try{var e=a.CleanCSS;"undefined"==typeof e&&"function"==typeof require&&(e=require("clean-css"));var f=new e(c);return d?w(f.minify(v(b)).styles):f.minify(b).styles}catch(g){Q(g)}return b}function N(a){var b;do b=Math.random().toString(36).slice(2);while(~a.indexOf(b));return b}function O(a,c,d){function i(a,b){return G(a)||c.canCollapseWhitespace(a,b)}function j(a,b){return H(a)||c.canTrimWhitespace(a,b)}function k(){for(var a=s.length-1;a>0&&!/^<[^\/!]/.test(s[a]);)a--;s.length=Math.max(0,a)}function l(){for(var a=s.length-1;a>0&&!/^<\//.test(s[a]);)a--;s.length=Math.max(0,a)}c=c||{};var m=[];J(c),a=c.collapseWhitespace?S(a):a;var o,p,q,r,s=[],t="",u="",v=[],w=[],E=[],K="",O="",T=c.lint,V=Date.now(),W=[],X=[];a=a.replace(/<!-- htmlmin:ignore -->([\s\S]*?)<!-- htmlmin:ignore -->/g,function(b,c){q||(q="<!--!"+N(a)+"-->");var d=q+W.length;return W.push(c),d});var Y=(c.ignoreCustomFragments||[/<%[\s\S]*?%>/,/<\?[\s\S]*?\?>/]).map(function(a){return a.source});if(Y.length){var Z=new RegExp("\\s*(?:"+Y.join("|")+")+\\s*","g");a=a.replace(Z,function(b){r||(r=N(a));var c=r+X.length;return X.push(b),"   "+c+"   "})}new R(a,{partialMarkup:d,html5:"undefined"!=typeof c.html5?c.html5:!0,start:function(a,b,d,e){var f=a.toLowerCase();if("svg"===f){m.push(c);var g={};for(var h in c)g[h]=c[h];g.keepClosingSlash=!0,g.caseSensitive=!0,c=g}a=c.caseSensitive?a:f,u=a,o=a,U(a)||(t=""),p=!1,v=b;var n=c.removeOptionalTags;if(n){var q=ra(a);q&&B(K,a)&&k(),K="",q&&D(O,a)&&(l(),n=!C(O,a)),O=""}c.collapseWhitespace&&(j(a,b)||w.push(a),i(a,b)||E.push(a));var r="<"+a,x=e&&c.keepClosingSlash;s.push(r),T&&T.testElement(a);for(var y,z=[],A=!0,F=b.length;--F>=0;)T&&T.testAttribute(a,b[F].name.toLowerCase(),b[F].value),y=I(b[F],b,a,x,F,c,A),y&&(A=!1,z.unshift(y));z.length>0?(s.push(" "),s.push.apply(s,z)):n&&ba(a)&&(K=a),s.push(s.pop()+(x?"/":"")+">")},end:function(a,b){var d=a.toLowerCase();if("svg"===d&&(c=m.pop()),a=c.caseSensitive?a:d,c.collapseWhitespace){if(w.length)a===w[w.length-1]&&w.pop();else{var e;s.length>1&&""===s[s.length-1]&&/\s+$/.test(s[s.length-2])?e=s.length-2:s.length>0&&/\s+$/.test(s[s.length-1])&&(e=s.length-1),e>0&&(s[e]=s[e].replace(/\s+$/,function(b){return f(b,"comment","/"+a,c)}))}E.length&&a===E[E.length-1]&&E.pop()}var g=!1;a===u&&(u="",g=!p),c.removeOptionalTags&&(g&&na(K)&&k(),K="",!ra(a)||!O||qa(O)||"p"===O&&ga(a)||l(),O=ca(a)?a:""),c.removeEmptyElements&&g&&F(a,b)?(k(),K="",O=""):(s.push("</"+a+">"),o="/"+a,U(a)||(t=""))},chars:function(a,d,g){if(d=""===d?"comment":d,g=""===g?"comment":g,c.collapseWhitespace){if(!w.length){if("comment"===d){var h=""===s[s.length-1];if(h&&(d=o),s.length>1&&(h||/ $/.test(t))){var i=s.length-2;s[i]=s[i].replace(/\s+$/,function(b){return a=b+a,""})}}if(d&&U("/"===d.charAt(0)?d.slice(1):d)&&(a=e(a,c,/(?:^|\s)$/.test(t))),a=d||g?f(a,d,g,c):S(a),!a&&/\s$/.test(t)&&d&&"/"===d.charAt(0))for(var m=s.length-2,q=d.slice(1);m>=0&&j(q);m--){var r=s[m],x=r.match(/^<\/([\w:-]+)>$/);if(x)q=x[1];else if(/>$/.test(r)||(s[m]=f(r,null,g,c)))break}}E.length||(a=d&&g||"html"===g?a:b(a))}"script"!==u&&"style"!==u||(c.removeCommentsFromCDATA&&(a=A(a,u)),c.removeCDATASectionsFromCDATA&&(a=y(a)),c.processScripts&&(a=z(a,c,v))),c.minifyJS&&n(u,v)&&(a=L(a,c.minifyJS),";"===a.charAt(a.length-1)&&(a=a.slice(0,-1))),"style"===u&&c.minifyCSS&&(a=M(a,c.minifyCSS)),c.removeOptionalTags&&a&&(("html"===K||"body"===K&&!/^\s/.test(a))&&k(),K="",(oa(O)||pa(O)&&!/^\s/.test(a))&&l(),O=""),o=/^\s*$/.test(a)?d:"comment",t+=a,a&&(p=!0),T&&T.testChars(a),s.push(a)},comment:function(a,b){var d=b?"<!":"<!--",e=b?">":"-->";a=g(a)?d+x(a,c)+e:c.removeComments?h(a,c)?"<!--"+a+"-->":"":d+a+e,c.removeOptionalTags&&a&&(K="",O=""),s.push(a)},doctype:function(a){s.push(c.useShortDoctype?"<!DOCTYPE html>":b(a))},customAttrAssign:c.customAttrAssign,customAttrSurround:c.customAttrSurround}),c.removeOptionalTags&&(na(K)&&k(),O&&!qa(O)&&l());var $=P(s,c);return r&&($=$.replace(new RegExp("(\\s*)"+r+"([0-9]+)(\\s*)","g"),function(a,b,d,f){var g=X[+d];return c.collapseWhitespace?("       "!==b&&(g=b+g),"        "!==f&&(g+=f),e(g,{preserveLineBreaks:c.preserveLineBreaks,conservativeCollapse:!0},/^\s/.test(g),/\s$/.test(g))):g})),q&&($=$.replace(new RegExp(q+"([0-9]+)","g"),function(a,b){return W[+b]})),Q("minified in: "+(Date.now()-V)+"ms"),$}function P(a,b){var c,d=b.maxLineLength;if(d){for(var e,f=[],g="",h=0,i=a.length;i>h;h++)e=a[h],g.length+e.length<d?g+=e:(f.push(g.replace(/^\n/,"")),g=e);f.push(g),c=f.join("\n")}else c=a.join("");return b.collapseWhitespace?S(c):c}var Q,R;Q=a.console&&a.console.log?function(b){a.console.log(b)}:function(){},a.HTMLParser?R=a.HTMLParser:"function"==typeof require&&(R=require("./htmlparser").HTMLParser);var S=function(a){return"string"!=typeof a?a:a.replace(/^\s+/,"").replace(/\s+$/,"")};String.prototype.trim&&(S=function(a){return"string"!=typeof a?a:a.trim()});var T=d("a,abbr,acronym,b,bdi,bdo,big,button,cite,code,del,dfn,em,font,i,ins,kbd,mark,math,q,rt,rp,s,samp,small,span,strike,strong,sub,sup,svg,time,tt,u,var"),U=d("a,abbr,acronym,b,big,del,em,font,i,ins,kbd,mark,s,samp,small,span,strike,strong,sub,sup,time,tt,u,var"),V=d("comment,img,input"),W=c(["text/javascript","text/ecmascript","text/jscript","application/javascript","application/x-javascript","application/ecmascript"]),X=d("allowfullscreen,async,autofocus,autoplay,checked,compact,controls,declare,default,defaultchecked,defaultmuted,defaultselected,defer,disabled,enabled,formnovalidate,hidden,indeterminate,inert,ismap,itemscope,loop,multiple,muted,nohref,noresize,noshade,novalidate,nowrap,open,pauseonexit,readonly,required,reversed,scoped,seamless,selected,sortable,truespeed,typemustmatch,visible"),Y=d("true,false"),Z="!function(){",$="}();",_={script:/^\s*(?:\/\/)?\s*<!--.*\n?/,style:/^\s*<!--\s*/},aa={script:/\s*(?:\/\/)?\s*-->\s*$/,style:/\s*-->\s*$/},ba=d("html,head,body,colgroup,tbody"),ca=d("html,head,body,li,dt,dd,p,rb,rt,rtc,rp,optgroup,option,colgroup,caption,thead,tbody,tfoot,tr,td,th"),da=d("meta,link,script,style,template,noscript"),ea=d("dt,dd"),fa=d("address,article,aside,blockquote,details,div,dl,fieldset,figcaption,figure,footer,form,h1,h2,h3,h4,h5,h6,header,hgroup,hr,main,menu,nav,ol,p,pre,section,table,ul"),ga=d("a,audio,del,ins,map,noscript,video"),ha=d("rb,rt,rtc,rp"),ia=d("rb,rtc,rp"),ja=d("option,optgroup"),ka=d("tbody,tfoot"),la=d("thead,tbody,tfoot"),ma=d("td,th"),na=d("html,head,body"),oa=d("html,body"),pa=d("head,colgroup,caption"),qa=d("dt,thead"),ra=d("a,abbr,acronym,address,applet,area,article,aside,audio,b,base,basefont,bdi,bdo,bgsound,big,blink,blockquote,body,br,button,canvas,caption,center,cite,code,col,colgroup,command,content,data,datalist,dd,del,details,dfn,dialog,dir,div,dl,dt,element,em,embed,fieldset,figcaption,figure,font,footer,form,frame,frameset,h1,h2,h3,h4,h5,h6,head,header,hgroup,hr,html,i,iframe,image,img,input,ins,isindex,kbd,keygen,label,legend,li,link,listing,main,map,mark,marquee,menu,menuitem,meta,meter,multicol,nav,nobr,noembed,noframes,noscript,object,ol,optgroup,option,output,p,param,picture,plaintext,pre,progress,q,rp,rt,rtc,ruby,s,samp,script,section,select,shadow,small,source,spacer,span,strike,strong,style,sub,summary,sup,table,tbody,td,template,textarea,tfoot,th,thead,time,title,tr,track,tt,u,ul,var,video,wbr,xmp"),sa=new RegExp("^(?:class|id|style|title|lang|dir|on(?:focus|blur|change|click|dblclick|mouse(?:down|up|over|move|out)|key(?:press|down|up)))$");a.minify=function(a,b){return O(a,b)}}("undefined"==typeof exports?this:exports),function(a){"use strict";function b(a){return/^(?:big|small|hr|blink|marquee)$/.test(a)}function c(a){return/^(?:applet|basefont|center|dir|font|isindex|strike)$/.test(a)}function d(a){return/^on[a-z]+/.test(a)}function e(a){return"style"===a.toLowerCase()}function f(a,b){return"align"===b&&/^(?:caption|applet|iframe|img|imput|object|legend|table|hr|div|h[1-6]|p)$/.test(a)||"alink"===b&&"body"===a||"alt"===b&&"applet"===a||"archive"===b&&"applet"===a||"background"===b&&"body"===a||"bgcolor"===b&&/^(?:table|t[rdh]|body)$/.test(a)||"border"===b&&/^(?:img|object)$/.test(a)||"clear"===b&&"br"===a||"code"===b&&"applet"===a||"codebase"===b&&"applet"===a||"color"===b&&/^(?:base(?:font)?)$/.test(a)||"compact"===b&&/^(?:dir|[dou]l|menu)$/.test(a)||"face"===b&&/^base(?:font)?$/.test(a)||"height"===b&&/^(?:t[dh]|applet)$/.test(a)||"hspace"===b&&/^(?:applet|img|object)$/.test(a)||"language"===b&&"script"===a||"link"===b&&"body"===a||"name"===b&&"applet"===a||"noshade"===b&&"hr"===a||"nowrap"===b&&/^t[dh]$/.test(a)||"object"===b&&"applet"===a||"prompt"===b&&"isindex"===a||"size"===b&&/^(?:hr|font|basefont)$/.test(a)||"start"===b&&"ol"===a||"text"===b&&"body"===a||"type"===b&&/^(?:li|ol|ul)$/.test(a)||"value"===b&&"li"===a||"version"===b&&"html"===a||"vlink"===b&&"body"===a||"vspace"===b&&/^(?:applet|img|object)$/.test(a)||"width"===b&&/^(?:hr|td|th|applet|pre)$/.test(a)}function g(a,b){return"href"===a&&/^\s*javascript\s*:\s*void\s*(\s+0|\(\s*0\s*\))\s*$/i.test(b)}function h(){this.log=[],this._lastElement=null,this._isElementRepeated=!1}h.prototype.testElement=function(a){c(a)?this.log.push('Found <span class="deprecated-element">deprecated</span> <strong><code>&lt;'+a+"&gt;</code></strong> element"):b(a)?this.log.push('Found <span class="presentational-element">presentational</span> <strong><code>&lt;'+a+"&gt;</code></strong> element"):this.checkRepeatingElement(a)},h.prototype.checkRepeatingElement=function(a){"br"===a&&"br"===this._lastElement?this._isElementRepeated=!0:this._isElementRepeated&&(this._reportRepeatingElement(),this._isElementRepeated=!1),this._lastElement=a},h.prototype._reportRepeatingElement=function(){this.log.push("Found <code>&lt;br></code> sequence. Try replacing it with styling.")},h.prototype.testAttribute=function(a,b,c){d(b)?this.log.push('Found <span class="event-attribute">event attribute</span> (<strong>'+b+"</strong>) on <strong><code>&lt;"+a+"&gt;</code></strong> element."):f(a,b)?this.log.push('Found <span class="deprecated-attribute">deprecated</span> <strong>'+b+"</strong> attribute on <strong><code>&lt;"+a+"&gt;</code></strong> element."):e(b)?this.log.push('Found <span class="style-attribute">style attribute</span> on <strong><code>&lt;'+a+"&gt;</code></strong> element."):g(b,c)&&this.log.push('Found <span class="inaccessible-attribute">inaccessible attribute</span> (on <strong><code>&lt;'+a+"&gt;</code></strong> element).")},h.prototype.testChars=function(a){this._lastElement="",/(&nbsp;\s*){2,}/.test(a)&&this.log.push("Found repeating <strong><code>&amp;nbsp;</code></strong> sequence. Try replacing it with styling.")},h.prototype.test=function(a,b,c){this.testElement(a),this.testAttribute(a,b,c)},h.prototype.populate=function(a){if(this._isElementRepeated&&this._reportRepeatingElement(),this.log.length)if(a)a.innerHTML="<ol><li>"+this.log.join("<li>")+"</ol>";else{var b=" - "+this.log.join("\n - ").replace(/(<([^>]+)>)/gi,"").replace(/&lt;/g,"<").replace(/&gt;/g,">");console.log(b)}},a.HTMLLint=h}("undefined"==typeof exports?this:exports);
\ No newline at end of file
+!function(a){"use strict";function b(a){var b,c=new RegExp("[\\w:\\.-]+(?:\\s*(?:"+d(a)+")\\s*(?:\"[^\"]*\"+?|'[^']*'+?|[^>\\s\"']+?))?");if(a.customAttrSurround){for(var e=[],f=a.customAttrSurround.length-1;f>=0;f--)e[f]="(?:"+a.customAttrSurround[f][0].source+"\\s*"+c.source+"\\s*"+a.customAttrSurround[f][1].source+")";e.unshift("(?:"+c.source+")"),b=new RegExp("((?:\\s*(?:"+e.join("|")+"))*)")}else b=new RegExp("((?:\\s*"+c.source+")*)");return new RegExp(j.source+b.source+k.source)}function c(a){var b=new RegExp(f.source+"(?:\\s*("+d(a)+")\\s*(?:"+i.join("|")+"))?");if(a.customAttrSurround){for(var c=[],e=a.customAttrSurround.length-1;e>=0;e--)c[e]="(?:("+a.customAttrSurround[e][0].source+")\\s*"+b.source+"\\s*("+a.customAttrSurround[e][1].source+"))";return c.unshift("(?:"+b.source+")"),new RegExp(c.join("|"),"g")}return new RegExp(b.source,"g")}function d(a){return h.concat(a.customAttrAssign||[]).map(function(a){return"(?:"+a.source+")"}).join("|")}function e(a){for(var b={},c=a.split(","),d=0;d<c.length;d++)b[c[d]]=!0,b[c[d].toUpperCase()]=!0;return b}var f=/([\w:\.-]+)/,g=/=/,h=[g],i=[/"((?:\\.|[^"])*)"/.source,/'((?:\\.|[^'])*)'/.source,/([^>\s]+)/.source],j=/^<([\w:-]+)/,k=/\s*(\/?)>/,l=/^<\/([\w:-]+)[^>]*>/,m=/\/>$/,n=/^<!DOCTYPE [^>]+>/i,o=!1;"x".replace(/x(.)?/g,function(a,b){o=""===b});var p=e("area,base,basefont,br,col,embed,frame,hr,img,input,isindex,keygen,link,meta,param,source,track,wbr"),q=e("a,abbr,acronym,applet,b,basefont,bdo,big,br,button,cite,code,del,dfn,em,font,i,iframe,img,input,ins,kbd,label,map,noscript,object,q,s,samp,script,select,small,span,strike,strong,sub,sup,svg,textarea,tt,u,var"),r=e("colgroup,dd,dt,li,options,p,td,tfoot,th,thead,tr,source"),s=e("checked,compact,declare,defer,disabled,ismap,multiple,nohref,noresize,noshade,nowrap,readonly,selected"),t=e("script,style"),u={},v=a.HTMLParser=function(a,d){function e(a,b,c,e){for(var h=!1;!d.html5&&g&&q[g];)f("",g);r[b]&&g===b&&f("",b),e=p[b]||"html"===b&&"head"===g||!!e;var i=[];c.replace(w,function(){var a,b,c,e,f,g,h,j=7;if(o&&-1===arguments[0].indexOf('""')&&(""===arguments[3]&&(arguments[3]=void 0),""===arguments[4]&&(arguments[4]=void 0),""===arguments[5]&&(arguments[5]=void 0)),a=arguments[1])g=arguments[2],c=arguments[3],b=c||arguments[4]||arguments[5],g&&(h=arguments[0].charAt(a.length+g.length),h="'"===h||'"'===h?h:"");else if(d.customAttrSurround)for(var k=d.customAttrSurround.length-1;k>=0;k--)if(a=arguments[k*j+7],g=arguments[k*j+8],a){c=arguments[k*j+9],b=c||arguments[k*j+10]||arguments[k*j+11],e=arguments[k*j+6],f=arguments[k*j+12];break}void 0===b&&(b=s[a]?a:c),i.push({name:a,value:b,customAssign:g||"=",customOpen:e||"",customClose:f||"",quote:h||""})}),e?h=a.match(m):(k.push({tag:b,attrs:i}),g=b),d.start&&d.start(b,i,e,h)}function f(a,b){var c;if(b){var e=b.toLowerCase();for(c=k.length-1;c>=0&&k[c].tag.toLowerCase()!==e;c--);}else c=0;if(c>=0){for(var f=k.length-1;f>=c;f--)d.end&&d.end(k[f].tag,k[f].attrs);k.length=c,g=c&&k[c-1].tag}}for(var g,h,i,j,k=[],v=b(d),w=c(d);a;){if(h=a,g&&t[g]){var x=g.toLowerCase(),y=u[x]||(u[x]=new RegExp("([\\s\\S]*?)</"+x+"[^>]*>","i"));a=a.replace(y,function(a,b){return"script"!==x&&"style"!==x&&"noscript"!==x&&(b=b.replace(/<!--([\s\S]*?)-->/g,"$1").replace(/<!\[CDATA\[([\s\S]*?)\]\]>/g,"$1")),d.chars&&d.chars(b),""}),f("",x)}else{var z=a.indexOf("<");if(0===z){if(/^<!--/.test(a)){var A=a.indexOf("-->");if(A>=0){d.comment&&d.comment(a.substring(4,A)),a=a.substring(A+3),i="";continue}}if(/^<!\[/.test(a)){var B=a.indexOf("]>");if(B>=0){d.comment&&d.comment(a.substring(2,B+1),!0),a=a.substring(B+2),i="";continue}}var C=a.match(n);if(C){d.doctype&&d.doctype(C[0]),a=a.substring(C[0].length),i="";continue}var D=a.match(l);if(D){a=a.substring(D[0].length),D[0].replace(l,f),i="/"+D[1].toLowerCase();continue}var E=a.match(v);if(E){a=a.substring(E[0].length),E[0].replace(v,e),i=E[1].toLowerCase();continue}}var F;z>=0?(F=a.substring(0,z),a=a.substring(z)):(F=a,a="");var G=a.match(v);G?j=G[1]:(G=a.match(l),j=G?"/"+G[1]:""),d.chars&&d.chars(F,i,j),i=""}if(a===h)throw"Parse Error: "+a}d.partialMarkup||f()};a.HTMLtoXML=function(a){var b="";return new v(a,{start:function(a,c,d){b+="<"+a;for(var e=0;e<c.length;e++)b+=" "+c[e].name+'="'+(c[e].value||"").replace(/"/g,"&#34;")+'"';b+=(d?"/":"")+">"},end:function(a){b+="</"+a+">"},chars:function(a){b+=a},comment:function(a){b+="<!--"+a+"-->"},ignore:function(a){b+=a}}),b},a.HTMLtoDOM=function(a,b){var c=e("html,head,body,title"),d={link:"head",base:"head"};b?b=b.ownerDocument||b.getOwnerDocument&&b.getOwnerDocument()||b:"undefined"!=typeof DOMDocument?b=new DOMDocument:"undefined"!=typeof document&&document.implementation&&document.implementation.createDocument?b=document.implementation.createDocument("","",null):"undefined"!=typeof ActiveX&&(b=new ActiveXObject("Msxml.DOMDocument"));var f=[],g=b.documentElement||b.getDocumentElement&&b.getDocumentElement();if(!g&&b.createElement&&!function(){var a=b.createElement("html"),c=b.createElement("head");c.appendChild(b.createElement("title")),a.appendChild(c),a.appendChild(b.createElement("body")),b.appendChild(a)}(),b.getElementsByTagName)for(var h in c)c[h]=b.getElementsByTagName(h)[0];var i=c.body;return new v(a,{start:function(a,e,g){if(c[a])return void(i=c[a]);var h=b.createElement(a);for(var j in e)h.setAttribute(e[j].name,e[j].value);d[a]&&"boolean"!=typeof c[d[a]]?c[d[a]].appendChild(h):i&&i.appendChild&&i.appendChild(h),g||(f.push(h),i=h)},end:function(){f.length-=1,i=f[f.length-1]},chars:function(a){i.appendChild(b.createTextNode(a))},comment:function(){},ignore:function(){}}),b}}("undefined"==typeof exports?this:exports),function(a){"use strict";function b(a){return a?a.replace(/[\t\n\r ]+/g," "):a}function c(a){var b={};return a.forEach(function(a){b[a]=1}),function(a){return 1===b[a]}}function d(a){return c(a.split(/,/))}function e(a,c,d,e,f){var g="",h="";return c.preserveLineBreaks&&(a=a.replace(/^[\t ]*[\n\r][\t\n\r ]*/,function(){return g="\n",""}).replace(/[\t\n\r ]*[\n\r][\t ]*$/,function(){return h="\n",""})),d&&(a=a.replace(/^\s+/,!g&&c.conservativeCollapse?" ":"")),e&&(a=a.replace(/\s+$/,!h&&c.conservativeCollapse?" ":"")),f&&(a=b(a)),g+a+h}function f(a,b,c,d){var f=b&&!V(b);f&&!d.collapseInlineTagWhitespace&&(f="/"===b.charAt(0)?!T(b.slice(1)):!U(b));var g=c&&!V(c);return g&&!d.collapseInlineTagWhitespace&&(g="/"===c.charAt(0)?!U(c.slice(1)):!T(c)),e(a,d,f,g,b&&c)}function g(a){return/^\[if\s[^\]]+\]|\[endif\]$/.test(a)}function h(a,b){if(/^!/.test(a))return!0;if(b.ignoreCustomComments)for(var c=0,d=b.ignoreCustomComments.length;d>c;c++)if(b.ignoreCustomComments[c].test(a))return!0;return!1}function i(a,b){var c=b.customEventAttributes;if(c){for(var d=c.length;d--;)if(c[d].test(a))return!0;return!1}return/^on[a-z]{3,}$/.test(a)}function j(a){return/^[^\x20\t\n\f\r"'`=<>]+$/.test(a)}function k(a,b){for(var c=a.length;c--;)if(a[c].name.toLowerCase()===b)return!0;return!1}function l(a,b,c,d){return c=c?S(c.toLowerCase()):"","script"===a&&"language"===b&&"javascript"===c||"form"===a&&"method"===b&&"get"===c||"input"===a&&"type"===b&&"text"===c||"script"===a&&"charset"===b&&!k(d,"src")||"a"===a&&"name"===b&&k(d,"id")||"area"===a&&"shape"===b&&"rect"===c}function m(a,b,c){return"script"===a&&"type"===b&&"text/javascript"===S(c.toLowerCase())}function n(a,b){if("script"!==a)return!1;for(var c=0,d=b.length;d>c;c++){var e=b[c].name.toLowerCase();if("type"===e){var f=S(b[c].value).split(/;/,2)[0].toLowerCase();return""===f||W(f)}}return!0}function o(a,b,c){return("style"===a||"link"===a)&&"type"===b&&"text/css"===S(c.toLowerCase())}function p(a,b){return X(a)||"draggable"===a&&!Y(b)}function q(a,b){return/^(?:a|area|link|base)$/.test(b)&&"href"===a||"img"===b&&/^(?:src|longdesc|usemap)$/.test(a)||"object"===b&&/^(?:classid|codebase|data|usemap)$/.test(a)||"q"===b&&"cite"===a||"blockquote"===b&&"cite"===a||("ins"===b||"del"===b)&&"cite"===a||"form"===b&&"action"===a||"input"===b&&("src"===a||"usemap"===a)||"head"===b&&"profile"===a||"script"===b&&("src"===a||"for"===a)}function r(a,b){return/^(?:a|area|object|button)$/.test(b)&&"tabindex"===a||"input"===b&&("maxlength"===a||"tabindex"===a)||"select"===b&&("size"===a||"tabindex"===a)||"textarea"===b&&/^(?:rows|cols|tabindex)$/.test(a)||"colgroup"===b&&"span"===a||"col"===b&&"span"===a||("th"===b||"td"===b)&&("rowspan"===a||"colspan"===a)}function s(a,b){if("link"!==a)return!1;for(var c=0,d=b.length;d>c;c++)if("rel"===b[c].name&&"canonical"===b[c].value)return!0}function t(a,c,d,e,f){if(d&&i(c,e)){if(d=S(d).replace(/^javascript:\s*/i,"").replace(/\s*;$/,""),e.minifyJS){var g=L(Z+d+$,e.minifyJS);return g.slice(Z.length,-$.length)}return d}return"class"===c?b(S(d)):q(c,a)?(d=S(d),e.minifyURLs&&!s(a,f)?K(d,e.minifyURLs):d):r(c,a)?S(d):"style"===c?(d=S(d),d&&(d=d.replace(/\s*;\s*$/,"")),e.minifyCSS?M(d,e.minifyCSS,!0):d):(u(a,f)&&"content"===c?d=d.replace(/\s+/g,"").replace(/[0-9]+\.[0-9]+/g,function(a){return(+a).toString()}):d&&e.customAttrCollapse&&e.customAttrCollapse.test(c)&&(d=d.replace(/\n+|\r+|\s{2,}/g,"")),d)}function u(a,b){if("meta"!==a)return!1;for(var c=0,d=b.length;d>c;c++)if("name"===b[c].name&&"viewport"===b[c].value)return!0}function v(a){return"*{"+a+"}"}function w(a){var b=a.match(/^\*\{([\s\S]*)\}$/m);return b&&b[1]?b[1]:a}function x(a,b){return a.replace(/^(\[if\s[^\]]+\]>)([\s\S]*?)(<!\[endif\])$/,function(a,c,d,e){return c+O(d,b,!0)+e})}function y(a){return a.replace(/^(?:\s*\/\*\s*<!\[CDATA\[\s*\*\/|\s*\/\/\s*<!\[CDATA\[.*)/,"").replace(/(?:\/\*\s*\]\]>\s*\*\/|\/\/\s*\]\]>)\s*$/,"")}function z(a,b,c){for(var d=0,e=c.length;e>d;d++)if("type"===c[d].name.toLowerCase()&&b.processScripts.indexOf(c[d].value)>-1)return O(a,b);return a}function A(a,b){return a.replace(_[b],"").replace(aa[b],"")}function B(a,b){switch(a){case"html":case"head":return!0;case"body":return!da(b);case"colgroup":return"col"===b;case"tbody":return"tr"===b}return!1}function C(a,b){switch(b){case"colgroup":return"colgroup"===a;case"tbody":return la(a)}return!1}function D(a,b){switch(a){case"html":case"head":case"body":case"colgroup":case"caption":return!0;case"li":case"optgroup":case"tr":return b===a;case"dt":case"dd":return ea(b);case"p":return fa(b);case"rb":case"rt":case"rp":return ha(b);case"rtc":return ia(b);case"option":return ja(b);case"thead":case"tbody":return ka(b);case"tfoot":return"tbody"===b;case"td":case"th":return ma(b)}return!1}function E(a,b,c){var d=!c||/^\s*$/.test(c);return d?"input"===a&&"value"===b||sa.test(b):!1}function F(a,b){if("textarea"===a)return!1;if("script"===a)for(var c=b.length-1;c>=0;c--)if("src"===b[c].name)return!1;return!0}function G(a){return!/^(?:script|style|pre|textarea)$/.test(a)}function H(a){return!/^(?:pre|textarea)$/.test(a)}function I(a,b,c,d,e,f,g){var h,i,k=f.caseSensitive?a.name:a.name.toLowerCase(),n=a.value,q=a.quote;if(f.removeRedundantAttributes&&l(c,k,n,b)||f.removeScriptTypeAttributes&&m(c,k,n)||f.removeStyleLinkTypeAttributes&&o(c,k,n))return"";if(n=t(c,k,n,f,b),f.removeEmptyAttributes&&E(c,k,n))return"";if(void 0!==n&&!f.removeAttributeQuotes||!j(n)){if(!f.preventAttributesEscaping){if(void 0!==f.quoteCharacter)q="'"===f.quoteCharacter?"'":'"';else{var r=(n.match(/'/g)||[]).length,s=(n.match(/"/g)||[]).length;q=s>r?"'":'"'}n='"'===q?n.replace(/"/g,"&#34;"):n.replace(/'/g,"&#39;")}i=q+n+q,g||f.removeTagWhitespace||(i+=" ")}else i=!g||d||/\/$/.test(n)?n+" ":n;return void 0===n||f.collapseBooleanAttributes&&p(k.toLowerCase(),n.toLowerCase())?(h=k,g||(h+=" ")):h=k+a.customAssign+i,a.customOpen+h+a.customClose}function J(a){for(var b=["canCollapseWhitespace","canTrimWhitespace"],c=0,d=b.length;d>c;c++)a[b[c]]||(a[b[c]]=function(){return!1})}function K(b,c){"object"!=typeof c&&(c={});try{var d=a.RelateUrl;return"undefined"==typeof d&&"function"==typeof require&&(d=require("relateurl")),d&&d.relate?d.relate(b,c):b}catch(e){Q(e)}return b}function L(b,c){"object"!=typeof c&&(c={}),c.fromString=!0;var d=c.output||{};d.inline_script=!0,c.output=d;try{var e=a.UglifyJS;if("undefined"==typeof e&&"function"==typeof require&&(e=require("uglify-js")),!e)return b;if(e.minify)return e.minify(b,c).code;if(e.parse){var f=e.parse(b);f.figure_out_scope();var g=e.Compressor(),h=f.transform(g);h.figure_out_scope(),h.compute_char_frequency(),c.mangle!==!1&&h.mangle_names();var i=e.OutputStream(c.output);return h.print(i),i.toString()}return b}catch(j){Q(j)}return b}function M(b,c,d){"object"!=typeof c&&(c={}),"undefined"==typeof c.advanced&&(c.advanced=!1);try{var e=a.CleanCSS;"undefined"==typeof e&&"function"==typeof require&&(e=require("clean-css"));var f=new e(c);return d?w(f.minify(v(b)).styles):f.minify(b).styles}catch(g){Q(g)}return b}function N(a){var b;do b=Math.random().toString(36).slice(2);while(~a.indexOf(b));return b}function O(a,c,d){function i(a,b){return G(a)||c.canCollapseWhitespace(a,b)}function j(a,b){return H(a)||c.canTrimWhitespace(a,b)}function k(){for(var a=s.length-1;a>0&&!/^<[^\/!]/.test(s[a]);)a--;s.length=Math.max(0,a)}function l(){for(var a=s.length-1;a>0&&!/^<\//.test(s[a]);)a--;s.length=Math.max(0,a)}c=c||{};var m=[];J(c),a=c.collapseWhitespace?S(a):a;var o,p,q,r,s=[],t="",u="",v=[],w=[],E=[],K="",O="",T=c.lint,V=Date.now(),W=[],X=[];a=a.replace(/<!-- htmlmin:ignore -->([\s\S]*?)<!-- htmlmin:ignore -->/g,function(b,c){q||(q="<!--!"+N(a)+"-->");var d=q+W.length;return W.push(c),d});var Y=(c.ignoreCustomFragments||[/<%[\s\S]*?%>/,/<\?[\s\S]*?\?>/]).map(function(a){return a.source});if(Y.length){var Z=new RegExp("\\s*(?:"+Y.join("|")+")+\\s*","g");a=a.replace(Z,function(b){r||(r=N(a));var c=r+X.length;return X.push(b),"    "+c+"   "})}new R(a,{partialMarkup:d,html5:"undefined"!=typeof c.html5?c.html5:!0,start:function(a,b,d,e){var f=a.toLowerCase();if("svg"===f){m.push(c);var g={};for(var h in c)g[h]=c[h];g.keepClosingSlash=!0,g.caseSensitive=!0,c=g}a=c.caseSensitive?a:f,u=a,o=a,U(a)||(t=""),p=!1,v=b;var n=c.removeOptionalTags;if(n){var q=ra(a);q&&B(K,a)&&k(),K="",q&&D(O,a)&&(l(),n=!C(O,a)),O=""}c.collapseWhitespace&&(j(a,b)||w.push(a),i(a,b)||E.push(a));var r="<"+a,x=e&&c.keepClosingSlash;s.push(r),T&&T.testElement(a);for(var y,z=[],A=!0,F=b.length;--F>=0;)T&&T.testAttribute(a,b[F].name.toLowerCase(),b[F].value),y=I(b[F],b,a,x,F,c,A),y&&(A=!1,z.unshift(y));z.length>0?(s.push(" "),s.push.apply(s,z)):n&&ba(a)&&(K=a),s.push(s.pop()+(x?"/":"")+">")},end:function(a,b){var d=a.toLowerCase();if("svg"===d&&(c=m.pop()),a=c.caseSensitive?a:d,c.collapseWhitespace){if(w.length)a===w[w.length-1]&&w.pop();else{var e;s.length>1&&""===s[s.length-1]&&/\s+$/.test(s[s.length-2])?e=s.length-2:s.length>0&&/\s+$/.test(s[s.length-1])&&(e=s.length-1),e>0&&(s[e]=s[e].replace(/\s+$/,function(b){return f(b,"comment","/"+a,c)}))}E.length&&a===E[E.length-1]&&E.pop()}var g=!1;a===u&&(u="",g=!p),c.removeOptionalTags&&(g&&na(K)&&k(),K="",!ra(a)||!O||qa(O)||"p"===O&&ga(a)||l(),O=ca(a)?a:""),c.removeEmptyElements&&g&&F(a,b)?(k(),K="",O=""):(s.push("</"+a+">"),o="/"+a,U(a)||(t=""))},chars:function(a,d,g){if(d=""===d?"comment":d,g=""===g?"comment":g,c.collapseWhitespace){if(!w.length){if("comment"===d){var h=""===s[s.length-1];if(h&&(d=o),s.length>1&&(h||/ $/.test(t))){var i=s.length-2;s[i]=s[i].replace(/\s+$/,function(b){return a=b+a,""})}}if(d&&U("/"===d.charAt(0)?d.slice(1):d)&&(a=e(a,c,/(?:^|\s)$/.test(t))),a=d||g?f(a,d,g,c):S(a),!a&&/\s$/.test(t)&&d&&"/"===d.charAt(0))for(var m=s.length-2,q=d.slice(1);m>=0&&j(q);m--){var r=s[m],x=r.match(/^<\/([\w:-]+)>$/);if(x)q=x[1];else if(/>$/.test(r)||(s[m]=f(r,null,g,c)))break}}E.length||(a=d&&g||"html"===g?a:b(a))}"script"!==u&&"style"!==u||(c.removeCommentsFromCDATA&&(a=A(a,u)),c.removeCDATASectionsFromCDATA&&(a=y(a)),c.processScripts&&(a=z(a,c,v))),c.minifyJS&&n(u,v)&&(a=L(a,c.minifyJS),";"===a.charAt(a.length-1)&&(a=a.slice(0,-1))),"style"===u&&c.minifyCSS&&(a=M(a,c.minifyCSS)),c.removeOptionalTags&&a&&(("html"===K||"body"===K&&!/^\s/.test(a))&&k(),K="",(oa(O)||pa(O)&&!/^\s/.test(a))&&l(),O=""),o=/^\s*$/.test(a)?d:"comment",t+=a,a&&(p=!0),T&&T.testChars(a),s.push(a)},comment:function(a,b){var d=b?"<!":"<!--",e=b?">":"-->";a=g(a)?d+x(a,c)+e:c.removeComments?h(a,c)?"<!--"+a+"-->":"":d+a+e,c.removeOptionalTags&&a&&(K="",O=""),s.push(a)},doctype:function(a){s.push(c.useShortDoctype?"<!DOCTYPE html>":b(a))},customAttrAssign:c.customAttrAssign,customAttrSurround:c.customAttrSurround}),c.removeOptionalTags&&(na(K)&&k(),O&&!qa(O)&&l());var $=P(s,c);return r&&($=$.replace(new RegExp("(\\s*)"+r+"([0-9]+)(\\s*)","g"),function(a,b,d,f){var g=X[+d];return c.collapseWhitespace?("       "!==b&&(g=b+g),"        "!==f&&(g+=f),e(g,{preserveLineBreaks:c.preserveLineBreaks,conservativeCollapse:!0},/^\s/.test(g),/\s$/.test(g))):g})),q&&($=$.replace(new RegExp(q+"([0-9]+)","g"),function(a,b){return W[+b]})),Q("minified in: "+(Date.now()-V)+"ms"),$}function P(a,b){var c,d=b.maxLineLength;if(d){for(var e,f=[],g="",h=0,i=a.length;i>h;h++)e=a[h],g.length+e.length<d?g+=e:(f.push(g.replace(/^\n/,"")),g=e);f.push(g),c=f.join("\n")}else c=a.join("");return b.collapseWhitespace?S(c):c}var Q,R;Q=a.console&&a.console.log?function(b){a.console.log(b)}:function(){},a.HTMLParser?R=a.HTMLParser:"function"==typeof require&&(R=require("./htmlparser").HTMLParser);var S=function(a){return"string"!=typeof a?a:a.replace(/^\s+/,"").replace(/\s+$/,"")};String.prototype.trim&&(S=function(a){return"string"!=typeof a?a:a.trim()});var T=d("a,abbr,acronym,b,bdi,bdo,big,button,cite,code,del,dfn,em,font,i,ins,kbd,mark,math,q,rt,rp,s,samp,small,span,strike,strong,sub,sup,svg,time,tt,u,var"),U=d("a,abbr,acronym,b,big,del,em,font,i,ins,kbd,mark,s,samp,small,span,strike,strong,sub,sup,time,tt,u,var"),V=d("comment,img,input"),W=c(["text/javascript","text/ecmascript","text/jscript","application/javascript","application/x-javascript","application/ecmascript"]),X=d("allowfullscreen,async,autofocus,autoplay,checked,compact,controls,declare,default,defaultchecked,defaultmuted,defaultselected,defer,disabled,enabled,formnovalidate,hidden,indeterminate,inert,ismap,itemscope,loop,multiple,muted,nohref,noresize,noshade,novalidate,nowrap,open,pauseonexit,readonly,required,reversed,scoped,seamless,selected,sortable,truespeed,typemustmatch,visible"),Y=d("true,false"),Z="!function(){",$="}();",_={script:/^\s*(?:\/\/)?\s*<!--.*\n?/,style:/^\s*<!--\s*/},aa={script:/\s*(?:\/\/)?\s*-->\s*$/,style:/\s*-->\s*$/},ba=d("html,head,body,colgroup,tbody"),ca=d("html,head,body,li,dt,dd,p,rb,rt,rtc,rp,optgroup,option,colgroup,caption,thead,tbody,tfoot,tr,td,th"),da=d("meta,link,script,style,template,noscript"),ea=d("dt,dd"),fa=d("address,article,aside,blockquote,details,div,dl,fieldset,figcaption,figure,footer,form,h1,h2,h3,h4,h5,h6,header,hgroup,hr,main,menu,nav,ol,p,pre,section,table,ul"),ga=d("a,audio,del,ins,map,noscript,video"),ha=d("rb,rt,rtc,rp"),ia=d("rb,rtc,rp"),ja=d("option,optgroup"),ka=d("tbody,tfoot"),la=d("thead,tbody,tfoot"),ma=d("td,th"),na=d("html,head,body"),oa=d("html,body"),pa=d("head,colgroup,caption"),qa=d("dt,thead"),ra=d("a,abbr,acronym,address,applet,area,article,aside,audio,b,base,basefont,bdi,bdo,bgsound,big,blink,blockquote,body,br,button,canvas,caption,center,cite,code,col,colgroup,command,content,data,datalist,dd,del,details,dfn,dialog,dir,div,dl,dt,element,em,embed,fieldset,figcaption,figure,font,footer,form,frame,frameset,h1,h2,h3,h4,h5,h6,head,header,hgroup,hr,html,i,iframe,image,img,input,ins,isindex,kbd,keygen,label,legend,li,link,listing,main,map,mark,marquee,menu,menuitem,meta,meter,multicol,nav,nobr,noembed,noframes,noscript,object,ol,optgroup,option,output,p,param,picture,plaintext,pre,progress,q,rp,rt,rtc,ruby,s,samp,script,section,select,shadow,small,source,spacer,span,strike,strong,style,sub,summary,sup,table,tbody,td,template,textarea,tfoot,th,thead,time,title,tr,track,tt,u,ul,var,video,wbr,xmp"),sa=new RegExp("^(?:class|id|style|title|lang|dir|on(?:focus|blur|change|click|dblclick|mouse(?:down|up|over|move|out)|key(?:press|down|up)))$");a.minify=function(a,b){return O(a,b)}}("undefined"==typeof exports?this:exports),function(a){"use strict";function b(a){return/^(?:big|small|hr|blink|marquee)$/.test(a)}function c(a){return/^(?:applet|basefont|center|dir|font|isindex|strike)$/.test(a)}function d(a){return/^on[a-z]+/.test(a)}function e(a){return"style"===a.toLowerCase()}function f(a,b){return"align"===b&&/^(?:caption|applet|iframe|img|imput|object|legend|table|hr|div|h[1-6]|p)$/.test(a)||"alink"===b&&"body"===a||"alt"===b&&"applet"===a||"archive"===b&&"applet"===a||"background"===b&&"body"===a||"bgcolor"===b&&/^(?:table|t[rdh]|body)$/.test(a)||"border"===b&&/^(?:img|object)$/.test(a)||"clear"===b&&"br"===a||"code"===b&&"applet"===a||"codebase"===b&&"applet"===a||"color"===b&&/^(?:base(?:font)?)$/.test(a)||"compact"===b&&/^(?:dir|[dou]l|menu)$/.test(a)||"face"===b&&/^base(?:font)?$/.test(a)||"height"===b&&/^(?:t[dh]|applet)$/.test(a)||"hspace"===b&&/^(?:applet|img|object)$/.test(a)||"language"===b&&"script"===a||"link"===b&&"body"===a||"name"===b&&"applet"===a||"noshade"===b&&"hr"===a||"nowrap"===b&&/^t[dh]$/.test(a)||"object"===b&&"applet"===a||"prompt"===b&&"isindex"===a||"size"===b&&/^(?:hr|font|basefont)$/.test(a)||"start"===b&&"ol"===a||"text"===b&&"body"===a||"type"===b&&/^(?:li|ol|ul)$/.test(a)||"value"===b&&"li"===a||"version"===b&&"html"===a||"vlink"===b&&"body"===a||"vspace"===b&&/^(?:applet|img|object)$/.test(a)||"width"===b&&/^(?:hr|td|th|applet|pre)$/.test(a)}function g(a,b){return"href"===a&&/^\s*javascript\s*:\s*void\s*(\s+0|\(\s*0\s*\))\s*$/i.test(b)}function h(){this.log=[],this._lastElement=null,this._isElementRepeated=!1}h.prototype.testElement=function(a){c(a)?this.log.push('Found <span class="deprecated-element">deprecated</span> <strong><code>&lt;'+a+"&gt;</code></strong> element"):b(a)?this.log.push('Found <span class="presentational-element">presentational</span> <strong><code>&lt;'+a+"&gt;</code></strong> element"):this.checkRepeatingElement(a)},h.prototype.checkRepeatingElement=function(a){"br"===a&&"br"===this._lastElement?this._isElementRepeated=!0:this._isElementRepeated&&(this._reportRepeatingElement(),this._isElementRepeated=!1),this._lastElement=a},h.prototype._reportRepeatingElement=function(){this.log.push("Found <code>&lt;br></code> sequence. Try replacing it with styling.")},h.prototype.testAttribute=function(a,b,c){d(b)?this.log.push('Found <span class="event-attribute">event attribute</span> (<strong>'+b+"</strong>) on <strong><code>&lt;"+a+"&gt;</code></strong> element."):f(a,b)?this.log.push('Found <span class="deprecated-attribute">deprecated</span> <strong>'+b+"</strong> attribute on <strong><code>&lt;"+a+"&gt;</code></strong> element."):e(b)?this.log.push('Found <span class="style-attribute">style attribute</span> on <strong><code>&lt;'+a+"&gt;</code></strong> element."):g(b,c)&&this.log.push('Found <span class="inaccessible-attribute">inaccessible attribute</span> (on <strong><code>&lt;'+a+"&gt;</code></strong> element).")},h.prototype.testChars=function(a){this._lastElement="",/(&nbsp;\s*){2,}/.test(a)&&this.log.push("Found repeating <strong><code>&amp;nbsp;</code></strong> sequence. Try replacing it with styling.")},h.prototype.test=function(a,b,c){this.testElement(a),this.testAttribute(a,b,c)},h.prototype.populate=function(a){if(this._isElementRepeated&&this._reportRepeatingElement(),this.log.length)if(a)a.innerHTML="<ol><li>"+this.log.join("<li>")+"</ol>";else{var b=" - "+this.log.join("\n - ").replace(/(<([^>]+)>)/gi,"").replace(/&lt;/g,"<").replace(/&gt;/g,">");console.log(b)}},a.HTMLLint=h}("undefined"==typeof exports?this:exports);
\ No newline at end of file
index 4830759..2368f5d 100644 (file)
@@ -67,7 +67,7 @@
   // Special Elements (can contain anything)
   var special = makeMap('script,style');
 
-  var reCache = {}, stackedTag, reStackedTag, tagMatch;
+  var reCache = {};
 
   function startTagForHandler( handler ) {
     var customStartTagAttrs;
   }
 
   var HTMLParser = global.HTMLParser = function( html, handler ) {
-    var index, match, stack = [], last = html, prevTag, nextTag;
-    stack.last = function() {
-      var last = this[ this.length - 1 ];
-      return last && last.tag;
-    };
-
+    var stack = [], lastTag;
     var startTag = startTagForHandler(handler);
     var attr = attrForHandler(handler);
-
+    var last, prevTag, nextTag;
     while ( html ) {
+      last = html;
       // Make sure we're not in a script or style element
-      if ( !stack.last() || !special[ stack.last() ] ) {
-
-        // Comment:
-        if ( /^<!--/.test( html ) ) {
-          index = html.indexOf('-->');
-
-          if ( index >= 0 ) {
-            if ( handler.comment ) {
-              handler.comment( html.substring( 4, index ) );
+      if ( !lastTag || !special[ lastTag ] ) {
+        var textEnd = html.indexOf('<');
+        if (textEnd === 0) {
+          // Comment:
+          if ( /^<!--/.test( html ) ) {
+            var commentEnd = html.indexOf('-->');
+
+            if ( commentEnd >= 0 ) {
+              if ( handler.comment ) {
+                handler.comment( html.substring( 4, commentEnd ) );
+              }
+              html = html.substring( commentEnd + 3 );
+              prevTag = '';
+              continue;
             }
-            html = html.substring( index + 3 );
-            prevTag = '';
-            continue;
           }
-        }
 
-        // http://en.wikipedia.org/wiki/Conditional_comment#Downlevel-revealed_conditional_comment
-        if ( /^<!\[/.test( html ) ) {
-          index = html.indexOf(']>');
+          // http://en.wikipedia.org/wiki/Conditional_comment#Downlevel-revealed_conditional_comment
+          if ( /^<!\[/.test( html ) ) {
+            var conditionalEnd = html.indexOf(']>');
+
+            if (conditionalEnd >= 0) {
+              if ( handler.comment ) {
+                handler.comment( html.substring(2, conditionalEnd + 1 ), true /* non-standard */ );
+              }
+              html = html.substring( conditionalEnd + 2 );
+              prevTag = '';
+              continue;
+            }
+          }
 
-          if (index >= 0) {
-            if ( handler.comment ) {
-              handler.comment( html.substring(2, index + 1 ), true /* non-standard */ );
+          // Doctype:
+          var doctypeMatch = html.match( doctype );
+          if ( doctypeMatch ) {
+            if ( handler.doctype ) {
+              handler.doctype( doctypeMatch[0] );
             }
-            html = html.substring( index + 2 );
+            html = html.substring( doctypeMatch[0].length );
             prevTag = '';
             continue;
           }
-        }
-
-        // Doctype:
-        if ( (match = doctype.exec( html )) ) {
-          if ( handler.doctype ) {
-            handler.doctype( match[0] );
-          }
-          html = html.substring( match[0].length );
-          prevTag = '';
-          continue;
-        }
 
-        // End tag:
-        if ( /^<\//.test( html ) ) {
-          match = html.match( endTag );
-          if ( match ) {
-            html = html.substring( match[0].length );
-            match[0].replace( endTag, parseEndTag );
-            prevTag = '/' + match[1].toLowerCase();
+          // End tag:
+          var endTagMatch = html.match( endTag );
+          if ( endTagMatch ) {
+            html = html.substring( endTagMatch[0].length );
+            endTagMatch[0].replace( endTag, parseEndTag );
+            prevTag = '/' + endTagMatch[1].toLowerCase();
             continue;
           }
 
-        }
-        // Start tag:
-        if ( /^</.test( html ) ) {
-          match = html.match( startTag );
-          if ( match ) {
-            html = html.substring( match[0].length );
-            match[0].replace( startTag, parseStartTag );
-            prevTag = match[1].toLowerCase();
+          // Start tag:
+          var startTagMatch = html.match( startTag );
+          if ( startTagMatch ) {
+            html = html.substring( startTagMatch[0].length );
+            startTagMatch[0].replace( startTag, parseStartTag );
+            prevTag = startTagMatch[1].toLowerCase();
             continue;
           }
         }
 
-        index = html.indexOf('<');
-
-        var text = index < 0 ? html : html.substring( 0, index );
-        html = index < 0 ? '' : html.substring( index );
+        var text;
+        if (textEnd >= 0) {
+          text = html.substring( 0, textEnd );
+          html = html.substring( textEnd );
+        }
+        else {
+          text = html;
+          html = '';
+        }
 
         // next tag
-        tagMatch = html.match( startTag );
-        if (tagMatch) {
-          nextTag = tagMatch[1];
+        var nextTagMatch = html.match( startTag );
+        if (nextTagMatch) {
+          nextTag = nextTagMatch[1];
         }
         else {
-          tagMatch = html.match( endTag );
-          if (tagMatch) {
-            nextTag = '/' + tagMatch[1];
+          nextTagMatch = html.match( endTag );
+          if (nextTagMatch) {
+            nextTag = '/' + nextTagMatch[1];
           }
           else {
             nextTag = '';
 
       }
       else {
-
-        stackedTag = stack.last().toLowerCase();
-        reStackedTag = reCache[stackedTag] || (reCache[stackedTag] = new RegExp('([\\s\\S]*?)<\/' + stackedTag + '[^>]*>', 'i'));
+        var stackedTag = lastTag.toLowerCase();
+        var reStackedTag = reCache[stackedTag] || (reCache[stackedTag] = new RegExp('([\\s\\S]*?)<\/' + stackedTag + '[^>]*>', 'i'));
 
         html = html.replace(reStackedTag, function(all, text) {
           if (stackedTag !== 'script' && stackedTag !== 'style' && stackedTag !== 'noscript') {
       if ( html === last ) {
         throw 'Parse Error: ' + html;
       }
-      last = html;
     }
 
     if (!handler.partialMarkup) {
     function parseStartTag( tag, tagName, rest, unary ) {
       var unarySlash = false;
 
-      while ( !handler.html5 && stack.last() && inline[ stack.last() ]) {
-        parseEndTag( '', stack.last() );
+      while ( !handler.html5 && lastTag && inline[ lastTag ]) {
+        parseEndTag( '', lastTag );
       }
 
-      if ( closeSelf[ tagName ] && stack.last() === tagName ) {
+      if ( closeSelf[ tagName ] && lastTag === tagName ) {
         parseEndTag( '', tagName );
       }
 
-      unary = empty[ tagName ] || tagName === 'html' && stack.last() === 'head' || !!unary;
+      unary = empty[ tagName ] || tagName === 'html' && lastTag === 'head' || !!unary;
 
       var attrs = [];
 
 
       if ( !unary ) {
         stack.push( { tag: tagName, attrs: attrs } );
+        lastTag = tagName;
       }
       else {
         unarySlash = tag.match( endingSlash );
 
         // Remove the open elements from the stack
         stack.length = pos;
+        lastTag = pos && stack[ pos - 1 ].tag;
       }
     }
   };