mirror of https://github.com/gogits/gogs.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
837 lines
55 KiB
837 lines
55 KiB
// CodeMirror, copyright (c) by Marijn Haverbeke and others |
|
// Distributed under an MIT license: http://codemirror.net/LICENSE |
|
|
|
// CodeMirror2 mode/perl/perl.js (text/x-perl) beta 0.10 (2011-11-08) |
|
// This is a part of CodeMirror from https://github.com/sabaca/CodeMirror_mode_perl (mail@sabaca.com) |
|
|
|
(function(mod) { |
|
if (typeof exports == "object" && typeof module == "object") // CommonJS |
|
mod(require("../../lib/codemirror")); |
|
else if (typeof define == "function" && define.amd) // AMD |
|
define(["../../lib/codemirror"], mod); |
|
else // Plain browser env |
|
mod(CodeMirror); |
|
})(function(CodeMirror) { |
|
"use strict"; |
|
|
|
CodeMirror.defineMode("perl",function(){ |
|
// http://perldoc.perl.org |
|
var PERL={ // null - magic touch |
|
// 1 - keyword |
|
// 2 - def |
|
// 3 - atom |
|
// 4 - operator |
|
// 5 - variable-2 (predefined) |
|
// [x,y] - x=1,2,3; y=must be defined if x{...} |
|
// PERL operators |
|
'->' : 4, |
|
'++' : 4, |
|
'--' : 4, |
|
'**' : 4, |
|
// ! ~ \ and unary + and - |
|
'=~' : 4, |
|
'!~' : 4, |
|
'*' : 4, |
|
'/' : 4, |
|
'%' : 4, |
|
'x' : 4, |
|
'+' : 4, |
|
'-' : 4, |
|
'.' : 4, |
|
'<<' : 4, |
|
'>>' : 4, |
|
// named unary operators |
|
'<' : 4, |
|
'>' : 4, |
|
'<=' : 4, |
|
'>=' : 4, |
|
'lt' : 4, |
|
'gt' : 4, |
|
'le' : 4, |
|
'ge' : 4, |
|
'==' : 4, |
|
'!=' : 4, |
|
'<=>' : 4, |
|
'eq' : 4, |
|
'ne' : 4, |
|
'cmp' : 4, |
|
'~~' : 4, |
|
'&' : 4, |
|
'|' : 4, |
|
'^' : 4, |
|
'&&' : 4, |
|
'||' : 4, |
|
'//' : 4, |
|
'..' : 4, |
|
'...' : 4, |
|
'?' : 4, |
|
':' : 4, |
|
'=' : 4, |
|
'+=' : 4, |
|
'-=' : 4, |
|
'*=' : 4, // etc. ??? |
|
',' : 4, |
|
'=>' : 4, |
|
'::' : 4, |
|
// list operators (rightward) |
|
'not' : 4, |
|
'and' : 4, |
|
'or' : 4, |
|
'xor' : 4, |
|
// PERL predefined variables (I know, what this is a paranoid idea, but may be needed for people, who learn PERL, and for me as well, ...and may be for you?;) |
|
'BEGIN' : [5,1], |
|
'END' : [5,1], |
|
'PRINT' : [5,1], |
|
'PRINTF' : [5,1], |
|
'GETC' : [5,1], |
|
'READ' : [5,1], |
|
'READLINE' : [5,1], |
|
'DESTROY' : [5,1], |
|
'TIE' : [5,1], |
|
'TIEHANDLE' : [5,1], |
|
'UNTIE' : [5,1], |
|
'STDIN' : 5, |
|
'STDIN_TOP' : 5, |
|
'STDOUT' : 5, |
|
'STDOUT_TOP' : 5, |
|
'STDERR' : 5, |
|
'STDERR_TOP' : 5, |
|
'$ARG' : 5, |
|
'$_' : 5, |
|
'@ARG' : 5, |
|
'@_' : 5, |
|
'$LIST_SEPARATOR' : 5, |
|
'$"' : 5, |
|
'$PROCESS_ID' : 5, |
|
'$PID' : 5, |
|
'$$' : 5, |
|
'$REAL_GROUP_ID' : 5, |
|
'$GID' : 5, |
|
'$(' : 5, |
|
'$EFFECTIVE_GROUP_ID' : 5, |
|
'$EGID' : 5, |
|
'$)' : 5, |
|
'$PROGRAM_NAME' : 5, |
|
'$0' : 5, |
|
'$SUBSCRIPT_SEPARATOR' : 5, |
|
'$SUBSEP' : 5, |
|
'$;' : 5, |
|
'$REAL_USER_ID' : 5, |
|
'$UID' : 5, |
|
'$<' : 5, |
|
'$EFFECTIVE_USER_ID' : 5, |
|
'$EUID' : 5, |
|
'$>' : 5, |
|
'$a' : 5, |
|
'$b' : 5, |
|
'$COMPILING' : 5, |
|
'$^C' : 5, |
|
'$DEBUGGING' : 5, |
|
'$^D' : 5, |
|
'${^ENCODING}' : 5, |
|
'$ENV' : 5, |
|
'%ENV' : 5, |
|
'$SYSTEM_FD_MAX' : 5, |
|
'$^F' : 5, |
|
'@F' : 5, |
|
'${^GLOBAL_PHASE}' : 5, |
|
'$^H' : 5, |
|
'%^H' : 5, |
|
'@INC' : 5, |
|
'%INC' : 5, |
|
'$INPLACE_EDIT' : 5, |
|
'$^I' : 5, |
|
'$^M' : 5, |
|
'$OSNAME' : 5, |
|
'$^O' : 5, |
|
'${^OPEN}' : 5, |
|
'$PERLDB' : 5, |
|
'$^P' : 5, |
|
'$SIG' : 5, |
|
'%SIG' : 5, |
|
'$BASETIME' : 5, |
|
'$^T' : 5, |
|
'${^TAINT}' : 5, |
|
'${^UNICODE}' : 5, |
|
'${^UTF8CACHE}' : 5, |
|
'${^UTF8LOCALE}' : 5, |
|
'$PERL_VERSION' : 5, |
|
'$^V' : 5, |
|
'${^WIN32_SLOPPY_STAT}' : 5, |
|
'$EXECUTABLE_NAME' : 5, |
|
'$^X' : 5, |
|
'$1' : 5, // - regexp $1, $2... |
|
'$MATCH' : 5, |
|
'$&' : 5, |
|
'${^MATCH}' : 5, |
|
'$PREMATCH' : 5, |
|
'$`' : 5, |
|
'${^PREMATCH}' : 5, |
|
'$POSTMATCH' : 5, |
|
"$'" : 5, |
|
'${^POSTMATCH}' : 5, |
|
'$LAST_PAREN_MATCH' : 5, |
|
'$+' : 5, |
|
'$LAST_SUBMATCH_RESULT' : 5, |
|
'$^N' : 5, |
|
'@LAST_MATCH_END' : 5, |
|
'@+' : 5, |
|
'%LAST_PAREN_MATCH' : 5, |
|
'%+' : 5, |
|
'@LAST_MATCH_START' : 5, |
|
'@-' : 5, |
|
'%LAST_MATCH_START' : 5, |
|
'%-' : 5, |
|
'$LAST_REGEXP_CODE_RESULT' : 5, |
|
'$^R' : 5, |
|
'${^RE_DEBUG_FLAGS}' : 5, |
|
'${^RE_TRIE_MAXBUF}' : 5, |
|
'$ARGV' : 5, |
|
'@ARGV' : 5, |
|
'ARGV' : 5, |
|
'ARGVOUT' : 5, |
|
'$OUTPUT_FIELD_SEPARATOR' : 5, |
|
'$OFS' : 5, |
|
'$,' : 5, |
|
'$INPUT_LINE_NUMBER' : 5, |
|
'$NR' : 5, |
|
'$.' : 5, |
|
'$INPUT_RECORD_SEPARATOR' : 5, |
|
'$RS' : 5, |
|
'$/' : 5, |
|
'$OUTPUT_RECORD_SEPARATOR' : 5, |
|
'$ORS' : 5, |
|
'$\\' : 5, |
|
'$OUTPUT_AUTOFLUSH' : 5, |
|
'$|' : 5, |
|
'$ACCUMULATOR' : 5, |
|
'$^A' : 5, |
|
'$FORMAT_FORMFEED' : 5, |
|
'$^L' : 5, |
|
'$FORMAT_PAGE_NUMBER' : 5, |
|
'$%' : 5, |
|
'$FORMAT_LINES_LEFT' : 5, |
|
'$-' : 5, |
|
'$FORMAT_LINE_BREAK_CHARACTERS' : 5, |
|
'$:' : 5, |
|
'$FORMAT_LINES_PER_PAGE' : 5, |
|
'$=' : 5, |
|
'$FORMAT_TOP_NAME' : 5, |
|
'$^' : 5, |
|
'$FORMAT_NAME' : 5, |
|
'$~' : 5, |
|
'${^CHILD_ERROR_NATIVE}' : 5, |
|
'$EXTENDED_OS_ERROR' : 5, |
|
'$^E' : 5, |
|
'$EXCEPTIONS_BEING_CAUGHT' : 5, |
|
'$^S' : 5, |
|
'$WARNING' : 5, |
|
'$^W' : 5, |
|
'${^WARNING_BITS}' : 5, |
|
'$OS_ERROR' : 5, |
|
'$ERRNO' : 5, |
|
'$!' : 5, |
|
'%OS_ERROR' : 5, |
|
'%ERRNO' : 5, |
|
'%!' : 5, |
|
'$CHILD_ERROR' : 5, |
|
'$?' : 5, |
|
'$EVAL_ERROR' : 5, |
|
'$@' : 5, |
|
'$OFMT' : 5, |
|
'$#' : 5, |
|
'$*' : 5, |
|
'$ARRAY_BASE' : 5, |
|
'$[' : 5, |
|
'$OLD_PERL_VERSION' : 5, |
|
'$]' : 5, |
|
// PERL blocks |
|
'if' :[1,1], |
|
elsif :[1,1], |
|
'else' :[1,1], |
|
'while' :[1,1], |
|
unless :[1,1], |
|
'for' :[1,1], |
|
foreach :[1,1], |
|
// PERL functions |
|
'abs' :1, // - absolute value function |
|
accept :1, // - accept an incoming socket connect |
|
alarm :1, // - schedule a SIGALRM |
|
'atan2' :1, // - arctangent of Y/X in the range -PI to PI |
|
bind :1, // - binds an address to a socket |
|
binmode :1, // - prepare binary files for I/O |
|
bless :1, // - create an object |
|
bootstrap :1, // |
|
'break' :1, // - break out of a "given" block |
|
caller :1, // - get context of the current subroutine call |
|
chdir :1, // - change your current working directory |
|
chmod :1, // - changes the permissions on a list of files |
|
chomp :1, // - remove a trailing record separator from a string |
|
chop :1, // - remove the last character from a string |
|
chown :1, // - change the owership on a list of files |
|
chr :1, // - get character this number represents |
|
chroot :1, // - make directory new root for path lookups |
|
close :1, // - close file (or pipe or socket) handle |
|
closedir :1, // - close directory handle |
|
connect :1, // - connect to a remote socket |
|
'continue' :[1,1], // - optional trailing block in a while or foreach |
|
'cos' :1, // - cosine function |
|
crypt :1, // - one-way passwd-style encryption |
|
dbmclose :1, // - breaks binding on a tied dbm file |
|
dbmopen :1, // - create binding on a tied dbm file |
|
'default' :1, // |
|
defined :1, // - test whether a value, variable, or function is defined |
|
'delete' :1, // - deletes a value from a hash |
|
die :1, // - raise an exception or bail out |
|
'do' :1, // - turn a BLOCK into a TERM |
|
dump :1, // - create an immediate core dump |
|
each :1, // - retrieve the next key/value pair from a hash |
|
endgrent :1, // - be done using group file |
|
endhostent :1, // - be done using hosts file |
|
endnetent :1, // - be done using networks file |
|
endprotoent :1, // - be done using protocols file |
|
endpwent :1, // - be done using passwd file |
|
endservent :1, // - be done using services file |
|
eof :1, // - test a filehandle for its end |
|
'eval' :1, // - catch exceptions or compile and run code |
|
'exec' :1, // - abandon this program to run another |
|
exists :1, // - test whether a hash key is present |
|
exit :1, // - terminate this program |
|
'exp' :1, // - raise I to a power |
|
fcntl :1, // - file control system call |
|
fileno :1, // - return file descriptor from filehandle |
|
flock :1, // - lock an entire file with an advisory lock |
|
fork :1, // - create a new process just like this one |
|
format :1, // - declare a picture format with use by the write() function |
|
formline :1, // - internal function used for formats |
|
getc :1, // - get the next character from the filehandle |
|
getgrent :1, // - get next group record |
|
getgrgid :1, // - get group record given group user ID |
|
getgrnam :1, // - get group record given group name |
|
gethostbyaddr :1, // - get host record given its address |
|
gethostbyname :1, // - get host record given name |
|
gethostent :1, // - get next hosts record |
|
getlogin :1, // - return who logged in at this tty |
|
getnetbyaddr :1, // - get network record given its address |
|
getnetbyname :1, // - get networks record given name |
|
getnetent :1, // - get next networks record |
|
getpeername :1, // - find the other end of a socket connection |
|
getpgrp :1, // - get process group |
|
getppid :1, // - get parent process ID |
|
getpriority :1, // - get current nice value |
|
getprotobyname :1, // - get protocol record given name |
|
getprotobynumber :1, // - get protocol record numeric protocol |
|
getprotoent :1, // - get next protocols record |
|
getpwent :1, // - get next passwd record |
|
getpwnam :1, // - get passwd record given user login name |
|
getpwuid :1, // - get passwd record given user ID |
|
getservbyname :1, // - get services record given its name |
|
getservbyport :1, // - get services record given numeric port |
|
getservent :1, // - get next services record |
|
getsockname :1, // - retrieve the sockaddr for a given socket |
|
getsockopt :1, // - get socket options on a given socket |
|
given :1, // |
|
glob :1, // - expand filenames using wildcards |
|
gmtime :1, // - convert UNIX time into record or string using Greenwich time |
|
'goto' :1, // - create spaghetti code |
|
grep :1, // - locate elements in a list test true against a given criterion |
|
hex :1, // - convert a string to a hexadecimal number |
|
'import' :1, // - patch a module's namespace into your own |
|
index :1, // - find a substring within a string |
|
'int' :1, // - get the integer portion of a number |
|
ioctl :1, // - system-dependent device control system call |
|
'join' :1, // - join a list into a string using a separator |
|
keys :1, // - retrieve list of indices from a hash |
|
kill :1, // - send a signal to a process or process group |
|
last :1, // - exit a block prematurely |
|
lc :1, // - return lower-case version of a string |
|
lcfirst :1, // - return a string with just the next letter in lower case |
|
length :1, // - return the number of bytes in a string |
|
'link' :1, // - create a hard link in the filesytem |
|
listen :1, // - register your socket as a server |
|
local : 2, // - create a temporary value for a global variable (dynamic scoping) |
|
localtime :1, // - convert UNIX time into record or string using local time |
|
lock :1, // - get a thread lock on a variable, subroutine, or method |
|
'log' :1, // - retrieve the natural logarithm for a number |
|
lstat :1, // - stat a symbolic link |
|
m :null, // - match a string with a regular expression pattern |
|
map :1, // - apply a change to a list to get back a new list with the changes |
|
mkdir :1, // - create a directory |
|
msgctl :1, // - SysV IPC message control operations |
|
msgget :1, // - get SysV IPC message queue |
|
msgrcv :1, // - receive a SysV IPC message from a message queue |
|
msgsnd :1, // - send a SysV IPC message to a message queue |
|
my : 2, // - declare and assign a local variable (lexical scoping) |
|
'new' :1, // |
|
next :1, // - iterate a block prematurely |
|
no :1, // - unimport some module symbols or semantics at compile time |
|
oct :1, // - convert a string to an octal number |
|
open :1, // - open a file, pipe, or descriptor |
|
opendir :1, // - open a directory |
|
ord :1, // - find a character's numeric representation |
|
our : 2, // - declare and assign a package variable (lexical scoping) |
|
pack :1, // - convert a list into a binary representation |
|
'package' :1, // - declare a separate global namespace |
|
pipe :1, // - open a pair of connected filehandles |
|
pop :1, // - remove the last element from an array and return it |
|
pos :1, // - find or set the offset for the last/next m//g search |
|
print :1, // - output a list to a filehandle |
|
printf :1, // - output a formatted list to a filehandle |
|
prototype :1, // - get the prototype (if any) of a subroutine |
|
push :1, // - append one or more elements to an array |
|
q :null, // - singly quote a string |
|
qq :null, // - doubly quote a string |
|
qr :null, // - Compile pattern |
|
quotemeta :null, // - quote regular expression magic characters |
|
qw :null, // - quote a list of words |
|
qx :null, // - backquote quote a string |
|
rand :1, // - retrieve the next pseudorandom number |
|
read :1, // - fixed-length buffered input from a filehandle |
|
readdir :1, // - get a directory from a directory handle |
|
readline :1, // - fetch a record from a file |
|
readlink :1, // - determine where a symbolic link is pointing |
|
readpipe :1, // - execute a system command and collect standard output |
|
recv :1, // - receive a message over a Socket |
|
redo :1, // - start this loop iteration over again |
|
ref :1, // - find out the type of thing being referenced |
|
rename :1, // - change a filename |
|
require :1, // - load in external functions from a library at runtime |
|
reset :1, // - clear all variables of a given name |
|
'return' :1, // - get out of a function early |
|
reverse :1, // - flip a string or a list |
|
rewinddir :1, // - reset directory handle |
|
rindex :1, // - right-to-left substring search |
|
rmdir :1, // - remove a directory |
|
s :null, // - replace a pattern with a string |
|
say :1, // - print with newline |
|
scalar :1, // - force a scalar context |
|
seek :1, // - reposition file pointer for random-access I/O |
|
seekdir :1, // - reposition directory pointer |
|
select :1, // - reset default output or do I/O multiplexing |
|
semctl :1, // - SysV semaphore control operations |
|
semget :1, // - get set of SysV semaphores |
|
semop :1, // - SysV semaphore operations |
|
send :1, // - send a message over a socket |
|
setgrent :1, // - prepare group file for use |
|
sethostent :1, // - prepare hosts file for use |
|
setnetent :1, // - prepare networks file for use |
|
setpgrp :1, // - set the process group of a process |
|
setpriority :1, // - set a process's nice value |
|
setprotoent :1, // - prepare protocols file for use |
|
setpwent :1, // - prepare passwd file for use |
|
setservent :1, // - prepare services file for use |
|
setsockopt :1, // - set some socket options |
|
shift :1, // - remove the first element of an array, and return it |
|
shmctl :1, // - SysV shared memory operations |
|
shmget :1, // - get SysV shared memory segment identifier |
|
shmread :1, // - read SysV shared memory |
|
shmwrite :1, // - write SysV shared memory |
|
shutdown :1, // - close down just half of a socket connection |
|
'sin' :1, // - return the sine of a number |
|
sleep :1, // - block for some number of seconds |
|
socket :1, // - create a socket |
|
socketpair :1, // - create a pair of sockets |
|
'sort' :1, // - sort a list of values |
|
splice :1, // - add or remove elements anywhere in an array |
|
'split' :1, // - split up a string using a regexp delimiter |
|
sprintf :1, // - formatted print into a string |
|
'sqrt' :1, // - square root function |
|
srand :1, // - seed the random number generator |
|
stat :1, // - get a file's status information |
|
state :1, // - declare and assign a state variable (persistent lexical scoping) |
|
study :1, // - optimize input data for repeated searches |
|
'sub' :1, // - declare a subroutine, possibly anonymously |
|
'substr' :1, // - get or alter a portion of a stirng |
|
symlink :1, // - create a symbolic link to a file |
|
syscall :1, // - execute an arbitrary system call |
|
sysopen :1, // - open a file, pipe, or descriptor |
|
sysread :1, // - fixed-length unbuffered input from a filehandle |
|
sysseek :1, // - position I/O pointer on handle used with sysread and syswrite |
|
system :1, // - run a separate program |
|
syswrite :1, // - fixed-length unbuffered output to a filehandle |
|
tell :1, // - get current seekpointer on a filehandle |
|
telldir :1, // - get current seekpointer on a directory handle |
|
tie :1, // - bind a variable to an object class |
|
tied :1, // - get a reference to the object underlying a tied variable |
|
time :1, // - return number of seconds since 1970 |
|
times :1, // - return elapsed time for self and child processes |
|
tr :null, // - transliterate a string |
|
truncate :1, // - shorten a file |
|
uc :1, // - return upper-case version of a string |
|
ucfirst :1, // - return a string with just the next letter in upper case |
|
umask :1, // - set file creation mode mask |
|
undef :1, // - remove a variable or function definition |
|
unlink :1, // - remove one link to a file |
|
unpack :1, // - convert binary structure into normal perl variables |
|
unshift :1, // - prepend more elements to the beginning of a list |
|
untie :1, // - break a tie binding to a variable |
|
use :1, // - load in a module at compile time |
|
utime :1, // - set a file's last access and modify times |
|
values :1, // - return a list of the values in a hash |
|
vec :1, // - test or set particular bits in a string |
|
wait :1, // - wait for any child process to die |
|
waitpid :1, // - wait for a particular child process to die |
|
wantarray :1, // - get void vs scalar vs list context of current subroutine call |
|
warn :1, // - print debugging info |
|
when :1, // |
|
write :1, // - print a picture record |
|
y :null}; // - transliterate a string |
|
|
|
var RXstyle="string-2"; |
|
var RXmodifiers=/[goseximacplud]/; // NOTE: "m", "s", "y" and "tr" need to correct real modifiers for each regexp type |
|
|
|
function tokenChain(stream,state,chain,style,tail){ // NOTE: chain.length > 2 is not working now (it's for s[...][...]geos;) |
|
state.chain=null; // 12 3tail |
|
state.style=null; |
|
state.tail=null; |
|
state.tokenize=function(stream,state){ |
|
var e=false,c,i=0; |
|
while(c=stream.next()){ |
|
if(c===chain[i]&&!e){ |
|
if(chain[++i]!==undefined){ |
|
state.chain=chain[i]; |
|
state.style=style; |
|
state.tail=tail;} |
|
else if(tail) |
|
stream.eatWhile(tail); |
|
state.tokenize=tokenPerl; |
|
return style;} |
|
e=!e&&c=="\\";} |
|
return style;}; |
|
return state.tokenize(stream,state);} |
|
|
|
function tokenSOMETHING(stream,state,string){ |
|
state.tokenize=function(stream,state){ |
|
if(stream.string==string) |
|
state.tokenize=tokenPerl; |
|
stream.skipToEnd(); |
|
return "string";}; |
|
return state.tokenize(stream,state);} |
|
|
|
function tokenPerl(stream,state){ |
|
if(stream.eatSpace()) |
|
return null; |
|
if(state.chain) |
|
return tokenChain(stream,state,state.chain,state.style,state.tail); |
|
if(stream.match(/^\-?[\d\.]/,false)) |
|
if(stream.match(/^(\-?(\d*\.\d+(e[+-]?\d+)?|\d+\.\d*)|0x[\da-fA-F]+|0b[01]+|\d+(e[+-]?\d+)?)/)) |
|
return 'number'; |
|
if(stream.match(/^<<(?=\w)/)){ // NOTE: <<SOMETHING\n...\nSOMETHING\n |
|
stream.eatWhile(/\w/); |
|
return tokenSOMETHING(stream,state,stream.current().substr(2));} |
|
if(stream.sol()&&stream.match(/^\=item(?!\w)/)){// NOTE: \n=item...\n=cut\n |
|
return tokenSOMETHING(stream,state,'=cut');} |
|
var ch=stream.next(); |
|
if(ch=='"'||ch=="'"){ // NOTE: ' or " or <<'SOMETHING'\n...\nSOMETHING\n or <<"SOMETHING"\n...\nSOMETHING\n |
|
if(prefix(stream, 3)=="<<"+ch){ |
|
var p=stream.pos; |
|
stream.eatWhile(/\w/); |
|
var n=stream.current().substr(1); |
|
if(n&&stream.eat(ch)) |
|
return tokenSOMETHING(stream,state,n); |
|
stream.pos=p;} |
|
return tokenChain(stream,state,[ch],"string");} |
|
if(ch=="q"){ |
|
var c=look(stream, -2); |
|
if(!(c&&/\w/.test(c))){ |
|
c=look(stream, 0); |
|
if(c=="x"){ |
|
c=look(stream, 1); |
|
if(c=="("){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[")"],RXstyle,RXmodifiers);} |
|
if(c=="["){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["]"],RXstyle,RXmodifiers);} |
|
if(c=="{"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["}"],RXstyle,RXmodifiers);} |
|
if(c=="<"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[">"],RXstyle,RXmodifiers);} |
|
if(/[\^'"!~\/]/.test(c)){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,[stream.eat(c)],RXstyle,RXmodifiers);}} |
|
else if(c=="q"){ |
|
c=look(stream, 1); |
|
if(c=="("){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[")"],"string");} |
|
if(c=="["){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["]"],"string");} |
|
if(c=="{"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["}"],"string");} |
|
if(c=="<"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[">"],"string");} |
|
if(/[\^'"!~\/]/.test(c)){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,[stream.eat(c)],"string");}} |
|
else if(c=="w"){ |
|
c=look(stream, 1); |
|
if(c=="("){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[")"],"bracket");} |
|
if(c=="["){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["]"],"bracket");} |
|
if(c=="{"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["}"],"bracket");} |
|
if(c=="<"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[">"],"bracket");} |
|
if(/[\^'"!~\/]/.test(c)){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,[stream.eat(c)],"bracket");}} |
|
else if(c=="r"){ |
|
c=look(stream, 1); |
|
if(c=="("){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[")"],RXstyle,RXmodifiers);} |
|
if(c=="["){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["]"],RXstyle,RXmodifiers);} |
|
if(c=="{"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,["}"],RXstyle,RXmodifiers);} |
|
if(c=="<"){ |
|
eatSuffix(stream, 2); |
|
return tokenChain(stream,state,[">"],RXstyle,RXmodifiers);} |
|
if(/[\^'"!~\/]/.test(c)){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,[stream.eat(c)],RXstyle,RXmodifiers);}} |
|
else if(/[\^'"!~\/(\[{<]/.test(c)){ |
|
if(c=="("){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,[")"],"string");} |
|
if(c=="["){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,["]"],"string");} |
|
if(c=="{"){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,["}"],"string");} |
|
if(c=="<"){ |
|
eatSuffix(stream, 1); |
|
return tokenChain(stream,state,[">"],"string");} |
|
if(/[\^'"!~\/]/.test(c)){ |
|
return tokenChain(stream,state,[stream.eat(c)],"string");}}}} |
|
if(ch=="m"){ |
|
var c=look(stream, -2); |
|
if(!(c&&/\w/.test(c))){ |
|
c=stream.eat(/[(\[{<\^'"!~\/]/); |
|
if(c){ |
|
if(/[\^'"!~\/]/.test(c)){ |
|
return tokenChain(stream,state,[c],RXstyle,RXmodifiers);} |
|
if(c=="("){ |
|
return tokenChain(stream,state,[")"],RXstyle,RXmodifiers);} |
|
if(c=="["){ |
|
return tokenChain(stream,state,["]"],RXstyle,RXmodifiers);} |
|
if(c=="{"){ |
|
return tokenChain(stream,state,["}"],RXstyle,RXmodifiers);} |
|
if(c=="<"){ |
|
return tokenChain(stream,state,[">"],RXstyle,RXmodifiers);}}}} |
|
if(ch=="s"){ |
|
var c=/[\/>\]})\w]/.test(look(stream, -2)); |
|
if(!c){ |
|
c=stream.eat(/[(\[{<\^'"!~\/]/); |
|
if(c){ |
|
if(c=="[") |
|
return tokenChain(stream,state,["]","]"],RXstyle,RXmodifiers); |
|
if(c=="{") |
|
return tokenChain(stream,state,["}","}"],RXstyle,RXmodifiers); |
|
if(c=="<") |
|
return tokenChain(stream,state,[">",">"],RXstyle,RXmodifiers); |
|
if(c=="(") |
|
return tokenChain(stream,state,[")",")"],RXstyle,RXmodifiers); |
|
return tokenChain(stream,state,[c,c],RXstyle,RXmodifiers);}}} |
|
if(ch=="y"){ |
|
var c=/[\/>\]})\w]/.test(look(stream, -2)); |
|
if(!c){ |
|
c=stream.eat(/[(\[{<\^'"!~\/]/); |
|
if(c){ |
|
if(c=="[") |
|
return tokenChain(stream,state,["]","]"],RXstyle,RXmodifiers); |
|
if(c=="{") |
|
return tokenChain(stream,state,["}","}"],RXstyle,RXmodifiers); |
|
if(c=="<") |
|
return tokenChain(stream,state,[">",">"],RXstyle,RXmodifiers); |
|
if(c=="(") |
|
return tokenChain(stream,state,[")",")"],RXstyle,RXmodifiers); |
|
return tokenChain(stream,state,[c,c],RXstyle,RXmodifiers);}}} |
|
if(ch=="t"){ |
|
var c=/[\/>\]})\w]/.test(look(stream, -2)); |
|
if(!c){ |
|
c=stream.eat("r");if(c){ |
|
c=stream.eat(/[(\[{<\^'"!~\/]/); |
|
if(c){ |
|
if(c=="[") |
|
return tokenChain(stream,state,["]","]"],RXstyle,RXmodifiers); |
|
if(c=="{") |
|
return tokenChain(stream,state,["}","}"],RXstyle,RXmodifiers); |
|
if(c=="<") |
|
return tokenChain(stream,state,[">",">"],RXstyle,RXmodifiers); |
|
if(c=="(") |
|
return tokenChain(stream,state,[")",")"],RXstyle,RXmodifiers); |
|
return tokenChain(stream,state,[c,c],RXstyle,RXmodifiers);}}}} |
|
if(ch=="`"){ |
|
return tokenChain(stream,state,[ch],"variable-2");} |
|
if(ch=="/"){ |
|
if(!/~\s*$/.test(prefix(stream))) |
|
return "operator"; |
|
else |
|
return tokenChain(stream,state,[ch],RXstyle,RXmodifiers);} |
|
if(ch=="$"){ |
|
var p=stream.pos; |
|
if(stream.eatWhile(/\d/)||stream.eat("{")&&stream.eatWhile(/\d/)&&stream.eat("}")) |
|
return "variable-2"; |
|
else |
|
stream.pos=p;} |
|
if(/[$@%]/.test(ch)){ |
|
var p=stream.pos; |
|
if(stream.eat("^")&&stream.eat(/[A-Z]/)||!/[@$%&]/.test(look(stream, -2))&&stream.eat(/[=|\\\-#?@;:&`~\^!\[\]*'"$+.,\/<>()]/)){ |
|
var c=stream.current(); |
|
if(PERL[c]) |
|
return "variable-2";} |
|
stream.pos=p;} |
|
if(/[$@%&]/.test(ch)){ |
|
if(stream.eatWhile(/[\w$\[\]]/)||stream.eat("{")&&stream.eatWhile(/[\w$\[\]]/)&&stream.eat("}")){ |
|
var c=stream.current(); |
|
if(PERL[c]) |
|
return "variable-2"; |
|
else |
|
return "variable";}} |
|
if(ch=="#"){ |
|
if(look(stream, -2)!="$"){ |
|
stream.skipToEnd(); |
|
return "comment";}} |
|
if(/[:+\-\^*$&%@=<>!?|\/~\.]/.test(ch)){ |
|
var p=stream.pos; |
|
stream.eatWhile(/[:+\-\^*$&%@=<>!?|\/~\.]/); |
|
if(PERL[stream.current()]) |
|
return "operator"; |
|
else |
|
stream.pos=p;} |
|
if(ch=="_"){ |
|
if(stream.pos==1){ |
|
if(suffix(stream, 6)=="_END__"){ |
|
return tokenChain(stream,state,['\0'],"comment");} |
|
else if(suffix(stream, 7)=="_DATA__"){ |
|
return tokenChain(stream,state,['\0'],"variable-2");} |
|
else if(suffix(stream, 7)=="_C__"){ |
|
return tokenChain(stream,state,['\0'],"string");}}} |
|
if(/\w/.test(ch)){ |
|
var p=stream.pos; |
|
if(look(stream, -2)=="{"&&(look(stream, 0)=="}"||stream.eatWhile(/\w/)&&look(stream, 0)=="}")) |
|
return "string"; |
|
else |
|
stream.pos=p;} |
|
if(/[A-Z]/.test(ch)){ |
|
var l=look(stream, -2); |
|
var p=stream.pos; |
|
stream.eatWhile(/[A-Z_]/); |
|
if(/[\da-z]/.test(look(stream, 0))){ |
|
stream.pos=p;} |
|
else{ |
|
var c=PERL[stream.current()]; |
|
if(!c) |
|
return "meta"; |
|
if(c[1]) |
|
c=c[0]; |
|
if(l!=":"){ |
|
if(c==1) |
|
return "keyword"; |
|
else if(c==2) |
|
return "def"; |
|
else if(c==3) |
|
return "atom"; |
|
else if(c==4) |
|
return "operator"; |
|
else if(c==5) |
|
return "variable-2"; |
|
else |
|
return "meta";} |
|
else |
|
return "meta";}} |
|
if(/[a-zA-Z_]/.test(ch)){ |
|
var l=look(stream, -2); |
|
stream.eatWhile(/\w/); |
|
var c=PERL[stream.current()]; |
|
if(!c) |
|
return "meta"; |
|
if(c[1]) |
|
c=c[0]; |
|
if(l!=":"){ |
|
if(c==1) |
|
return "keyword"; |
|
else if(c==2) |
|
return "def"; |
|
else if(c==3) |
|
return "atom"; |
|
else if(c==4) |
|
return "operator"; |
|
else if(c==5) |
|
return "variable-2"; |
|
else |
|
return "meta";} |
|
else |
|
return "meta";} |
|
return null;} |
|
|
|
return { |
|
startState: function() { |
|
return { |
|
tokenize: tokenPerl, |
|
chain: null, |
|
style: null, |
|
tail: null |
|
}; |
|
}, |
|
token: function(stream, state) { |
|
return (state.tokenize || tokenPerl)(stream, state); |
|
}, |
|
lineComment: '#' |
|
}; |
|
}); |
|
|
|
CodeMirror.registerHelper("wordChars", "perl", /[\w$]/); |
|
|
|
CodeMirror.defineMIME("text/x-perl", "perl"); |
|
|
|
// it's like "peek", but need for look-ahead or look-behind if index < 0 |
|
function look(stream, c){ |
|
return stream.string.charAt(stream.pos+(c||0)); |
|
} |
|
|
|
// return a part of prefix of current stream from current position |
|
function prefix(stream, c){ |
|
if(c){ |
|
var x=stream.pos-c; |
|
return stream.string.substr((x>=0?x:0),c);} |
|
else{ |
|
return stream.string.substr(0,stream.pos-1); |
|
} |
|
} |
|
|
|
// return a part of suffix of current stream from current position |
|
function suffix(stream, c){ |
|
var y=stream.string.length; |
|
var x=y-stream.pos+1; |
|
return stream.string.substr(stream.pos,(c&&c<y?c:x)); |
|
} |
|
|
|
// eating and vomiting a part of stream from current position |
|
function eatSuffix(stream, c){ |
|
var x=stream.pos+c; |
|
var y; |
|
if(x<=0) |
|
stream.pos=0; |
|
else if(x>=(y=stream.string.length-1)) |
|
stream.pos=y; |
|
else |
|
stream.pos=x; |
|
} |
|
|
|
});
|
|
|