diff options
Diffstat (limited to 'elpa/emmet-mode-20210820.1124/emmet-mode.el')
-rw-r--r-- | elpa/emmet-mode-20210820.1124/emmet-mode.el | 4274 |
1 files changed, 4274 insertions, 0 deletions
diff --git a/elpa/emmet-mode-20210820.1124/emmet-mode.el b/elpa/emmet-mode-20210820.1124/emmet-mode.el new file mode 100644 index 0000000..77544f6 --- /dev/null +++ b/elpa/emmet-mode-20210820.1124/emmet-mode.el @@ -0,0 +1,4274 @@ +;;; emmet-mode.el --- Unofficial Emmet's support for emacs + +;; Copyright (C) 2014- Dmitry Mukhutdinov (@flyingleafe https://github.com/flyingleafe) +;; Copyright (C) 2014- William David Mayo (@pbocks https://github.com/pobocks) +;; Copyright (C) 2013- Shin Aoyama (@smihica https://github.com/smihica) +;; Copyright (C) 2009-2012 Chris Done + +;; Version: 1.0.10 +;; Package-Version: 20210820.1124 +;; Package-Commit: 6b2e554f7fd27f732810f4b14ea01e3c54b7b3da +;; Author: Shin Aoyama <smihica@gmail.com> +;; URL: https://github.com/smihica/emmet-mode +;; Last-Updated: 2014-08-11 Mon +;; Keywords: convenience + +;; This file is free software; you can redistribute it and/or modify +;; it under the terms of the GNU General Public License as published by +;; the Free Software Foundation; either version 3, or (at your option) +;; any later version. +;; +;; This file is distributed in the hope that it will be useful, +;; but WITHOUT ANY WARRANTY; without even the implied warranty of +;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +;; GNU General Public License for more details. +;; +;; You should have received a copy of the GNU General Public License +;; along with GNU Emacs; see the file COPYING. If not, write to +;; the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, +;; Boston, MA 02110-1301, USA. + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; +;;; Commentary: +;; +;; Unfold CSS-selector-like expressions to markup. Intended to be used +;; with sgml-like languages; xml, html, xhtml, xsl, etc. +;; +;; See `emmet-mode' for more information. +;; +;; Copy emmet-mode.el to your load-path and add to your .emacs: +;; +;; (require 'emmet-mode) +;; +;; Example setup: +;; +;; (add-to-list 'load-path "~/Emacs/emmet/") +;; (require 'emmet-mode) +;; (add-hook 'sgml-mode-hook 'emmet-mode) ;; Auto-start on any markup modes +;; (add-hook 'html-mode-hook 'emmet-mode) +;; (add-hook 'css-mode-hook 'emmet-mode) +;; +;; Enable the minor mode with M-x emmet-mode. +;; +;; See ``Test cases'' section for a complete set of expression types. +;; +;; If you are hacking on this project, eval (emmet-test-cases) to +;; ensure that your changes have not broken anything. Feel free to add +;; new test cases if you add new features. +;; +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; +;;; History: +;; +;; This is a fork of zencoding-mode to support Emmet's feature. +;; zencoding-mode (https://github.com/rooney/zencoding) + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; +;;; Code: + +(defconst emmet-mode:version "1.0.10") + +(with-no-warnings + (require 'cl)) + +;; for portability with < 24.3 EMACS +(unless (fboundp 'cl-labels) (fset 'cl-labels 'labels)) +(unless (fboundp 'cl-flet) (fset 'cl-flet 'flet)) +;; < 22.1 +(unless (fboundp 'string-to-number) (fset 'string-to-number 'string-to-int)) + +(defmacro emmet-defparameter (symbol &optional initvalue docstring) + `(progn + (defvar ,symbol nil ,docstring) + (setq ,symbol ,initvalue))) + +(defun emmet-join-string (lis joiner) + (mapconcat 'identity lis joiner)) + +(defun emmet-get-keys-of-hash (hash) + (let ((ks nil)) + (maphash #'(lambda (k v) (setq ks (cons k ks))) hash) + ks)) + +(defun emmet-get-vals-of-hash (hash) + (let ((vs nil)) + (maphash #'(lambda (k v) (setq vs (cons v vs))) hash) + vs)) + +(defun emmet-jsx-prop-value-var? (prop-value) + (string-match "{.+}" prop-value)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Generic parsing macros and utilities + +(defmacro emmet-aif (test-form then-form &rest else-forms) + "Anaphoric if. Temporary variable `it' is the result of test-form." + `(let ((it ,test-form)) + (if it ,then-form ,@(or else-forms '(it))))) + +(defmacro emmet-pif (test-form then-form &rest else-forms) + "Parser anaphoric if. Temporary variable `it' is the result of test-form." + `(let ((it ,test-form)) + (if (not (eq 'error (car it))) ,then-form ,@(or else-forms '(it))))) + +(defmacro emmet-parse (regex nums label &rest body) + "Parse according to a regex and update the `input' variable." + `(emmet-aif (emmet-regex ,regex input ',(number-sequence 0 nums)) + (let ((input (elt it ,nums))) + ,@body) + `,`(error ,(concat "expected " ,label)))) + +(defmacro emmet-run (parser then-form &rest else-forms) + "Run a parser and update the input properly, extract the parsed + expression." + `(emmet-pif (,parser input) + (let ((input (cdr it)) + (expr (car it))) + ,then-form) + ,@(or else-forms '(it)))) + +(defmacro emmet-por (parser1 parser2 then-form &rest else-forms) + "OR two parsers. Try one parser, if it fails try the next." + `(emmet-pif (,parser1 input) + (let ((input (cdr it)) + (expr (car it))) + ,then-form) + (emmet-pif (,parser2 input) + (let ((input (cdr it)) + (expr (car it))) + ,then-form) + ,@else-forms))) + +(defmacro emmet-find (direction regexp &optional limit-of-search repeat-count) + "Regexp-search in given direction, returning the position (or nil) +and leaving the point in place." + `(save-excursion + (if (,(intern (concat "re-search-" direction)) + ,regexp ,limit-of-search t ,repeat-count) + (match-beginning 0)))) + +(defun emmet-regex (regexp string refs) + "Return a list of (`ref') matches for a `regex' on a `string' or nil." + (if (string-match (concat "^" regexp "\\([^\n]*\\)$") string) + (mapcar (lambda (ref) (match-string ref string)) + (if (sequencep refs) refs (list refs))) + nil)) +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Emmet minor mode + +(defgroup emmet nil + "Customization group for emmet-mode." + :group 'convenience) + +(defun emmet-expr-on-line () + "Extract a emmet expression and the corresponding bounds + for the current line." + (let* ((end (point)) + (start (emmet-find-left-bound)) + (line (buffer-substring-no-properties start end)) + (expr (emmet-regex "\\([ \t]*\\)\\([^\n]+\\)" line 2))) + (if (first expr) + (list (first expr) start end)))) + +(defun emmet-find-left-bound () + "Find the left bound of an emmet expr" + (save-excursion (save-match-data + (let ((char (char-before)) + (in-style-attr (looking-back "style=[\"'][^\"']*" nil)) + (syn-tab (make-syntax-table))) + (modify-syntax-entry ?\\ "\\") + (while char + (cond ((and in-style-attr (member char '(?\" ?\'))) + (setq char nil)) + ((member char '(?\} ?\] ?\))) + (with-syntax-table syn-tab + (backward-sexp) (setq char (char-before)))) + ((eq char ?\>) + (if (looking-back "<[^>]+>" (line-beginning-position)) + (setq char nil) + (progn (backward-char) (setq char (char-before))))) + ((not (string-match-p "[[:space:]\n;]" (string char))) + (backward-char) (setq char (char-before))) + (t + (setq char nil)))) + (point))))) + +(defcustom emmet-indentation 4 + "Number of spaces used for indentation." + :type '(number :tag "Spaces") + :group 'emmet) + +(defcustom emmet-indent-after-insert t + "Indent region after insert?" + :type 'boolean + :group 'emmet) + +(defcustom emmet-use-style-tag-and-attr-detection t + "When true, enables detection of style tags and attributes in HTML +to provide proper CSS abbreviations completion." + :type 'boolean + :group 'emmet) + +(defcustom emmet-self-closing-tag-style "/" + "Self-closing tags style. + +This determines how Emmet expands self-closing tags. + +E.g., FOO is a self-closing tag. When expanding \"FOO\": + +When \" /\", the expansion is \"<FOO />\". +When \"/\", the expansion is \"<FOO/>\". +When \"\", the expansion is \"<FOO>\". + +Default value is \"/\". + +NOTE: only \" /\", \"/\" and \"\" are valid." + :type '(choice (const :tag " />" " /") + (const :tag "/>" "/") + (const :tag ">" "")) + :group 'emmet) + +(defvar emmet-use-css-transform nil + "When true, transform Emmet snippets into CSS, instead of the usual HTML.") +(make-variable-buffer-local 'emmet-use-css-transform) + +(defvar emmet-use-sass-syntax nil + "When true, uses Sass syntax for CSS abbreviations expanding, +e. g. without semicolons") +(make-variable-buffer-local 'emmet-use-sass-syntax) + + +(defvar emmet-css-major-modes + '(css-mode + scss-mode + sass-mode + less-mode + less-css-mode) + "Major modes that use emmet for CSS, rather than HTML.") + +(defvar emmet-fallback-filter '("html") + "Fallback filter for `emmet-default-filter', if none is found.") + +(defvar emmet-file-filter nil + "File local filter used by `emmet-default-filter'.") +(make-variable-buffer-local 'emmet-file-filter) + +(defvar emmet-jsx-major-modes + '(rjsx-mode + typescript-tsx-mode + js-jsx-mode + js2-jsx-mode + jsx-mode + js-mode) + "Which modes to check before using jsx class expansion") + +(defun emmet-transform (input) + (if (or (emmet-detect-style-tag-and-attr) emmet-use-css-transform) + (emmet-css-transform input) + (emmet-html-transform input))) + +(defun emmet-detect-style-tag-and-attr () + (let* ((style-attr-end "[^=][\"']") + (style-attr-begin "style=[\"']") + (style-tag-end "</style>") + (style-tag-begin "<style>")) + (and emmet-use-style-tag-and-attr-detection + (or + (emmet-check-if-between style-attr-begin style-attr-end) ; style attr + (emmet-check-if-between style-tag-begin style-tag-end))))) ; style tag + +(defun emmet-check-if-between (begin end) + (let ((begin-back (emmet-find "backward" begin)) + (end-back (emmet-find "backward" end)) + (begin-front (emmet-find "forward" begin)) + (end-front (emmet-find "forward" end))) + (and begin-back end-front + (or (not end-back) (> begin-back end-back)) + (or (not begin-front) (< end-front begin-front))))) + +(defcustom emmet-preview-default nil + "If non-nil then preview is the default action. +This determines how `emmet-expand-line' works by default." + :type 'boolean + :group 'emmet) + +;;;###autoload +(defun emmet-expand-line (arg) + "Replace the current line's emmet expression with the corresponding expansion. +If prefix ARG is given or region is visible call `emmet-preview' to start an +interactive preview. + +Otherwise expand line directly. + +For more information see `emmet-mode'." + (interactive "P") + (let* ((here (point)) + (preview (if emmet-preview-default (not arg) arg)) + (beg (if preview + (emmet-find-left-bound) + (when (use-region-p) (region-beginning)))) + (end (if preview + here + (when (use-region-p) (region-end))))) + (if (and preview beg) + (progn + (goto-char here) + (emmet-preview beg end)) + (let ((expr (emmet-expr-on-line))) + (if expr + (let ((markup (emmet-transform (first expr)))) + (when markup + (delete-region (second expr) (third expr)) + (emmet-insert-and-flash markup) + (emmet-reposition-cursor expr)))))))) + +(defvar emmet-mode-keymap + (let + ((map (make-sparse-keymap))) + (define-key map (kbd "C-j") 'emmet-expand-line) + (define-key map (kbd "<C-return>") 'emmet-expand-line) + (define-key map (kbd "<C-M-right>") 'emmet-next-edit-point) + (define-key map (kbd "<C-M-left>") 'emmet-prev-edit-point) + (define-key map (kbd "C-c C-c w") 'emmet-wrap-with-markup) + map) + "Keymap for emmet minor mode.") + +(defun emmet-after-hook () + "Initialize Emmet's buffer-local variables." + (if (memq major-mode emmet-css-major-modes) + (setq emmet-use-css-transform t)) + (if (eq major-mode 'sass-mode) + (setq emmet-use-sass-syntax t))) + +;;;###autoload +(define-minor-mode emmet-mode + "Minor mode for writing HTML and CSS markup. +With emmet for HTML and CSS you can write a line like + + ul#name>li.item*2 + +and have it expanded to + + <ul id=\"name\"> + <li class=\"item\"></li> + <li class=\"item\"></li> + </ul> + +This minor mode defines keys for quick access: + +\\{emmet-mode-keymap} + +Home page URL `http://www.emacswiki.org/emacs/Emmet'. + +See also `emmet-expand-line'." + :lighter (" Emmet" (:eval (if emmet-preview-mode "[P]" ""))) + :keymap emmet-mode-keymap + :after-hook (emmet-after-hook)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Emmet yasnippet integration + +(defun emmet-transform-yas (input) + (let* ((leaf-count 0) + (emmet-leaf-function + (lambda () + (format "$%d" (cl-incf leaf-count))))) + (emmet-transform input))) + +;;;###autoload +(defun emmet-expand-yas () + (interactive) + (let ((expr (emmet-expr-on-line))) + (if expr + (let* ((markup (emmet-transform-yas (first expr))) + (filled (replace-regexp-in-string "><" ">\n<" markup))) + (delete-region (second expr) (third expr)) + (insert filled) + (indent-region (second expr) (point)) + (if (fboundp 'yas/expand-snippet) + (yas/expand-snippet + (buffer-substring (second expr) (point)) + (second expr) (point))))))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;;; Real-time preview +;; + +;;;;;;;;;; +;; Lennart's version + +(defvar emmet-preview-input nil) +(make-local-variable 'emmet-preview-input) +(defvar emmet-preview-output nil) +(make-local-variable 'emmet-preview-output) +(defvar emmet-old-show-paren nil) +(make-local-variable 'emmet-old-show-paren) + +(defface emmet-preview-input + '((default :box t :inherit secondary-selection)) + "Face for preview input field." + :group 'emmet) + +(defface emmet-preview-output + '((default :inherit highlight)) + "Face for preview output field." + :group 'emmet) + +(defvar emmet-preview-keymap + (let ((map (make-sparse-keymap))) + (define-key map (kbd "RET") 'emmet-preview-accept) + (define-key map (kbd "<return>") 'emmet-preview-accept) + (define-key map [(control ?g)] 'emmet-preview-abort) + map)) + +(defun emmet-html-text-p (markup) + (string-match "^[\s|\t|\n|\r]*<.*$" markup)) + +(defun emmet-preview-accept () + (interactive) + (let ((ovli emmet-preview-input) + (expr (emmet-expr-on-line))) + (if (not (and (overlayp ovli) + (bufferp (overlay-buffer ovli)))) + (message "Preview is not active") + (let* ((indent (current-indentation)) + (markup (emmet-preview-transformed indent))) + (when markup + (delete-region (overlay-start ovli) (overlay-end ovli)) + (emmet-insert-and-flash markup) + (emmet-reposition-cursor expr))))) + (emmet-preview-abort)) + +(defun emmet-html-next-insert-point (str) + (with-temp-buffer + (insert str) + (goto-char (point-min)) + (or + (emmet-aif (emmet-go-to-edit-point 1 t) (- it 1)) ; try to find an edit point + (emmet-aif (re-search-forward ".+</" nil t) (- it 3)) ; try to place cursor after tag contents + (length str)))) ; ok, just go to the end + +(defun emmet-css-next-insert-point (str) + (let ((regexp (if emmet-use-sass-syntax ": *\\($\\)" ": *\\(;\\)$"))) + (save-match-data + (set-match-data nil t) + (string-match regexp str) + (or (match-beginning 1) (length str))))) + +(defvar emmet-flash-ovl nil) +(make-variable-buffer-local 'emmet-flash-ovl) + +(defun emmet-remove-flash-ovl (buf) + (with-current-buffer buf + (when (overlayp emmet-flash-ovl) + (delete-overlay emmet-flash-ovl)) + (setq emmet-flash-ovl nil))) + +(defcustom emmet-insert-flash-time 0.5 + "Time to flash insertion. +Set this to a negative number if you do not want flashing the +expansion after insertion." + :type '(number :tag "Seconds") + :group 'emmet) + +(defcustom emmet-move-cursor-after-expanding t + "If non-nil the the cursor position is +moved to before the first closing tag when the exp was expanded." + :type 'boolean + :group 'emmet) + +(defcustom emmet-move-cursor-between-quotes nil + "If emmet-move-cursor-after-expands is non-nil and this is non-nil then +cursor position will be moved to after the first quote." + :type 'boolean + :group 'emmet) + +(defun emmet-reposition-cursor (expr) + (let ((output-markup (buffer-substring-no-properties (second expr) (point)))) + (when emmet-move-cursor-after-expanding + (let ((p (point)) + (new-pos (if (emmet-html-text-p output-markup) + (emmet-html-next-insert-point output-markup) + (emmet-css-next-insert-point output-markup)))) + (goto-char + (+ (- p (length output-markup)) + new-pos)))))) + +(defun emmet-insert-and-flash (markup) + (emmet-remove-flash-ovl (current-buffer)) + (let ((here (point))) + (insert markup) + (when emmet-indent-after-insert + (indent-region here (point)) + (setq here + (save-excursion + (goto-char here) + (skip-chars-forward "[:space:]") + (point)))) + (setq emmet-flash-ovl (make-overlay here (point))) + (overlay-put emmet-flash-ovl 'face 'emmet-preview-output) + (when (< 0 emmet-insert-flash-time) + (run-with-idle-timer emmet-insert-flash-time + nil 'emmet-remove-flash-ovl (current-buffer))))) + +;;;###autoload +(defun emmet-preview (beg end) + "Expand emmet between BEG and END interactively. +This will show a preview of the expanded emmet code and you can +accept it or skip it." + (interactive (if (use-region-p) + (list (region-beginning) (region-end)) + (list nil nil))) + (emmet-preview-abort) + (if (not beg) + (message "Region not active") + (setq emmet-old-show-paren show-paren-mode) + (show-paren-mode -1) + (let ((here (point))) + (goto-char beg) + (forward-line 1) + (unless (= 0 (current-column)) + (insert "\n")) + (let* ((opos (point)) + (ovli (make-overlay beg end nil nil t)) + (ovlo (make-overlay opos opos)) + (info (propertize " Emmet preview. Choose with RET. Cancel by stepping out. \n" + 'face 'tooltip))) + (overlay-put ovli 'face 'emmet-preview-input) + (overlay-put ovli 'keymap emmet-preview-keymap) + (overlay-put ovlo 'face 'emmet-preview-output) + (overlay-put ovlo 'before-string info) + (setq emmet-preview-input ovli) + (setq emmet-preview-output ovlo) + (add-hook 'before-change-functions 'emmet-preview-before-change t t) + (goto-char here) + (add-hook 'post-command-hook 'emmet-preview-post-command t t))))) + +(defun emmet-preview-online () + "Display `emmet-preview' on the fly as the user types. + +To use this, add the function as a local hook: + + (add-hook 'post-self-insert-hook 'emmet-preview-online t t) + +or enable `emmet-preview-mode'." + (ignore-errors + (let* ((expr (emmet-expr-on-line)) + (text (nth 0 expr)) + (beg (nth 1 expr)) + (end (nth 2 expr))) + (let ((wap (thing-at-point 'word 'no-properties))) + (when (and (not (equal wap text)) + (emmet-transform text)) + (emmet-preview beg end)))))) + +(define-minor-mode emmet-preview-mode + "When enabled, automatically show `emmet-preview' as the user types. + +See `emmet-preview-online'." + :init-value nil + :group 'emmet + (if emmet-preview-mode + (add-hook 'post-self-insert-hook 'emmet-preview-online :append :local) + (remove-hook 'post-self-insert-hook 'emmet-preview-online :local))) + +(defvar emmet-preview-pending-abort nil) +(make-variable-buffer-local 'emmet-preview-pending-abort) + +(defun emmet-preview-before-change (beg end) + (when + (or (> beg (overlay-end emmet-preview-input)) + (< beg (overlay-start emmet-preview-input)) + (> end (overlay-end emmet-preview-input)) + (< end (overlay-start emmet-preview-input))) + (setq emmet-preview-pending-abort t))) + +(defun emmet-preview-abort () + "Abort emmet code preview." + (interactive) + (setq emmet-preview-pending-abort nil) + (remove-hook 'before-change-functions 'emmet-preview-before-change t) + (when (overlayp emmet-preview-input) + (delete-overlay emmet-preview-input)) + (setq emmet-preview-input nil) + (when (overlayp emmet-preview-output) + (delete-overlay emmet-preview-output)) + (setq emmet-preview-output nil) + (remove-hook 'post-command-hook 'emmet-preview-post-command t) + (when emmet-old-show-paren (show-paren-mode 1))) + +(defun emmet-preview-post-command () + (condition-case err + (emmet-preview-post-command-1) + (error (message "emmet-preview-post: %s" err)))) + +(defun emmet-preview-post-command-1 () + (if (and (not emmet-preview-pending-abort) + (<= (point) (overlay-end emmet-preview-input)) + (>= (point) (overlay-start emmet-preview-input))) + (emmet-update-preview (current-indentation)) + (emmet-preview-abort))) + +(defun emmet-preview-transformed (indent) + (let* ((string (buffer-substring-no-properties + (overlay-start emmet-preview-input) + (overlay-end emmet-preview-input)))) + (let ((output (emmet-transform string))) + (when output + output)))) + +(defun emmet-update-preview (indent) + (let* ((pretty (emmet-preview-transformed indent)) + (show (when pretty + (propertize pretty 'face 'highlight)))) + (when show + (overlay-put emmet-preview-output 'after-string + (concat show "\n"))))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Implementation of "Go to Edit Point" functionality ;; +;; http://docs.emmet.io/actions/go-to-edit-point/ ;; +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defun emmet-go-to-edit-point (count &optional only-before-closed-tag) + (let* + ((between-tags + (if only-before-closed-tag "\\(><\\)/" "\\(><\\)")) + (indented-line "\\(^[[:blank:]]+$\\)") + (between-quotes + (if emmet-move-cursor-between-quotes "\\(=\\(\"\\|'\\)\\{2\\}\\)" nil)) + (whole-regex + (mapconcat 'identity + (delq nil + (list between-tags indented-line between-quotes)) + "\\|")) + (edit-point (format "\\(%s\\)" whole-regex))) + (if (> count 0) + (progn + (forward-char) + (let + ((search-result (re-search-forward edit-point nil t count))) + (if search-result + (progn + (cond + ((match-string 2) (goto-char (- (match-end 2) 1))) + ((match-string 3) (end-of-line)) + ((match-string 4) (backward-char))) + (point)) + (backward-char)))) + (progn + (backward-char) + (let + ((search-result (re-search-backward edit-point nil t (- count)))) + (if search-result + (progn + (cond + ((match-string 2) (goto-char (- (match-end 2) 1))) + ((match-string 3) (end-of-line)) + ((match-string 4) (forward-char 2))) + (point)) + (forward-char))))))) + +(defcustom emmet-postwrap-goto-edit-point nil + "Goto first edit point after wrapping markup?" + :type 'boolean + :group 'emmet) + +;;;###autoload +(defun emmet-wrap-with-markup (wrap-with) + "Wrap region with markup." + (interactive "sExpression to wrap with: ") + (let* ((multi (string-match "\\*$" wrap-with)) + (txt (buffer-substring-no-properties (region-beginning) (region-end))) + (to-wrap (if multi + (split-string txt "\n") + (list txt))) + (initial-elements (replace-regexp-in-string + "\\(.*\\(\\+\\|>\\)\\)?[^>*]+\\*?[[:digit:]]*$" + "\\1" wrap-with t)) + (terminal-element (replace-regexp-in-string + "\\(.*>\\)?\\([^>*]+\\)\\(\\*[[:digit:]]+$\\)?\\*?$" + "\\2" wrap-with t)) + (multiplier-expr (replace-regexp-in-string + "\\(.*>\\)?\\([^>*]+\\)\\(\\*[[:digit:]]+$\\)?\\*?$" + "\\3" wrap-with t)) + (expr (concat + initial-elements + (mapconcat (lambda (el) + (concat terminal-element + "{!!!" + (secure-hash 'sha1 el) + "!!!}" + multiplier-expr)) + to-wrap + "+"))) + (markup + (cl-reduce + (lambda (result text) + (replace-regexp-in-string + (concat "!!!" (secure-hash 'sha1 text) "!!!") + text + result t t)) + to-wrap + :initial-value (emmet-transform expr)))) + (when markup + (delete-region (region-beginning) (region-end)) + (insert markup) + (indent-region (region-beginning) (region-end)) + (if emmet-postwrap-goto-edit-point + (let ((end (region-end))) + (goto-char (region-beginning)) + (unless (ignore-errors (progn (emmet-next-edit-point 1) t)) + (goto-char end))) + )))) + +;;;###autoload +(defun emmet-next-edit-point (count) + (interactive "^p") + (unless (or emmet-use-css-transform (emmet-go-to-edit-point count)) + (error "Last edit point reached."))) + +;;;###autoload +(defun emmet-prev-edit-point (count) + (interactive "^p") + (unless (or emmet-use-css-transform (emmet-go-to-edit-point (- count))) + (error "First edit point reached."))) + +(provide 'emmet-mode) +;; src/snippets.el +;; This file is generated from conf/snippets.json +;; Don't edit. +(emmet-defparameter emmet-snippets +(let ((tbl (make-hash-table :test 'equal))) +(puthash "css" (let ((tbl (make-hash-table :test 'equal))) +(puthash "snippets" (let ((tbl (make-hash-table :test 'equal))) +(puthash "!" "!important" tbl) +(puthash "@f" "@font-face {\n\tfont-family:|;\n\tsrc:url(|);\n}" tbl) +(puthash "@f+" "@font-face {\n\tfont-family: '${1:FontName}';\n\tsrc: url('${2:FileName}.eot');\n\tsrc: url('${2:FileName}.eot?#iefix') format('embedded-opentype'),\n\t\t url('${2:FileName}.woff') format('woff'),\n\t\t url('${2:FileName}.ttf') format('truetype'),\n\t\t url('${2:FileName}.svg#${1:FontName}') format('svg');\n\tfont-style: ${3:normal};\n\tfont-weight: ${4:normal};\n}" tbl) +(puthash "@i" "@import url(|);" tbl) +(puthash "@import" "@import url(|);" tbl) +(puthash "@kf" "@-webkit-keyframes ${1:identifier} {\n\t${2:from} { ${3} }${6}\n\t${4:to} { ${5} }\n}\n@-o-keyframes ${1:identifier} {\n\t${2:from} { ${3} }${6}\n\t${4:to} { ${5} }\n}\n@-moz-keyframes ${1:identifier} {\n\t${2:from} { ${3} }${6}\n\t${4:to} { ${5} }\n}\n@keyframes ${1:identifier} {\n\t${2:from} { ${3} }${6}\n\t${4:to} { ${5} }\n}" tbl) +(puthash "@m" "@media ${1:screen} {\n\t|\n}" tbl) +(puthash "@media" "@media ${1:screen} {\n\t|\n}" tbl) +(puthash "ac" "align-content:|;" tbl) +(puthash "ac:c" "align-content:center;" tbl) +(puthash "ac:fe" "align-content:flex-end;" tbl) +(puthash "ac:fs" "align-content:flex-start;" tbl) +(puthash "ac:s" "align-content:stretch;" tbl) +(puthash "ac:sa" "align-content:space-around;" tbl) +(puthash "ac:sb" "align-content:space-between;" tbl) +(puthash "ai" "align-items:|;" tbl) +(puthash "ai:b" "align-items:baseline;" tbl) +(puthash "ai:c" "align-items:center;" tbl) +(puthash "ai:fe" "align-items:flex-end;" tbl) +(puthash "ai:fs" "align-items:flex-start;" tbl) +(puthash "ai:s" "align-items:stretch;" tbl) +(puthash "anim" "animation:|;" tbl) +(puthash "anim-" "animation:${1:name} ${2:duration} ${3:timing-function} ${4:delay} ${5:iteration-count} ${6:direction} ${7:fill-mode};" tbl) +(puthash "animdel" "animation-delay:${1:time};" tbl) +(puthash "animdir" "animation-direction:${1:normal};" tbl) +(puthash "animdir:a" "animation-direction:alternate;" tbl) +(puthash "animdir:ar" "animation-direction:alternate-reverse;" tbl) +(puthash "animdir:n" "animation-direction:normal;" tbl) +(puthash "animdir:r" "animation-direction:reverse;" tbl) +(puthash "animdur" "animation-duration:${1:0}s;" tbl) +(puthash "animfm" "animation-fill-mode:${1:both};" tbl) +(puthash "animfm:b" "animation-fill-mode:backwards;" tbl) +(puthash "animfm:bh" "animation-fill-mode:both;" tbl) +(puthash "animfm:bt" "animation-fill-mode:both;" tbl) +(puthash "animfm:f" "animation-fill-mode:forwards;" tbl) +(puthash "animic" "animation-iteration-count:${1:1};" tbl) +(puthash "animic:i" "animation-iteration-count:infinite;" tbl) +(puthash "animn" "animation-name:${1:none};" tbl) +(puthash "animps" "animation-play-state:${1:running};" tbl) +(puthash "animps:p" "animation-play-state:paused;" tbl) +(puthash "animps:r" "animation-play-state:running;" tbl) +(puthash "animtf" "animation-timing-function:${1:linear};" tbl) +(puthash "animtf:cb" "animation-timing-function:cubic-bezier(${1:0.1}, ${2:0.7}, ${3:1.0}, ${3:0.1});" tbl) +(puthash "animtf:e" "animation-timing-function:ease;" tbl) +(puthash "animtf:ei" "animation-timing-function:ease-in;" tbl) +(puthash "animtf:eio" "animation-timing-function:ease-in-out;" tbl) +(puthash "animtf:eo" "animation-timing-function:ease-out;" tbl) +(puthash "animtf:l" "animation-timing-function:linear;" tbl) +(puthash "ap" "appearance:${none};" tbl) +(puthash "as" "align-self:|;" tbl) +(puthash "as:a" "align-self:auto;" tbl) +(puthash "as:b" "align-self:baseline;" tbl) +(puthash "as:c" "align-self:center;" tbl) +(puthash "as:fe" "align-self:flex-end;" tbl) +(puthash "as:fs" "align-self:flex-start;" tbl) +(puthash "as:s" "align-self:stretch;" tbl) +(puthash "b" "bottom:|;" tbl) +(puthash "b:a" "bottom:auto;" tbl) +(puthash "bb" "border-bottom:|;" tbl) +(puthash "bd" "border:|;" tbl) +(puthash "bd+" "border:${1:1px} ${2:solid} ${3:#000};" tbl) +(puthash "bd:n" "border:none;" tbl) +(puthash "bdb" "border-bottom:|;" tbl) +(puthash "bdb+" "border-bottom:${1:1px} ${2:solid} ${3:#000};" tbl) +(puthash "bdb:n" "border-bottom:none;" tbl) +(puthash "bdbc" "border-bottom-color:${1:#000};" tbl) +(puthash "bdbc:t" "border-bottom-color:transparent;" tbl) +(puthash "bdbi" "border-bottom-image:url(|);" tbl) +(puthash "bdbi:n" "border-bottom-image:none;" tbl) +(puthash "bdbk" "border-break:${1:close};" tbl) +(puthash "bdbk:c" "border-break:close;" tbl) +(puthash "bdbli" "border-bottom-left-image:url(|);" tbl) +(puthash "bdbli:c" "border-bottom-left-image:continue;" tbl) +(puthash "bdbli:n" "border-bottom-left-image:none;" tbl) +(puthash "bdblrs" "border-bottom-left-radius:|;" tbl) +(puthash "bdbri" "border-bottom-right-image:url(|);" tbl) +(puthash "bdbri:c" "border-bottom-right-image:continue;" tbl) +(puthash "bdbri:n" "border-bottom-right-image:none;" tbl) +(puthash "bdbrrs" "border-bottom-right-radius:|;" tbl) +(puthash "bdbs" "border-bottom-style:|;" tbl) +(puthash "bdbs:n" "border-bottom-style:none;" tbl) +(puthash "bdbw" "border-bottom-width:|;" tbl) +(puthash "bdc" "border-color:${1:#000};" tbl) +(puthash "bdc:t" "border-color:transparent;" tbl) +(puthash "bdci" "border-corner-image:url(|);" tbl) +(puthash "bdci:c" "border-corner-image:continue;" tbl) +(puthash "bdci:n" "border-corner-image:none;" tbl) +(puthash "bdcl" "border-collapse:|;" tbl) +(puthash "bdcl:c" "border-collapse:collapse;" tbl) +(puthash "bdcl:s" "border-collapse:separate;" tbl) +(puthash "bdf" "border-fit:${1:repeat};" tbl) +(puthash "bdf:c" "border-fit:clip;" tbl) +(puthash "bdf:of" "border-fit:overflow;" tbl) +(puthash "bdf:ow" "border-fit:overwrite;" tbl) +(puthash "bdf:r" "border-fit:repeat;" tbl) +(puthash "bdf:sc" "border-fit:scale;" tbl) +(puthash "bdf:sp" "border-fit:space;" tbl) +(puthash "bdf:st" "border-fit:stretch;" tbl) +(puthash "bdi" "border-image:url(|);" tbl) +(puthash "bdi:n" "border-image:none;" tbl) +(puthash "bdl" "border-left:|;" tbl) +(puthash "bdl+" "border-left:${1:1px} ${2:solid} ${3:#000};" tbl) +(puthash "bdl:n" "border-left:none;" tbl) +(puthash "bdlc" "border-left-color:${1:#000};" tbl) +(puthash "bdlc:t" "border-left-color:transparent;" tbl) +(puthash "bdlen" "border-length:|;" tbl) +(puthash "bdlen:a" "border-length:auto;" tbl) +(puthash "bdli" "border-left-image:url(|);" tbl) +(puthash "bdli:n" "border-left-image:none;" tbl) +(puthash "bdls" "border-left-style:|;" tbl) +(puthash "bdls:n" "border-left-style:none;" tbl) +(puthash "bdlw" "border-left-width:|;" tbl) +(puthash "bdr" "border-right:|;" tbl) +(puthash "bdr+" "border-right:${1:1px} ${2:solid} ${3:#000};" tbl) +(puthash "bdr:n" "border-right:none;" tbl) +(puthash "bdrc" "border-right-color:${1:#000};" tbl) +(puthash "bdrc:t" "border-right-color:transparent;" tbl) +(puthash "bdri" "border-right-image:url(|);" tbl) +(puthash "bdri:n" "border-right-image:none;" tbl) +(puthash "bdrs" "border-radius:|;" tbl) +(puthash "bdrst" "border-right-style:|;" tbl) +(puthash "bdrst:n" "border-right-style:none;" tbl) +(puthash "bdrw" "border-right-width:|;" tbl) +(puthash "bds" "border-style:|;" tbl) +(puthash "bds:db" "border-style:double;" tbl) +(puthash "bds:ds" "border-style:dashed;" tbl) +(puthash "bds:dt" "border-style:dotted;" tbl) +(puthash "bds:dtds" "border-style:dot-dash;" tbl) +(puthash "bds:dtdtds" "border-style:dot-dot-dash;" tbl) +(puthash "bds:g" "border-style:groove;" tbl) +(puthash "bds:h" "border-style:hidden;" tbl) +(puthash "bds:i" "border-style:inset;" tbl) +(puthash "bds:n" "border-style:none;" tbl) +(puthash "bds:o" "border-style:outset;" tbl) +(puthash "bds:r" "border-style:ridge;" tbl) +(puthash "bds:s" "border-style:solid;" tbl) +(puthash "bds:w" "border-style:wave;" tbl) +(puthash "bdsp" "border-spacing:|;" tbl) +(puthash "bdt" "border-top:|;" tbl) +(puthash "bdt+" "border-top:${1:1px} ${2:solid} ${3:#000};" tbl) +(puthash "bdt:n" "border-top:none;" tbl) +(puthash "bdtc" "border-top-color:${1:#000};" tbl) +(puthash "bdtc:t" "border-top-color:transparent;" tbl) +(puthash "bdti" "border-top-image:url(|);" tbl) +(puthash "bdti:n" "border-top-image:none;" tbl) +(puthash "bdtli" "border-top-left-image:url(|);" tbl) +(puthash "bdtli:c" "border-top-left-image:continue;" tbl) +(puthash "bdtli:n" "border-top-left-image:none;" tbl) +(puthash "bdtlrs" "border-top-left-radius:|;" tbl) +(puthash "bdtri" "border-top-right-image:url(|);" tbl) +(puthash "bdtri:c" "border-top-right-image:continue;" tbl) +(puthash "bdtri:n" "border-top-right-image:none;" tbl) +(puthash "bdtrrs" "border-top-right-radius:|;" tbl) +(puthash "bdts" "border-top-style:|;" tbl) +(puthash "bdts:n" "border-top-style:none;" tbl) +(puthash "bdtw" "border-top-width:|;" tbl) +(puthash "bdw" "border-width:|;" tbl) +(puthash "bfv" "backface-visibility:|;" tbl) +(puthash "bfv:h" "backface-visibility:hidden;" tbl) +(puthash "bfv:v" "backface-visibility:visible;" tbl) +(puthash "bg" "background:#${1:000};" tbl) +(puthash "bg+" "background:${1:#fff} url(${2}) ${3:0} ${4:0} ${5:no-repeat};" tbl) +(puthash "bg:ie" "filter:progid:DXImageTransform.Microsoft.AlphaImageLoader(src='${1:x}.png',sizingMethod='${2:crop}');" tbl) +(puthash "bg:n" "background:none;" tbl) +(puthash "bga" "background-attachment:|;" tbl) +(puthash "bga:f" "background-attachment:fixed;" tbl) +(puthash "bga:s" "background-attachment:scroll;" tbl) +(puthash "bgbk" "background-break:|;" tbl) +(puthash "bgbk:bb" "background-break:bounding-box;" tbl) +(puthash "bgbk:c" "background-break:continuous;" tbl) +(puthash "bgbk:eb" "background-break:each-box;" tbl) +(puthash "bgc" "background-color:${1:fff};" tbl) +(puthash "bgc:t" "background-color:transparent;" tbl) +(puthash "bgcp" "background-clip:${1:padding-box};" tbl) +(puthash "bgcp:bb" "background-clip:border-box;" tbl) +(puthash "bgcp:cb" "background-clip:content-box;" tbl) +(puthash "bgcp:nc" "background-clip:no-clip;" tbl) +(puthash "bgcp:pb" "background-clip:padding-box;" tbl) +(puthash "bgi" "background-image:url(|);" tbl) +(puthash "bgi:n" "background-image:none;" tbl) +(puthash "bgo" "background-origin:|;" tbl) +(puthash "bgo:bb" "background-origin:border-box;" tbl) +(puthash "bgo:cb" "background-origin:content-box;" tbl) +(puthash "bgo:pb" "background-origin:padding-box;" tbl) +(puthash "bgp" "background-position:${1:0} ${2:0};" tbl) +(puthash "bgpx" "background-position-x:|;" tbl) +(puthash "bgpy" "background-position-y:|;" tbl) +(puthash "bgr" "background-repeat:|;" tbl) +(puthash "bgr:n" "background-repeat:no-repeat;" tbl) +(puthash "bgr:rd" "background-repeat:round;" tbl) +(puthash "bgr:sp" "background-repeat:space;" tbl) +(puthash "bgr:x" "background-repeat:repeat-x;" tbl) +(puthash "bgr:y" "background-repeat:repeat-y;" tbl) +(puthash "bgsz" "background-size:|;" tbl) +(puthash "bgsz:a" "background-size:auto;" tbl) +(puthash "bgsz:ct" "background-size:contain;" tbl) +(puthash "bgsz:cv" "background-size:cover;" tbl) +(puthash "bl" "border-left:|;" tbl) +(puthash "br" "border-right:|;" tbl) +(puthash "bt" "border-top:|;" tbl) +(puthash "bxsh" "box-shadow:${1:inset }${2:hoff} ${3:voff} ${4:blur} ${5:color};" tbl) +(puthash "bxsh:n" "box-shadow:none;" tbl) +(puthash "bxsh:r" "box-shadow:${1:inset }${2:hoff} ${3:voff} ${4:blur} ${5:spread }rgb(${6:0}, ${7:0}, ${8:0});" tbl) +(puthash "bxsh:ra" "box-shadow:${1:inset }${2:h} ${3:v} ${4:blur} ${5:spread }rgba(${6:0}, ${7:0}, ${8:0}, .${9:5});" tbl) +(puthash "bxz" "box-sizing:${1:border-box};" tbl) +(puthash "bxz:bb" "box-sizing:border-box;" tbl) +(puthash "bxz:cb" "box-sizing:content-box;" tbl) +(puthash "c" "color:${1:#000};" tbl) +(puthash "c:r" "color:rgb(${1:0}, ${2:0}, ${3:0});" tbl) +(puthash "c:ra" "color:rgba(${1:0}, ${2:0}, ${3:0}, .${4:5});" tbl) +(puthash "cl" "clear:${1:both};" tbl) +(puthash "cl:b" "clear:both;" tbl) +(puthash "cl:l" "clear:left;" tbl) +(puthash "cl:n" "clear:none;" tbl) +(puthash "cl:r" "clear:right;" tbl) +(puthash "cm" "/* |${child} */" tbl) +(puthash "cnt" "content:'|';" tbl) +(puthash "cnt:a" "content:attr(|);" tbl) +(puthash "cnt:c" "content:counter(|);" tbl) +(puthash "cnt:cq" "content:close-quote;" tbl) +(puthash "cnt:cs" "content:counters(|);" tbl) +(puthash "cnt:n" "content:normal;" tbl) +(puthash "cnt:ncq" "content:no-close-quote;" tbl) +(puthash "cnt:noq" "content:no-open-quote;" tbl) +(puthash "cnt:oq" "content:open-quote;" tbl) +(puthash "coi" "counter-increment:|;" tbl) +(puthash "colm" "columns:|;" tbl) +(puthash "colmc" "column-count:|;" tbl) +(puthash "colmf" "column-fill:|;" tbl) +(puthash "colmg" "column-gap:|;" tbl) +(puthash "colmr" "column-rule:|;" tbl) +(puthash "colmrc" "column-rule-color:|;" tbl) +(puthash "colmrs" "column-rule-style:|;" tbl) +(puthash "colmrw" "column-rule-width:|;" tbl) +(puthash "colms" "column-span:|;" tbl) +(puthash "colmw" "column-width:|;" tbl) +(puthash "cor" "counter-reset:|;" tbl) +(puthash "cp" "clip:|;" tbl) +(puthash "cp:a" "clip:auto;" tbl) +(puthash "cp:r" "clip:rect(${1:top} ${2:right} ${3:bottom} ${4:left});" tbl) +(puthash "cps" "caption-side:|;" tbl) +(puthash "cps:b" "caption-side:bottom;" tbl) +(puthash "cps:t" "caption-side:top;" tbl) +(puthash "ct" "content:|;" tbl) +(puthash "ct:a" "content:attr(|);" tbl) +(puthash "ct:c" "content:counter(|);" tbl) +(puthash "ct:cq" "content:close-quote;" tbl) +(puthash "ct:cs" "content:counters(|);" tbl) +(puthash "ct:n" "content:normal;" tbl) +(puthash "ct:ncq" "content:no-close-quote;" tbl) +(puthash "ct:noq" "content:no-open-quote;" tbl) +(puthash "ct:oq" "content:open-quote;" tbl) +(puthash "cur" "cursor:${pointer};" tbl) +(puthash "cur:a" "cursor:auto;" tbl) +(puthash "cur:c" "cursor:crosshair;" tbl) +(puthash "cur:d" "cursor:default;" tbl) +(puthash "cur:ha" "cursor:hand;" tbl) +(puthash "cur:he" "cursor:help;" tbl) +(puthash "cur:m" "cursor:move;" tbl) +(puthash "cur:p" "cursor:pointer;" tbl) +(puthash "cur:t" "cursor:text;" tbl) +(puthash "d" "display:${1:block};" tbl) +(puthash "d:b" "display:block;" tbl) +(puthash "d:cp" "display:compact;" tbl) +(puthash "d:f" "display:flex;" tbl) +(puthash "d:i" "display:inline;" tbl) +(puthash "d:ib" "display:inline-block;" tbl) +(puthash "d:ib+" "display: inline-block;\n*display: inline;\n*zoom: 1;" tbl) +(puthash "d:if" "display:inline-flex;" tbl) +(puthash "d:itb" "display:inline-table;" tbl) +(puthash "d:li" "display:list-item;" tbl) +(puthash "d:n" "display:none;" tbl) +(puthash "d:rb" "display:ruby;" tbl) +(puthash "d:rbb" "display:ruby-base;" tbl) +(puthash "d:rbbg" "display:ruby-base-group;" tbl) +(puthash "d:rbt" "display:ruby-text;" tbl) +(puthash "d:rbtg" "display:ruby-text-group;" tbl) +(puthash "d:ri" "display:run-in;" tbl) +(puthash "d:tb" "display:table;" tbl) +(puthash "d:tbc" "display:table-cell;" tbl) +(puthash "d:tbcl" "display:table-column;" tbl) +(puthash "d:tbclg" "display:table-column-group;" tbl) +(puthash "d:tbcp" "display:table-caption;" tbl) +(puthash "d:tbfg" "display:table-footer-group;" tbl) +(puthash "d:tbhg" "display:table-header-group;" tbl) +(puthash "d:tbr" "display:table-row;" tbl) +(puthash "d:tbrg" "display:table-row-group;" tbl) +(puthash "ec" "empty-cells:|;" tbl) +(puthash "ec:h" "empty-cells:hide;" tbl) +(puthash "ec:s" "empty-cells:show;" tbl) +(puthash "f" "font:|;" tbl) +(puthash "f+" "font:${1:1em} ${2:Arial,sans-serif};" tbl) +(puthash "fef" "font-effect:|;" tbl) +(puthash "fef:eb" "font-effect:emboss;" tbl) +(puthash "fef:eg" "font-effect:engrave;" tbl) +(puthash "fef:n" "font-effect:none;" tbl) +(puthash "fef:o" "font-effect:outline;" tbl) +(puthash "fem" "font-emphasize:|;" tbl) +(puthash "femp" "font-emphasize-position:|;" tbl) +(puthash "femp:a" "font-emphasize-position:after;" tbl) +(puthash "femp:b" "font-emphasize-position:before;" tbl) +(puthash "fems" "font-emphasize-style:|;" tbl) +(puthash "fems:ac" "font-emphasize-style:accent;" tbl) +(puthash "fems:c" "font-emphasize-style:circle;" tbl) +(puthash "fems:ds" "font-emphasize-style:disc;" tbl) +(puthash "fems:dt" "font-emphasize-style:dot;" tbl) +(puthash "fems:n" "font-emphasize-style:none;" tbl) +(puthash "ff" "font-family:|;" tbl) +(puthash "ff:a" "font-family: Arial, \"Helvetica Neue\", Helvetica, sans-serif;" tbl) +(puthash "ff:c" "font-family:cursive;" tbl) +(puthash "ff:f" "font-family:fantasy;" tbl) +(puthash "ff:m" "font-family:monospace;" tbl) +(puthash "ff:s" "font-family:serif;" tbl) +(puthash "ff:ss" "font-family:sans-serif;" tbl) +(puthash "ff:t" "font-family: \"Times New Roman\", Times, Baskerville, Georgia, serif;" tbl) +(puthash "ff:v" "font-family: Verdana, Geneva, sans-serif;" tbl) +(puthash "fl" "float:${1:left};" tbl) +(puthash "fl:l" "float:left;" tbl) +(puthash "fl:n" "float:none;" tbl) +(puthash "fl:r" "float:right;" tbl) +(puthash "fs" "font-style:italic;" tbl) +(puthash "fs:i" "font-style:italic;" tbl) +(puthash "fs:n" "font-style:normal;" tbl) +(puthash "fs:o" "font-style:oblique;" tbl) +(puthash "fsm" "font-smooth:|;" tbl) +(puthash "fsm:a" "font-smooth:auto;" tbl) +(puthash "fsm:aw" "font-smooth:always;" tbl) +(puthash "fsm:n" "font-smooth:never;" tbl) +(puthash "fst" "font-stretch:|;" tbl) +(puthash "fst:c" "font-stretch:condensed;" tbl) +(puthash "fst:e" "font-stretch:expanded;" tbl) +(puthash "fst:ec" "font-stretch:extra-condensed;" tbl) +(puthash "fst:ee" "font-stretch:extra-expanded;" tbl) +(puthash "fst:n" "font-stretch:normal;" tbl) +(puthash "fst:sc" "font-stretch:semi-condensed;" tbl) +(puthash "fst:se" "font-stretch:semi-expanded;" tbl) +(puthash "fst:uc" "font-stretch:ultra-condensed;" tbl) +(puthash "fst:ue" "font-stretch:ultra-expanded;" tbl) +(puthash "fv" "font-variant:|;" tbl) +(puthash "fv:n" "font-variant:normal;" tbl) +(puthash "fv:sc" "font-variant:small-caps;" tbl) +(puthash "fw" "font-weight:|;" tbl) +(puthash "fw:b" "font-weight:bold;" tbl) +(puthash "fw:br" "font-weight:bolder;" tbl) +(puthash "fw:lr" "font-weight:lighter;" tbl) +(puthash "fw:n" "font-weight:normal;" tbl) +(puthash "fx" "flex:|;" tbl) +(puthash "fxb" "flex-basis:|;" tbl) +(puthash "fxd" "flex-direction:|;" tbl) +(puthash "fxd:c" "flex-direction:column;" tbl) +(puthash "fxd:cr" "flex-direction:column-reverse;" tbl) +(puthash "fxd:r" "flex-direction:row;" tbl) +(puthash "fxd:rr" "flex-direction:row-reverse;" tbl) +(puthash "fxf" "flex-flow:|;" tbl) +(puthash "fxg" "flex-grow:|;" tbl) +(puthash "fxsh" "flex-shrink:|;" tbl) +(puthash "fxw" "flex-wrap: |;" tbl) +(puthash "fxw:n" "flex-wrap:nowrap;" tbl) +(puthash "fxw:w" "flex-wrap:wrap;" tbl) +(puthash "fxw:wr" "flex-wrap:wrap-reverse;" tbl) +(puthash "fz" "font-size:|;" tbl) +(puthash "fza" "font-size-adjust:|;" tbl) +(puthash "fza:n" "font-size-adjust:none;" tbl) +(puthash "h" "height:|;" tbl) +(puthash "h:a" "height:auto;" tbl) +(puthash "jc" "justify-content:|;" tbl) +(puthash "jc:c" "justify-content:center;" tbl) +(puthash "jc:fe" "justify-content:flex-end;" tbl) +(puthash "jc:fs" "justify-content:flex-start;" tbl) +(puthash "jc:sa" "justify-content:space-around;" tbl) +(puthash "jc:sb" "justify-content:space-between;" tbl) +(puthash "l" "left:|;" tbl) +(puthash "l:a" "left:auto;" tbl) +(puthash "lh" "line-height:|;" tbl) +(puthash "lis" "list-style:|;" tbl) +(puthash "lis:n" "list-style:none;" tbl) +(puthash "lisi" "list-style-image:|;" tbl) +(puthash "lisi:n" "list-style-image:none;" tbl) +(puthash "lisp" "list-style-position:|;" tbl) +(puthash "lisp:i" "list-style-position:inside;" tbl) +(puthash "lisp:o" "list-style-position:outside;" tbl) +(puthash "list" "list-style-type:|;" tbl) +(puthash "list:c" "list-style-type:circle;" tbl) +(puthash "list:d" "list-style-type:disc;" tbl) +(puthash "list:dc" "list-style-type:decimal;" tbl) +(puthash "list:dclz" "list-style-type:decimal-leading-zero;" tbl) +(puthash "list:lr" "list-style-type:lower-roman;" tbl) +(puthash "list:n" "list-style-type:none;" tbl) +(puthash "list:s" "list-style-type:square;" tbl) +(puthash "list:ur" "list-style-type:upper-roman;" tbl) +(puthash "lts" "letter-spacing:|;" tbl) +(puthash "lts-n" "letter-spacing:normal;" tbl) +(puthash "m" "margin:|;" tbl) +(puthash "m:a" "margin:auto;" tbl) +(puthash "mah" "max-height:|;" tbl) +(puthash "mah:n" "max-height:none;" tbl) +(puthash "mar" "max-resolution:${1:res};" tbl) +(puthash "maw" "max-width:|;" tbl) +(puthash "maw:n" "max-width:none;" tbl) +(puthash "mb" "margin-bottom:|;" tbl) +(puthash "mb:a" "margin-bottom:auto;" tbl) +(puthash "mih" "min-height:|;" tbl) +(puthash "mir" "min-resolution:${1:res};" tbl) +(puthash "miw" "min-width:|;" tbl) +(puthash "ml" "margin-left:|;" tbl) +(puthash "ml:a" "margin-left:auto;" tbl) +(puthash "mr" "margin-right:|;" tbl) +(puthash "mr:a" "margin-right:auto;" tbl) +(puthash "mt" "margin-top:|;" tbl) +(puthash "mt:a" "margin-top:auto;" tbl) +(puthash "ol" "outline:|;" tbl) +(puthash "ol:n" "outline:none;" tbl) +(puthash "olc" "outline-color:${1:#000};" tbl) +(puthash "olc:i" "outline-color:invert;" tbl) +(puthash "olo" "outline-offset:|;" tbl) +(puthash "ols" "outline-style:|;" tbl) +(puthash "ols:db" "outline-style:double;" tbl) +(puthash "ols:ds" "outline-style:dashed;" tbl) +(puthash "ols:dt" "outline-style:dotted;" tbl) +(puthash "ols:g" "outline-style:groove;" tbl) +(puthash "ols:i" "outline-style:inset;" tbl) +(puthash "ols:n" "outline-style:none;" tbl) +(puthash "ols:o" "outline-style:outset;" tbl) +(puthash "ols:r" "outline-style:ridge;" tbl) +(puthash "ols:s" "outline-style:solid;" tbl) +(puthash "olw" "outline-width:|;" tbl) +(puthash "olw:m" "outline-width:medium;" tbl) +(puthash "olw:tc" "outline-width:thick;" tbl) +(puthash "olw:tn" "outline-width:thin;" tbl) +(puthash "op" "opacity:|;" tbl) +(puthash "op+" "opacity: $1;\nfilter: alpha(opacity=$2);" tbl) +(puthash "op:ie" "filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=100);" tbl) +(puthash "op:ms" "-ms-filter:'progid:DXImageTransform.Microsoft.Alpha(Opacity=100)';" tbl) +(puthash "ord" "order:|;" tbl) +(puthash "ori" "orientation:|;" tbl) +(puthash "ori:l" "orientation:landscape;" tbl) +(puthash "ori:p" "orientation:portrait;" tbl) +(puthash "orp" "orphans:|;" tbl) +(puthash "ov" "overflow:${1:hidden};" tbl) +(puthash "ov:a" "overflow:auto;" tbl) +(puthash "ov:h" "overflow:hidden;" tbl) +(puthash "ov:s" "overflow:scroll;" tbl) +(puthash "ov:v" "overflow:visible;" tbl) +(puthash "ovs" "overflow-style:${1:scrollbar};" tbl) +(puthash "ovs:a" "overflow-style:auto;" tbl) +(puthash "ovs:m" "overflow-style:move;" tbl) +(puthash "ovs:mq" "overflow-style:marquee;" tbl) +(puthash "ovs:p" "overflow-style:panner;" tbl) +(puthash "ovs:s" "overflow-style:scrollbar;" tbl) +(puthash "ovx" "overflow-x:${1:hidden};" tbl) +(puthash "ovx:a" "overflow-x:auto;" tbl) +(puthash "ovx:h" "overflow-x:hidden;" tbl) +(puthash "ovx:s" "overflow-x:scroll;" tbl) +(puthash "ovx:v" "overflow-x:visible;" tbl) +(puthash "ovy" "overflow-y:${1:hidden};" tbl) +(puthash "ovy:a" "overflow-y:auto;" tbl) +(puthash "ovy:h" "overflow-y:hidden;" tbl) +(puthash "ovy:s" "overflow-y:scroll;" tbl) +(puthash "ovy:v" "overflow-y:visible;" tbl) +(puthash "p" "padding:|;" tbl) +(puthash "pb" "padding-bottom:|;" tbl) +(puthash "pgba" "page-break-after:|;" tbl) +(puthash "pgba:al" "page-break-after:always;" tbl) +(puthash "pgba:au" "page-break-after:auto;" tbl) +(puthash "pgba:l" "page-break-after:left;" tbl) +(puthash "pgba:r" "page-break-after:right;" tbl) +(puthash "pgbb" "page-break-before:|;" tbl) +(puthash "pgbb:al" "page-break-before:always;" tbl) +(puthash "pgbb:au" "page-break-before:auto;" tbl) +(puthash "pgbb:l" "page-break-before:left;" tbl) +(puthash "pgbb:r" "page-break-before:right;" tbl) +(puthash "pgbi" "page-break-inside:|;" tbl) +(puthash "pgbi:au" "page-break-inside:auto;" tbl) +(puthash "pgbi:av" "page-break-inside:avoid;" tbl) +(puthash "pl" "padding-left:|;" tbl) +(puthash "pos" "position:${1:relative};" tbl) +(puthash "pos:a" "position:absolute;" tbl) +(puthash "pos:f" "position:fixed;" tbl) +(puthash "pos:r" "position:relative;" tbl) +(puthash "pos:s" "position:static;" tbl) +(puthash "pr" "padding-right:|;" tbl) +(puthash "pt" "padding-top:|;" tbl) +(puthash "q" "quotes:|;" tbl) +(puthash "q:en" "quotes:'\\201C' '\\201D' '\\2018' '\\2019';" tbl) +(puthash "q:n" "quotes:none;" tbl) +(puthash "q:ru" "quotes:'\\00AB' '\\00BB' '\\201E' '\\201C';" tbl) +(puthash "r" "right:|;" tbl) +(puthash "r:a" "right:auto;" tbl) +(puthash "rsz" "resize:|;" tbl) +(puthash "rsz:b" "resize:both;" tbl) +(puthash "rsz:h" "resize:horizontal;" tbl) +(puthash "rsz:n" "resize:none;" tbl) +(puthash "rsz:v" "resize:vertical;" tbl) +(puthash "t" "top:|;" tbl) +(puthash "t:a" "top:auto;" tbl) +(puthash "ta" "text-align:${1:left};" tbl) +(puthash "ta-lst" "text-align-last:|;" tbl) +(puthash "ta:c" "text-align:center;" tbl) +(puthash "ta:j" "text-align:justify;" tbl) +(puthash "ta:l" "text-align:left;" tbl) +(puthash "ta:r" "text-align:right;" tbl) +(puthash "tal:a" "text-align-last:auto;" tbl) +(puthash "tal:c" "text-align-last:center;" tbl) +(puthash "tal:l" "text-align-last:left;" tbl) +(puthash "tal:r" "text-align-last:right;" tbl) +(puthash "tbl" "table-layout:|;" tbl) +(puthash "tbl:a" "table-layout:auto;" tbl) +(puthash "tbl:f" "table-layout:fixed;" tbl) +(puthash "td" "text-decoration:${1:none};" tbl) +(puthash "td:l" "text-decoration:line-through;" tbl) +(puthash "td:n" "text-decoration:none;" tbl) +(puthash "td:o" "text-decoration:overline;" tbl) +(puthash "td:u" "text-decoration:underline;" tbl) +(puthash "te" "text-emphasis:|;" tbl) +(puthash "te:a" "text-emphasis:after;" tbl) +(puthash "te:ac" "text-emphasis:accent;" tbl) +(puthash "te:b" "text-emphasis:before;" tbl) +(puthash "te:c" "text-emphasis:circle;" tbl) +(puthash "te:ds" "text-emphasis:disc;" tbl) +(puthash "te:dt" "text-emphasis:dot;" tbl) +(puthash "te:n" "text-emphasis:none;" tbl) +(puthash "th" "text-height:|;" tbl) +(puthash "th:a" "text-height:auto;" tbl) +(puthash "th:f" "text-height:font-size;" tbl) +(puthash "th:m" "text-height:max-size;" tbl) +(puthash "th:t" "text-height:text-size;" tbl) +(puthash "ti" "text-indent:|;" tbl) +(puthash "ti:-" "text-indent:-9999px;" tbl) +(puthash "tj" "text-justify:|;" tbl) +(puthash "tj:a" "text-justify:auto;" tbl) +(puthash "tj:d" "text-justify:distribute;" tbl) +(puthash "tj:ic" "text-justify:inter-cluster;" tbl) +(puthash "tj:ii" "text-justify:inter-ideograph;" tbl) +(puthash "tj:iw" "text-justify:inter-word;" tbl) +(puthash "tj:k" "text-justify:kashida;" tbl) +(puthash "tj:t" "text-justify:tibetan;" tbl) +(puthash "to" "text-outline:|;" tbl) +(puthash "to+" "text-outline:${1:0} ${2:0} ${3:#000};" tbl) +(puthash "to:n" "text-outline:none;" tbl) +(puthash "tov" "text-overflow:${ellipsis};" tbl) +(puthash "tov:c" "text-overflow:clip;" tbl) +(puthash "tov:e" "text-overflow:ellipsis;" tbl) +(puthash "tr" "text-replace:|;" tbl) +(puthash "tr:n" "text-replace:none;" tbl) +(puthash "trf" "transform:|;" tbl) +(puthash "trf:r" "transform: rotate(${1:angle});" tbl) +(puthash "trf:rx" "transform: rotateX(${1:angle});" tbl) +(puthash "trf:ry" "transform: rotateY(${1:angle});" tbl) +(puthash "trf:rz" "transform: rotateZ(${1:angle});" tbl) +(puthash "trf:sc" "transform: scale(${1:x}, ${2:y});" tbl) +(puthash "trf:sc3" "transform: scale3d(${1:x}, ${2:y}, ${3:z});" tbl) +(puthash "trf:scx" "transform: scaleX(${1:x});" tbl) +(puthash "trf:scy" "transform: scaleY(${1:y});" tbl) +(puthash "trf:scz" "transform: scaleZ(${1:z});" tbl) +(puthash "trf:skx" "transform: skewX(${1:angle});" tbl) +(puthash "trf:sky" "transform: skewY(${1:angle});" tbl) +(puthash "trf:t" "transform: translate(${1:x}, ${2:y});" tbl) +(puthash "trf:t3" "transform: translate3d(${1:tx}, ${2:ty}, ${3:tz});" tbl) +(puthash "trf:tx" "transform: translateX(${1:x});" tbl) +(puthash "trf:ty" "transform: translateY(${1:y});" tbl) +(puthash "trf:tz" "transform: translateZ(${1:z});" tbl) +(puthash "trfo" "transform-origin:|;" tbl) +(puthash "trfs" "transform-style:${1:preserve-3d};" tbl) +(puthash "trs" "transition:${1:prop} ${2:time};" tbl) +(puthash "trsde" "transition-delay:${1:time};" tbl) +(puthash "trsdu" "transition-duration:${1:time};" tbl) +(puthash "trsp" "transition-property:${1:prop};" tbl) +(puthash "trstf" "transition-timing-function:${1:tfunc};" tbl) +(puthash "tsh" "text-shadow:${1:hoff} ${2:voff} ${3:blur} ${4:#000};" tbl) +(puthash "tsh+" "text-shadow:${1:0} ${2:0} ${3:0} ${4:#000};" tbl) +(puthash "tsh:n" "text-shadow:none;" tbl) +(puthash "tsh:r" "text-shadow:${1:h} ${2:v} ${3:blur} rgb(${4:0}, ${5:0}, ${6:0});" tbl) +(puthash "tsh:ra" "text-shadow:${1:h} ${2:v} ${3:blur} rgba(${4:0}, ${5:0}, ${6:0}, .${7:5});" tbl) +(puthash "tt" "text-transform:${1:uppercase};" tbl) +(puthash "tt:c" "text-transform:capitalize;" tbl) +(puthash "tt:l" "text-transform:lowercase;" tbl) +(puthash "tt:n" "text-transform:none;" tbl) +(puthash "tt:u" "text-transform:uppercase;" tbl) +(puthash "tw" "text-wrap:|;" tbl) +(puthash "tw:n" "text-wrap:normal;" tbl) +(puthash "tw:no" "text-wrap:none;" tbl) +(puthash "tw:s" "text-wrap:suppress;" tbl) +(puthash "tw:u" "text-wrap:unrestricted;" tbl) +(puthash "us" "user-select:${none};" tbl) +(puthash "v" "visibility:${1:hidden};" tbl) +(puthash "v:c" "visibility:collapse;" tbl) +(puthash "v:h" "visibility:hidden;" tbl) +(puthash "v:v" "visibility:visible;" tbl) +(puthash "va" "vertical-align:${1:top};" tbl) +(puthash "va:b" "vertical-align:bottom;" tbl) +(puthash "va:bl" "vertical-align:baseline;" tbl) +(puthash "va:m" "vertical-align:middle;" tbl) +(puthash "va:sub" "vertical-align:sub;" tbl) +(puthash "va:sup" "vertical-align:super;" tbl) +(puthash "va:t" "vertical-align:top;" tbl) +(puthash "va:tb" "vertical-align:text-bottom;" tbl) +(puthash "va:tt" "vertical-align:text-top;" tbl) +(puthash "w" "width:|;" tbl) +(puthash "w:a" "width:auto;" tbl) +(puthash "wfsm" "-webkit-font-smoothing:${antialiased};" tbl) +(puthash "wfsm:a" "-webkit-font-smoothing:antialiased;" tbl) +(puthash "wfsm:n" "-webkit-font-smoothing:none;" tbl) +(puthash "wfsm:s" "-webkit-font-smoothing:subpixel-antialiased;" tbl) +(puthash "wfsm:sa" "-webkit-font-smoothing:subpixel-antialiased;" tbl) +(puthash "whs" "white-space:|;" tbl) +(puthash "whs:n" "white-space:normal;" tbl) +(puthash "whs:nw" "white-space:nowrap;" tbl) +(puthash "whs:p" "white-space:pre;" tbl) +(puthash "whs:pl" "white-space:pre-line;" tbl) +(puthash "whs:pw" "white-space:pre-wrap;" tbl) +(puthash "whsc" "white-space-collapse:|;" tbl) +(puthash "whsc:ba" "white-space-collapse:break-all;" tbl) +(puthash "whsc:bs" "white-space-collapse:break-strict;" tbl) +(puthash "whsc:k" "white-space-collapse:keep-all;" tbl) +(puthash "whsc:l" "white-space-collapse:loose;" tbl) +(puthash "whsc:n" "white-space-collapse:normal;" tbl) +(puthash "wid" "widows:|;" tbl) +(puthash "wm" "writing-mode:${1:lr-tb};" tbl) +(puthash "wm:btl" "writing-mode:bt-lr;" tbl) +(puthash "wm:btr" "writing-mode:bt-rl;" tbl) +(puthash "wm:lrb" "writing-mode:lr-bt;" tbl) +(puthash "wm:lrt" "writing-mode:lr-tb;" tbl) +(puthash "wm:rlb" "writing-mode:rl-bt;" tbl) +(puthash "wm:rlt" "writing-mode:rl-tb;" tbl) +(puthash "wm:tbl" "writing-mode:tb-lr;" tbl) +(puthash "wm:tbr" "writing-mode:tb-rl;" tbl) +(puthash "wob" "word-break:|;" tbl) +(puthash "wob:ba" "word-break:break-all;" tbl) +(puthash "wob:k" "word-break:keep-all;" tbl) +(puthash "wob:n" "word-break:normal;" tbl) +(puthash "wos" "word-spacing:|;" tbl) +(puthash "wow" "word-wrap:|;" tbl) +(puthash "wow:b" "word-wrap:break-word;" tbl) +(puthash "wow:n" "word-wrap:none;" tbl) +(puthash "wow:nm" "word-wrap:normal;" tbl) +(puthash "wow:s" "word-wrap:suppress;" tbl) +(puthash "wow:u" "word-wrap:unrestricted;" tbl) +(puthash "z" "z-index:|;" tbl) +(puthash "z:a" "z-index:auto;" tbl) +(puthash "zm" "zoom:1;" tbl) +(puthash "zoo" "zoom:1;" tbl) +tbl) tbl) +tbl) tbl) +(puthash "html" (let ((tbl (make-hash-table :test 'equal))) +(puthash "aliases" (let ((tbl (make-hash-table :test 'equal))) +(puthash "!" "html:5" tbl) +(puthash "!mvp" "html>(head>meta[charset=UTF-8]+meta:vp+title{Document})+body" tbl) +(puthash "a:link" "a[href=http://]" tbl) +(puthash "a:mail" "a[href=mailto:]" tbl) +(puthash "acr" "acronym" tbl) +(puthash "adr" "address" tbl) +(puthash "area:c" "area[shape=circle coords href alt]" tbl) +(puthash "area:d" "area[shape=default href alt]" tbl) +(puthash "area:p" "area[shape=poly coords href alt]" tbl) +(puthash "area:r" "area[shape=rect coords href alt]" tbl) +(puthash "art" "article" tbl) +(puthash "bdo:l" "bdo[dir=ltr]" tbl) +(puthash "bdo:r" "bdo[dir=rtl]" tbl) +(puthash "bq" "blockquote" tbl) +(puthash "btn" "button" tbl) +(puthash "btn:b" "button[type=button]" tbl) +(puthash "btn:r" "button[type=reset]" tbl) +(puthash "btn:s" "button[type=submit]" tbl) +(puthash "cap" "caption" tbl) +(puthash "cmd" "command" tbl) +(puthash "colg" "colgroup" tbl) +(puthash "colg+" "colgroup>col" tbl) +(puthash "colgroup+" "colgroup>col" tbl) +(puthash "datag" "datagrid" tbl) +(puthash "datal" "datalist" tbl) +(puthash "det" "details" tbl) +(puthash "dl+" "dl>dt+dd" tbl) +(puthash "dlg" "dialog" tbl) +(puthash "doc" "html>(head>meta[charset=UTF-8]+title{Document})+body" tbl) +(puthash "doc4" "html>(head>meta[http-equiv=\"Content-Type\" content=\"text/html;charset=UTF-8\"]+title{Document})" tbl) +(puthash "emb" "embed" tbl) +(puthash "fig" "figure" tbl) +(puthash "figc" "figcaption" tbl) +(puthash "form:get" "form[action method=get]" tbl) +(puthash "form:post" "form[action method=post]" tbl) +(puthash "fset" "fieldset" tbl) +(puthash "fst" "fieldset" tbl) +(puthash "ftr" "footer" tbl) +(puthash "hdr" "header" tbl) +(puthash "html:4s" "!!!4s+doc4[lang=en]" tbl) +(puthash "html:4t" "!!!4t+doc4[lang=en]" tbl) +(puthash "html:5" "!!!+doc[lang=en]" tbl) +(puthash "html:xml" "html[xmlns=http://www.w3.org/1999/xhtml]" tbl) +(puthash "html:xs" "!!!xs+doc4[xmlns=http://www.w3.org/1999/xhtml xml:lang=en]" tbl) +(puthash "html:xt" "!!!xt+doc4[xmlns=http://www.w3.org/1999/xhtml xml:lang=en]" tbl) +(puthash "html:xxs" "!!!xxs+doc4[xmlns=http://www.w3.org/1999/xhtml xml:lang=en]" tbl) +(puthash "ifr" "iframe" tbl) +(puthash "input:b" "input:button" tbl) +(puthash "input:button" "input[type=button]" tbl) +(puthash "input:c" "input:checkbox" tbl) +(puthash "input:checkbox" "input[type=checkbox]" tbl) +(puthash "input:color" "input[type=color]" tbl) +(puthash "input:date" "input[type=date]" tbl) +(puthash "input:datetime" "input[type=datetime]" tbl) +(puthash "input:datetime-local" "input[type=datetime-local]" tbl) +(puthash "input:email" "input[type=email]" tbl) +(puthash "input:f" "input:file" tbl) +(puthash "input:file" "input[type=file]" tbl) +(puthash "input:h" "input:hidden" tbl) +(puthash "input:hidden" "input[type=hidden]" tbl) +(puthash "input:i" "input:image" tbl) +(puthash "input:image" "input[type=image src alt]" tbl) +(puthash "input:month" "input[type=month]" tbl) +(puthash "input:number" "input[type=number]" tbl) +(puthash "input:p" "input:password" tbl) +(puthash "input:password" "input[type=password]" tbl) +(puthash "input:r" "input:radio" tbl) +(puthash "input:radio" "input[type=radio]" tbl) +(puthash "input:range" "input[type=range]" tbl) +(puthash "input:reset" "input[type=reset]" tbl) +(puthash "input:s" "input:submit" tbl) +(puthash "input:search" "input[type=search]" tbl) +(puthash "input:submit" "input[type=submit]" tbl) +(puthash "input:t" "input" tbl) +(puthash "input:text" "input" tbl) +(puthash "input:time" "input[type=time]" tbl) +(puthash "input:url" "input[type=url]" tbl) +(puthash "input:week" "input[type=week]" tbl) +(puthash "kg" "keygen" tbl) +(puthash "leg" "legend" tbl) +(puthash "link:atom" "link[rel=alternate type=\"application/atom+xml\" title=Atom href=atom.xml]" tbl) +(puthash "link:css" "link[rel=stylesheet href=style.css]" tbl) +(puthash "link:favicon" "link[icon rel=shortcut type=image/x-icon href=favicon.ico]" tbl) +(puthash "link:print" "link[rel=stylesheet href=print.css media=print]" tbl) +(puthash "link:rss" "link[rel=alternate type=application/rss+xml title=RSS href=rss.xml]" tbl) +(puthash "link:touch" "link[rel=apple-touch-icon href=favicon.png]" tbl) +(puthash "map+" "map>area" tbl) +(puthash "menu:c" "menu:context" tbl) +(puthash "menu:context" "menu[type=context]" tbl) +(puthash "menu:t" "menu:toolbar" tbl) +(puthash "menu:toolbar" "menu[type=toolbar]" tbl) +(puthash "meta:compat" "meta[http-equiv=X-UA-Compatible content=\"IE=edge,chrome=1\"]" tbl) +(puthash "meta:utf" "meta[http-equiv=Content-Type content=\"text/html;charset=UTF-8\"]" tbl) +(puthash "meta:vp" "meta[name=viewport content=\"width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0\"]" tbl) +(puthash "meta:win" "meta[http-equiv=Content-Type content=\"text/html;charset=windows-1251\"]" tbl) +(puthash "obj" "object" tbl) +(puthash "ol+" "ol>li" tbl) +(puthash "opt" "option" tbl) +(puthash "optg" "optgroup" tbl) +(puthash "optg+" "optgroup>option" tbl) +(puthash "optgroup+" "optgroup>option" tbl) +(puthash "out" "output" tbl) +(puthash "prog" "progress" tbl) +(puthash "script:src" "script[src]" tbl) +(puthash "sect" "section" tbl) +(puthash "select+" "select>option" tbl) +(puthash "src" "source" tbl) +(puthash "str" "strong" tbl) +(puthash "table+" "table>tr>td" tbl) +(puthash "tarea" "textarea" tbl) +(puthash "tr+" "tr>td" tbl) +(puthash "ul+" "ul>li" tbl) +tbl) tbl) +(puthash "snippets" (let ((tbl (make-hash-table :test 'equal))) +(puthash "!!!" "<!doctype html>" tbl) +(puthash "!!!4s" "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01//EN\" \"http://www.w3.org/TR/html4/strict.dtd\">" tbl) +(puthash "!!!4t" "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\" \"http://www.w3.org/TR/html4/loose.dtd\">" tbl) +(puthash "!!!xs" "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">" tbl) +(puthash "!!!xt" "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">" tbl) +(puthash "!!!xxs" "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.1//EN\" \"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd\">" tbl) +(puthash "cc:ie" "<!--[if IE]>\n\t${child}\n<![endif]-->" tbl) +(puthash "cc:ie6" "<!--[if lte IE 6]>\n\t${child}\n<![endif]-->" tbl) +(puthash "cc:noie" "<!--[if !IE]><!-->\n\t${child}\n<!--<![endif]-->" tbl) +tbl) tbl) +tbl) tbl) +(puthash "sass" (let ((tbl (make-hash-table :test 'equal))) +(puthash "snippets" (let ((tbl (make-hash-table :test 'equal))) +(puthash "@f" "@font-face\n\tfont-family:|\n\tsrc:url(|)\n" tbl) +(puthash "@f+" "@font-face\n\tfont-family: '${1:FontName}'\n\tsrc: url('${2:FileName}.eot')\n\tsrc: url('${2:FileName}.eot?#iefix') format('embedded-opentype'), url('${2:FileName}.woff') format('woff'), url('${2:FileName}.ttf') format('truetype'), url('${2:FileName}.svg#${1:FontName}') format('svg')\n\tfont-style: ${3:normal}\n\tfont-weight: ${4:normal}\n" tbl) +(puthash "@kf" "@-webkit-keyframes ${1:identifier}\n\t${2:from}\n\t\t${3}${6}\n\t${4:to}\n\t\t${5}\n\n@-o-keyframes ${1:identifier}\n\t${2:from}\n\t\t${3}${6}\n\t${4:to}\n\t\t${5}\n\n@-moz-keyframes ${1:identifier}\n\t${2:from}\n\t\t${3}${6}\n\t${4:to}\n\t\t${5}\n\n@keyframes ${1:identifier}\n\t${2:from}\n\t\t${3}${6}\n\t${4:to}\n\t\t${5}\n" tbl) +(puthash "@m" "@media ${1:screen}\n\t|\n" tbl) +(puthash "@media" "@media ${1:screen}\n\t|\n" tbl) +tbl) tbl) +tbl) tbl) +tbl)) +;; src/preferences.el +;; This file is generated from conf/preferences.json +;; Don't edit. +(emmet-defparameter emmet-preferences +(let ((tbl (make-hash-table :test 'equal))) +(puthash "css" (let ((tbl (make-hash-table :test 'equal))) +(puthash "color" (let ((tbl (make-hash-table :test 'equal))) +(puthash "case" "auto" tbl) +(puthash "shortenIfPossible" t tbl) +(puthash "trailingAliases" (let ((tbl (make-hash-table :test 'equal))) +(puthash "h" "hidden" tbl) +(puthash "n" "none" tbl) +(puthash "s" "solid" tbl) +(puthash "t" "dotted" tbl) +tbl) tbl) +tbl) tbl) +(puthash "floatUnit" "em" tbl) +(puthash "intUnit" "px" tbl) +(puthash "keywordAliases" (let ((tbl (make-hash-table :test 'equal))) +(puthash "a" "auto" tbl) +(puthash "da" "dashed" tbl) +(puthash "do" "dotted" tbl) +(puthash "i" "inherit" tbl) +(puthash "s" "solid" tbl) +(puthash "t" "transparent" tbl) +tbl) tbl) +(puthash "keywords" (vector +"auto" +"inherit" +) + tbl) +(puthash "unitAliases" (let ((tbl (make-hash-table :test 'equal))) +(puthash "-" "px" tbl) +(puthash "e" "em" tbl) +(puthash "p" "%" tbl) +(puthash "r" "rem" tbl) +(puthash "x" "ex" tbl) +tbl) tbl) +(puthash "unitlessProperties" (vector +"z-index" +"line-height" +"opacity" +"font-weight" +"zoom" +) + tbl) +(puthash "vendorPrefixesProperties" (let ((tbl (make-hash-table :test 'equal))) +(puthash "accelerator" (vector +"ms" +) + tbl) +(puthash "accesskey" (vector +"o" +) + tbl) +(puthash "animation" (vector +"webkit" +"o" +) + tbl) +(puthash "animation-delay" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "animation-direction" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "animation-duration" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "animation-fill-mode" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "animation-iteration-count" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "animation-name" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "animation-play-state" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "animation-timing-function" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "appearance" (vector +"webkit" +"moz" +) + tbl) +(puthash "backface-visibility" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "background-clip" (vector +"webkit" +"moz" +) + tbl) +(puthash "background-composite" (vector +"webkit" +) + tbl) +(puthash "background-inline-policy" (vector +"moz" +) + tbl) +(puthash "background-origin" (vector +"webkit" +) + tbl) +(puthash "background-position-x" (vector +"ms" +) + tbl) +(puthash "background-position-y" (vector +"ms" +) + tbl) +(puthash "background-size" (vector +"webkit" +) + tbl) +(puthash "behavior" (vector +"ms" +) + tbl) +(puthash "binding" (vector +"moz" +) + tbl) +(puthash "block-progression" (vector +"ms" +) + tbl) +(puthash "border-bottom-colors" (vector +"moz" +) + tbl) +(puthash "border-fit" (vector +"webkit" +) + tbl) +(puthash "border-horizontal-spacing" (vector +"webkit" +) + tbl) +(puthash "border-image" (vector +"webkit" +"moz" +"o" +) + tbl) +(puthash "border-left-colors" (vector +"moz" +) + tbl) +(puthash "border-radius" (vector +"webkit" +"moz" +) + tbl) +(puthash "border-right-colors" (vector +"moz" +) + tbl) +(puthash "border-top-colors" (vector +"moz" +) + tbl) +(puthash "border-vertical-spacing" (vector +"webkit" +) + tbl) +(puthash "box-align" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "box-direction" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "box-flex" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "box-flex-group" (vector +"webkit" +) + tbl) +(puthash "box-line-progression" (vector +"ms" +) + tbl) +(puthash "box-lines" (vector +"webkit" +"ms" +) + tbl) +(puthash "box-ordinal-group" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "box-orient" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "box-pack" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "box-reflect" (vector +"webkit" +) + tbl) +(puthash "box-shadow" (vector +"webkit" +"moz" +) + tbl) +(puthash "box-sizing" (vector +"webkit" +"moz" +) + tbl) +(puthash "color-correction" (vector +"webkit" +) + tbl) +(puthash "column-break-after" (vector +"webkit" +) + tbl) +(puthash "column-break-before" (vector +"webkit" +) + tbl) +(puthash "column-break-inside" (vector +"webkit" +) + tbl) +(puthash "column-count" (vector +"webkit" +"moz" +) + tbl) +(puthash "column-gap" (vector +"webkit" +"moz" +) + tbl) +(puthash "column-rule-color" (vector +"webkit" +"moz" +) + tbl) +(puthash "column-rule-style" (vector +"webkit" +"moz" +) + tbl) +(puthash "column-rule-width" (vector +"webkit" +"moz" +) + tbl) +(puthash "column-span" (vector +"webkit" +) + tbl) +(puthash "column-width" (vector +"webkit" +"moz" +) + tbl) +(puthash "content-zoom-boundary" (vector +"ms" +) + tbl) +(puthash "content-zoom-boundary-max" (vector +"ms" +) + tbl) +(puthash "content-zoom-boundary-min" (vector +"ms" +) + tbl) +(puthash "content-zoom-chaining" (vector +"ms" +) + tbl) +(puthash "content-zoom-snap" (vector +"ms" +) + tbl) +(puthash "content-zoom-snap-points" (vector +"ms" +) + tbl) +(puthash "content-zoom-snap-type" (vector +"ms" +) + tbl) +(puthash "content-zooming" (vector +"ms" +) + tbl) +(puthash "dashboard-region" (vector +"webkit" +"o" +) + tbl) +(puthash "filter" (vector +"ms" +) + tbl) +(puthash "float-edge" (vector +"moz" +) + tbl) +(puthash "flow-from" (vector +"ms" +) + tbl) +(puthash "flow-into" (vector +"ms" +) + tbl) +(puthash "font-feature-settings" (vector +"moz" +"ms" +) + tbl) +(puthash "font-language-override" (vector +"moz" +) + tbl) +(puthash "font-smoothing" (vector +"webkit" +) + tbl) +(puthash "force-broken-image-icon" (vector +"moz" +) + tbl) +(puthash "grid-column" (vector +"ms" +) + tbl) +(puthash "grid-column-align" (vector +"ms" +) + tbl) +(puthash "grid-column-span" (vector +"ms" +) + tbl) +(puthash "grid-columns" (vector +"ms" +) + tbl) +(puthash "grid-layer" (vector +"ms" +) + tbl) +(puthash "grid-row" (vector +"ms" +) + tbl) +(puthash "grid-row-align" (vector +"ms" +) + tbl) +(puthash "grid-row-span" (vector +"ms" +) + tbl) +(puthash "grid-rows" (vector +"ms" +) + tbl) +(puthash "high-contrast-adjust" (vector +"ms" +) + tbl) +(puthash "highlight" (vector +"webkit" +) + tbl) +(puthash "hyphenate-character" (vector +"webkit" +) + tbl) +(puthash "hyphenate-limit-after" (vector +"webkit" +) + tbl) +(puthash "hyphenate-limit-before" (vector +"webkit" +) + tbl) +(puthash "hyphenate-limit-chars" (vector +"ms" +) + tbl) +(puthash "hyphenate-limit-lines" (vector +"ms" +) + tbl) +(puthash "hyphenate-limit-zone" (vector +"ms" +) + tbl) +(puthash "hyphens" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "image-region" (vector +"moz" +) + tbl) +(puthash "ime-mode" (vector +"ms" +) + tbl) +(puthash "input-format" (vector +"o" +) + tbl) +(puthash "input-required" (vector +"o" +) + tbl) +(puthash "interpolation-mode" (vector +"ms" +) + tbl) +(puthash "layout-flow" (vector +"ms" +) + tbl) +(puthash "layout-grid" (vector +"ms" +) + tbl) +(puthash "layout-grid-char" (vector +"ms" +) + tbl) +(puthash "layout-grid-line" (vector +"ms" +) + tbl) +(puthash "layout-grid-mode" (vector +"ms" +) + tbl) +(puthash "layout-grid-type" (vector +"ms" +) + tbl) +(puthash "line-box-contain" (vector +"webkit" +) + tbl) +(puthash "line-break" (vector +"webkit" +"ms" +) + tbl) +(puthash "line-clamp" (vector +"webkit" +) + tbl) +(puthash "link" (vector +"o" +) + tbl) +(puthash "link-source" (vector +"o" +) + tbl) +(puthash "locale" (vector +"webkit" +) + tbl) +(puthash "margin-after-collapse" (vector +"webkit" +) + tbl) +(puthash "margin-before-collapse" (vector +"webkit" +) + tbl) +(puthash "marquee-dir" (vector +"o" +) + tbl) +(puthash "marquee-direction" (vector +"webkit" +) + tbl) +(puthash "marquee-increment" (vector +"webkit" +) + tbl) +(puthash "marquee-loop" (vector +"o" +) + tbl) +(puthash "marquee-repetition" (vector +"webkit" +) + tbl) +(puthash "marquee-speed" (vector +"o" +) + tbl) +(puthash "marquee-style" (vector +"webkit" +"o" +) + tbl) +(puthash "mask-attachment" (vector +"webkit" +) + tbl) +(puthash "mask-box-image" (vector +"webkit" +) + tbl) +(puthash "mask-box-image-outset" (vector +"webkit" +) + tbl) +(puthash "mask-box-image-repeat" (vector +"webkit" +) + tbl) +(puthash "mask-box-image-slice" (vector +"webkit" +) + tbl) +(puthash "mask-box-image-source" (vector +"webkit" +) + tbl) +(puthash "mask-box-image-width" (vector +"webkit" +) + tbl) +(puthash "mask-clip" (vector +"webkit" +) + tbl) +(puthash "mask-composite" (vector +"webkit" +) + tbl) +(puthash "mask-image" (vector +"webkit" +) + tbl) +(puthash "mask-origin" (vector +"webkit" +) + tbl) +(puthash "mask-position" (vector +"webkit" +) + tbl) +(puthash "mask-repeat" (vector +"webkit" +) + tbl) +(puthash "mask-size" (vector +"webkit" +) + tbl) +(puthash "nbsp-mode" (vector +"webkit" +) + tbl) +(puthash "object-fit" (vector +"o" +) + tbl) +(puthash "object-position" (vector +"o" +) + tbl) +(puthash "orient" (vector +"moz" +) + tbl) +(puthash "outline-radius-bottomleft" (vector +"moz" +) + tbl) +(puthash "outline-radius-bottomright" (vector +"moz" +) + tbl) +(puthash "outline-radius-topleft" (vector +"moz" +) + tbl) +(puthash "outline-radius-topright" (vector +"moz" +) + tbl) +(puthash "overflow-style" (vector +"ms" +) + tbl) +(puthash "perspective" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "perspective-origin" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "perspective-origin-x" (vector +"ms" +) + tbl) +(puthash "perspective-origin-y" (vector +"ms" +) + tbl) +(puthash "rtl-ordering" (vector +"webkit" +) + tbl) +(puthash "scroll-boundary" (vector +"ms" +) + tbl) +(puthash "scroll-boundary-bottom" (vector +"ms" +) + tbl) +(puthash "scroll-boundary-left" (vector +"ms" +) + tbl) +(puthash "scroll-boundary-right" (vector +"ms" +) + tbl) +(puthash "scroll-boundary-top" (vector +"ms" +) + tbl) +(puthash "scroll-chaining" (vector +"ms" +) + tbl) +(puthash "scroll-rails" (vector +"ms" +) + tbl) +(puthash "scroll-snap-points-x" (vector +"ms" +) + tbl) +(puthash "scroll-snap-points-y" (vector +"ms" +) + tbl) +(puthash "scroll-snap-type" (vector +"ms" +) + tbl) +(puthash "scroll-snap-x" (vector +"ms" +) + tbl) +(puthash "scroll-snap-y" (vector +"ms" +) + tbl) +(puthash "scrollbar-arrow-color" (vector +"ms" +) + tbl) +(puthash "scrollbar-base-color" (vector +"ms" +) + tbl) +(puthash "scrollbar-darkshadow-color" (vector +"ms" +) + tbl) +(puthash "scrollbar-face-color" (vector +"ms" +) + tbl) +(puthash "scrollbar-highlight-color" (vector +"ms" +) + tbl) +(puthash "scrollbar-shadow-color" (vector +"ms" +) + tbl) +(puthash "scrollbar-track-color" (vector +"ms" +) + tbl) +(puthash "stack-sizing" (vector +"moz" +) + tbl) +(puthash "svg-shadow" (vector +"webkit" +) + tbl) +(puthash "tab-size" (vector +"moz" +"o" +) + tbl) +(puthash "table-baseline" (vector +"o" +) + tbl) +(puthash "text-align-last" (vector +"ms" +) + tbl) +(puthash "text-autospace" (vector +"ms" +) + tbl) +(puthash "text-blink" (vector +"moz" +) + tbl) +(puthash "text-combine" (vector +"webkit" +) + tbl) +(puthash "text-decoration-color" (vector +"moz" +) + tbl) +(puthash "text-decoration-line" (vector +"moz" +) + tbl) +(puthash "text-decoration-style" (vector +"moz" +) + tbl) +(puthash "text-decorations-in-effect" (vector +"webkit" +) + tbl) +(puthash "text-emphasis-color" (vector +"webkit" +) + tbl) +(puthash "text-emphasis-position" (vector +"webkit" +) + tbl) +(puthash "text-emphasis-style" (vector +"webkit" +) + tbl) +(puthash "text-fill-color" (vector +"webkit" +) + tbl) +(puthash "text-justify" (vector +"ms" +) + tbl) +(puthash "text-kashida-space" (vector +"ms" +) + tbl) +(puthash "text-orientation" (vector +"webkit" +) + tbl) +(puthash "text-overflow" (vector +"ms" +) + tbl) +(puthash "text-security" (vector +"webkit" +) + tbl) +(puthash "text-size-adjust" (vector +"moz" +"ms" +) + tbl) +(puthash "text-stroke-color" (vector +"webkit" +) + tbl) +(puthash "text-stroke-width" (vector +"webkit" +) + tbl) +(puthash "text-underline-position" (vector +"ms" +) + tbl) +(puthash "touch-action" (vector +"ms" +) + tbl) +(puthash "transform" (vector +"webkit" +"moz" +"ms" +"o" +) + tbl) +(puthash "transform-origin" (vector +"webkit" +"moz" +"ms" +"o" +) + tbl) +(puthash "transform-origin-x" (vector +"ms" +) + tbl) +(puthash "transform-origin-y" (vector +"ms" +) + tbl) +(puthash "transform-origin-z" (vector +"ms" +) + tbl) +(puthash "transform-style" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "transition" (vector +"webkit" +"moz" +"ms" +"o" +) + tbl) +(puthash "transition-delay" (vector +"webkit" +"moz" +"ms" +"o" +) + tbl) +(puthash "transition-duration" (vector +"webkit" +"moz" +"ms" +"o" +) + tbl) +(puthash "transition-property" (vector +"webkit" +"moz" +"ms" +"o" +) + tbl) +(puthash "transition-timing-function" (vector +"webkit" +"moz" +"ms" +"o" +) + tbl) +(puthash "user-drag" (vector +"webkit" +) + tbl) +(puthash "user-focus" (vector +"moz" +) + tbl) +(puthash "user-input" (vector +"moz" +) + tbl) +(puthash "user-modify" (vector +"webkit" +"moz" +) + tbl) +(puthash "user-select" (vector +"webkit" +"moz" +"ms" +) + tbl) +(puthash "window-shadow" (vector +"moz" +) + tbl) +(puthash "word-break" (vector +"ms" +) + tbl) +(puthash "word-wrap" (vector +"ms" +) + tbl) +(puthash "wrap-flow" (vector +"ms" +) + tbl) +(puthash "wrap-margin" (vector +"ms" +) + tbl) +(puthash "wrap-through" (vector +"ms" +) + tbl) +(puthash "writing-mode" (vector +"webkit" +"ms" +) + tbl) +tbl) tbl) +tbl) tbl) +(puthash "html" (let ((tbl (make-hash-table :test 'equal))) +(puthash "tags" (let ((tbl (make-hash-table :test 'equal))) +(puthash "a" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "href" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "abbr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "title" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "acronym" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "title" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "address" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "applet" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "area" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "alt" "" tbl) +(puthash "coords" "" tbl) +(puthash "href" "" tbl) +(puthash "shape" "" tbl) +tbl) tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "article" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "aside" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "audio" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "src" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "b" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "base" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "href" "" tbl) +tbl) tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "basefont" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "bdi" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "bdo" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "dir" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "big" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "blockquote" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "body" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "br" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "button" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "canvas" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "caption" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "center" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "cite" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "code" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "col" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "colgroup" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "command" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "datalist" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "dd" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "del" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "details" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "dfn" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "dialog" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "dir" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "div" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "dl" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "dt" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "em" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "embed" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "src" "" tbl) +(puthash "type" "" tbl) +tbl) tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "fieldset" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "figcaption" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "figure" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "font" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "footer" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "form" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "frame" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "frameset" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "h1" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "h2" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "h3" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "h4" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "h5" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "h6" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "head" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "header" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "hgroup" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "hr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "html" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "i" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "iframe" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "frameborder" "0" tbl) +(puthash "src" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "img" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "alt" "" tbl) +(puthash "src" "" tbl) +tbl) tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "input" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "name" "" tbl) +(puthash "type" "text" tbl) +(puthash "value" "" tbl) +tbl) tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "ins" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "kbd" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "keygen" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "label" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "for" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "legend" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "li" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "link" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "href" "" tbl) +(puthash "rel" "stylesheet" tbl) +tbl) tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "map" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "name" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "mark" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "menu" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "meta" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "meter" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "nav" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "noframes" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "noscript" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "object" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "data" "" tbl) +(puthash "type" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "ol" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "optgroup" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "option" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "value" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "output" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "p" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "param" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "name" "" tbl) +(puthash "value" "" tbl) +tbl) tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "pre" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "progress" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "q" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "rp" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "rt" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "ruby" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "s" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "samp" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "script" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "section" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "select" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "id" "" tbl) +(puthash "name" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "small" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "source" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "span" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "strike" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "strong" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "style" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "sub" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "summary" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "sup" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "table" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "tbody" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "td" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "textarea" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "cols" "30" tbl) +(puthash "id" "" tbl) +(puthash "name" "" tbl) +(puthash "rows" "10" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "tfoot" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "th" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "thead" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "time" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "title" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "tr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "track" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" t tbl) +tbl) tbl) +(puthash "tt" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "u" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "ul" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "var" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "video" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" t tbl) +(puthash "defaultAttr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "src" "" tbl) +tbl) tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +(puthash "wbr" (let ((tbl (make-hash-table :test 'equal))) +(puthash "block" nil tbl) +(puthash "selfClosing" nil tbl) +tbl) tbl) +tbl) tbl) +tbl) tbl) +tbl)) +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; XML abbrev + +(require 'cl-lib) + +(emmet-defparameter + emmet-tag-aliases-table + (gethash "aliases" (gethash "html" emmet-snippets))) + +(defun emmet-expr (input) + "Parse a zen coding expression with optional filters." + (emmet-pif (emmet-extract-expr-filters input) + (let ((input (elt it 1)) + (filters (elt it 2))) + (emmet-pif (emmet-extract-filters filters) + (emmet-filter input it) + it)) + (emmet-filter input (emmet-default-filter)))) + +(defun emmet-subexpr (input) + "Parse a zen coding expression with no filter. This pretty much defines precedence." + (emmet-run emmet-siblings + it + (emmet-run emmet-parent-child + it + (emmet-run emmet-multiplier + it + (emmet-run emmet-pexpr + it + (emmet-run emmet-text + it + (emmet-run emmet-tag + it + '(error "no match, expecting ( or a-zA-Z0-9")))))))) + +(defun emmet-extract-expr-filters (input) + (cl-flet ((string-match-reverse (str regexp) + (emmet-aif (string-match regexp (reverse str)) (- (length str) 1 it)))) + (let ((err '(error "expected expr|filter"))) + (emmet-aif (string-match-reverse input "|") + (if (string-match "[\"}]" (substring input (+ it 1))) + err + `(,input + ,(substring input 0 it) + ,(substring input (+ it 1)))) + err)))) + +(defun emmet-extract-filters (input) + "Extract filters from expression." + (emmet-pif (emmet-parse "\\([^\\|]+?\\)|" 2 "" it) + (let ((filter-name (elt it 1)) + (more-filters (elt it 2))) + (emmet-pif (emmet-extract-filters more-filters) + (cons filter-name it) + it)) + (emmet-parse "\\([^\\|]+\\)" 1 "filter name" `(,(elt it 1))))) + +(defun emmet-extract-inner-text (input) + "Extract inner-text in the form of {inner_text}... +Right curly braces can be escaped by backslash, i.e. '\\}' +Return `(,inner-text ,input-without-inner-text) if succeeds, otherwise return +`(error ,error-message)" + (cl-labels ( + (string-find-paired-right-curly-brace + (str) + (cl-labels ((char-at-pos (str pos) (string-to-char (substring str pos))) + (helper (str pos cnt) + (let ((len (length str))) + (if (>= pos len) nil + (let ((c (char-at-pos str pos))) + (cond ((char-equal c ?{) (setq cnt (+ cnt 1))) + ((and (char-equal c ?}) (not (char-equal (char-at-pos str (- pos 1)) ?\\))) (setq cnt (- cnt 1)))) + (if (= cnt 0) pos (helper str (+ pos 1) cnt))))) + )) + (helper str 0 0)))) + (let ((err '(error "expected inner text"))) + (if (or (< (length input) 2) (not (char-equal (string-to-char input) ?{))) err + (emmet-aif (string-find-paired-right-curly-brace input) + `(,input + ,(substring input 1 it) + ,(substring input (+ it 1))) + err))))) + +(defun emmet-filter (input filters) + "Construct AST with specified filters." + (emmet-pif (emmet-subexpr input) + (let ((result (car it)) + (rest (cdr it))) + `((filter ,filters ,result) . ,rest)) + it)) + +(defun emmet-default-filter () + "Default filter(s) to be used if none is specified." + (or emmet-file-filter + (let* ((file-ext (car (emmet-regex ".*\\(\\..*\\)" (or (buffer-file-name) "") 1))) + (defaults '(".html" ("html") + ".htm" ("html") + ".haml" ("haml") + ".clj" ("hic") + ".cljs" ("hic"))) + (default-else emmet-fallback-filter) + (selected-default (member file-ext defaults))) + (if selected-default + (cadr selected-default) + default-else)))) + +(defun emmet-numbering (input) + (emmet-parse + "\\(\\$+\\)" 2 "numbering, $" + (let ((doller (elt it 1))) + (emmet-pif (emmet-parse + "@\\([0-9-][0-9]*\\)" 2 "numbering args" + (let* ((args (read (elt it 1))) + (direction (not (or (eq '- args) (minusp args)))) + (base (if (eq '- args) 1 (abs args)))) + `((n ,(length doller) ,direction ,base) . ,input))) + it + `((n ,(length doller) t 1) . ,input))))) + +(defun emmet-split-numbering-expressions (input) + (cl-labels + ((iter (input) + (emmet-aif (emmet-regex "\\([^$]*\\)\\(\\$.*\\)" input '(1 2)) + (let ((prefix (car it)) + (input (cadr it))) + (if (and (< 0 (length prefix)) ; check if ..\\$... or ...$... + (string-equal (substring prefix -1) "\\")) + `(,(store-substring prefix (- (length prefix) 1) ?$) + ,@(iter (substring input 1))) + (let ((res (emmet-numbering input))) + `(,prefix ,(car res) ,@(iter (cdr res)))))) + (list input)))) + (let ((res (iter input))) + (if (cl-every #'stringp res) + (apply #'concat res) + `(numberings ,@res))))) + +(defun emmet-instantiate-numbering-expression (i lim exp) + (cl-labels ((instantiate + (i lim exps) + (apply #'concat + (mapcar + (lambda (exp) + (if (listp exp) + (let ((digits (second exp)) + (direction (third exp)) + (base (fourth exp))) + (let ((num (if direction (+ base i) + (- (+ lim (- base 1)) i)))) + (format (concat "%0" (format "%d" digits) "d") num))) + exp)) exps))) + (cl-search + (i lim exp) + (if (listp exp) + (if (eql (car exp) 'numberings) + (instantiate i lim (cdr exp)) + ;; Should do like this for real searching. + ;; But stack overflow occurs. + ;; (cons (search-numberings i lim (car exp)) + ;; (search-numberings i lim (cdr exp))) + (mapcar (lambda (exp) (cl-search i lim exp)) exp)) + exp))) + (cl-search i lim exp))) + +(defun emmet-multiply-expression (multiplicand exp) + (cl-loop for i to (- multiplicand 1) collect + (emmet-instantiate-numbering-expression i multiplicand exp))) + +(defun emmet-multiplier (input) + (emmet-pif (emmet-run emmet-pexpr + it + (emmet-run emmet-tag + it + (emmet-run emmet-text + it + '(error "expected *n multiplier")))) + (let* ((expr (car it)) (input (cdr it)) + (multiplier expr)) + (emmet-parse "\\*\\([0-9]+\\)" 2 "*n where n is a number" + (let ((multiplicand (read (elt it 1)))) + `((list ,(emmet-multiply-expression + multiplicand + multiplier)) . ,input)))))) + +(defun emmet-tag (input) + "Parse a tag." + (emmet-run + emmet-tagname + (let ((tagname (cadr expr)) + (has-body? (cddr expr))) + (emmet-pif + (emmet-run emmet-identifier + (emmet-tag-classes + `(tag (,tagname ,has-body? ,(cddr expr))) input) + (emmet-tag-classes + `(tag (,tagname ,has-body? nil)) input)) + (let ((tag-data (cadar it)) (input (cdr it))) + (emmet-pif (emmet-run + emmet-properties + (let ((props (cdr expr))) + `((tag ,(append tag-data (list props))) . ,input)) + `((tag ,(append tag-data '(nil))) . ,input)) + (let ((expr (car it)) (input (cdr it))) + (cl-destructuring-bind (expr . input) + (emmet-tag-text expr input) + (or + (emmet-expand-lorem expr input) + (emmet-expand-tag-alias expr input)))))))) + (emmet-default-tag input))) + +(defun emmet-get-first-tag (expr) + (if (listp expr) + (if (listp (car expr)) + (emmet-get-first-tag (car expr)) + (if (eql (car expr) 'tag) + expr + (emmet-get-first-tag (cdr expr)))) + nil)) + +(defun emmet-lorem (input) + (emmet-aif + (and (stringp input) (emmet-regex "\\(?:lorem\\|ipsum\\)\\([0-9]*\\)" input '(0 1))) + (let ((w (elt it 1))) + (let ((word-num (if (string-equal w "") 30 (read w)))) + word-num)))) + +(defun emmet-expand-lorem (tag input) + (let ((tag-data (cadr tag))) + (let ((tag-name (car tag-data))) + (emmet-aif (emmet-lorem tag-name) + (if (cl-equalp (cdr tag-data) '(t nil nil nil nil)) + `((text (lorem ,it)) . ,input) + `((tag ("div" ,@(cl-subseq tag-data 1 -1) (lorem ,it))) . ,input)))))) + +(defun emmet-expand-tag-alias (tag input) + (let ((tag-data (cadr tag))) + (let ((tag-name (car tag-data))) + (emmet-aif + (gethash tag-name emmet-tag-aliases-table) + (let ((expr (if (stringp it) + (emmet-subexpr it) + it))) + (prog1 + (let ((rt (copy-tree expr))) + (let ((first-tag-data (cadr (emmet-get-first-tag rt)))) + (setf (second first-tag-data) (second tag-data)) + (setf (third first-tag-data) (third tag-data)) + (setf (fourth first-tag-data) + (cl-remove-duplicates + (append (fourth first-tag-data) + (fourth tag-data)) :test #'string=)) + (setf (fifth first-tag-data) + (cl-remove-duplicates + (append (fifth first-tag-data) + (fifth tag-data)) + :test #'(lambda (p1 p2) + (eql (car p1) (car p2))))) + (setf (cl-sixth first-tag-data) (cl-sixth tag-data)) + (setf (cdr rt) (concat (cdr rt) input)) + rt)) + (puthash tag-name expr emmet-tag-aliases-table))) + `(,tag . ,input))))) + +(defun emmet-default-tag (input) + "Parse a #id or .class" + (emmet-parse "\\([#|\\.]\\)" 1 "tagname" + (emmet-tag (concat "div" (elt it 0))))) + +(defun emmet-tag-text (tag input) + (let ((tag-data (cadr tag))) + (emmet-run emmet-text + (let ((txt (cadr expr))) + `((tag ,(append tag-data (list txt))) . ,input)) + `((tag ,(append tag-data '(nil))) . ,input)))) + +(defun emmet-tag-props (tag input) + (let ((tag-data (cadr tag))) + (emmet-run emmet-properties + (let ((props (cdr expr))) + `((tag ,(append tag-data (list props))) . ,input)) + `((tag ,(append tag-data '(nil))) . ,input)))) + +(defun emmet-props (input) + "Parse many props." + (emmet-run emmet-prop + (emmet-pif (emmet-props input) + `((props . ,(cons expr (cdar it))) . ,(cdr it)) + `((props . ,(list expr)) . ,input)))) + +(defun emmet-prop (input) + (emmet-parse + " *" 1 "space" + (emmet-run + emmet-name + (let ((name (cdr expr))) + (emmet-pif (emmet-parse "\\.\\(.*?\\)" 2 "." + `((,(read name)) . ,input)) + it + (emmet-pif (emmet-prop-value name input) + it + `((,(read name) "") . ,input))))))) + +(defun emmet-prop-value (name input) + (emmet-pif (emmet-parse + "=\"\\(.*?\\)\"" 2 + "=\"property value\"" + (let ((value (elt it 1)) + (input (elt it 2))) + `((,(read name) ,(emmet-split-numbering-expressions value)) . ,input))) + it + (let ((emmet--prop-value-parse-any + (lambda () (emmet-parse + "=\\([^\\,\\+\\>\\{\\}\\ )]*\\)" 2 + "=property value" + (let ((value (elt it 1)) + (input (elt it 2))) + `((,(read name) ,(emmet-split-numbering-expressions value)) . ,input)))))) + (if (emmet-jsx-supported-mode?) + (emmet-pif + (emmet-parse "=\\({.*?}\\)" 2 + "=\"property value\"" + (let ((value (elt it 1)) + (input (elt it 2))) + `((,(read name) ,(emmet-split-numbering-expressions value)) . ,input))) + it + (funcall emmet--prop-value-parse-any)) + (funcall emmet--prop-value-parse-any))))) + +(defun emmet-tag-classes (tag input) + (let ((tag-data (cadr tag))) + (emmet-run emmet-classes + (let ((classes (mapcar (lambda (cls) (cdadr cls)) + (cdr expr)))) + `((tag ,(append tag-data (list classes))) . ,input)) + `((tag ,(append tag-data '(nil))) . ,input)))) + +(defun emmet-tagname (input) + "Parse a tagname a-zA-Z0-9 tagname (e.g. html/head/xsl:if/br)." + (emmet-parse "\\([a-zA-Z!][a-zA-Z0-9:!$@-]*\/?\\)" 2 "tagname, a-zA-Z0-9" + (let* ((tag-spec (elt it 1)) + (empty-tag (emmet-regex "\\([^\/]*\\)\/" tag-spec 1)) + (tag (emmet-split-numbering-expressions + (if empty-tag (car empty-tag) tag-spec)))) + `((tagname . (,tag . ,(not empty-tag))) . ,input)))) + +(defun emmet-text (input) + "A zen coding expression innertext." + (emmet-pif (emmet-extract-inner-text input) + (let ((txt (emmet-split-numbering-expressions (elt it 1)))) + (if (listp txt) + (setq txt (cons (car txt) (cons (replace-regexp-in-string "\\\\\\(.\\)" "\\1" (cadr txt)) (cddr txt)))) + (setq txt (replace-regexp-in-string "\\\\\\(.\\)" "\\1" txt))) + `((text ,txt) . ,(elt it 2))) + '(error "expected inner text"))) + +;; (defun emmet-text (input) +;; "A zen coding expression innertext." +;; (emmet-parse "{\\(\\(?:\\\\.\\|[^\\\\}]\\|\\\\}\\)*?\\)}" 2 "inner text" +;; (let ((txt (emmet-split-numbering-expressions (elt it 1)))) +;; (if (listp txt) +;; (setq txt (cons (car txt) (cons (replace-regexp-in-string "\\\\\\(.\\)" "\\1" (cadr txt)) (cddr txt)))) +;; (setq txt (replace-regexp-in-string "\\\\\\(.\\)" "\\1" txt))) +;; `((text ,txt) . ,input)) +;; '(error "expected inner text"))) + +(defun emmet-properties (input) + "A bracketed emmet property expression." + (emmet-parse "\\[\\(.*?\\)\\]" 2 "properties" + `(,(car (emmet-props (elt it 1))) . ,input))) + + +(defun emmet-pexpr (input) + "A zen coding expression with parentheses around it." + (emmet-parse "(" 1 "(" + (emmet-run emmet-subexpr + (emmet-aif (emmet-regex ")" input '(0 1)) + `(,expr . ,(elt it 1)) + '(error "expecting `)'"))))) + +(defun emmet-parent-child (input) + "Parse an tag>e expression, where `n' is an tag and `e' is any + expression." + (cl-labels + ((listing (parents child input) + (let ((len (length parents))) + `((list ,(map 'list + (lambda (parent i) + `(parent-child ,parent + ,(emmet-instantiate-numbering-expression i len child))) + parents + (cl-loop for i to (- len 1) collect i))) . ,input)))) + (emmet-run + emmet-multiplier + (let* ((items (cadr expr)) + (rest (emmet-child-sans expr input))) + (if (not (eq (car rest) 'error)) + (let ((child (car rest)) + (input (cdr rest))) + + (emmet-aif (emmet-regex "^" input '(0 1)) + (let ((input (elt it 1))) + (emmet-run + emmet-subexpr + `((sibling ,(car (listing items child "")) ,expr) . ,input) + (listing items child input))) + (listing items child input))) + '(error "expected child"))) + (emmet-run emmet-tag + (emmet-child expr input) + '(error "expected parent"))))) + +(defun emmet-child-sans (parent input) + (emmet-parse ">" 1 ">" + (emmet-run emmet-subexpr + it + '(error "expected child")))) + +(defun emmet-child (parent input) + (emmet-parse ">" 1 ">" + (emmet-run emmet-subexpr + (let ((child expr)) + (emmet-aif (emmet-regex "^" input '(0 1)) + (let ((input (elt it 1))) + (emmet-run emmet-subexpr + `((sibling (parent-child ,parent ,child) ,expr) . ,input) + `((parent-child ,parent ,child) . ,input))) + `((parent-child ,parent ,child) . ,input))) + '(error "expected child")))) + +(defun emmet-sibling (input) + (emmet-por emmet-pexpr emmet-multiplier + it + (emmet-run emmet-tag + it + (emmet-run emmet-text + it + '(error "expected sibling"))))) + +(defun emmet-siblings (input) + "Parse an e+e expression, where e is an tag or a pexpr." + (emmet-run emmet-sibling + (let ((parent expr)) + (emmet-parse + "\\+" 1 "+" + (emmet-run + emmet-subexpr + (let ((child expr)) + `((sibling ,parent ,child) . ,input)) + (emmet-expand parent input)))) + '(error "expected first sibling"))) + +(defun emmet-expand (parent input) + "Parse an e+ expression, where e is an expandable tag" + (let* ((parent-tag (car (cadr parent)))) + (setf (caadr parent) (concat parent-tag "+")) + (cl-destructuring-bind (parent . input) + (emmet-expand-tag-alias parent input) + (emmet-pif (emmet-parse "+\\(.*\\)" 1 "+expr" + (emmet-subexpr (elt it 1))) + `((sibling ,parent ,@it)) + `(,parent . ,input))))) + +(defun emmet-name (input) + "Parse a class or identifier name, e.g. news, footer, mainimage" + (emmet-parse "\\([a-zA-Z$@][a-zA-Z0-9$@_:-]*\\)" 2 "class or identifer name" + `((name . ,(emmet-split-numbering-expressions + (elt it 1))) . ,input))) + +(defun emmet-class (input) + "Parse a classname expression, e.g. .foo" + (emmet-parse "\\." 1 "." + (emmet-run emmet-name + `((class ,expr) . ,input) + '(error "expected class name")))) +(defun emmet-identifier (input) + "Parse an identifier expression, e.g. #foo" + (emmet-parse "#" 1 "#" + (emmet-run emmet-name + `((identifier . ,expr) . ,input)))) + +(defun emmet-classes (input) + "Parse many classes." + (emmet-run emmet-class + (emmet-pif (emmet-classes input) + `((classes . ,(cons expr (cdar it))) . ,(cdr it)) + `((classes . ,(list expr)) . ,input)) + '(error "expected class"))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Zen coding transformer from AST to string + +(defvar emmet-leaf-function nil + "Function to execute when expanding a leaf node in the + Emmet AST.") + +(defvar emmet-jsx-className-braces? nil + "Wether to wrap classNames in {} instead of \"\"") + +(emmet-defparameter + emmet-tag-settings-table + (gethash "tags" (gethash "html" emmet-preferences))) + +(emmet-defparameter + emmet-tag-snippets-table + (gethash "snippets" (gethash "html" emmet-snippets))) + +(defvar emmet-filters + '("html" (emmet-primary-filter emmet-make-html-tag) + "c" (emmet-primary-filter emmet-make-commented-html-tag) + "haml" (emmet-primary-filter emmet-make-haml-tag) + "hic" (emmet-primary-filter emmet-make-hiccup-tag) + "e" (emmet-escape-xml))) + +(defun emmet-instantiate-lorem-expression (input) + (if input + (if (consp input) + (if (and (eql (car input) 'lorem) (numberp (cadr input))) + (emmet-lorem-generate (cadr input)) + (cons (emmet-instantiate-lorem-expression (car input)) + (emmet-instantiate-lorem-expression (cdr input)))) + input))) + +(defun emmet-primary-filter (input proc) + "Process filter that needs to be executed first, ie. not given output from other filter." + (if (listp input) + (let ((tag-maker (cadr proc))) + (emmet-transform-ast + (emmet-instantiate-lorem-expression input) + tag-maker)) + nil)) + +(defun emmet-process-filter (filters input) + "Process filters, chain one filter output as the input of the next filter." + (let ((filter-data (member (car filters) emmet-filters)) + (more-filters (cdr filters))) + (if filter-data + (let* ((proc (cadr filter-data)) + (fun (car proc)) + (filter-output (funcall fun input proc))) + (if more-filters + (emmet-process-filter more-filters filter-output) + filter-output)) + nil))) + +(defun emmet-make-tag (tag-maker tag-info &optional content) + "Extract tag info and pass them to tag-maker." + (let* ((name (pop tag-info)) + (has-body? (pop tag-info)) + (id (pop tag-info)) + (classes (pop tag-info)) + (props (pop tag-info)) + (txt (pop tag-info)) + (settings (gethash name emmet-tag-settings-table)) + (self-closing? + (and (not (or txt content)) + (or (not has-body?) + (and settings (gethash "selfClosing" settings)))))) + (funcall tag-maker name has-body? id classes props txt settings + (if content content + (if (and emmet-leaf-function (not self-closing?)) + (funcall emmet-leaf-function)))))) + +(defun emmet-hash-to-list (hash &optional proc) + (unless proc (setq proc #'cons)) + (cl-loop for key being the hash-keys of hash using (hash-values val) + collect (funcall proc key val))) + +(defun emmet-merge-tag-props (default-table tag-props) + (if default-table + (let ((tbl (copy-hash-table default-table))) + (cl-loop for prop in tag-props do + (puthash (symbol-name (car prop)) (cadr prop) tbl)) + (emmet-hash-to-list tbl 'list)) + tag-props)) + +(defun emmet-html-snippets-instantiate-lambda (src) + (let ((lines (mapcar + #'(lambda (src) + (if (string-match "^\\(.*\\)${child}\\(.*\\)$" src) + (mapcar (lambda (i) + (match-string i src)) + '(1 2)) + (list src))) + (split-string src "\n")))) + (cl-labels + ((iter + (l m a b) + (if l + (if (< 1 (length (car l))) + (iter (cdr l) + 'b + (cons (caar l) a) + (cons (cadar l) b)) + (if (eql m 'a) + (iter (cdr l) m (cons (caar l) a) b) + (iter (cdr l) m a (cons (caar l) b)))) + (if b + `(lambda (contents) + (concat + ,(emmet-join-string (reverse a) "\n") + contents + ,(emmet-join-string (reverse b) "\n"))) + `(lambda (contents) + (concat + ,(emmet-join-string (reverse a) "\n") + contents)))))) + (eval (iter lines 'a nil nil))))) + +(defun emmet-jsx-supported-mode? () + "Is the current mode we're on enabled for jsx class attribute expansion?" + (member major-mode emmet-jsx-major-modes)) + +(defun emmet-make-html-tag (tag-name tag-has-body? tag-id tag-classes tag-props tag-txt settings content) + "Create HTML markup string" + (emmet-aif + (gethash tag-name emmet-tag-snippets-table) + + (let ((fn (if (stringp it) + (emmet-html-snippets-instantiate-lambda it) + it))) + (prog1 + (funcall fn content) + (puthash tag-name fn emmet-tag-snippets-table))) + + (let* ((id (emmet-concat-or-empty " id=\"" tag-id "\"")) + (class-attr (if (emmet-jsx-supported-mode?) + (if emmet-jsx-className-braces? + " className={" + " className=\"") + " class=\"")) + (class-list-closer (if emmet-jsx-className-braces? "}" "\"")) + (class-list-delimiter (if emmet-jsx-className-braces? "." " ")) + (classes (emmet-mapconcat-or-empty class-attr tag-classes class-list-delimiter class-list-closer)) + (props (let* ((tag-props-default + (and settings (gethash "defaultAttr" settings))) + (merged-tag-props + (emmet-merge-tag-props + tag-props-default + tag-props))) + (emmet-mapconcat-or-empty + " " merged-tag-props " " nil + (lambda (prop) + (let* ((key-raw (car prop)) + (key-str + (if (symbolp key-raw) + (symbol-name key-raw) + key-raw)) + (key + (if (and (emmet-jsx-supported-mode?) + (string= key-str "for")) + "htmlFor" + key-str)) + (val (cadr prop)) + (format-string + (if (and (emmet-jsx-supported-mode?) + (emmet-jsx-prop-value-var? val)) + "%s=%s" + "%s=\"%s\""))) + (if val (format format-string key val) key)))))) + (content-multiline? (and content (string-match "\n" content))) + (block-tag? (and settings (gethash "block" settings))) + (self-closing? (and (not (or tag-txt content)) + (or (not tag-has-body?) + (and settings (gethash "selfClosing" settings))))) + (block-indentation? (or content-multiline? (and block-tag? content))) + (lf (if block-indentation? "\n"))) + (concat "<" tag-name id classes props + (if self-closing? + (concat emmet-self-closing-tag-style ">") + (concat ">" + (if tag-txt + (if block-indentation? + (emmet-indent tag-txt) + tag-txt)) + (if content + (if block-indentation? + (emmet-indent content) + content)) + lf + "</" tag-name ">")))))) + +(defun emmet-make-commented-html-tag (tag-name tag-has-body? tag-id tag-classes tag-props tag-txt settings content) + "Create HTML markup string with extra comments for elements with #id or .classes" + (let ((body (emmet-make-html-tag tag-name tag-has-body? tag-id tag-classes tag-props tag-txt settings content))) + (if (or tag-id tag-classes) + (let ((id (emmet-concat-or-empty "#" tag-id)) + (classes (emmet-mapconcat-or-empty "." tag-classes "."))) + (concat "<!-- " id classes " -->\n" + body + "\n<!-- /" id classes " -->")) + body))) + +(defun emmet-make-haml-tag (tag-name tag-has-body? tag-id tag-classes tag-props tag-txt settings content) + "Create HAML string" + (let ((name (if (and (equal tag-name "div") + (or tag-id tag-classes)) + "" + (concat "%" tag-name))) + (id (emmet-concat-or-empty "#" tag-id)) + (classes (emmet-mapconcat-or-empty "." tag-classes ".")) + (props (emmet-mapconcat-or-empty + "{" tag-props ", " "}" + (lambda (prop) + (concat ":" (symbol-name (car prop)) " => \"" (cadr prop) "\""))))) + (concat name id classes props + (if tag-txt + (emmet-indent tag-txt)) + (if content + (emmet-indent content))))) + +(defun emmet-make-hiccup-tag (tag-name tag-has-body? tag-id tag-classes tag-props tag-txt settings content) + "Create Hiccup string" + (let* ((id (emmet-concat-or-empty "#" tag-id)) + (classes (emmet-mapconcat-or-empty "." tag-classes ".")) + (props (emmet-mapconcat-or-empty + " {" tag-props ", " "}" + (lambda (prop) + (concat ":" (symbol-name (car prop)) " \"" (cadr prop) "\"")))) + (content-multiline? (and content (string-match "\n" content))) + (block-tag? (and settings (gethash "block" settings))) + (block-indentation? (or content-multiline? (and block-tag? content)))) + (concat "[:" tag-name id classes props + (if tag-txt + (let ((tag-txt-quoted (concat "\"" tag-txt "\""))) + (if block-indentation? + (emmet-indent tag-txt-quoted) + (concat " " tag-txt-quoted)))) + (if content + (if block-indentation? + (emmet-indent content) + (concat " " content))) + "]"))) + +(defun emmet-make-text (tag-maker text) + (cond + ((eq tag-maker 'emmet-make-hiccup-tag) + (concat "\"" text "\"")) + (t text))) + +(defun emmet-concat-or-empty (prefix body &optional suffix) + "Return prefixed suffixed text or empty string." + (if body + (concat prefix body suffix) + "")) + +(defun emmet-mapconcat-or-empty (prefix list-body delimiter &optional suffix map-fun) + "Return prefixed suffixed mapconcated text or empty string." + (if list-body + (let* ((mapper (if map-fun map-fun 'identity)) + (body (mapconcat mapper list-body delimiter))) + (concat prefix body suffix)) + "")) + +(defun emmet-escape-xml (input proc) + "Escapes XML-unsafe characters: <, > and &." + (replace-regexp-in-string + "<" "<" + (replace-regexp-in-string + ">" ">" + (replace-regexp-in-string + "&" "&" + (if (stringp input) + input + (emmet-process-filter (emmet-default-filter) input)))))) + +(defun emmet-html-transform (input) + (let ((ast (car (emmet-expr input)))) + (when (not (eq ast 'error)) + (emmet-transform-ast-with-filters ast)))) + +(defun emmet-transform-ast-with-filters (ast-with-filters) + "Transform AST (containing filter data) into string." + (let ((filters (cadr ast-with-filters)) + (ast (caddr ast-with-filters))) + (emmet-process-filter filters ast))) + +(defun emmet-transform-ast (ast tag-maker) + "Transform AST (without filter data) into string." + (let ((type (car ast))) + (cond + ((eq type 'list) + (mapconcat (lexical-let ((make-tag-fun tag-maker)) + #'(lambda (sub-ast) + (emmet-transform-ast sub-ast make-tag-fun))) + (cadr ast) + "\n")) + ((eq type 'tag) + (emmet-make-tag tag-maker (cadr ast))) + ((eq type 'text) + (emmet-make-text tag-maker (cadr ast))) + ((eq type 'parent-child) + (let ((parent (cadadr ast)) + (children (emmet-transform-ast (caddr ast) tag-maker))) + (emmet-make-tag tag-maker parent children))) + ((eq type 'sibling) + (let ((sib1 (emmet-transform-ast (cadr ast) tag-maker)) + (sib2 (emmet-transform-ast (caddr ast) tag-maker))) + (concat sib1 "\n" sib2)))))) + +;; Indents text rigidly by inserting spaces +;; Only matters if emmet-indent-after-insert is set to nil +(defun emmet-indent (text) + "Indent the text" + (if text + (replace-regexp-in-string "\n" (concat "\n" (make-string emmet-indentation ?\ )) (concat "\n" text)) + nil)) +(defvar emmet-lorem-words + '("lorem" "ipsum" "dolor" "sit" "amet," "consectetur" "adipiscing" "elit" "ut" "aliquam," "purus" "sit" "amet" "luctus" "venenatis," + "lectus" "magna" "fringilla" "urna," "porttitor" "rhoncus" "dolor" "purus" "non" "enim" "praesent" "elementum" "facilisis" "leo," + "vel" "fringilla" "est" "ullamcorper" "eget" "nulla" "facilisi" "etiam" "dignissim" "diam" "quis" "enim" "lobortis" "scelerisque" + "fermentum" "dui" "faucibus" "in" "ornare" "quam" "viverra" "orci" "sagittis" "eu" "volutpat" "odio" "facilisis" "mauris" "sit" "amet" + "massa" "vitae" "tortor" "condimentum" "lacinia" "quis" "vel" "eros" "donec" "ac" "odio" "tempor" "orci" "dapibus" "ultrices" "in" + "iaculis" "nunc" "sed" "augue" "lacus," "viverra" "vitae" "congue" "eu," "consequat" "ac" "felis" "donec" "et" "odio" "pellentesque" + "diam" "volutpat" "commodo" "sed" "egestas" "egestas" "fringilla" "phasellus" "faucibus" "scelerisque" "eleifend" "donec" "pretium" + "vulputate" "sapien" "nec" "sagittis" "aliquam" "malesuada" "bibendum" "arcu" "vitae" "elementum" "curabitur" "vitae" "nunc" "sed" + "velit" "dignissim" "sodales" "ut" "eu" "sem" "integer" "vitae" "justo" "eget" "magna" "fermentum" "iaculis" "eu" "non" "diam" + "phasellus" "vestibulum" "lorem" "sed" "risus" "ultricies" "tristique" "nulla" "aliquet" "enim" "tortor," "at" "auctor" "urna" "nunc" + "id" "cursus" "metus" "aliquam" "eleifend" "mi" "in" "nulla" "posuere" "sollicitudin" "aliquam" "ultrices" "sagittis" "orci," "a" + "scelerisque" "purus" "semper" "eget" "duis" "at" "tellus" "at" "urna" "condimentum" "mattis" "pellentesque" "id" "nibh" "tortor," + "id" "aliquet" "lectus" "proin" "nibh" "nisl," "condimentum" "id" "venenatis" "a," "condimentum" "vitae" "sapien" "pellentesque" + "habitant" "morbi" "tristique" "senectus" "et" "netus" "et" "malesuada" "fames" "ac" "turpis" "egestas" "sed" "tempus," "urna" "et" + "pharetra" "pharetra," "massa" "massa" "ultricies" "mi," "quis" "hendrerit" "dolor" "magna" "eget" "est" "lorem" "ipsum" "dolor" "sit" + "amet," "consectetur" "adipiscing" "elit" "pellentesque" "habitant" "morbi" "tristique" "senectus" "et" "netus" "et" "malesuada" "fames" + "ac" "turpis" "egestas" "integer" "eget" "aliquet" "nibh" "praesent" "tristique" "magna" "sit" "amet" "purus" "gravida" "quis" "blandit" + "turpis" "cursus" "in" "hac" "habitasse" "platea" "dictumst" "quisque" "sagittis," "purus" "sit" "amet" "volutpat" "consequat," "mauris" + "nunc" "congue" "nisi," "vitae" "suscipit" "tellus" "mauris" "a" "diam" "maecenas" "sed" "enim" "ut" "sem" "viverra" "aliquet" "eget" + "sit" "amet" "tellus" "cras" "adipiscing" "enim" "eu" "turpis" "egestas" "pretium" "aenean" "pharetra," "magna" "ac" "placerat" + "vestibulum," "lectus" "mauris" "ultrices" "eros," "in" "cursus" "turpis" "massa" "tincidunt" "dui" "ut" "ornare" "lectus" "sit" "amet" + "est" "placerat" "in" "egestas" "erat" "imperdiet" "sed" "euismod" "nisi" "porta" "lorem" "mollis" "aliquam" "ut" "porttitor" "leo" "a" + "diam" "sollicitudin" "tempor" "id" "eu" "nisl" "nunc" "mi" "ipsum," "faucibus" "vitae" "aliquet" "nec," "ullamcorper" "sit" "amet" + "risus" "nullam" "eget" "felis" "eget" "nunc" "lobortis" "mattis" "aliquam" "faucibus" "purus" "in" "massa" "tempor" "nec" "feugiat" + "nisl" "pretium" "fusce" "id" "velit" "ut" "tortor" "pretium" "viverra" "suspendisse" "potenti" "nullam" "ac" "tortor" "vitae" "purus" + "faucibus" "ornare" "suspendisse" "sed" "nisi" "lacus," "sed" "viverra" "tellus" "in" "hac" "habitasse" "platea" "dictumst" "vestibulum" + "rhoncus" "est" "pellentesque" "elit" "ullamcorper" "dignissim" "cras" "tincidunt" "lobortis" "feugiat" "vivamus" "at" "augue" "eget" + "arcu" "dictum" "varius" "duis" "at" "consectetur" "lorem" "donec" "massa" "sapien," "faucibus" "et" "molestie" "ac," "feugiat" "sed" + "lectus" "vestibulum" "mattis" "ullamcorper" "velit" "sed" "ullamcorper" "morbi" "tincidunt" "ornare" "massa," "eget" "egestas" "purus" + "viverra" "accumsan" "in" "nisl" "nisi," "scelerisque" "eu" "ultrices" "vitae," "auctor" "eu" "augue" "ut" "lectus" "arcu," "bibendum" + "at" "varius" "vel," "pharetra" "vel" "turpis" "nunc" "eget" "lorem" "dolor," "sed" "viverra" "ipsum" "nunc" "aliquet" "bibendum" "enim," + "facilisis" "gravida" "neque" "convallis" "a" "cras" "semper" "auctor" "neque," "vitae" "tempus" "quam" "pellentesque" "nec" "nam" + "aliquam" "sem" "et" "tortor" "consequat" "id" "porta" "nibh" "venenatis" "cras" "sed" "felis" "eget" "velit" "aliquet" "sagittis" + "id" "consectetur" "purus" "ut" "faucibus" "pulvinar" "elementum" "integer" "enim" "neque," "volutpat" "ac" "tincidunt" "vitae," + "semper" "quis" "lectus" "nulla" "at" "volutpat" "diam" "ut" "venenatis" "tellus" "in" "metus" "vulputate" "eu" "scelerisque" "felis" + "imperdiet" "proin" "fermentum" "leo" "vel" "orci" "porta" "non" "pulvinar" "neque" "laoreet" "suspendisse" "interdum" "consectetur" + "libero," "id" "faucibus" "nisl" "tincidunt" "eget" "nullam" "non" "nisi" "est," "sit" "amet" "facilisis" "magna" "etiam" "tempor," + "orci" "eu" "lobortis" "elementum," "nibh" "tellus" "molestie" "nunc," "non" "blandit" "massa" "enim" "nec" "dui" "nunc" "mattis" + "enim" "ut" "tellus" "elementum" "sagittis" "vitae" "et" "leo" "duis" "ut" "diam" "quam" "nulla" "porttitor" "massa" "id" "neque" + "aliquam" "vestibulum" "morbi" "blandit" "cursus" "risus," "at" "ultrices" "mi" "tempus" "imperdiet" "nulla" "malesuada" "pellentesque" + "elit" "eget" "gravida" "cum" "sociis" "natoque" "penatibus" "et" "magnis" "dis" "parturient" "montes," "nascetur" "ridiculus" "mus" + "mauris" "vitae" "ultricies" "leo" "integer" "malesuada" "nunc" "vel" "risus" "commodo" "viverra" "maecenas" "accumsan," "lacus" "vel" + "facilisis" "volutpat," "est" "velit" "egestas" "dui," "id" "ornare" "arcu" "odio" "ut" "sem" "nulla" "pharetra" "diam" "sit" "amet" + "nisl" "suscipit" "adipiscing" "bibendum" "est" "ultricies" "integer" "quis" "auctor" "elit" "sed" "vulputate" "mi" "sit" "amet" "mauris" + "commodo" "quis" "imperdiet" "massa" "tincidunt" "nunc" "pulvinar" "sapien" "et" "ligula" "ullamcorper" "malesuada" "proin" "libero" + "nunc," "consequat" "interdum" "varius" "sit" "amet," "mattis" "vulputate" "enim" "nulla" "aliquet" "porttitor" "lacus," "luctus" + "accumsan" "tortor" "posuere" "ac" "ut" "consequat" "semper" "viverra" "nam" "libero" "justo," "laoreet" "sit" "amet" "cursus" "sit" + "amet," "dictum" "sit" "amet" "justo" "donec" "enim" "diam," "vulputate" "ut" "pharetra" "sit" "amet," "aliquam" "id" "diam" "maecenas" + "ultricies" "mi" "eget" "mauris" "pharetra" "et" "ultrices" "neque" "ornare" "aenean" "euismod" "elementum" "nisi," "quis" "eleifend" + "quam" "adipiscing" "vitae" "proin" "sagittis," "nisl" "rhoncus" "mattis" "rhoncus," "urna" "neque" "viverra" "justo," "nec" "ultrices" + "dui" "sapien" "eget" "mi" "proin" "sed" "libero" "enim," "sed" "faucibus" "turpis" "in" "eu" "mi" "bibendum" "neque" "egestas" "congue" + "quisque" "egestas" "diam" "in" "arcu" "cursus" "euismod" "quis" "viverra" "nibh" "cras" "pulvinar" "mattis" "nunc," "sed" "blandit" + "libero" "volutpat" "sed" "cras" "ornare" "arcu" "dui" "vivamus" "arcu" "felis," "bibendum" "ut" "tristique" "et," "egestas" "quis" + "ipsum" "suspendisse" "ultrices" "gravida" "dictum" "fusce" "ut" "placerat" "orci" "nulla" "pellentesque" "dignissim" "enim," "sit" + "amet" "venenatis" "urna" "cursus" "eget" "nunc" "scelerisque" "viverra" "mauris," "in" "aliquam" "sem" "fringilla" "ut" "morbi" + "tincidunt" "augue" "interdum" "velit" "euismod" "in" "pellentesque" "massa" "placerat" "duis" "ultricies" "lacus" "sed" "turpis" + "tincidunt" "id" "aliquet" "risus" "feugiat" "in" "ante" "metus," "dictum" "at" "tempor" "commodo," "ullamcorper" "a" "lacus" "vestibulum" + "sed" "arcu" "non" "odio" "euismod" "lacinia" "at" "quis" "risus" "sed" "vulputate" "odio" "ut" "enim" "blandit" "volutpat" "maecenas" + "volutpat" "blandit" "aliquam" "etiam" "erat" "velit," "scelerisque" "in" "dictum" "non," "consectetur" "a" "erat" "nam" "at" "lectus" + "urna" "duis" "convallis" "convallis" "tellus," "id" "interdum" "velit" "laoreet" "id" "donec" "ultrices" "tincidunt" "arcu," "non" + "sodales" "neque" "sodales" "ut" "etiam" "sit" "amet" "nisl" "purus," "in" "mollis" "nunc" "sed" "id" "semper" "risus" "in" "hendrerit" + "gravida" "rutrum" "quisque" "non" "tellus" "orci," "ac" "auctor" "augue" "mauris" "augue" "neque," "gravida" "in" "fermentum" "et," + "sollicitudin" "ac" "orci" "phasellus" "egestas" "tellus" "rutrum" "tellus" "pellentesque" "eu" "tincidunt" "tortor" "aliquam" "nulla" + "facilisi" "cras" "fermentum," "odio" "eu" "feugiat" "pretium," "nibh" "ipsum" "consequat" "nisl," "vel" "pretium" "lectus" "quam" "id" + "leo" "in" "vitae" "turpis" "massa" "sed" "elementum" "tempus" "egestas" "sed" "sed" "risus" "pretium" "quam" "vulputate" "dignissim" + "suspendisse" "in" "est" "ante" "in" "nibh" "mauris," "cursus" "mattis" "molestie" "a," "iaculis" "at" "erat" "pellentesque" "adipiscing" + "commodo" "elit," "at" "imperdiet" "dui" "accumsan" "sit" "amet" "nulla" "facilisi" "morbi" "tempus" "iaculis" "urna," "id" "volutpat" + "lacus" "laoreet" "non" "curabitur" "gravida" "arcu" "ac" "tortor" "dignissim" "convallis" "aenean" "et" "tortor" "at" "risus" "viverra" + "adipiscing" "at" "in" "tellus" "integer" "feugiat" "scelerisque" "varius" "morbi" "enim" "nunc," "faucibus" "a" "pellentesque" "sit" + "amet," "porttitor" "eget" "dolor" "morbi" "non" "arcu" "risus," "quis" "varius" "quam" "quisque" "id" "diam" "vel" "quam" "elementum" + "pulvinar" "etiam" "non" "quam" "lacus" "suspendisse" "faucibus" "interdum" "posuere" "lorem" "ipsum" "dolor" "sit" "amet," "consectetur" + "adipiscing" "elit" "duis" "tristique" "sollicitudin" "nibh" "sit" "amet" "commodo" "nulla" "facilisi" "nullam" "vehicula" "ipsum" "a" + "arcu" "cursus" "vitae" "congue" "mauris" "rhoncus" "aenean" "vel" "elit" "scelerisque" "mauris" "pellentesque" "pulvinar" "pellentesque" + "habitant" "morbi" "tristique" "senectus" "et" "netus" "et" "malesuada" "fames" "ac" "turpis" "egestas" "maecenas" "pharetra" "convallis" + "posuere" "morbi" "leo" "urna," "molestie" "at" "elementum" "eu," "facilisis" "sed" "odio" "morbi" "quis" "commodo" "odio" "aenean" "sed" + "adipiscing" "diam" "donec" "adipiscing" "tristique" "risus" "nec" "feugiat" "in" "fermentum" "posuere" "urna" "nec" "tincidunt" "praesent" + "semper" "feugiat" "nibh" "sed" "pulvinar" "proin" "gravida" "hendrerit" "lectus" "a" "molestie")) + +(defun emmet-random-range (min max) + (+ min (random (+ (- max min) 1)))) + +(defun emmet-lorem-choice-words (count &optional s) + (let* ((l (length emmet-lorem-words)) + (s (if s s (random l))) + (f (+ s count)) + (e (if (< l f) l f))) + (append + (cl-subseq emmet-lorem-words s e) + (if (= e l) (emmet-lorem-choice-words (- f l) 0))))) + +(defvar emmet-lorem-min-sentence 5) + +(defvar emmet-lorem-max-sentence 30) + +(defun emmet-upcase-first (s) + (concat (upcase (cl-subseq s 0 1)) (cl-subseq s 1))) + +(defun emmet-lorem-generate (count) + (if (<= count 0) "" + (let ((sl (if (< count emmet-lorem-max-sentence) count + (emmet-random-range + emmet-lorem-min-sentence + (min (- count emmet-lorem-min-sentence) + emmet-lorem-max-sentence)))) + (last (let ((r (random 4))) + (if (< 1 r) "." (if (< 0 r) "?" "!"))))) + (let ((words (let ((w (emmet-lorem-choice-words sl))) + (let ((l (car (last w)))) + (if (string-equal (substring l -1) ",") + (append (cl-subseq w 0 -1) (list (substring l 0 -1))) + w))))) + (concat (emmet-upcase-first (emmet-join-string words " ")) last + (let ((next (emmet-lorem-generate (- count sl)))) + (if (string-equal next "") "" + (concat " " next)))))))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; +;;; CSS abbrev: + +(emmet-defparameter + emmet-css-unit-aliases + (gethash "unitAliases" (gethash "css" emmet-preferences))) +(defun emmet-css-arg-number (input) + (emmet-parse + " *\\(\\(?:-\\|\\)[0-9.]+\\)\\(-\\|[A-Za-z]*\\)" 3 "css number arguments" + (cons (list (elt it 1) + (let ((unit (elt it 2))) + (if (= (length unit) 0) + (if (cl-find ?. (elt it 1)) "em" "px") + (gethash unit emmet-css-unit-aliases unit)))) + input))) + +(emmet-defparameter + emmet-css-color-shorten-if-possible + (gethash "shortenIfPossible" (gethash "color" (gethash "css" emmet-preferences)))) +(emmet-defparameter + emmet-css-color-case + (gethash "case" (gethash "color" (gethash "css" emmet-preferences)))) +(emmet-defparameter + emmet-css-color-trailing-aliases + (gethash "trailingAliases" (gethash "color" (gethash "css" emmet-preferences)))) +(defun emmet-css-arg-color (input) + (emmet-parse + (concat " *#\\([0-9a-fA-F]\\{1,6\\}\\)\\(rgb\\|\\)\\([" + (emmet-join-string + (emmet-get-keys-of-hash emmet-css-color-trailing-aliases) "") + "]\\|\\)") + 4 "css color argument" + (let ((color + (let* ((n (elt it 1)) + (l (length n))) + (substring + (cond ((= l 1) (concat (make-list 6 (string-to-char n)))) + ((= l 2) (concat n n n)) + ((= l 3) (concat + (cl-loop for c in (string-to-list n) + append (list c c)))) + (t (concat n n))) + 0 6)))) + (cons + (let ((rgb-mode (string= (elt it 2) "rgb"))) + (if rgb-mode + (format "rgb(%d,%d,%d)" + (string-to-number (substring color 0 2) 16) + (string-to-number (substring color 2 4) 16) + (string-to-number (substring color 4 6) 16)) + (concat + "#" + (let ((filter (cond ((string= emmet-css-color-case "auto") #'identity) + ((string= emmet-css-color-case "up") #'upcase) + (t #'downcase)))) + (funcall + filter + (if (and emmet-css-color-shorten-if-possible + (eql (aref color 0) (aref color 1)) + (eql (aref color 2) (aref color 3)) + (eql (aref color 4) (aref color 5))) + (concat (mapcar #'(lambda (i) (aref color i)) '(0 2 4))) + color)))))) + (if (< 0 (length (elt it 3))) + (cons (gethash (elt it 3) emmet-css-color-trailing-aliases) input) + input))))) + +(defun emmet-css-arg-something (input) + (emmet-parse + " *\\([^ ]+\\)" 2 "css argument" + (cons (elt it 1) input))) + +(defun emmet-css-parse-arg (input) + (emmet-run emmet-css-arg-number it + (emmet-run emmet-css-arg-color it + (emmet-run emmet-css-arg-something it + (if (equal input "") + it + (cons input "")))))) + +(defun emmet-css-important-p (input) + (let ((len (length input))) + (and (< 0 len) + (char-equal (aref input (1- len)) ?!)))) + +(defun emmet-css-parse-args (args) + (when args + (let ((rt nil)) + (cl-loop + (emmet-pif + (emmet-css-parse-arg args) + (cl-loop for i on it do (push (car i) rt) + while (consp (cdr i)) + finally (setq args (cdr i))) + (cl-return (nreverse rt))))))) + +(defun emmet-css-split-args (exp) + (emmet-aif + (string-match "\\(?:[ #0-9$]\\|-[0-9]\\)" exp) + (list (substring exp 0 it) (substring exp it)) + (list exp nil))) + +(defun emmet-css-split-vendor-prefixes (input) + (emmet-parse + "\\(-[wmso]+-\\|-\\|\\)\\(.*\\)" 3 "css vendor prefixes" + (list (elt it 2) + (let ((vp (elt it 1))) + (if (not (string= vp "")) + (if (string= vp "-") 'auto + (string-to-list (cl-subseq vp 1 -1)))))))) + +(defun emmet-css-subexpr (exp) + (let* ((importantp (emmet-css-important-p exp))) + (cl-destructuring-bind (exp vp) + (emmet-css-split-vendor-prefixes exp) + (cl-destructuring-bind (key args) + (emmet-css-split-args (if importantp (cl-subseq exp 0 -1) exp)) + `(,key ,vp + ,importantp + ,@(emmet-css-parse-args args)))))) + +(defun emmet-css-toknize (str) + (let* ((i (split-string str "+")) + (rt nil)) + (cl-loop + (let ((f (first i)) + (s (second i))) + (if f + (if (and s (or (string= s "") + (string-match "^\\(?:[ #0-9$]\\|-[0-9]\\)" s))) + (progn + (setf rt (cons (concat f "+" s) rt)) + (setf i (cddr i))) + (progn + (setf rt (cons f rt)) + (setf i (cdr i)))) + (cl-return (nreverse rt))))))) + +(defun emmet-css-expr (input) + (mapcar #'emmet-css-subexpr + (emmet-css-toknize input))) + +(emmet-defparameter + emmet-css-snippets + (gethash "snippets" (gethash "css" emmet-snippets))) + +(emmet-defparameter + emmet-sass-snippets + (gethash "snippets" (gethash "sass" emmet-snippets))) + +(emmet-defparameter + emmet-css-unitless-properties + (gethash "unitlessProperties" (gethash "css" emmet-preferences))) + +(emmet-defparameter + emmet-css-unitless-properties-regex + (concat "^\\(:?" (emmet-join-string + emmet-css-unitless-properties "\\|") + "\\):.*$")) + +(defun emmet-css-instantiate-lambda (str) + (cl-flet ((insert-space-between-name-and-body + (str) + (if (string-match "^\\([a-z-]+:\\)\\(.+\\)$" str) + (emmet-join-string + (mapcar (lambda (ref) (match-string ref str)) '(1 2)) " ") + str)) + (split-string-to-body + (str args-sym) + (let ((rt '(concat)) (idx-max 0)) + (cl-loop for i from 0 to 255 do + (emmet-aif + (string-match "\\(?:|\\|${\\(?:\\([0-9]\\)\\|\\)\\(?::\\(.+?\\)\\|\\)}\\)" str) + (cl-destructuring-bind (mat idx def) + (mapcar (lambda (ref) (match-string ref str)) '(0 1 2)) + (setf rt + `((or + (nth ,(let ((cur-idx (if idx (1- (string-to-number idx)) i))) + (setf idx-max (max cur-idx idx-max))) + ,args-sym) + ,(or def "")) + ,(substring str 0 it) ;; ordered to reverse + ,@rt)) + (setf str (substring str (+ it (length mat))))) + ;; don't use nreverse. cause bug in emacs-lisp. + (cl-return (cons idx-max (reverse (cons str rt))))))))) + (let ((args (gensym)) + (str (insert-space-between-name-and-body str))) + (cl-destructuring-bind (idx-max . body) (split-string-to-body str args) + (eval + `(lambda (&rest ,args) + (progn + (when (nthcdr ,idx-max ,args) + (setf (nthcdr ,idx-max ,args) + (list (emmet-join-string + (nthcdr ,idx-max ,args) " ")))) + ,body))))))) + +(emmet-defparameter + emmet-vendor-prefixes-properties + (gethash "vendorPrefixesProperties" (gethash "css" emmet-preferences))) +(emmet-defparameter + emmet-vendor-prefixes-default + (list "webkit" "moz" "ms" "o")) +(defun emmet-css-transform-vendor-prefixes (line vp) + (let ((key (cl-subseq line 0 (or (cl-position ?: line) (length line))))) + (let ((vps (if (eql vp 'auto) + (gethash key + emmet-vendor-prefixes-properties + emmet-vendor-prefixes-default) + (mapcar (lambda (v) + (cond ((= v ?w) "webkit") + ((= v ?m) "moz") + ((= v ?s) "ms") + ((= v ?o) "o"))) + vp)))) + (emmet-join-string + (append (mapcar (lambda (v) (concat "-" v "-" line)) vps) + (list line)) + "\n")))) + +(defun emmet-css-transform-exprs (exprs) + (emmet-join-string + (mapcar + #'(lambda (expr) + (let* + ((hash-map (if emmet-use-sass-syntax emmet-sass-snippets emmet-css-snippets)) + (basement + (emmet-aif + (or (gethash (car expr) hash-map) (gethash (car expr) emmet-css-snippets)) + (let ((set it) (fn nil) (unitlessp nil)) + (if (stringp set) + (progn + ;; new pattern + ;; creating print function + (setf fn (emmet-css-instantiate-lambda set)) + ;; get unitless or no + (setf unitlessp + (not (null (string-match + emmet-css-unitless-properties-regex set)))) + ;; caching + (puthash (car expr) (cons fn unitlessp) hash-map)) + (progn + ;; cache hit. + (setf fn (car set)) + (setf unitlessp (cdr set)))) + (apply fn + (mapcar + #'(lambda (arg) + (if (listp arg) + (if unitlessp (car arg) + (apply #'concat arg)) + arg)) + (cdddr expr)))) + (concat (car expr) ": " + (emmet-join-string + (mapcar #'(lambda (arg) + (if (listp arg) (apply #'concat arg) arg)) + (cdddr expr)) " ") + ";")))) + (let ((line + (if (caddr expr) + (concat (cl-subseq basement 0 -1) " !important;") + basement))) + ;; remove trailing semicolon while editing Sass files + (if (and emmet-use-sass-syntax (equal ";" (cl-subseq line -1))) + (setq line (cl-subseq line 0 -1))) + (emmet-aif + (cadr expr) + (emmet-css-transform-vendor-prefixes line it) + line)))) + exprs) + "\n")) + +(defun emmet-css-transform (input) + (emmet-css-transform-exprs (emmet-css-expr input))) + +;;; emmet-mode.el ends here |