plugins/ZenCoding/zen_textarea.js

3114 lines
89 KiB
JavaScript
Raw Normal View History

2013-12-06 09:22:36 +08:00
(function(){
/**
* Zen Coding settings
* @author Sergey Chikuyonok (serge.che@gmail.com)
* @link http://chikuyonok.ru
*/
var zen_settings = {
/**
* Variables that can be placed inside snippets or abbreviations as ${variable}
* ${child} variable is reserved, don't use it
*/
'variables': {
'lang': 'en',
'locale': 'en-US',
'charset': 'UTF-8',
'profile': 'xhtml',
/** Inner element indentation */
'indentation': '\t' // TODO take from Aptana settings
},
'css': {
'snippets': {
"@i": "@import url(|);",
"@m": "@media print {\n\t|\n}",
"@f": "@font-face {\n\tfont-family:|;\n\tsrc:url(|);\n}",
"!": "!important",
"pos": "position:|;",
"pos:s": "position:static;",
"pos:a": "position:absolute;",
"pos:r": "position:relative;",
"pos:f": "position:fixed;",
"t": "top:|;",
"t:a": "top:auto;",
"r": "right:|;",
"r:a": "right:auto;",
"b": "bottom:|;",
"b:a": "bottom:auto;",
"l": "left:|;",
"l:a": "left:auto;",
"z": "z-index:|;",
"z:a": "z-index:auto;",
"fl": "float:|;",
"fl:n": "float:none;",
"fl:l": "float:left;",
"fl:r": "float:right;",
"cl": "clear:|;",
"cl:n": "clear:none;",
"cl:l": "clear:left;",
"cl:r": "clear:right;",
"cl:b": "clear:both;",
"d": "display:|;",
"d:n": "display:none;",
"d:b": "display:block;",
"d:ib": "display:inline;",
"d:li": "display:list-item;",
"d:ri": "display:run-in;",
"d:cp": "display:compact;",
"d:tb": "display:table;",
"d:itb": "display:inline-table;",
"d:tbcp": "display:table-caption;",
"d:tbcl": "display:table-column;",
"d:tbclg": "display:table-column-group;",
"d:tbhg": "display:table-header-group;",
"d:tbfg": "display:table-footer-group;",
"d:tbr": "display:table-row;",
"d:tbrg": "display:table-row-group;",
"d:tbc": "display:table-cell;",
"d:rb": "display:ruby;",
"d:rbb": "display:ruby-base;",
"d:rbbg": "display:ruby-base-group;",
"d:rbt": "display:ruby-text;",
"d:rbtg": "display:ruby-text-group;",
"v": "visibility:|;",
"v:v": "visibility:visible;",
"v:h": "visibility:hidden;",
"v:c": "visibility:collapse;",
"ov": "overflow:|;",
"ov:v": "overflow:visible;",
"ov:h": "overflow:hidden;",
"ov:s": "overflow:scroll;",
"ov:a": "overflow:auto;",
"ovx": "overflow-x:|;",
"ovx:v": "overflow-x:visible;",
"ovx:h": "overflow-x:hidden;",
"ovx:s": "overflow-x:scroll;",
"ovx:a": "overflow-x:auto;",
"ovy": "overflow-y:|;",
"ovy:v": "overflow-y:visible;",
"ovy:h": "overflow-y:hidden;",
"ovy:s": "overflow-y:scroll;",
"ovy:a": "overflow-y:auto;",
"ovs": "overflow-style:|;",
"ovs:a": "overflow-style:auto;",
"ovs:s": "overflow-style:scrollbar;",
"ovs:p": "overflow-style:panner;",
"ovs:m": "overflow-style:move;",
"ovs:mq": "overflow-style:marquee;",
"zoo": "zoom:1;",
"cp": "clip:|;",
"cp:a": "clip:auto;",
"cp:r": "clip:rect(|);",
"bxz": "box-sizing:|;",
"bxz:cb": "box-sizing:content-box;",
"bxz:bb": "box-sizing:border-box;",
"bxsh": "box-shadow:|;",
"bxsh:n": "box-shadow:none;",
"bxsh:w": "-webkit-box-shadow:0 0 0 #000;",
"bxsh:m": "-moz-box-shadow:0 0 0 0 #000;",
"m": "margin:|;",
"m:a": "margin:auto;",
"m:0": "margin:0;",
"m:2": "margin:0 0;",
"m:3": "margin:0 0 0;",
"m:4": "margin:0 0 0 0;",
"mt": "margin-top:|;",
"mt:a": "margin-top:auto;",
"mr": "margin-right:|;",
"mr:a": "margin-right:auto;",
"mb": "margin-bottom:|;",
"mb:a": "margin-bottom:auto;",
"ml": "margin-left:|;",
"ml:a": "margin-left:auto;",
"p": "padding:|;",
"p:0": "padding:0;",
"p:2": "padding:0 0;",
"p:3": "padding:0 0 0;",
"p:4": "padding:0 0 0 0;",
"pt": "padding-top:|;",
"pr": "padding-right:|;",
"pb": "padding-bottom:|;",
"pl": "padding-left:|;",
"w": "width:|;",
"w:a": "width:auto;",
"h": "height:|;",
"h:a": "height:auto;",
"maw": "max-width:|;",
"maw:n": "max-width:none;",
"mah": "max-height:|;",
"mah:n": "max-height:none;",
"miw": "min-width:|;",
"mih": "min-height:|;",
"o": "outline:|;",
"o:n": "outline:none;",
"oo": "outline-offset:|;",
"ow": "outline-width:|;",
"os": "outline-style:|;",
"oc": "outline-color:#000;",
"oc:i": "outline-color:invert;",
"bd": "border:|;",
"bd+": "border:1px solid #000;",
"bd:n": "border:none;",
"bdbk": "border-break:|;",
"bdbk:c": "border-break:close;",
"bdcl": "border-collapse:|;",
"bdcl:c": "border-collapse:collapse;",
"bdcl:s": "border-collapse:separate;",
"bdc": "border-color:#000;",
"bdi": "border-image:url(|);",
"bdi:n": "border-image:none;",
"bdi:w": "-webkit-border-image:url(|) 0 0 0 0 stretch stretch;",
"bdi:m": "-moz-border-image:url(|) 0 0 0 0 stretch stretch;",
"bdti": "border-top-image:url(|);",
"bdti:n": "border-top-image:none;",
"bdri": "border-right-image:url(|);",
"bdri:n": "border-right-image:none;",
"bdbi": "border-bottom-image:url(|);",
"bdbi:n": "border-bottom-image:none;",
"bdli": "border-left-image:url(|);",
"bdli:n": "border-left-image:none;",
"bdci": "border-corner-image:url(|);",
"bdci:n": "border-corner-image:none;",
"bdci:c": "border-corner-image:continue;",
"bdtli": "border-top-left-image:url(|);",
"bdtli:n": "border-top-left-image:none;",
"bdtli:c": "border-top-left-image:continue;",
"bdtri": "border-top-right-image:url(|);",
"bdtri:n": "border-top-right-image:none;",
"bdtri:c": "border-top-right-image:continue;",
"bdbri": "border-bottom-right-image:url(|);",
"bdbri:n": "border-bottom-right-image:none;",
"bdbri:c": "border-bottom-right-image:continue;",
"bdbli": "border-bottom-left-image:url(|);",
"bdbli:n": "border-bottom-left-image:none;",
"bdbli:c": "border-bottom-left-image:continue;",
"bdf": "border-fit:|;",
"bdf:c": "border-fit:clip;",
"bdf:r": "border-fit:repeat;",
"bdf:sc": "border-fit:scale;",
"bdf:st": "border-fit:stretch;",
"bdf:ow": "border-fit:overwrite;",
"bdf:of": "border-fit:overflow;",
"bdf:sp": "border-fit:space;",
"bdl": "border-length:|;",
"bdl:a": "border-length:auto;",
"bdsp": "border-spacing:|;",
"bds": "border-style:|;",
"bds:n": "border-style:none;",
"bds:h": "border-style:hidden;",
"bds:dt": "border-style:dotted;",
"bds:ds": "border-style:dashed;",
"bds:s": "border-style:solid;",
"bds:db": "border-style:double;",
"bds:dtds": "border-style:dot-dash;",
"bds:dtdtds": "border-style:dot-dot-dash;",
"bds:w": "border-style:wave;",
"bds:g": "border-style:groove;",
"bds:r": "border-style:ridge;",
"bds:i": "border-style:inset;",
"bds:o": "border-style:outset;",
"bdw": "border-width:|;",
"bdt": "border-top:|;",
"bdt+": "border-top:1px solid #000;",
"bdt:n": "border-top:none;",
"bdtw": "border-top-width:|;",
"bdts": "border-top-style:|;",
"bdts:n": "border-top-style:none;",
"bdtc": "border-top-color:#000;",
"bdr": "border-right:|;",
"bdr+": "border-right:1px solid #000;",
"bdr:n": "border-right:none;",
"bdrw": "border-right-width:|;",
"bdrs": "border-right-style:|;",
"bdrs:n": "border-right-style:none;",
"bdrc": "border-right-color:#000;",
"bdb": "border-bottom:|;",
"bdb+": "border-bottom:1px solid #000;",
"bdb:n": "border-bottom:none;",
"bdbw": "border-bottom-width:|;",
"bdbs": "border-bottom-style:|;",
"bdbs:n": "border-bottom-style:none;",
"bdbc": "border-bottom-color:#000;",
"bdl": "border-left:|;",
"bdl+": "border-left:1px solid #000;",
"bdl:n": "border-left:none;",
"bdlw": "border-left-width:|;",
"bdls": "border-left-style:|;",
"bdls:n": "border-left-style:none;",
"bdlc": "border-left-color:#000;",
"bdrs": "border-radius:|;",
"bdtrrs": "border-top-right-radius:|;",
"bdtlrs": "border-top-left-radius:|;",
"bdbrrs": "border-bottom-right-radius:|;",
"bdblrs": "border-bottom-left-radius:|;",
"bg": "background:|;",
"bg+": "background:#FFF url(|) 0 0 no-repeat;",
"bg:n": "background:none;",
"bg:ie": "filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='|x.png');",
"bgc": "background-color:#FFF;",
"bgi": "background-image:url(|);",
"bgi:n": "background-image:none;",
"bgr": "background-repeat:|;",
"bgr:n": "background-repeat:no-repeat;",
"bgr:x": "background-repeat:repeat-x;",
"bgr:y": "background-repeat:repeat-y;",
"bga": "background-attachment:|;",
"bga:f": "background-attachment:fixed;",
"bga:s": "background-attachment:scroll;",
"bgp": "background-position:0 0;",
"bgpx": "background-position-x:|;",
"bgpy": "background-position-y:|;",
"bgbk": "background-break:|;",
"bgbk:bb": "background-break:bounding-box;",
"bgbk:eb": "background-break:each-box;",
"bgbk:c": "background-break:continuous;",
"bgcp": "background-clip:|;",
"bgcp:bb": "background-clip:border-box;",
"bgcp:pb": "background-clip:padding-box;",
"bgcp:cb": "background-clip:content-box;",
"bgcp:nc": "background-clip:no-clip;",
"bgo": "background-origin:|;",
"bgo:pb": "background-origin:padding-box;",
"bgo:bb": "background-origin:border-box;",
"bgo:cb": "background-origin:content-box;",
"bgz": "background-size:|;",
"bgz:a": "background-size:auto;",
"bgz:ct": "background-size:contain;",
"bgz:cv": "background-size:cover;",
"c": "color:#000;",
"tbl": "table-layout:|;",
"tbl:a": "table-layout:auto;",
"tbl:f": "table-layout:fixed;",
"cps": "caption-side:|;",
"cps:t": "caption-side:top;",
"cps:b": "caption-side:bottom;",
"ec": "empty-cells:|;",
"ec:s": "empty-cells:show;",
"ec:h": "empty-cells:hide;",
"lis": "list-style:|;",
"lis:n": "list-style:none;",
"lisp": "list-style-position:|;",
"lisp:i": "list-style-position:inside;",
"lisp:o": "list-style-position:outside;",
"list": "list-style-type:|;",
"list:n": "list-style-type:none;",
"list:d": "list-style-type:disc;",
"list:c": "list-style-type:circle;",
"list:s": "list-style-type:square;",
"list:dc": "list-style-type:decimal;",
"list:dclz": "list-style-type:decimal-leading-zero;",
"list:lr": "list-style-type:lower-roman;",
"list:ur": "list-style-type:upper-roman;",
"lisi": "list-style-image:|;",
"lisi:n": "list-style-image:none;",
"q": "quotes:|;",
"q:n": "quotes:none;",
"q:ru": "quotes:'\00AB' '\00BB' '\201E' '\201C';",
"q:en": "quotes:'\201C' '\201D' '\2018' '\2019';",
"ct": "content:|;",
"ct:n": "content:normal;",
"ct:oq": "content:open-quote;",
"ct:noq": "content:no-open-quote;",
"ct:cq": "content:close-quote;",
"ct:ncq": "content:no-close-quote;",
"ct:a": "content:attr(|);",
"ct:c": "content:counter(|);",
"ct:cs": "content:counters(|);",
"coi": "counter-increment:|;",
"cor": "counter-reset:|;",
"va": "vertical-align:|;",
"va:sup": "vertical-align:super;",
"va:t": "vertical-align:top;",
"va:tt": "vertical-align:text-top;",
"va:m": "vertical-align:middle;",
"va:bl": "vertical-align:baseline;",
"va:b": "vertical-align:bottom;",
"va:tb": "vertical-align:text-bottom;",
"va:sub": "vertical-align:sub;",
"ta": "text-align:|;",
"ta:l": "text-align:left;",
"ta:c": "text-align:center;",
"ta:r": "text-align:right;",
"tal": "text-align-last:|;",
"tal:a": "text-align-last:auto;",
"tal:l": "text-align-last:left;",
"tal:c": "text-align-last:center;",
"tal:r": "text-align-last:right;",
"td": "text-decoration:|;",
"td:n": "text-decoration:none;",
"td:u": "text-decoration:underline;",
"td:o": "text-decoration:overline;",
"td:l": "text-decoration:line-through;",
"te": "text-emphasis:|;",
"te:n": "text-emphasis:none;",
"te:ac": "text-emphasis:accent;",
"te:dt": "text-emphasis:dot;",
"te:c": "text-emphasis:circle;",
"te:ds": "text-emphasis:disc;",
"te:b": "text-emphasis:before;",
"te:a": "text-emphasis:after;",
"th": "text-height:|;",
"th:a": "text-height:auto;",
"th:f": "text-height:font-size;",
"th:t": "text-height:text-size;",
"th:m": "text-height:max-size;",
"ti": "text-indent:|;",
"ti:-": "text-indent:-9999px;",
"tj": "text-justify:|;",
"tj:a": "text-justify:auto;",
"tj:iw": "text-justify:inter-word;",
"tj:ii": "text-justify:inter-ideograph;",
"tj:ic": "text-justify:inter-cluster;",
"tj:d": "text-justify:distribute;",
"tj:k": "text-justify:kashida;",
"tj:t": "text-justify:tibetan;",
"to": "text-outline:|;",
"to+": "text-outline:0 0 #000;",
"to:n": "text-outline:none;",
"tr": "text-replace:|;",
"tr:n": "text-replace:none;",
"tt": "text-transform:|;",
"tt:n": "text-transform:none;",
"tt:c": "text-transform:capitalize;",
"tt:u": "text-transform:uppercase;",
"tt:l": "text-transform:lowercase;",
"tw": "text-wrap:|;",
"tw:n": "text-wrap:normal;",
"tw:no": "text-wrap:none;",
"tw:u": "text-wrap:unrestricted;",
"tw:s": "text-wrap:suppress;",
"tsh": "text-shadow:|;",
"tsh+": "text-shadow:0 0 0 #000;",
"tsh:n": "text-shadow:none;",
"lh": "line-height:|;",
"whs": "white-space:|;",
"whs:n": "white-space:normal;",
"whs:p": "white-space:pre;",
"whs:nw": "white-space:nowrap;",
"whs:pw": "white-space:pre-wrap;",
"whs:pl": "white-space:pre-line;",
"whsc": "white-space-collapse:|;",
"whsc:n": "white-space-collapse:normal;",
"whsc:k": "white-space-collapse:keep-all;",
"whsc:l": "white-space-collapse:loose;",
"whsc:bs": "white-space-collapse:break-strict;",
"whsc:ba": "white-space-collapse:break-all;",
"wob": "word-break:|;",
"wob:n": "word-break:normal;",
"wob:k": "word-break:keep-all;",
"wob:l": "word-break:loose;",
"wob:bs": "word-break:break-strict;",
"wob:ba": "word-break:break-all;",
"wos": "word-spacing:|;",
"wow": "word-wrap:|;",
"wow:nm": "word-wrap:normal;",
"wow:n": "word-wrap:none;",
"wow:u": "word-wrap:unrestricted;",
"wow:s": "word-wrap:suppress;",
"lts": "letter-spacing:|;",
"f": "font:|;",
"f+": "font:1em Arial,sans-serif;",
"fw": "font-weight:|;",
"fw:n": "font-weight:normal;",
"fw:b": "font-weight:bold;",
"fw:br": "font-weight:bolder;",
"fw:lr": "font-weight:lighter;",
"fs": "font-style:|;",
"fs:n": "font-style:normal;",
"fs:i": "font-style:italic;",
"fs:o": "font-style:oblique;",
"fv": "font-variant:|;",
"fv:n": "font-variant:normal;",
"fv:sc": "font-variant:small-caps;",
"fz": "font-size:|;",
"fza": "font-size-adjust:|;",
"fza:n": "font-size-adjust:none;",
"ff": "font-family:|;",
"ff:s": "font-family:serif;",
"ff:ss": "font-family:sans-serif;",
"ff:c": "font-family:cursive;",
"ff:f": "font-family:fantasy;",
"ff:m": "font-family:monospace;",
"fef": "font-effect:|;",
"fef:n": "font-effect:none;",
"fef:eg": "font-effect:engrave;",
"fef:eb": "font-effect:emboss;",
"fef:o": "font-effect:outline;",
"fem": "font-emphasize:|;",
"femp": "font-emphasize-position:|;",
"femp:b": "font-emphasize-position:before;",
"femp:a": "font-emphasize-position:after;",
"fems": "font-emphasize-style:|;",
"fems:n": "font-emphasize-style:none;",
"fems:ac": "font-emphasize-style:accent;",
"fems:dt": "font-emphasize-style:dot;",
"fems:c": "font-emphasize-style:circle;",
"fems:ds": "font-emphasize-style:disc;",
"fsm": "font-smooth:|;",
"fsm:a": "font-smooth:auto;",
"fsm:n": "font-smooth:never;",
"fsm:aw": "font-smooth:always;",
"fst": "font-stretch:|;",
"fst:n": "font-stretch:normal;",
"fst:uc": "font-stretch:ultra-condensed;",
"fst:ec": "font-stretch:extra-condensed;",
"fst:c": "font-stretch:condensed;",
"fst:sc": "font-stretch:semi-condensed;",
"fst:se": "font-stretch:semi-expanded;",
"fst:e": "font-stretch:expanded;",
"fst:ee": "font-stretch:extra-expanded;",
"fst:ue": "font-stretch:ultra-expanded;",
"op": "opacity:|;",
"op:ie": "filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);",
"op:ms": "-ms-filter:'progid:DXImageTransform.Microsoft.Alpha(Opacity=100)';",
"rz": "resize:|;",
"rz:n": "resize:none;",
"rz:b": "resize:both;",
"rz:h": "resize:horizontal;",
"rz:v": "resize:vertical;",
"cur": "cursor:|;",
"cur:a": "cursor:auto;",
"cur:d": "cursor:default;",
"cur:c": "cursor:crosshair;",
"cur:ha": "cursor:hand;",
"cur:he": "cursor:help;",
"cur:m": "cursor:move;",
"cur:p": "cursor:pointer;",
"cur:t": "cursor:text;",
"pgbb": "page-break-before:|;",
"pgbb:au": "page-break-before:auto;",
"pgbb:al": "page-break-before:always;",
"pgbb:l": "page-break-before:left;",
"pgbb:r": "page-break-before:right;",
"pgbi": "page-break-inside:|;",
"pgbi:au": "page-break-inside:auto;",
"pgbi:av": "page-break-inside:avoid;",
"pgba": "page-break-after:|;",
"pgba:au": "page-break-after:auto;",
"pgba:al": "page-break-after:always;",
"pgba:l": "page-break-after:left;",
"pgba:r": "page-break-after:right;",
"orp": "orphans:|;",
"wid": "widows:|;"
}
},
'html': {
'snippets': {
'cc:ie6': '<!--[if lte IE 6]>\n\t${child}|\n<![endif]-->',
'cc:ie': '<!--[if IE]>\n\t${child}|\n<![endif]-->',
'cc:noie': '<!--[if !IE]><!-->\n\t${child}|\n<!--<![endif]-->',
'html:4t': '<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">\n' +
'<html lang="${lang}">\n' +
'<head>\n' +
' <title></title>\n' +
' <meta http-equiv="Content-Type" content="text/html;charset=${charset}">\n' +
'</head>\n' +
'<body>\n\t${child}|\n</body>\n' +
'</html>',
'html:4s': '<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">\n' +
'<html lang="${lang}">\n' +
'<head>\n' +
' <title></title>\n' +
' <meta http-equiv="Content-Type" content="text/html;charset=${charset}">\n' +
'</head>\n' +
'<body>\n\t${child}|\n</body>\n' +
'</html>',
'html:xt': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">\n' +
'<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="${lang}">\n' +
'<head>\n' +
' <title></title>\n' +
' <meta http-equiv="Content-Type" content="text/html;charset=${charset}" />\n' +
'</head>\n' +
'<body>\n\t${child}|\n</body>\n' +
'</html>',
'html:xs': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">\n' +
'<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="${lang}">\n' +
'<head>\n' +
' <title></title>\n' +
' <meta http-equiv="Content-Type" content="text/html;charset=${charset}" />\n' +
'</head>\n' +
'<body>\n\t${child}|\n</body>\n' +
'</html>',
'html:xxs': '<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">\n' +
'<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="${lang}">\n' +
'<head>\n' +
' <title></title>\n' +
' <meta http-equiv="Content-Type" content="text/html;charset=${charset}" />\n' +
'</head>\n' +
'<body>\n\t${child}|\n</body>\n' +
'</html>',
'html:5': '<!DOCTYPE HTML>\n' +
'<html lang="${locale}">\n' +
'<head>\n' +
' <title></title>\n' +
' <meta charset="${charset}">\n' +
'</head>\n' +
'<body>\n\t${child}|\n</body>\n' +
'</html>',
'more': '\n<!--more-->\n'
},
'abbreviations': {
'a': '<a href=""></a>',
'a:link': '<a href="http://|"></a>',
'a:mail': '<a href="mailto:|"></a>',
'abbr': '<abbr title=""></abbr>',
'acronym': '<acronym title=""></acronym>',
'base': '<base href="" />',
'bdo': '<bdo dir=""></bdo>',
'bdo:r': '<bdo dir="rtl"></bdo>',
'bdo:l': '<bdo dir="ltr"></bdo>',
'link:css': '<link rel="stylesheet" type="text/css" href="|style.css" media="all" />',
'link:print': '<link rel="stylesheet" type="text/css" href="|print.css" media="print" />',
'link:favicon': '<link rel="shortcut icon" type="image/x-icon" href="|favicon.ico" />',
'link:touch': '<link rel="apple-touch-icon" href="|favicon.png" />',
'link:rss': '<link rel="alternate" type="application/rss+xml" title="RSS" href="|rss.xml" />',
'link:atom': '<link rel="alternate" type="application/atom+xml" title="Atom" href="atom.xml" />',
'meta:utf': '<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />',
'meta:win': '<meta http-equiv="Content-Type" content="text/html;charset=windows-1251" />',
'meta:compat': '<meta http-equiv="X-UA-Compatible" content="IE=7" />',
'style': '<style type="text/css"></style>',
'script': '<script type="text/javascript"></script>',
'script:src': '<script type="text/javascript" src=""></script>',
'img': '<img src="" alt="" />',
'iframe': '<iframe src="" frameborder="0"></iframe>',
'embed': '<embed src="" type="" />',
'object': '<object data="" type=""></object>',
'param': '<param name="" value="" />',
'map': '<map name=""></map>',
'area': '<area shape="" coords="" href="" alt="" />',
'area:d': '<area shape="default" href="" alt="" />',
'area:c': '<area shape="circle" coords="" href="" alt="" />',
'area:r': '<area shape="rect" coords="" href="" alt="" />',
'area:p': '<area shape="poly" coords="" href="" alt="" />',
'link': '<link rel="stylesheet" href="" />',
'form': '<form action=""></form>',
'form:get': '<form action="" method="get"></form>',
'form:post': '<form action="" method="post"></form>',
'label': '<label for=""></label>',
'input': '<input type="" />',
'input:hidden': '<input type="hidden" name="" />',
'input:h': '<input type="hidden" name="" />',
'input:text': '<input type="text" name="" id="" />',
'input:t': '<input type="text" name="" id="" />',
'input:search': '<input type="search" name="" id="" />',
'input:email': '<input type="email" name="" id="" />',
'input:url': '<input type="url" name="" id="" />',
'input:password': '<input type="password" name="" id="" />',
'input:p': '<input type="password" name="" id="" />',
'input:datetime': '<input type="datetime" name="" id="" />',
'input:date': '<input type="date" name="" id="" />',
'input:datetime-local': '<input type="datetime-local" name="" id="" />',
'input:month': '<input type="month" name="" id="" />',
'input:week': '<input type="week" name="" id="" />',
'input:time': '<input type="time" name="" id="" />',
'input:number': '<input type="number" name="" id="" />',
'input:color': '<input type="color" name="" id="" />',
'input:checkbox': '<input type="checkbox" name="" id="" />',
'input:c': '<input type="checkbox" name="" id="" />',
'input:radio': '<input type="radio" name="" id="" />',
'input:r': '<input type="radio" name="" id="" />',
'input:range': '<input type="range" name="" id="" />',
'input:file': '<input type="file" name="" id="" />',
'input:f': '<input type="file" name="" id="" />',
'input:submit': '<input type="submit" value="" />',
'input:s': '<input type="submit" value="" />',
'input:image': '<input type="image" src="" alt="" />',
'input:i': '<input type="image" src="" alt="" />',
'input:reset': '<input type="reset" value="" />',
'input:button': '<input type="button" value="" />',
'input:b': '<input type="button" value="" />',
'select': '<select name="" id=""></select>',
'option': '<option value=""></option>',
'textarea': '<textarea name="" id="" cols="30" rows="10"></textarea>',
'menu:context': '<menu type="context"></menu>',
'menu:c': '<menu type="context"></menu>',
'menu:toolbar': '<menu type="toolbar"></menu>',
'menu:t': '<menu type="toolbar"></menu>',
'video': '<video src=""></video>',
'audio': '<audio src=""></audio>',
'html:xml': '<html xmlns="http://www.w3.org/1999/xhtml"></html>',
'bq': '<blockquote></blockquote>',
'acr': '<acronym></acronym>',
'fig': '<figure></figure>',
'ifr': '<iframe></iframe>',
'emb': '<embed></embed>',
'obj': '<object></object>',
'src': '<source></source>',
'cap': '<caption></caption>',
'colg': '<colgroup></colgroup>',
'fst': '<fieldset></fieldset>',
'btn': '<button></button>',
'optg': '<optgroup></optgroup>',
'opt': '<option></option>',
'tarea': '<textarea></textarea>',
'leg': '<legend></legend>',
'sect': '<section></section>',
'art': '<article></article>',
'hdr': '<header></header>',
'ftr': '<footer></footer>',
'adr': '<address></address>',
'dlg': '<dialog></dialog>',
'str': '<strong></strong>',
'prog': '<progress></progress>',
'fset': '<fieldset></fieldset>',
'datag': '<datagrid></datagrid>',
'datal': '<datalist></datalist>',
'kg': '<keygen></keygen>',
'out': '<output></output>',
'det': '<details></details>',
'cmd': '<command></command>',
// expandos
'ol+': 'ol>li',
'ul+': 'ul>li',
'dl+': 'dl>dt+dd',
'map+': 'map>area',
'table+': 'table>tr>td',
'colgroup+': 'colgroup>col',
'colg+': 'colgroup>col',
'tr+': 'tr>td',
'select+': 'select>option',
'optgroup+': 'optgroup>option',
'optg+': 'optgroup>option'
},
'element_types': {
'empty': 'area,base,basefont,br,col,frame,hr,img,input,isindex,link,meta,param,embed,keygen,command',
'block_level': 'address,applet,blockquote,button,center,dd,del,dir,div,dl,dt,fieldset,form,frameset,hr,iframe,ins,isindex,li,link,map,menu,noframes,noscript,object,ol,p,pre,script,table,tbody,td,tfoot,th,thead,tr,ul,h1,h2,h3,h4,h5,h6',
'inline_level': 'a,abbr,acronym,applet,b,basefont,bdo,big,br,button,cite,code,del,dfn,em,font,i,iframe,img,input,ins,kbd,label,map,object,q,s,samp,script,select,small,span,strike,strong,sub,sup,textarea,tt,u,var'
}
},
'xsl': {
'extends': 'html',
'abbreviations': {
'tm': '<xsl:template match="" mode=""></xsl:template>',
'tmatch': 'tm',
'tn': '<xsl:template name=""></xsl:template>',
'tname': 'tn',
'xsl:when': '<xsl:when test=""></xsl:when>',
'wh': 'xsl:when',
'var': '<xsl:variable name="">|</xsl:variable>',
'vare': '<xsl:variable name="" select=""/>',
'if': '<xsl:if test=""></xsl:if>',
'call': '<xsl:call-template name=""/>',
'attr': '<xsl:attribute name=""></xsl:attribute>',
'wp': '<xsl:with-param name="" select=""/>',
'par': '<xsl:param name="" select=""/>',
'val': '<xsl:value-of select=""/>',
'co': '<xsl:copy-of select=""/>',
'each': '<xsl:for-each select=""></xsl:for-each>',
'ap': '<xsl:apply-templates select="" mode=""/>',
//expandos
'choose+': 'xsl:choose>xsl:when+xsl:otherwise'
}
}
};/**
* @author Sergey Chikuyonok (serge.che@gmail.com)
* @link http://chikuyonok.ru
*/ (function(){
// Regular Expressions for parsing tags and attributes
var start_tag = /^<([\w\:\-]+)((?:\s+[\w\-:]+(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*)\s*(\/?)>/,
end_tag = /^<\/([\w\:\-]+)[^>]*>/,
attr = /([\w\-:]+)(?:\s*=\s*(?:(?:"((?:\\.|[^"])*)")|(?:'((?:\\.|[^'])*)')|([^>\s]+)))?/g;
// Empty Elements - HTML 4.01
var empty = makeMap("area,base,basefont,br,col,frame,hr,img,input,isindex,link,meta,param,embed");
// Block Elements - HTML 4.01
var block = makeMap("address,applet,blockquote,button,center,dd,dir,div,dl,dt,fieldset,form,frameset,hr,iframe,isindex,li,map,menu,noframes,noscript,object,ol,p,pre,script,table,tbody,td,tfoot,th,thead,tr,ul");
// Inline Elements - HTML 4.01
var inline = makeMap("a,abbr,acronym,applet,b,basefont,bdo,big,br,button,cite,code,del,dfn,em,font,i,iframe,img,input,ins,kbd,label,map,object,q,s,samp,select,small,span,strike,strong,sub,sup,textarea,tt,u,var");
// Elements that you can, intentionally, leave open
// (and which close themselves)
var close_self = makeMap("colgroup,dd,dt,li,options,p,td,tfoot,th,thead,tr");
/** Last matched HTML pair */
var last_match = {
opening_tag: null, // tag() or comment() object
closing_tag: null, // tag() or comment() object
start_ix: -1,
end_ix: -1
};
function tag(match, ix) {
var name = match[1].toLowerCase();
return {
name: name,
full_tag: match[0],
start: ix,
end: ix + match[0].length,
unary: Boolean(match[3]) || (name in empty),
type: 'tag',
close_self: (name in close_self)
};
}
function comment(start, end) {
return {
start: start,
end: end,
type: 'comment'
};
}
function makeMap(str){
var obj = {}, items = str.split(",");
for ( var i = 0; i < items.length; i++ )
obj[ items[i] ] = true;
return obj;
}
/**
* Makes selection ranges for matched tag pair
* @param {tag} opening_tag
* @param {tag} closing_tag
* @param {Number} ix
*/
function makeRange(opening_tag, closing_tag, ix) {
ix = ix || 0;
var start_ix = -1,
end_ix = -1;
if (opening_tag && !closing_tag) { // unary element
start_ix = opening_tag.start;
end_ix = opening_tag.end;
} else if (opening_tag && closing_tag) { // complete element
if (
(opening_tag.start < ix && opening_tag.end > ix) ||
(closing_tag.start <= ix && closing_tag.end > ix)
) {
start_ix = opening_tag.start;
end_ix = closing_tag.end;
} else {
start_ix = opening_tag.end;
end_ix = closing_tag.start;
}
}
return [start_ix, end_ix];
}
/**
* Save matched tag for later use and return found indexes
* @param {tag} opening_tag
* @param {tag} closing_tag
* @param {Number} ix
* @return {Array}
*/
function saveMatch(opening_tag, closing_tag, ix) {
ix = ix || 0;
last_match.opening_tag = opening_tag;
last_match.closing_tag = closing_tag;
var range = makeRange(opening_tag, closing_tag, ix);
last_match.start_ix = range[0];
last_match.end_ix = range[1];
return last_match.start_ix != -1 ? [last_match.start_ix, last_match.end_ix] : null;
}
/**
* Search for matching tags in <code>html</code>, starting from
* <code>start_ix</code> position
* @param {String} html Code to search
* @param {Number} start_ix Character index where to start searching pair
* (commonly, current caret position)
* @param {Function} action Function that creates selection range
* @return {Array|null}
*/
function findPair(html, start_ix, action) {
action = action || makeRange;
var forward_stack = [],
backward_stack = [],
/** @type {tag()} */
opening_tag = null,
/** @type {tag()} */
closing_tag = null,
range = null,
html_len = html.length,
m,
ix,
tmp_tag;
forward_stack.last = backward_stack.last = function() {
return this[this.length - 1];
}
function hasMatch(str, start) {
if (arguments.length == 1)
start = ix;
return html.substr(start, str.length) == str;
}
function searchCommentStart(from) {
while (from--) {
if (html.charAt(from) == '<' && hasMatch('<!--', from))
break;
}
return from;
}
// find opening tag
ix = start_ix;
while (ix-- && ix >= 0) {
var ch = html.charAt(ix);
if (ch == '<') {
var check_str = html.substring(ix, html_len);
if ( (m = check_str.match(end_tag)) ) { // found closing tag
tmp_tag = tag(m, ix);
if (tmp_tag.start < start_ix && tmp_tag.end > start_ix) // direct hit on searched closing tag
closing_tag = tmp_tag;
else
backward_stack.push(tmp_tag);
} else if ( (m = check_str.match(start_tag)) ) { // found opening tag
tmp_tag = tag(m, ix);
if (tmp_tag.unary) {
if (tmp_tag.start < start_ix && tmp_tag.end > start_ix) // exact match
return saveMatch(tmp_tag, null, start_ix);
} else if (backward_stack.last() && backward_stack.last().name == tmp_tag.name) {
backward_stack.pop();
} else { // found nearest unclosed tag
opening_tag = tmp_tag;
break;
}
} else if (check_str.indexOf('<!--') == 0) { // found comment start
var end_ix = check_str.search('-->') + ix + 3;
if (ix < start_ix && end_ix >= start_ix)
return saveMatch( comment(ix, end_ix) );
}
} else if (ch == '-' && hasMatch('-->')) { // found comment end
// search left until comment start is reached
ix = searchCommentStart(ix);
}
}
if (!opening_tag)
return action(null);
// find closing tag
if (!closing_tag) {
for (ix = start_ix; ix < html_len; ix++) {
var ch = html.charAt(ix);
if (ch == '<') {
var check_str = html.substring(ix, html_len);
if ( (m = check_str.match(start_tag)) ) { // found opening tag
tmp_tag = tag(m, ix);
if (!tmp_tag.unary)
forward_stack.push( tmp_tag );
} else if ( (m = check_str.match(end_tag)) ) { // found closing tag
var tmp_tag = tag(m, ix);
if (forward_stack.last() && forward_stack.last().name == tmp_tag.name)
forward_stack.pop();
else { // found matched closing tag
closing_tag = tmp_tag;
break;
}
} else if (hasMatch('<!--')) { // found comment
ix += check_str.search('-->') + 3;
}
} else if (ch == '-' && hasMatch('-->')) {
// looks like cursor was inside comment with invalid HTML
if (!forward_stack.last() || forward_stack.last().type != 'comment') {
var end_ix = ix + 3;
return action(comment( searchCommentStart(ix), end_ix ));
}
}
}
}
return action(opening_tag, closing_tag, start_ix);
}
/**
* Search for matching tags in <code>html</code>, starting
* from <code>start_ix</code> position. The result is automatically saved in
* <code>last_match</code> property
*
* @return {Array|null}
*/
var HTMLPairMatcher = this.HTMLPairMatcher = function(/* String */ html, /* Number */ start_ix){
return findPair(html, start_ix, saveMatch);
}
HTMLPairMatcher.start_tag = start_tag;
HTMLPairMatcher.end_tag = end_tag;
/**
* Search for matching tags in <code>html</code>, starting from
* <code>start_ix</code> position. The difference between
* <code>HTMLPairMatcher</code> function itself is that <code>find</code>
* method doesn't save matched result in <code>last_match</code> property.
* This method is generally used for lookups
*/
HTMLPairMatcher.find = function(html, start_ix) {
return findPair(html, start_ix);
};
/**
* Search for matching tags in <code>html</code>, starting from
* <code>start_ix</code> position. The difference between
* <code>HTMLPairMatcher</code> function itself is that <code>getTags</code>
* method doesn't save matched result in <code>last_match</code> property
* and returns array of opening and closing tags
* This method is generally used for lookups
*/
HTMLPairMatcher.getTags = function(html, start_ix) {
return findPair(html, start_ix, function(opening_tag, closing_tag){
return [opening_tag, closing_tag];
});
};
HTMLPairMatcher.last_match = last_match;
})();/**
* @author Sergey Chikuyonok (serge.che@gmail.com)
* @link http://chikuyonok.ru
* @include "settings.js"
* @include "/EclipseMonkey/scripts/monkey-doc.js"
*/ var zen_coding = (function(){
var re_tag = /<\/?[\w:\-]+(?:\s+[\w\-:]+(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*\s*(\/?)>$/;
var TYPE_ABBREVIATION = 'zen-tag',
TYPE_EXPANDO = 'zen-expando',
/** Reference to another abbreviation or tag */
TYPE_REFERENCE = 'zen-reference',
content_placeholder = '{%::zen-content::%}',
newline = '\n';
var default_profile = {
tag_case: 'lower',
attr_case: 'lower',
attr_quotes: 'double',
// each tag on new line
tag_nl: 'decide',
place_cursor: true,
// indent tags
indent: true,
// use self-closing style for writing empty elements, e.g. <br /> or <br>
self_closing_tag: 'xhtml'
};
var profiles = {};
/**
* Проверяет, является ли символ допустимым в аббревиатуре
* @param {String} ch
* @return {Boolean}
*/
function isAllowedChar(ch) {
var char_code = ch.charCodeAt(0),
special_chars = '#.>+*:$-_!@';
return (char_code > 64 && char_code < 91) // uppercase letter
|| (char_code > 96 && char_code < 123) // lowercase letter
|| (char_code > 47 && char_code < 58) // number
|| special_chars.indexOf(ch) != -1; // special character
}
/**
* Возвращает символ перевода строки, используемый в редакторе
* @return {String}
*/
function getNewline() {
return zen_coding.getNewline();
}
/**
* Split text into lines. Set <code>remove_empty</code> to true to filter
* empty lines
* @param {String} text
* @param {Boolean} [remove_empty]
* @return {Array}
*/
function splitByLines(text, remove_empty) {
// IE fails to split string by regexp,
// need to normalize newlines first
var lines = text.replace(/\r\n/g, '\n').replace(/\n\r/g, '\n').split('\n');
// var nl = getNewline(),
// lines = text.split(new RegExp('\\r?\\n|\\n\\r|\\r|' + nl));
if (remove_empty) {
for (var i = lines.length; i >= 0; i--) {
if (!trim(lines[i]))
lines.splice(i, 1);
}
}
return lines;
}
/**
* Trim whitespace from string
* @param {String} text
* @return {String}
*/
function trim(text) {
return (text || "").replace( /^\s+|\s+$/g, "" );
}
function createProfile(options) {
var result = {};
for (var p in default_profile)
result[p] = (p in options) ? options[p] : default_profile[p];
return result;
}
function setupProfile(name, options) {
profiles[name.toLowerCase()] = createProfile(options || {});
}
/**
* Helper function that transforms string into hash
* @return {Object}
*/
function stringToHash(str){
var obj = {}, items = str.split(",");
for ( var i = 0; i < items.length; i++ )
obj[ items[i] ] = true;
return obj;
}
/**
* Отбивает текст отступами
* @param {String} text Текст, который нужно отбить
* @param {String|Number} pad Количество отступов или сам отступ
* @return {String}
*/
function padString(text, pad, verbose) {
var pad_str = '', result = '';
if (typeof(pad) == 'number')
for (var i = 0; i < pad; i++)
pad_str += zen_settings.variables.indentation;
else
pad_str = pad;
var lines = splitByLines(text),
nl = getNewline();
result += lines[0];
for (var j = 1; j < lines.length; j++)
result += nl + pad_str + lines[j];
return result;
}
/**
* Check if passed abbreviation is snippet
* @param {String} abbr
* @param {String} type
* @return {Boolean}
*/
function isShippet(abbr, type) {
return getSnippet(type, abbr) ? true : false;
}
/**
* Проверяет, закачивается ли строка полноценным тэгом. В основном
* используется для проверки принадлежности символа '>' аббревиатуре
* или тэгу
* @param {String} str
* @return {Boolean}
*/
function isEndsWithTag(str) {
return re_tag.test(str);
}
/**
* Returns specified elements collection (like 'empty', 'block_level') from
* <code>resource</code>. If collections wasn't found, returns empty object
* @param {Object} resource
* @param {String} type
* @return {Object}
*/
function getElementsCollection(resource, type) {
if (resource && resource.element_types)
return resource.element_types[type] || {}
else
return {};
}
/**
* Replace variables like ${var} in string
* @param {String} str
* @param {Object} [vars] Variable set (default is <code>zen_settings.variables</code>)
* @return {String}
*/
function replaceVariables(str, vars) {
vars = vars || zen_settings.variables;
return str.replace(/\$\{([\w\-]+)\}/g, function(str, p1){
return (p1 in vars) ? vars[p1] : str;
});
}
/**
* Тэг
* @class
* @param {String} name Имя тэга
* @param {Number} count Сколько раз вывести тэг (по умолчанию: 1)
* @param {String} type Тип тэга (html, xml)
*/
function Tag(name, count, type) {
name = name.toLowerCase();
type = type || 'html';
var abbr = getAbbreviation(type, name);
if (abbr && abbr.type == TYPE_REFERENCE)
abbr = getAbbreviation(type, abbr.value);
this.name = (abbr) ? abbr.value.name : name.replace('+', '');
this.count = count || 1;
this.children = [];
this.attributes = [];
this._attr_hash = {};
this._abbr = abbr;
this._res = zen_settings[type];
this._content = '';
this.repeat_by_lines = false;
// add default attributes
if (this._abbr && this._abbr.value.attributes) {
var def_attrs = this._abbr.value.attributes; if (def_attrs) {
for (var i = 0; i < def_attrs.length; i++) {
var attr = def_attrs[i];
this.addAttribute(attr.name, attr.value);
}
}
}
}
Tag.prototype = {
/**
* Добавляет нового потомка
* @param {Tag} tag
*/
addChild: function(tag) {
this.children.push(tag);
},
/**
* Добавляет атрибут
* @param {String} name Название атрибута
* @param {String} value Значение атрибута
*/
addAttribute: function(name, value) {
var a;
if (name in this._attr_hash) {
// attribute already exists, decide what to do
a = this._attr_hash[name];
if (name == 'class') {
// 'class' is a magic attribute
a.value += ((a.value) ? ' ' : '') + value;
} else {
a.value = value;
}
} else {
a = {name: name, value: value};
this._attr_hash[name] = a
this.attributes.push(a);
}
},
/**
* Проверяет, является ли текущий элемент пустым
* @return {Boolean}
*/
isEmpty: function() {
return (this._abbr && this._abbr.value.is_empty) || (this.name in getElementsCollection(this._res, 'empty'));
},
/**
* Проверяет, является ли текущий элемент строчным
* @return {Boolean}
*/
isInline: function() {
return (this.name in getElementsCollection(this._res, 'inline_level'));
},
/**
* Проверяет, является ли текущий элемент блочным
* @return {Boolean}
*/
isBlock: function() {
return (this.name in getElementsCollection(this._res, 'block_level'));
},
/**
* This function tests if current tags' content contains xHTML tags.
* This function is mostly used for output formatting
*/
hasTagsInContent: function() {
return this.getContent() && re_tag.test(this.getContent());
},
/**
* Проверяет, есть ли блочные потомки у текущего тэга.
* Используется для форматирования
* @return {Boolean}
*/
hasBlockChildren: function() {
if (this.hasTagsInContent() && this.isBlock()) {
return true;
}
for (var i = 0; i < this.children.length; i++) {
if (this.children[i].isBlock())
return true;
}
return false;
},
/**
* Set textual content for tag
* @param {String} str Tag's content
*/
setContent: function(str) {
this._content = str;
},
/**
* Returns tag's textual content
* @return {String}
*/
getContent: function() {
return this._content;
},
/**
* Search for deepest and latest child of current element
* @return {Tag|null} Returns null if there's no children
*/
findDeepestChild: function() {
if (!this.children.length)
return null;
var deepest_child = this;
while (true) {
deepest_child = deepest_child.children[ deepest_child.children.length - 1 ];
if (!deepest_child.children.length)
break;
}
return deepest_child;
},
/**
* Transforms and formats tag into string using profile
* @param {String} profile Profile name
* @return {String}
* TODO Function is too large, need refactoring
*/
toString: function(profile_name) {
var result = [],
profile = (profile_name in profiles) ? profiles[profile_name] : profiles['plain'],
attrs = '',
content = '',
start_tag = '',
end_tag = '',
cursor = profile.place_cursor ? '|' : '',
self_closing = '',
attr_quote = profile.attr_quotes == 'single' ? "'" : '"',
attr_name,
is_empty = (this.isEmpty() && !this.children.length);
if (profile.self_closing_tag == 'xhtml')
self_closing = ' /';
else if (profile.self_closing_tag === true)
self_closing = '/';
function allowNewline(tag) {
return (profile.tag_nl === true || (profile.tag_nl == 'decide' && tag.isBlock()))
}
// make attribute string
for (var i = 0; i < this.attributes.length; i++) {
var a = this.attributes[i];
attr_name = (profile.attr_case == 'upper') ? a.name.toUpperCase() : a.name.toLowerCase();
attrs += ' ' + attr_name + '=' + attr_quote + (a.value || cursor) + attr_quote;
}
var deepest_child = this.findDeepestChild();
// output children
if (!is_empty) {
if (deepest_child && this.repeat_by_lines)
deepest_child.setContent(content_placeholder);
for (var j = 0; j < this.children.length; j++) {
//
content += this.children[j].toString(profile_name);
if (
(j != this.children.length - 1) &&
( allowNewline(this.children[j]) || allowNewline(this.children[j + 1]) )
)
content += getNewline();
}
}
// define opening and closing tags
if (this.name) {
var tag_name = (profile.tag_case == 'upper') ? this.name.toUpperCase() : this.name.toLowerCase();
if (is_empty) {
start_tag = '<' + tag_name + attrs + self_closing + '>';
} else {
start_tag = '<' + tag_name + attrs + '>';
end_tag = '</' + tag_name + '>';
}
}
// formatting output
if (profile.tag_nl !== false) {
if (
this.name &&
(
profile.tag_nl === true ||
this.hasBlockChildren()
)
) {
if (end_tag) { // non-empty tag: add indentation
start_tag += getNewline() + zen_settings.variables.indentation;
end_tag = getNewline() + end_tag;
} else { // empty tag
}
}
if (this.name) {
if (content)
content = padString(content, profile.indent ? 1 : 0);
else if (!is_empty)
start_tag += cursor;
}
}
// repeat tag by lines count
var cur_content = '';
if (this.repeat_by_lines) {
var lines = splitByLines( trim(this.getContent()) , true);
for (var j = 0; j < lines.length; j++) {
cur_content = deepest_child ? '' : content_placeholder;
if (content && !deepest_child)
cur_content += getNewline();
var elem_str = start_tag.replace(/\$/g, j + 1) + cur_content + content + end_tag;
result.push(elem_str.replace(content_placeholder, trim(lines[j])));
}
}
// repeat tag output
if (!result.length) {
if (this.getContent()) {
var pad = (profile.tag_nl === true || (this.hasTagsInContent() && this.isBlock())) ? 1 : 0;
content = padString(this.getContent(), pad) + content;
}
for (var i = 0; i < this.count; i++)
result.push(start_tag.replace(/\$/g, i + 1) + content + end_tag);
}
var glue = '';
if (allowNewline(this))
glue = getNewline();
return result.join(glue);
}
};
// TODO inherit from Tag
function Snippet(name, count, type) {
/** @type {String} */
this.name = name;
this.count = count || 1;
this.children = [];
this._content = '';
this.repeat_by_lines = false;
this.attributes = {'id': '|', 'class': '|'};
this.value = getSnippet(type, name);
}
Snippet.prototype = {
/**
* Добавляет нового потомка
* @param {Tag} tag
*/
addChild: function(tag) {
this.children.push(tag);
},
addAttribute: function(name, value){
this.attributes[name] = value;
},
isBlock: function() {
return true;
},
/**
* Set textual content for snippet
* @param {String} str Tag's content
*/
setContent: function(str) {
this._content = str;
},
/**
* Returns snippet's textual content
* @return {String}
*/
getContent: function() {
return this._content;
},
/**
* Search for deepest and latest child of current element
* @return {Tag|null} Returns null if there's no children
*/
findDeepestChild: function() {
if (!this.children.length)
return null;
var deepest_child = this;
while (true) {
deepest_child = deepest_child.children[ deepest_child.children.length - 1 ];
if (!deepest_child.children.length)
break;
}
return deepest_child;
},
toString: function(profile_name) {
var content = '',
profile = (profile_name in profiles) ? profiles[profile_name] : profiles['plain'],
result = [],
data = this.value,
begin = '',
end = '',
child_padding = '',
child_token = '${child}';
if (data) {
if (profile.tag_nl !== false) {
var nl = getNewline();
data = data.replace(/\n/g, nl);
// figuring out indentation for children
var lines = data.split(nl), m;
for (var j = 0; j < lines.length; j++) {
if (lines[j].indexOf(child_token) != -1) {
child_padding = (m = lines[j].match(/(^\s+)/)) ? m[1] : '';
break;
}
}
}
var parts = data.split(child_token);
begin = parts[0] || '';
end = parts[1] || '';
}
for (var i = 0; i < this.children.length; i++) {
content += this.children[i].toString(profile_name);
if (
i != this.children.length - 1 &&
(
profile.tag_nl === true ||
(profile.tag_nl == 'decide' && this.children[i].isBlock())
)
)
content += getNewline();
}
if (child_padding)
content = padString(content, child_padding);
// substitute attributes
begin = replaceVariables(begin, this.attributes);
end = replaceVariables(end, this.attributes);
if (this.getContent()) {
content = padString(this.getContent(), 1) + content;
}
// выводим тэг нужное количество раз
for (var i = 0; i < this.count; i++)
result.push(begin + content + end);
// result.push(begin.replace(/\$(?!\{)/g, i + 1) + content + end);
return result.join((profile.tag_nl !== false) ? getNewline() : '');
}
}
/**
* Returns abbreviation value from data set
* @param {String} type Resource type (html, css, ...)
* @param {String} abbr Abbreviation name
* @return {Object|null}
*/
function getAbbreviation(type, abbr) {
return getSettingsResource(type, abbr, 'abbreviations');
}
/**
* Returns snippet value from data set
* @param {String} type Resource type (html, css, ...)
* @param {String} snippet_name Snippet name
* @return {Object|null}
*/
function getSnippet(type, snippet_name) {
return getSettingsResource(type, snippet_name, 'snippets');
}
/**
* Returns resurce value from data set with respect of inheritance
* @param {String} type Resource type (html, css, ...)
* @param {String} abbr Abbreviation name
* @param {String} res_name Resource name ('snippets' or 'abbreviation')
* @return {Object|null}
*/
function getSettingsResource(type, abbr, res_name) {
var resource = zen_settings[type];
if (resource) {
if (res_name in resource && abbr in resource[res_name])
return resource[res_name][abbr];
else if ('extends' in resource) {
// find abbreviation in ancestors
for (var i = 0; i < resource['extends'].length; i++) {
var type = resource['extends'][i];
if (
zen_settings[type] &&
zen_settings[type][res_name] &&
zen_settings[type][res_name][abbr]
)
return zen_settings[type][res_name][abbr];
}
}
}
return null;
}
// create default profiles
setupProfile('xhtml');
setupProfile('html', {self_closing_tag: false});
setupProfile('xml', {self_closing_tag: true, tag_nl: true});
setupProfile('plain', {tag_nl: false, indent: false, place_cursor: false});
return {
expandAbbreviation: function(abbr, type, profile) {
var tree = this.parseIntoTree(abbr, type || 'html');
return replaceVariables(tree ? tree.toString(profile) : '');
},
/**
* Extracts abbreviations from text stream, starting from the end
* @param {String} str
* @return {String} Abbreviation or empty string
*/
extractAbbreviation: function(str) {
var cur_offset = str.length,
start_index = -1;
while (true) {
cur_offset--;
if (cur_offset < 0) {
// дошли до начала строки
start_index = 0;
break;
}
var ch = str.charAt(cur_offset);
if (!isAllowedChar(ch) || (ch == '>' && isEndsWithTag(str.substring(0, cur_offset + 1)))) {
start_index = cur_offset + 1;
break;
}
}
if (start_index != -1)
// что-то нашли, возвращаем аббревиатуру
return str.substring(start_index);
else
return '';
},
/**
* Parses abbreviation into a node set
* @param {String} abbr Abbreviation
* @param {String} type Document type (xsl, html, etc.)
* @return {Tag}
*/
parseIntoTree: function(abbr, type) {
type = type || 'html';
var root = new Tag('', 1, type),
parent = root,
last = null,
multiply_elem = null,
res = zen_settings[type],
re = /([\+>])?([a-z@\!][a-z0-9:\-]*)(#[\w\-\$]+)?((?:\.[\w\-\$]+)*)(\*(\d*))?(\+$)?/ig;
if (!abbr)
return null;
// replace expandos
abbr = abbr.replace(/([a-z][\w\:\-]*)\+$/i, function(str){
var a = getAbbreviation(type, str);
return a ? a.value : str;
});
abbr = abbr.replace(re, function(str, operator, tag_name, id, class_name, has_multiplier, multiplier, has_expando){
var multiply_by_lines = (has_multiplier && !multiplier);
multiplier = multiplier ? parseInt(multiplier) : 1;
if (has_expando)
tag_name += '+';
var current = isShippet(tag_name, type) ? new Snippet(tag_name, multiplier, type) : new Tag(tag_name, multiplier, type);
if (id)
current.addAttribute('id', id.substr(1));
if (class_name)
current.addAttribute('class', class_name.substr(1).replace(/\./g, ' '));
// dive into tree
if (operator == '>' && last)
parent = last;
parent.addChild(current);
last = current;
if (multiply_by_lines)
multiply_elem = current;
return '';
});
root.last = last;
root.multiply_elem = multiply_elem;
// empty 'abbr' string means that abbreviation was successfully expanded,
// if not — abbreviation wasn't valid
return (!abbr) ? root : null;
},
/**
* Отбивает текст отступами
* @param {String} text Текст, который нужно отбить
* @param {String|Number} pad Количество отступов или сам отступ
* @return {String}
*/
padString: padString,
setupProfile: setupProfile,
getNewline: function(){
return newline;
},
setNewline: function(str) {
newline = str;
},
/**
* Returns range for matched tag pair inside document
* @requires HTMLParser
* @param {String} html Full xHTML document
* @param {Number} cursor_pos Cursor position inside document
* @return {Object} Pair of indicies (<code>start</code> and <code>end</code>).
* Returns 'null' if match wasn't found
*/
getPairRange: function(html, cursor_pos) {
var tags = {},
ranges = [],
result = null;
function inRange(start, end) {
return cursor_pos > start && cursor_pos < end;
}
var handler = {
start: function(name, attrs, unary, ix_start, ix_end) {
if (unary && inRange(ix_start, ix_end)) {
// this is the exact range for cursor position, stop searching
result = {start: ix_start, end: ix_end};
this.stop = true;
} else {
if (!tags.hasOwnProperty(name))
tags[name] = [];
tags[name].push(ix_start);
}
},
end: function(name, ix_start, ix_end) {
if (tags.hasOwnProperty(name)) {
var start = tags[name].pop();
if (inRange(start, ix_end))
ranges.push({start: start, end: ix_end});
}
},
comment: function(data, ix_start, ix_end) {
if (inRange(ix_start, ix_end)) {
// this is the exact range for cursor position, stop searching
result = {start: ix_start, end: ix_end};
this.stop = true;
}
}
};
// scan document
try {
HTMLParser(html, handler);
} catch(e) {}
if (!result && ranges.length) {
// because we have overlaped ranges only, we have to sort array by
// length: the shorter range length, the most probable match
result = ranges.sort(function(a, b){
return (a.end - a.start) - (b.end - b.start);
})[0];
}
return result;
},
/**
* Wraps passed text with abbreviation. Text will be placed inside last
* expanded element
* @param {String} abbr Abbreviation
* @param {String} text Text to wrap
* @param {String} [type] Document type (html, xml, etc.). Default is 'html'
* @param {String} [profile] Output profile's name. Default is 'plain'
* @return {String}
*/
wrapWithAbbreviation: function(abbr, text, type, profile) {
var tree = this.parseIntoTree(abbr, type || 'html');
if (tree) {
var repeat_elem = tree.multiply_elem || tree.last;
repeat_elem.setContent(text);
repeat_elem.repeat_by_lines = !!tree.multiply_elem;
return tree.toString(profile);
} else {
return null;
}
},
splitByLines: splitByLines,
/**
* Check if cursor is placed inside xHTML tag
* @param {String} html Contents of the document
* @param {Number} cursor_pos Current caret position inside tag
* @return {Boolean}
*/
isInsideTag: function(html, cursor_pos) {
var re_tag = /^<\/?\w[\w\:\-]*.*?>/;
// search left to find opening brace
var pos = cursor_pos;
while (pos > -1) {
if (html.charAt(pos) == '<')
break;
pos--;
}
if (pos != -1) {
var m = re_tag.exec(html.substring(pos));
if (m && cursor_pos > pos && cursor_pos < pos + m[0].length)
return true;
}
return false;
},
settings_parser: (function(){
/**
* Unified object for parsed data
*/
function entry(type, key, value) {
return {
type: type,
key: key,
value: value
};
}
/** Regular expression for XML tag matching */
var re_tag = /^<(\w+\:?[\w\-]*)((?:\s+[\w\:\-]+\s*=\s*(['"]).*?\3)*)\s*(\/?)>/,
re_attrs = /([\w\-]+)\s*=\s*(['"])(.*?)\2/g;
/**
* Make expando from string
* @param {String} key
* @param {String} value
* @return {Object}
*/
function makeExpando(key, value) {
return entry(TYPE_EXPANDO, key, value);
}
/**
* Make abbreviation from string
* @param {String} key Abbreviation key
* @param {String} tag_name Expanded element's tag name
* @param {String} attrs Expanded element's attributes
* @param {Boolean} is_empty Is expanded element empty or not
* @return {Object}
*/
function makeAbbreviation(key, tag_name, attrs, is_empty) {
var result = {
name: tag_name,
is_empty: Boolean(is_empty)
};
if (attrs) {
var m;
result.attributes = [];
while (m = re_attrs.exec(attrs)) {
result.attributes.push({
name: m[1],
value: m[3]
});
}
}
return entry(TYPE_ABBREVIATION, key, result);
}
/**
* Parses all abbreviations inside object
* @param {Object} obj
*/
function parseAbbreviations(obj) {
for (var key in obj) {
var value = obj[key], m;
key = trim(key);
if (key.substr(-1) == '+') {
// this is expando, leave 'value' as is
obj[key] = makeExpando(key, value);
} else if (m = re_tag.exec(value)) {
obj[key] = makeAbbreviation(key, m[1], m[2], m[4] == '/');
} else {
// assume it's reference to another abbreviation
obj[key] = entry(TYPE_REFERENCE, key, value);
}
}
}
return {
/**
* Parse user's settings
* @param {Object} settings
*/
parse: function(settings) {
for (var p in settings) {
if (p == 'abbreviations')
parseAbbreviations(settings[p]);
else if (p == 'extends') {
var ar = settings[p].split(',');
for (var i = 0; i < ar.length; i++)
ar[i] = trim(ar[i]);
settings[p] = ar;
}
else if (typeof(settings[p]) == 'object')
arguments.callee(settings[p]);
}
},
extend: function(parent, child) {
for (var p in child) {
if (typeof(child[p]) == 'object' && parent.hasOwnProperty(p))
arguments.callee(parent[p], child[p]);
else
parent[p] = child[p];
}
},
/**
* Create hash maps on certain string properties
* @param {Object} obj
*/
createMaps: function(obj) {
for (var p in obj) {
if (p == 'element_types') {
for (var k in obj[p])
obj[p][k] = stringToHash(obj[p][k]);
} else if (typeof(obj[p]) == 'object') {
arguments.callee(obj[p]);
}
}
},
TYPE_ABBREVIATION: TYPE_ABBREVIATION,
TYPE_EXPANDO: TYPE_EXPANDO,
/** Reference to another abbreviation or tag */
TYPE_REFERENCE: TYPE_REFERENCE
}
})()
}
})();
if ('zen_settings' in this || zen_settings) {
// first we need to expand some strings into hashes
zen_coding.settings_parser.createMaps(zen_settings);
if ('my_zen_settings' in this) {
// we need to extend default settings with user's
zen_coding.settings_parser.createMaps(my_zen_settings);
zen_coding.settings_parser.extend(zen_settings, my_zen_settings);
}
// now we need to parse final set of settings
zen_coding.settings_parser.parse(zen_settings);
}/**
* High-level editor interface which communicates with other editor (like
* TinyMCE, CKEditor, etc.) or browser.
* Before using any of editor's methods you should initialize it with
* <code>editor.setTarget(elem)</code> method and pass reference to
* &lt;textarea&gt; element.
* @example
* var textarea = document.getElemenetsByTagName('textarea')[0];
* editor.setTarget(textarea);
* //now you are ready to use editor object
* editor.getSelectionRange()
*
* @author Sergey Chikuyonok (serge.che@gmail.com)
* @link http://chikuyonok.ru
* @include "../../aptana/lib/zen_coding.js"
*/
var zen_editor = (function(){
/** @param {Element} Source element */
var target = null,
/** Textual placeholder that identifies cursor position in pasted text */
caret_placeholder = '|';
// different browser uses different newlines, so we have to figure out
// native browser newline and sanitize incoming text with them
var tx = document.createElement('textarea');
tx.value = '\n';
zen_coding.setNewline(tx.value);
tx = null;
/**
* Returns content of current target element
*/
function getContent() {
return target.value || '';
}
/**
* Returns selection indexes from element
*/
function getSelectionRange() {
if ('selectionStart' in target) { // W3C's DOM
var length = target.selectionEnd - target.selectionStart;
return {
start: target.selectionStart,
end: target.selectionEnd
};
} else if (document.selection) { // IE
target.focus();
var range = document.selection.createRange();
if (range === null) {
return {
start: 0,
end: getContent().length
};
}
var re = target.createTextRange();
var rc = re.duplicate();
re.moveToBookmark(range.getBookmark());
rc.setEndPoint('EndToStart', re);
return {
start: rc.text.length,
end: rc.text.length + range.text.length
};
} else {
return null;
}
}
/**
* Creates text selection on target element
* @param {Number} start
* @param {Number} end
*/
function createSelection(start, end) {
// W3C's DOM
if (typeof(end) == 'undefined')
end = start;
if ('setSelectionRange' in target) {
target.setSelectionRange(start, end);
} else if ('createTextRange' in target) {
var t = target.createTextRange();
t.collapse(true);
var delta = zen_coding.splitByLines(getContent().substring(0, start)).length - 1;
// IE has an issue with handling newlines while creating selection,
// so we need to adjust start and end indexes
end -= delta + zen_coding.splitByLines(getContent().substring(start, end)).length - 1;
start -= delta;
t.moveStart('character', start);
t.moveEnd('character', end - start);
t.select();
}
}
/**
* Find start and end index of text line for <code>from</code> index
* @param {String} text
* @param {Number} from
*/
function findNewlineBounds(text, from) {
var len = text.length,
start = 0,
end = len - 1;
// search left
for (var i = from - 1; i > 0; i--) {
var ch = text.charAt(i);
if (ch == '\n' || ch == '\r') {
start = i + 1;
break;
}
}
// search right
for (var j = from; j < len; j++) {
var ch = text.charAt(j);
if (ch == '\n' || ch == '\r') {
end = j;
break;
}
}
return {start: start, end: end};
}
/**
* Returns current caret position
*/
function getCaretPos() {
var selection = getSelectionRange();
return selection ? selection.start : null;
}
/**
* Returns whitrespace padding of string
* @param {String} str String line
* @return {String}
*/
function getStringPadding(str) {
return (str.match(/^(\s+)/) || [''])[0];
}
return {
setTarget: function(elem) {
target = elem;
},
getSelectionRange: getSelectionRange,
createSelection: createSelection,
/**
* Returns current line's start and end indexes
*/
getCurrentLineRange: function() {
var caret_pos = getCaretPos(),
content = getContent();
if (caret_pos === null) return null;
return findNewlineBounds(content, caret_pos);
},
/**
* Returns current caret position
* @return {Number}
*/
getCaretPos: getCaretPos,
/**
* Returns content of current line
* @return {String}
*/
getCurrentLine: function() {
var range = this.getCurrentLineRange();
return range.start < range.end ? this.getContent().substring(range.start, range.end) : '';
},
/**
* Replace editor's content or it's part (from <code>start</code> to
* <code>end</code> index). If <code>value</code> contains
* <code>caret_placeholder</code>, the editor will put caret into
* this position. If you skip <code>start</code> and <code>end</code>
* arguments, the whole target's content will be replaced with
* <code>value</code>.
*
* If you pass <code>start</code> argument only,
* the <code>value</code> will be placed at <code>start</code> string
* index of current content.
*
* If you pass <code>start</code> and <code>end</code> arguments,
* the corresponding substring of current target's content will be
* replaced with <code>value</code>.
* @param {String} value Content you want to paste
* @param {Number} [start] Start index of editor's content
* @param {Number} [end] End index of editor's content
*/
replaceContent: function(value, start, end) {
var content = getContent(),
caret_pos = getCaretPos(),
has_start = typeof(start) !== 'undefined',
has_end = typeof(end) !== 'undefined';
// indent new value
value = zen_coding.padString(value, getStringPadding(this.getCurrentLine()));
// find new caret position
var new_pos = value.indexOf(caret_placeholder);
if (new_pos != -1) {
caret_pos = (start || 0) + new_pos;
value = value.split(caret_placeholder).join('');
} else {
caret_pos += value.length;
}
try {
if (has_start && has_end) {
content = content.substring(0, start) + value + content.substring(end);
} else if (has_start) {
content = content.substring(0, start) + value + content.substring(start);
}
target.value = content;
createSelection(caret_pos, caret_pos);
} catch(e){}
},
/**
* Returns editor's content
* @return {String}
*/
getContent: getContent
}
})();
/**
* Middleware layer that communicates between editor and Zen Coding.
* This layer describes all available Zen Coding actions, like
* "Expand Abbreviation".
* @author Sergey Chikuyonok (serge.che@gmail.com)
* @link http://chikuyonok.ru
*
* @include "editor.js"
* @include "../../aptana/lib/html_matcher.js"
* @include "../../aptana/lib/zen_coding.js"
*/
/**
* Search for abbreviation in editor from current caret position
* @param {zen_editor} editor Editor instance
* @return {String|null}
*/
function findAbbreviation(editor) {
var range = editor.getSelectionRange();
if (range.start != range.end) {
// abbreviation is selected by user
return editor.getContent().substring(range.start, range.end);
}
// search for new abbreviation from current caret position
var cur_line = editor.getCurrentLineRange();
return zen_coding.extractAbbreviation(editor.getContent().substring(cur_line.start, range.start));
}
/**
* Find from current caret position and expand abbreviation in editor
* @param {zen_editor} editor Editor instance
* @param {String} type Syntax type (html, css, etc.)
* @param {String} profile_name Output profile name (html, xml, xhtml)
* @return {Boolean} Returns <code>true</code> if abbreviation was expanded
* successfully
*/
function expandAbbreviation(editor, type, profile_name) {
profile_name = profile_name || 'xhtml';
var caret_pos = editor.getSelectionRange().end,
abbr,
content = '';
if ( (abbr = findAbbreviation(editor)) ) {
content = zen_coding.expandAbbreviation(abbr, type, profile_name);
if (content) {
editor.replaceContent(content, caret_pos - abbr.length, caret_pos);
return true;
}
}
return false;
}
/**
* A special version of <code>expandAbbreviation</code> function: if it can't
* find abbreviation, it will place Tab character at caret position
* @param {zen_editor} editor Editor instance
* @param {String} type Syntax type (html, css, etc.)
* @param {String} profile_name Output profile name (html, xml, xhtml)
*/
function expandAbbreviationWithTab(editor, type, profile_name) {
if (!expandAbbreviation(editor, type, profile_name))
editor.replaceContent('\t', editor.getCaretPos());
}
/**
* Find and select HTML tag pair
* @param {zen_editor} editor Editor instance
* @param {String} [direction] Direction of pair matching: 'in' or 'out'.
* Default is 'out'
*/
function matchPair(editor, direction) {
direction = (direction || 'out').toLowerCase();
var range = editor.getSelectionRange(),
cursor = range.end,
range_start = range.start,
range_end = range.end,
// content = zen_coding.splitByLines(editor.getContent()).join('\n'),
content = editor.getContent(),
range = null,
_r,
old_open_tag = HTMLPairMatcher.last_match['opening_tag'],
old_close_tag = HTMLPairMatcher.last_match['closing_tag'];
if (direction == 'in' && old_open_tag && range_start != range_end) {
// user has previously selected tag and wants to move inward
if (!old_close_tag) {
// unary tag was selected, can't move inward
return false;
} else if (old_open_tag.start == range_start) {
if (content[old_open_tag.end] == '<') {
// test if the first inward tag matches the entire parent tag's content
_r = HTMLPairMatcher.find(content, old_open_tag.end + 1);
if (_r[0] == old_open_tag.end && _r[1] == old_close_tag.start) {
range = HTMLPairMatcher(content, old_open_tag.end + 1);
} else {
range = [old_open_tag.end, old_close_tag.start];
}
} else {
range = [old_open_tag.end, old_close_tag.start];
}
} else {
var new_cursor = content.substring(0, old_close_tag.start).indexOf('<', old_open_tag.end);
var search_pos = new_cursor != -1 ? new_cursor + 1 : old_open_tag.end;
range = HTMLPairMatcher(content, search_pos);
}
} else {
range = HTMLPairMatcher(content, cursor);
}
if (range !== null && range[0] != -1) {
// alert(range[0] + ', '+ range[1]);
editor.createSelection(range[0], range[1]);
return true;
} else {
return false;
}
}
/**
* Wraps content with abbreviation
* @param {zen_editor} Editor instance
* @param {String} type Syntax type (html, css, etc.)
* @param {String} profile_name Output profile name (html, xml, xhtml)
*/
function wrapWithAbbreviation(editor, abbr, type, profile_name) {
profile_name = profile_name || 'xhtml';
var range = editor.getSelectionRange(),
start_offset = range.start,
end_offset = range.end,
content = editor.getContent();
if (!abbr)
return null;
if (start_offset == end_offset) {
// no selection, find tag pair
range = HTMLPairMatcher(content, start_offset);
if (!range || range[0] == -1) // nothing to wrap
return null;
start_offset = range[0];
end_offset = range[1];
// narrow down selection until first non-space character
var re_space = /\s|\n|\r/;
function isSpace(ch) {
return re_space.test(ch);
}
while (start_offset < end_offset) {
if (!isSpace(content.charAt(start_offset)))
break;
start_offset++;
}
while (end_offset > start_offset) {
end_offset--;
if (!isSpace(content.charAt(end_offset))) {
end_offset++;
break;
}
}
}
var new_content = content.substring(start_offset, end_offset),
result = zen_coding.wrapWithAbbreviation(abbr, unindent(editor, new_content), type, profile_name);
if (result) {
editor.createSelection(end_offset);
editor.replaceContent(result, start_offset, end_offset);
}
}
/**
* Unindent content, thus preparing text for tag wrapping
* @param {zen_editor} Editor instance
* @param {String} text
* @return {String}
*/
function unindent(editor, text) {
var pad = getCurrentLinePadding(editor);
var lines = zen_coding.splitByLines(text);
for (var i = 0; i < lines.length; i++) {
if (lines[i].search(pad) == 0)
lines[i] = lines[i].substr(pad.length);
}
return lines.join(zen_coding.getNewline());
}
/**
* Returns padding of current editor's line
* @param {zen_editor} Editor instance
* @return {String}
*/
function getCurrentLinePadding(editor) {
return (editor.getCurrentLine().match(/^(\s+)/) || [''])[0];
}
/**
* Search for new caret insertion point
* @param {zen_editor} editor Editor instance
* @param {Number} Search increment: -1 search left, 1 search right
* @param {Number} Initial offset relative to current caret position
* @return {Number} Returns -1 if insertion point wasn't found
*/
function findNewEditPoint(editor, inc, offset) {
inc = inc || 1;
offset = offset || 0;
var cur_point = editor.getCaretPos() + offset,
content = editor.getContent(),
max_len = content.length,
next_point = -1,
re_empty_line = /^\s+$/;
function ch(ix) {
return content.charAt(ix);
}
function getLine(ix) {
var start = ix;
while (start >= 0) {
var c = ch(start);
if (c == '\n' || c == '\r')
break;
start--;
}
return content.substring(start, ix);
}
while (cur_point < max_len && cur_point > 0) {
cur_point += inc;
var cur_char = ch(cur_point),
next_char = ch(cur_point + 1),
prev_char = ch(cur_point - 1);
switch (cur_char) {
case '"':
case '\'':
if (next_char == cur_char && prev_char == '=') {
// empty attribute
next_point = cur_point + 1;
}
break;
case '>':
if (next_char == '<') {
// between tags
next_point = cur_point + 1;
}
break;
case '\n':
case '\r':
// empty line
if (re_empty_line.test(getLine(cur_point - 1))) {
next_point = cur_point;
}
break;
}
if (next_point != -1)
break;
}
return next_point;
}
/**
* Move caret to previous edit point
* @param {zen_editor} editor Editor instance
*/
function prevEditPoint(editor) {
var cur_pos = editor.getCaretPos(),
new_point = findNewEditPoint(editor, -1);
if (new_point == cur_pos)
// we're still in the same point, try searching from the other place
new_point = findNewEditPoint(editor, -1, -2);
if (new_point != -1)
editor.createSelection(new_point);
}
/**
* Move caret to next edit point
* @param {zen_editor} editor Editor instance
*/
function nextEditPoint(editor) {
var new_point = findNewEditPoint(editor, 1);
if (new_point != -1)
editor.createSelection(new_point);
}
/**
* Inserts newline character with proper indentation
* @param {zen_editor} editor Editor instance
* @param {String} mode Syntax mode (only 'html' is implemented)
*/
function insertFormattedNewline(editor, mode) {
mode = mode || 'html';
var pad = getCurrentLinePadding(editor),
caret_pos = editor.getCaretPos();
function insert_nl() {
editor.replaceContent('\n', caret_pos);
}
switch (mode) {
case 'html':
// let's see if we're breaking newly created tag
var pair = HTMLPairMatcher.getTags(editor.getContent(), editor.getCaretPos());
if (pair[0] && pair[1] && pair[0].type == 'tag' && pair[0].end == caret_pos && pair[1].start == caret_pos) {
editor.replaceContent('\n\t|\n', caret_pos);
} else {
insert_nl();
}
break;
default:
insert_nl();
}
}
/**
* Select line under cursor
* @param {zen_editor} editor Editor instance
*/
function selectLine(editor) {
var range = editor.getCurrentLineRange();
editor.createSelection(range.start, range.end);
}/**
* http://www.openjs.com/scripts/events/keyboard_shortcuts/
* Version : 2.01.B
* By Binny V A
* License : BSD
*/
shortcut = {
'all_shortcuts':{},//All the shortcuts are stored in this array
'add': function(shortcut_combination,callback,opt) {
var is_opera = !!window.opera,
is_mac = /mac\s+os/i.test(navigator.userAgent);
//Provide a set of default options
var default_options = {
'type':is_opera ? 'keypress' : 'keydown',
'propagate':false,
'disable_in_input':false,
'target':document,
'keycode':false
}
if(!opt) opt = default_options;
else {
for(var dfo in default_options) {
if(typeof opt[dfo] == 'undefined') opt[dfo] = default_options[dfo];
}
}
var ele = opt.target;
if(typeof opt.target == 'string') ele = document.getElementById(opt.target);
var ths = this;
shortcut_combination = shortcut_combination.toLowerCase();
//The function to be called at keypress
var func = function(e) {
e = e || window.event;
var code;
if(opt['disable_in_input']) { //Don't enable shortcut keys in Input, Textarea fields
var element;
if(e.target) element=e.target;
else if(e.srcElement) element=e.srcElement;
if(element.nodeType==3) element=element.parentNode;
if(element.tagName == 'INPUT' || element.tagName == 'TEXTAREA') return;
}
//Find Which key is pressed
if (e.keyCode) code = e.keyCode;
else if (e.which) code = e.which;
var character = String.fromCharCode(code).toLowerCase();
if(code == 188) character=","; //If the user presses , when the type is onkeydown
if(code == 190) character="."; //If the user presses , when the type is onkeydown
var keys = shortcut_combination.split("+");
//Key Pressed - counts the number of valid keypresses - if it is same as the number of keys, the shortcut function is invoked
var kp = 0;
//Work around for stupid Shift key bug created by using lowercase - as a result the shift+num combination was broken
var shift_nums = {
"`":"~",
"1":"!",
"2":"@",
"3":"#",
"4":"$",
"5":"%",
"6":"^",
"7":"&",
"8":"*",
"9":"(",
"0":")",
"-":"_",
"=":"+",
";":":",
"'":"\"",
",":"<",
".":">",
"/":"?",
"\\":"|"
}
//Special Keys - and their codes
var special_keys = {
'esc':27,
'escape':27,
'tab':9,
'space':32,
'return':13,
'enter':13,
'backspace':8,
'scrolllock':145,
'scroll_lock':145,
'scroll':145,
'capslock':20,
'caps_lock':20,
'caps':20,
'numlock':144,
'num_lock':144,
'num':144,
'pause':19,
'break':19,
'insert':45,
'home':36,
'delete':46,
'end':35,
'pageup':33,
'page_up':33,
'pu':33,
'pagedown':34,
'page_down':34,
'pd':34,
'plus': 187,
'minus': 189,
'left':37,
'up':38,
'right':39,
'down':40,
'f1':112,
'f2':113,
'f3':114,
'f4':115,
'f5':116,
'f6':117,
'f7':118,
'f8':119,
'f9':120,
'f10':121,
'f11':122,
'f12':123
}
var modifiers = {
shift: { wanted:false, pressed:false},
ctrl : { wanted:false, pressed:false},
alt : { wanted:false, pressed:false},
meta : { wanted:false, pressed:false} //Meta is Mac specific
};
if(e.ctrlKey) modifiers.ctrl.pressed = true;
if(e.shiftKey) modifiers.shift.pressed = true;
if(e.altKey) modifiers.alt.pressed = true;
if(e.metaKey) modifiers.meta.pressed = true;
var k;
for(var i=0; k=keys[i], i<keys.length; i++) {
// Due to stupid Opera bug I have to swap Ctrl and Meta keys
if (is_mac && is_opera) {
if (k == 'ctrl' || k == 'control')
k = 'meta';
else if (k == 'meta')
k = 'ctrl';
} else if (!is_mac && k == 'meta') {
k = 'ctrl';
}
//Modifiers
if(k == 'ctrl' || k == 'control') {
kp++;
modifiers.ctrl.wanted = true;
} else if(k == 'shift') {
kp++;
modifiers.shift.wanted = true;
} else if(k == 'alt') {
kp++;
modifiers.alt.wanted = true;
} else if(k == 'meta') {
kp++;
modifiers.meta.wanted = true;
} else if(k.length > 1) { //If it is a special key
if(special_keys[k] == code) kp++;
} else if(opt['keycode']) {
if(opt['keycode'] == code) kp++;
} else { //The special keys did not match
if(character == k) kp++;
else {
if(shift_nums[character] && e.shiftKey) { //Stupid Shift key bug created by using lowercase
character = shift_nums[character];
if(character == k) kp++;
}
}
}
}
if(kp == keys.length &&
modifiers.ctrl.pressed == modifiers.ctrl.wanted &&
modifiers.shift.pressed == modifiers.shift.wanted &&
modifiers.alt.pressed == modifiers.alt.wanted &&
modifiers.meta.pressed == modifiers.meta.wanted) {
var result = callback(e);
if(result !== true && !opt['propagate']) { //Stop the event
//e.cancelBubble is supported by IE - this will kill the bubbling process.
e.cancelBubble = true;
e.returnValue = false;
//e.stopPropagation works in Firefox.
if (e.stopPropagation) {
e.stopPropagation();
e.preventDefault();
}
return false;
}
}
}
this.all_shortcuts[shortcut_combination] = {
'callback':func,
'target':ele,
'event': opt['type']
};
//Attach the function with the event
if(ele.addEventListener) ele.addEventListener(opt['type'], func, false);
else if(ele.attachEvent) ele.attachEvent('on'+opt['type'], func);
else ele['on'+opt['type']] = func;
},
//Remove the shortcut - just specify the shortcut and I will remove the binding
'remove':function(shortcut_combination) {
shortcut_combination = shortcut_combination.toLowerCase();
var binding = this.all_shortcuts[shortcut_combination];
delete(this.all_shortcuts[shortcut_combination])
if(!binding) return;
var type = binding['event'];
var ele = binding['target'];
var callback = binding['callback'];
if(ele.detachEvent) ele.detachEvent('on'+type, callback);
else if(ele.removeEventListener) ele.removeEventListener(type, callback, false);
else ele['on'+type] = false;
}
}/**
* Editor manager that handles all incoming events and runs Zen Coding actions.
* This manager is also used for setting up editor preferences
* @author Sergey Chikuyonok (serge.che@gmail.com)
* @link http://chikuyonok.ru
*
* @include "actions.js"
* @include "editor.js"
* @include "shortcut.js"
*/ zen_textarea = (function(){ // should be global
var default_options = {
profile: 'xhtml',
syntax: 'html',
use_tab: false,
pretty_break: false
},
mac_char_map = {
'ctrl': '⌃',
'control': '⌃',
'meta': '⌘',
'shift': '⇧',
'alt': '⌥',
'enter': '⏎',
'tab': '⇥',
'left': '←',
'right': '→'
},
pc_char_map = {
'left': '←',
'right': '→'
},
shortcuts = {},
is_mac = /mac\s+os/i.test(navigator.userAgent),
/** Zen Coding parameter name/value regexp for getting options from element */
re_param = /\bzc\-(\w+)\-(\w+)/g;
/** @type {default_options} */
var options = {};
function copyOptions(opt) {
opt = opt || {};
var result = {};
for (var p in default_options) if (default_options.hasOwnProperty(p)) {
result[p] = (p in opt) ? opt[p] : default_options[p];
}
return result;
}
options = copyOptions();
/**
* Makes first letter of string in uppercase
* @param {String} str
*/
function capitalize(str) {
return str.charAt().toUpperCase() + str.substring(1);
}
function humanize(str) {
return capitalize(str.replace(/_(\w)/g, function(s, p){return ' ' + p.toUpperCase()}));
}
function formatShortcut(char_map, glue) {
var result = [];
if (typeof(glue) == 'undefined')
glue = '+';
for (var p in shortcuts) if (shortcuts.hasOwnProperty(p)) {
var keys = p.split('+'),
ar = [],
lp = p.toLowerCase();
if (lp == 'tab' || lp == 'enter')
continue;
for (var i = 0; i < keys.length; i++) {
var key = keys[i].toLowerCase();
ar.push(key in char_map ? char_map[key] : capitalize(key));
}
result.push({
'keystroke': ar.join(glue),
'action_name': humanize(shortcuts[p])
});
}
return result;
}
/**
* Get Zen Coding options from element's class name
* @param {Element} elem
*/
function getOptionsFromElement(elem) {
var param_str = elem.className || '',
m,
result = copyOptions(options);
while ( (m = re_param.exec(param_str)) ) {
var key = m[1].toLowerCase(),
value = m[2].toLowerCase();
if (value == 'true' || value == 'yes' || value == '1')
value = true;
else if (value == 'false' || value == 'no' || value == '0')
value = false;
result[key] = value;
}
return result;
}
/**
* Returns normalized action name
* @param {String} name Action name (like 'Expand Abbreviation')
* @return Normalized name for coding (like 'expand_abbreviation')
*/
function normalizeActionName(name) {
return name
.replace(/(^\s+|\s+$)/g, '') // remove trailing spaces
.replace(/\s+/g, '_')
.toLowerCase();
}
/**
* Runs actions called by user
* @param {String} name Normalized action name
* @param {Event} evt Event object
*/
function runAction(name, evt) {
/** @type {Element} */
var target_elem = evt.target || evt.srcElement,
key_code = evt.keyCode || evt.which;
if (target_elem && target_elem.nodeType == 1 && target_elem.nodeName == 'TEXTAREA') {
zen_editor.setTarget(target_elem);
var options = getOptionsFromElement(target_elem),
syntax = options.syntax,
profile_name = options.profile;
switch (name) {
case 'expand_abbreviation':
if (key_code == 9) {
if (options.use_tab)
expandAbbreviationWithTab(zen_editor, syntax, profile_name);
else
// user pressed Tab key but it's forbidden in
// Zen Coding: bubble up event
return true;
} else {
expandAbbreviation(zen_editor, syntax, profile_name);
}
break;
case 'match_pair_inward':
case 'balance_tag_inward':
matchPair(zen_editor, 'in');
break;
case 'match_pair_outward':
case 'balance_tag_outward':
matchPair(zen_editor, 'out');
break;
case 'wrap_with_abbreviation':
var abbr = prompt('Enter abbreviation', 'div');
if (abbr)
wrapWithAbbreviation(zen_editor, abbr, syntax, profile_name);
break;
case 'next_edit_point':
nextEditPoint(zen_editor);
break;
case 'previous_edit_point':
case 'prev_edit_point':
prevEditPoint(zen_editor);
break;
case 'pretty_break':
case 'format_line_break':
if (key_code == 13) {
if (options.pretty_break)
insertFormattedNewline(zen_editor);
else
// user pressed Enter but it's forbidden in
// Zen Coding: bubble up event
return true;
} else {
insertFormattedNewline(zen_editor);
}
break;
case 'select_line':
selectLine(zen_editor);
}
} else {
// allow event bubbling
return true;
}
}
/**
* Bind shortcut to Zen Coding action
* @param {String} keystroke
* @param {String} action_name
*/
function addShortcut(keystroke, action_name) {
action_name = normalizeActionName(action_name);
shortcuts[keystroke.toLowerCase()] = action_name;
shortcut.add(keystroke, function(evt){
return runAction(action_name, evt);
});
}
// add default shortcuts
addShortcut('Meta+E', 'Expand Abbreviation');
addShortcut('Tab', 'Expand Abbreviation');
addShortcut('Meta+D', 'Balance Tag Outward');
addShortcut('Shift+Meta+D', 'Balance Tag inward');
addShortcut('Shift+Meta+A', 'Wrap with Abbreviation');
addShortcut('Ctrl+RIGHT', 'Next Edit Point');
addShortcut('Ctrl+LEFT', 'Previous Edit Point');
addShortcut('Meta+L', 'Select Line');
addShortcut('Enter', 'Format Line Break');
return {
shortcut: addShortcut,
/**
* Removes shortcut binding
* @param {String} keystroke
*/
unbindShortcut: function(keystroke) {
keystroke = keystroke.toLowerCase();
if (keystroke in shortcuts)
delete shortcuts[keystroke];
shortcut.remove(keystroke);
},
/**
* Setup editor. Pass object with values defined in
* <code>default_options</code>
*/
setup: function(opt) {
options = copyOptions(opt);
},
/**
* Returns option value
*/
getOption: function(name) {
return options[name];
},
/**
* Returns array of binded actions and their keystrokes
* @return {Array}
*/
getShortcuts: function() {
return formatShortcut(is_mac ? mac_char_map : pc_char_map, is_mac ? '' : '+');
},
/**
* Show info window about Zen Coding
*/
showInfo: function() {
var message = 'All textareas on this page are powered by Zen Coding project: ' +
'a set of tools for fast HTML coding.\n\n' +
'Available shortcuts:\n';
var sh = this.getShortcuts(),
actions = [];
for (var i = 0; i < sh.length; i++) {
actions.push(sh[i].keystroke + ' — ' + sh[i].action_name)
}
message += actions.join('\n') + '\n\n';
message += 'More info on http://code.google.com/p/zen-coding/';
alert(message);
}
}
})();
})();