# -*- coding: utf-8 -*- #
module Rouge
module Lexers
class Haxe < RegexLexer
title "Haxe"
desc "Haxe Cross-platform Toolkit (http://haxe.org)"
tag 'haxe'
aliases 'hx', 'haxe'
filenames '*.hx'
mimetypes 'text/haxe', 'text/x-haxe', 'text/x-hx'
def self.detect?(text)
return true if text.shebang? "haxe"
end
def self.keywords
@keywords ||= Set.new %w(
as break case cast catch class continue default do else enum false for
function if import in interface macro new null override package private
public return switch this throw true try untyped while
)
end
def self.imports
@imports ||= Set.new %w(
package import using
)
end
def self.declarations
@declarations ||= Set.new %w(
abstract dynamic extern extends from implements inline static to
typedef var
)
end
def self.reserved
@reserved ||= Set.new %w(
super trace inline build autoBuild enum
)
end
def self.constants
@constants ||= Set.new %w(true false null)
end
def self.builtins
@builtins ||= %w(
Void Dynamic Math Class Any Float Int UInt String StringTools Sys
EReg isNaN parseFloat parseInt this Array Map Date DateTools Bool
Lambda Reflect Std File FileSystem
)
end
id = /[$a-zA-Z_][a-zA-Z0-9_]*/
dotted_id = /[$a-zA-Z_][a-zA-Z0-9_.]*/
state :comments_and_whitespace do
rule %r/\s+/, Text
rule %r(//.*?$), Comment::Single
rule %r(/\*.*?\*/)m, Comment::Multiline
end
state :expr_start do
mixin :comments_and_whitespace
rule %r/#(?:if|elseif|else|end).*/, Comment::Preproc
rule %r(~) do
token Str::Regex
goto :regex
end
rule %r/[{]/, Punctuation, :object
rule %r//, Text, :pop!
end
state :namespace do
mixin :comments_and_whitespace
rule %r/
(#{dotted_id})
(\s+)(in|as)(\s+)
(#{id})
/x do
groups(Name::Namespace, Text::Whitespace, Keyword, Text::Whitespace, Name)
end
rule %r/#{dotted_id}/, Name::Namespace
rule(//) { pop! }
end
state :regex do
rule %r(/) do
token Str::Regex
goto :regex_end
end
rule %r([^/]\n), Error, :pop!
rule %r/\n/, Error, :pop!
rule %r/\[\^/, Str::Escape, :regex_group
rule %r/\[/, Str::Escape, :regex_group
rule %r/\\./, Str::Escape
rule %r{[(][?][:=<!]}, Str::Escape
rule %r/[{][\d,]+[}]/, Str::Escape
rule %r/[()?]/, Str::Escape
rule %r/./, Str::Regex
end
state :regex_end do
rule %r/[gim]+/, Str::Regex, :pop!
rule(//) { pop! }
end
state :regex_group do
# specially highlight / in a group to indicate that it doesn't
# close the regex
rule %r/\//, Str::Escape
rule %r([^/]\n) do
token Error
pop! 2
end
rule %r/\]/, Str::Escape, :pop!
rule %r/\\./, Str::Escape
rule %r/./, Str::Regex
end
state :bad_regex do
rule %r/[^\n]+/, Error, :pop!
end
state :root do
rule %r/\n/, Text, :statement
rule %r(\{), Punctuation, :expr_start
mixin :comments_and_whitespace
rule %r/@/, Name::Decorator, :metadata
rule %r(\+\+ | -- | ~ | && | \|\| | \\(?=\n) | << | >> | ==
| != )x,
Operator, :expr_start
rule %r([-:<>+*%&|\^/!=]=?), Operator, :expr_start
rule %r/[(\[,]/, Punctuation, :expr_start
rule %r/;/, Punctuation, :statement
rule %r/[)\]}.]/, Punctuation
rule %r/[?]/ do
token Punctuation
push :ternary
push :expr_start
end
rule id do |m|
match = m[0]
if self.class.imports.include?(match)
token Keyword::Namespace
push :namespace
elsif self.class.keywords.include?(match)
token Keyword
push :expr_start
elsif self.class.declarations.include?(match)
token Keyword::Declaration
push :expr_start
elsif self.class.reserved.include?(match)
token Keyword::Reserved
elsif self.class.constants.include?(match)
token Keyword::Constant
elsif self.class.builtins.include?(match)
token Name::Builtin
else
token Name::Other
end
end
rule %r/\-?\d+\.\d+(?:[eE]\d+)?[fd]?/, Num::Float
rule %r/0x\h+/, Num::Hex
rule %r/\-?[0-9]+/, Num::Integer
rule %r/"/, Str::Double, :str_double
rule %r/'/, Str::Single, :str_single
end
# braced parts that aren't object literals
state :statement do
rule %r/(#{id})(\s*)(:)/ do
groups Name::Label, Text, Punctuation
end
mixin :expr_start
end
# object literals
state :object do
mixin :comments_and_whitespace
rule %r/[}]/ do
token Punctuation
goto :statement
end
rule %r/(#{id})(\s*)(:)/ do
groups Name::Attribute, Text, Punctuation
push :expr_start
end
rule %r/:/, Punctuation
mixin :root
end
state :metadata do
rule %r/(#{id})(\()?/ do |m|
groups Name::Decorator, Punctuation
pop! unless m[2]
end
rule %r/:#{id}(?:\.#{id})*/, Name::Decorator, :pop!
rule %r/\)/, Name::Decorator, :pop!
mixin :root
end
# ternary expressions, where <id>: is not a label!
state :ternary do
rule %r/:/ do
token Punctuation
goto :expr_start
end
mixin :root
end
state :str_double do
mixin :str_escape
rule %r/"/, Str::Double, :pop!
rule %r/[^\\"]+/, Str::Double
end
state :str_single do
mixin :str_escape
rule %r/'/, Str::Single, :pop!
rule %r/\$\$/, Str::Single
rule %r/\$#{id}/, Str::Interpol
rule %r/\$\{/, Str::Interpol, :str_interpol
rule %r/[^\\$']+/, Str::Single
end
state :str_escape do
rule %r/\\[\\tnr'"]/, Str::Escape
rule %r/\\[0-7]{3}/, Str::Escape
rule %r/\\x\h{2}/, Str::Escape
rule %r/\\u\h{4}/, Str::Escape
rule %r/\\u\{\h{1,6}\}/, Str::Escape
end
state :str_interpol do
rule %r/\}/, Str::Interpol, :pop!
mixin :root
end
end
end
end