522 lines
18 KiB
YAML
522 lines
18 KiB
YAML
%YAML 1.2
|
|
---
|
|
# http://www.sublimetext.com/docs/3/syntax.html
|
|
name: Elixir
|
|
comment: Textmate bundle for Elixir Programming Language.
|
|
file_extensions:
|
|
- ex
|
|
- exs
|
|
first_line_match: ^#!/.*\belixir
|
|
scope: source.elixir
|
|
contexts:
|
|
main:
|
|
- match: \b(fn)\b(?!.*->)
|
|
captures:
|
|
1: keyword.control.elixir
|
|
push:
|
|
- match: $
|
|
pop: true
|
|
- include: core_syntax
|
|
- match: \b(fn)\b(?=.*->)
|
|
captures:
|
|
1: keyword.control.elixir
|
|
push:
|
|
- match: (?>(->)|(when)|(\)))
|
|
captures:
|
|
1: keyword.operator.other.elixir
|
|
2: keyword.control.elixir
|
|
3: punctuation.section.function.elixir
|
|
pop: true
|
|
- include: core_syntax
|
|
- include: core_syntax
|
|
- match: '^(?=.*->)((?![^"'']*("|'')[^"'']*->)|(?=.*->[^"'']*("|'')[^"'']*->))((?!.*\([^\)]*->)|(?=[^\(\)]*->)|(?=\s*\(.*\).*->))((?!.*\b(fn)\b)|(?=.*->.*\bfn\b))'
|
|
captures:
|
|
1: keyword.control.elixir
|
|
push:
|
|
- match: (?>(->)|(when)|(\)))
|
|
captures:
|
|
1: keyword.operator.other.elixir
|
|
2: keyword.control.elixir
|
|
3: punctuation.section.function.elixir
|
|
pop: true
|
|
- include: core_syntax
|
|
core_syntax:
|
|
- match: ^\s*(defmodule)\b
|
|
captures:
|
|
1: keyword.control.module.elixir
|
|
push:
|
|
- meta_scope: meta.module.elixir
|
|
- match: \b(do)\b
|
|
captures:
|
|
1: keyword.control.module.elixir
|
|
pop: true
|
|
- match: '\b[A-Z]\w*\b'
|
|
scope: entity.name.class.elixir
|
|
- match: ^\s*(defprotocol)\b
|
|
captures:
|
|
1: keyword.control.protocol.elixir
|
|
push:
|
|
- meta_scope: meta.protocol_declaration.elixir
|
|
- match: \b(do)\b
|
|
captures:
|
|
1: keyword.control.protocol.elixir
|
|
pop: true
|
|
- match: '\b[A-Z]\w*\b'
|
|
scope: entity.name.protocol.elixir
|
|
- match: ^\s*(defimpl)\b
|
|
captures:
|
|
1: keyword.control.protocol.elixir
|
|
push:
|
|
- meta_scope: meta.protocol_implementation.elixir
|
|
- match: \b(do)\b
|
|
captures:
|
|
1: keyword.control.protocol.elixir
|
|
pop: true
|
|
- match: '\b[A-Z]\w*\b'
|
|
scope: entity.name.protocol.elixir
|
|
- match: '^\s*(def|defmacro)\s+((?>[a-zA-Z_]\w*(?>\.|::))?(?>[a-zA-Z_]\w*(?>[?!]|=(?!>))?|===?|>[>=]?|<=>|<[<=]?|[%&`/\|]|\*\*?|=?~|[-+]@?|\[\]=?))((\()|\s*)'
|
|
captures:
|
|
1: keyword.control.module.elixir
|
|
2: entity.name.function.public.elixir
|
|
4: punctuation.section.function.elixir
|
|
push:
|
|
- meta_scope: meta.function.public.elixir
|
|
- match: (\bdo:)|(\bdo\b)|(?=\s+(def|defmacro)\b)
|
|
captures:
|
|
1: constant.other.keywords.elixir
|
|
2: keyword.control.module.elixir
|
|
pop: true
|
|
- include: main
|
|
- match: \s(\\\\)
|
|
captures:
|
|
1: keyword.operator.other.elixir
|
|
push:
|
|
- match: ',|\)|$'
|
|
pop: true
|
|
- include: main
|
|
- match: \b(is_atom|is_binary|is_bitstring|is_boolean|is_float|is_function|is_integer|is_list|is_map|is_nil|is_number|is_pid|is_port|is_record|is_reference|is_tuple|is_exception|abs|bit_size|byte_size|div|elem|hd|length|map_size|node|rem|round|tl|trunc|tuple_size)\b
|
|
scope: keyword.control.elixir
|
|
- match: '^\s*(defp|defmacrop)\s+((?>[a-zA-Z_]\w*(?>\.|::))?(?>[a-zA-Z_]\w*(?>[?!]|=(?!>))?|===?|>[>=]?|<=>|<[<=]?|[%&`/\|]|\*\*?|=?~|[-+]@?|\[\]=?))((\()|\s*)'
|
|
captures:
|
|
1: keyword.control.module.elixir
|
|
2: entity.name.function.private.elixir
|
|
4: punctuation.section.function.elixir
|
|
push:
|
|
- meta_scope: meta.function.private.elixir
|
|
- match: (\bdo:)|(\bdo\b)|(?=\s+(defp|defmacrop)\b)
|
|
captures:
|
|
1: constant.other.keywords.elixir
|
|
2: keyword.control.module.elixir
|
|
pop: true
|
|
- include: main
|
|
- match: \s(\\\\)
|
|
captures:
|
|
1: keyword.operator.other.elixir
|
|
push:
|
|
- match: ',|\)|$'
|
|
pop: true
|
|
- include: main
|
|
- match: \b(is_atom|is_binary|is_bitstring|is_boolean|is_float|is_function|is_integer|is_list|is_map|is_nil|is_number|is_pid|is_port|is_record|is_reference|is_tuple|is_exception|abs|bit_size|byte_size|div|elem|hd|length|map_size|node|rem|round|tl|trunc|tuple_size)\b
|
|
scope: keyword.control.elixir
|
|
- match: '@(module|type)?doc (~[a-z])?"""'
|
|
comment: "@doc with heredocs is treated as documentation"
|
|
push:
|
|
- meta_scope: comment.documentation.heredoc
|
|
- match: \s*"""
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '@(module|type)?doc ~[A-Z]"""'
|
|
comment: "@doc with heredocs is treated as documentation"
|
|
push:
|
|
- meta_scope: comment.documentation.heredoc
|
|
- match: \s*"""
|
|
pop: true
|
|
- match: "@(module|type)?doc (~[a-z])?'''"
|
|
comment: "@doc with heredocs is treated as documentation"
|
|
push:
|
|
- meta_scope: comment.documentation.heredoc
|
|
- match: \s*'''
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: "@(module|type)?doc ~[A-Z]'''"
|
|
comment: "@doc with heredocs is treated as documentation"
|
|
push:
|
|
- meta_scope: comment.documentation.heredoc
|
|
- match: \s*'''
|
|
pop: true
|
|
- match: "@(module|type)?doc false"
|
|
comment: "@doc false is treated as documentation"
|
|
scope: comment.documentation.false
|
|
- match: '@(module|type)?doc "'
|
|
comment: "@doc with string is treated as documentation"
|
|
push:
|
|
- meta_scope: comment.documentation.string
|
|
- match: '"'
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '(?<!\.)\b(do|end|case|bc|lc|for|if|cond|unless|try|receive|fn|defmodule|defp?|defprotocol|defimpl|defrecord|defstruct|defmacrop?|defdelegate|defcallback|defmacrocallback|defexception|defoverridable|exit|after|rescue|catch|else|raise|throw|import|require|alias|use|quote|unquote|super|with)\b(?![?!:])'
|
|
scope: keyword.control.elixir
|
|
- match: (?<!\.)\b(and|not|or|when|xor|in)\b
|
|
comment: as above, just doesn't need a 'end' and does a logic operation
|
|
scope: keyword.operator.elixir
|
|
- match: '\b[A-Z]\w*\b'
|
|
scope: entity.name.class.elixir
|
|
- match: '\b(nil|true|false)\b(?![?!])'
|
|
scope: constant.language.elixir
|
|
- match: '\b(__(CALLER|ENV|MODULE|DIR)__)\b(?![?!])'
|
|
scope: variable.language.elixir
|
|
- match: '(@)[a-zA-Z_]\w*'
|
|
scope: variable.other.readwrite.module.elixir
|
|
captures:
|
|
1: punctuation.definition.variable.elixir
|
|
- match: (&)\d+
|
|
scope: variable.other.anonymous.elixir
|
|
captures:
|
|
1: punctuation.definition.variable.elixir
|
|
- match: '\^[a-z_]\w*'
|
|
scope: variable.other.capture.elixir
|
|
captures:
|
|
1: punctuation.definition.variable.elixir
|
|
- match: '\b(0x[0-9A-Fa-f](?>_?[0-9A-Fa-f])*|\d(?>_?\d)*(\.(?![^[:space:][:digit:]])(?>_?\d)*)?([eE][-+]?\d(?>_?\d)*)?|0b[01]+|0o[0-7]+)\b'
|
|
scope: constant.numeric.elixir
|
|
- match: ":'"
|
|
captures:
|
|
0: punctuation.definition.constant.elixir
|
|
push:
|
|
- meta_scope: constant.other.symbol.single-quoted.elixir
|
|
- match: "'"
|
|
captures:
|
|
0: punctuation.definition.constant.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: ':"'
|
|
captures:
|
|
0: punctuation.definition.constant.elixir
|
|
push:
|
|
- meta_scope: constant.other.symbol.double-quoted.elixir
|
|
- match: '"'
|
|
captures:
|
|
0: punctuation.definition.constant.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: (?>''')
|
|
comment: Single-quoted heredocs
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.single.heredoc.elixir
|
|
- match: ^\s*'''
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: "'"
|
|
comment: single quoted string (allows for interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.single.elixir
|
|
- match: "'"
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: (?>""")
|
|
comment: Double-quoted heredocs
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.double.heredoc.elixir
|
|
- match: ^\s*"""
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '"'
|
|
comment: double quoted string (allows for interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.double.elixir
|
|
- match: '"'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '~[a-z](?>""")'
|
|
comment: Double-quoted heredocs sigils
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.double.heredoc.elixir
|
|
- match: ^\s*"""
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '~[a-z]\{'
|
|
comment: sigil (allow for interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.interpolated.elixir
|
|
- match: '\}[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '~[a-z]\['
|
|
comment: sigil (allow for interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.interpolated.elixir
|
|
- match: '\][a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '~[a-z]\<'
|
|
comment: sigil (allow for interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.interpolated.elixir
|
|
- match: '\>[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '~[a-z]\('
|
|
comment: sigil (allow for interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.interpolated.elixir
|
|
- match: '\)[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '~[a-z]([^\w])'
|
|
comment: sigil (allow for interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.interpolated.elixir
|
|
- match: '\1[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- include: escaped_char
|
|
- match: '~[A-Z](?>""")'
|
|
comment: Double-quoted heredocs sigils
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.other.literal.upper.elixir
|
|
- match: ^\s*"""
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- match: '~[A-Z]\{'
|
|
comment: sigil (without interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.other.literal.upper.elixir
|
|
- match: '\}[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- match: '~[A-Z]\['
|
|
comment: sigil (without interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.other.literal.upper.elixir
|
|
- match: '\][a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- match: '~[A-Z]\<'
|
|
comment: sigil (without interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.other.literal.upper.elixir
|
|
- match: '\>[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- match: '~[A-Z]\('
|
|
comment: sigil (without interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.other.literal.upper.elixir
|
|
- match: '\)[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- match: '~[A-Z]([^\w])'
|
|
comment: sigil (without interpolation)
|
|
captures:
|
|
0: punctuation.definition.string.begin.elixir
|
|
push:
|
|
- meta_scope: string.quoted.other.literal.upper.elixir
|
|
- match: '\1[a-z]*'
|
|
captures:
|
|
0: punctuation.definition.string.end.elixir
|
|
pop: true
|
|
- match: '(?<!:)(:)(?>[a-zA-Z_][\w@]*(?>[?!]|=(?![>=]))?|\<\>|===?|!==?|<<>>|<<<|>>>|~~~|::|<\-|\|>|=>|=~|=|/|\\\\|\*\*?|\.\.?\.?|>=?|<=?|&&?&?|\+\+?|\-\-?|\|\|?\|?|\!|@|\%?\{\}|%|\[\]|\^(\^\^)?)'
|
|
comment: symbols
|
|
scope: constant.other.symbol.elixir
|
|
captures:
|
|
1: punctuation.definition.constant.elixir
|
|
- match: '(?>[a-zA-Z_][\w@]*(?>[?!])?)(:)(?!:)'
|
|
comment: symbols
|
|
scope: constant.other.keywords.elixir
|
|
captures:
|
|
1: punctuation.definition.constant.elixir
|
|
- match: ^\s*(##).*$\n?
|
|
scope: comment.line.section.elixir
|
|
captures:
|
|
1: punctuation.definition.comment.elixir
|
|
- match: '(?:^[ \t]+)?(#).*$\n?'
|
|
scope: comment.line.number-sign.elixir
|
|
captures:
|
|
1: punctuation.definition.comment.elixir
|
|
- match: '(?<!\w)\?(\\(x[0-9A-Fa-f]{1,2}(?![0-9A-Fa-f])\b|[^xMC])|[^\s\\])'
|
|
comment: |
|
|
matches questionmark-letters.
|
|
|
|
examples (1st alternation = hex):
|
|
?\x1 ?\x61
|
|
|
|
examples (2rd alternation = escaped):
|
|
?\n ?\b
|
|
|
|
examples (3rd alternation = normal):
|
|
?a ?A ?0
|
|
?* ?" ?(
|
|
?. ?#
|
|
|
|
the negative lookbehind prevents against matching
|
|
p(42.tainted?)
|
|
scope: constant.numeric.elixir
|
|
- match: \+\+|\-\-|<\|>
|
|
scope: keyword.operator.concatenation.elixir
|
|
- match: \|\>|<~>|<>|<<<|>>>|~>>|<<~|~>|<~|<\|>
|
|
scope: keyword.operator.sigils_1.elixir
|
|
- match: "&&&|&&"
|
|
scope: keyword.operator.sigils_2.elixir
|
|
- match: <\-|\\\\
|
|
scope: keyword.operator.sigils_3.elixir
|
|
- match: "===?|!==?|<=?|>=?"
|
|
scope: keyword.operator.comparison.elixir
|
|
- match: (\|\|\||&&&|^^^|<<<|>>>|~~~)
|
|
scope: keyword.operator.bitwise.elixir
|
|
- match: '(?<=[ \t])!+|\bnot\b|&&|\band\b|\|\||\bor\b|\bxor\b'
|
|
scope: keyword.operator.logical.elixir
|
|
- match: (\*|\+|\-|/)
|
|
scope: keyword.operator.arithmetic.elixir
|
|
- match: \||\+\+|\-\-|\*\*|\\\\|\<\-|\<\>|\<\<|\>\>|\:\:|\.\.|\|>|~|=>|&
|
|
scope: keyword.operator.other.elixir
|
|
- match: "="
|
|
scope: keyword.operator.assignment.elixir
|
|
- match: ":"
|
|
scope: punctuation.separator.other.elixir
|
|
- match: \;
|
|
scope: punctuation.separator.statement.elixir
|
|
- match: ","
|
|
scope: punctuation.separator.object.elixir
|
|
- match: \.
|
|
scope: punctuation.separator.method.elixir
|
|
- match: '\{|\}'
|
|
scope: punctuation.section.scope.elixir
|
|
- match: '\[|\]'
|
|
scope: punctuation.section.array.elixir
|
|
- match: \(|\)
|
|
scope: punctuation.section.function.elixir
|
|
escaped_char:
|
|
- match: '\\(x[\da-fA-F]{1,2}|.)'
|
|
scope: constant.character.escaped.elixir
|
|
interpolated_elixir:
|
|
- match: '#\{(\})'
|
|
scope: source.elixir.embedded.source
|
|
captures:
|
|
0: punctuation.section.embedded.elixir
|
|
1: source.elixir.embedded.source.empty
|
|
- match: '#\{'
|
|
captures:
|
|
0: punctuation.section.embedded.elixir
|
|
push:
|
|
- meta_scope: source.elixir.embedded.source
|
|
- match: '\}'
|
|
captures:
|
|
0: punctuation.section.embedded.elixir
|
|
pop: true
|
|
- include: nest_curly_and_self
|
|
- include: main
|
|
nest_curly_and_self:
|
|
- match: '\{'
|
|
captures:
|
|
0: punctuation.section.scope.elixir
|
|
push:
|
|
- match: '\}'
|
|
captures:
|
|
0: punctuation.section.scope.elixir
|
|
pop: true
|
|
- include: nest_curly_and_self
|
|
- include: main
|
|
regex_sub:
|
|
- include: interpolated_elixir
|
|
- include: escaped_char
|
|
- match: '(\{)\d+(,\d+)?(\})'
|
|
scope: string.regexp.arbitrary-repitition.elixir
|
|
captures:
|
|
1: punctuation.definition.arbitrary-repitition.elixir
|
|
3: punctuation.definition.arbitrary-repitition.elixir
|
|
- match: '\[(?:\^?\])?'
|
|
captures:
|
|
0: punctuation.definition.character-class.elixir
|
|
push:
|
|
- meta_scope: string.regexp.character-class.elixir
|
|
- match: '\]'
|
|
captures:
|
|
0: punctuation.definition.character-class.elixir
|
|
pop: true
|
|
- include: escaped_char
|
|
- match: \(
|
|
captures:
|
|
0: punctuation.definition.group.elixir
|
|
push:
|
|
- meta_scope: string.regexp.group.elixir
|
|
- match: \)
|
|
captures:
|
|
0: punctuation.definition.group.elixir
|
|
pop: true
|
|
- include: regex_sub
|
|
- match: '(?<=^|\s)(#)\s[[a-zA-Z0-9,. \t?!-][^\x{00}-\x{7F}]]*$'
|
|
comment: We are restrictive in what we allow to go after the comment character to avoid false positives, since the availability of comments depend on regexp flags.
|
|
scope: comment.line.number-sign.elixir
|
|
captures:
|
|
1: punctuation.definition.comment.elixir
|