;;; web-mode.el --- major mode for editing html templates
;;; -*- coding: utf-8 -*-
;; Copyright 2011-2014 François-Xavier Bois
;; Version: 20141018.216
;; X-Original-Version: 10.0.3
;; Author: François-Xavier Bois
;; Maintainer: François-Xavier Bois
;; Created: July 2011
;; Keywords: html template javascript css web php django erb jsp
;; URL: http://web-mode.org
;; Repository: http://github.com/fxbois/web-mode
;; License: GNU General Public License >= 2
;; Distribution: This file is not part of Emacs
;; =========================================================================
;; This work is sponsored by Kernix : Digital Agency (Web & Mobile) in Paris
;; =========================================================================
;; Code goes here
;;---- CONSTS ------------------------------------------------------------------
(defconst web-mode-version "10.0.3"
"Web Mode version.")
;;---- GROUPS ------------------------------------------------------------------
(defgroup web-mode nil
"Major mode for editing web templates:
html files embedding parts (css/javascript)
and blocks (php, erb, django/twig, smarty, jsp, asp, etc.)"
:group 'languages
:prefix "web-"
:link '(url-link :tag "Site" "http://web-mode.org")
:link '(url-link :tag "Repository" "https://github.com/fxbois/web-mode"))
(defgroup web-mode-faces nil
"Faces for syntax highlighting."
:group 'web-mode
:group 'faces)
;;---- CUSTOMS -----------------------------------------------------------------
(defcustom web-mode-script-padding 1
"Script element left padding."
:type 'integer
:group 'web-mode)
(defcustom web-mode-style-padding 1
"Style element left padding."
:type 'integer
:group 'web-mode)
(defcustom web-mode-block-padding 0
"Multi-line block (php, ruby, java, python, asp, etc.) left padding."
:type 'integer
:group 'web-mode)
(defcustom web-mode-markup-indent-offset 2
"Html indentation level."
:type 'integer
:group 'web-mode)
(defcustom web-mode-css-indent-offset 2
"CSS indentation level."
:type 'integer
:group 'web-mode)
(defcustom web-mode-code-indent-offset 2
"Code (javascript, php, etc.) indentation level."
:type 'integer
:group 'web-mode)
(defcustom web-mode-sql-indent-offset 4
"Sql (inside strings) indentation level."
:type 'integer
:group 'web-mode)
(defcustom web-mode-enable-css-colorization (display-graphic-p)
"In a CSS part, set background according to the color: #xxx, rgb(x,x,x)."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-auto-indentation (display-graphic-p)
"Auto-indentation."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-auto-pairing (display-graphic-p)
"Auto-pairing."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-auto-opening (display-graphic-p)
"Html element auto-opening."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-block-partial-invalidation t
"Partial invalidation in blocks (php and asp at the moment)."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-part-partial-invalidation t
"Partial invalidation in js/css parts."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-current-element-highlight nil
"Disable element highlight."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-current-column-highlight nil
"Show current region."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-whitespace-fontification nil
"Enable whitespaces."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-block-face nil
"Enable block face (useful for setting a background for example).
See web-mode-block-face."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-part-face nil
"Enable part face (useful for setting a background for example).
See web-mode-part-face."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-inlays nil
"Enable inlays (e.g. LaTeX) highlighting."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-string-interpolation t
"Enable string interpolation fontification (php and erb)."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-heredoc-fontification t
"Enable heredoc fontification. The identifier should contain JS, JAVASCRIPT or HTML."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-element-content-fontification nil
"Enable element content fontification. The content of an element can have a face associated."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-element-tag-fontification nil
"Enable tag name fontification."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-engine-detection nil
"Detect directive like -*- engine: ENGINE -*- at the top of the file."
:type 'boolean
:group 'web-mode)
(defcustom web-mode-enable-comment-keywords nil
"Enable highlight of keywords like FIXME, TODO, etc. in comments."
:type 'list
:group 'web-mode)
(defcustom web-mode-comment-style 1
"Comment style : 1 = default, 2 = force server comments outside a block."
:group 'web-mode
:type '(choice (const :tag "default" 1)
(const :tag "force engine comments" 2)))
(defcustom web-mode-indent-style 2
"Indentation style."
:group 'web-mode
:type '(choice (const :tag "default (all lines are indented)" 2)
(const :tag "text at the beginning of line is not indented" 1)))
(defcustom web-mode-tag-auto-close-style (if (display-graphic-p) 1 0)
"Tag auto-close style:
0=no auto-closing
1=auto-close with
2=auto-close with > and )."
:group 'web-mode-faces)
(defface web-mode-html-tag-bracket-face
'((((class color) (min-colors 88) (background dark)) :foreground "Snow3")
(((class color) (min-colors 88) (background light)) :foreground "Grey14")
(((class color) (min-colors 16) (background dark)) :foreground "Snow3")
(((class color) (min-colors 16) (background light)) :foreground "Grey14")
(((class color) (min-colors 8)) :foreground "Snow3")
(((type tty) (class mono)) :inverse-video t)
(t :foreground "Snow3"))
"Face for HTML tags angle brackets (< and >)."
:group 'web-mode-faces)
(defface web-mode-html-attr-name-face
'((((class color) (min-colors 88) (background dark)) :foreground "Snow3")
(((class color) (min-colors 88) (background light)) :foreground "Snow4")
(((class color) (min-colors 16) (background dark)) :foreground "Snow3")
(((class color) (min-colors 16) (background light)) :foreground "Grey13")
(((class color) (min-colors 8)) :foreground "Snow3")
(((type tty) (class mono)) :inverse-video t)
(t :foreground "Snow4"))
"Face for HTML attribute names."
:group 'web-mode-faces)
(defface web-mode-html-attr-custom-face
'((t :inherit web-mode-html-attr-name-face))
"Face for custom attribute names (e.g. data-*)."
:group 'web-mode-faces)
(defface web-mode-html-attr-engine-face
'((t :inherit web-mode-html-attr-custom-face))
"Face for custom engine attribute names (e.g. ng-*)."
:group 'web-mode-faces)
(defface web-mode-html-attr-equal-face
'((t :inherit web-mode-html-attr-name-face))
"Face for the = character between name and value."
:group 'web-mode-faces)
(defface web-mode-html-attr-value-face
'((t :inherit font-lock-string-face))
"Face for HTML attribute values."
:group 'web-mode-faces)
(defface web-mode-block-attr-name-face
'((t :foreground "#8fbc8f"))
"Face for block attribute names."
:group 'web-mode-faces)
(defface web-mode-block-attr-value-face
'((t :foreground "#5f9ea0"))
"Face for block attribute values."
:group 'web-mode-faces)
(defface web-mode-variable-name-face
'((t :inherit font-lock-variable-name-face))
"Face for variable names."
:group 'web-mode-faces)
(defface web-mode-css-selector-face
'((t :inherit font-lock-keyword-face))
"Face for CSS rules."
:group 'web-mode-faces)
(defface web-mode-css-pseudo-class-face
'((t :inherit font-lock-builtin-face))
"Face for CSS pseudo-classes."
:group 'web-mode-faces)
(defface web-mode-css-at-rule-face
'((t :inherit font-lock-constant-face))
"Face for CSS at-rules."
:group 'web-mode-faces)
(defface web-mode-css-property-name-face
'((t :inherit font-lock-variable-name-face))
"Face for CSS props."
:group 'web-mode-faces)
(defface web-mode-css-color-face
'((t :inherit font-lock-builtin-face))
"Face for CSS colors (#xxx)."
:group 'web-mode-faces)
(defface web-mode-css-priority-face
'((t :inherit font-lock-builtin-face))
"Face for CSS priority (!important)."
:group 'web-mode-faces)
(defface web-mode-css-function-face
'((t :inherit font-lock-builtin-face))
"Face for CSS functions."
:group 'web-mode-faces)
(defface web-mode-css-variable-face
'((t :inherit web-mode-variable-name-face :slant italic))
"Face for CSS vars."
:group 'web-mode-faces)
(defface web-mode-function-name-face
'((t :inherit font-lock-function-name-face))
"Face for function names."
:group 'web-mode-faces)
(defface web-mode-function-call-face
'((t :inherit font-lock-function-name-face))
"Face for function calls."
:group 'web-mode-faces)
(defface web-mode-string-face
'((t :inherit font-lock-string-face))
"Face for strings."
:group 'web-mode-faces)
(defface web-mode-block-string-face
'((t :inherit web-mode-string-face))
"Face for block strings."
:group 'web-mode-faces)
(defface web-mode-part-string-face
'((t :inherit web-mode-string-face))
"Face for part strings."
:group 'web-mode-faces)
(defface web-mode-javascript-string-face
'((t :inherit web-mode-string-face))
"Face for javascript strings."
:group 'web-mode-faces)
(defface web-mode-css-string-face
'((t :inherit web-mode-string-face))
"Face for css strings."
:group 'web-mode-faces)
(defface web-mode-json-key-face
'((t :foreground "plum"))
"Face for json key strings."
:group 'web-mode-faces)
(defface web-mode-json-context-face
'((t :foreground "orchid3"))
"Face for json context strings."
:group 'web-mode-faces)
(defface web-mode-json-string-face
'((t :inherit web-mode-string-face))
"Face for json strings."
:group 'web-mode-faces)
(defface web-mode-comment-face
'((t :inherit font-lock-comment-face))
"Face for comments."
:group 'web-mode-faces)
(defface web-mode-block-comment-face
'((t :inherit web-mode-comment-face))
"Face for server comments."
:group 'web-mode-faces)
(defface web-mode-part-comment-face
'((t :inherit web-mode-comment-face))
"Face for part comments."
:group 'web-mode-faces)
(defface web-mode-json-comment-face
'((t :inherit web-mode-comment-face))
"Face for json comments."
:group 'web-mode-faces)
(defface web-mode-javascript-comment-face
'((t :inherit web-mode-comment-face))
"Face for javascript comments."
:group 'web-mode-faces)
(defface web-mode-css-comment-face
'((t :inherit web-mode-comment-face))
"Face for css comments."
:group 'web-mode-faces)
(defface web-mode-constant-face
'((t :inherit font-lock-constant-face))
"Face for language constants."
:group 'web-mode-faces)
(defface web-mode-type-face
'((t :inherit font-lock-type-face))
"Face for language types."
:group 'web-mode-faces)
(defface web-mode-keyword-face
'((t :inherit font-lock-keyword-face))
"Face for language keywords."
:group 'web-mode-faces)
(defface web-mode-param-name-face
'((t :foreground "Snow3"))
"Face for server attribute names."
:group 'web-mode-faces)
(defface web-mode-whitespace-face
'((t :background "DarkOrchid4"))
"Face for whitespaces."
:group 'web-mode-faces)
(defface web-mode-inlay-face
'((((class color) (min-colors 88) (background dark)) :background "Black")
(((class color) (min-colors 88) (background light)) :background "LightYellow1")
(((class color) (min-colors 16) (background dark)) :background "Brey18")
(((class color) (min-colors 16) (background light)) :background "LightYellow1")
(((class color) (min-colors 8)) :background "Black")
(((type tty) (class mono)) :inverse-video t)
(t :background "Grey"))
"Face for inlays. Must be used in conjunction with web-mode-enable-inlays."
:group 'web-mode-faces)
(defface web-mode-block-face
'((((class color) (min-colors 88) (background dark)) :background "Black")
(((class color) (min-colors 88) (background light)) :background "LightYellow1")
(((class color) (min-colors 16) (background dark)) :background "Grey18")
(((class color) (min-colors 16) (background light)) :background "LightYellow1")
(((class color) (min-colors 8)) :background "Black")
(((type tty) (class mono)) :inverse-video t)
(t :background "Grey"))
"Face for blocks (useful for setting a background for example).
Must be used in conjunction with web-mode-enable-block-face."
:group 'web-mode-faces)
(defface web-mode-part-face
'((t :inherit web-mode-block-face))
"Face for parts."
:group 'web-mode-faces)
(defface web-mode-folded-face
'((t :underline t))
"Overlay face for folded."
:group 'web-mode-faces)
(defface web-mode-bold-face
'((t :weight bold))
"bold face."
:group 'web-mode-faces)
(defface web-mode-italic-face
'((t :slant italic))
"bold face."
:group 'web-mode-faces)
(defface web-mode-underline-face
'((t :underline t))
"bold face."
:group 'web-mode-faces)
(defface web-mode-current-element-highlight-face
'((t :background "#000000"))
"Overlay face for element highlight."
:group 'web-mode-faces)
(defface web-mode-current-column-highlight-face
'((t :background "#292929"))
"Overlay face for current column."
:group 'web-mode-faces)
(defface web-mode-comment-keyword-face
'((t :weight bold :box t))
"Comment keywords."
:group 'web-mode-faces)
(defface web-mode-sql-keyword-face
'((t :weight bold :slant italic))
"Sql keywords."
:group 'web-mode-faces)
;;---- VARS --------------------------------------------------------------------
(defvar font-lock-beg)
(defvar font-lock-end)
(defvar web-mode-auto-pairs nil)
(defvar web-mode-block-regexp nil)
(defvar web-mode-chunk-length 64)
(defvar web-mode-closing-blocks nil)
(defvar web-mode-column-overlays nil)
(defvar web-mode-comments-invisible nil)
(defvar web-mode-content-type "")
(defvar web-mode-end-tag-overlay nil)
(defvar web-mode-engine nil)
(defvar web-mode-engine-attr-regexp nil)
(defvar web-mode-engine-font-lock-keywords nil)
(defvar web-mode-engine-token-regexp nil)
(defvar web-mode-expand-initial-pos nil)
(defvar web-mode-expand-previous-state "")
(defvar web-mode-font-lock-keywords '(web-mode-font-lock-highlight))
(defvar web-mode-change-beg nil)
(defvar web-mode-change-end nil)
(defvar web-mode-hl-line-mode-flag nil)
(defvar web-mode-hook nil)
(defvar web-mode-inlay-regexp nil)
(defvar web-mode-is-scratch nil)
(defvar web-mode-jshint-errors 0)
(defvar web-mode-obarray nil)
(defvar web-mode-snippets nil)
(defvar web-mode-start-tag-overlay nil)
(defvar web-mode-time (current-time))
(defvar web-mode-pre-elements
'("code" "pre" "textarea"))
(defvar web-mode-void-elements
'("area" "base" "br" "col" "command" "embed" "hr" "img" "input" "keygen"
"link" "meta" "param" "source" "track" "wbr"))
(defvar web-mode-scan-properties
(list 'tag-beg nil 'tag-end nil 'tag-name nil 'tag-type nil
'tag-attr nil 'tag-attr-end nil
'part-side nil 'part-token nil 'part-element nil 'part-expr nil
'block-side nil 'block-token nil 'block-controls nil
'block-beg nil 'block-end nil
;;'face nil
;;'syntax-table nil
)
"Text properties used for code regions/tokens and html nodes.")
(defvar web-mode-start-tag-regexp "<\\([[:alpha:]][[:alnum:]-]*\\)"
"Regular expression for HTML/XML start tag.")
(defvar web-mode-whitespaces-regexp
"^[ \t]\\{2,\\}$\\| \t\\|\t \\|[ \t]+$\\|^[ \n\t]+\\'\\|^[ \t]?[\n]\\{2,\\}"
"Regular expression for whitespaces.")
(defvar web-mode-imenu-regexp-list
'(("<\\(h[1-9]\\)\\([^>]*\\)>\\([^<]*\\)" 1 3 ">")
("^[ \t]*<\\([@a-z]+\\)[^>]*>? *$" 1 "id=\"\\([a-zA-Z0-9_]+\\)\"" "#" ">"))
"Regexps to match imenu items (see http://web-mode.org/doc/imenu.txt)")
(defvar web-mode-engines
'(("angular" . ("angularjs" "angular.js"))
("asp" . ())
("aspx" . ())
("blade" . ("laravel"))
("clip" . ())
("closure" . ("soy"))
("ctemplate" . ("mustache" "handlebars" "hapax" "ngtemplate" "ember" "kite" "meteor" "blaze"))
("django" . ("dtl" "twig" "swig" "jinja" "jinja2" "erlydtl" "liquid" "clabango" "selmer"))
("dust" . ("dustjs"))
("erb" . ("eruby" "erubis" "ejs"))
("go" . ("gtl"))
("jsp" . ("grails"))
("mason" . ("poet"))
("lsp" . ())
("mojolicious" . ())
("python" . ())
("razor" . ("play" "play2"))
("underscore" . ("underscore.js"))
("velocity" . ("vtl" "cheetah"))
("web2py" . ()))
"Engine name aliases")
(defvar web-mode-content-types
'(("css" . "\\.\\(s?css\\|css\\.erb\\)\\'")
("javascript" . "\\.\\(js\\|js\\.erb\\)\\'")
("json" . "\\.\\(json\\|jsonld\\)\\'")
("jsx" . "\\.jsx\\'")
("html" . "."))
"content types")
(defvar web-mode-engine-attr-regexps
'(("angular" . "ng-"))
"Engine custom attributes")
(defvar web-mode-engine-file-regexps
'(("asp" . "\\.asp\\'")
("aspx" . "\\.as[cp]x\\'")
("blade" . "\\.blade\\.php\\'")
("clip" . "\\.ctml\\'")
("closure" . "\\.soy\\'")
("ctemplate" . "\\.\\(chtml\\|mustache\\)\\'")
("django" . "\\.\\(djhtml\\|tmpl\\|dtl\\|liquid\\)\\'")
("erb" . "\\.\\(erb\\|rhtml\\|ejs\\|erb\\.html\\)\\'")
("freemarker" . "\\.ftl\\'")
("go" . "\\.go\\(html\\|tmpl\\)\\'")
("handlebars" . "\\.\\(hb\\.html\\|hbs\\)\\'")
("jsp" . "\\.[gj]sp\\'")
("lsp" . "\\.lsp\\'")
("mako" . "\\.mako?\\'")
("mason" . "\\.mas\\'")
("mojolicious" . "\\.epl?\\'")
("php" . "\\.\\(php\\|ctp\\|psp\\|inc\\)\\'")
("python" . "\\.pml\\'")
("razor" . "\\.\\(cshtml\\|vbhtml\\)\\'")
("smarty" . "\\.tpl\\'")
("template-toolkit" . "\\.tt.?\\'")
("velocity" . "\\.v\\(sl\\|tl\\|m\\)\\'")
("django" . "twig")
("razor" . "scala")
)
"Engine file extensions.")
(defvar web-mode-smart-quotes
'("«" . "»")
"Preferred smart quotes")
(defvar web-mode-xml-chars
'((?\& . "&")
(?\< . "<")
(?\> . ">"))
"XML chars")
(defvar web-mode-html-entities
'(("egrave" . 232)
("eacute" . 233)
("ecirc" . 234)
("euml" . 235)
("agrave" . 224)
("aacute" . 225)
("aelig" . 230)
("ccedil" . 231)
("times" . 215)
("quot" . 34)
("amp" . 38)
("lt" . 60)
("gt" . 62)
("laquo" . 171)
("raquo" . 187)
("lsquo" . 8249)
("rsquo" . 8250)
("ldquo" . 8220)
("rdquo" . 8221)
("lsaquo" . 8249)
("rsaquo" . 8250)
("apos" . 39)
("frac14" . 188)
("frac12" . 189)
("frac34" . 190)
("para" . 182)
("middot" . 183)
("ndash" . 8211)
("mdash" . 8212)
("bull" . 8226)
("hellip" . 8230)
("trade" . 8482)
("larr" . 8592)
("uarr" . 8593)
("rarr" . 8594)
("darr" . 8595)
("harr" . 8596)
("crarr" . 8629)
("and" . 8743)
("or" . 8744)
("sdot" . 8901)))
(defvar web-mode-engines-alternate-delimiters
(if (boundp 'web-mode-engines-alternate-delimiters)
web-mode-engines-alternate-delimiters
'())
"Engine delimiters. Useful for engines that provide alternate delimiters.")
(defun web-mode-engine-delimiter-open (engine default)
"alternative open delimiter"
(let (delim)
(setq delim (car (cdr (assoc engine web-mode-engines-alternate-delimiters))))
(or delim default)
))
(defun web-mode-engine-delimiter-close (engine default)
"alternative close delimiter"
(let (delim)
(setq delim (cdr (cdr (assoc engine web-mode-engines-alternate-delimiters))))
(or delim default)
))
;; http://webdesign.about.com/od/localization/l/blhtmlcodes-ascii.htm
(defvar web-mode-display-table
(let ((table (make-display-table)))
(aset table 9 (vector ?\xBB ?\t)) ;tab
(aset table 10 (vector ?\xB6 ?\n)) ;line feed
(aset table 32 (vector ?\xB7))
table)
"Display table used when switching to the internal whitespace mode.")
(defvar web-mode-engines-closing-blocks
'(
("php" . (("if" . "")
("for" . "")
("foreach" . "")
("while" . "")))
)
"Closing blocks (see web-mode-block-close)")
(defvar web-mode-engines-auto-pairs
'(("angular" . (("{{ " " }}")))
("asp" . (("<% " " %>")))
("aspx" . (("<% " " %>")
("<%=" "%>")
("<%#" "%>")
("<%$" "%>")
("<%@" "%>")
("<%:" "%>")
("<%-" "- " " --%>")))
("blade" . (("{{{" " " " }}}")
("{{ " " }}")
("{{-" "- " " --}}")))
("ctemplate" . (("{{ " "}}")
("{{{" "}}}")
("{~{" "}}")
("{{~" "{ " "}}}")
("{{/" "}}")
("{{#" "}}")))
("django" . (("{{ " " }}")
("{% " " %}")
("{%-" " " " %}")
("{# " " #}")))
("erb" . (("<% " " %>")
("<%=" "%>")
("<%#" "%>")
("<%-" "%>")))
("freemarker" . (("<% " " %>")
("${ " " }")
("[% " " %]")
("[# " " #]")
("[#-" "- " " --]")))
("jsp" . (("<% " " %>")
("<%-" "- " " %>")
("<%=" "%>")
("<%!" "%>")
("<%@" "%>")
("${ " " }")))
("lsp" . (("<% " " %>")
("<%%" " " " %>")
("<%#" " " " %>")))
("mako" . (("<% " " %>")
("<%!" " " " %>")
("${ " " }")))
("mason" . (("<% " " %>")))
("mojolicious" . (("<% " " %>")
("<%=" " " " %>")
("<%%" " " " %>")
("<%#" " " "%>")))
("php" . (("")
(" " " ?>")
("=" "?>")))
("underscore" . (("<% " " %>")))
("web2py" . (("{{ " " }}")
("{{=" "}}")))
(nil . ((""))))
"Engines auto-pairs")
(defvar web-mode-engines-snippets
'(("erb" . (("if" . ("<% if " . " %>\n\n<% end %>"))))
("php" . (("if" . ("\n\n"))
("while" . ("\n\n"))
("for" . ("\n\n"))
("foreach" . ("\n\n"))
("switch" . ("\n\n\n\n\n\n\n"))))
("django" . (("block" . ("{% block " . " %}\n\n{% endblock %}"))
("comment" . ("{% comment " . " %}\n\n{% endcomment %}"))
("cycle" . ("{% cycle " . " as %}\n\n{% endcycle %}"))
("filter" . ("{% filter " . " %}\n\n{% endfilter %}"))
("for" . ("{% for " . " in %}\n\n{% endfor %}"))
("if" . ("{% if " . " %}\n\n{% endif %}"))
("ifequal" . ("{% ifequal " . " %}\n\n{% endifequal %}"))
("ifnotequal" . ("{% ifnotequal " . " %}\n\n{% endifnotequal %}"))
("safe" . ("{% safe " . " %}\n\n{% endsafe %}"))
))
(nil . (("html5" . ("\n\n\n\n\n\n\n" . "\n\n"))
("table" . (""))
("ul" . (""))))
)
"Engines snippets")
(defvar web-mode-engine-token-regexps
(list
'("asp" . "//\\|/\\*\\|\"\\|'")
'("erb" . "\"\\|'\\|#\\|<<[-]?['\"]?\\([[:alnum:]_]+\\)['\"]?")
'("lsp" . "\"\\|#|\\|;")
'("mako" . "\"\\|'\\|#")
'("mason" . "\"\\|'\\|#")
'("mojolicious" . "\"\\|'")
'("php" . "//\\|/\\*\\|#\\|\"\\|'\\|<<<['\"]?\\([[:alnum:]]+\\)['\"]?")
'("python" . "\"\\|'\\|#")
'("web2py" . "\"\\|'"))
"Engine regexps used to identify tokens (strings / comments) in blocks.")
(defvar web-mode-engine-open-delimiter-regexps
(list
'("angular" . "{{")
'("asp" . "<%\\|?[[:alpha:]]+:[[:alpha:]]+\\|?[[:alpha:]]+Template")
'("aspx" . "<%.")
'("blade" . "{{.\\|^[ \t]*@[[:alpha:]]")
'("closure" . "{.\\|/\\*\\| //")
'("clip" . "?c:[[:alpha:]-]+")
'("ctemplate" . "[$]?{[{~].")
'("django" . "{[#{%]")
'("dust" . "{.")
'("erb" . "<%\\|^%.")
'("freemarker" . "<%\\|${\\|?[[:alpha:]]+:[[:alpha:]]\\|?[@#].\\|\\[/?[@#].")
'("go" . "{{.")
'("jsp" . "<%\\|${\\|?[[:alpha:]]+:[[:alpha:]]+")
'("lsp" . "<%")
'("mako" . "?%\\|${\\|^[ \t]*%.\\|^[ \t]*##")
'("mason" . "?[&%]\\|^%.")
'("mojolicious" . "<%.\\|^[ \t]*%.")
'("php" . "<\\?")
'("python" . "<\\?")
'("razor" . "@.\\|^[ \t]*}")
(cons "smarty" (concat (web-mode-engine-delimiter-open "smarty" "{") "[[:alpha:]#$/*\"]"))
'("template-toolkit" . "\\[[%#]")
'("underscore" . "<%")
'("velocity" . "^[ \t]*#[[:alpha:]#*]\\|$[[:alpha:]!{]")
'("web2py" . "{{"))
"Engine regexps used to identify blocks.")
(defvar web-mode-normalization-rules
'(("tag-case" . "lower-case")
("attr-case" . "lower-case")
("special-chars" . "unicode") ; "unicode" "entities"
("css-indentation" . t)
("smart-apostrophes" . t)
("smart-quotes" . t)
("whitespaces" . t)
("indentation" . t))
"Normalization rules")
(defvar web-mode-element-tag-faces
(list
'("h1" . web-mode-underline-face)
'("h2" . web-mode-underline-face)
'("h3" . web-mode-underline-face)
'("h4" . web-mode-underline-face)
'("title" . web-mode-underline-face)
'("em" . web-mode-italic-face)
'("strong" . web-mode-bold-face)
))
(defvar web-mode-element-content-faces
(list
'("h1" . web-mode-underline-face)
'("h2" . web-mode-underline-face)
'("h3" . web-mode-underline-face)
'("h4" . web-mode-underline-face)
'("title" . web-mode-underline-face)
'("em" . web-mode-italic-face)
'("strong" . web-mode-bold-face)
))
(defvar web-mode-comment-keywords
(regexp-opt
(append
(cdr (assoc "comment" web-mode-extra-keywords))
'("FIXME" "TODO" "BUG" "KLUDGE" "WORKAROUND" "OPTIMIZE" "HACK" "REFACTOR" "REVIEW"))))
(defvar web-mode-sql-queries
(regexp-opt
'("SELECT" "INSERT" "UPDATE" "DELETE")))
(defvar web-mode-sql-keywords
(regexp-opt
(append
(cdr (assoc "sql" web-mode-extra-keywords))
'("SELECT" "INSERT" "UPDATE" "DELETE"
"FROM" "WHERE" "GROUP BY" "LIMIT" "HAVING" "JOIN" "LEFT" "INNER"
"FULL" "OUTER" "VALUES" "ORDER BY" "SEPARATOR" "ASC" "DESC"
"AND" "OR" "ON"))))
(defvar web-mode-python-constants
(regexp-opt
(append
(cdr (assoc "python" web-mode-extra-constants))
'("True" "False" "None" "__debug__" "NotImplemented" "Ellipsis"))))
(defvar web-mode-lsp-constants
(regexp-opt
'("nil" "t")))
(defvar web-mode-lsp-keywords
(regexp-opt
'("dolist" "let" "while" "cond" "when" "progn" "if"
"dotimes" "unless" "lambda"
"loop" "for" "and" "or" "in" "do" "defun")))
(defvar web-mode-php-constants
(regexp-opt
(append
(cdr (assoc "php" web-mode-extra-constants))
'("TRUE" "FALSE" "NULL" "true" "false" "null"
"STR_PAD_LEFT" "STR_PAD_RIGHT"
"ENT_COMPAT" "ENT_QUOTES" "ENT_NOQUOTES" "ENT_IGNORE"
"ENT_SUBSTITUTE" "ENT_DISALLOWED" "ENT_HTML401" "ENT_XML1"
"ENT_XHTML" "ENT_HTML5"
"LIBXML_NOBLANKS"))))
(defvar web-mode-php-keywords
(regexp-opt
(append
(cdr (assoc "php" web-mode-extra-keywords))
'("and" "array" "as" "break"
"callable" "case" "catch" "catch all" "class" "const" "continue"
"default" "die" "do" "echo" "else" "elseif" "empty"
"endfor" "endforeach" "endif" "endswitch" "endwhile" "exit" "extends"
"finally" "for" "foreach" "function" "global" "goto"
"if" "include" "include_once" "instanceof" "interface" "isset"
"list" "next" "new" "or" "private" "protected" "public"
"require" "require_once" "return" "static" "switch" "try" "throw" "unset" "use"
"var" "when" "while" "xor" "yield"))))
(defvar web-mode-php-types
(eval-when-compile
(regexp-opt
'("array" "bool" "boolean" "char" "const" "double" "float"
"int" "integer" "long" "mixed" "object" "real" "string"))))
(defvar web-mode-css-at-rules
(eval-when-compile
(regexp-opt
'("charset" "import" "media" "page" "font-face"
"namespace" "supports" "document"
"keyframes" "-moz-keyframes" "-webkit-keyframes"
"mixin"))))
(defvar web-mode-css-pseudo-classes
(eval-when-compile
(regexp-opt
'("active" "after" "before" "checked" "disabled" "empty" "enabled"
"first" "first-child" "first-letter" "first-line" "first-of-type" "focus"
"hover" "lang" "last-child" "last-of-type" "left" "link"
"not" "nth-child" "nth-last-child" "nth-last-of-type" "nth-of-type"
"only-child" "only-of-type"
"right" "root" "selection" "target" "visited"))))
(defvar web-mode-python-keywords
(regexp-opt
(append
(cdr (assoc "python" web-mode-extra-keywords))
'("and" "as" "assert" "break" "class" "continue" "def" "del"
"elif" "else" "except" "finally" "for" "from" "global"
"if" "import" "in" "is" "lambda" "nonlocal" "not" "or" "pass"
"raise" "return" "try" "while" "with" "yield"))))
(defvar web-mode-jsp-keywords
(regexp-opt
(append
(cdr (assoc "jsp" web-mode-extra-keywords))
'("case" "catch" "do" "else" "end" "false" "for" "function" "if" "in" "include"
"new" "package" "page" "private" "protected" "public"
"return" "tag" "taglib" "throw" "throws" "true" "try" "void" "while"))))
(defvar web-mode-erb-keywords
(regexp-opt
(append
(cdr (assoc "erb" web-mode-extra-keywords))
'("alias" "and" "begin" "break" "case" "class" "def" "defined?" "do"
"elsif" "else" "end" "ensure" "fail" "for" "if" "in"
"module" "next" "not" "or" "redo" "rescue" "retry" "return"
"then" "super" "unless" "undef" "until" "when" "while" "yield"
"__ENCODING__" "__FILE__" "__LINE__"))))
(defvar web-mode-mason-keywords
(regexp-opt
(append
(cdr (assoc "mason" web-mode-extra-keywords))
'("and" "base" "close" "die" "each" "else" "elsif" "eval" "exists"
"foreach" "grep" "if" "length" "local" "my" "next" "open" "or"
"package" "pop" "ref" "return" "stat" "sub" "tie"
"undef" "unless" "use" "while"))))
(defvar web-mode-erb-builtins
(regexp-opt
(append
(cdr (assoc "erb" web-mode-extra-builtins))
'("__callee__" "__dir__" "__method__"
"abort" "at_exit" "autoload" "autoload?"
"binding" "block_given?" "caller" "catch"
"eval" "exec" "exit" "exit!" "fail" "fork" "format"
"lambda" "load" "loop" "open"
"p" "print" "printf" "proc" "putc" "puts"
"raise" "rand" "readline" "readlines" "require" "require_relative"
"sleep" "spawn" "sprintf" "srand" "syscall" "system"
"throw" "trap" "warn"
"alias_method" "attr" "attr_accessor" "attr_reader" "attr_writer"
"define_method" "extend" "include" "module_function"
"prepend" "private" "protected" "public"
"refine" "using"
"error_message_on" "error_messages_for" "form" "input"
"auto_discovery_link_tag" "image_tag" "javascript_include_tag"
"stylesheet_link_tag" "image_path" "path_to_image"" "
"javascript_path" "path_to_javascript" "register_javascript_expansion"
"register_javascript_include_default" "register_stylesheet_expansion"
"stylesheet_path" "path_to_stylesheet" "atom_feed" "entry" "updated"
"benchmark" "cache" "capture" "content_for" "distance_of_time_in_words"
"distance_of_time_in_words_to_now" "time_ago_in_words" "date_select"
"datetime_select" "time_select" "select_date" "select_datetime"
"select_day" "select_hour" "select_minute" "select_month" "select_second"
"select_time" "select_year" "debug"
"check_box" "fields_for" "file_field" "form_for" "hidden_field"
"label" "password_field" "radio_button" "text_area" "text_field"
"check_box_tag" "field_set_tag" "file_field_tag" "form_tag"
"hidden_field_tag" "image_submit_tag" "label_tag" "password_field_tag"
"radio_button_tag" "select_tag" "submit_tag" "text_area_tag"
"text_field_tag"
"collection_select" "country_options_for_select" "country_select"
"option_groups_from_collection_for_select" "options_for_select"
"options_from_collection_for_select" "select"
"time_zone_options_for_select"
"time_zone_select" "button_to_function" "define_javascript_functions"
"escape_javascript" "javascript_tag" "link_to_function"" "
"number_to_currency" "number_to_human_size" "number_to_percentage"
"number_to_phone" "number_with_delimiter" "number_with_precision"
"evaluate_remote_response" "form_remote_for" "form_remote_tag"
"link_to_remote" "observe_field" "observe_field"
"periodically_call_remote"
"remote_form_for" "remote_function" "submit_to_remote" "update_page"
"update_page_tag" "dom_class" "dom_id" "partial_path" "sanitize"
"sanitize_css" "strip_links" "strip_tags"
"cdata_section" "content_tag" "escape_once" "tag"
"auto_link" "concat" "cycle" "excerpt" "highlight" "markdown" "pluralize"
"reset_cycle" "simple_format" "textilize" "textilize_without_paragraph"
"truncate" "word_wrap" "button_to" "current_page?" "link_to" "link_to_if"
"link_to_unless" "link_to_unless_current" "mail_to" "url_for"
"action_name" "atom_feed" "audio_path" "audio_tag"
"content_tag_for" "controller" "controller_name" "action_name"
"controller_path" "convert_to_model" "cookies" "csrf_meta_tag"
"csrf_meta_tags" "headers"
"current_cycle" "div_for" "email_field" "email_field_tag"
"favicon_link_tag" "flash" "l" "button_tag"
"grouped_collection_select" "grouped_options_for_select"
"image_alt" "j" "javascript_cdata_section"
"localize" "logger" "number_field"
"number_field_tag" "number_to_human" "params" "path_to_audio"
"path_to_video" "phone_field" "phone_field_tag" "provide"
"range_field" "range_field_tag" "raw" "render" "request"
"request_forgery_protection_token" "response" "safe_concat"
"safe_join" "search_field" "search_field_tag"
"session" "t" "telephone_field" "telephone_field_tag"
"time_tag" "translate" "url_field" "url_field_tag"
"url_options" "video_path" "video_tag"
))))
(defvar web-mode-asp-constants
(regexp-opt
(append
(cdr (assoc "asp" web-mode-extra-constants))
'("adAsyncExecute" "adAsyncFetch" "adAsyncFetchNonBlocking" "adCmdFile"
"adCmdStoredProc" "adCmdTable" "adCmdTableDirect" "adCmdText" "adCmdUnknown"
"adCmdUnspecified" "adExecuteNoRecords" "adExecuteRecord" "adExecuteStream"
"adLockBatchOptimistic" "adLockOptimistic" "adLockPessimistic"
"adLockReadOnly" "adLockUnspecified" "adOpenDynamic" "adOpenForwardOnly"
"adOpenKeyset" "adOpenStatic" "adOpenUnspecified" "adOptionUnspecified"
"Empty" "Nothing" "Null" "True" "False"
"vbBack" "vbCr" "vbCrLf" "vbFormFeed" "vbLf" "vbNewLine" "vbNullChar"
"vbNullString" "vbObjectError" "vbScript" "vbTab" "vbVerticalTab"))))
(defvar web-mode-asp-keywords
(regexp-opt
(append
(cdr (assoc "asp" web-mode-extra-keywords))
'("Abs" "And" "Array" "Asc" "Atn"
"CBool" "CByte" "CCur" "CDate" "CDbl" "CInt" "CLng" "CSng" "CStr"
"Call" "Case" "Chr" "Class" "Const" "Cos" "CreateObject"
"Date" "DateAdd" "DateDiff" "DatePart" "DateSerial" "DateValue"
"Day" "Dim" "Do"
"Each" "Else" "ElseIf" "End" "Erase" "Err" "Eval" "Exit" "Exp"
"Explicit"
"Filter" "Fix" "For" "FormatCurrency" "FormatDateTime"
"FormatNumber" "FormatPercent" "Function"
"GetLocale" "GetObject" "GetRef" "Hex" "Hour"
"If" "In" "InStr" "InStrRev" "InputBox" "Int" "IsArray" "IsDate"
"IsEmpty" "IsNull" "IsNumeric" "IsObject" "Join"
"LBound" "LCase" "LTrim" "Language" "Left" "Len" "Let"
"LoadPicture" "Log" "Loop"
"Mid" "Minute" "Month" "MonthName" "MsgBox"
"New" "Next" "Not" "Now"
"Oct" "On" "Option" "Or" "Preserve" "Private" "Public"
"RGB" "RTrim" "Redim" "Rem" "Replace" "Right" "Rnd" "Round"
"ScriptEngine" "ScriptEngineBuildVersion"
"ScriptEngineMajorVersion" "ScriptEngineMinorVersion"
"Second" "Select" "Set" "SetLocale" "Sgn" "Sin" "Space" "Split"
"Sqr" "StrComp" "StrReverse" "String" "Sub"
"Tan" "Then" "Time" "TimeSerial" "TimeValue" "Timer" "To" "Trim"
"TypeName"
"UBound" "UCase" "Until" "VarType"
"Weekday" "WeekdayName" "Wend" "With" "While" "Year"))))
(defvar web-mode-asp-types
(regexp-opt
(append
(cdr (assoc "asp" web-mode-extra-types))
'("Application" "ASPError" "Request" "Response" "Server" "Session"))))
(defvar web-mode-aspx-keywords
(regexp-opt
(append
(cdr (assoc "aspx" web-mode-extra-keywords))
'("case" "catch" "do" "else" "end" "for" "foreach" "function"
"if" "in" "include" "new" "package" "page" "return"
"tag" "throw" "throws" "try" "while"))))
(defvar web-mode-smarty-keywords
(regexp-opt '("as")))
(defvar web-mode-velocity-keywords
(eval-when-compile
(regexp-opt '("in"))))
(defvar web-mode-freemarker-keywords
(eval-when-compile
(regexp-opt '("as" "list"))))
(defvar web-mode-go-keywords
(eval-when-compile
(regexp-opt
'("define" "else" "end" "if" "pipeline" "range" "template" "with"))))
(defvar web-mode-go-functions
(eval-when-compile
(regexp-opt
'("and" "call" "html" "index" "js" "len" "not" "or"
"print" "printf" "println" "urlquery"))))
(defvar web-mode-closure-keywords
(eval-when-compile
(regexp-opt '("in" "and" "not" "or"))))
(defvar web-mode-django-control-blocks
(regexp-opt
'("assets" "autoescape" "block" "blocktrans" "cache" "call" "comment"
"elif" "else" "elseif" "elsif" "embed" "empty" "filter" "foreach" "for"
"ifchanged" "ifequal" "ifnotequal" "if" "with"
"macro" "draw" "random" "safe" "sandbox" "spaceless" "verbatim"
"form" "unless" "capture" ;; liquid
)
t))
(defvar web-mode-django-keywords
(eval-when-compile
(regexp-opt
'("and" "as" "autoescape" "block" "blocktrans" "break"
"cache" "call" "comment" "context" "continue" "csrf_token" "cycle"
"debug" "do" "embed" "empty" "else" "elseif" "elsif" "elif"
"endautoescape" "endblock" "endblocktrans" "endcomment"
"endcache" "endcall" "endembed" "endfilter" "endfor" "endif"
"endifchanged" "endifequal" "endifnotequal" "endmacro" "endrandom" "endraw"
"endsandbox" "endset" "endspaceless" "endtrans" "endverbatim" "endwith"
"extends" "filter" "firstof" "flush" "for" "from"
"if" "ifchanged" "ifequal" "ifnotequal" "ignore" "import"
"in" "include" "is" "load" "macro" "missing" "none" "not" "now" "or"
"pluralize" "random" "raw" "regroup" "trans"
"sandbox" "set" "spaceless" "ssi" "static" "templatetag" "trans"
"use" "url" "var" "verbatim" "widthratio" "with"
"assign" "capture" "endcapture" "case" "layout" "tablerow" "endtablerow" ;;liquid
"unless" "endunless" "form" "endform" ;; liquid
))))
(defvar web-mode-django-types
(eval-when-compile
(regexp-opt '("null" "empty" "false" "true"))))
(defvar web-mode-directives
(eval-when-compile
(regexp-opt
'("include" "page" "taglib"
"Assembly" "Control" "Implements" "Import"
"Master" "OutputCache" "Page" "Reference" "Register"))))
(defvar web-mode-template-toolkit-keywords
(regexp-opt
'("block" "call" "case" "catch" "clear" "default" "do"
"else" "elsif" "end" "filter" "final" "for"
"foreach" "get" "if" "in" "include" "insert" "is" "last"
"macro" "meta" "or" "perl" "process" "rawperl" "return"
"set" "stop" "switch" "tags" "throw" "try"
"unless" "use" "while" "wrapper")))
(defvar web-mode-perl-keywords
(regexp-opt
'("__DATA__" "__END__" "__FILE__" "__LINE__" "__PACKAGE__"
"and" "cmp" "continue" "CORE" "do" "else" "elsif" "eq" "exp"
"for" "foreach" "ge" "gt" "if" "le" "lock" "lt" "m" "ne" "no"
"or" "package" "q" "qq" "qr" "qw" "qx" "s" "sub"
"tr" "unless" "until" "while" "xor" "y"
"my")))
(defvar web-mode-javascript-keywords
(regexp-opt
(append
(cdr (assoc "javascript" web-mode-extra-keywords))
'("break" "case" "catch" "class" "const" "continue"
"debugger" "default" "delete" "do" "else" "enum" "eval"
"export" "extends" "finally" "for" "function" "if"
"implements" "import" "in" "instanceof" "interface" "let"
"new" "package" "private" "protected" "public"
"return" "static" "super" "switch" "throw"
"try" "typeof" "var" "void" "while" "with" "yield"))))
(defvar web-mode-javascript-constants
(regexp-opt
'("false" "null" "undefined" "Infinity" "NaN" "true" "arguments" "this")))
(defvar web-mode-razor-keywords
(regexp-opt
(append
(cdr (assoc "razor" web-mode-extra-keywords))
'("false" "true" "foreach" "if" "else" "in" "var" "for" "display"
"match" "case"
"Html"))))
(defvar web-mode-selector-font-lock-keywords
(list
'("$[[:alnum:]-]+" 0 'web-mode-css-variable-face)
(cons (concat "@\\(" web-mode-css-at-rules "\\)\\>")
'(1 'web-mode-css-at-rule-face))
'("\\<\\(all\|braille\\|embossed\\|handheld\\|print\\|projection\\|screen\\|speech\\|tty\\|tv\\|and\\|or\\)\\>" 1 'web-mode-keyword-face)
(cons (concat ":\\(" web-mode-css-pseudo-classes "\\)\\>")
'(1 'web-mode-css-pseudo-class-face))
'("[[:alnum:]-]+" 0 'web-mode-css-selector-face)
'("\\[.*?\\]\\|(.*?)" 0 nil t t)
'("url(\\(.+?\\))" 1 'web-mode-string-face)
))
(defvar web-mode-declaration-font-lock-keywords
(list
'("--[[:alnum:]-]+" 0 'web-mode-css-variable-face)
'("$[[:alnum:]-]+" 0 'web-mode-css-variable-face)
(cons (concat "@\\(" web-mode-css-at-rules "\\)\\>") '(1 'web-mode-css-at-rule-face))
'("url(\\([^)]+\\)" 1 'web-mode-string-face)
'("\\([[:alpha:]-]+\\)[ ]?:" 1 'web-mode-css-property-name-face)
'("\\([[:alpha:]-]+\\)[ ]?(" 1 'web-mode-css-function-face)
'("#[[:alnum:]]\\{1,6\\}" 0 'web-mode-css-color-face t t)
'("![ ]?important" 0 'web-mode-css-priority-face t t)
'("\\([[:alnum:]-.]+\\)[ ]+{" 1 'web-mode-css-selector-face)
))
(defvar web-mode-html-font-lock-keywords
(list
'("?[[:alnum:]]+[ >]\\|>" 0 'web-mode-html-tag-face t)
'(" \\([[:alnum:]-]+=\\)\\(\"[^\"]+\"\\)"
(1 'web-mode-html-attr-name-face)
(2 'web-mode-html-attr-value-face))
))
(defvar web-mode-javascript-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-javascript-keywords "\\)\\>")
'(0 'web-mode-keyword-face))
(cons (concat "\\<\\(" web-mode-javascript-constants "\\)\\>")
'(0 'web-mode-constant-face))
'("\\<\\(new\\|instanceof\\) \\([[:alnum:]_.]+\\)\\>" 2 'web-mode-type-face)
'("\\<\\([[:alnum:]_]+\\):[ ]*function[ ]*(" 1 'web-mode-function-name-face)
'("\\" 0 'web-mode-html-tag-bracket-face)
))
(defvar web-mode-dust-font-lock-keywords
(list
'("{[#:/?@><+^]\\([[:alpha:]_]+\\)" 1 'web-mode-block-control-face)
'(":\\([[:alpha:]]+\\)" 1 'web-mode-keyword-face)
'("\\<\\([[:alpha:]_]+=\\)\\(\"[^\"]*\"\\|[[:alnum:]_]*\\)"
(1 'web-mode-block-attr-name-face)
(2 'web-mode-block-attr-value-face))
'("\\\([[:alnum:]_]+\\)" 0 'web-mode-variable-name-face)
))
(defvar web-mode-template-toolkit-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-template-toolkit-keywords "\\)\\>")
'(1 'web-mode-keyword-face))
'("\\\([[:alpha:]][[:alnum:]_]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("\\\([[:alpha:]][[:alnum:]_]+\\)" 0 'web-mode-variable-name-face)
))
(defvar web-mode-smarty-font-lock-keywords
(list
(cons (concat "[ ]\\(" web-mode-smarty-keywords "\\)[ ]")
'(1 'web-mode-keyword-face))
(cons (concat (web-mode-engine-delimiter-open "smarty" "{") "/?\\([[:alpha:]_]+\\)")
'(1 'web-mode-block-control-face))
'("\\([}{]\\)" 0 'web-mode-block-delimiter-face)
'("\\<\\([$]\\)\\([[:alnum:]_]+\\)" (1 nil) (2 'web-mode-variable-name-face))
'("\\<\\(\\sw+\\)[ ]?(" 1 'web-mode-function-call-face)
'(" \\(\\sw+[ ]?=\\)" 1 'web-mode-param-name-face)
'(" \\(\\sw+\\)[ }]" 1 'web-mode-param-name-face)
'("|\\([[:alnum:]_]+\\)" 1 'web-mode-function-call-face)
'("\\(->\\)\\(\\sw+\\)" (1 nil) (2 'web-mode-variable-name-face))
'("[.]\\([[:alnum:]_-]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("[.]\\([[:alnum:]_]+\\)" 1 'web-mode-variable-name-face)
'("#\\([[:alnum:]_]+\\)#" 1 'web-mode-variable-name-face)
))
(defvar web-mode-velocity-font-lock-keywords
(list
'("#\\([[:alpha:]_]+\\)\\>" (1 'web-mode-block-control-face))
(cons (concat "[ ]\\(" web-mode-velocity-keywords "\\)[ ]") '(1 'web-mode-keyword-face t t))
'("#macro([ ]*\\([[:alpha:]]+\\)[ ]+" 1 'web-mode-function-name-face)
'("[.]\\([[:alnum:]_-]+\\)" 1 'web-mode-variable-name-face)
'("\\<\\($[!]?[{]?\\)\\([[:alnum:]_-]+\\)[}]?" (1 nil) (2 'web-mode-variable-name-face))
))
(defvar web-mode-mako-tag-font-lock-keywords
(list
'("?%\\([[:alpha:]:]+\\)" 1 'web-mode-block-control-face)
'("\\<\\([[:alpha:]]+=\\)\\(\"[^\"]*\"\\)"
(1 'web-mode-block-attr-name-face t t)
(2 'web-mode-block-attr-value-face t t))
))
(defvar web-mode-mako-block-font-lock-keywords
(list
'("\\<\\(\\sw+\\)[ ]?(" 1 'web-mode-function-call-face)
(cons (concat "\\<\\(" web-mode-python-constants "\\)\\>") '(1 'web-mode-constant-face))
(cons (concat "\\<\\(" web-mode-python-keywords "\\)\\>") '(1 'web-mode-keyword-face))
(cons (concat "\\<\\(endfor\\|endif\\|endwhile\\)\\>") '(1 'web-mode-keyword-face))
))
(defvar web-mode-web2py-font-lock-keywords
(list
'("\\<\\(\\sw+\\)[ ]?(" 1 'web-mode-function-call-face)
(cons (concat "\\<\\(" web-mode-python-constants "\\)\\>") '(1 'web-mode-constant-face))
(cons (concat "\\<\\(" web-mode-python-keywords "\\)\\>") '(1 'web-mode-keyword-face))
(cons (concat "\\<\\(block\\|extend\\|super\\|end\\|include\\)\\>") '(1 'web-mode-keyword-face))
))
(defvar web-mode-django-expr-font-lock-keywords
(list
'("|[ ]?\\([[:alpha:]_]+\\)\\>" 1 'web-mode-function-call-face)
(cons (concat "\\<\\(" web-mode-django-types "\\)\\>") '(1 'web-mode-type-face))
'("\\<\\([[:alpha:]_]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("[[:alnum:]_]+" 0 'web-mode-variable-name-face)
))
(defvar web-mode-django-code-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-django-keywords "\\)\\>") '(1 'web-mode-keyword-face))
(cons (concat "\\<\\(" web-mode-django-types "\\)\\>") '(1 'web-mode-type-face))
'("|[ ]?\\([[:alpha:]_]+\\)\\>" 1 'web-mode-function-call-face)
'("\\<\\([[:alpha:]_]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("[[:alnum:]_]+" 0 'web-mode-variable-name-face)
))
(defvar web-mode-ctemplate-font-lock-keywords
(list
'("{{[#/>][ ]*\\([[:alnum:]_-]+\\)" 1 'web-mode-block-control-face)
;; '(" \\([[:alnum:]_]\\)\\(=\\)\"" (1 'web-mode-variable-name-face))
'("[[:alnum:]_]" 0 'web-mode-variable-name-face)
'("[ ]+\\([[:alnum:]_]+=\\)" 1 'web-mode-param-name-face t t)
'("[:=]\\([[:alpha:]_]+\\)" 1 'web-mode-function-call-face t t)
))
(defvar web-mode-razor-font-lock-keywords
(list
'("@\\([[:alnum:]_.]+\\)[ ]*[({]" 1 'web-mode-block-control-face)
(cons (concat "\\<\\(" web-mode-razor-keywords "\\)\\>") '(1 'web-mode-keyword-face))
;; '("\\([[:alnum:]]+\\):" 1 'web-mode-symbol-face)
;; '("@\\([[:alnum:]_.]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("@\\([[:alnum:]_.]+\\)" 1 'web-mode-variable-name-face)
;; '("<\\([[:alnum:]_]+\\)>" 1 'web-mode-type-face)
;; '("\\<\\([[:alnum:].]+\\)[ ]+[{[:alpha:]]+" 1 'web-mode-type-face)
;; '("[[:alnum:]_]+" 0 'web-mode-variable-name-face)
))
(defvar web-mode-closure-font-lock-keywords
(list
'("{/?\\([[:alpha:]]+\\)" 1 'web-mode-block-control-face)
'("{param[ ]+\\([[:alnum:]]+\\)" 1 'web-mode-symbol-face)
'("\\<\\(true\\|false\\|null\\)\\>" 1 'web-mode-type-face)
(cons (concat "\\<\\(" web-mode-closure-keywords "\\)\\>")
'(1 'web-mode-keyword-face))
'("{\\(alias\\|call\\|delcall\\|delpackage\\|deltemplate\\|namespace\\|template\\)[ ]+\\([[:alnum:].]+\\)" 2 'web-mode-constant-face)
'("\\(allowemptydefault\\|data\\|desc\\|meaning\\|autoescape\\|private\\|variant\\)=" 0 'web-mode-block-attr-name-face)
'("|\\([[:alpha:]]+\\)" 1 'web-mode-function-call-face)
'("\\<\\([[:alnum:]]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("$\\([[:alnum:]._]+\\)" 1 'web-mode-variable-name-face)
))
(defvar web-mode-go-font-lock-keywords
(list
'("{{\\([[:alpha:]]+\\)" 1 'web-mode-block-control-face)
(cons (concat "\\<\\(" web-mode-go-keywords "\\)\\>")
'(1 'web-mode-keyword-face))
(cons (concat "\\<\\(" web-mode-go-functions "\\)\\>")
'(1 'web-mode-function-call-face))
'("[$.]\\([[:alnum:]_]+\\)" 1 'web-mode-variable-name-face t t)
))
(defvar web-mode-expression-font-lock-keywords
(list
'("[[:alpha:]_]" 0 'web-mode-variable-name-face)
))
(defvar web-mode-angular-font-lock-keywords
(list
'("[[:alpha:]_]" 0 'web-mode-variable-name-face)
))
(defvar web-mode-underscore-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-javascript-keywords "\\)\\>")
'(0 'web-mode-keyword-face))
'("\\<\\(_\.[[:alpha:]]+\\)(" 1 'web-mode-function-call-face)
'("\\" 1 'web-mode-type-face)
'("\\<\\([[:alnum:]_]+\\):[ ]*function[ ]*(" 1 'web-mode-function-name-face)
'("\\<\\(var\\)\\>[ ]+\\([[:alnum:]_]+\\)"
(1 'web-mode-keyword-face)
(2 'web-mode-variable-name-face))
))
(defvar web-mode-engine-tag-font-lock-keywords
(list
'("?\\([[:alpha:]]+\\(?:Template\\|:[[:alpha:]-]+\\)\\)" 1 'web-mode-block-control-face)
'("\\<\\([[:alpha:]-]+=\\)\\(\"[^\"]*\"\\)"
(1 'web-mode-block-attr-name-face t t)
(2 'web-mode-block-attr-value-face t t))
))
(defvar web-mode-jsp-font-lock-keywords
(list
'("\\(throws\\|new\\|extends\\)[ ]+\\([[:alnum:].]+\\)" 2 'web-mode-type-face)
(cons (concat "\\<\\(" web-mode-jsp-keywords "\\)\\>") '(0 'web-mode-keyword-face))
'("\\<\\([[:alnum:]._]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("@\\(\\sw*\\)" 1 'web-mode-variable-name-face)
'("\\<\\([[:alnum:].]+\\)[ ]+[{[:alpha:]]+" 1 'web-mode-type-face)
))
(defvar web-mode-asp-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-asp-keywords "\\)\\>")
'(0 'web-mode-keyword-face))
(cons (concat "\\<\\(" web-mode-asp-types "\\)\\>")
'(0 'web-mode-type-face))
(cons (concat "\\<\\(" web-mode-asp-constants "\\)\\>")
'(0 'web-mode-constant-face))
'("\\(Class\\|new\\) \\([[:alnum:]_]+\\)" 2 'web-mode-type-face)
'("Const \\([[:alnum:]_]+\\)" 1 'web-mode-constant-face)
'("\\"
(0 'web-mode-keyword-face)
("[[:alnum:]_]+" nil nil (0 'web-mode-variable-name-face)))
'("\\<\\(public\\|private\\|sub\\|function\\)\\> \\([[:alnum:]_]+\\)[ ]*("
2 'web-mode-function-name-face)
'("\\<\\(public\\|private\\|dim\\)\\> \\([[:alnum:]_]+\\)"
2 'web-mode-variable-name-face)
))
(defvar web-mode-aspx-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-aspx-keywords "\\)\\>") '(0 'web-mode-keyword-face))
'("\\<\\([[:alnum:].]+\\)[ ]+[[:alpha:]]+" 1 'web-mode-type-face)
))
(defvar web-mode-uel-font-lock-keywords
(list
'("[$#{]{\\|}" 0 'web-mode-preprocessor-face)
'("\\([[:alpha:]_]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("|[ ]*\\(trim\\|x\\|u\\)" 1 'web-mode-function-call-face)
'("[[:alpha:]_]" 0 'web-mode-variable-name-face)
))
(defvar web-mode-php-var-interpolation-font-lock-keywords
(list
'("[[:alpha:]_]" 0 'web-mode-variable-name-face)
'("\".+\"\\|'.*'" 0 'web-mode-string-face)
))
(defvar web-mode-freemarker-square-font-lock-keywords
(list
'("\\[/?[#@]\\([[:alpha:]_.]*\\)" 1 'web-mode-block-control-face)
'("#\\(macro\\|function\\) \\([[:alpha:]]+\\)" 2 'web-mode-function-name-face)
(cons (concat "\\<\\(" web-mode-freemarker-keywords "\\)\\>")
'(1 'web-mode-keyword-face))
'("\\<\\([[:alnum:]._]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("[[:alpha:]]\\([[:alnum:]_]+\\)?" 0 'web-mode-variable-name-face)
))
(defvar web-mode-freemarker-font-lock-keywords
(list
'("?[#@]\\([[:alpha:]_.]*\\)" 1 'web-mode-block-control-face)
'("#\\(macro\\|function\\) \\([[:alpha:]]+\\)" 2 'web-mode-function-name-face)
(cons (concat "\\<\\(" web-mode-freemarker-keywords "\\)\\>") '(1 'web-mode-keyword-face))
'("\\<\\([[:alnum:]._]+\\)[ ]?(" 1 'web-mode-function-call-face)
'("[[:alpha:]]\\([[:alnum:]_]+\\)?" 0 'web-mode-variable-name-face)
))
(defvar web-mode-directive-font-lock-keywords
(list
'("<%@[ ]*\\([[:alpha:]]+\\)[ ]+" 1 'web-mode-block-control-face)
'("\\<\\([[:alpha:]]+=\\)\\(\"[^\"]*\"\\)"
(1 'web-mode-block-attr-name-face t t)
(2 'web-mode-block-attr-value-face t t))
))
(defvar web-mode-erb-font-lock-keywords
(list
'("[^:]\\(:[[:alnum:]_]+\\)" 1 'web-mode-symbol-face)
'("\\([[:alnum:]_]+:\\)[ ]+" 1 'web-mode-symbol-face)
(cons (concat "\\<\\(" web-mode-erb-builtins "\\)\\>") '(0 'web-mode-builtin-face))
(cons (concat "\\<\\(" web-mode-erb-keywords "\\)\\>") '(0 'web-mode-keyword-face))
'("\\<\\(self\\|true\\|false\\|nil\\)\\>" 0 'web-mode-variable-name-face)
'("[@$]@?\\([[:alnum:]_]+\\)" 0 'web-mode-variable-name-face)
'("class[ ]+\\([[:alnum:]_]+\\)" 1 'web-mode-type-face)
'("def[ ]+\\([[:alnum:]_]+\\)" 1 'web-mode-function-name-face)
'("\\(?:\\_<\\|::\\)\\([A-Z]+[[:alnum:]_]+\\)" 1 (unless (eq (char-after) ?\() 'web-mode-type-face))
'("/[^/]+/" 0 'web-mode-string-face)
))
(defvar web-mode-python-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-python-keywords "\\)\\>") '(0 'web-mode-keyword-face))
))
(defvar web-mode-mason-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-mason-keywords "\\)\\>")
'(0 'web-mode-keyword-face))
'("sub[ ]+\\([[:alnum:]_]+\\)" 1 'web-mode-function-name-face)
'(" | \\([hun]+\\) " 1 'web-mode-function-name-face)
'("\\<\\([[:alnum:]_]+\\)[ ]?::" 1 'web-mode-type-face)
'("\\([@]\\)\\([[:alnum:]#_]*\\)" (1 nil) (2 'web-mode-variable-name-face))
'("\\<\\([$%]\\)\\([[:alnum:]@#_]*\\)" (1 nil) (2 'web-mode-variable-name-face))
'("{\\([[:alnum:]_]+\\)}" 1 'web-mode-variable-name-face)
'("\\<\\(\\sw+\\)[ ]?(" 1 'web-mode-function-call-face)
'("[[:alnum:]_][ ]?::[ ]?\\([[:alnum:]_]+\\)" 1 'web-mode-variable-name-face)
'("->[ ]?\\([[:alnum:]_]+\\)" 1 'web-mode-variable-name-face)
'("\\(method\\|def\\)" 1 'web-mode-block-control-face)
'("\\(?:method\\|def\\) \\([[:alnum:]._]+\\)" 1 'web-mode-function-name-face)
))
(defvar web-mode-mojolicious-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-perl-keywords "\\)\\>") '(0 'web-mode-keyword-face))
'("\\<\\([$]\\)\\([[:alnum:]_]*\\)" (1 nil) (2 'web-mode-variable-name-face))
))
(defvar web-mode-lsp-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-lsp-keywords "\\)\\>") '(0 'web-mode-keyword-face))
(cons (concat "\\<\\(" web-mode-lsp-constants "\\)\\>") '(1 'web-mode-constant-face))
'("[ ]\\(:[[:alnum:]-_]+\\)" 1 'web-mode-symbol-face)
))
(defvar web-mode-php-font-lock-keywords
(list
(cons (concat "\\<\\(" web-mode-php-keywords "\\)\\>") '(0 'web-mode-keyword-face))
(cons (concat "(\\<\\(" web-mode-php-types "\\)\\>") '(1 'web-mode-type-face))
(cons (concat "\\<\\(" web-mode-php-constants "\\)\\>") '(0 'web-mode-constant-face))
'("function[ ]+\\([[:alnum:]_]+\\)" 1 'web-mode-function-name-face)
'("\\<\\(\\sw+\\)[ ]?(" 1 'web-mode-function-call-face)
'("[[:alnum:]_][ ]?::[ ]?\\([[:alnum:]_]+\\)" 1 'web-mode-constant-face)
'("->[ ]?\\([[:alnum:]_]+\\)" 1 'web-mode-variable-name-face)
'("\\<\\([[:alnum:]_]+\\)[ ]?::" 1 'web-mode-type-face)
'("\\<\\(instanceof\\|class\\|extends\\|new\\)[ ]+\\([[:alnum:]_]+\\)" 2 'web-mode-type-face)
'("\\<\\([$]\\)\\([[:alnum:]_]*\\)" (1 nil) (2 'web-mode-variable-name-face))
))
(defvar web-mode-latex-font-lock-keywords
(list
'("[[:alnum:]_]+" 0 'web-mode-function-name-face t t)
))
(defvar web-mode-blade-font-lock-keywords
(append
(list
'("@\\([[:alpha:]_]+\\)" (1 'web-mode-block-control-face)))
web-mode-php-font-lock-keywords))
(defvar web-mode-engines-font-lock-keywords
'(("angular" . web-mode-angular-font-lock-keywords)
;; ("asp" . web-mode-asp-font-lock-keywords)
("blade" . web-mode-blade-font-lock-keywords)
("closure" . web-mode-closure-font-lock-keywords)
("ctemplate" . web-mode-ctemplate-font-lock-keywords)
("dust" . web-mode-dust-font-lock-keywords)
("erb" . web-mode-erb-font-lock-keywords)
("go" . web-mode-go-font-lock-keywords)
("lsp" . web-mode-lsp-font-lock-keywords)
;; ("mason" . web-mode-mason-font-lock-keywords)
("mojolicious" . web-mode-mojolicious-font-lock-keywords)
("php" . web-mode-php-font-lock-keywords)
("python" . web-mode-python-font-lock-keywords)
("razor" . web-mode-razor-font-lock-keywords)
("smarty" . web-mode-smarty-font-lock-keywords)
("template-toolkit" . web-mode-template-toolkit-font-lock-keywords)
("underscore" . web-mode-underscore-font-lock-keywords)
("web2py" . web-mode-web2py-font-lock-keywords)
("velocity" . web-mode-velocity-font-lock-keywords))
"Engines font-lock keywords")
(defvar web-mode-before-auto-complete-hooks nil
"List of functions to run before triggering the auto-complete library.
Auto-complete sources will sometimes need some tweaking to work
nicely with web-mode. This hook gives users the chance to adjust
the environment as needed for ac-sources, right before they're used.")
(defvar web-mode-ac-sources-alist nil
"alist mapping language names to ac-sources for that language.")
(defvar web-mode-syntax-table
(let ((table (make-syntax-table)))
(modify-syntax-entry ?_ "w" table)
(modify-syntax-entry ?< "." table)
(modify-syntax-entry ?> "." table)
(modify-syntax-entry ?& "." table)
(modify-syntax-entry ?/ "." table)
(modify-syntax-entry ?= "." table)
(modify-syntax-entry ?% "." table)
table)
"Syntax table used to reveal whitespaces.")
(defvar web-mode-map
(let ((map (make-sparse-keymap)))
(define-key map [menu-bar wm] (cons "Web-Mode" (make-sparse-keymap)))
(define-key map [menu-bar wm dom] (cons "Dom" (make-sparse-keymap)))
(define-key map [menu-bar wm blk] (cons "Block" (make-sparse-keymap)))
(define-key map [menu-bar wm attr] (cons "Html Attr" (make-sparse-keymap)))
(define-key map [menu-bar wm tag] (cons "Html Tag" (make-sparse-keymap)))
(define-key map [menu-bar wm elt] (cons "Html Element" (make-sparse-keymap)))
(define-key map [menu-bar wm sep-1] '(menu-item "--"))
(define-key map [menu-bar wm dom dom-xpa] '(menu-item "XPath" web-mode-dom-xpath))
(define-key map [menu-bar wm dom dom-tra] '(menu-item "Traverse" web-mode-dom-traverse))
(define-key map [menu-bar wm dom dom-err] '(menu-item "Show error(s)" web-mode-dom-errors-show))
(define-key map [menu-bar wm dom dom-ent] '(menu-item "Replace HTML entities" web-mode-dom-entities-replace))
(define-key map [menu-bar wm dom dom-quo] '(menu-item "Replace dumb quotes" web-mode-dom-quotes-replace))
(define-key map [menu-bar wm dom dom-apo] '(menu-item "Replace apostrophes" web-mode-dom-apostrophes-replace))
(define-key map [menu-bar wm dom dom-nor] '(menu-item "Normalise" web-mode-dom-normalize))
(define-key map [menu-bar wm blk blk-sel] '(menu-item "Select" web-mode-block-select))
(define-key map [menu-bar wm blk blk-pre] '(menu-item "Previous" web-mode-block-previous))
(define-key map [menu-bar wm blk blk-nex] '(menu-item "Next" web-mode-block-next))
(define-key map [menu-bar wm blk blk-kil] '(menu-item "Kill" web-mode-block-kill))
(define-key map [menu-bar wm blk blk-end] '(menu-item "End" web-mode-block-end))
(define-key map [menu-bar wm blk blk-clo] '(menu-item "Close" web-mode-block-close))
(define-key map [menu-bar wm blk blk-beg] '(menu-item "Beginning" web-mode-block-beginning))
(define-key map [menu-bar wm attr attr-end] '(menu-item "End" web-mode-attribute-end))
(define-key map [menu-bar wm attr attr-beg] '(menu-item "Beginning" web-mode-attribute-beginning))
(define-key map [menu-bar wm attr attr-sel] '(menu-item "Select" web-mode-attribute-select))
(define-key map [menu-bar wm attr attr-nex] '(menu-item "Next" web-mode-attribute-next))
(define-key map [menu-bar wm attr attr-tra] '(menu-item "Next" web-mode-attribute-transpose))
(define-key map [menu-bar wm tag tag-beg] '(menu-item "Sort Attributes" web-mode-tag-attributes-sort))
(define-key map [menu-bar wm tag tag-sel] '(menu-item "Select" web-mode-tag-select))
(define-key map [menu-bar wm tag tag-pre] '(menu-item "Previous" web-mode-tag-previous))
(define-key map [menu-bar wm tag tag-nex] '(menu-item "Next" web-mode-tag-next))
(define-key map [menu-bar wm tag tag-mat] '(menu-item "Match" web-mode-tag-match))
(define-key map [menu-bar wm tag tag-end] '(menu-item "End" web-mode-tag-end))
(define-key map [menu-bar wm tag tag-beg] '(menu-item "Beginning" web-mode-tag-beginning))
(define-key map [menu-bar wm elt elt-wra] '(menu-item "Wrap" web-mode-element-wrap))
(define-key map [menu-bar wm elt elt-van] '(menu-item "Vanish" web-mode-element-vanish))
(define-key map [menu-bar wm elt elt-exc] '(menu-item "Transpose" web-mode-element-transpose))
(define-key map [menu-bar wm elt elt-sel] '(menu-item "Select" web-mode-element-select))
(define-key map [menu-bar wm elt elt-ren] '(menu-item "Rename" web-mode-element-rename))
(define-key map [menu-bar wm elt elt-pre] '(menu-item "Previous" web-mode-element-previous))
(define-key map [menu-bar wm elt elt-par] '(menu-item "Parent" web-mode-element-parent))
(define-key map [menu-bar wm elt elt-nex] '(menu-item "Next" web-mode-element-next))
(define-key map [menu-bar wm elt elt-mut] '(menu-item "Mute blanks" web-mode-element-mute-blanks))
(define-key map [menu-bar wm elt elt-del] '(menu-item "Kill" web-mode-element-kill))
(define-key map [menu-bar wm elt elt-end] '(menu-item "End" web-mode-element-end))
(define-key map [menu-bar wm elt elt-inn] '(menu-item "Content (select)" web-mode-element-content-select))
(define-key map [menu-bar wm elt elt-clo] '(menu-item "Close" web-mode-element-close))
(define-key map [menu-bar wm elt elt-dup] '(menu-item "Clone" web-mode-element-clone))
(define-key map [menu-bar wm elt elt-cfo] '(menu-item "Children fold" web-mode-element-children-fold-or-unfold))
(define-key map [menu-bar wm elt elt-chi] '(menu-item "Child" web-mode-element-child))
(define-key map [menu-bar wm elt elt-beg] '(menu-item "Beginning" web-mode-element-beginning))
(define-key map [menu-bar wm fol] '(menu-item "Fold/Unfold" web-mode-fold-or-unfold))
(define-key map [menu-bar wm hig] '(menu-item "Highlight buffer" web-mode-buffer-highlight))
(define-key map [menu-bar wm ind] '(menu-item "Indent buffer" web-mode-buffer-indent))
(define-key map [menu-bar wm nav] '(menu-item "Tag/Block navigation" web-mode-navigate))
(define-key map [menu-bar wm exp] '(menu-item "Mark and Expand" web-mode-mark-and-expand))
(define-key map [menu-bar wm spa] '(menu-item "Toggle whitespaces" web-mode-whitespaces-show))
(define-key map [menu-bar wm sni] '(menu-item "Insert snippet" web-mode-snippet-insert))
;;--------------------------------------------------------------------------
;; "C-c letter" are reserved for users
(define-key map (kbd "C-c C-a b") 'web-mode-attribute-beginning)
(define-key map (kbd "C-c C-a e") 'web-mode-attribute-end)
(define-key map (kbd "C-c C-a s") 'web-mode-attribute-select)
(define-key map (kbd "C-c C-a t") 'web-mode-attribute-transpose)
(define-key map (kbd "C-c C-a n") 'web-mode-attribute-next)
(define-key map (kbd "C-c C-b c") 'web-mode-block-close)
(define-key map (kbd "C-c C-b b") 'web-mode-block-beginning)
(define-key map (kbd "C-c C-b e") 'web-mode-block-end)
(define-key map (kbd "C-c C-b k") 'web-mode-block-kill)
(define-key map (kbd "C-c C-b n") 'web-mode-block-next)
(define-key map (kbd "C-c C-b p") 'web-mode-block-previous)
(define-key map (kbd "C-c C-b s") 'web-mode-block-select)
(define-key map (kbd "C-c C-d a") 'web-mode-dom-apostrophes-replace)
(define-key map (kbd "C-c C-d n") 'web-mode-dom-normalize)
(define-key map (kbd "C-c C-d d") 'web-mode-dom-errors-show)
(define-key map (kbd "C-c C-d e") 'web-mode-dom-entities-replace)
(define-key map (kbd "C-c C-d q") 'web-mode-dom-quotes-replace)
(define-key map (kbd "C-c C-d t") 'web-mode-dom-traverse)
(define-key map (kbd "C-c C-d x") 'web-mode-dom-xpath)
(define-key map (kbd "C-c C-e b") 'web-mode-element-beginning)
(define-key map (kbd "C-c C-e c") 'web-mode-element-clone)
(define-key map (kbd "C-c C-e d") 'web-mode-element-child)
(define-key map (kbd "C-c C-e e") 'web-mode-element-end)
(define-key map (kbd "C-c C-e f") 'web-mode-element-children-fold-or-unfold)
(define-key map (kbd "C-c C-e i") 'web-mode-element-content-select)
(define-key map (kbd "C-c C-e k") 'web-mode-element-kill)
(define-key map (kbd "C-c C-e m") 'web-mode-element-mute-blanks)
(define-key map (kbd "C-c C-e n") 'web-mode-element-next)
(define-key map (kbd "C-c C-e p") 'web-mode-element-previous)
(define-key map (kbd "C-c C-e r") 'web-mode-element-rename)
(define-key map (kbd "C-c C-e s") 'web-mode-element-select)
(define-key map (kbd "C-c C-e t") 'web-mode-element-transpose)
(define-key map (kbd "C-c C-e u") 'web-mode-element-parent)
(define-key map (kbd "C-c C-e v") 'web-mode-element-vanish)
(define-key map (kbd "C-c C-e w") 'web-mode-element-wrap)
(define-key map (kbd "C-c C-t a") 'web-mode-tag-attributes-sort)
(define-key map (kbd "C-c C-t b") 'web-mode-tag-beginning)
(define-key map (kbd "C-c C-t e") 'web-mode-tag-end)
(define-key map (kbd "C-c C-t m") 'web-mode-tag-match)
(define-key map (kbd "C-c C-t n") 'web-mode-tag-next)
(define-key map (kbd "C-c C-t p") 'web-mode-tag-previous)
(define-key map (kbd "C-c C-t s") 'web-mode-tag-select)
;;--------------------------------------------------------------------------
(define-key map (kbd "M-;") 'web-mode-comment-or-uncomment)
;;C-c C-a : attribute
;;C-c C-b : block
;;C-c C-d : dom
;;C-c C-e : element
(define-key map (kbd "C-c C-f") 'web-mode-fold-or-unfold)
(define-key map (kbd "C-c C-h") 'web-mode-buffer-highlight)
(define-key map (kbd "C-c C-i") 'web-mode-buffer-indent)
(define-key map (kbd "C-c C-j") 'web-mode-jshint)
(define-key map (kbd "C-c C-m") 'web-mode-mark-and-expand)
(define-key map (kbd "C-c C-n") 'web-mode-navigate)
(define-key map (kbd "C-c C-s") 'web-mode-snippet-insert)
;;C-c C-t : tag
(define-key map (kbd "C-c C-w") 'web-mode-whitespaces-show)
map)
"Keymap for `web-mode'.")
;;---- COMPATIBILITY -----------------------------------------------------------
(eval-and-compile
(defalias 'web-mode-prog-mode
(if (fboundp 'prog-mode) 'prog-mode 'fundamental-mode))
(if (fboundp 'with-silent-modifications)
(defalias 'web-mode-with-silent-modifications 'with-silent-modifications)
(defmacro web-mode-with-silent-modifications (&rest body)
"For compatibility with Emacs pre 23.3"
`(let ((old-modified-p (buffer-modified-p))
(inhibit-modification-hooks t)
(buffer-undo-list t))
(unwind-protect
,@body
(set-buffer-modified-p old-modified-p)))))
(defun web-mode-buffer-narrowed-p ()
"For compatibility with Emacs pre 24.3."
(if (fboundp 'buffer-narrowed-p)
(buffer-narrowed-p)
(/= (- (point-max) (point-min)) (buffer-size))))
) ;eval-and-compile
;;---- MAJOR MODE --------------------------------------------------------------
;;;###autoload
(define-derived-mode web-mode web-mode-prog-mode "Web"
"Major mode for editing web templates."
(make-local-variable 'web-mode-auto-pairs)
(make-local-variable 'web-mode-block-regexp)
(make-local-variable 'web-mode-engine-open-delimiter-regexps)
(make-local-variable 'web-mode-change-beg)
(make-local-variable 'web-mode-change-end)
(make-local-variable 'web-mode-column-overlays)
(make-local-variable 'web-mode-comment-style)
(make-local-variable 'web-mode-content-type)
(make-local-variable 'web-mode-display-table)
(make-local-variable 'web-mode-enable-block-face)
(make-local-variable 'web-mode-enable-inlays)
(make-local-variable 'web-mode-enable-part-face)
(make-local-variable 'web-mode-end-tag-overlay)
(make-local-variable 'web-mode-engine)
(make-local-variable 'web-mode-engine-attr-regexp)
(make-local-variable 'web-mode-engine-file-regexps)
(make-local-variable 'web-mode-engine-token-regexp)
(make-local-variable 'web-mode-expand-initial-pos)
(make-local-variable 'web-mode-expand-previous-state)
(make-local-variable 'web-mode-hl-line-mode-flag)
(make-local-variable 'web-mode-indent-style)
(make-local-variable 'web-mode-is-scratch)
(make-local-variable 'web-mode-jshint-errors)
(make-local-variable 'web-mode-start-tag-overlay)
(make-local-variable 'web-mode-time)
(make-local-variable 'fill-paragraph-function)
(make-local-variable 'font-lock-beg)
(make-local-variable 'font-lock-defaults)
(make-local-variable 'font-lock-end)
(make-local-variable 'font-lock-support-mode)
(make-local-variable 'imenu-case-fold-search)
(make-local-variable 'imenu-create-index-function)
(make-local-variable 'imenu-generic-expression)
(make-local-variable 'indent-line-function)
(make-local-variable 'parse-sexp-lookup-properties)
(make-local-variable 'text-property-default-nonsticky)
(make-local-variable 'yank-excluded-properties)
;; required for block-code-beg|end
(add-to-list 'text-property-default-nonsticky '(block-token . t))
;; (add-to-list 'text-property-default-nonsticky '((block-token . t)
;; (tag-end . t)))
(setq fill-paragraph-function 'web-mode-fill-paragraph
font-lock-defaults '(web-mode-font-lock-keywords t)
font-lock-support-mode nil
imenu-case-fold-search t
imenu-create-index-function 'web-mode-imenu-index
indent-line-function 'web-mode-indent-line
parse-sexp-lookup-properties t
yank-excluded-properties t)
(if (and (fboundp 'global-hl-line-mode)
global-hl-line-mode)
(setq web-mode-hl-line-mode-flag t))
(add-hook 'after-change-functions 'web-mode-on-after-change nil t)
(add-hook 'after-save-hook 'web-mode-on-after-save t t)
(add-hook 'change-major-mode-hook 'web-mode-on-exit nil t)
(add-hook 'post-command-hook 'web-mode-on-post-command nil t)
(cond
((boundp 'yas-after-exit-snippet-hook)
(add-hook 'yas-after-exit-snippet-hook
'web-mode-yasnippet-exit-hook
t t))
((boundp 'yas/after-exit-snippet-hook)
(add-hook 'yas/after-exit-snippet-hook
'web-mode-yasnippet-exit-hook
t t))
)
(when web-mode-enable-whitespace-fontification
(web-mode-whitespaces-on))
(web-mode-guess-engine-and-content-type)
(setq web-mode-change-beg (point-min)
web-mode-change-end (point-max))
(when (> (point-max) 256000)
(web-mode-buffer-highlight))
;;(web-mode-trace "buffer loaded")
)
;;---- DEFUNS ------------------------------------------------------------------
(defun web-mode-scan-region (beg end &optional content-type)
"Identify nodes/parts/blocks and syntactic symbols (strings/comments)."
;; (message "scan-region: beg(%d) end(%d) content-type(%S)" beg end content-type)
(web-mode-with-silent-modifications
(save-excursion
(save-restriction
(save-match-data
(let ((inhibit-point-motion-hooks t)
(inhibit-quit t))
(remove-text-properties beg end web-mode-scan-properties)
;; (remove-text-properties beg end '(face nil))
(cond
((and content-type (string= content-type "php"))
;; (web-mode-block-scan beg end)
)
((and content-type
(member content-type '("javascript" "json" "jsx" "css")))
(put-text-property beg end 'part-side
(cond
((string= content-type "javascript") 'javascript)
((string= content-type "json") 'json)
((string= content-type "jsx") 'jsx)
((string= content-type "css") 'css)
))
(web-mode-scan-blocks beg end)
(web-mode-part-scan beg end content-type)
)
((member web-mode-content-type '("javascript" "json" "jsx" "css"))
(web-mode-scan-blocks beg end)
(web-mode-part-scan beg end)
)
((string= web-mode-engine "none")
(web-mode-scan-elements beg end)
(web-mode-process-parts beg end 'web-mode-part-scan)
)
(t
(web-mode-scan-blocks beg end)
(web-mode-scan-elements beg end)
(web-mode-process-parts beg end 'web-mode-part-scan)
)
) ;cond
(cons beg end)
))))))
(defun web-mode-scan-blocks (reg-beg reg-end)
"Identifies blocks (with block-side, block-beg, block-end text properties)."
(save-excursion
(let ((i 0) open close closing-string start sub1 sub2 pos tagopen l tmp delim-open delim-close part-beg part-end tagclose)
(goto-char reg-beg)
;; (message "%S: %Sx%S" (point) reg-beg reg-end)
;; (message "regexp=%S" web-mode-block-regexp)
(while (and (< i 1200)
(> reg-end (point))
web-mode-block-regexp
(re-search-forward web-mode-block-regexp reg-end t)
(not (eobp)))
(setq i (1+ i)
closing-string nil
close nil
tagopen (match-string 0)
open (match-beginning 0)
delim-open nil
delim-close nil
pos nil)
(setq l (length tagopen))
(when (member (string-to-char tagopen) '(?\s ?\t))
(setq tagopen (replace-regexp-in-string "\\`[ \t]*" "" tagopen))
;; (message "tagopen=%s (%S)" tagopen (point))
(setq open (+ open (- l (length tagopen))))
(setq l (length tagopen))
)
;; (message "tagopen=%s (%S)" tagopen (point))
(setq sub1 (substring tagopen 0 1)
sub2 (substring tagopen 0 (if (>= l 2) 2 1)))
(cond
((string= web-mode-engine "php")
(unless (member (char-after) '(?x ?X))
(setq closing-string '("<\\?". "\\?>")))
(cond
;; ((eq (char-after) ?p)
((looking-at-p "")
) ;php
((string= web-mode-engine "django")
(cond
((string= sub2 "{{")
(setq closing-string '("{{" . "}}")
delim-open "{{"
delim-close "}}"))
((string= sub2 "{%")
(setq closing-string "%}"
delim-open "{%[+-]?"
delim-close "[-]?%}"))
(t
(setq closing-string "#}"))
)
) ;django
((string= web-mode-engine "erb")
(cond
((string= sub2 "<%")
(setq closing-string "%>"
delim-open "<%[=-]?"
delim-close "[-]?%>")
)
(t
(setq closing-string "EOL"
delim-open "%"))
)
) ;erb
((string= web-mode-engine "lsp")
(setq closing-string "%>"
delim-open "<%[%#]?"
delim-close "%>")
) ;lsp
((string= web-mode-engine "mako")
(cond
((member tagopen '("<% " "<%!"))
(setq closing-string "%>"
delim-open "<%[!]?"
delim-close "%>"))
((member sub2 '("<%" ""))
(setq closing-string ">"
delim-open "?%"
delim-close "/?>"))
((string= sub2 "${")
(setq closing-string "}"
delim-open "${"
delim-close "}"))
(t
(setq closing-string "EOL"
delim-open "%"))
)
) ;mako
((string= web-mode-engine "mojolicious")
(cond
((string= tagopen "<%#")
(setq closing-string "%>"))
((string= sub2 "<%")
(setq closing-string "%>"
delim-open "<%\\(==\\|[=%]\\)?"
delim-close "%>"))
((string= sub2 "%#")
(setq closing-string "EOL"))
(t
(setq closing-string "EOL"
delim-open "%\\(==\\|[=%]\\)?"))
)
) ;mojolicious
((string= web-mode-engine "ctemplate")
;;(message "ctemplate")
(cond
((member tagopen '("{{{" "{{~"))
(setq closing-string "}~?}}"
delim-open "{{~?{"
delim-close "}~?}}")
)
((string= tagopen "{~{")
(setq closing-string "}~?}"
delim-open "{~{"
delim-close "}~?}")
)
((string= sub2 "{{")
(setq closing-string "}~?}"
delim-open "{{[>#/%^&]?"
delim-close "}~?}"))
(t
(setq closing-string "}}"
delim-open "${{"
delim-close "}}"))
)
) ;ctemplate
((string= web-mode-engine "aspx")
(setq closing-string "%>"
delim-open "<%[:=#@$]?"
delim-close "%>")
) ;aspx
((string= web-mode-engine "asp")
(cond
((string= sub2 "<%")
(setq closing-string "%>"
delim-open "<%[:=#@$]?"
delim-close "%>"))
(t
(setq closing-string ">"
delim-open "?"
delim-close "/?>"))
)
) ;asp
((string= web-mode-engine "jsp")
(cond
((string= sub2 "<%")
(setq closing-string "%>"
delim-open "<%\\([!=@]\\|#=\\)?"
delim-close "[-]?%>"))
((string= sub2 "${")
(setq closing-string "}"
delim-open "${"
delim-close "}"))
(t
(setq closing-string ">"
delim-open "?"
delim-close "/?>"))
)
) ;jsp
((string= web-mode-engine "clip")
(setq closing-string ">"
delim-open "?"
delim-close "/?>")
) ;clip
((string= web-mode-engine "blade")
(cond
((string= tagopen "{{-")
(setq closing-string "--}}"))
((string= tagopen "{{{")
(setq closing-string "}}}"
delim-open "{{{"
delim-close "}}}"))
((string= sub2 "{{")
(setq closing-string "}}"
delim-open "{{"
delim-close "}}"))
((string= sub1 "@")
(setq closing-string "EOL"
delim-open "@"))
)
) ;blade
((string= web-mode-engine "smarty")
(cond
((string= tagopen (concat (web-mode-engine-delimiter-open web-mode-engine "{") "*"))
(setq closing-string (concat "*" (web-mode-engine-delimiter-close web-mode-engine "}")))
)
((string= tagopen (concat (web-mode-engine-delimiter-open web-mode-engine "{") "#"))
(setq closing-string (concat "#" (web-mode-engine-delimiter-close web-mode-engine "}"))
delim-open (concat (web-mode-engine-delimiter-open web-mode-engine "{") "#")
delim-close (concat "#" (web-mode-engine-delimiter-close web-mode-engine "}")))
)
(t
(setq closing-string (cons (web-mode-engine-delimiter-open web-mode-engine "{")
(web-mode-engine-delimiter-close web-mode-engine "}"))
delim-open (concat (web-mode-engine-delimiter-open web-mode-engine "{") "/?")
delim-close (web-mode-engine-delimiter-close web-mode-engine "}"))
;; (message "delim-open=%s delim-close=%s" delim-open delim-close)
) ;t
) ;cond
) ;smarty
((string= web-mode-engine "web2py")
(setq closing-string "}}"
delim-open "{{[=]?"
delim-close "}}")
) ;web2py
((string= web-mode-engine "dust")
(cond
((string= sub2 "{!")
(setq closing-string "!}"))
(t
(setq closing-string "}"
delim-open "{[#/:?@><+^]?"
delim-close "/?}")
)
)
) ;dust
((string= web-mode-engine "closure")
(cond
((string= sub2 "//")
(setq closing-string "EOL")
)
((string= sub2 "/*")
(setq closing-string "*/")
)
(t
(setq closing-string "}"
delim-open "{/?"
delim-close "/?}")
)
)
) ;closure
((string= web-mode-engine "go")
(setq closing-string "}}"
delim-open "{{"
delim-close "}}")
) ;go
((string= web-mode-engine "angular")
(setq closing-string "}}"
delim-open "{{"
delim-close "}}")
) ;angular
((string= web-mode-engine "mason")
(cond
((and (member sub2 '("<%" ""))
(looking-at "[[:alpha:]]+"))
(if (member (match-string-no-properties 0) '("def" "method"))
(setq closing-string ">"
delim-open "<[/]?%"
delim-close ">")
;;delim-open "<[^>]+>")
(setq closing-string (concat "%" (match-string-no-properties 0) ">")
delim-open "<[^>]+>"
delim-close "<[^>]+>")
) ;if
)
((and (string= sub2 "<%")
(eq (char-after) ?\s))
(setq closing-string "%>"
delim-open "<%"
delim-close "%>"))
((string= tagopen "&")
(setq closing-string ">"
delim-open "&"
delim-close ">")
)
((string= sub2 "<&")
(setq closing-string "&>"
delim-open "<&[|]?"
delim-close "&>"))
(t
(setq closing-string "EOL"
delim-open "%"))
)
) ;mason
((string= web-mode-engine "underscore")
(setq closing-string "%>"
delim-open "<%"
delim-close "%>")
) ;underscore
((string= web-mode-engine "template-toolkit")
(cond
((string= sub2 "[#")
(setq closing-string "#]"))
(t
(setq closing-string "%]"
delim-open "\\[%[-+]?"
delim-close "[-=+]?%\\]"))
)
) ;template-toolkit
((string= web-mode-engine "freemarker")
(cond
((string= sub1 "<")
(setq closing-string ">"
delim-open "?[#@]?"
delim-close "/?>"))
((string= sub1 "[")
(setq closing-string "]"
delim-open "\\[/?[#@]"
delim-close "/?\\]"))
(t
(setq closing-string "}"
delim-open "${"
delim-close "}"))
)
) ;freemarker
((string= web-mode-engine "velocity")
(cond
((string= sub2 "##")
(setq closing-string "EOL"))
((string= sub2 "#*")
(setq closing-string "*#"))
(t
(setq closing-string "EOV"
delim-open "#"))
)
) ;velocity
((string= web-mode-engine "razor")
;; (message "sub2=%S" sub2)
(cond
((string= sub2 "@@")
(forward-char 2)
(setq closing-string nil))
((string= sub2 "@*")
(setq closing-string "*@"))
((string= sub1 "@")
(setq closing-string "EOR"
delim-open "@"))
((string= sub1 "}")
(save-excursion
(let (paren-pos)
(setq paren-pos (web-mode-opening-paren-position (1- (point))))
(if (and paren-pos (get-text-property paren-pos 'block-side))
(setq closing-string "EOR")
(setq closing-string nil)
) ;if
) ;let
) ;let
) ; case }
) ;cond
) ;razor
) ;cond
(when closing-string
(cond
((listp closing-string)
(cond
((web-mode-rsf-balanced (car closing-string) (cdr closing-string) reg-end t)
(setq close (match-end 0)
pos (point))
)
((and (string= web-mode-engine "php")
(string= "" sub2))
(if (text-property-not-all (+ open 2) (point-max) 'tag-beg nil)
(setq close (line-end-position)
delim-close nil
pos (line-end-position))
(setq close (point-max)
delim-close nil
pos (point-max))
) ;if
) ;case
) ;cond
) ;case listp
((and (string= web-mode-engine "smarty")
(string= closing-string (web-mode-engine-delimiter-close web-mode-engine "}")))
(goto-char open)
(setq tmp (web-mode-closing-delimiter-position
(web-mode-engine-delimiter-close web-mode-engine "}")
(point)
(line-end-position)))
(if tmp
(setq tmp (1+ tmp))
(setq tmp (line-end-position)))
(goto-char tmp)
(setq close (point)
pos (point))
)
((and (member web-mode-engine '("closure" "dust"))
(string= closing-string "}"))
(goto-char open)
(setq tmp (web-mode-closing-paren-position (point) (line-end-position)))
(if tmp
(setq tmp (1+ tmp))
(setq tmp (line-end-position)))
(goto-char tmp)
(setq close (point)
pos (point))
)
((string= closing-string "EOL")
(end-of-line)
(setq close (point)
pos (point)))
((string= closing-string "EOR")
(web-mode-razor-skip-forward open)
(setq close (if (> (point) reg-end) reg-end (point))
pos (if (> (point) reg-end) reg-end (point)))
(goto-char pos))
((string= closing-string "EOV")
(web-mode-velocity-skip-forward open)
(setq close (point)
pos (point)))
((and (member web-mode-engine '("ctemplate"))
;; (progn (message "ici%S %S" (point) closing-string) t)
(re-search-forward closing-string reg-end t))
(setq close (match-end 0)
pos (point)))
((search-forward closing-string reg-end t)
(setq close (match-end 0)
pos (point)))
) ;cond
;; (message "close=%S reg-end=%S pos=%S" close reg-end pos)
(when (and close (>= reg-end pos))
;;(message "pos(%S) : open(%S) close(%S)" pos open close)
(put-text-property open (1+ open) 'block-beg 0)
(put-text-property open (1+ open) 'block-controls 0)
(if nil ;;(= close (point-max))
(progn
(put-text-property open (1+ close) 'block-side t)
(put-text-property close (1+ close) 'block-end t))
(put-text-property open close 'block-side t)
(put-text-property (1- close) close 'block-end t)
)
(when delim-open
(web-mode-block-delimiters-set open close delim-open delim-close))
(web-mode-block-scan open close)
(when (and (member tagopen '(" part-end part-beg))
(put-text-property part-beg part-end
'part-side
(cond
((member tagopen '("= i 1200)
(message "scan-blocks ** crazy loop **"))
((string= web-mode-engine "razor")
(web-mode-process-blocks reg-beg reg-end 'web-mode-block-scan))
((string= web-mode-engine "django")
(web-mode-scan-engine-comments reg-beg reg-end "{% comment %}" "{% endcomment %}"))
((string= web-mode-engine "mako")
(web-mode-scan-engine-comments reg-beg reg-end "<%doc>" "%doc>"))
) ;cond
)))
(defun web-mode-block-delimiters-set (reg-beg reg-end delim-open delim-close)
"Set text-property 'block-token to 'delimiter on block delimiters (e.g. )"
;; (message "reg-beg(%S) reg-end(%S) delim-open(%S) delim-close(%S)" reg-beg reg-end delim-open delim-close)
(when (member web-mode-engine '("asp" "aspx" "clip" "closure" "ctemplate" "django" "dust"
"erb" "freemarker" "jsp" "lsp" "mako" "mason" "mojolicious"
"smarty" "template-toolkit" "web2py"))
(save-excursion
(when delim-open
(goto-char reg-beg)
(looking-at delim-open)
(setq delim-open (match-string-no-properties 0)))
(when delim-close
(goto-char reg-end)
(looking-back delim-close reg-beg t)
(setq delim-close (match-string-no-properties 0)))
))
;; todo : prefer 'delimiter-open 'delimiter-close, stronger : =?>
(when delim-open
(put-text-property reg-beg (+ reg-beg (length delim-open)) 'block-token 'delimiter))
(when delim-close
(put-text-property (- reg-end (length delim-close)) reg-end 'block-token 'delimiter))
)
(defun web-mode-process-blocks (reg-beg reg-end func)
"Process blocks. The scan relies on the 'block-beg and 'block-end text-properties."
(let ((i 0) (continue t) (block-beg reg-beg) (block-end nil))
(while continue
(setq block-end nil)
(unless (get-text-property block-beg 'block-beg)
(setq block-beg (web-mode-block-next-position block-beg)))
(when (and block-beg (< block-beg reg-end))
(setq block-end (web-mode-block-end-position block-beg)))
(cond
((> (setq i (1+ i)) 1200)
(message "process-blocks ** crazy loop **")
(setq continue nil))
((or (null block-end) (> block-end reg-end))
(setq continue nil))
(t
(setq block-end (1+ block-end))
(funcall func block-beg block-end)
(setq block-beg block-end)
) ;t
) ;cond
) ;while
))
(defun web-mode-process-parts (reg-beg reg-end func)
"Process parts. The scan relies on the 'part-side text-property."
(let ((i 0) (continue t) (part-beg reg-beg) (part-end nil))
(while continue
(setq part-end nil)
(unless (get-text-property part-beg 'part-side)
(setq part-beg (web-mode-part-next-position part-beg)))
(when (and part-beg (< part-beg reg-end))
(setq part-end (web-mode-part-end-position part-beg)))
(cond
((> (setq i (1+ i)) 1200)
(setq continue nil)
(message "process-parts ** crazy loop **"))
((or (null part-end) (> part-end reg-end))
(setq continue nil))
(t
(setq part-end (1+ part-end))
(funcall func part-beg part-end)
(setq part-beg part-end)
)
) ;cond
) ;while
))
(defun web-mode-block-scan (block-beg block-end)
"Scan a block."
(let (sub1 sub2 sub3 regexp token-type)
;;(message "block-beg=%S block-end=%S" block-beg block-end)
;;(remove-text-properties block-beg block-end web-mode-scan-properties)
(goto-char block-beg)
(cond
((>= (point-max) (+ block-beg 3))
(setq sub3 (buffer-substring-no-properties block-beg (+ block-beg 3))
sub2 (buffer-substring-no-properties block-beg (+ block-beg 2))
sub1 (buffer-substring-no-properties block-beg (+ block-beg 1)))
)
((>= (point-max) (+ block-beg 2))
(setq sub3 (buffer-substring-no-properties block-beg (+ block-beg 2))
sub2 (buffer-substring-no-properties block-beg (+ block-beg 2))
sub1 (buffer-substring-no-properties block-beg (+ block-beg 1)))
)
(t
(setq sub1 (buffer-substring-no-properties block-beg (+ block-beg 1)))
(setq sub2 sub1
sub3 sub1)
)
)
(cond
((member web-mode-engine '("php" "lsp" "python" "web2py" "mason"))
(setq regexp web-mode-engine-token-regexp)
;; (message "%S %S" (point) web-mode-engine-token-regexp)
)
((string= web-mode-engine "mako")
(cond
((string= sub2 "##")
(setq token-type 'comment)
)
(t
(setq regexp web-mode-engine-token-regexp))
)
) ;mako
((string= web-mode-engine "django")
(cond
((member sub2 '("{{" "{%"))
(setq regexp "\"\\|'"))
((string= sub2 "{#")
(setq token-type 'comment))
)
) ;django
((string= web-mode-engine "ctemplate")
(cond
((string= sub3 "{{!")
(setq token-type 'comment)
)
((member sub2 '("{{"))
(setq regexp "\"\\|'")
)
)
) ;ctemplate
((string= web-mode-engine "go")
(cond
((string= sub3 "{{/")
(setq token-type 'comment))
((string= sub2 "{{")
(setq regexp "\"\\|'"))
)
) ;go
((string= web-mode-engine "razor")
(cond
((string= sub2 "@*")
(setq token-type 'comment))
(t
(setq regexp "//\\|@\\*\\|\"\\|'"))
)
) ;razor
((string= web-mode-engine "blade")
(cond
((string= sub3 "{{-")
(setq token-type 'comment))
(t
(setq regexp "\"\\|'"))
)
) ;blade
((string= web-mode-engine "mojolicious")
(cond
((or (string= sub2 "%#") (string= sub3 "<%#"))
(setq token-type 'comment))
(t
(setq regexp "\"\\|'"))
)
) ;mojolicious
((string= web-mode-engine "velocity")
(cond
((member sub2 '("##" "#*"))
(setq token-type 'comment))
((member sub1 '("$" "#"))
(setq regexp "\"\\|'"))
)
) ;velocity
((string= web-mode-engine "jsp")
(cond
((string= sub3 "<%-")
(setq token-type 'comment))
((string= sub3 "<%@")
(setq regexp "/\\*"))
((member sub2 '("${" "#{"))
(setq regexp "\"\\|'"))
((string= sub2 "<%")
(setq regexp "//\\|/\\*\\|\"\\|'"))
)
) ;jsp
((string= web-mode-engine "clip")
(setq regexp nil)
) ;clip
((and (string= web-mode-engine "asp")
(string= sub2 "<%"))
(setq regexp "//\\|/\\*\\|\"\\|'")
) ; asp
((string= web-mode-engine "aspx")
(cond
((string= sub3 "<%-")
(setq token-type 'comment))
((string= sub3 "<%@")
(setq regexp "/\\*"))
((string= sub3 "<%$")
(setq regexp "\"\\|'"))
(t
(setq regexp "//\\|/\\*\\|\"\\|'"))
)
) ;aspx
((string= web-mode-engine "freemarker")
(cond
((member sub3 '("<#-" "[#-"))
(setq token-type 'comment))
((member sub2 '("${" "#{"))
(setq regexp "\"\\|'"))
((or (member sub2 '("<@" "[@" "<#" "[#"))
(member sub3 '("@" "[/@" "#" "[/#")))
(setq regexp "\""))
)
) ;freemarker
((string= web-mode-engine "erb")
(cond
((string= sub3 "<%#")
(setq token-type 'comment))
(t
(setq regexp web-mode-engine-token-regexp))
)
) ;erb
((string= web-mode-engine "template-toolkit")
(cond
((string= sub2 "[#")
(setq token-type 'comment))
(t
(setq regexp "#\\|\"\\|'"))
)
) ;template-toolkit
((string= web-mode-engine "underscore")
(setq regexp "/\\*\\|\"\\|'")
) ;underscore
((string= web-mode-engine "angular")
) ;angular
((string= web-mode-engine "smarty")
(cond
((string= sub2 (concat (web-mode-engine-delimiter-open web-mode-engine "{") "*"))
(setq token-type 'comment))
(t
(setq regexp "\"\\|'")))
) ;smarty
((string= web-mode-engine "dust")
(cond
((string= sub2 "{!")
(setq token-type 'comment))
(t
(setq regexp "\"\\|'"))
)
) ;dust
((string= web-mode-engine "closure")
(cond
((member sub2 '("/*" "//"))
(setq token-type 'comment))
(t
(setq regexp "\"\\|'"))
)
) ;closure
) ;cond
(cond
(token-type
(put-text-property block-beg block-end 'block-token token-type)
(when (eq token-type 'comment)
(web-mode-block-flags-add block-beg 4)))
((and regexp
(or (> (- block-end block-beg) 7)
(text-property-not-all block-beg block-end 'block-token 'delimiter)))
(web-mode-block-tokenize
(web-mode-block-code-beginning-position block-beg)
(web-mode-block-code-end-position block-beg)
regexp)
)
) ;cond
))
(defun web-mode-block-tokenize (reg-beg reg-end &optional regexp)
"Tokenize block chunk."
(unless regexp (setq regexp web-mode-engine-token-regexp))
;; (message "tokenize: reg-beg(%S) reg-end(%S) regexp(%S)" reg-beg reg-end regexp)
(save-excursion
(let ((pos reg-beg) beg char match continue (flags 0) token-type)
(remove-text-properties reg-beg reg-end '(block-token nil))
(goto-char reg-beg)
(when (> reg-beg reg-end)
(message "block-tokenise: %S %S" reg-beg reg-end))
;;FIXME
;;(setq reg-end (1+ reg-end))
(while (and (< reg-beg reg-end) (re-search-forward regexp reg-end t))
(setq beg (match-beginning 0)
match (match-string 0)
continue t
flags (logior flags 1)
token-type nil)
(setq char (aref match 0))
(cond
((and (string= web-mode-engine "asp")
(eq char ?\'))
(setq token-type 'comment)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
((eq char ?\')
(setq token-type 'string)
(while (and continue (search-forward "'" reg-end t))
(if (looking-back "\\\\+'" reg-beg t)
(setq continue (= (mod (- (point) (match-beginning 0)) 2) 0))
(setq continue nil))
)
)
((eq char ?\")
(setq token-type 'string)
(while (and continue (search-forward "\"" reg-end t))
(if (looking-back "\\\\+\"" reg-beg t)
(setq continue (= (mod (- (point) (match-beginning 0)) 2) 0))
(setq continue nil))
)
)
((string= match "//")
(setq token-type 'comment)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
((eq char ?\;)
(setq token-type 'comment)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
((string= match "#|")
(setq token-type 'comment)
(unless (search-forward "|#" reg-end t)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
)
((eq char ?\#)
(setq token-type 'comment)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
((string= match "/*")
(setq token-type 'comment)
(unless (search-forward "*/" reg-end t)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
)
((string= match "@*")
(setq token-type 'comment)
(unless (search-forward "*@" reg-end t)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
)
((eq char ?\<)
(when (and web-mode-enable-heredoc-fontification
(string-match-p "JS\\|JAVASCRIPT\\|HTM\\|CSS" (match-string 1)))
(setq flags (logior flags 2))
)
(setq token-type 'string)
(re-search-forward (concat "^[ ]*" (match-string 1)) reg-end t)
)
) ;cond
;; (message "%S %S %S" beg (point) token-type)
(put-text-property beg (point) 'block-token token-type)
(when (eq token-type 'comment)
(put-text-property beg (1+ beg) 'syntax-table (string-to-syntax "<"))
(put-text-property (1- (point)) (point) 'syntax-table (string-to-syntax ">"))
)
) ;while
(web-mode-block-flags-add pos flags)
(web-mode-block-controls-unset pos)
)))
;; TODO: SUPPRIMER les flags 1 et 2
;; (1)has token (2)has heredoc (4)comment-block
(defun web-mode-block-flags-add (pos flags)
"Set block flags. Block flags are associated with the 'block-beg text-property."
(unless (get-text-property pos 'block-beg)
(setq pos (web-mode-block-beginning-position pos)))
(cond
((null pos)
(message "block-flags-add ** pos(%S) flags(%S) **" pos flags)
)
(t
(setq flags (logior (or (get-text-property pos 'block-beg) 0) flags))
(put-text-property pos (1+ pos) 'block-beg flags))
)
)
(defun web-mode-block-flags-detect (pos flags)
"Checks if current block has flags."
(unless (get-text-property pos 'block-beg)
(setq pos (web-mode-block-beginning-position pos)))
(= (logand (or (get-text-property pos 'block-beg) 0) flags) flags)
)
(defun web-mode-set-php-controls (reg-beg reg-end)
"web-mode-set-php-controls"
(goto-char reg-beg)
(let (match controls
(continue t)
(regexp "endif\\|endforeach\\|endfor\\|endwhile\\|elseif\\|else\\|if\\|foreach\\|for\\|while"))
(while continue
(if (not (web-mode-block-rsf regexp reg-end))
(setq continue nil)
(setq match (match-string-no-properties 0))
;; (message "%S %S" match (point))
(cond
((and (member match '("else" "elseif"))
(looking-at-p "[ ]*[:(]"))
(setq controls (append controls (list (cons 'inside "if"))))
)
((and (>= (length match) 3)
(string= (substring match 0 3) "end"))
(setq controls (append controls (list (cons 'close (substring match 3)))))
)
((and (progn (skip-chars-forward "[ ]") t)
(eq (char-after) ?\()
(web-mode-closing-paren reg-end)
;; (progn (message "ixi%S" (point)))
(looking-at-p ")[ ]*:"))
(setq controls (append controls (list (cons 'open match))))
)
) ; cond
) ;if
) ;while
;;(message "%S-%S %S" reg-beg reg-end controls)
(when (and controls (> (length controls) 1))
(setq controls (web-mode-block-controls-reduce controls)))
controls))
;;todo
;; nettoyer
;;
;; ((open . "if") (close . "if"))
;; -> nil
(defun web-mode-block-controls-reduce (controls)
"clean block controls"
(when (and (eq (car (car controls)) 'open)
(member (cons 'close (cdr (car controls))) controls))
(setq controls nil))
controls)
(defun web-mode-block-controls-unset (pos)
"Unset block controls"
(cond
((null (get-text-property pos 'block-side))
(message "block-controls-unset ** invalid value (%S) **" pos))
((or (get-text-property pos 'block-beg)
(setq pos (web-mode-block-beginning-position pos)))
(put-text-property pos (1+ pos) 'block-controls 0))
(t
(message "block-controls-unset ** failure (%S) **" (point)))
))
(defun web-mode-block-controls-get (pos)
"Get block controls"
(web-mode-with-silent-modifications
(let ((controls nil))
(cond
((null (get-text-property pos 'block-side))
(message "block-controls-get ** invalid value (%S) **" pos))
((or (get-text-property pos 'block-beg)
(setq pos (web-mode-block-beginning-position pos)))
(setq controls (get-text-property pos 'block-controls))
(when (integerp controls)
(web-mode-block-controls-set pos (web-mode-block-end-position pos))
(setq controls (get-text-property pos 'block-controls))
;; (message "controls=%S" controls)
)
)
(t
(message "block-controls-get ** failure (%S) **" (point)))
) ;cond
controls)))
(defun web-mode-block-controls-set (reg-beg reg-end)
"Set block properties"
(save-excursion
(goto-char reg-beg)
(let (controls pos type control)
(cond
((null web-mode-engine)
(message "block-controls-set ** unknown engine (%S) **" web-mode-engine)
)
((string= web-mode-engine "php")
(setq controls (web-mode-set-php-controls reg-beg reg-end))
(when (web-mode-block-starts-with "}" reg-beg)
(setq controls (append controls (list (cons 'close "{")))))
(when (web-mode-block-ends-with (cons "{" "}") reg-beg)
(setq controls (append controls (list (cons 'open "{")))))
) ; php
((string= web-mode-engine "erb")
(cond
((web-mode-block-starts-with "else\\|elsif\\|when" reg-beg)
(setq controls (append controls (list (cons 'inside "ctrl")))))
((web-mode-block-starts-with "end" reg-beg)
(setq controls (append controls (list (cons 'close "ctrl")))))
((web-mode-block-starts-with ".* do \\|for\\|if\\|unless\\|case" reg-beg)
(setq controls (append controls (list (cons 'open "ctrl")))))
)
) ; erb
((string= web-mode-engine "django")
(when (eq (char-after (1+ reg-beg)) ?\%)
(cond
((web-mode-block-starts-with "\\(else\\|els?if\\)" reg-beg)
(let ((continue t)
(pos reg-beg)
(ctrl nil))
(while continue
(cond
((null (setq pos (web-mode-block-control-previous-position 'open pos)))
(setq continue nil))
((member (setq ctrl (cdr (car (get-text-property pos 'block-controls)))) '("if" "ifequal" "ifnotequal"))
(setq continue nil)
)
) ;cond
) ;while
(setq controls (append controls (list (cons 'inside (or ctrl "if")))))
) ;let
) ;case else
((web-mode-block-starts-with "\\(empty\\)" reg-beg)
(setq controls (append controls (list (cons 'inside "for")))))
((web-mode-block-starts-with "end\\([[:alpha:]]+\\)" reg-beg)
(setq controls (append controls (list (cons 'close (match-string-no-properties 1))))))
((web-mode-block-starts-with web-mode-django-control-blocks reg-beg)
(setq controls (append controls (list (cons 'open (match-string-no-properties 1))))))
)
)
) ;django
((string= web-mode-engine "smarty")
(cond
((and (eq (char-after (+ (length (web-mode-engine-delimiter-open web-mode-engine "{")) reg-beg)) ?\/)
(web-mode-block-starts-with "\\([[:alpha:]]+\\)" reg-beg))
(setq controls (append controls (list (cons 'close (match-string-no-properties 1))))))
((web-mode-block-starts-with "\\(else\\|elseif\\)" reg-beg)
(setq controls (append controls (list (cons 'inside "if")))))
((web-mode-block-starts-with "\\(block\\|foreach\\|for\\|if\\|section\\|while\\)")
(setq controls (append controls (list (cons 'open (match-string-no-properties 1))))))
)
) ;smarty
((string= web-mode-engine "web2py")
(cond
((web-mode-block-starts-with "def" reg-beg)
(setq controls (append controls (list (cons 'open "def")))))
((web-mode-block-starts-with "return" reg-beg)
(setq controls (append controls (list (cons 'close "def")))))
((web-mode-block-starts-with "block" reg-beg)
(setq controls (append controls (list (cons 'open "block")))))
((web-mode-block-starts-with "end" reg-beg)
(setq controls (append controls (list (cons 'close "block")))))
((web-mode-block-starts-with "pass" reg-beg)
(setq controls (append controls (list (cons 'close "ctrl")))))
((web-mode-block-starts-with "\\(except\\|finally\\|els\\)" reg-beg)
(setq controls (append controls (list (cons 'inside "ctrl")))))
((web-mode-block-starts-with "\\(if\\|for\\|try\\|while\\)")
(setq controls (append controls (list (cons 'open "ctrl")))))
)
) ;web2py
((string= web-mode-engine "dust")
(cond
((eq (char-after (1- reg-end)) ?\/)
)
((eq (char-after (1+ reg-beg)) ?\:)
(setq pos (web-mode-block-control-previous-position 'open reg-beg))
(when pos
(setq controls (append controls
(list
(cons 'inside
(cdr (car (web-mode-block-controls-get pos))))))))
)
((looking-at "{/\\([[:alpha:]]+\\)")
(setq controls (append controls (list (cons 'close (match-string-no-properties 1))))))
((looking-at "{[#?@><+^]\\([[:alpha:]]+\\)")
(setq controls (append controls (list (cons 'open (match-string-no-properties 1))))))
) ;cond
) ;dust
((member web-mode-engine '("aspx" "underscore" "mojolicious"))
(cond
((web-mode-block-starts-with "}" reg-beg)
(setq controls (append controls (list (cons 'close "{")))))
((web-mode-block-ends-with "{" reg-beg)
(setq controls (append controls (list (cons 'open "{")))))
)
) ;aspx underscore
((member web-mode-engine '("jsp" "asp" "clip"))
(cond
((eq (char-after (1- reg-end)) ?\/)
)
((looking-at "?\\([[:alpha:]]+\\(?:[:][[:alpha:]]+\\)\\|[[:alpha:]]+Template\\)")
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 1) ?\/) 'close 'open))
(when (not (member control '("h:inputtext" "jsp:usebean" "jsp:forward" "struts:property")))
(setq controls (append controls (list (cons type control)))))
)
(t
(when (web-mode-block-starts-with "}" reg-beg)
(setq controls (append controls (list (cons 'close "{")))))
;; todo : bug qd <% } else { %>
(when (web-mode-block-ends-with "{" reg-beg)
(setq controls (append controls (list (cons 'open "{")))))
)
)
) ;jsp asp
((string= web-mode-engine "mako")
(cond
((looking-at "?%\\([[:alpha:]]+\\(?:[:][[:alpha:]]+\\)?\\)")
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 1) ?\/) 'close 'open))
(setq controls (append controls (list (cons type control))))
)
((web-mode-block-starts-with "\\(else\\|elif\\)" reg-beg)
(setq controls (append controls (list (cons 'inside "if")))))
((web-mode-block-starts-with "end\\(if\\|for\\)" reg-beg)
(setq controls (append controls (list (cons 'close (match-string-no-properties 1))))))
((web-mode-block-starts-with "if\\|for" reg-beg)
(setq controls (append controls (list (cons 'open (match-string-no-properties 0))))))
)
) ;mako
((string= web-mode-engine "mason")
(cond
((looking-at "?%\\(def\\|method\\)")
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 1) ?\/) 'close 'open))
(setq controls (append controls (list (cons type control))))
)
) ;mason
)
((string= web-mode-engine "ctemplate")
(cond
((looking-at-p "{{else")
(setq controls (append controls (list (cons 'inside "if")))))
((looking-at "{{[#^/][ ]*\\([[:alpha:]-]+\\)")
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 2) ?\/) 'close 'open))
(setq controls (append controls (list (cons type control))))
)
)
) ;ctemplate
((string= web-mode-engine "blade")
(cond
((not (eq (char-after) ?\@))
)
((web-mode-block-starts-with
"\\(?:end\\)?\\(foreach\\|forelse\\|for\\|if\\|section\\|unless\\|while\\)"
reg-beg)
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 0) ?e) 'close 'open))
(setq controls (append controls (list (cons type control))))
)
((web-mode-block-starts-with "stop\\|show" reg-beg)
(setq controls (append controls (list (cons 'close "section")))))
((web-mode-block-starts-with "else\\|elseif" reg-beg)
(setq controls (append controls (list (cons 'inside "if")))))
)
) ;blade
((string= web-mode-engine "closure")
(cond
((eq (char-after (1- reg-end)) ?\/)
)
((looking-at "alias\\|namespace")
)
((web-mode-block-starts-with "ifempty" reg-beg)
(setq controls (append controls (list (cons 'inside "foreach")))))
((web-mode-block-starts-with "else\\|elseif" reg-beg)
(setq controls (append controls (list (cons 'inside "if")))))
((web-mode-block-starts-with "case\\|default" reg-beg)
(setq controls (append controls (list (cons 'inside "switch")))))
((looking-at
"{/?\\(call\\|deltemplate\\|for\\|foreach\\|if\\|let\\|literal\\|msg\\|param\\|switch\\|template\\)")
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 1) ?\/) 'close 'open))
(setq controls (append controls (list (cons type control))))
)
)
) ;closure
((string= web-mode-engine "go")
(cond
((web-mode-block-starts-with "end" reg-beg)
(setq controls (append controls (list (cons 'close "ctrl")))))
((web-mode-block-starts-with "else" reg-beg)
(setq controls (append controls (list (cons 'inside "ctrl")))))
((web-mode-block-starts-with "range\\|with" reg-beg)
(setq controls (append controls (list (cons 'open "ctrl")))))
)
) ;go
((string= web-mode-engine "template-toolkit")
(cond
((web-mode-block-starts-with "end" reg-beg)
(setq controls (append controls (list (cons 'close "ctrl")))))
((web-mode-block-starts-with "els" reg-beg)
(setq controls (append controls (list (cons 'inside "ctrl")))))
((web-mode-block-starts-with "if\\|foreach\\|filter" reg-beg)
(setq controls (append controls (list (cons 'open "ctrl")))))
)
) ;template-toolkit
((string= web-mode-engine "velocity")
(cond
((web-mode-block-starts-with "end" reg-beg)
(setq controls (append controls (list (cons 'close "ctrl")))))
((web-mode-block-starts-with "els" reg-beg)
(setq controls (append controls (list (cons 'inside "ctrl")))))
((web-mode-block-starts-with "define\\|if\\|for\\|foreach\\|macro" reg-beg)
(setq controls (append controls (list (cons 'open "ctrl")))))
)
) ;velocity
((string= web-mode-engine "freemarker")
(cond
((eq (char-after (1- reg-end)) ?\/)
)
((looking-at "[<[]#\\(break\\|case\\|default\\)")
(setq controls (append controls (list (cons 'inside "switch"))))
)
((looking-at "[<[]#els")
(setq controls (append controls (list (cons 'inside "if"))))
)
((looking-at "?\\([[:alpha:]]+\\(?:[:][[:alpha:]]+\\)?\\)")
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 1) ?\/) 'close 'open))
(setq controls (append controls (list (cons type control))))
)
((looking-at "[<[]/?[#@]\\([[:alpha:]]+\\(?:[:][[:alpha:]]+\\)?\\)")
(setq control (match-string-no-properties 1)
type (if (eq (aref (match-string-no-properties 0) 1) ?\/) 'close 'open))
(setq controls (append controls (list (cons type control))))
)
(t
(when (web-mode-block-starts-with "}" reg-beg)
(setq controls (append controls (list (cons 'close "{")))))
;; todo : bug qd <% } else { %>
(when (web-mode-block-ends-with "{" reg-beg)
(setq controls (append controls (list (cons 'open "{")))))
)
)
) ;freemarker
((string= web-mode-engine "razor")
(when (web-mode-block-starts-with "}" reg-beg)
(setq controls (append controls (list (cons 'close "{")))))
(when (web-mode-block-ends-with "{" reg-beg)
(setq controls (append controls (list (cons 'open "{")))))
) ;razor
((string= web-mode-engine "lsp")
(when (web-mode-block-starts-with ")" reg-beg)
(setq controls (append controls (list (cons 'close "(")))))
(when (web-mode-block-is-opened-sexp reg-beg reg-end)
(setq controls (append controls (list (cons 'open "(")))))
) ;lsp
) ;cond
(put-text-property reg-beg (1+ reg-beg) 'block-controls controls)
;; (message "(%S) controls=%S" reg-beg controls)
)))
(defun web-mode-block-is-opened-sexp (reg-beg reg-end)
"web-mode-block-is-opened-sexp"
(let ((n 0))
(save-excursion
(goto-char reg-beg)
(while (web-mode-block-rsf "[()]" reg-end)
(if (eq (char-before) ?\()
(setq n (1+ n))
(setq n (1- n))
)
)
)
(> n 0)
))
(defvar web-mode-regexp1 "<\\(/?[[:alpha:]][[:alnum:]-]*\\|!--\\|!\\[CDATA\\[\\|!doctype\\|!DOCTYPE\\|\?xml\\)")
;;(defvar web-mode-regexp1 "<\\(/?[[:alpha:]][[:alnum:]-]*\\|\?xml\\|!\\(?:--\\|\\[CDATA\\[\\|doctype\\|DOCTYPE\\)\\)")
(defvar web-mode-regexp2 "<\\(/?[[:alpha:]][[:alnum:]-]*\\|!--\\|!\\[CDATA\\[\\)")
;;(defvar web-mode-regexp2 "<\\(/?[[:alpha:]][[:alnum:]-]*\\|\?xml\\|!\\(?:--\\|\\[CDATA\\[\\)\\)")
(defun web-mode-scan-elements (reg-beg reg-end)
"Scan html nodes (tags/attrs/comments/doctype)."
(save-excursion
(let (part-beg part-end flags limit close-expr props tname tbeg tend element-content-type (regexp web-mode-regexp1) part-close-tag char)
(goto-char reg-beg)
(while (web-mode-dom-rsf regexp reg-end)
(setq flags 0
tname (downcase (match-string-no-properties 1))
char (aref tname 0)
tbeg (match-beginning 0)
tend nil
element-content-type nil
limit reg-end
part-beg nil
part-end nil
props nil
close-expr nil
part-close-tag nil)
(cond
((not (member char '(?\! ?\?)))
(when (string-match-p "-" tname)
(setq flags (logior flags 2)))
(cond
((eq char ?\/)
(setq props (list 'tag-name (substring tname 1) 'tag-type 'end)
flags (logior flags 4)
limit (if (> reg-end (line-end-position)) (line-end-position) reg-end))
)
((web-mode-element-is-void tname)
(setq props (list 'tag-name tname 'tag-type 'void)))
(t
;; (intern tname web-mode-obarray)
(setq props (list 'tag-name tname 'tag-type 'start)))
;; (setq props (list 'tag-name (intern tname web-mode-obarray) 'tag-type 'start)))
) ;cond
)
((and (eq char ?\!) (eq (aref tname 1) ?\-))
(setq close-expr "-->"
props '(tag-type comment)))
((string= tname "?xml")
(setq regexp web-mode-regexp2
close-expr "?>"
props '(tag-type declaration)))
((string= tname "![cdata[")
(setq close-expr "]]>"
props '(tag-type cdata)))
((string= tname "!doctype")
(setq regexp web-mode-regexp2
props '(tag-type doctype)))
) ;cond
(cond
((and (null close-expr) (eq (char-after) ?\>))
(setq flags (logior flags 16)
tend (1+ (point)))
)
((null close-expr)
(setq flags (logior flags (web-mode-tag-skip reg-end)))
(when (> (logand flags 8) 0)
(setq props (plist-put props 'tag-type 'void)))
(setq tend (point)))
((web-mode-dom-sf close-expr limit t)
(setq tend (point)))
(t
(setq tend (line-end-position)))
)
(cond
((string= tname "style")
(setq element-content-type "css"
part-close-tag ""))
((string= tname "script")
(let (script)
(setq script (buffer-substring-no-properties tbeg tend)
part-close-tag "")
(cond
((string-match-p " type[ ]*=[ ]*[\"']text/jsx" script)
(setq element-content-type "jsx"))
((string-match-p " type[ ]*=[ ]*[\"']text/\\(x-handlebars\\|html\\|ng-template\\|template\\)" script)
(setq element-content-type "html"
part-close-tag nil))
((string-match-p " type[ ]*=[ ]*[\"']application/\\(ld\\+json\\|json\\)" script)
(setq element-content-type "json"))
(t
(setq element-content-type "javascript"))
) ;cond
) ;let
) ;script
)
;; (message "ect=%S" element-content-type)
;; (message "%S %S %S %S" tname (point) part-close-tag reg-end)
(add-text-properties tbeg tend props)
(put-text-property tbeg (1+ tbeg) 'tag-beg flags)
(put-text-property (1- tend) tend 'tag-end t)
(when (and part-close-tag
(web-mode-dom-sf part-close-tag reg-end t)
(setq part-beg tend)
(setq part-end (match-beginning 0))
(> part-end part-beg))
(put-text-property part-beg part-end 'part-side
(intern element-content-type web-mode-obarray))
(setq tend part-end)
) ;when
(goto-char tend)
) ;while
)))
;; tag flags
;; (1)attrs (2)custom (4)slash-beg (8)slash-end (16)brackend-end
;; start-tag, end-tag, tag-name, element (xsx, an element is delimited by tags), void-element
;; http://www.w3schools.com/jsref/prop_node_nodetype.asp
;; http://dev.w3.org/html5/html-author/#tags
;; http://www.w3.org/TR/html-markup/syntax.html#syntax-elements
;; http://www.w3.org/TR/html-markup/syntax.html#syntax-attributes
;; attr states:
;; (0)nil (1)space (2)name (3)space-before (4)equal (5)space-after (6)value-uq (7)value-sq (8)value-dq
;; (9)value-bq : jsx {}
(defun web-mode-tag-skip (limit)
"Scan attributes and fetch end of current tag."
(let ((tag-flags 0) (attr-flags 0) (continue t) (attrs 0) (counter 0) (brace-depth 0)
(pos-ori (point)) (state 0) (equal-offset 0) (go-back nil)
name-beg name-end val-beg char pos escaped spaced quoted)
(while continue
(setq pos (point)
char (char-after)
spaced (eq char ?\s))
(when quoted (setq quoted (1+ quoted)))
(cond
((>= pos limit)
(setq continue nil)
(setq go-back t)
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
)
((or (and (= state 8) (not (member char '(?\" ?\\))))
(and (= state 7) (not (member char '(?\' ?\\))))
(and (= state 9) (not (member char '(?} ?\\))))
)
(when (and (= state 9) (eq char ?\{))
(setq brace-depth (1+ brace-depth)))
)
((and (= state 9) (eq char ?\}) (> brace-depth 1))
(setq brace-depth (1- brace-depth)))
((get-text-property pos 'block-side)
(when (= state 2)
(setq name-end pos))
)
((or (and (= state 8) (eq ?\" char) (not escaped))
(and (= state 7) (eq ?\' char) (not escaped))
(and (= state 9) (eq ?\} char) (= brace-depth 1))
)
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
(setq state 0
attr-flags 0
equal-offset 0
name-beg nil
name-end nil
val-beg nil)
)
((and (member state '(4 5)) (member char '(?\' ?\" ?\{)))
(setq val-beg pos)
(setq quoted 1)
(setq state (cond
((eq ?\' char) 7)
((eq ?\" char) 8)
(t 9)))
(when (= state 9)
(setq brace-depth 1))
)
((and (eq ?\= char) (member state '(2 3)))
;; (message "%S" (get-text-property pos 'part-token))
(setq equal-offset (- pos name-beg))
(setq state 4)
)
((and spaced (= state 0))
(setq state 1)
)
((and (eq char ?\<) (not (member state '(7 8 9))))
(setq continue nil)
(setq go-back t)
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
)
((and (eq char ?\>) (not (member state '(7 8 9))))
(setq tag-flags (logior tag-flags 16))
(when (eq (char-before) ?\/)
(setq tag-flags (logior tag-flags 8))
)
(setq continue nil)
(when name-beg
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset))))
)
((and spaced (member state '(1 3 5)))
)
((and spaced (= state 2))
(setq state 3)
)
((and (eq char ?\/) (member state '(4 5)))
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
(setq state 1
attr-flags 0
equal-offset 0
name-beg nil
name-end nil
val-beg nil)
)
((and (eq char ?\/) (member state '(0 1)))
)
((and spaced (= state 4))
(setq state 5)
)
((and (= state 3)
(or (and (>= char 97) (<= char 122)) ;a - z
(and (>= char 65) (<= char 90)) ;A - Z
(eq char ?\-)))
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
(setq state 2
attr-flags 0
equal-offset 0
name-beg pos
name-end pos
val-beg nil)
)
((and (eq char ?\n) (not (member state '(7 8 9))))
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
(setq state 1
attr-flags 0
equal-offset 0
name-beg nil
name-end nil
val-beg nil)
)
((and (= state 6) (member char '(?\s ?\n ?\/)))
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
(setq state 1
attr-flags 0
equal-offset 0
name-beg nil
name-end nil
val-beg nil)
)
((and quoted (= quoted 2) (member char '(?\s ?\n ?\>)))
(when (eq char ?\>)
(setq tag-flags (logior tag-flags 16))
(setq continue nil))
(setq state 6)
(setq attrs (+ attrs (web-mode-scan-attr state char name-beg name-end val-beg attr-flags equal-offset)))
(setq state 1
attr-flags 0
equal-offset 0
name-beg nil
name-end nil
val-beg nil)
)
((and (not spaced) (= state 1))
(setq state 2)
(setq name-beg pos
name-end pos)
)
((member state '(4 5))
(setq val-beg pos)
(setq state 6)
)
((= state 1)
(setq state 2)
)
((= state 2)
(setq name-end pos)
(when (and (= attr-flags 0) (eq char ?\-))
(setq attr-flags (logior attr-flags 1)))
(when (= (logand attr-flags 1) 1)
(let (attr)
(setq attr (buffer-substring-no-properties name-beg (1+ name-end)))
(cond
((member attr '("http-equiv"))
(setq attr-flags (1- attr-flags))
)
((and web-mode-engine-attr-regexp
(string-match-p web-mode-engine-attr-regexp attr))
;; (message "%S" web-mode-engine-attr-regexp)
(setq attr-flags (logior attr-flags 2))
(setq attr-flags (1- attr-flags))
)
) ;cond
) ;let
) ;when attr-flags = 1
) ;state=2
) ;cond
;;(message "point(%S) end(%S) state(%S) c(%S) name-beg(%S) name-end(%S) val-beg(%S) attr-flags(%S) equal-offset(%S)" pos end state char name-beg name-end val-beg attr-flags equal-offset)
(when (and quoted (>= quoted 2))
(setq quoted nil))
(setq escaped (eq ?\\ char))
(when (null go-back)
(forward-char))
(when (> (setq counter (1+ counter)) 2000)
(message "tag-skip ** too much attr ** pos-ori(%S) limit(%S)" pos-ori limit)
(setq continue nil))
) ;while
(when (> attrs 0)
(setq tag-flags (logior tag-flags 1)))
tag-flags))
;; attr flags
;; (1)custom-attr (2)engine-attr
;; states:
;; (0)nil (1)space (2)name (3)space-before (4)equal (5)space-after (6)value-uq (7)value-sq (8)value-dq
(defun web-mode-scan-attr (state char name-beg name-end val-beg flags equal-offset)
"scan attr."
;; (message "point(%S) state(%S) c(%c) name-beg(%S) name-end(%S) val-beg(%S) flags(%S) equal-offset(%S)"
;; (point) state char name-beg name-end val-beg flags equal-offset)
(if (null flags) (setq flags 0))
(cond
((or (null name-beg))
;; (message "name-beg is null (%S)" (point))
0)
((or (and (= state 8) (not (eq ?\" char)))
(and (= state 7) (not (eq ?\' char))))
(put-text-property name-beg val-beg 'tag-attr flags)
(put-text-property (1- val-beg) val-beg 'tag-attr-end equal-offset)
1)
((and (member state '(4 5)) (null val-beg))
(put-text-property name-beg (+ name-beg equal-offset 1) 'tag-attr flags)
(put-text-property (+ name-beg equal-offset) (+ name-beg equal-offset 1) 'tag-attr-end equal-offset)
1)
(t
(let (val-end)
(if (null val-beg)
(setq val-end name-end)
(setq val-end (point))
(when (or (null char) (member char '(?\s ?\n ?\> ?\/)))
(setq val-end (1- val-end))
;; (message "val-end=%S" val-end)
)
) ;if
(put-text-property name-beg (1+ val-end) 'tag-attr flags)
(put-text-property val-end (1+ val-end) 'tag-attr-end equal-offset)
) ;let
1) ;t
(t
0)
) ;cond
)
(defun web-mode-part-scan (reg-beg reg-end &optional content-type)
"Scan client part (e.g. javascript, json, css)."
(save-excursion
(let (token-re ch-before ch-at ch-next token-type beg end continue)
(cond
(content-type
)
((member web-mode-content-type '("javascript" "json" "jsx" "css"))
(setq content-type web-mode-content-type))
(t
(setq content-type (symbol-name (get-text-property reg-beg 'part-side))))
) ;cond
(goto-char reg-beg)
;;(message "part-scan: reg-beg(%S) reg-end(%S) content-type(%S)" reg-beg reg-end content-type)
;; (when (string= content-type "jsx")
;; (message "scan-literals")
;; (web-mode-scan-literals reg-beg reg-end)
;; )
(cond
((member content-type '("javascript" "json"))
(setq token-re "//\\|/\\*\\|\"\\|'"))
((member content-type '("jsx"))
(setq token-re "//\\|/\\*\\|\"\\|'\\|?[[:alpha:]]"))
((string= content-type "css")
(setq token-re "/\\*\\|\"\\|'"))
(t
(setq token-re "/\\*\\|\"\\|'"))
)
;;(message "%S" token-re)
(while (and token-re (web-mode-dom-rsf token-re reg-end t))
(setq beg (match-beginning 0)
end nil
token-type nil
continue t
ch-at (char-after beg)
ch-next (or (char-after (1+ beg)) ?\d)
ch-before (or (char-before beg) ?\d))
;;(message "beg=%S : before(%c) at(%c) next(%c)" beg ch-before ch-at ch-next)
(cond
;;((and (get-text-property beg 'part-expr) (string= content-type "jsx"))
;; (setq beg nil))
;;((and (eq (get-text-property beg 'part-token) 'html)
;; (string= content-type "jsx"))
;; (setq beg nil))
((eq ?\' ch-at)
(while (and continue (search-forward "'" reg-end t))
(cond
((get-text-property (1- (point)) 'block-side)
(setq continue t))
((looking-back "\\\\+'" reg-beg t)
(setq continue (= (mod (- (point) (match-beginning 0)) 2) 0)))
(t
(setq continue nil))
)
) ;while
(cond
((string= content-type "javascript")
(setq token-type 'string))
((string= content-type "css")
(setq token-type 'string))
((string= content-type "json")
(setq token-type 'string))
(t
(setq token-type 'string))
) ;cond
) ;eq '
((eq ?\" ch-at)
(while (and continue (search-forward "\"" reg-end t))
(cond
((get-text-property (1- (point)) 'block-side)
(setq continue t))
((looking-back "\\\\+\"" reg-beg t)
(setq continue (= (mod (- (point) (match-beginning 0)) 2) 0)))
(t
(setq continue nil))
)
)
(cond
((string= content-type "json")
(if (looking-at-p "[ ]*:")
(cond
((eq ?\@ (char-after (1+ beg)))
(setq token-type 'context))
(t
(setq token-type 'key))
)
(setq token-type 'string))
)
(t
(cond
((string= content-type "javascript")
(setq token-type 'string))
((string= content-type "css")
(setq token-type 'string))
(t
(setq token-type 'string))
) ;cond
) ;t
) ;cond
)
((eq ?\< ch-at)
(when (web-mode-jsx-skip-forward reg-end)
(setq end (point))
(put-text-property beg end 'part-element t)
(web-mode-scan-elements beg end)
(web-mode-scan-expr-literal beg end)
) ;when
)
((eq ?\/ ch-next)
(unless (eq ?\\ ch-before)
(setq token-type 'comment)
(goto-char (if (< reg-end (line-end-position)) reg-end (line-end-position)))
)
)
((eq ?\* ch-next)
(unless (eq ?\\ ch-before)
(setq token-type 'comment)
(search-forward "*/" reg-end t)
)
)
((and (member content-type '("javascript" "jsx"))
(eq ?\/ ch-at)
(progn (or (bobp) (backward-char)) t)
(looking-back "[(=][ ]*/")
(looking-at-p ".+/")
;; (not (eq ?\s ch-next))
)
;; (message "regexp literal at (%S)" (1- (point)))
(while (and continue (search-forward "/" reg-end t))
(setq continue (or (get-text-property (1- (point)) 'block-side)
(eq ?\\ (char-before (1- (point))))))
)
(setq token-type 'string)
(skip-chars-forward "gimy")
)
) ;cond
(when (and beg (>= reg-end (point)) token-type)
;; (message "%S %S %S" beg (point) token-type)
(put-text-property beg (point) 'part-token token-type)
(when (eq token-type 'comment)
(put-text-property beg (1+ beg) 'syntax-table (string-to-syntax "<"))
(put-text-property (1- (point)) (point) 'syntax-table (string-to-syntax ">"))
)
)
) ;while
)))
(defun web-mode-jsx-skip-forward (reg-end)
"web-mode-jsx-elt-skip-forward"
(let ((continue t) (pos nil) (i 0))
(save-excursion
(while continue
(cond
((> (setq i (1+ i)) 100)
(message "jsx-skip-forward ** crazy loop **")
(setq continue nil))
((not (web-mode-dom-rsf ">\\([ \t\n]*[;,)']\\)\\|{" reg-end))
;;(backward-char)
(setq continue nil))
((eq (char-before) ?\{)
(backward-char)
(if (web-mode-closing-paren reg-end)
(forward-char)
(setq continue nil))
)
(t
(setq continue nil)
(setq pos (match-beginning 1))
) ;t
) ;cond
) ;while
) ;save-excursion
(when pos (goto-char pos))
pos))
(defun web-mode-scan-expr-literal (reg-beg reg-end)
"web-mode-scan-expr-literal"
(let ((continue t) beg end)
(save-excursion
(goto-char reg-beg)
;; (message "reg-beg=%S reg-end=%S" reg-beg reg-end)
(while (and continue (search-forward "{" reg-end t))
;; (message "pt=%S" (point))
(backward-char)
(setq beg (point)
end (web-mode-closing-paren reg-end)
;; end (search-forward "}" reg-end t)
)
;; (message "beg(%S) end(%S)" beg end)
(if (not end)
(setq continue nil)
(setq end (1+ end))
;; KEEPME: garder { et } en part-token est util pour l'indentation
(put-text-property (1+ beg) (1- end) 'part-token nil)
(put-text-property beg end 'part-expr t)
(web-mode-part-scan (1+ beg) (1- end) "javascript")
)
)
)))
;; css rule = selector(s) + declaration (properties)
(defun web-mode-css-rule-next (limit)
"Fetch next css rule."
(let (at-rule sel-beg sel-end dec-beg dec-end chunk)
(skip-chars-forward "\n\t ")
(setq sel-beg (point))
(when (and (< (point) limit)
(web-mode-part-rsf "[{;]" limit))
(setq sel-end (1- (point)))
(cond
((eq (char-before) ?\{)
(setq dec-beg (point))
(setq dec-end (web-mode-closing-paren-position (1- dec-beg) limit))
(if dec-end
(progn
(goto-char dec-end)
(forward-char))
(setq dec-end limit)
(goto-char limit))
)
(t
)
) ;cond
(setq chunk (buffer-substring-no-properties sel-beg sel-end))
(when (string-match "@\\([[:alpha:]-]+\\)" chunk)
(setq at-rule (match-string-no-properties 1 chunk))
;; (message "%S at-rule=%S" chunk at-rule)
)
) ;when
(if (not sel-end)
(progn (goto-char limit) nil)
(list :at-rule at-rule
:sel-beg sel-beg
:sel-end sel-end
:dec-beg dec-beg
:dec-end dec-end)
) ;if
))
(defun web-mode-css-rule-current (&optional pos part-beg part-end)
"Current CSS rule boundaries."
(interactive)
(unless pos (setq pos (point)))
(unless part-beg (setq part-beg (web-mode-part-beginning-position pos)))
(unless part-end (setq part-end (web-mode-part-end-position pos)))
(save-excursion
(let (beg end)
(goto-char pos)
(if (not (web-mode-part-sb "{" part-beg))
(progn
(setq beg part-beg)
(if (web-mode-part-sf ";" part-end)
(setq end (1+ (point)))
(setq end part-end))
) ;progn
(setq beg (point))
(setq end (web-mode-closing-paren-position beg part-end))
(if end
(setq end (1+ end))
(setq end (line-end-position)))
;; (message "%S >>beg%S >>end%S" pos beg end)
(if (> pos end)
;;selectors
(progn
(goto-char pos)
(if (web-mode-part-rsb "[};]" part-beg)
(setq beg (1+ (point)))
(setq beg part-beg)
) ;if
(goto-char pos)
(if (web-mode-part-rsf "[{;]" part-end)
(cond
((eq (char-before) ?\;)
(setq end (point))
)
(t
(setq end (web-mode-closing-paren-position (1- (point)) part-end))
(if end
(setq end (1+ end))
(setq end part-end))
)
) ;cond
(setq end part-end)
)
) ;progn selectors
;; declaration
(goto-char beg)
(if (web-mode-part-rsb "[}{;]" part-beg)
(setq beg (1+ (point)))
(setq beg part-beg)
) ;if
) ; if > pos end
)
;; (message "beg(%S) end(%S)" beg end)
(when (eq (char-after beg) ?\n)
(setq beg (1+ beg)))
(cons beg end)
)))
(defun web-mode-velocity-skip-forward (pos)
"find the end of a velocity block."
(goto-char pos)
(let ((continue t) (i 0))
(when (eq ?\# (char-after))
(forward-char))
(when (member (char-after) '(?\$ ?\@))
(forward-char))
(when (member (char-after) '(?\!))
(forward-char))
(if (member (char-after) '(?\{))
(search-forward "}")
(setq continue t)
(while continue
(skip-chars-forward "a-zA-Z0-9_-")
(when (> (setq i (1+ i)) 500)
(message "velocity-skip-forward ** crazy loop (%S) **" pos)
(setq continue nil))
(when (member (char-after) '(?\())
(search-forward ")" nil t))
(if (member (char-after) '(?\.))
(forward-char)
(setq continue nil))
) ;while
) ;if
))
(defun web-mode-razor-skip-forward (pos)
"Find the end of a razor block."
(goto-char pos)
(let ((continue t) tmp (i 0))
(while continue
(skip-chars-forward " =@a-zA-Z0-9_-")
(cond
((> (setq i (1+ i)) 500)
(message "razor-skip-forward ** crazy loop (%S) **" pos)
(setq continue nil))
((and (eq (char-after) ?\*)
(eq (char-before) ?@))
(when (not (search-forward "*@" nil t))
(setq continue nil)
)
)
((looking-at-p "@[({]")
(forward-char)
(setq tmp (web-mode-closing-paren-position (point)))
(when tmp
(goto-char tmp))
(forward-char)
)
((and (not (eobp)) (eq ?\( (char-after)))
(if (looking-at-p "[ \n]*<")
(setq continue nil)
(setq tmp (web-mode-closing-paren-position))
(when tmp
(goto-char tmp))
(forward-char)
) ;if
)
((and (not (eobp)) (eq ?\. (char-after)))
(forward-char))
((looking-at-p "[ \n]*{")
(search-forward "{")
(if (looking-at-p "[ \n]*<")
(setq continue nil)
(backward-char)
(setq tmp (web-mode-closing-paren-position))
(when tmp
(goto-char tmp))
(forward-char)
) ;if
)
((looking-at-p "}")
(forward-char)
)
(t
(setq continue nil))
) ;cond
) ;while
))
(defun web-mode-scan-engine-comments (reg-beg reg-end tag-start tag-end)
"Scan engine comments (mako, django)."
(save-excursion
(let (beg end (continue t))
(goto-char reg-beg)
(while (and continue
(< (point) reg-end)
(re-search-forward tag-start reg-end t))
(goto-char (match-beginning 0))
(setq beg (point))
(if (not (re-search-forward tag-end reg-end t))
(setq continue nil)
(setq end (point))
(remove-text-properties beg end web-mode-scan-properties)
(add-text-properties beg end '(block-side t block-token comment))
(put-text-property beg (1+ beg) 'block-beg t)
(put-text-property (1- end) end 'block-end t)
) ;if
) ;while
)))
(defun web-mode-propertize (&optional beg end)
"Propertize."
;; (message "propertize: beg(%S) end(%S)" web-mode-change-beg web-mode-change-end)
(unless beg (setq beg web-mode-change-beg))
(unless end (setq end web-mode-change-end))
(setq web-mode-change-beg nil
web-mode-change-end nil)
(cond
((or (null beg) (null end))
;; (message "nothing todo")
nil)
((and web-mode-enable-block-partial-invalidation
web-mode-engine-token-regexp
(get-text-property beg 'block-side)
(get-text-property end 'block-side)
(> beg (point-min))
(not (eq (get-text-property (1- beg) 'block-token) 'delimiter))
(not (eq (get-text-property end 'block-token) 'delimiter))
;; (not (looking-back "\\*/\\|\\?>"))
;; (progn
;; (setq chunk (buffer-substring-no-properties beg end))
;; (not (string-match-p "\\*/\\|\\?>" chunk))
;; )
)
;; (message "invalidate block")
(web-mode-invalidate-block-region beg end))
((and web-mode-enable-part-partial-invalidation
(or (member web-mode-content-type '("jsx" "javascript"))
(and (get-text-property beg 'part-side)
(get-text-property end 'part-side)
(> beg (point-min))
(get-text-property (1- beg) 'part-side)
(get-text-property end 'part-side))
))
;; (not (looking-back "\\*/\\|"))
;; (progn
;; (setq chunk (buffer-substring-no-properties beg end))
;; (not (string-match-p "\\*/\\|" chunk))
;; )
;; (message "invalidate part")
(web-mode-invalidate-part-region beg end))
(t
(web-mode-invalidate-region beg end))
) ;cond
)
;; Note : il est important d'identifier des caractères en fin de ligne
;; web-mode-block-tokenize travaille en effet sur les fins de lignes pour
;; les commentaires de type //
(defun web-mode-invalidate-block-region (pos-beg pos-end)
"Invalidate block region."
(save-excursion
(let (beg end code-beg code-end)
;; (message "invalidate-block-region: pos-beg(%S)=%S" pos-beg (get-text-property pos 'block-side))
(setq code-beg (web-mode-block-code-beginning-position pos-beg)
code-end (web-mode-block-code-end-position pos-beg))
;; (message "code-beg(%S) code-end(%S) pos-beg(%S) pos-end(%S)" code-beg code-end pos-beg pos-end)
(if (and code-beg code-end
(>= pos-beg code-beg)
(<= pos-end code-end)
(> code-end code-beg))
(cond
((member web-mode-engine '("asp"))
(goto-char pos-beg)
(forward-line -1)
(setq beg (line-beginning-position))
(when (> code-beg beg)
(setq beg code-beg))
(goto-char pos-beg)
(forward-line)
(setq end (line-end-position))
(when (< code-end end)
(setq end code-end))
;;(message "beg(%S) end(%S)" beg end)
(cons beg end)
)
(t
(goto-char pos-beg)
(if (web-mode-block-rsb "[;{}(][ ]*\n" code-beg)
(setq beg (match-end 0))
(setq beg code-beg))
(goto-char pos-end)
(if (web-mode-block-rsf "[;{})][ ]*\n" code-end)
(setq end (1- (match-end 0)))
(setq end code-end))
(web-mode-block-tokenize beg end)
(cons beg end)
)
) ;cond
(web-mode-invalidate-region pos-beg pos-end)
) ;if
)))
(defun web-mode-invalidate-part-region (pos-beg pos-end)
"Invalidate part."
(save-excursion
(let (beg end part-beg part-end language)
(if (member web-mode-content-type '("css" "javascript" "json"))
(setq language web-mode-content-type)
(setq language (symbol-name (get-text-property pos-beg 'part-side))))
(setq part-beg (web-mode-part-beginning-position pos-beg)
part-end (web-mode-part-end-position pos-beg))
;; (message "language(%S) pos-beg(%S) pos-end(%S) part-beg(%S) part-end(%S)"
;; language pos-beg pos-end part-beg part-end)
(if (and part-beg part-end
(>= pos-beg part-beg)
(<= pos-end part-end)
(> part-end part-beg))
(progn
(goto-char pos-beg)
(cond
((member language '("javascript" "json"))
(if (web-mode-part-rsb "[;{}(][ ]*\n" part-beg)
(setq beg (match-end 0))
(setq beg part-beg))
(goto-char pos-end)
(if (web-mode-part-rsf "[;{})][ ]*\n" part-end)
(setq end (1- (match-end 0)))
(setq end part-end))
)
((string= language "css")
(let (rule1 rule2)
(setq rule1 (web-mode-css-rule-current pos-beg))
(setq rule2 rule1)
(when (> pos-end (cdr rule1))
(setq rule2 (web-mode-css-rule-current pos-end)))
(setq beg (car rule1)
end (cdr rule2))
)
;; (message "rule-beg(%S) rule-end(%S)" beg end)
)
(t
(setq beg part-beg
end part-end))
) ;cond
;; (setq web-mode-highlight-beg beg
;; web-mode-highlight-end end)
(web-mode-scan-region beg end language)
;; (message "[%S] scan-region beg=%S end=%S" language beg end)
) ;progn
;; nil
;; (setq web-mode-change-flags 0)
(web-mode-invalidate-region pos-beg pos-end)
) ;if
)))
(defun web-mode-invalidate-region (reg-beg reg-end)
"Invalidate region"
(setq reg-beg (or (web-mode-previous-tag-at-bol-pos reg-beg)
(point-min))
reg-end (or (web-mode-next-tag-at-eol-pos reg-end)
(point-max)))
;; (message "invalidate-region: reg-beg(%S) reg-end(%S)" reg-beg reg-end)
(web-mode-scan-region reg-beg reg-end))
(defun web-mode-buffer-scan ()
"Scan entine buffer."
(interactive)
(web-mode-scan-region (point-min) (point-max)))
;;---- HIGHLIGHTING ------------------------------------------------------------
;;RQ : nous sommes sûrs d'être passés par propertize
(defun web-mode-font-lock-highlight (limit)
"font-lock matcher"
;;(message "font-lock-highlight: point(%S) limit(%S) change-beg(%S) change-end(%S)" (point) limit web-mode-change-beg web-mode-change-end)
;; (when (or (null web-mode-change-beg) (null web-mode-change-end))
;; (message "font-lock-highlight: untouched buffer (%S)" this-command))
(let ((inhibit-modification-hooks t)
(buffer-undo-list t)
(region nil))
(if (and web-mode-change-beg web-mode-change-end)
(setq region (web-mode-propertize))
(message "font-lock-highlight ** untouched buffer (%S) **" this-command)
(setq region (web-mode-propertize (point) limit)))
(when (and region (car region))
(web-mode-highlight-region (car region) (cdr region))
))
nil)
(defun web-mode-buffer-highlight ()
"Scan entine buffer."
(interactive)
(setq web-mode-change-beg (point-min)
web-mode-change-end (point-max))
(web-mode-font-lock-highlight (point-max))
)
(defun web-mode-highlight-region (&optional beg end content-type)
"Highlight region (relying on text-properties setted during the scanning phase)."
;; (unless beg (setq beg web-mode-highlight-beg))
;; (unless end (setq end web-mode-highlight-end))
;; (message "highlight-region: beg(%S) end(%S) ct(%S)" beg end content-type)
(web-mode-with-silent-modifications
(save-excursion
(save-restriction
(save-match-data
(let ((inhibit-modification-hooks t)
(inhibit-point-motion-hooks t)
(inhibit-quit t))
(remove-text-properties beg end '(font-lock-face nil))
(cond
((and (get-text-property beg 'block-side)
(not (get-text-property beg 'block-beg)))
(web-mode-block-highlight beg end))
((or (member web-mode-content-type '("javascript" "json" "jsx" "css"))
(member content-type '("javascript" "json" "jsx" "css"))
(get-text-property beg 'part-side))
(web-mode-part-highlight beg end)
(web-mode-process-blocks beg end 'web-mode-block-highlight))
((string= web-mode-engine "none")
(web-mode-highlight-tags beg end)
(web-mode-process-parts beg end 'web-mode-part-highlight))
(t
(web-mode-highlight-tags beg end)
(web-mode-process-parts beg end 'web-mode-part-highlight)
(web-mode-process-blocks beg end 'web-mode-block-highlight))
) ;cond
(when web-mode-enable-element-content-fontification
(web-mode-highlight-elements beg end))
(when web-mode-enable-whitespace-fontification
(web-mode-highlight-whitespaces beg end))
))))))
(defun web-mode-highlight-tags (reg-beg reg-end)
"web-mode-highlight-nodes"
(let ((continue t))
(goto-char reg-beg)
(when (and (not (get-text-property (point) 'tag-beg))
(not (web-mode-tag-next)))
(setq continue nil))
(when (and continue (>= (point) reg-end))
(setq continue nil))
(while continue
(web-mode-tag-highlight)
(when (or (not (web-mode-tag-next))
(>= (point) reg-end))
(setq continue nil))
) ;while
(when web-mode-enable-inlays
(when (null web-mode-inlay-regexp)
(setq web-mode-inlay-regexp (regexp-opt '("\\[" "\\(" "\\begin{align}"))))
(let (beg end expr)
(goto-char reg-beg)
(while (web-mode-dom-rsf web-mode-inlay-regexp reg-end)
(setq beg (match-beginning 0)
end nil
expr (substring (match-string-no-properties 0) 0 2))
(setq expr (cond
((string= expr "\\[") "\\]")
((string= expr "\\(") "\\)")
(t "\\end{align}")))
(when (and (web-mode-dom-sf expr reg-end)
(setq end (match-end 0))
(not (text-property-any beg end 'tag-end t)))
(font-lock-append-text-property beg end 'font-lock-face 'web-mode-inlay-face)
) ;when
) ;while
) ;let
) ;when
))
;; flags
;; (1)attrs (2)custom (4)slash-beg (8)slash-end (16)brackend-end
(defun web-mode-tag-highlight (&optional beg end)
"web-mode-highlight-nodes"
(unless beg (setq beg (point)))
(unless end (setq end (1+ (web-mode-tag-end-position beg))))
;; (message "tag-highlight: %S %S" beg end)
(let (name type face flags slash-beg slash-end bracket-end)
(setq flags (get-text-property beg 'tag-beg)
type (get-text-property beg 'tag-type)
name (get-text-property beg 'tag-name))
(cond
((eq type 'comment)
(put-text-property beg end 'font-lock-face 'web-mode-comment-face)
;; (message "web-mode-enable-comment-keywords=%S beg(%S) end(%S)" web-mode-enable-comment-keywords beg end)
(when (and web-mode-enable-comment-keywords (> (- end beg) 5))
(web-mode-interpolate-comment beg end nil))
)
((eq type 'cdata)
(put-text-property beg end 'font-lock-face 'web-mode-doctype-face))
((eq type 'doctype)
(put-text-property beg end 'font-lock-face 'web-mode-doctype-face))
((eq type 'declaration)
(put-text-property beg end 'font-lock-face 'web-mode-doctype-face))
(name
;; todo : se passer des vars intermédiaires
(setq face (cond
((and web-mode-enable-element-tag-fontification
(setq face (cdr (assoc name web-mode-element-tag-faces))))
face)
((> (logand flags 2) 0) 'web-mode-html-tag-custom-face)
(t 'web-mode-html-tag-face))
slash-beg (> (logand flags 4) 0)
slash-end (> (logand flags 8) 0)
bracket-end (> (logand flags 16) 0))
(put-text-property beg (+ beg (if slash-beg 2 1)) 'font-lock-face 'web-mode-html-tag-bracket-face)
(put-text-property (+ beg (if slash-beg 2 1))
(+ beg (if slash-beg 2 1) (length name))
'font-lock-face face)
(when (or slash-end bracket-end)
(put-text-property (- end (if slash-end 2 1)) end 'font-lock-face 'web-mode-html-tag-bracket-face)
) ;when
(when (> (logand flags 1) 0)
(web-mode-highlight-attrs beg end)
)
) ;name
) ;cond
))
;; ATTENTION remplacement de font-lock-face par face
;; todo : optimisation des zones reg-beg et reg-end
(defun web-mode-highlight-attrs (reg-beg reg-end)
"Highlight attributes."
(let ((continue t) (pos reg-beg) beg end flags offset face)
(while continue
(setq beg (next-single-property-change pos 'tag-attr))
(if (and beg (< beg reg-end))
(progn
(setq flags (or (get-text-property beg 'tag-attr) 0))
;; (message "beg=%S flags=%S" beg flags)
(setq face (cond
((= (logand flags 1) 1) 'web-mode-html-attr-custom-face)
((= (logand flags 2) 2) 'web-mode-html-attr-engine-face)
(t 'web-mode-html-attr-name-face)))
;; (message "attr-face=%S" face)
(if (get-text-property beg 'tag-attr-end)
(setq end beg)
(setq end (next-single-property-change beg 'tag-attr-end)))
;; (message "beg=%S end=%S" beg end)
(if (and end (< end reg-end))
(progn
(setq offset (get-text-property end 'tag-attr-end))
;; (message "offset=%S" offset)
(if (= offset 0)
(put-text-property beg (1+ end) 'font-lock-face face)
(put-text-property beg (+ beg offset) 'font-lock-face face)
(put-text-property (+ beg offset) (+ beg offset 1)
'font-lock-face
'web-mode-html-attr-equal-face)
(put-text-property (+ beg offset 1) (1+ end)
'font-lock-face
(if (get-text-property (+ beg offset 1) 'part-expr)
nil
'web-mode-html-attr-value-face))
;; (message "attr-value=%S %S" (+ beg offset 1) (1+ end))
) ;if offset
(setq pos (1+ end))
) ;progn
(setq continue nil)
) ;if end
) ;progn beg
(setq continue nil)
) ;if beg
) ;while
))
(defun web-mode-block-highlight (reg-beg reg-end)
"Highlight block."
(let (sub1 sub2 sub3 continue char keywords token-type face beg end (buffer (current-buffer)))
;;(message "reg-beg=%S reg-end=%S" reg-beg reg-end)
;;NOTE: required for block inside tag attr
(remove-text-properties reg-beg reg-end '(font-lock-face nil))
(goto-char reg-beg)
(when (null web-mode-engine-font-lock-keywords)
(setq sub1 (buffer-substring-no-properties
reg-beg (+ reg-beg 1))
sub2 (buffer-substring-no-properties
reg-beg (+ reg-beg 2))
sub3 (buffer-substring-no-properties
reg-beg (+ reg-beg (if (>= (point-max) (+ reg-beg 3)) 3 2))))
)
(cond
((and (get-text-property reg-beg 'block-beg)
(eq (get-text-property reg-beg 'block-token) 'comment))
(put-text-property reg-beg reg-end 'font-lock-face 'web-mode-comment-face)
) ;comment block
(web-mode-engine-font-lock-keywords
(setq keywords web-mode-engine-font-lock-keywords)
)
((string= web-mode-engine "django")
(cond
((string= sub2 "{{")
(setq keywords web-mode-django-expr-font-lock-keywords))
((string= sub2 "{%")
(setq keywords web-mode-django-code-font-lock-keywords))
)) ;django
((string= web-mode-engine "mako")
(cond
((member sub3 '("<% " "<%\n" "<%!"))
(setq keywords web-mode-mako-block-font-lock-keywords))
((eq (aref sub2 0) ?\%)
(setq keywords web-mode-mako-block-font-lock-keywords))
((member sub2 '("<%" ""))
(setq keywords web-mode-mako-tag-font-lock-keywords))
((member sub2 '("${"))
(setq keywords web-mode-uel-font-lock-keywords))
)) ;mako
((string= web-mode-engine "mason")
(cond
((member sub3 '("<% " "<%\n" "<&|"))
(setq keywords web-mode-mason-font-lock-keywords))
((eq (aref sub2 0) ?\%)
(setq keywords web-mode-mason-font-lock-keywords))
(t
(setq keywords web-mode-mason-font-lock-keywords))
)) ;mason
((string= web-mode-engine "jsp")
(cond
((string= sub3 "<%@")
(setq keywords web-mode-directive-font-lock-keywords))
((member sub2 '("${" "#{"))
(setq keywords web-mode-uel-font-lock-keywords))
((string= sub2 "<%")
(setq keywords web-mode-jsp-font-lock-keywords))
(t
(setq keywords web-mode-engine-tag-font-lock-keywords))
)) ;jsp
((string= web-mode-engine "asp")
(cond
((or (string= sub2 "<%")
(not (string= sub1 "<")))
(setq keywords web-mode-asp-font-lock-keywords))
(t
(setq keywords web-mode-engine-tag-font-lock-keywords))
)) ;asp
((string= web-mode-engine "clip")
(setq keywords web-mode-engine-tag-font-lock-keywords)
) ;clip
((string= web-mode-engine "aspx")
(cond
((string= sub3 "<%@")
(setq keywords web-mode-directive-font-lock-keywords))
((string= sub3 "<%$")
(setq keywords web-mode-expression-font-lock-keywords))
(t
(setq keywords web-mode-aspx-font-lock-keywords))
)) ;aspx
((string= web-mode-engine "freemarker")
(cond
((member sub2 '("${" "#{"))
(setq keywords web-mode-uel-font-lock-keywords))
((or (member sub2 '("<@" "[@" "<#" "[#"))
(member sub3 '("@" "[/@" "#" "[/#")))
(setq keywords (if (eq ?\[ (aref sub2 0))
web-mode-freemarker-square-font-lock-keywords
web-mode-freemarker-font-lock-keywords)))
(t
(setq keywords web-mode-engine-tag-font-lock-keywords))
)) ;freemarker
) ;cond
(when keywords
(web-mode-fontify-region reg-beg reg-end keywords)
(setq continue t)
(setq end reg-beg)
(while continue
(if (get-text-property end 'block-token)
(setq beg end)
(setq beg (next-single-property-change end 'block-token buffer reg-end)))
(setq end nil)
(when beg (setq char (char-after beg)))
(if (and beg (< beg reg-end))
(progn
(setq token-type (get-text-property beg 'block-token))
(setq face (cond
((eq token-type 'string) 'web-mode-block-string-face)
((eq token-type 'comment) 'web-mode-block-comment-face)
(t 'web-mode-block-delimiter-face)))
(setq end (next-single-property-change beg 'block-token buffer reg-end))
;; (message "end=%S" end)
(if (and end (<= end reg-end))
(progn
;; (message "%S > %S face(%S)" beg end face)
(remove-text-properties beg end '(face nil))
(put-text-property beg end 'font-lock-face face)
)
(setq continue nil
end nil)
) ;if end
) ;progn beg
(setq continue nil
end nil)
) ;if beg
(when (and beg end)
(save-match-data
(when (and web-mode-enable-heredoc-fontification
(eq char ?\<)
(> (- end beg) 8)
(string-match-p "JS\\|JAVASCRIPT\\|HTM\\|CSS" (buffer-substring-no-properties beg end)))
(setq keywords (if (eq ?H (char-after (+ beg 3)))
web-mode-html-font-lock-keywords
web-mode-javascript-font-lock-keywords))
;; (remove-text-properties beg end '(font-lock-face nil))
(web-mode-fontify-region beg end keywords)
))
;; (message "%S %c %S beg=%S end=%S" web-mode-enable-string-interpolation char web-mode-engine beg end)
(when (and web-mode-enable-string-interpolation
(member char '(?\" ?\<))
(member web-mode-engine '("php" "erb"))
(> (- end beg) 4))
(web-mode-interpolate-block-string beg end)
) ;when
(when (and web-mode-enable-comment-keywords
(eq token-type 'comment)
(> (- end beg) 3))
(web-mode-interpolate-comment beg end t)
) ;when
(when (and (eq token-type 'string)
(> (- end beg) 6)
(web-mode-looking-at-p (concat "[ \n]*" web-mode-sql-queries) (1+ beg)))
(web-mode-interpolate-sql-string beg end)
) ;when
) ;when beg end
) ;while continue
) ;when keywords
(when (and (member web-mode-engine '("jsp" "mako"))
(> (- reg-end reg-beg) 12)
(eq ?\< (char-after reg-beg)))
(web-mode-interpolate-block-tag reg-beg reg-end))
(when web-mode-enable-block-face
;; (message "block-face %S %S" reg-beg reg-end)
(font-lock-append-text-property reg-beg reg-end 'face 'web-mode-block-face))
))
(defun web-mode-part-highlight (reg-beg reg-end)
"Highlight part (e.g. javascript, json, css)."
(save-excursion
(let (start continue token-type face beg end string-face comment-face content-type)
;; (message "part-highlight: reg-beg(%S) reg-end(%S)" reg-beg reg-end)
(if (member web-mode-content-type '("javascript" "json" "jsx" "css"))
(setq content-type web-mode-content-type)
(setq content-type (symbol-name (get-text-property reg-beg 'part-side)))
)
(goto-char reg-beg)
(cond
((member content-type '("javascript" "jsx"))
(setq string-face 'web-mode-javascript-string-face
comment-face 'web-mode-javascript-comment-face)
(web-mode-fontify-region reg-beg reg-end web-mode-javascript-font-lock-keywords)
)
((string= content-type "json")
(setq string-face 'web-mode-json-string-face
comment-face 'web-mode-json-comment-face)
(web-mode-fontify-region reg-beg reg-end web-mode-javascript-font-lock-keywords))
((string= content-type "css")
(setq string-face 'web-mode-css-string-face
comment-face 'web-mode-css-comment-face)
(web-mode-css-rules-highlight reg-beg reg-end)
)
(t
(setq string-face 'web-mode-part-string-face
comment-face 'web-mode-part-comment-face)
)
)
(when (string= content-type "jsx")
(web-mode-highlight-tags reg-beg reg-end)
)
(setq continue t
end reg-beg)
(while continue
;; (setq char (char-after beg))
(if (and (= reg-beg end)
(get-text-property end 'part-token))
(setq beg reg-beg)
(setq beg (next-single-property-change end 'part-token)))
(setq end nil)
(if (and beg (< beg reg-end))
(progn
(setq token-type (get-text-property beg 'part-token))
(setq face (cond
((eq token-type 'string) string-face)
((eq token-type 'comment) comment-face)
((eq token-type 'context) 'web-mode-json-context-face)
((eq token-type 'key) 'web-mode-json-key-face)
(t nil)))
(setq end (or (next-single-property-change beg 'part-token) (point-max)))
;; (message "end=%S reg-end=%S" end reg-end)
(if (<= end reg-end)
(cond
(face
(remove-text-properties beg end '(face nil))
(put-text-property beg end 'font-lock-face face)
(when (and web-mode-enable-string-interpolation
(string= content-type "javascript")
(>= (- end beg) 6))
(web-mode-interpolate-javascript-string beg end))
) ;face
;;((eq token-type 'html)
;; (message "html : %S %S" beg end)
;; ;; (remove-text-properties beg end '(face nil))
;; (web-mode-highlight-tags beg end)
;; ;;TODO : placer le traitement des part expr ici
;; )
) ;cond
(setq continue nil
end nil)
) ;if end
) ;progn beg
(setq continue nil
end nil)
) ;if beg
(when (and beg end
web-mode-enable-comment-keywords
(eq token-type 'comment)
(> (- end beg) 3))
(web-mode-interpolate-comment beg end t))
) ;while
(when web-mode-enable-part-face
(font-lock-append-text-property reg-beg reg-end 'face 'web-mode-part-face))
(when (string= content-type "jsx")
(goto-char reg-beg)
;;(web-mode-highlight-tags reg-beg reg-end)
(setq continue t
end reg-beg)
(while continue
(setq beg (next-single-property-change end 'part-expr)
end nil)
;; (message "beg=%S reg-end=%S" beg reg-end)
(if (and beg (< beg reg-end))
(progn
(setq end (next-single-property-change beg 'part-expr))
;; (message "bounds %S %S" beg end)
(if (and end (< end reg-end))
(progn
;; (remove-text-properties beg end '(face nil))
(put-text-property beg (1+ beg) 'font-lock-face 'web-mode-block-delimiter-face)
(put-text-property (1- end) end 'font-lock-face 'web-mode-block-delimiter-face)
(web-mode-fontify-region (1+ beg) (1- end) web-mode-javascript-font-lock-keywords)
) ;progn
(setq continue nil
end nil))
) ;progn
(setq continue nil
end nil))
) ;while
) ;when jsx
)))
(defun web-mode-css-rules-highlight (part-beg part-end)
"Scan CSS rules."
(save-excursion
(goto-char part-beg)
(let (rule (continue t) (i 0) (at-rule nil))
(while continue
(setq rule (web-mode-css-rule-next part-end))
(cond
((> (setq i (1+ i)) 1000)
(message "css-rules-highlight ** too much rules **")
(setq continue nil))
((null rule)
(setq continue nil))
((and (setq at-rule (plist-get rule :at-rule))
(not (member at-rule '("charset" "font-face" "import")))
(plist-get rule :dec-end))
(web-mode-css-rule-highlight (plist-get rule :sel-beg)
(plist-get rule :sel-end)
nil nil)
(web-mode-css-rules-highlight (plist-get rule :dec-beg)
(plist-get rule :dec-end))
)
(t
(web-mode-css-rule-highlight (plist-get rule :sel-beg)
(plist-get rule :sel-end)
(plist-get rule :dec-beg)
(plist-get rule :dec-end))
)
) ;cond
)
)
))
(defun web-mode-css-rule-highlight (sel-beg sel-end dec-beg dec-end)
"Fontify css rule."
(save-excursion
;; (message "sel-beg=%S sel-end=%S dec-beg=%S dec-end=%S" sel-beg sel-end dec-beg dec-end)
(web-mode-fontify-region sel-beg sel-end
web-mode-selector-font-lock-keywords)
(when (and dec-beg dec-end)
(web-mode-fontify-region dec-beg dec-end
web-mode-declaration-font-lock-keywords)
(goto-char dec-beg)
(while (and web-mode-enable-css-colorization
(re-search-forward "#[0-9a-fA-F]\\{6\\}\\|#[0-9a-fA-F]\\{3\\}\\|rgb([ ]*\\([[:digit:]]\\{1,3\\}\\)[ ]*,[ ]*\\([[:digit:]]\\{1,3\\}\\)[ ]*,[ ]*\\([[:digit:]]\\{1,3\\}\\)\\(.*?\\))" dec-end t)
(< (point) dec-end))
(web-mode-colorize (match-beginning 0) (match-end 0))
)
)))
(defun web-mode-fontify-region (beg end keywords)
"Font-lock region according to the keywords."
;; (message "beg=%S end=%S" beg end);; (symbol-name keywords))
(save-excursion
(let ((font-lock-keywords keywords)
(font-lock-multiline nil)
(font-lock-keywords-case-fold-search
(member web-mode-engine '("asp" "template-toolkit")))
(font-lock-keywords-only t)
(font-lock-extend-region-functions nil))
;; (message "%S" keywords)
(when (listp font-lock-keywords)
(font-lock-fontify-region beg end)
)
)
))
(defun web-mode-colorize-foreground (color)
"Colorize foreground based on background luminance."
(let* ((values (x-color-values color))
(r (car values))
(g (cadr values))
(b (car (cdr (cdr values)))))
(if (> 128.0 (floor (+ (* .3 r) (* .59 g) (* .11 b)) 256))
"white" "black")))
(defun web-mode-colorize (beg end)
"Colorize CSS colors."
(let (str plist len)
(setq str (buffer-substring-no-properties beg end))
(setq len (length str))
(cond
((string= (substring str 0 1) "#")
(setq plist (list :background str
:foreground (web-mode-colorize-foreground str)))
(put-text-property beg end 'face plist))
((string= (substring str 0 4) "rgb(")
(setq str (format "#%02X%02X%02X"
(string-to-number (match-string-no-properties 1))
(string-to-number (match-string-no-properties 2))
(string-to-number (match-string-no-properties 3))))
(setq plist (list :background str
:foreground (web-mode-colorize-foreground str)))
(put-text-property beg end 'face plist))
) ;cond
))
(defun web-mode-interpolate-block-tag (beg end)
"Scan a block tag (jsp / mako) to fontify ${ } blocks"
(save-excursion
(goto-char (+ 4 beg))
(setq end (1- end))
(while (re-search-forward "${.*?}" end t)
(remove-text-properties (match-beginning 0) (match-end 0)
'(font-lock-face nil))
(web-mode-fontify-region (match-beginning 0) (match-end 0)
web-mode-uel-font-lock-keywords))
))
(defun web-mode-interpolate-javascript-string (beg end)
"Scan a js string to fontify ${ } vars"
(save-excursion
(goto-char (1+ beg))
(setq end (1- end))
(while (re-search-forward "${.*?}" end t)
(put-text-property (match-beginning 0) (match-end 0)
'font-lock-face
'web-mode-variable-name-face)
)
))
;; todo : parsing plus compliqué: {$obj->values[3]->name}
(defun web-mode-interpolate-block-string (beg end)
"Interpolate php/erb strings."
(save-excursion
(goto-char (1+ beg))
(setq end (1- end))
(cond
((string= web-mode-engine "php")
(while (re-search-forward "$[[:alnum:]_]+\\(->[[:alnum:]_]+\\)*\\|{[ ]*$.+?}" end t)
;; (message "%S > %S" (match-beginning 0) (match-end 0))
(remove-text-properties (match-beginning 0) (match-end 0) '(font-lock-face nil))
(web-mode-fontify-region (match-beginning 0) (match-end 0)
web-mode-php-var-interpolation-font-lock-keywords)
))
((string= web-mode-engine "erb")
(while (re-search-forward "#{.*?}" end t)
(remove-text-properties (match-beginning 0) (match-end 0) '(font-lock-face nil))
(put-text-property (match-beginning 0) (match-end 0)
'font-lock-face 'web-mode-variable-name-face)
))
) ;cond
))
(defun web-mode-interpolate-comment (beg end block-side)
"Interpolate comment"
(save-excursion
(let ((regexp (concat "\\<\\(" web-mode-comment-keywords "\\)\\>")))
(goto-char beg)
(while (re-search-forward regexp end t)
(font-lock-prepend-text-property (match-beginning 1) (match-end 1)
'font-lock-face
'web-mode-comment-keyword-face)
) ;while
)))
(defun web-mode-interpolate-sql-string (beg end)
"Interpolate comment"
(save-excursion
(let ((regexp (concat "\\<\\(" web-mode-sql-keywords "\\)\\>")))
(goto-char beg)
(while (re-search-forward regexp end t)
(font-lock-prepend-text-property (match-beginning 1) (match-end 1)
'font-lock-face
'web-mode-sql-keyword-face)
) ;while
)))
(defun web-mode-make-tag-overlays ()
(unless web-mode-start-tag-overlay
(setq web-mode-start-tag-overlay (make-overlay 1 1)
web-mode-end-tag-overlay (make-overlay 1 1))
(overlay-put web-mode-start-tag-overlay
'font-lock-face
'web-mode-current-element-highlight-face)
(overlay-put web-mode-end-tag-overlay
'font-lock-face
'web-mode-current-element-highlight-face)))
(defun web-mode-delete-tag-overlays ()
(when web-mode-start-tag-overlay
(delete-overlay web-mode-start-tag-overlay)
(delete-overlay web-mode-end-tag-overlay)))
(defun web-mode-column-overlay-factory (index)
(let (overlay)
(when (null web-mode-column-overlays)
(dotimes (i 100)
(setq overlay (make-overlay 1 1))
(overlay-put overlay 'font-lock-face 'web-mode-current-column-highlight-face)
(setq web-mode-column-overlays (append web-mode-column-overlays (list overlay)))
)
)
(setq overlay (nth index web-mode-column-overlays))
(when (null overlay)
;; (message "new overlay(%S)" index)
(setq overlay (make-overlay 1 1))
(overlay-put overlay 'font-lock-face 'web-mode-current-column-highlight-face)
(setq web-mode-column-overlays (append web-mode-column-overlays (list overlay)))
) ;when
overlay))
(defun web-mode-column-hide ()
(remove-overlays (point-min) (point-max) 'font-lock-face 'web-mode-current-column-highlight-face))
(defun web-mode-column-show (column line-from line-to)
(let (current index overlay)
(when (> line-from line-to)
(let (tmp)
(setq tmp line-from)
(setq line-from line-to)
(setq line-to tmp)
))
(setq current line-from
index 0)
;; (message "column(%S) from(%S) to(%S)" column line-from line-to)
(web-mode-with-silent-modifications
(save-excursion
(goto-char (point-min))
(when (> line-from 1)
(forward-line (1- line-from)))
(while (<= current line-to)
(move-to-column (1+ column) t)
(backward-char)
(setq overlay (web-mode-column-overlay-factory index))
(move-overlay overlay (point) (1+ (point)))
(setq current (1+ current))
(forward-line)
(setq index (1+ index))
)
) ;save-excursion
) ;silent
))
(defun web-mode-highlight-current-element ()
(let ((ctx (web-mode-element-boundaries)))
;; (message "%S" ctx)
(if (null ctx)
(web-mode-delete-tag-overlays)
(web-mode-make-tag-overlays)
(move-overlay web-mode-start-tag-overlay (caar ctx) (1+ (cdar ctx)))
(move-overlay web-mode-end-tag-overlay (cadr ctx) (1+ (cddr ctx))))
))
(defun web-mode-fill-paragraph (&optional justify)
"fill paragraph"
(save-excursion
(let ((pos (point)) fill-coll
prop pair beg end delim-beg delim-end chunk fill-col)
(cond
((or (eq (get-text-property pos 'part-token) 'comment)
(eq (get-text-property pos 'block-token) 'comment))
(setq prop
(if (get-text-property pos 'part-token) 'part-token 'block-token))
(setq pair (web-mode-property-boundaries prop pos))
(when (and pair (> (- (cdr pair) (car pair)) 6))
(setq fill-coll (if (< fill-column 10) 70 fill-column))
(setq beg (car pair)
end (cdr pair))
(goto-char beg)
(setq chunk (buffer-substring-no-properties beg (+ beg 2)))
(cond
((string= chunk "//")
(setq delim-beg "//"
delim-end "EOL"))
((string= chunk "/*")
(setq delim-beg "/*"
delim-end "*/"))
((string= chunk "{#")
(setq delim-beg "{#"
delim-end "#}"))
((string= chunk ""))
)
)
) ;comment - case
((web-mode-is-content)
(setq pair (web-mode-content-boundaries pos))
(setq beg (car pair)
end (cdr pair))
)
) ;cond
;; (message "beg(%S) end(%S)" beg end)
(when (and beg end)
(fill-region beg end))
t)))
(defun web-mode-property-boundaries (prop &optional pos)
"property boundaries (cdr is 1+)"
(unless pos (setq pos (point)))
(let (beg end val)
(setq val (get-text-property pos prop))
(if (null val)
val
(if (or (bobp)
(not (eq (get-text-property (1- pos) prop) val)))
(setq beg pos)
(setq beg (previous-single-property-change pos prop))
(when (null beg) (setq beg (point-min))))
(if (or (eobp)
(not (eq (get-text-property (1+ pos) prop) val)))
(setq end pos)
(setq end (next-single-property-change pos prop))
(when (null end) (setq end (point-min))))
(cons beg end))))
;; verifier avec text-property-any si 'block-side
(defun web-mode-content-apply (&optional fun)
"web-mode-content-apply"
(interactive)
(save-excursion
(let ((beg nil) (i 0) (continue t))
(goto-char (point-min))
(when (get-text-property (point) 'tag-type)
(web-mode-tag-end)
(setq beg (point)))
(while (and continue
(or (get-text-property (point) 'tag-beg)
(web-mode-tag-next)))
(when (> (setq i (1+ i)) 2000)
(message "content-apply ** crazy loop **")
(setq continue nil))
(when (and beg (> (point) beg))
(message "content=%S > %S" beg (point)))
(if (web-mode-tag-end)
(setq beg (point))
(setq continue nil))
) ;while
)))
(defun web-mode-content-boundaries (&optional pos)
"Text boundaries"
(unless pos (setq pos (point)))
(let (beg end)
(setq beg (or (previous-property-change pos (current-buffer))
(point-max)))
(setq end (or (next-property-change pos (current-buffer))
(point-min)))
(while (and (< beg end) (member (char-after beg) '(?\s ?\n)))
(setq beg (1+ beg)))
(while (and (> end beg) (member (char-after (1- end)) '(?\s ?\n)))
(setq end (1- end)))
;; (message "beg(%S) end(%S)" beg end)
(cons beg end)
))
(defun web-mode-coord-pos (line column)
"Return pos at Line / Column pos"
(save-excursion
(when (stringp line) (setq line (string-to-number line)))
(when (stringp column) (setq column (string-to-number column)))
(goto-char (point-min))
(forward-line (1- line))
(move-to-column (1- column))
(point)))
(defun web-mode-engine-syntax-check ()
(interactive)
(let ((proc nil)
(errors nil)
(file (concat temporary-file-directory "emacs-web-mode-tmp")))
(write-region (point-min) (point-max) file)
(cond
;; ((null (buffer-file-name))
;; )
((string= web-mode-engine "php")
(setq proc (start-process "php-proc" nil "php" "-l" file))
(set-process-filter proc
(lambda (proc output)
(cond
((string-match-p "No syntax errors" output)
(message "No syntax errors")
)
(t
;; (setq output (replace-regexp-in-string temporary-file-directory "" output))
;; (message output)
(message "Syntax error")
(setq errors t))
) ;cond
;; (delete-file file)
) ;lambda
)
) ;php
(t
(message "no syntax checker found")
) ;t
) ;cond
errors))
(defun web-mode-jshint ()
"Run JSHint on all the JavaScript parts."
(interactive)
(let (proc lines)
(when (buffer-file-name)
(setq proc (start-process
"jshint-proc"
nil
"jshint" "--extract=auto" (buffer-file-name)))
(setq web-mode-jshint-errors 0)
(set-process-filter proc
(lambda (proc output)
(let ((offset 0) overlay pos (old 0) msg)
(remove-overlays (point-min) (point-max) 'font-lock-face 'web-mode-error-face)
(while (string-match
"line \\([[:digit:]]+\\), col \\([[:digit:]]+\\), \\(.+\\)\\.$"
output offset)
(setq web-mode-jshint-errors (1+ web-mode-jshint-errors))
(setq offset (match-end 0))
(setq pos (web-mode-coord-pos (match-string-no-properties 1 output)
(match-string-no-properties 2 output)))
(when (get-text-property pos 'tag-beg)
(setq pos (1- pos)))
(when (not (= pos old))
(setq old pos)
(setq overlay (make-overlay pos (1+ pos)))
(overlay-put overlay 'font-lock-face 'web-mode-error-face)
)
(setq msg (or (overlay-get overlay 'help-echo)
(concat "l="
(match-string-no-properties 1 output)
" c="
(match-string-no-properties 2 output)
)))
(overlay-put overlay 'help-echo
(concat msg " ## " (match-string-no-properties 3 output)))
) ;while
))
)
) ;when
))
(defun web-mode-dom-errors-show ()
"Show unclosed tags."
(interactive)
(let (beg end tag pos l n tags i cont cell overlay overlays first
(ori (point))
(errors 0)
(continue t)
)
(setq overlays (overlays-in (point-min) (point-max)))
(when overlays
(dolist (overlay overlays)
(when (eq (overlay-get overlay 'face) 'web-mode-warning-face)
(delete-overlay overlay)
)
)
)
(goto-char (point-min))
(when (not (or (get-text-property (point) 'tag-beg)
(web-mode-tag-next)))
(setq continue nil))
(while continue
(setq pos (point))
(setq tag (get-text-property pos 'tag-name))
(cond
((eq (get-text-property (point) 'tag-type) 'start)
(setq tags (add-to-list 'tags (list tag pos)))
;; (message "(%S) opening %S" pos tag)
)
((eq (get-text-property (point) 'tag-type) 'end)
(setq i 0
l (length tags)
cont t)
(while (and (< i l) cont)
(setq cell (nth i tags))
;; (message "cell=%S" cell)
(setq i (1+ i))
(cond
((string= tag (nth 0 cell))
(setq cont nil)
)
(t
(setq errors (1+ errors))
(setq beg (nth 1 cell))
(setq end (web-mode-tag-end-position beg))
(unless first
(setq first beg))
(setq overlay (make-overlay beg (1+ end)))
(overlay-put overlay 'font-lock-face 'web-mode-warning-face)
;; (message "invalid <%S> at %S" (nth 0 cell) (nth 1 cell))
)
) ;cond
) ;while
(dotimes (i i)
(setq tags (cdr tags)))
)
) ;cond
(when (not (web-mode-tag-next))
(setq continue nil))
) ;while
(message "%S error(s) detected" errors)
(if (> errors 0)
(progn (goto-char first)
(recenter))
(goto-char ori)
) ;if
;; (message "%S" tags)
))
(defun web-mode-highlight-elements (beg end)
"web-mode-highlight-elements"
(save-excursion
(goto-char beg)
(let ((continue (or (get-text-property (point) 'tag-beg) (web-mode-tag-next)))
(i 0) (ctx nil) (face nil))
(while continue
(cond
((> (setq i (1+ i)) 1000)
(message "highlight-elements ** too much tags **")
(setq continue nil))
((> (point) end)
(setq continue nil))
((not (get-text-property (point) 'tag-beg))
(setq continue nil))
((eq (get-text-property (point) 'tag-type) 'start)
(when (and (setq ctx (web-mode-element-boundaries (point)))
(<= (car (cdr ctx)) end)
(setq face (cdr (assoc (get-text-property (point) 'tag-name) web-mode-element-content-faces))))
(font-lock-prepend-text-property (1+ (cdr (car ctx))) (car (cdr ctx))
'font-lock-face face))
)
) ;cond
(when (not (web-mode-tag-next))
(setq continue nil))
) ;while
)))
(defun web-mode-highlight-whitespaces (beg end)
"Scan whitespaces."
(save-excursion
(goto-char beg)
(while (re-search-forward web-mode-whitespaces-regexp end t)
(add-text-properties (match-beginning 0) (match-end 0)
'(face web-mode-whitespace-face))
) ;while
))
(defun web-mode-whitespaces-show ()
"Toggle whitespaces."
(interactive)
(if web-mode-enable-whitespace-fontification
(web-mode-whitespaces-off)
(web-mode-whitespaces-on)))
(defun web-mode-whitespaces-on ()
"Show whitespaces."
(interactive)
(when web-mode-hl-line-mode-flag
(global-hl-line-mode -1))
(when web-mode-display-table
(setq buffer-display-table web-mode-display-table))
(setq web-mode-enable-whitespace-fontification t))
(defun web-mode-whitespaces-off ()
"Hide whitespaces."
(when web-mode-hl-line-mode-flag
(global-hl-line-mode 1))
(setq buffer-display-table nil)
(setq web-mode-enable-whitespace-fontification nil))
(defun web-mode-buffer-indent ()
"Indent all buffer."
(interactive)
(indent-region (point-min) (point-max))
(delete-trailing-whitespace))
(defun web-mode-buffer-change-tag-case (&optional type)
"Change HTML tag case."
(interactive)
(save-excursion
(goto-char (point-min))
(let ((continue t) f)
(setq f (if (member type '("upper" "uppercase" "upper-case")) 'uppercase 'downcase))
(when (and (not (get-text-property (point) 'tag-beg))
(not (web-mode-tag-next)))
(setq continue nil))
(while continue
(skip-chars-forward "= pos-close pos)))
(if ret
(cons pos-open pos-close)
ret)
)))
;; voir line-number-at-pos
(defun web-mode-line-number (&optional pos)
"Return line number at point."
(unless pos (setq pos (point)))
(let (ret)
(setq ret (+ (count-lines 1 pos)
(if (= (web-mode-column-at-pos pos) 0) 1 0)))
ret))
(defun web-mode-clean-client-line (input)
"Remove comments and server scripts."
(let ((out "")
(beg 0)
(keep t)
(n (length input)))
(dotimes (i n)
(if (or (get-text-property i 'block-side input)
(eq (get-text-property i 'part-token input) 'comment)
(eq (get-text-property i 'tag-type input) 'comment))
(when keep
(setq out (concat out (substring input beg i))
beg 0
keep nil))
(when (null keep)
(setq beg i
keep t))
) ;if
) ;dotimes
(if (> beg 0) (setq out (concat out (substring input beg n))))
(setq out (if (= (length out) 0) input out))
(web-mode-trim out)
;; (message "%S [%s] > [%s]" beg input out)
))
(defun web-mode-clean-server-line (input)
"Remove comments from server line."
(let ((out "")
(beg 0)
(keep t)
(n (length input)))
(dotimes (i n)
(if (or (not (get-text-property i 'block-side input))
(member (get-text-property i 'block-token input) '(comment delimiter)))
(when keep
(setq out (concat out (substring input beg i))
beg 0
keep nil))
(when (null keep)
(setq beg i
keep t))
) ;if
) ;dotimes
(if (> beg 0) (setq out (concat out (substring input beg n))))
(setq out (if (= (length out) 0) input out))
(web-mode-trim out)
;; (message "%S [%s] > [%s]" beg input out)
))
(defun web-mode-language-at-pos (&optional pos)
"Return the language at pos."
(unless pos (setq pos (point)))
(cond
((get-text-property pos 'block-side)
web-mode-engine)
((get-text-property pos 'part-side)
(symbol-name (get-text-property pos 'part-side)))
(t
web-mode-content-type)
) ;cond
)
(defun web-mode-column-at-pos (&optional pos)
"Column at point"
(unless pos (setq pos (point)))
(save-excursion
(goto-char pos)
(current-column)
))
;; renommer block-beg en reg-beg
(defun web-mode-point-context (pos)
"POS should be at the beginning of the indentation.
Return ctx = plist containing
:block-beg, :block-column,
:first-char, :line (trimmed)
:type (live, comment, string),
:language (html, php, jsp, aspx, asp + javascript, css),
:indent-offset
:prev-line :prev-char :prev-props :prev-indentation"
(save-excursion
(let (ctx pos-min
block-beg block-column first-char line type language indent-offset
prev prev-line prev-char prev-props prev-indentation)
(setq pos-min (point-min))
(setq block-beg pos-min
block-column 0
type "live"
language ""
prev-line ""
prev-char 0)
(cond
((bobp)
(setq language "html")
)
((string= web-mode-content-type "css")
(setq language "css"
indent-offset web-mode-css-indent-offset))
((member web-mode-content-type '("javascript" "json"))
(setq language "javascript"
indent-offset web-mode-code-indent-offset))
((member web-mode-content-type '("jsx"))
(setq language "jsx"
indent-offset web-mode-code-indent-offset)
(when (and (> pos (point-min))
(get-text-property pos 'part-expr)
(get-text-property (1- pos) 'part-expr))
(setq language "javascript")
(setq block-beg (1+ (previous-single-property-change pos 'part-expr)))
(goto-char block-beg)
(setq block-column (current-column))
)
)
((string= web-mode-content-type "php")
(setq language "php"
indent-offset web-mode-code-indent-offset))
((or (string= web-mode-content-type "xml"))
(setq language "xml"
indent-offset web-mode-markup-indent-offset))
((and (get-text-property pos 'tag-beg)
(get-text-property pos 'tag-name)
(not (get-text-property pos 'part-side)))
(setq language "html"
indent-offset web-mode-markup-indent-offset)
)
((and (get-text-property pos 'block-side)
(not (get-text-property pos 'block-beg)))
(setq block-beg (or (web-mode-block-beginning-position pos) pos-min))
(goto-char block-beg)
(setq block-column (current-column))
(setq language web-mode-engine)
(setq indent-offset web-mode-code-indent-offset)
(cond
((and (string= web-mode-engine "php")
(not (web-mode-looking-at "<\\?\\(=\\|php\\)?[ ]*$" block-beg)))
(save-excursion
(goto-char block-beg)
(looking-at "<\\?\\(=\\|php\\)?[ ]*")
(goto-char (match-end 0))
(setq block-column (current-column))
)
)
((string= web-mode-engine "blade")
(setq block-beg (+ block-beg 2)
block-column (+ block-column 2))
)
((string= web-mode-engine "razor")
(setq block-beg (+ block-beg 2))
)
((string= web-mode-engine "template-toolkit")
(setq block-beg (+ block-beg 3)
block-column (+ block-column 3))
)
((and (string= web-mode-engine "jsp")
(web-mode-looking-at "<%@\\|<[[:alpha:]]" block-beg))
(save-excursion
(goto-char block-beg)
(looking-at "<%@[ ]*[[:alpha:]]+[ ]+\\|?[[:alpha:]]+:[[:alpha:]]+[ ]+")
(goto-char (match-end 0))
(setq block-column (current-column))
)
)
((and (string= web-mode-engine "django")
(not (web-mode-looking-at "{{[{]?[ ]*$" block-beg)))
(save-excursion
(goto-char block-beg)
(looking-at "{{[{]?[ ]*")
(goto-char (match-end 0))
(setq block-column (current-column))
)
)
((and (string= web-mode-engine "freemarker")
(web-mode-looking-at "<@\\|<%@\\|<[[:alpha:]]" block-beg))
(save-excursion
(goto-char block-beg)
(looking-at "<@[[:alpha:].]+[ ]+\\|<%@[ ]*[[:alpha:]]+[ ]+\\|<[[:alpha:]]+:[[:alpha:]]+[ ]+")
(goto-char (match-end 0))
(setq block-column (current-column))
)
)
) ;cond
) ;block-side
((get-text-property pos 'part-side)
;; (message "pos-min=%S block-beg=%S part-beg=%S" pos-min block-beg (web-mode-part-beginning-position pos))
(setq block-beg (web-mode-part-beginning-position pos))
;; (message "block-beg %S" block-beg)
(setq block-beg (or block-beg pos-min))
(goto-char block-beg)
(search-backward "<" nil t)
(setq block-column (current-column))
(setq language (symbol-name (get-text-property pos 'part-side)))
(cond
((string= language "css")
(setq indent-offset web-mode-css-indent-offset)
)
((string= language "jsx")
(setq indent-offset web-mode-code-indent-offset)
)
(t
(setq language "javascript"
indent-offset web-mode-code-indent-offset)
)
)
) ;part-side
(t
(setq language "html"
indent-offset web-mode-markup-indent-offset)
)
) ;cond
(cond
((or (and (> pos (point-min))
(eq (get-text-property pos 'part-token) 'comment)
(eq (get-text-property (1- pos) 'part-token) 'comment)
(progn
(setq block-beg (previous-single-property-change pos 'part-token))
t))
(and (> pos (point-min))
(eq (get-text-property pos 'block-token) 'comment)
(eq (get-text-property (1- pos) 'block-token) 'comment)
(progn
(setq block-beg (previous-single-property-change pos 'block-token))
t)))
(setq type "comment"))
((or (and (> pos (point-min))
(member (get-text-property pos 'part-token) '(string context key))
(member (get-text-property (1- pos) 'part-token) '(string context key)))
(and (eq (get-text-property pos 'block-token) 'string)
(eq (get-text-property (1- pos) 'block-token) 'string)))
(setq type "string"))
)
(goto-char pos)
(setq line (web-mode-trim (buffer-substring-no-properties (line-beginning-position)
(line-end-position))))
(setq first-char (if (string= line "") 0 (aref line 0)))
(when (or (member language '("php" "javascript" "jsx" "razor"))
(and (string= language "html")
(not (eq ?\< first-char))))
(cond
((member language '("html" "javascript" "jsx"))
(setq prev (web-mode-previous-usable-client-line))
;; (message "prev-line=%S" prev-line)
(when prev
(setq prev-line (car prev)
prev-indentation (cdr prev))
(setq prev-line (web-mode-clean-client-line prev-line))
;; (message "prev-line[%s]" prev-line)
(setq prev-props (text-properties-at (1- (length prev-line)) prev-line)))
)
(t
(setq prev (web-mode-block-previous-live-line))
(when prev
(setq prev-line (car prev)
prev-indentation (cdr prev))
(setq prev-line (web-mode-clean-server-line prev-line))
;; (message "pl=%s" prev-line)
) ;when
)
) ;cond
(when (>= (length prev-line) 1)
(setq prev-char (aref prev-line (1- (length prev-line))))
(setq prev-line (substring-no-properties prev-line))
)
)
(when (string= web-mode-content-type "html")
(cond
((member language '("javascript" "jsx"))
(setq block-column (+ block-column web-mode-script-padding)))
((string= language "css")
(setq block-column (+ block-column web-mode-style-padding)))
((not (member language '("html" "razor")))
(setq block-column (+ block-column web-mode-block-padding)))
)
)
(setq ctx (list :block-beg block-beg
:block-column block-column
:first-char first-char
:line line
:type type
:language language
:indent-offset indent-offset
:prev-line prev-line
:prev-char prev-char
:prev-props prev-props
:prev-indentation prev-indentation))
;; (message "%S" ctx)
ctx
)))
(defun web-mode-indent-line ()
"Indent current line according to language."
(web-mode-propertize)
(let ((offset nil)
(char nil)
(inhibit-modification-hooks t))
(save-excursion
(back-to-indentation)
(setq char (char-after))
(let* ((pos (point))
(ctx (web-mode-point-context pos))
(block-beg (plist-get ctx :block-beg))
(block-column (plist-get ctx :block-column))
(first-char (plist-get ctx :first-char))
(line (plist-get ctx :line))
(type (plist-get ctx :type))
(language (plist-get ctx :language))
(indent-offset (plist-get ctx :indent-offset))
(prev-line (plist-get ctx :prev-line))
(prev-char (plist-get ctx :prev-char))
(prev-props (plist-get ctx :prev-props))
(prev-indentation (plist-get ctx :prev-indentation)))
(cond
((or (bobp) (= (line-number-at-pos pos) 1))
(setq offset 0)
)
((string= type "string")
(cond
((and (get-text-property pos 'block-token)
(web-mode-block-token-starts-with (concat "[ \n]*" web-mode-sql-queries)))
(save-excursion
(let (col)
;; (web-mode-block-rsb "SELECT")
(web-mode-block-token-beginning)
(skip-chars-forward "[ \"'\n]")
(setq col (current-column))
(goto-char pos)
(if (looking-at-p "\\(SELECT\\|INSERT\\|DELETE\\|UPDATE\\|FROM\\|LEFT\\|JOIN\\|WHERE\\|GROUP BY\\|LIMIT\\|HAVING\\|\)\\)")
(setq offset col)
(setq offset (+ col web-mode-sql-indent-offset)))
))
)
(t
(setq offset nil))
)
)
((string= type "comment")
(goto-char (car
(web-mode-property-boundaries
(if (eq (get-text-property pos 'part-token) 'comment)
'part-token
'block-token)
pos)))
(setq offset (current-column))
(cond
((member (buffer-substring-no-properties (point) (+ (point) 2)) '("/*" "{*" "@*"))
(cond
((eq ?\* first-char)
(setq offset (1+ offset)))
(t
(setq offset (+ offset 3)))
) ;cond
)
((and (string= web-mode-engine "django")
(looking-back "{% comment %}"))
(setq offset (- offset 12))
)
((and (string= web-mode-engine "mako")
(looking-back "<%doc%>"))
(setq offset (- offset 6))
)
) ;cond
) ;case comment
;; ((and (string= language "html")
;; (string= web-mode-engine "razor")
;; (get-text-property pos 'block-side)
;; (string-match-p "^}" line))
;; (goto-char (web-mode-opening-paren-position (point)))
;; (back-to-indentation)
;; (setq offset (current-column))
;; )
((and (or (web-mode-block-is-close pos)
(web-mode-block-is-inside pos))
(web-mode-block-match))
(setq offset (current-indentation))
)
((eq (get-text-property pos 'tag-type) 'end)
(when (web-mode-tag-match)
(setq offset (current-indentation)))
)
;; TODO: il suffit de regarder si on est dans un 'tag-name
((and prev-props (plist-get prev-props 'tag-attr))
(web-mode-tag-beginning)
(let ((skip (next-single-property-change (point) 'tag-attr)))
(when skip
(goto-char skip)
(setq offset (current-column))
))
)
((and (get-text-property pos 'tag-beg)
(not (get-text-property pos 'part-side))
(not (member web-mode-content-type '("jsx"))))
(setq offset (web-mode-markup-indentation pos))
)
((member language '("css"))
(cond
((member first-char '(?\} ?\) ?\]))
(let ((ori (web-mode-opening-paren-position pos)))
(cond
((null ori)
(message "indent-line ** invalid closing bracket (%S) **" pos)
(setq offset block-column)
)
(t
(goto-char ori)
(back-to-indentation)
(setq offset (current-indentation))
(when (> block-column offset)
(setq offset block-column))
)
)
) ;let
)
(t
(setq offset (car (web-mode-part-indentation pos
block-column
indent-offset
language
block-beg)))
)
) ;cond
)
((member language '("asp" "aspx" "blade" "code" "django" "erb"
"freemarker" "javascript" "jsp" "jsx" "lsp"
"mako" "mason" "mojolicious"
"php" "python" "razor"
"template-toolkit" "web2py"))
(cond
((and (get-text-property (1- pos) 'block-side)
(eq (get-text-property pos 'block-token) 'delimiter))
(if (web-mode-block-beginning)
(setq offset (current-column)))
)
((string= language "lsp")
(setq offset (web-mode-bracket-indentation pos
block-column
indent-offset
language
block-beg))
)
((member first-char '(?\} ?\) ?\]))
(let ((ori (web-mode-opening-paren-position pos)))
(cond
((null ori)
(message "indent-line ** invalid closing bracket (%S) **" pos)
(setq offset block-column))
(t
(goto-char ori)
(back-to-indentation)
(setq offset (current-indentation))
(when (> block-column offset)
(setq offset block-column))
) ;t
) ;cond
) ;let
)
((and (string= language "mason")
(string-match-p "%" line))
(if (web-mode-block-beginning)
(setq offset (current-column)))
)
((and (string= language "razor")
(string-match-p "^\\." line)
(string-match-p "^\\." prev-line))
(setq offset prev-indentation)
)
((and (string= language "razor")
(string-match-p "^case " line)
(string-match-p "^case " prev-line))
(search-backward "case ")
(setq offset (current-column))
)
((and (string-match-p "^[=]?%]" line)
(string= web-mode-engine "template-toolkit"))
(if (web-mode-block-beginning)
(setq offset (current-column)))
)
((and (string= language "php") (string-match-p "^->" line))
(when (web-mode-translate-backward pos "->" language block-beg)
(setq offset (current-column)))
)
((and (string= language "php") (string-match-p "\\.$" prev-line))
(cond
((and (string-match-p "\\(=\\|echo \\|return \\)" prev-line)
(web-mode-rsb "\\(=\\|echo\\|return\\)[ ]+" block-beg))
(goto-char (match-end 0))
(setq offset (current-column))
)
((string-match-p "^['\"$0-9.]" prev-line)
(setq offset prev-indentation)
)
(t
(setq offset block-column)
)
)
)
((and (member language '("php" "javascript" "jsx"))
(or (string-match-p "^else$" prev-line)
(string-match-p "^\\(if\\|for\\|foreach\\|while\\)[ ]*(.+)$" prev-line))
(not (string-match-p "^{" line)))
(setq offset (+ prev-indentation web-mode-code-indent-offset))
)
((and (member language '("javascript" "jsx"))
(eq ?\. first-char))
(when (web-mode-translate-backward pos "[[:alnum:][:blank:]]\\.[[:alpha:]]" language block-beg)
(setq offset (1+ (current-column))))
)
((and (member first-char '(?\? ?\. ?\:))
(not (member language '("erb"))))
(web-mode-rsb "[^!=][=(]" block-beg)
(setq offset (1+ (current-column)))
(when (and (string= web-mode-engine "php")
(looking-at-p " =>"))
(setq offset (1+ offset))
)
)
((and (member prev-char '(?\? ?\:))
(not (get-text-property pos 'part-element))
(not (string-match-p "^\\(case\\|default\\)[ :]" prev-line)))
;; (message "ici")
(web-mode-sb "?" block-beg)
(when (looking-back ")[ ]*")
(web-mode-sb ")" block-beg)
(goto-char (web-mode-opening-paren-position (point)))
)
(web-mode-rsb "[=(]" block-beg)
(if (eq (char-after) ?\=) (skip-chars-forward "= ") (skip-chars-forward "( "))
(setq offset (current-column))
)
((and (member prev-char '(?\. ?\+ ?\? ?\:))
(not (get-text-property pos 'part-element))
(not (string-match-p "^\\(case\\|default\\)[ :]" prev-line)))
;; (message "coucou")
(web-mode-rsb "=\\|(" block-beg)
(if (eq (char-after) ?\=) (skip-chars-forward "= ") (skip-chars-forward "( "))
(setq offset (current-column))
)
((string= language "erb")
(setq offset (web-mode-ruby-indentation pos
line
block-column
indent-offset
block-beg))
)
((member language '("mako" "web2py"))
(setq offset (web-mode-python-indentation pos
line
block-column
indent-offset
block-beg))
)
((string= language "asp")
(setq offset (web-mode-asp-indentation pos
line
block-column
indent-offset
block-beg))
)
((and (string= language "jsx")
(eq (get-text-property pos 'tag-type) 'end)
(web-mode-tag-match))
(setq offset (current-indentation))
)
((and (string= language "jsx")
(get-text-property pos 'part-element)
;;(get-text-property pos 'tag-type)
(not (and (get-text-property pos 'part-expr)
(get-text-property (1- pos) 'part-expr)))
(and prev-props (plist-get prev-props 'tag-type))
;;(eq (get-text-property (1- pos) 'part-token) 'html)
)
;;(message "jsx %S" prev-props)
(setq offset (web-mode-markup-indentation pos))
)
((member language '("javascript" "jsx"))
(setq offset (car (web-mode-part-indentation pos
block-column
indent-offset
language
block-beg)))
) ;javascript
(t
;;(message "bracket indent")
;;TODO : prendre l'origine du bracket indent
(setq offset (web-mode-bracket-indentation pos
block-column
indent-offset
language
block-beg))
) ;t
)) ;end case script block
(t ; case html
(cond
((and (string= web-mode-engine "mason")
(string-match-p "^%" line))
(setq offset 0)
)
((and web-mode-pre-elements
(null (get-text-property pos 'block-side))
(null (get-text-property pos 'part-side))
(and (null (get-text-property pos 'tag-beg))
(save-excursion
(and (web-mode-element-parent)
(member (get-text-property (point) 'tag-name) web-mode-pre-elements))))
)
;; (message "ici%S" pos)
(setq offset nil))
((or (eq (length line) 0)
(= web-mode-indent-style 2)
(get-text-property pos 'tag-beg)
(get-text-property pos 'block-beg))
;; (message "ici")
(setq offset (web-mode-markup-indentation pos))
)
) ;cond
) ;end case html block
) ;end switch language block
)) ;save-excursion
;; (message "offset=%S" offset)
(when offset
(let ((diff (- (current-column) (current-indentation))))
(when (not (= offset (current-indentation)))
(setq web-mode-change-beg (line-beginning-position)
web-mode-change-end (+ web-mode-change-beg offset)))
;; (message "offset=%S ci=%S" )
(setq offset (max 0 offset))
(indent-line-to offset)
(if (> diff 0) (forward-char diff))
(when (and (string= web-mode-engine "mason")
(= offset 0)
(eq char ?\%))
(web-mode-highlight-region (line-beginning-position) (line-end-position)))
) ;let
) ;when
))
(defun web-mode-block-is-control (pos)
"web-mode-block-is-control"
(save-excursion
(let (control state controls pair)
(goto-char pos)
(setq controls (web-mode-block-controls-get pos))
(setq pair (car controls))
(cond
((eq (car pair) 'inside)
)
((eq (car pair) 'open)
(setq state t
control (cdr pair)))
((eq (car pair) 'close)
(setq state nil
control (cdr pair)))
) ;cond
;; (message "engine=%S control=%S state=%S" web-mode-engine control state)
(if control (cons control state) nil)
)))
(defun web-mode-block-is-opening-control (pos)
"web-mode-block-is-control"
(save-excursion
(let (controls pair)
(goto-char pos)
;; (setq pair (car controls))
(if (and (setq controls (web-mode-block-controls-get pos))
(= (length controls) 1)
(setq pair (car controls))
(eq (car pair) 'open))
(cdr pair)
nil)
)))
(defun web-mode-markup-indentation-origin ()
"web-mode-indentation-origin-pos"
(let* ((continue (not (bobp)))
(pos (point))
(part-side (not (null (get-text-property pos 'part-side))))
(types '(start end void)))
(while continue
(forward-line -1)
(back-to-indentation)
(setq pos (point))
(setq continue (not (or (bobp)
(and (null part-side)
(null (get-text-property pos 'part-side))
(get-text-property pos 'tag-beg)
(member (get-text-property pos 'tag-type) types))
(and part-side
(get-text-property pos 'part-side)
(get-text-property pos 'tag-beg)
(member (get-text-property pos 'tag-type) types))
(and (get-text-property pos 'block-beg)
(web-mode-block-is-control pos)
(not (looking-at-p "{% comment"))))))
) ;while
;;(message "indent-origin=%S" pos)
pos))
;;TODO : prendre en compte part-token
;; state=t <=> start tag
(defun web-mode-element-is-opened (pos limit)
"Is there any HTML element (or control block) without a closing tag ?"
(interactive)
(let (tag
last-end-tag
tag-pos block-pos
state
n
ret
(continue t)
controls
control
(buffer (current-buffer))
(h (make-hash-table :test 'equal))
(h2 (make-hash-table :test 'equal)))
;; (message "pos-ori=%S limit=%S" pos limit)
(while continue
(setq control nil
controls nil
last-end-tag nil
tag nil)
(cond
((get-text-property pos 'tag-beg)
(when (member (get-text-property pos 'tag-type) '(start end))
(setq tag (get-text-property pos 'tag-name)
state (eq (get-text-property pos 'tag-type) 'start))
(if (null state) (setq last-end-tag (cons tag pos)))
(setq n (gethash tag h 0))
(if (null state)
(progn
(when (> n 0) (puthash tag (1- n) h))
(puthash tag (1- n) h2))
(puthash tag (1+ n) h)
(puthash tag (1+ n) h2))
) ;when
(when (setq pos (web-mode-tag-end-position pos))
(setq tag-pos nil)
(when (and block-pos (> pos block-pos))
(setq block-pos nil))
) ;when
)
((get-text-property pos 'block-beg)
(when (setq controls (web-mode-block-controls-get pos))
(dolist (control controls)
(setq tag (cdr control))
(setq n (gethash tag h 0))
(cond
((eq (car control) 'inside)
)
((eq (car control) 'open)
(puthash tag (1+ n) h))
((> n 0)
(puthash tag (1- n) h))
) ;cond
) ;dolist
)
(when (setq pos (web-mode-block-end-position pos))
(setq block-pos nil)
(when (and tag-pos (> pos tag-pos))
(setq tag-pos nil))
)
)
) ;cond
;; (message "tag=%S end-pos=%S" tag pos)
(when (and pos (< pos limit))
(when (or (null tag-pos) (>= pos tag-pos))
(setq tag-pos (web-mode-tag-next-position pos limit))
;; (message "from=%S tag-next-pos=%S" pos tag-pos)
)
(when (or (null block-pos) (>= pos block-pos))
(setq block-pos (web-mode-block-next-position pos limit))
;; (message "from=%S block-next-pos=%S" pos block-pos)
)
)
(cond
((null pos)
)
((and (null tag-pos)
(null block-pos))
(setq pos nil))
((and tag-pos block-pos)
(if (< tag-pos block-pos)
(progn
(setq pos tag-pos)
(setq tag-pos nil))
(setq pos block-pos)
(setq block-pos nil))
)
((null tag-pos)
(setq pos block-pos)
(setq block-pos nil))
(t
(setq pos tag-pos)
(setq tag-pos nil))
)
(when (or (null pos)
(>= pos limit))
(setq continue nil))
) ;while
;; (message "hashtable=%S" h)
(maphash (lambda (k v) (if (> v 0) (setq ret t))) h)
(when (and (null ret)
last-end-tag
(> (hash-table-count h2) 1)
(< (gethash (car last-end-tag) h2) 0))
;; (message "last-end-tag=%S" last-end-tag)
(save-excursion
(goto-char (cdr last-end-tag))
(web-mode-tag-match)
(when (not (= (point) (cdr last-end-tag)))
(setq n (point))
(back-to-indentation)
(if (= n (point)) (setq ret (current-indentation))))
))
ret))
(defun web-mode-markup-indentation (pos)
"markup indentation"
(save-excursion
(goto-char pos)
(let ((offset 0) beg ret)
(setq beg (web-mode-markup-indentation-origin))
(when beg
(goto-char beg)
(setq ret (web-mode-element-is-opened beg pos))
(cond
((null ret)
(setq offset (current-indentation)))
((eq ret t)
(setq offset (+ (current-indentation) web-mode-markup-indent-offset)))
(t
(setq offset ret))
) ;cond
) ;when
offset)))
;; :opened-blocks :col-num :inline-pos
(defun web-mode-ruby-indentation (pos line initial-column language-offset limit)
"Calc indent column."
(interactive)
(unless limit (setq limit nil))
(let (h out prev-line prev-indentation ctx)
(setq ctx (web-mode-count-opened-brackets pos "ruby" limit))
(if (plist-get ctx :inline-pos)
(setq out (plist-get ctx :col-num))
(setq h (web-mode-previous-line pos limit))
(setq out initial-column)
(when h
(setq prev-line (car h))
(setq prev-indentation (cdr h))
(cond
((string-match-p "^\\(end\\|else\\|elsif\\|when\\)" line)
(setq out (- prev-indentation language-offset))
)
((string-match-p "\\(when\\|if\\|else\\|elsif\\|unless\\|for\\|while\\|def\\|class\\)" prev-line)
(setq out (+ prev-indentation language-offset))
)
(t
(setq out prev-indentation)
)
)
) ;when
) ;if
out))
(defun web-mode-python-indentation (pos line initial-column language-offset limit)
"Calc indent column."
(interactive)
(unless limit (setq limit nil))
(let (h out prev-line prev-indentation ctx)
(setq h (web-mode-previous-line pos limit))
(setq out initial-column)
(when h
(setq prev-line (car h))
(setq prev-indentation (cdr h))
(cond
((string-match-p "^\\(pass\\|else\\|elif\\|when\\)" line)
(setq out (- prev-indentation language-offset))
)
((string-match-p "\\(if\\|else\\|elif\\|for\\|while\\)" prev-line)
(setq out (+ prev-indentation language-offset))
)
(t
(setq out prev-indentation)
)
) ;cond
) ;when
out))
(defun web-mode-asp-indentation (pos line initial-column language-offset limit)
"Calc indent column."
(interactive)
(unless limit (setq limit nil))
(let (h out prev-line prev-indentation)
(setq h (web-mode-previous-line pos limit))
(setq out initial-column)
(when h
(setq prev-line (car h))
(setq prev-indentation (cdr h))
(cond
;; ----------------------------------------------------------------------
;; unindent
((string-match-p "\\<\\(\\(end \\(if\\|function\\|class\\|sub\\|with\\)\\)\\|else\\|elseif\\|next\\|loop\\)\\>" line)
(setq out (- prev-indentation language-offset)))
;; ----------------------------------------------------------------------
;; select case statement
((string-match-p "\\<\\(select case\\)\\>" line)
(setq out (- prev-indentation 0)))
((string-match-p "\\<\\(end select\\)" line)
(setq out (- prev-indentation (* 2 language-offset))))
((and (string-match-p "\\<\\(case\\)\\>" line) (not (string-match-p "\\<\\(select case\\)\\>" prev-line)))
(setq out (- prev-indentation language-offset)))
;; ----------------------------------------------------------------------
;; do nothing
((string-match-p "\\<\\(\\(end \\(if\\|function\\|class\\|sub\\|select\\|with\\)\\)\\|loop\\( until\\| while\\)?\\)\\>" prev-line)
(setq out (+ prev-indentation 0)))
;; indent
((string-match-p "\\<\\(\\(select \\)?case\\|else\\|elseif\\|unless\\|for\\|class\\|with\\|do\\( until\\| while\\)?\\|while\\|\\(public \\|private \\)?\\(function\\|sub\\|class\\)\\)\\>" prev-line)
(setq out (+ prev-indentation language-offset)))
;; single line if statement
((string-match-p "\\.*\\[ \t]*[[:alpha:]]+" prev-line)
(setq out (+ prev-indentation 0)))
;; normal if statement
((string-match-p "\\<\\if\\>" prev-line)
(setq out (+ prev-indentation language-offset)))
(t
(setq out prev-indentation))
)
) ;when
out))
(defun web-mode-previous-line (pos limit)
"Previous line"
(save-excursion
(let (beg end line (continue t))
(goto-char pos)
(while continue
(forward-line -1)
(setq end (line-end-position))
(setq line (buffer-substring-no-properties (point) end))
(when (or (not (string-match-p "^[ \t]*$" line))
(bobp)
(<= (point) limit))
(setq continue nil))
)
(if (<= (point) limit)
;;todo : affiner (le + 3 n est pas générique cf. (length regexp) 3)
(setq searcher 'web-mode-rsb)
(setq searcher 'web-mode-sb))
(while continue
(cond
((> (setq i (1+ i)) 200)
(setq continue nil
pos nil)
(message "translate-backward-pos ** crazy loop **")
)
((not (funcall searcher regexp limit))
(setq continue nil
pos nil)
)
((= depth (web-mode-bracket-depth (point) language limit))
(setq continue nil
pos (point))
)
) ;cond
) ;while
;; (message "%S: %S" regexp pos)
pos)))
(defun web-mode-bracket-depth (pos language &optional limit)
"Count opened brackets at POS."
(interactive)
(unless limit (setq limit nil))
(save-excursion
(goto-char pos)
(let ((assoc (make-hash-table :test 'equal))
(char nil)
(continue t)
(regexp "[\]\[}{)(]"))
(while (and continue (re-search-backward regexp limit t))
(setq char (aref (match-string-no-properties 0) 0))
(cond
((web-mode-is-comment-or-string))
((member char '(?\{ ?\( ?\[))
(puthash char (1+ (gethash char assoc 0)) assoc))
(t ;;(member char '(?\} ?\) ?\]))
(cond
((eq char ?\)) (setq char ?\())
((eq char ?\}) (setq char ?\{))
(t (setq char ?\[)))
(puthash char (1- (gethash char assoc 0)) assoc))
) ;cond
;; (message "(%S) %c : %S" (point) char (gethash char assoc 0))
) ;while
(+ (gethash ?\( assoc 0)
(gethash ?\{ assoc 0)
(gethash ?\[ assoc 0))
)))
(defun web-mode-part-indentation (pos initial-column language-offset language &optional limit)
"Part indentation"
(let ((ctx (web-mode-bracket-up pos language limit)) indentation)
;; (message "pos(%S) initial-column(%S) language-offset(%S) language(%S) limit(%S)" pos initial-column language-offset language limit)
;; (message "part-indentation: %S" ctx)
(cond
((or (null ctx) (null (plist-get ctx :pos)))
(setq indentation initial-column))
((not (plist-get ctx :eol)) ;bracket is not at EOL
(save-excursion
(goto-char (1+ (plist-get ctx :pos)))
(skip-chars-forward "[ \t]")
(setq indentation (current-column))
)
)
((and (member language '("javascript" "jsx" "php"))
(eq (plist-get ctx :char) ?\{)
(web-mode-looking-back "switch[ ]*(.*)[ ]*" (plist-get ctx :pos))
(not (looking-at-p "case\\|default")))
(setq indentation (+ (plist-get ctx :indentation) (* language-offset 2)))
)
(t
(setq indentation (+ (plist-get ctx :indentation) language-offset))
;; (message "indentation:%S" indentation)
)
)
(cons (if (< indentation initial-column) initial-column indentation) ctx)
))
;; optim ? qd on passe à -1 remonter directement à la paren ouvrante
(defun web-mode-bracket-up (pos language &optional limit)
(unless limit (setq limit nil))
(save-excursion
(goto-char pos)
(let ((continue t)
(regexp "[\]\[}{)(]")
(key nil)
(char nil)
(n 0)
(queue nil))
(while (and continue (web-mode-part-rsb regexp limit))
(setq char (aref (match-string-no-properties 0) 0))
(setq key (cond
((eq char ?\)) ?\()
((eq char ?\}) ?\{)
((eq char ?\]) ?\[)
(t char)))
(setq n (or (plist-get queue key) 0))
(setq n (if (member char '(?\( ?\{ ?\[)) (1+ n) (1- n)))
(setq queue (plist-put queue key n))
(setq continue (< n 1))
;; (message "pos=%S char=%c n=%S queue=%S" (point) char n queue)
) ;while
(if (> n 0)
(list :pos (point)
:char char
:column (current-column)
:indentation (current-indentation)
:eol (web-mode-is-void-after))
nil)
) ;let
))
;; (plist-get ctx :block-beg)
;; :opened-blocks :col-num :inline-pos
;; cddr
(defun web-mode-bracket-indentation (pos initial-column language-offset language &optional limit)
"Calc indent column."
(interactive)
(unless limit (setq limit nil))
;; (message "limit=%S" limit)
(save-excursion
(let (offset n first-char block-info positions col block-column (continue t) ori)
(goto-char pos)
(setq first-char (char-after)
block-column initial-column)
;; (message "first-char=%c" first-char)
(while continue
(forward-line -1)
(back-to-indentation)
(cond
((or (> limit (point))
(bobp))
(setq continue nil)
)
((and (= (current-indentation) initial-column)
(not (get-text-property (point) 'block-token))
(not (eolp)))
(setq continue nil)
(setq limit (point))
)
) ;cond
) ;while
(goto-char pos)
(setq block-info (web-mode-count-opened-brackets pos language limit))
;; (message "%S" block-info)
(setq col initial-column)
(cond
((plist-get block-info :inline-arg) ;;lsp
(cond
((numberp (plist-get block-info :inline-arg))
(setq col (+ (plist-get block-info :inline-arg) (plist-get block-info :col-num))))
;; ((string= (plist-get block-info :inline-arg) "loop")
;; (setq col (+ (plist-get block-info :col-num) 5)))
(t
(setq col (+ (plist-get block-info :col-num) web-mode-code-indent-offset)))
)
)
((and (plist-get block-info :inline-pos) (not (eq first-char ?\))))
(setq col (plist-get block-info :col-num)))
((and (eq first-char ?\))
(setq ori (web-mode-opening-paren-position pos)))
(goto-char ori)
(back-to-indentation)
(setq col (current-indentation))
;; (message "col%S" col)
)
(t
(setq n (plist-get block-info :opened-blocks))
;; (message "n = %S" n)
(setq col initial-column)
(when (and (member first-char '(?\} ?\) ?\])) (> n 0))
(setq n (1- n)))
(setq col (+ initial-column (* n language-offset)))
) ;t
) ;cond
(if (< col block-column) block-column col)
)
) ;save-excursion
)
(defun web-mode-count-opened-blocks (pos &optional limit)
"Count opened opened control blocks at POS."
(save-excursion
(goto-char pos)
(let ((continue t) pair controls control type (counter 0))
(when (null limit)
(cond
((get-text-property pos 'part-side)
(setq limit (web-mode-part-beginning-position pos)))
((get-text-property pos 'block-side)
(setq limit (web-mode-block-beginning-position pos)))
) ;cond
) ;when
(while continue
(cond
((bobp)
(setq continue nil))
((not (web-mode-block-previous))
(setq continue nil)
)
((null (setq controls (web-mode-block-controls-get (point))))
)
(t
(setq pair (car (last controls)))
;; (message "pair=%S" pair)
(cond
((eq (car pair) 'open)
(setq counter (1+ counter)))
((eq (car pair) 'inside)
)
(t
(setq counter (1- counter)))
)
) ;t
) ;cond
) ;while
(if (>= counter 0) counter 0)
)))
(defun web-mode-count-opened-brackets (pos language &optional limit)
"Count opened brackets at POS."
(interactive)
(unless limit (setq limit nil))
;; (message "limit=%S" limit)
(save-excursion
(goto-char pos)
(let ((continue t)
(match "")
(switch-level 0)
(queues (make-hash-table :test 'equal))
(opened-blocks 0)
(col-num 0)
(num-opened 0)
regexp
close-char n queue inline-pos inline-checked inline-arg char line lines reg-end)
(cond
((string= language "css")
(setq regexp "[\]\[}{)(]"))
((string= language "lsp")
(setq regexp "[)(]"))
(t
(setq regexp "[\]\[}{)(]\\|[ ;\t]\\(switch\\)"))
)
;; (message "limit=%S" limit)
(while (and continue (re-search-backward regexp limit t))
;; (message "%S: %c" (point) (char-after))
(unless (web-mode-is-comment-or-string)
(setq match (match-string-no-properties 0))
;; (message "match=%S" match)
(when (> (length match) 1)
;; (message "match=%S" (match-string-no-properties 0))
(skip-chars-forward "[ \t]")
(setq match (replace-regexp-in-string "\\`[ ;\t]*" "" (replace-regexp-in-string "[ ]*\\'" "" match)))
)
(setq char (aref match 0))
;; (message "match:%S" match)
;; (when (member char '(?\{ ?\( ?\[ ?\} ?\) ?\])) (message "(%S) c=%c" (point) char))
(cond
((member char '(?\{ ?\( ?\[))
(cond
((eq char ?\() (setq close-char ?\)))
((eq char ?\{) (setq close-char ?\}))
((eq char ?\[) (setq close-char ?\])))
(setq queue (gethash char queues nil))
(setq queue (push (cons (point) (web-mode-line-number)) queue))
(puthash char queue queues)
;; (message "%c queue=%S" char queue)
(setq queue (gethash close-char queues nil))
(setq n (length queue))
(cond
((> n 0)
(setq queue (cdr queue))
(puthash close-char queue queues)
;;(message "%c queue=%S" close-char queue)
(setq queue (gethash char queues nil))
(setq queue (cdr queue))
(puthash char queue queues)
;; (message "(%S) %c queue=%S" (point) char queue)
)
((= n 0)
(setq num-opened (1+ num-opened))
;; (message "num-opened=%S %S" num-opened (point))
)
)
(when (and (= num-opened 1) (null inline-checked))
(setq inline-checked t)
;; (message "la%S" (point))
(when (not (web-mode-is-void-after)) ;(not (looking-at-p ".[ ]*$"))
(setq inline-pos t
col-num (1+ (current-column)))
(when (string= language "lsp")
(cond
((looking-at "(\\(let\\|when\\|def\\|lambda\\|with\\)")
(setq inline-arg (match-string-no-properties 1)))
((looking-at "(\\([[:alpha:]-]+[ ]+\\).+$")
(setq inline-arg (length (match-string-no-properties 1))))
)
;; (message "pos=%S %S" (point) inline-arg)
)
(when (looking-at ".[ ]+")
(setq col-num (+ col-num (1- (length (match-string-no-properties 0)))))
)
)
)
;; (message "%c queue=%S" char queue)
) ;case (?\{ ?\( ?\[)
((member char '(?\} ?\) ?\]))
(setq queue (gethash char queues nil))
(setq queue (push (point) queue))
(puthash char queue queues)
;; (message "%c queue=%S" char queue)
)
((and (string= match "switch")
(looking-at-p "switch[ ]*("))
(let (tmp)
(when (null reg-end)
(cond
((member language '("css" "javascript" "jsx"))
(setq reg-end (web-mode-part-end-position pos))
;; (message "reg-end%S" reg-end)
)
(t
(setq reg-end (web-mode-block-end-position pos))
)
)
) ;when
;; (message "reg-end=%S" reg-end)
(setq tmp (web-mode-bracket-block-end-position (point) reg-end))
(when (and tmp (< pos tmp))
;; (message "bol(%S) is inside switch(%S)" pos (point))
(setq switch-level (1+ switch-level))
)
)
) ;case switch
) ;cond
) ;unless
) ;while
(maphash
(lambda (char queue)
(when (member char '(?\{ ?\( ?\[))
;;(message "%c : " char queue)
(dolist (pair queue)
;;(message " %S" pair)
(if (not (web-mode-is-void-after (car pair)))
()
(setq line (cdr pair))
(unless (member line lines)
(push line lines))
) ;if
) ;dolist
) ;when
)
queues)
(setq opened-blocks (length lines)) ;; il faut calculer opened-blocks meme lorsque inline-pos pour code-depth
(goto-char pos)
(when (and (> switch-level 0)
(not (looking-at-p "\\(case[ ]\\|default[ :]\\)")))
(setq opened-blocks (+ opened-blocks switch-level)))
(when (member language '("jsx"))
(setq opened-blocks (+ opened-blocks (web-mode-count-opened-blocks pos)))
)
(list :opened-blocks opened-blocks
:col-num col-num
:inline-pos inline-pos
:inline-arg inline-arg)
)))
(defun web-mode-bracket-block-end-position (pos limit)
"web-mode-blk-end-pos"
(save-excursion
(goto-char pos)
(setq pos nil)
(when (and (web-mode-sf "{" limit)
(progn (backward-char) t)
(web-mode-closing-paren limit))
(setq pos (point)))
pos))
(defun web-mode-is-void-after (&optional pos)
"Only spaces or comment after (1+ pos)"
(unless pos (setq pos (point)))
(save-excursion
(setq pos (1+ pos))
(goto-char pos)
;; (message "after pos=%S" pos)
(let ((eol (line-end-position)) (continue t) c (ret t) part-side)
(setq part-side (or (member web-mode-content-type '("javascript" "css"))
(not (null (get-text-property pos 'part-side)))))
(while continue
(setq c (char-after pos))
;; (message "%S c='%c'" pos c)
(cond
((member c '(?\s ?\n)) )
((and part-side (eq (get-text-property pos 'part-token) 'comment)) )
((and part-side (get-text-property pos 'block-side)) )
((and (not part-side) (eq (get-text-property pos 'block-token) 'comment)) )
(t (setq ret nil
continue nil))
)
(when continue
(setq pos (1+ pos))
(when (>= pos eol) (setq continue nil)))
) ;while
ret)))
(defun web-mode-count-char-in-string (char string)
"Count char in string."
(let ((n 0))
(dotimes (i (length string))
(if (eq (elt string i) char)
(setq n (1+ n))))
n))
(defun web-mode-mark-and-expand ()
"Mark and expand."
(interactive)
(when (not (eq last-command this-command))
(setq web-mode-expand-previous-state nil))
(web-mode-mark (point)))
(defun web-mode-mark (pos)
"Mark at point."
(let ((beg pos) (end pos) prop reg-beg boundaries)
(if mark-active
(setq reg-beg (region-beginning))
(setq web-mode-expand-initial-pos (point)))
;; (message "regs=%S %S %S %S" (region-beginning) (region-end) (point-min) (point-max))
;; (message "before=%S" web-mode-expand-previous-state)
(cond
((and mark-active
(= (region-beginning) (point-min))
(or (= (region-end) (point-max))
(= (1+ (region-end)) (point-max))))
(deactivate-mark)
(goto-char (or web-mode-expand-initial-pos (point-min)))
(setq web-mode-expand-previous-state nil)
(recenter))
((and (member (get-text-property pos 'block-token) '(comment string))
(not (member web-mode-expand-previous-state '("block-token" "block-body" "block-side"))))
(when (eq (get-text-property pos 'block-token) (get-text-property (1- pos) 'block-token))
(setq beg (or (previous-single-property-change pos 'block-token) (point-min))))
(when (eq (get-text-property pos 'block-token) (get-text-property (1+ pos) 'block-token))
(setq end (next-single-property-change pos 'block-token)))
(set-mark beg)
(goto-char end)
(exchange-point-and-mark)
(setq web-mode-expand-previous-state "block-token"))
((and (get-text-property pos 'block-side)
(not (member web-mode-expand-previous-state '("block-body" "block-side")))
(not (member web-mode-engine '(django go)))
(setq boundaries (web-mode-in-code-block "{" "}" 'block-side)))
(set-mark (car boundaries))
(goto-char (cdr boundaries))
(exchange-point-and-mark)
(setq web-mode-expand-previous-state "block-body"))
((and (get-text-property pos 'block-side)
(not (member web-mode-expand-previous-state '("block-side"))))
(set-mark (web-mode-block-beginning-position pos))
(goto-char (1+ (web-mode-block-end-position pos)))
(exchange-point-and-mark)
(setq web-mode-expand-previous-state "block-side"))
((and (get-text-property pos 'part-token)
(not (string= web-mode-expand-previous-state "part-token")))
(when (eq (get-text-property pos 'part-token) (get-text-property (1- pos) 'part-token))
(setq beg (previous-single-property-change pos 'part-token)))
(when (eq (get-text-property pos 'part-token) (get-text-property (1+ pos) 'part-token))
(setq end (next-single-property-change pos 'part-token)))
(set-mark beg)
(goto-char end)
(exchange-point-and-mark)
(setq web-mode-expand-previous-state "part-token"))
((and (get-text-property pos 'part-side)
(not (string= web-mode-expand-previous-state "client-part"))
(setq boundaries (web-mode-in-code-block "{" "}" 'part-side)))
(set-mark (car boundaries))
(goto-char (cdr boundaries))
(exchange-point-and-mark)
(setq web-mode-expand-previous-state "client-part"))
((and (get-text-property pos 'part-side)
(not (string= web-mode-expand-previous-state "part-side")))
(when (eq (get-text-property pos 'part-side) (get-text-property (1- pos) 'part-side))
(setq beg (previous-single-property-change pos 'part-side)))
(when (eq (get-text-property pos 'part-side) (get-text-property (1+ pos) 'part-side))
(setq end (next-single-property-change pos 'part-side)))
(when (eq (char-after beg) ?\n)
(setq beg (1+ beg)))
(set-mark beg)
(goto-char end)
(when (looking-back "^[ \t]+")
(beginning-of-line))
(exchange-point-and-mark)
(setq web-mode-expand-previous-state "part-side"))
((and (get-text-property pos 'tag-attr)
(not (member web-mode-expand-previous-state '("html-attr" "html-tag"))))
(web-mode-attribute-select pos)
(setq web-mode-expand-previous-state "html-attr"))
((and (eq (get-text-property pos 'tag-type) 'comment)
(not (member web-mode-expand-previous-state '("html-tag" "html-comment" "html-elt" "html-parent"))))
(web-mode-tag-select)
(setq web-mode-expand-previous-state "html-comment"))
((and (get-text-property pos 'tag-name)
(not (member web-mode-expand-previous-state '("html-tag" "html-elt" "html-parent"))))
(web-mode-tag-select)
(setq web-mode-expand-previous-state "html-tag"))
((and (get-text-property pos 'tag-beg)
(string= web-mode-expand-previous-state "html-tag"))
(web-mode-element-select)
(setq web-mode-expand-previous-state "html-elt"))
(t
(web-mode-element-parent)
(if (and reg-beg (= reg-beg (region-beginning)))
(progn
(push-mark (point))
(push-mark (point-max) nil t)
(goto-char (point-min))
(setq web-mode-expand-previous-state "mark-whole"))
(web-mode-element-select)
(setq web-mode-expand-previous-state "html-parent"))
) ;t
) ;cond
;; (message "after=%S\n-----------------------" web-mode-expand-previous-state)
))
(defun web-mode-block-select ()
"Select the current block."
(interactive)
(let ((beg (web-mode-block-beginning-position (point))))
(when beg
(goto-char beg)
(set-mark (point))
(web-mode-block-end)
(exchange-point-and-mark))
beg))
(defun web-mode-tag-select ()
"Select the current HTML tag."
(interactive)
(let ((beg (web-mode-tag-beginning-position (point))))
(when beg
(goto-char beg)
(set-mark (point))
(web-mode-tag-end)
(exchange-point-and-mark))
beg))
(defun web-mode-element-content-select ()
"Select the content of a HTML element."
(interactive)
(let (pos beg end)
(web-mode-element-select)
(when mark-active
(setq pos (point))
(deactivate-mark)
(web-mode-tag-match)
(setq end (point))
(goto-char pos)
(web-mode-tag-end)
(set-mark (point))
(goto-char end)
(exchange-point-and-mark)
)))
(defun web-mode-element-select ()
"Select the current HTML element (including opening and closing tags)."
(interactive)
(let* ((pos (point))
(type (get-text-property pos 'tag-type)))
(if type
(cond
((member type '(start void))
(web-mode-tag-beginning)
(set-mark (point))
(web-mode-tag-match)
(web-mode-tag-end)
(exchange-point-and-mark))
(t
(web-mode-tag-match)
(set-mark (point))
(web-mode-tag-match)
(web-mode-tag-end)
(exchange-point-and-mark))
) ;cond
(web-mode-element-parent)
(unless (= (point) pos) (web-mode-element-select))
) ;if
))
(defun web-mode-element-is-collapsed (&optional pos)
"Is the HTML element collapsed ?"
(unless pos (setq pos (point)))
(let (boundaries ret)
(setq ret (and (setq boundaries (web-mode-element-boundaries pos))
(or (= (car (car boundaries)) (car (cdr boundaries)))
(= (cdr (car boundaries)) (1- (car (cdr boundaries)))))
))
;; (when ret (message "elt(%S) is collapsed" pos))
ret))
(defun web-mode-element-transpose ()
"Transpose two HTML elements."
(interactive)
(let (pos start1 end1 start2 end2)
(save-excursion
(setq pos (point))
(cond
((get-text-property pos 'tag-type)
(setq start1 (web-mode-element-beginning-position pos)
end1 (1+ (web-mode-element-end-position pos)))
)
((setq start1 (web-mode-element-parent-position pos))
(setq end1 (1+ (web-mode-element-end-position pos)))
)
) ;cond
(when (and start1 end1 (> end1 0))
(goto-char end1)
(unless (get-text-property (point) 'tag-beg)
(skip-chars-forward "\n\t "))
(when (get-text-property (point) 'tag-beg)
(setq start2 (web-mode-element-beginning-position (point))
end2 (1+ (web-mode-element-end-position (point))))
)
)
;; (message "start1(%S) end1(%S) start2(%S) end2(%S)"
;; start1 end1 start2 end2)
(transpose-regions start1 end1 start2 end2)
) ;save-excursion
start2))
(defun web-mode-element-children-fold-or-unfold (&optional pos)
"Fold/Unfold all the children of the current HTML element."
(interactive)
(unless pos (setq pos (point)))
(let (child children)
(save-excursion
(setq children (reverse (web-mode-element-children pos)))
(dolist (child children)
;; (message "child(%S)" child)
(goto-char child)
(web-mode-fold-or-unfold)
)
)))
(defun web-mode-element-mute-blanks ()
"Mute blanks."
(interactive)
(let (pos parent beg end child children elt)
(setq pos (point))
(save-excursion
(when (and (setq parent (web-mode-element-boundaries pos))
(setq child (web-mode-element-child-position (point))))
(setq children (reverse (web-mode-element-children)))
;; (message "%S %S" parent children)
;; (setq end (car (cdr parent)))
;; (message "end=%S" end)
(goto-char (car (cdr parent)))
(dolist (child children)
(setq elt (web-mode-element-boundaries child))
;; (message "child=%S elt=%S" child elt)
(when (> (point) (1+ (cddr elt)))
;;(message "%S-->" (point))
;;(message "%S" (point))
(web-mode-insert-text-at-pos "" (point))
(web-mode-insert-text-at-pos ""))
)
)
) ;case html
((member language '("php" "javascript" "css"))
(web-mode-insert-and-indent (concat "/* " sel " */")))
((member language '("erb"))
(web-mode-insert-and-indent (replace-regexp-in-string "^" "#" sel)))
((member language '("asp"))
(web-mode-insert-and-indent (replace-regexp-in-string "^" "''" sel)))
(t
(web-mode-insert-and-indent (concat "/* " sel " */")))
) ;cond
) ;t
) ;cond
)
) ;save-excursion
;; (message "%S" (point))
;; (goto-char pos)
)
(defun web-mode-uncomment (&optional pos)
"Uncomment line(s) at point."
(interactive)
(unless pos (setq pos (point)))
(let ((beg pos) (end pos) (sub2 "") comment prop)
(cond
((and (get-text-property pos 'block-side)
(intern-soft (concat "web-mode-uncomment-" web-mode-engine "-block")))
(funcall (intern (concat "web-mode-uncomment-" web-mode-engine "-block")) pos)
)
(t
(setq prop
(cond
((eq (get-text-property pos 'block-token) 'comment) 'block-token)
((eq (get-text-property pos 'tag-type) 'comment) 'tag-type)
((eq (get-text-property pos 'part-token) 'comment) 'part-token)
))
(if (and (not (bobp))
(eq (get-text-property pos prop) (get-text-property (1- pos) prop)))
(setq beg (or (previous-single-property-change pos prop)
(point-min))))
(if (and (not (eobp))
(eq (get-text-property pos prop) (get-text-property (1+ pos) prop)))
(setq end (or (next-single-property-change pos prop)
(point-max))))
(when (> (- end beg) 4)
(setq comment (buffer-substring-no-properties beg end))
(setq sub2 (substring comment 0 2))
(cond
((member sub2 '("$\\)" "" comment)))
((string= sub2 "{#")
(setq comment (replace-regexp-in-string "\\(^{#[ ]?\\|[ ]?#}$\\)" "" comment)))
((string= sub2 "/*")
(setq comment (replace-regexp-in-string "\\(^/\\*[ ]?\\|[ ]?\\*/$\\)" "" comment)))
((string= sub2 "//")
(setq comment (replace-regexp-in-string "\\(^//\\)" "" comment)))
)
(delete-region beg end)
(web-mode-insert-and-indent comment)
(goto-char beg)
) ;when
) ;t
) ;cond
(indent-according-to-mode)
;;(indent-for-tab-command)
))
(defun web-mode-comment-erb-block (pos)
"Turn an erb block into a comment block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
;; (message "%S %S" beg end)
(web-mode-insert-text-at-pos "#" (+ beg 2))
))
(defun web-mode-comment-django-block (pos)
"Turn a django block into a comment block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-insert-text-at-pos "#" end)
(web-mode-insert-text-at-pos "#" (1+ beg))
))
(defun web-mode-comment-dust-block (pos)
"Turn a dust block into a comment block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-insert-text-at-pos "!" end)
(web-mode-insert-text-at-pos "!" (1+ beg))
))
(defun web-mode-comment-aspx-block (pos)
"Turn a aspx block into a comment block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-insert-text-at-pos "#" end)
(web-mode-insert-text-at-pos "#" (1+ beg))
))
(defun web-mode-comment-jsp-block (pos)
"Turn a jsp block into a comment block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-insert-text-at-pos "--" (+ beg 2))
))
(defun web-mode-comment-go-block (pos)
"Turn a go block into a comment block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-insert-text-at-pos "/" (+ beg 2))
))
(defun web-mode-comment-php-block (pos)
"Turn a php block into a comment block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-insert-text-at-pos "*/" (- end 1))
(web-mode-insert-text-at-pos "/*" (+ beg (if (web-mode-looking-at "<\\?php" beg) 5 3)))
))
(defun web-mode-uncomment-erb-block (pos)
"Uncomment an erb block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-remove-text-at-pos 1 (+ beg 2))
))
(defun web-mode-uncomment-django-block (pos)
"Uncomment a django block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-remove-text-at-pos 2 (1- end))
(web-mode-remove-text-at-pos 2 beg)
))
(defun web-mode-uncomment-dust-block (pos)
"Uncomment a dust block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-remove-text-at-pos 1 (1- end))
(web-mode-remove-text-at-pos 1 (1+ beg))
))
(defun web-mode-uncomment-aspx-block (pos)
"Uncomment a aspx block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-remove-text-at-pos 1 (1- end))
(web-mode-remove-text-at-pos 1 (1+ beg))
))
(defun web-mode-uncomment-jsp-block (pos)
"Uncomment a jsp block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-remove-text-at-pos 2 (+ beg 2))
))
(defun web-mode-uncomment-go-block (pos)
"Uncomment a go block."
(let (beg end)
(setq beg (web-mode-block-beginning-position pos)
end (web-mode-block-end-position pos))
(web-mode-remove-text-at-pos 1 (+ beg 2))
))
(defun web-mode-snippet-names ()
"Return the snippet names."
(interactive)
(let (codes snippet)
(dolist (snippet web-mode-snippets)
(add-to-list 'codes (car snippet) t))
codes))
(defun web-mode-snippet-insert (code)
"Insert snippet."
(interactive
(list (completing-read
"Snippet: " (web-mode-snippet-names))))
(let (beg
(continue t)
(counter 0)
end
sel
snippet
(l (length web-mode-snippets))
pos)
(when mark-active
(setq sel (web-mode-trim
(buffer-substring-no-properties
(region-beginning) (region-end))))
(delete-region (region-beginning) (region-end)))
(while (and continue (< counter l))
(setq snippet (nth counter web-mode-snippets))
(when (string= (car snippet) code)
(setq continue nil))
(setq counter (1+ counter)))
(when snippet
(setq snippet (cdr snippet))
(setq beg (point-at-bol))
(insert (car snippet))
;; (message "insert[1] (%S)" (point))
(setq pos (point))
(when sel
(insert sel)
(setq pos (point)))
(if (cdr snippet) (insert (cdr snippet)))
;; (message "insert[2] (%S)" (point))
(setq end (point-at-eol))
(unless sel (goto-char pos))
(indent-region beg end))
;; (message "indent : beg(%S) end(%S)" beg end)
))
(defun web-mode-looking-at (regexp pos)
"Performs a looking-at at POS."
(save-excursion
(goto-char pos)
(looking-at regexp)
))
(defun web-mode-looking-at-p (regexp pos)
"Performs a looking-at at POS."
(save-excursion
(goto-char pos)
(looking-at-p regexp)
))
(defun web-mode-looking-back (regexp pos)
"Performs a looking-at at POS."
(save-excursion
(goto-char pos)
(looking-back regexp)
))
(defun web-mode-insert-text-at-pos (text pos)
"Insert TEXT at POS."
(let ((mem web-mode-enable-auto-pairing))
(setq web-mode-enable-auto-pairing nil)
(save-excursion
(goto-char pos)
(insert text)
(setq web-mode-enable-auto-pairing mem)
)))
(defun web-mode-remove-text-at-pos (n &optional pos)
"Remove N chars at POS."
(unless pos (setq pos (point)))
(delete-region pos (+ pos n)))
(defun web-mode-insert-and-indent (text)
"Insert and indent text."
(interactive)
(let (beg end)
(setq beg (point-at-bol))
(insert text)
(setq end (point-at-eol))
(indent-region beg end)))
(defun web-mode-navigate (&optional pos)
"Match tag."
(interactive)
(unless pos (setq pos (point)))
(let (init)
(goto-char pos)
(setq init (point))
(when (> (current-indentation) (current-column))
(back-to-indentation))
(setq pos (point))
(cond
((and (get-text-property pos 'block-side)
(web-mode-block-beginning)
(web-mode-block-controls-get (point)))
(web-mode-block-match))
((member (get-text-property pos 'tag-type) '(start end))
(web-mode-tag-beginning)
(web-mode-tag-match))
(t
(goto-char init))
)
))
(defun web-mode-block-match (&optional pos)
"Block match"
(unless pos (setq pos (point)))
(let (pos-ori controls control (counter 1) type (continue t) pair)
(setq pos-ori pos)
(goto-char pos)
(setq controls (web-mode-block-controls-get pos))
;; (message "controls=%S" controls)
(cond
(controls
(setq pair (car controls))
(setq control (cdr pair))
(setq type (car pair))
(when (eq type 'inside) (setq type 'close))
(while continue
(cond
((and (> pos-ori 1) (bobp))
(setq continue nil))
((or (and (eq type 'open) (not (web-mode-block-next)))
(and (eq type 'close) (not (web-mode-block-previous))))
;; (message "ici%S" (point))
(setq continue nil)
)
((null (setq controls (web-mode-block-controls-get (point))))
)
(t
;; TODO : est il nécessaire de faire un reverse sur controls si on doit matcher backward
(dolist (pair controls)
(cond
((not (string= (cdr pair) control))
)
((eq (car pair) 'inside)
)
((eq (car pair) type)
(setq counter (1+ counter)))
(t
(setq counter (1- counter)))
)
) ;dolist
(when (= counter 0)
(setq continue nil))
) ;t
) ;cond
) ;while
(if (= counter 0) (point) nil)
) ;controls
(t
(goto-char pos-ori)
nil
) ;controls = nul
) ;conf
))
(defun web-mode-tag-match (&optional pos)
"Fetch HTML block."
(unless pos (setq pos (point)))
(let (regexp)
(setq regexp (concat "?" (get-text-property pos 'tag-name)))
(when (member (get-text-property pos 'tag-type) '(start end))
(web-mode-tag-beginning)
(setq pos (point)))
(if (eq (get-text-property pos 'tag-type) 'end)
(web-mode-tag-fetch-opening regexp pos)
(web-mode-tag-fetch-closing regexp pos))
t))
(defun web-mode-tag-fetch-opening (regexp pos)
"Fetch opening HTML block."
(let ((counter 1) (n 0))
(goto-char pos)
(while (and (> counter 0) (re-search-backward regexp nil t))
(when (get-text-property (point) 'tag-beg)
(setq n (1+ n))
(if (eq (get-text-property (point) 'tag-type) 'end)
(setq counter (1+ counter))
(setq counter (1- counter))))
)
(if (= n 0) (goto-char pos))
))
(defun web-mode-tag-fetch-closing (regexp pos)
"Fetch closing HTML closing block."
(let ((counter 1) (n 0))
(goto-char pos)
(web-mode-tag-end)
(while (and (> counter 0) (re-search-forward regexp nil t))
(when (get-text-property (match-beginning 0) 'tag-beg)
(setq n (1+ n))
(if (eq (get-text-property (point) 'tag-type) 'end)
(setq counter (1- counter))
(setq counter (1+ counter))))
)
(if (> n 0)
(web-mode-tag-beginning)
(goto-char pos))
))
(defun web-mode-element-tag-name (&optional pos)
(unless pos (setq pos (point)))
(save-excursion
(goto-char pos)
(if (and (web-mode-tag-beginning)
(looking-at "<\\(/?[[:alpha:]][[:alnum:]-]*\\)"))
(match-string-no-properties 1)
nil)))
(defun web-mode-element-close ()
"Close HTML element."
(interactive)
(let (jump epp ins tag)
(setq epp (web-mode-element-parent-position))
(when epp
;; (setq tag (get-text-property epp 'tag-name))
(setq tag (web-mode-element-tag-name epp))
(cond
((null tag)
(setq epp nil))
((looking-back "")
(setq ins tag))
((looking-back "<")
(setq ins (concat "/" tag)))
(t
;; auto-close-style = 2
;; (message "%S %c" (point) (char-after))
(when (and (looking-at-p "[[:alpha:]]") (> (length tag) 4))
(dolist (elt '("div" "span" "strong" "pre" "li"))
(when (and (string-match-p (concat "^" elt) tag) (not (string= tag elt)))
(setq tag elt)
(put-text-property epp (point) 'tag-name tag))
)
) ;when
(if (web-mode-element-is-void (get-text-property (point) 'tag-name))
(setq ins nil
epp nil)
(setq ins (concat "" tag)))
)
) ;cond
(when ins
;; (message "ins=%S" ins)
(unless (looking-at-p "[ ]*>")
(setq ins (concat ins ">")))
(insert ins)
(save-excursion
(search-backward "<")
(setq jump (and (eq (char-before) ?\>)
(string= (get-text-property (1- (point)) 'tag-name) tag)))
(if jump (setq jump (point)))
;; (setq jump (looking-back (concat "<" tag ">")))
) ;save-excursion
(if jump (goto-char jump))
) ;when not ins
) ;when epp
epp
))
(defun web-mode-detect-content-type ()
"Update content type"
(cond
((and (string= web-mode-engine "none")
(< (point) 16)
(eq (char-after 1) ?\#)
(string-match-p "php" (buffer-substring-no-properties
(line-beginning-position)
(line-end-position))))
(web-mode-set-engine "php")
)
((and (string= web-mode-content-type "javascript")
(< (point) web-mode-chunk-length)
(eq (char-after (point-min)) ?\/)
(string-match-p "@jsx" (buffer-substring-no-properties
(line-beginning-position)
(line-end-position))))
(web-mode-set-content-type "jsx")
)
) ;cond
)
(defun web-mode-detect-engine ()
"Engine detection"
(save-excursion
(let (engine)
;; (message "detect-engine: %S" (point))
(goto-char (point-min))
(when (and (re-search-forward "-\\*- engine:[ ]*\\([[:alnum:]-]+\\)[ ]*-\\*-" web-mode-chunk-length t)
(or t
(eq (get-text-property (point) 'part-token) 'comment)
(eq (get-text-property (point) 'block-token) 'comment)
(eq (get-text-property (point) 'tag-type) 'comment)))
(setq engine (web-mode-engine-canonical-name (match-string-no-properties 1))
web-mode-engine engine)
)
engine)
))
(defun web-mode-complete ()
"Autocomple at point."
(interactive)
(let ((beg (1- (point)))
(end (point))
(chunk (buffer-substring-no-properties (- (point) 2) (point)))
(auto-closed nil)
(auto-paired nil))
;;-- auto-closing
(when (and (> web-mode-tag-auto-close-style 0)
(or (and (= web-mode-tag-auto-close-style 2)
(not (get-text-property end 'part-side))
(string-match-p "[[:alnum:]'\"]>" chunk))
(string= "" chunk))
(not (get-text-property (- beg 1) 'block-side)))
(when (web-mode-element-close)
;; (message "auto-close %S" web-mode-change-end)
(setq auto-closed t))
)
;;-- auto-pairing
(when (and web-mode-enable-auto-pairing
(not auto-closed)
(not (get-text-property end 'part-side)))
(let ((i 0) expr p after pos-end (l (length web-mode-auto-pairs)))
(setq pos-end (if (> (+ end 32) (line-end-position))
(line-end-position)
(+ end 10)))
(setq chunk (buffer-substring-no-properties (- beg 2) end)
after (buffer-substring-no-properties end pos-end))
(while (and (< i l) (not auto-paired))
(setq expr (elt web-mode-auto-pairs i))
(setq i (1+ i))
(when (and (string= (elt expr 0) chunk)
(not (string-match-p (or (elt expr 2) (elt expr 1)) after)))
(setq auto-paired t)
(insert (elt expr 1))
(if (not (elt expr 2))
(goto-char end)
(setq p (point))
(insert (elt expr 2))
(goto-char p))
) ;when
) ;while
) ;let
) ; end auto-pairing
(cond
((or auto-closed auto-paired)
(when (and web-mode-change-end (>= (line-end-position) web-mode-change-end))
(setq web-mode-change-end (line-end-position))
;; (message "new change-end=%S" web-mode-change-end)
)
(list :auto-closed auto-closed :auto-paired auto-paired))
(t
nil)
)
))
(defun web-mode-on-after-change (beg end len)
"Handles auto-pairing, auto-closing, and region-refresh after buffer alteration."
;;(message "after-change: pos=%d, beg=%d, end=%d, len=%d, cur=%d, cmd=%S" (point) beg end len (current-column) this-original-command)
;; (backtrace)
(when (or (null web-mode-change-beg) (< beg web-mode-change-beg))
(setq web-mode-change-beg beg))
(when (or (null web-mode-change-end) (> end web-mode-change-end))
(setq web-mode-change-end end))
)
(defun web-mode-on-post-command ()
(let (ctx)
(when (< (point) 16)
(web-mode-detect-content-type))
(when (and web-mode-enable-engine-detection
(or (null web-mode-engine) (string= web-mode-engine "none"))
(< (point) web-mode-chunk-length)
(web-mode-detect-engine))
(web-mode-on-engine-setted)
(web-mode-buffer-highlight))
(cond
((<= (point) 3)
)
((and (member this-command '(self-insert-command))
(not (get-text-property (point) 'part-side)))
(setq ctx (web-mode-complete))
)
((and web-mode-enable-auto-opening
(member this-command '(newline))
;; (not (web-mode-buffer-narrowed-p))
(or (and (eq (char-before) ?\n) ;;(string= ">\n" chunk)
(eq (char-before (1- (point))) ?\>)
(not (eobp))
(eq (get-text-property (- (point) 2) 'tag-type) 'start)
(eq (get-text-property (point) 'tag-type) 'end)
(string= (get-text-property (- (point) 2) 'tag-name)
(get-text-property (point) 'tag-name)))
(and (get-text-property (1- (point)) 'block-side)
(string= web-mode-engine "php")
(looking-back "<\\?php[ ]*\n")
(looking-at-p "[ ]*\\?>"))))
(newline-and-indent)
(forward-line -1)
(indent-according-to-mode)
;; (indent-according-to-mode)
;; (indent-for-tab-command)
)
;;-- auto-indentation
) ;cond
(when (and web-mode-enable-auto-indentation
(member this-command '(self-insert-command))
;;(not auto-opened)
(or (and ctx (plist-get ctx :auto-closed))
(and (> (point) (point-min))
(get-text-property (1- (point)) 'tag-end)
(get-text-property (line-beginning-position) 'tag-beg))))
(indent-according-to-mode)
;; (indent-for-tab-command)
(when (and web-mode-change-end (> web-mode-change-end (point-max)))
(message "post-command: enlarge web-mode-change-end")
(setq web-mode-change-end (point-max))
)
) ; when auto-indent
(when web-mode-enable-current-element-highlight
(web-mode-highlight-current-element))
(when (and web-mode-enable-current-column-highlight
(not (web-mode-buffer-narrowed-p)))
(web-mode-column-hide)
(save-excursion
(let (pos column line-to line-from)
(back-to-indentation)
(setq pos (point)
column (current-column)
line-to (web-mode-line-number))
(when (and (get-text-property (point) 'tag-beg)
(member (get-text-property (point) 'tag-type) '(start end))
(web-mode-tag-match)
(setq line-from (web-mode-line-number))
;; (progn (message "cool %S %S" line-from line-to) t)
(not (= line-from line-to)))
(web-mode-column-show column line-from line-to)
) ;if
) ;let
) ;save-excursion
)
;; (message "post-command (%S) (%S)" web-mode-change-end web-mode-change-end)
))
(defun web-mode-dom-apostrophes-replace ()
"Replace ' with ’."
(interactive)
(save-excursion
(let ((min (point-min)) (max (point-max)))
(when mark-active
(setq min (region-beginning)
max (region-end))
(deactivate-mark))
(goto-char min)
(while (web-mode-rsf-content "\\([[:alpha:]]\\)'\\([[:alpha:]]\\)" max)
(replace-match "\\1’\\2")
) ;while
)))
(defun web-mode-dom-entities-encode ()
"Replace special chars with HTML entities (e.g. é becomes é)"
(save-excursion
(let (regexp ms pair elt (min (point-min)) (max (point-max)))
(when mark-active
(setq min (region-beginning)
max (region-end))
(deactivate-mark))
(goto-char min)
(setq regexp "[")
(dolist (pair web-mode-html-entities)
(setq regexp (concat regexp (char-to-string (cdr pair))))
)
(setq regexp (concat regexp "]"))
(while (web-mode-rsf-content regexp max)
(setq elt (match-string-no-properties 0))
(setq elt (aref elt 0))
(setq elt (car (rassoc elt web-mode-html-entities)))
;; (message "%S" elt)
(replace-match (concat "&" elt ";"))
) ;while
)))
;; ½ ½ ½ ½
(defun web-mode-dom-entities-replace ()
"Replace HTML entities e.g. entities é é é become é"
(interactive)
(save-excursion
(let (ms pair elt (min (point-min)) (max (point-max)))
(when mark-active
(setq min (region-beginning)
max (region-end))
(deactivate-mark))
(goto-char min)
(while (web-mode-rsf-content "&\\([#]?[[:alnum:]]\\{2,8\\}\\);" max)
(setq elt nil)
;; (message "E=%S" (match-string 1))
(setq ms (match-string-no-properties 1))
(if (eq (aref ms 0) ?\#)
(if (eq (aref ms 1) ?x)
(progn
(setq elt (substring ms 2))
(setq elt (downcase elt))
(setq elt (string-to-number elt 16))
(setq elt (char-to-string elt))
)
(setq elt (substring ms 1))
(setq elt (char-to-string (string-to-number elt)))
)
(setq pair (assoc ms web-mode-html-entities))
;; (message "pos=%S name=%S pair=%S" (point) name pair)
(if pair (setq elt (cdr pair)))
(if elt (setq elt (char-to-string elt)))
) ;if
(if elt (replace-match elt))
) ;while
)))
(defun web-mode-dom-xml-replace ()
"Replace &, > and < in HTML content."
(interactive)
(save-excursion
(let (expr (min (point-min)) (max (point-max)))
(when mark-active
(setq min (region-beginning)
max (region-end))
(deactivate-mark))
(goto-char min)
(while (web-mode-rsf-content "[&<>]" max)
(replace-match (cdr (assq (char-before) web-mode-xml-chars)) t t))
)))
(defun web-mode-dom-quotes-replace ()
"Replace dumb quotes."
(interactive)
(save-excursion
(let (expr (min (point-min)) (max (point-max)))
(when mark-active
(setq min (region-beginning)
max (region-end))
(deactivate-mark))
(goto-char min)
(setq expr (concat (car web-mode-smart-quotes) "\\2" (cdr web-mode-smart-quotes)))
(while (web-mode-rsf-content "\\(\"\\)\\(.\\{1,200\\}\\)\\(\"\\)" max)
(replace-match expr)
) ;while
)))
(defun web-mode-dom-xpath (&optional pos)
"HTML path"
(interactive)
(unless pos (setq pos (point)))
(save-excursion
(goto-char pos)
(let (path)
(while (web-mode-element-parent)
(setq path (cons (get-text-property (point) 'tag-name) path))
)
(message "/%s" (mapconcat 'identity path "/"))
)))
(defun web-mode-block-ends-with (regexp &optional pos)
"Check if current block ends with regexp"
(unless pos (setq pos (point)))
(save-excursion
(goto-char pos)
(save-match-data
(if (stringp regexp)
(and (web-mode-block-end)
(progn (backward-char) t)
(web-mode-block-skip-blank-backward)
(progn (forward-char) t)
(looking-back regexp))
(let ((pair regexp)
(block-beg (web-mode-block-beginning-position pos))
(block-end (web-mode-block-end-position pos)))
(and (web-mode-block-end)
(web-mode-block-sb (car pair) block-beg)
(not (web-mode-sf (cdr pair) block-end)))
) ;let
) ;if
)))
(defun web-mode-block-token-starts-with (regexp &optional pos)
"Check if current token starts with regexp"
(unless pos (setq pos (point)))
(save-excursion
(and (goto-char pos)
(web-mode-block-token-beginning)
(skip-chars-forward "[\"']")
(looking-at regexp))
))
(defun web-mode-block-starts-with (regexp &optional pos)
"Check if current block starts with regexp"
(unless pos (setq pos (point)))
(save-excursion
(and (web-mode-block-beginning)
(web-mode-block-skip-blank-forward)
(looking-at regexp))
))
(defun web-mode-block-skip-blank-backward (&optional pos)
"skip backward"
(unless pos (setq pos (point)))
(let ((continue t))
(goto-char pos)
(while continue
(if (and (get-text-property (point) 'block-side)
(not (bobp))
(or (member (char-after) '(?\s ?\n))
(member (get-text-property (point) 'block-token) '(delimiter comment))))
(backward-char)
(setq continue nil))
) ;while
(point)))
(defun web-mode-block-skip-blank-forward (&optional pos)
"skip forward"
(unless pos (setq pos (point)))
(let ((continue t))
(goto-char pos)
(while continue
(if (and (get-text-property (point) 'block-side)
(or (member (char-after) '(?\s ?\n ?\t))
(member (get-text-property (point) 'block-token) '(delimiter comment))))
(forward-char)
(setq continue nil))
) ;while
;; (message "pt=%S" (point))
(point)
))
(defun web-mode-tag-attributes-sort (&optional pos)
"Sort attributes"
(interactive)
(unless pos (setq pos (point)))
(save-excursion
(let (attrs (continue t) min max tag-beg tag-end attr attr-name attr-beg attr-end indent indentation sorter ins)
(if (not (member (get-text-property pos 'tag-type) '(start void)))
nil
(setq tag-beg (web-mode-tag-beginning-position pos)
tag-end (web-mode-tag-end-position))
;; (message "%S %S" tag-beg tag-end)
(goto-char tag-beg)
(while continue
(if (or (not (web-mode-attribute-next))
(>= (point) tag-end))
(setq continue nil)
;;(message "attr=%S" (point))
(setq attr-beg (web-mode-attribute-beginning-position)
attr-end (1+ (web-mode-attribute-end-position)))
(when (null min)
(setq min attr-beg))
(setq max attr-end)
(goto-char attr-beg)
(setq attr (buffer-substring-no-properties attr-beg attr-end))
(if (string-match "^\\([[:alnum:]-]+\\)=" attr)
(setq attr-name (match-string-no-properties 1 attr))
(setq attr-name attr))
(setq indent (looking-back "^[ \t]*"))
(setq attrs (append attrs (list (list attr-beg attr-end attr-name attr indent))))
) ;if
) ;while
) ;if in tag
(when attrs
(setq sorter (function
(lambda (elt1 elt2)
(string< (nth 2 elt1) (nth 2 elt2))
)))
(setq attrs (sort attrs sorter))
(delete-region (1- min) max)
(setq ins "")
(dolist (elt attrs)
(if (and (nth 4 elt) (> (length ins) 1))
(setq ins (concat ins "\n"))
(setq ins (concat ins " ")))
(setq ins (concat ins (nth 3 elt)))
)
(goto-char (1- min))
(insert ins)
(web-mode-tag-beginning)
(setq min (line-beginning-position))
(web-mode-tag-end)
(setq max (line-end-position))
(indent-region min max)
)
;;(message "attrs=%S" attrs)
)))
(defun web-mode-attribute-transpose (&optional pos)
"Transpose attribute"
(interactive)
(unless pos (setq pos (point)))
(let (ret attr-beg attr-end next-beg next-end tag-end)
(when (and (get-text-property pos 'tag-attr)
(setq next-beg (web-mode-attribute-next-position pos))
(setq next-end (web-mode-attribute-end-position next-beg))
(setq tag-end (web-mode-tag-end-position pos))
(> tag-end next-end))
(setq attr-beg (web-mode-attribute-beginning-position pos)
attr-end (web-mode-attribute-end-position pos))
;; (message "%S %S - %S %S" attr-beg attr-end next-beg next-end)
(transpose-regions attr-beg (1+ attr-end) next-beg (1+ next-end))
)))
(defun web-mode-attribute-select (&optional pos)
"Select attribute."
(interactive)
(unless pos (setq pos (point)))
(if (null (get-text-property pos 'tag-attr))
nil
(goto-char pos)
(web-mode-attribute-beginning)
(set-mark (point))
(web-mode-attribute-end)
(point)
))
(defun web-mode-block-close (&optional pos)
"Close the first opened control block."
(interactive)
(unless pos (setq pos (point)))
(let ((continue t) ctx h ctrl n closing-block)
(save-excursion
(setq h (make-hash-table :test 'equal))
(while (and continue (web-mode-block-previous))
;; (message "ici")
(when (setq ctx (web-mode-block-is-control (point)))
(setq ctrl (car ctx))
(setq n (gethash ctrl h 0))
(if (cdr ctx)
(puthash ctrl (1+ n) h)
(puthash ctrl (1- n) h)
)
(when (> (gethash ctrl h) 0)
(setq continue nil))
;; (if ctx (message "(%S) %S : %S" (point) ctrl (gethash ctrl h)))
)
) ;while
) ;save-excursion
(when (and (null continue)
(setq closing-block (web-mode-closing-block ctrl)))
(insert closing-block)
(indent-according-to-mode)
;; (indent-for-tab-command)
)
))
(defun web-mode-closing-block (type)
"Return the closing block corresponding to TYPE"
(cond
((string= web-mode-engine "django")
(concat "{% end" type " %}"))
((string= web-mode-engine "ctemplate")
(concat "{{/" type "}}"))
((string= web-mode-engine "blade")
(concat "@end" type))
((string= web-mode-engine "dust")
(concat "{/" type "}"))
((string= web-mode-engine "underscore")
"<% } %>")
((string= web-mode-engine "erb")
"<% end %>")
(t
(cdr (assoc type web-mode-closing-blocks)))
) ;cond
)
;;---- POSITION ----------------------------------------------------------------
(defun web-mode-opening-paren-position (&optional pos limit)
"Opening paren POS."
(save-restriction
(unless pos (setq pos (point)))
(unless limit (setq limit nil))
(goto-char pos)
(let* ((n -1)
(block-side (and (get-text-property pos 'block-side)
(not (string= web-mode-engine "razor"))))
(paren (char-after))
(pairs '((?\) . "[)(]")
(?\] . "[\]\[]")
(?\} . "[}{]")
(?\> . "[><]")))
(regexp (cdr (assoc paren pairs)))
(continue (not (null regexp)))
(counter 0))
(while (and continue (re-search-backward regexp limit t))
(cond
((> (setq counter (1+ counter)) 500)
(message "opening-paren-position ** crazy loop **")
(setq continue nil))
((or (web-mode-is-comment-or-string)
(and block-side (not (get-text-property (point) 'block-side))))
)
((eq (char-after) paren)
(setq n (1- n)))
(t
(setq n (1+ n))
(setq continue (not (= n 0))))
)
)
(if (= n 0) (point) nil)
)))
(defun web-mode-closing-paren-position (&optional pos limit)
(save-excursion
(unless pos (setq pos (point)))
(unless limit (setq limit nil))
(goto-char pos)
(let* ((n 0)
(block-side (and (get-text-property pos 'block-side)
(not (string= web-mode-engine "razor"))))
(paren (char-after))
(pairs '((?\( . "[)(]")
(?\[ . "[\]\[]")
(?\{ . "[}{]")
(?\< . "[><]")))
(regexp (cdr (assoc paren pairs)))
(continue (not (null regexp))))
(while (and continue (re-search-forward regexp limit t))
(cond
((or (web-mode-is-comment-or-string (1- (point)))
(and block-side (not (get-text-property (point) 'block-side))))
;;(message "pt=%S" (point))
)
((eq (char-before) paren)
(setq n (1+ n)))
(t
(setq n (1- n))
(setq continue (not (= n 0)))
)
) ;cond
) ;while
(if (= n 0) (1- (point)) nil)
)))
(defun web-mode-closing-delimiter-position (delimiter &optional pos limit)
"Fetch closing delimiter."
(save-excursion
(unless pos (setq pos (point)))
(unless limit (setq limit nil))
(goto-char pos)
(setq pos nil)
(let ((continue t))
(while (and continue (re-search-forward delimiter limit t))
(setq continue nil
pos (1- (point)))
) ;while
pos)))
(defun web-mode-previous-tag-at-bol-pos (pos)
"Line beginning with an HTML tag. BOL is returned or nil."
(save-excursion
(goto-char pos)
(setq pos nil)
(let ((continue t))
(back-to-indentation)
(if (and (get-text-property (point) 'tag-beg)
;; (not (member web-mode-content-type '("jsx")))
(not (get-text-property (point) 'part-side))
(not (get-text-property (point) 'block-side))
)
(setq pos (line-beginning-position))
(while continue
(forward-line -1)
(setq pos (point))
(when (bobp)
(setq continue nil))
(back-to-indentation)
(if (and (get-text-property (point) 'tag-beg)
(not (get-text-property (point) 'part-side))
(not (get-text-property (point) 'block-side))
)
(setq continue nil)
(setq pos nil))
) ;while
) ;if
;; (message "pos=%S" pos)
pos)))
(defun web-mode-next-tag-at-eol-pos (pos)
"Line ending with an HTML tag. EOL is returned or nil."
(save-excursion
(goto-char pos)
(let ((continue t))
(while continue
(end-of-line)
(setq pos (point))
(when (eobp)
(setq continue nil))
(skip-chars-backward " ")
(if (and (> (point) (point-min))
(get-text-property (1- (point)) 'tag-end)
(not (and (member (get-text-property (1- (point)) 'tag-name) '("script" "style"))
(eq (get-text-property (1- (point)) 'tag-type) 'start)))
(not (get-text-property (1- (point)) 'part-side))
(not (get-text-property (1- (point)) 'block-side)))
(progn
;; (message "point(%S) %S %S %S"
;; (point)
;; (get-text-property (1- (point)) 'tag-end)
;; (get-text-property (1- (point)) 'part-side)
;; (get-text-property (1- (point)) 'block-side))
(setq continue nil))
(setq pos nil))
(if continue (forward-line))
) ;while
pos)))
(defun web-mode-tag-match-position (&optional pos)
"Html tag match position."
(unless pos (setq pos (point)))
(save-excursion
(web-mode-tag-match pos)
(if (= pos (point)) nil (point))))
(defun web-mode-tag-beginning-position (&optional pos)
"Beginning position of the current tag. POINT is at <."
(unless pos (setq pos (point)))
(let (beg)
(cond
((get-text-property pos 'tag-beg)
(setq beg pos))
((and (> pos 1) (get-text-property (1- pos) 'tag-beg))
(setq beg (1- pos)))
((get-text-property pos 'tag-type)
(setq beg (1- (previous-single-property-change pos 'tag-beg)))
(when (not (get-text-property beg 'tag-beg))
(setq beg nil)))
(t
(setq beg nil))
) ;cond
beg))
(defun web-mode-tag-end-position (&optional pos)
"End position of the current tag. POINT is at >."
(unless pos (setq pos (point)))
(let (end)
(cond
((null pos)
(setq end nil))
((get-text-property pos 'tag-end)
(setq end pos))
((get-text-property pos 'tag-type)
(setq end (next-single-property-change pos 'tag-end))
(when (not (get-text-property end 'tag-end))
(setq end nil)))
(t
(setq end nil))
) ;cond
end))
(defun web-mode-tag-next-position (&optional pos limit)
(unless pos (setq pos (point)))
(unless limit (setq limit (point-max)))
(save-excursion
(goto-char pos)
(cond
((eobp)
(setq pos nil))
(t
(when (get-text-property pos 'tag-beg)
(setq pos (1+ pos)))
(setq pos (next-single-property-change pos 'tag-beg))
(when (and pos (> pos limit)) (setq pos nil))
)
)
pos))
(defun web-mode-tag-previous-position (&optional pos limit)
(unless pos (setq pos (point)))
(unless limit (setq limit (point-min)))
(save-excursion
(goto-char pos)
(cond
((bobp)
(setq pos nil))
(t
(when (get-text-property pos 'tag-beg)
(setq pos (1- pos)))
(setq pos (previous-single-property-change pos 'tag-beg))
(when pos
(setq pos (1- pos))
(goto-char pos))
)
)
pos))
(defun web-mode-attribute-beginning-position (&optional pos)
"Beginning position of the current attr."
(unless pos (setq pos (point)))
(cond
((null (get-text-property pos 'tag-attr))
nil)
((null (get-text-property (1- pos) 'tag-attr))
pos)
(t
(previous-single-property-change pos 'tag-attr))
))
(defun web-mode-attribute-end-position (&optional pos)
"End position of the current attr."
(unless pos (setq pos (point)))
(let (end)
(cond
((null pos)
(setq end nil))
((get-text-property pos 'tag-attr-end)
(setq end pos))
((get-text-property pos 'tag-attr)
(setq end (next-single-property-change pos 'tag-attr-end))
(when (not (get-text-property end 'tag-attr-end))
(setq end nil)))
(t
(setq end nil))
) ;cond
end))
(defun web-mode-attribute-next-position (&optional pos)
"Next attribute position."
(unless pos (setq pos (point)))
(let ((continue t))
(while continue
(setq pos (next-single-property-change pos 'tag-attr))
(cond
((null pos)
(setq continue nil
pos nil))
((get-text-property pos 'tag-attr)
(setq continue nil))
)
) ;while
pos))
(defun web-mode-element-beginning-position (&optional pos)
"Beginning of element pos."
(unless pos (setq pos (point)))
(cond
((null (get-text-property pos 'tag-type))
(setq pos (web-mode-element-parent-position)))
((eq (get-text-property pos 'tag-type) 'end)
(setq pos (web-mode-tag-match-position pos))
(setq pos (if (get-text-property pos 'tag-beg) pos nil)))
((member (get-text-property pos 'tag-type) '(start void))
(setq pos (web-mode-tag-beginning-position pos)))
(t
(setq pos nil))
) ;cond
pos)
(defun web-mode-element-end-position (&optional pos)
"End of element pos."
(unless pos (setq pos (point)))
(cond
((null (get-text-property pos 'tag-type))
(setq pos (web-mode-element-parent-position pos))
(when pos
(setq pos (web-mode-tag-match-position pos))
(when pos (setq pos (web-mode-tag-end-position pos)))
)
)
((member (get-text-property pos 'tag-type) '(end void))
(setq pos (web-mode-tag-end-position pos))
)
((member (get-text-property pos 'tag-type) '(start))
(setq pos (web-mode-tag-match-position pos))
(when pos (setq pos (web-mode-tag-end-position pos))))
(t
(setq pos nil))
) ;cond
pos)
(defun web-mode-element-child-position (&optional pos)
"Child element pos."
(save-excursion
(let (child close)
(unless pos (setq pos (point)))
(goto-char pos)
(cond
((eq (get-text-property pos 'tag-type) 'start)
(web-mode-tag-match)
(setq close (point))
(goto-char pos)
)
((eq (get-text-property pos 'tag-type) 'void)
)
((eq (get-text-property pos 'tag-type) 'end)
(web-mode-tag-beginning)
(setq close (point))
(web-mode-tag-match)
)
((web-mode-element-parent-position pos)
(setq pos (point))
(web-mode-tag-match)
(setq close (point))
(goto-char pos)
)
) ;cond
(when (and close
(web-mode-element-next)
(< (point) close))
(setq child (point))
)
child
)))
(defun web-mode-element-parent-position (&optional pos)
"Parent element pos."
(let (n
tag-type
tag-name
(continue t)
(h (make-hash-table :test 'equal)))
(save-excursion
(if pos (goto-char pos))
(while (and continue (web-mode-tag-previous))
(setq pos (point))
(setq tag-type (get-text-property pos 'tag-type)
tag-name (get-text-property pos 'tag-name))
(setq n (gethash tag-name h 0))
(when (member tag-type '(end start))
(if (eq tag-type 'end)
(puthash tag-name (1- n) h)
(puthash tag-name (1+ n) h)
(when (= n 0) (setq continue nil))
) ;if
) ;when
) ;while
) ;save-excursion
(if (null continue) pos nil)
))
(defun web-mode-element-next-position (&optional pos limit)
(unless pos (setq pos (point)))
(unless limit (setq limit (point-max)))
(save-excursion
(goto-char pos)
(let ((continue (not (eobp)))
(props '(start void comment)))
(while continue
(setq pos (web-mode-tag-next))
(cond
((or (null pos) (> (point) limit))
(setq continue nil
pos nil))
((member (get-text-property (point) 'tag-type) props)
(setq continue nil))
)
) ;while
;; (message "pos=%S" pos)
pos)))
(defun web-mode-part-end-position (&optional pos)
"End position of the current part."
(unless pos (setq pos (point)))
(cond
((member web-mode-content-type '("css" "javascript" "json" "jsx"))
(setq pos (point-max)))
((not (get-text-property pos 'part-side))
(setq pos nil))
((= pos (point-max))
(setq pos nil))
((not (get-text-property (1+ pos) 'part-side))
pos)
(t
(setq pos (next-single-property-change pos 'part-side)))
) ;cond
pos)
(defun web-mode-part-beginning-position (&optional pos)
"Beginning of part"
(unless pos (setq pos (point)))
(cond
((member web-mode-content-type '("css" "javascript" "json" "jsx"))
(setq pos (point-min)))
((not (get-text-property pos 'part-side))
(setq pos nil))
((= pos (point-min))
(setq pos nil))
((not (get-text-property (1- pos) 'part-side))
pos)
(t
(setq pos (previous-single-property-change pos 'part-side)))
) ;cond
pos)
(defun web-mode-part-next-position (&optional pos)
"web-mode-part-next-position"
(unless pos (setq pos (point)))
(if (get-text-property pos 'part-side)
(if (= pos (point-min))
(set pos (point-min))
(setq pos (web-mode-part-end-position pos))
(when (and pos (> (point-max) pos))
(setq pos (1+ pos))
(if (not (get-text-property pos 'part-side))
(setq pos (next-single-property-change pos 'part-side)))
) ;when
)
(setq pos (next-single-property-change pos 'part-side))
)
pos)
(defun web-mode-block-match-position (&optional pos)
"Match block position."
(unless pos (setq pos (point)))
(save-excursion
(web-mode-block-match pos)
(if (= pos (point)) nil (point))))
(defun web-mode-block-control-previous-position (type &optional pos)
"web-mode-block-control-previous-open"
(unless pos (setq pos (point)))
(let ((continue t) controls)
(while continue
(setq pos (web-mode-block-previous-position pos))
(cond
((null pos)
(setq continue nil
pos nil))
((and (setq controls (web-mode-block-controls-get pos))
(eq (car (car controls)) type))
(setq continue nil))
) ;cond
) ;while
pos))
(defun web-mode-block-opening-paren-position (pos limit)
"Is opened code line."
(save-excursion
(goto-char pos)
(let (c
n
pt
(continue t)
(pairs '((")" . "(")
("]" . "[")
("}" . "{")))
(h (make-hash-table :test 'equal))
(regexp "[\]\[)(}{]"))
(while (and continue (re-search-backward regexp limit t))
(unless (web-mode-is-comment-or-string)
(setq c (string (char-after)))
(cond
((member c '("(" "{" "["))
(setq n (gethash c h 0))
(if (= n 0)
(setq continue nil
pt (point))
(puthash c (1+ n) h)
))
(t
(setq c (cdr (assoc c pairs)))
(setq n (gethash c h 0))
(puthash c (1- n) h))
) ;cond
) ;unless
) ;while
;; (message "h=%S pt=%S" h pt)
pt
)))
(defun web-mode-block-code-beginning-position (&optional pos)
"Block beginning position, just after the start delimiter (e.g. just after pos (point-min))
(get-text-property (1- pos) 'block-beg))
(setq pos (1- pos))
)
((get-text-property pos 'block-side)
(setq pos (previous-single-property-change pos 'block-beg))
(setq pos (if (and pos (> pos (point-min))) (1- pos) (point-min)))
)
(t
(setq pos nil))
) ;cond
;; (message "web-mode-block-beginning-position=%S" pos)
pos)
(defun web-mode-block-token-beginning-position (&optional pos)
(unless pos (setq pos (point)))
(cond
((and (get-text-property pos 'block-token)
(= pos (point-min)))
)
((and (> pos (point-min))
(get-text-property pos 'block-token)
(not (get-text-property (1- pos) 'block-token)))
;; (setq pos (1- pos))
)
((get-text-property pos 'block-token)
(setq pos (previous-single-property-change pos 'block-token))
(setq pos (if (and pos (> pos (point-min))) pos (point-min)))
)
(t
(setq pos nil))
) ;cond
;;(message "web-mode-block-token-beginning-position=%S" pos)
pos)
(defun web-mode-block-code-end-position (&optional pos)
(unless pos (setq pos (point)))
(setq pos (web-mode-block-end-position pos))
(cond
((and pos
(eq (get-text-property pos 'block-token) 'delimiter)
(eq (get-text-property (1- pos) 'block-token) 'delimiter))
;;TODO : danger string enchaine avec delimiter
(setq pos (previous-single-property-change pos 'block-token)))
((= pos (1- (point-max))) ;; TODO: comparer plutot avec line-end-position
(setq pos (point-max)))
)
pos)
(defun web-mode-block-end-position (&optional pos)
"web-mode-block-end-position"
(unless pos (setq pos (point)))
(cond
((get-text-property pos 'block-end)
pos)
((get-text-property pos 'block-side)
;; (message "pos=%S %S" pos (or (next-single-property-change pos 'block-end) (point-max)))
(or (next-single-property-change pos 'block-end)
(point-max)))
(t
nil)
))
(defun web-mode-block-previous-position (&optional pos)
(unless pos (setq pos (point)))
(cond
((= pos (point-min))
(setq pos nil))
((get-text-property pos 'block-side)
(setq pos (web-mode-block-beginning-position pos))
(cond
((or (null pos) (= pos (point-min)))
(setq pos nil)
)
((and (setq pos (previous-single-property-change pos 'block-beg))
(> pos (point-min)))
(setq pos (1- pos))
)
)
) ;block-side
((get-text-property (1- pos) 'block-side)
(setq pos (web-mode-block-beginning-position (1- pos)))
(cond
((or (null pos) (= pos (point-min)))
(setq pos nil)
)
((and (setq pos (previous-single-property-change pos 'block-beg))
(> pos (point-min)))
(setq pos (1- pos))
)
)
) ;block-side
(t
(setq pos (previous-single-property-change pos 'block-side))
;; (message "pos(%S)" pos)
(cond
((and (null pos) (get-text-property (point-min) 'block-beg))
(setq pos (point-min)))
((and pos (> pos (point-min)))
(setq pos (web-mode-block-beginning-position (1- pos))))
)
)
) ;conf
pos)
(defun web-mode-block-next-position (&optional pos limit)
(unless pos (setq pos (point)))
(unless limit (setq limit (point-max)))
(cond
((and (get-text-property pos 'block-side)
(setq pos (web-mode-block-end-position pos))
(> (point-max) pos)
(setq pos (1+ pos)))
(if (get-text-property pos 'block-beg)
pos
(setq pos (next-single-property-change pos 'block-side)))
)
(t
(setq pos (next-single-property-change pos 'block-side)))
) ;cond
(when (and pos (> pos limit))
(setq pos nil))
pos)
;;---- MOVING ------------------------------------------------------------------
(defun web-mode-closing-paren (limit)
"web-mode-closing-paren"
(let ((pos (web-mode-closing-paren-position (point) limit)))
(if (or (null pos) (> pos limit))
nil
(goto-char pos)
pos)
))
(defun web-mode-tag-beginning ()
"Fetch html tag beg."
(interactive)
(let ((pos (web-mode-tag-beginning-position (point))))
(when pos (goto-char pos))
pos))
(defun web-mode-tag-end ()
"Fetch html tag end."
(interactive)
(let ((pos (web-mode-tag-end-position (point))))
(when pos
(setq pos (1+ pos))
(goto-char pos))
pos))
(defun web-mode-tag-previous ()
"Fetch previous tag."
(interactive)
(let ((pos (web-mode-tag-previous-position (point))))
(when pos (goto-char pos))
pos))
(defun web-mode-tag-next ()
"Fetch next tag. Might be HTML comment or server tag (ie. JSP)."
(interactive)
(let ((pos (web-mode-tag-next-position (point))))
(when pos (goto-char pos))
pos))
(defun web-mode-attribute-beginning ()
"Fetch html attribute end."
(interactive)
(let ((pos (web-mode-attribute-beginning-position (point))))
(when pos (goto-char pos))
pos))
(defun web-mode-attribute-end ()
"Fetch html attribute end."
(interactive)
(let ((pos (web-mode-attribute-end-position (point))))
(when pos
(setq pos (1+ pos))
(goto-char pos))
pos))
(defun web-mode-attribute-next ()
"Fetch next attribute."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-attribute-next-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-element-previous ()
"Fetch previous element."
(interactive)
(let ((continue (not (bobp)))
(ret)
(pos (point))
(props '(start void)))
(while continue
(setq ret (web-mode-tag-previous))
(when (or (null ret)
(member (get-text-property (point) 'tag-type) props))
(setq continue nil))
) ;while
(unless ret (goto-char pos))
ret))
(defun web-mode-element-next ()
"Fetch next element."
(interactive)
(let ((pos (web-mode-element-next-position (point))))
(when pos (goto-char pos))
pos))
(defun web-mode-element-sibling-next ()
"Fetch next element."
(interactive)
(let (parent ret (pos (point)))
(save-excursion
(cond
((not (get-text-property pos 'tag-type))
(when (and (web-mode-element-parent)
(web-mode-tag-match)
(web-mode-element-next))
(setq ret (point))
)
)
((eq (get-text-property pos 'tag-type) 'start)
(when (and (web-mode-tag-match)
(web-mode-element-next))
(setq ret (point))
)
)
((web-mode-element-next)
(setq ret (point))
)
) ;cond
) ;save
(if ret (goto-char ret))
))
(defun web-mode-element-beginning ()
"Move to beginning of element."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-element-beginning-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-element-end ()
"Move to end of element."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-element-end-position pos))
(when pos
(setq pos (1+ pos))
(goto-char pos))
pos))
(defun web-mode-element-parent ()
"Fetch parent element."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-element-parent-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-element-child ()
"Fetch child element."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-element-child-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-dom-traverse ()
"Traverse html dom tree."
(interactive)
(cond
((web-mode-element-child)
)
((web-mode-element-sibling-next)
)
((web-mode-element-parent)
(unless (web-mode-element-sibling-next)
(goto-char (point-min)))
)
) ;cond
)
(defun web-mode-part-next ()
"Move point to the beginning of the next part."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-part-next-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-block-opening-paren (limit)
"opening paren"
(let ((pos (point)))
(setq pos (web-mode-block-opening-paren-position pos limit))
(when pos (goto-char pos))
))
(defun web-mode-block-previous ()
"Move point to the beginning of the previous block."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-block-previous-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-block-next ()
"Move point to the beginning of the next block."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-block-next-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-block-beginning ()
"Move point to the beginning of the current block."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-block-beginning-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-block-token-beginning ()
"Move point to the beginning of the current block token."
(interactive)
(let ((pos (point)))
(setq pos (web-mode-block-token-beginning-position pos))
(when pos (goto-char pos))
pos))
(defun web-mode-block-end ()
"web-mode-block-beg"
(interactive)
(let ((pos (point)))
(setq pos (web-mode-block-end-position pos))
(when pos
(setq pos (1+ pos))
(goto-char pos))
pos))
;;---- SEARCH ------------------------------------------------------------------
(defun web-mode-rsf-balanced (regexp-open regexp-close &optional limit noerror)
"web-mode-rsf-balanced in client."
(unless noerror (setq noerror t))
(let ((continue t)
(level 1)
(pos (point))
ret
(regexp (concat regexp-open "\\|" regexp-close)))
(while continue
(setq ret (re-search-forward regexp limit noerror))
(cond
((null ret)
(setq continue nil)
)
(t
;; (message "%S" (match-string-no-properties 0))
(if (string-match-p regexp-open (match-string-no-properties 0))
(setq level (1+ level))
(setq level (1- level)))
(when (< level 1)
(setq continue nil)
)
) ;t
) ;cond
) ;while
(when (not (= level 0)) (goto-char pos))
;; (message "ret=%S level=%S" ret level)
ret))
(defun web-mode-block-sb (expr &optional limit noerror)
"re-search-backward inside a block (outside tokens)."
(unless limit (setq limit (web-mode-block-beginning-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (search-backward expr limit noerror))
(when (or (null ret)
(not (get-text-property (point) 'block-token)))
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-block-sf (expr &optional limit noerror)
"re-search-backward inside a block (outside tokens)."
(unless limit (setq limit (web-mode-block-end-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (search-forward expr limit noerror))
(when (or (null ret)
(not (get-text-property (point) 'block-token)))
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-block-rsb (regexp &optional limit noerror)
"re-search-backward inside a block (outside tokens)."
(unless limit (setq limit (web-mode-block-beginning-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (re-search-backward regexp limit noerror))
(when (or (null ret)
(not (get-text-property (point) 'block-token)))
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-block-rsf (regexp &optional limit noerror)
"re-search-backward inside a block (outside tokens)."
(unless limit (setq limit (web-mode-block-end-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (re-search-forward regexp limit noerror))
(when (or (null ret)
(not (get-text-property (point) 'block-token)))
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-part-sb (expr &optional limit noerror)
"search-backward inside a part (outside part tokens and blocks)."
(unless limit (setq limit (web-mode-part-beginning-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (search-backward expr limit noerror))
(when (or (null ret)
(and (not (get-text-property (point) 'part-token))
(not (get-text-property (point) 'block-side)))
)
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-part-sf (expr &optional limit noerror)
"search-forward inside a part (outside part tokens and block)."
(unless limit (setq limit (web-mode-part-end-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (search-forward expr limit noerror))
(when (or (null ret)
(and (not (get-text-property (point) 'part-token))
(not (get-text-property (point) 'block-side)))
)
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-part-rsb (regexp &optional limit noerror)
"re-search-backward inside a part (outside part tokens and blocks)."
(unless limit (setq limit (web-mode-part-beginning-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (re-search-backward regexp limit noerror))
(when (or (null ret)
(and (not (get-text-property (point) 'part-token))
(not (get-text-property (point) 'block-side)))
)
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-part-rsf (regexp &optional limit noerror)
"re-search-forward inside a part (outside part tokens and block)."
(unless limit (setq limit (web-mode-part-end-position (point))))
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (re-search-forward regexp limit t))
(when (or (null ret)
(and (not (get-text-property (point) 'part-token))
(not (get-text-property (point) 'block-side)))
)
(setq continue nil)
) ;when
) ;while
ret))
(defun web-mode-dom-sf (expr &optional limit noerror)
"search-forward outside blocks."
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (search-forward expr limit noerror))
(if (or (null ret)
(not (get-text-property (- (point) (length expr)) 'block-side)))
(setq continue nil))
)
ret))
(defun web-mode-dom-rsf (regexp &optional limit noerror)
"re-search-forward outside blocks."
(unless noerror (setq noerror t))
(let ((continue t) (ret nil))
(while continue
(setq ret (re-search-forward regexp limit noerror))
;; (message "ret=%S point=%S limit=%S i=%S" ret (point) limit 0)
(when (or (null ret)
(not (get-text-property (match-beginning 0) 'block-side)))
(setq continue nil))
) ;while
ret))
(defun web-mode-rsb (regexp &optional limit noerror)
"re-search-backward not in comment or string."
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (re-search-backward regexp limit noerror))
(if (or (null ret)
(not (web-mode-is-comment-or-string)))
(setq continue nil)))
ret))
(defun web-mode-rsf (regexp &optional limit noerror)
"re-search-forward not in comment or string."
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (re-search-forward regexp limit noerror))
(if (or (null ret)
(not (web-mode-is-comment-or-string)))
(setq continue nil))
)
ret))
(defun web-mode-sb (expr &optional limit noerror)
"re-search-backward not in comment or string."
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (search-backward expr limit noerror))
(if (or (null ret)
(not (web-mode-is-comment-or-string)))
(setq continue nil)))
ret))
(defun web-mode-sf (expr &optional limit noerror)
"re-search-backward not in comment or string."
(unless noerror (setq noerror t))
(let ((continue t) ret)
(while continue
(setq ret (search-forward expr limit noerror))
(if (or (null ret)
(not (web-mode-is-comment-or-string)))
(setq continue nil)))
ret))
(defun web-mode-rsf-content (regexp &optional limit noerror)
"re-search-forward only in html content."
(unless noerror (setq noerror t))
(let ((continue t) ret beg end)
(while continue
(setq ret (re-search-forward regexp limit noerror)
beg (if (null ret) (point) (match-beginning 0))
end (if (null ret) (point) (1- (match-end 0))))
(if (or (null ret)
(and (web-mode-is-content beg)
(web-mode-is-content end)))
(setq continue nil)))
ret))
(defun web-mode-is-comment-or-string-line ()
"Detect if current line is in a comment or in a string."
(save-excursion
(let ((continue t) (counter 0))
(beginning-of-line)
(back-to-indentation)
(while (and continue (not (eolp)))
(cond
((web-mode-is-comment-or-string)
(setq counter (1+ counter)))
((not (eq ?\s (following-char)))
(setq continue nil
counter 0))
) ;cond
(forward-char)
) ;while
(> counter 0)
)))
(defun web-mode-block-is-token-line ()
"Detect if current line is a comment or a string."
(save-excursion
(let ((continue t) (counter 0))
(beginning-of-line)
(back-to-indentation)
(while (and continue (not (eolp)))
(cond
((web-mode-block-is-token)
(setq counter (1+ counter)))
((not (eq ?\s (following-char)))
(setq continue nil
counter 0))
) ;cond
(forward-char)
) ;while
(> counter 0)
)))
(defun web-mode-is-part-token-or-server (&optional pos)
"Detect if POS is in a comment, a string or in server script."
(unless pos (setq pos (point)))
(not (null (or (get-text-property pos 'block-side)
(member (get-text-property pos 'part-token) '(comment string)))))
)
(defun web-mode-is-part-token-line ()
"Detect if current line has only client tokens (string/comment) or server blocks."
(save-excursion
(let ((continue t) (counter 0))
(beginning-of-line)
(while (and continue (not (eolp)))
(if (web-mode-is-part-token-or-server)
(setq counter (1+ counter))
(when (not (eq ?\s (following-char)))
(setq continue nil
counter 0))
) ;if
(forward-char)
) ;while
(> counter 0)
)))
(defun web-mode-is-content (&optional pos)
"Is pos in a html text."
(unless pos (setq pos (point)))
(not (or (get-text-property pos 'part-side)
(get-text-property pos 'tag-type)
(get-text-property pos 'block-side)
)))
(defun web-mode-block-is-token (&optional pos)
"Detect if point is in a comment or in a string."
(unless pos (setq pos (point)))
(not (null (get-text-property pos 'block-token)))
)
(defun web-mode-is-comment-or-string (&optional pos)
"Detect if point is in a comment or in a string."
(unless pos (setq pos (point)))
(not (null (or (eq (get-text-property pos 'tag-type) 'comment)
(member (get-text-property pos 'block-token) '(comment string))
(member (get-text-property pos 'part-token) '(comment string)))))
)
(defun web-mode-is-comment (&optional pos)
"Detect if point is in a comment."
(unless pos (setq pos (point)))
(not (null (or (eq (get-text-property pos 'tag-type) 'comment)
(eq (get-text-property pos 'block-token) 'comment)
(eq (get-text-property pos 'part-token) 'comment))))
)
;;---- ADVICES -----------------------------------------------------------------
(defadvice ac-start (before web-mode-set-up-ac-sources activate)
"Set `ac-sources' based on current language before running auto-complete."
(if (equal major-mode 'web-mode)
(progn
(run-hooks 'web-mode-before-auto-complete-hooks)
(let ((new-web-mode-ac-sources
(assoc (web-mode-language-at-pos)
web-mode-ac-sources-alist)))
(setq ac-sources (cdr new-web-mode-ac-sources))))))
;;---- MINOR MODE ADDONS -------------------------------------------------------
(defun web-mode-yasnippet-exit-hook ()
"Yasnippet exit hook"
(when (and (boundp 'yas-snippet-beg) (boundp 'yas-snippet-end))
(indent-region yas-snippet-beg yas-snippet-end)))
(defun web-mode-imenu-index ()
(interactive)
"return imenu items"
(let (toc-index
line)
(save-excursion
(goto-char (point-min))
(while (not (eobp))
(setq line (buffer-substring-no-properties
(line-beginning-position)
(line-end-position)))
(let (found
(i 0)
item
regexp
type
type-idx
content
content-idx
content-regexp
close-tag-regexp
concat-str
jumpto
str)
(while (and (not found ) (< i (length web-mode-imenu-regexp-list)))
(setq item (nth i web-mode-imenu-regexp-list))
(setq regexp (nth 0 item))
(setq type-idx (nth 1 item))
(setq content-idx (nth 2 item))
(setq concat-str (nth 3 item))
(when (not (numberp content-idx))
(setq content-regexp (nth 2 item)
close-tag-regexp (nth 4 item)
content-idx nil))
(when (string-match regexp line)
(cond
(content-idx
(setq type (match-string type-idx line))
(setq content (match-string content-idx line))
(setq str (concat type concat-str content))
(setq jumpto (line-beginning-position)))
(t
(let (limit)
(setq type (match-string type-idx line))
(goto-char (line-beginning-position))
(save-excursion
(setq limit (re-search-forward close-tag-regexp (point-max) t)))
(when limit
(when (re-search-forward content-regexp limit t)
(setq content (match-string 1))
(setq str (concat type concat-str content))
(setq jumpto (line-beginning-position))
)
)))
)
(when str (setq toc-index
(cons (cons str jumpto)
toc-index)
)
(setq found t))
)
(setq i (1+ i))))
(forward-line)
(goto-char (line-end-position)) ;; make sure we are at eobp
))
(nreverse toc-index)))
;;---- UNIT TESTING ------------------------------------------------------------
(defun web-mode-test ()
"Exec web-mode unit tests. See web-mode-tests-directory."
(interactive)
(let (files ret regexp)
(setq regexp "^[[:alnum:]][[:alnum:].]+\\'")
;; (setq regexp "a\\.jsx\\'")
(setq files (directory-files web-mode-tests-directory t regexp))
(dolist (file files)
(setq ret (web-mode-test-process file)))
))
(defun web-mode-test-process (file)
(with-temp-buffer
(let (out sig1 sig2 success err)
(setq-default indent-tabs-mode nil)
(insert-file-contents file)
(set-visited-file-name file)
(web-mode)
(setq sig1 (md5 (current-buffer)))
(delete-horizontal-space)
(while (not (eobp))
(forward-line)
(delete-horizontal-space)
(end-of-line))
(web-mode-buffer-indent)
(setq sig2 (md5 (current-buffer)))
(setq success (string= sig1 sig2))
(setq out (concat (if success "ok" "ko") " : " (file-name-nondirectory file)))
(message out)
(setq err (concat (file-name-directory file) "_err." (file-name-nondirectory file)))
(if success
(when (file-readable-p err)
(delete-file err))
(write-file err)
(message "[%s]" (buffer-string))
) ;if
out)))
;;---- MISC --------------------------------------------------------------------
(defun web-mode-set-engine (engine)
"set engine"
(interactive
(list (completing-read
"Engine: "
(let (engines elt)
(dolist (elt web-mode-engines)
(setq engines (append engines (list (car elt)))))
engines))))
(setq web-mode-content-type "html"
web-mode-engine engine)
(web-mode-on-engine-setted)
(web-mode-buffer-highlight))
(defun web-mode-set-content-type (content-type)
"set engine"
(setq web-mode-content-type content-type)
(web-mode-buffer-highlight))
(defun web-mode-on-engine-setted ()
"engine setted"
(let (elt elts engines)
(when (string= web-mode-engine "razor") (setq web-mode-enable-block-face t))
(setq web-mode-engine-attr-regexp (cdr (assoc web-mode-engine web-mode-engine-attr-regexps)))
(setq web-mode-engine-token-regexp (cdr (assoc web-mode-engine web-mode-engine-token-regexps)))
(setq elt (assoc web-mode-engine web-mode-engine-open-delimiter-regexps))
(if elt
(setq web-mode-block-regexp (cdr elt))
(setq web-mode-engine "none"))
(unless (boundp 'web-mode-extra-auto-pairs)
(setq web-mode-extra-auto-pairs nil))
(setq web-mode-auto-pairs
(append
(cdr (assoc web-mode-engine web-mode-engines-auto-pairs))
(cdr (assoc nil web-mode-engines-auto-pairs))
(cdr (assoc web-mode-engine web-mode-extra-auto-pairs))
(cdr (assoc nil web-mode-extra-auto-pairs))))
(unless (boundp 'web-mode-extra-snippets)
(setq web-mode-extra-snippets nil))
(setq elts
(append
(cdr (assoc web-mode-engine web-mode-extra-snippets))
(cdr (assoc nil web-mode-extra-snippets))
(cdr (assoc web-mode-engine web-mode-engines-snippets))
(cdr (assoc nil web-mode-engines-snippets))))
(dolist (elt elts)
(unless (assoc (car elt) web-mode-snippets)
(setq web-mode-snippets (append (list elt) web-mode-snippets)))
)
;; (message "wms=%S" web-mode-snippets)
(setq web-mode-closing-blocks (cdr (assoc web-mode-engine web-mode-engines-closing-blocks)))
(setq web-mode-engine-font-lock-keywords
(symbol-value (cdr (assoc web-mode-engine web-mode-engines-font-lock-keywords))))
;; (message "%S" (symbol-value (cdr (assoc web-mode-engine web-mode-engines-font-lock-keywords))))
))
(defun web-mode-guess-engine-and-content-type ()
"Try to guess the server engine and the content type."
(let (buff-name elt found)
(setq buff-name (buffer-file-name))
(unless buff-name (setq buff-name (buffer-name)))
(setq web-mode-is-scratch (string= buff-name "*scratch*"))
(setq web-mode-content-type nil)
(when (boundp 'web-mode-content-types-alist)
(setq found nil)
(dolist (elt web-mode-content-types-alist)
(when (and (not found) (string-match-p (cdr elt) buff-name))
(setq web-mode-content-type (car elt)
found t))
) ;dolist
) ;when
(unless web-mode-content-type
(setq found nil)
(dolist (elt web-mode-content-types)
(when (and (not found) (string-match-p (cdr elt) buff-name))
(setq web-mode-content-type (car elt)
found t))
) ;dolist
) ;unless
(when (boundp 'web-mode-engines-alist)
(setq found nil)
(dolist (elt web-mode-engines-alist)
(cond
((stringp (cdr elt))
(when (string-match-p (cdr elt) buff-name)
(setq web-mode-engine (car elt))))
((functionp (cdr elt))
(when (funcall (cdr elt))
(setq web-mode-engine (car elt))))
) ;cond
) ;dolist
) ;when
(unless web-mode-engine
(setq found nil)
(dolist (elt web-mode-engine-file-regexps)
;; (message "%S %S" (cdr elt) buff-name)
(when (and (not found) (string-match-p (cdr elt) buff-name))
(setq web-mode-engine (car elt)
found t))
)
)
(unless web-mode-engine
(setq found nil)
(dolist (elt web-mode-engines)
;; (message "%S %S" (car elt) buff-name)
(when (and (not found) (string-match-p (car elt) buff-name))
(setq web-mode-engine (car elt)
found t))
)
)
;; TODO : remplacer par web-mode-engine-canonical-name
(when web-mode-engine
(setq found nil)
(dolist (elt web-mode-engines)
;; (message "%S" elt)
(when (and (not found) (member web-mode-engine (cdr elt)))
(setq web-mode-engine (car elt)
found t))
)
)
(when (and (null found)
(string-match-p "php" (buffer-substring-no-properties
(line-beginning-position)
(line-end-position))))
(setq web-mode-engine "php"
found t)
)
(when (and (string= web-mode-content-type "javascript")
(string-match-p "@jsx"
(buffer-substring-no-properties
(point-min)
(if (< (point-max) web-mode-chunk-length)
(point-max)
web-mode-chunk-length)
)))
(setq web-mode-content-type "jsx")
) ;when
(when (and web-mode-enable-engine-detection
(or (null web-mode-engine)
(string= web-mode-engine "none")))
;;(message "%S %S" web-mode-enable-engine-detection web-mode-engine)
(web-mode-detect-engine))
(web-mode-on-engine-setted)
))
(defun web-mode-engine-canonical-name (name)
(let (engine)
(cond
((null name)
)
((assoc name web-mode-engines)
(setq engine name))
(t
(dolist (elt web-mode-engines)
(when (and (null engine) (member name (cdr elt)))
(setq engine (car elt)))
) ;dolist
) ;t
)
;;(message "name=%S engine=%S" name engine)
engine))
(defun web-mode-on-after-save ()
(when web-mode-is-scratch
(web-mode-guess-engine-and-content-type)
(web-mode-buffer-highlight))
nil)
(defun web-mode-on-exit ()
"Exit web-mode."
(interactive)
(web-mode-with-silent-modifications
(put-text-property (point-min) (point-max) 'invisible nil)
(remove-overlays)
(remove-hook 'change-major-mode-hook 'web-mode-on-exit t)
))
(defun web-mode-reload ()
"Reload web-mode."
(interactive)
(web-mode-with-silent-modifications
(put-text-property (point-min) (point-max) 'invisible nil)
(remove-overlays)
(unload-feature 'web-mode t)
(load "web-mode.el")
(setq web-mode-change-beg nil
web-mode-change-end nil)
(web-mode)
(when (fboundp 'web-mode-hook)
(web-mode-hook))
) ;silent
)
(defun web-mode-trace (msg)
"Benchmark."
(interactive)
(let (sub)
;; (when (null web-mode-time) (setq web-mode-time (current-time)))
(setq sub (time-subtract (current-time) web-mode-time))
(when nil
(save-excursion
(let ((n 0))
(goto-char (point-min))
(while (web-mode-tag-next)
(setq n (1+ n))
)
(message "%S tags found" n)
)))
(message "%18s: time elapsed = %Ss %9Sµs" msg (nth 1 sub) (nth 2 sub))
))
(defun web-mode-reveal ()
"Display text properties at point"
(interactive)
(let (symbol symbols out)
(setq out (format "[point=%S engine=%S content-type=%S language-at-pos=%S]\n"
(point)
web-mode-engine
web-mode-content-type
(web-mode-language-at-pos (point))))
(setq symbols (append web-mode-scan-properties '(font-lock-face)))
(dolist (symbol symbols)
(when symbol
(setq out (concat out (format "%s(%S) " (symbol-name symbol) (get-text-property (point) symbol)))))
)
(message "%s\n" out)
(message nil)
))
(defun web-mode-debug ()
"Display informations useful for debuging"
(interactive)
(let (modes)
(message "\n")
(message "--- WEB-MODE DEBUG BEG ---")
(message "versions: emacs(%S.%S) web-mode(%S)"
emacs-major-version emacs-minor-version web-mode-version)
(message "vars: engine(%S) content-type(%S) file(%S)"
web-mode-engine
web-mode-content-type
(or (buffer-file-name) (buffer-name)))
(message "system: window(%S) config(%S)" window-system system-configuration)
(message "colors: fg(%S) bg(%S) "
(cdr (assoc 'foreground-color default-frame-alist))
(cdr (assoc 'background-color default-frame-alist)))
(mapc (lambda (mode)
(condition-case nil
(if (and (symbolp mode) (symbol-value mode))
(add-to-list 'modes mode))
(error nil))
) ;lambda
minor-mode-list)
(message "minor modes: %S" modes)
(message "--- WEB-MODE DEBUG END ---")
(switch-to-buffer "*Messages*")
(goto-char (point-max))
(recenter)
))
;;; web-mode.el ends here
(provide 'web-mode)
;; Local Variables:
;; coding: utf-8
;; indent-tabs-mode: nil
;; End:
;;---- TODO --------------------------------------------------------------------
;;- parameter for function chaining
;;- supprimer 2 flags sur blocks
;;- phphint
;;- tag-name uniquement sur les html tag
;;- Stickiness of Text Properties
;;- screenshot : http://www.cockos.com/licecap/
;;- tester shortcut A -> pour pomme
;; (defun web-mode-indent-cycle (regex-line regex-sym block-beg indent-offset)
;; "Returns next position in the indent cycle for REGEX-SYM on
;; positions from the previous line matching REGEX-LINE withing
;; BLOCK-BEGIN. Loops to start at INDENT-OFFSET."
;; (letrec
;; ((match-indices-all (lambda (regex string)
;; (let ((i (string-match-p regex string)))
;; (if i (cons
;; i
;; (mapcar (lambda (x) (+ x i 1))
;; (funcall match-indices-all regex
;; (substring string (+ i 1)))))))))
;; (filter (lambda (condp lst)
;; (delq nil
;; (mapcar (lambda (x)
;; (and (funcall condp x) x)) lst))))
;; (this-line (thing-at-point 'line))
;; (rsb-prev-line (progn
;; (web-mode-rsb regex-line block-beg)
;; (thing-at-point 'line)))
;; (pos-of-this-sym (string-match-p regex-sym this-line))
;; (prev-sym-locations (funcall match-indices-all regex-sym rsb-prev-line))
;; (farther-syms (progn
;; (add-to-list 'prev-sym-locations (+ indent-offset web-mode-code-indent-offset))
;; (funcall filter (lambda (i) (> i pos-of-this-sym))
;; (sort prev-sym-locations '<)))))
;; (cond ((null farther-syms) indent-offset)
;; ((or web-mode-indent-cycle-left-first
;; (equal last-command 'indent-for-tab-command)) (car farther-syms))
;; (t (car (last farther-syms))))))