forth.js 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. // CodeMirror, copyright (c) by Marijn Haverbeke and others
  2. // Distributed under an MIT license: https://codemirror.net/LICENSE
  3. // Author: Aliaksei Chapyzhenka
  4. ;(function (mod) {
  5. if (typeof exports == 'object' && typeof module == 'object')
  6. // CommonJS
  7. mod(require('../../lib/codemirror'))
  8. else if (typeof define == 'function' && define.amd)
  9. // AMD
  10. define(['../../lib/codemirror'], mod)
  11. // Plain browser env
  12. else mod(CodeMirror)
  13. })(function (CodeMirror) {
  14. 'use strict'
  15. function toWordList(words) {
  16. var ret = []
  17. words.split(' ').forEach(function (e) {
  18. ret.push({ name: e })
  19. })
  20. return ret
  21. }
  22. var coreWordList = toWordList(
  23. 'INVERT AND OR XOR\
  24. 2* 2/ LSHIFT RSHIFT\
  25. 0= = 0< < > U< MIN MAX\
  26. 2DROP 2DUP 2OVER 2SWAP ?DUP DEPTH DROP DUP OVER ROT SWAP\
  27. >R R> R@\
  28. + - 1+ 1- ABS NEGATE\
  29. S>D * M* UM*\
  30. FM/MOD SM/REM UM/MOD */ */MOD / /MOD MOD\
  31. HERE , @ ! CELL+ CELLS C, C@ C! CHARS 2@ 2!\
  32. ALIGN ALIGNED +! ALLOT\
  33. CHAR [CHAR] [ ] BL\
  34. FIND EXECUTE IMMEDIATE COUNT LITERAL STATE\
  35. ; DOES> >BODY\
  36. EVALUATE\
  37. SOURCE >IN\
  38. <# # #S #> HOLD SIGN BASE >NUMBER HEX DECIMAL\
  39. FILL MOVE\
  40. . CR EMIT SPACE SPACES TYPE U. .R U.R\
  41. ACCEPT\
  42. TRUE FALSE\
  43. <> U> 0<> 0>\
  44. NIP TUCK ROLL PICK\
  45. 2>R 2R@ 2R>\
  46. WITHIN UNUSED MARKER\
  47. I J\
  48. TO\
  49. COMPILE, [COMPILE]\
  50. SAVE-INPUT RESTORE-INPUT\
  51. PAD ERASE\
  52. 2LITERAL DNEGATE\
  53. D- D+ D0< D0= D2* D2/ D< D= DMAX DMIN D>S DABS\
  54. M+ M*/ D. D.R 2ROT DU<\
  55. CATCH THROW\
  56. FREE RESIZE ALLOCATE\
  57. CS-PICK CS-ROLL\
  58. GET-CURRENT SET-CURRENT FORTH-WORDLIST GET-ORDER SET-ORDER\
  59. PREVIOUS SEARCH-WORDLIST WORDLIST FIND ALSO ONLY FORTH DEFINITIONS ORDER\
  60. -TRAILING /STRING SEARCH COMPARE CMOVE CMOVE> BLANK SLITERAL'
  61. )
  62. var immediateWordList = toWordList('IF ELSE THEN BEGIN WHILE REPEAT UNTIL RECURSE [IF] [ELSE] [THEN] ?DO DO LOOP +LOOP UNLOOP LEAVE EXIT AGAIN CASE OF ENDOF ENDCASE')
  63. CodeMirror.defineMode('forth', function () {
  64. function searchWordList(wordList, word) {
  65. var i
  66. for (i = wordList.length - 1; i >= 0; i--) {
  67. if (wordList[i].name === word.toUpperCase()) {
  68. return wordList[i]
  69. }
  70. }
  71. return undefined
  72. }
  73. return {
  74. startState: function () {
  75. return {
  76. state: '',
  77. base: 10,
  78. coreWordList: coreWordList,
  79. immediateWordList: immediateWordList,
  80. wordList: [],
  81. }
  82. },
  83. token: function (stream, stt) {
  84. var mat
  85. if (stream.eatSpace()) {
  86. return null
  87. }
  88. if (stt.state === '') {
  89. // interpretation
  90. if (stream.match(/^(\]|:NONAME)(\s|$)/i)) {
  91. stt.state = ' compilation'
  92. return 'builtin compilation'
  93. }
  94. mat = stream.match(/^(\:)\s+(\S+)(\s|$)+/)
  95. if (mat) {
  96. stt.wordList.push({ name: mat[2].toUpperCase() })
  97. stt.state = ' compilation'
  98. return 'def' + stt.state
  99. }
  100. mat = stream.match(/^(VARIABLE|2VARIABLE|CONSTANT|2CONSTANT|CREATE|POSTPONE|VALUE|WORD)\s+(\S+)(\s|$)+/i)
  101. if (mat) {
  102. stt.wordList.push({ name: mat[2].toUpperCase() })
  103. return 'def' + stt.state
  104. }
  105. mat = stream.match(/^(\'|\[\'\])\s+(\S+)(\s|$)+/)
  106. if (mat) {
  107. return 'builtin' + stt.state
  108. }
  109. } else {
  110. // compilation
  111. // ; [
  112. if (stream.match(/^(\;|\[)(\s)/)) {
  113. stt.state = ''
  114. stream.backUp(1)
  115. return 'builtin compilation'
  116. }
  117. if (stream.match(/^(\;|\[)($)/)) {
  118. stt.state = ''
  119. return 'builtin compilation'
  120. }
  121. if (stream.match(/^(POSTPONE)\s+\S+(\s|$)+/)) {
  122. return 'builtin'
  123. }
  124. }
  125. // dynamic wordlist
  126. mat = stream.match(/^(\S+)(\s+|$)/)
  127. if (mat) {
  128. if (searchWordList(stt.wordList, mat[1]) !== undefined) {
  129. return 'variable' + stt.state
  130. }
  131. // comments
  132. if (mat[1] === '\\') {
  133. stream.skipToEnd()
  134. return 'comment' + stt.state
  135. }
  136. // core words
  137. if (searchWordList(stt.coreWordList, mat[1]) !== undefined) {
  138. return 'builtin' + stt.state
  139. }
  140. if (searchWordList(stt.immediateWordList, mat[1]) !== undefined) {
  141. return 'keyword' + stt.state
  142. }
  143. if (mat[1] === '(') {
  144. stream.eatWhile(function (s) {
  145. return s !== ')'
  146. })
  147. stream.eat(')')
  148. return 'comment' + stt.state
  149. }
  150. // // strings
  151. if (mat[1] === '.(') {
  152. stream.eatWhile(function (s) {
  153. return s !== ')'
  154. })
  155. stream.eat(')')
  156. return 'string' + stt.state
  157. }
  158. if (mat[1] === 'S"' || mat[1] === '."' || mat[1] === 'C"') {
  159. stream.eatWhile(function (s) {
  160. return s !== '"'
  161. })
  162. stream.eat('"')
  163. return 'string' + stt.state
  164. }
  165. // numbers
  166. if (mat[1] - 0xfffffffff) {
  167. return 'number' + stt.state
  168. }
  169. // if (mat[1].match(/^[-+]?[0-9]+\.[0-9]*/)) {
  170. // return 'number' + stt.state;
  171. // }
  172. return 'atom' + stt.state
  173. }
  174. },
  175. }
  176. })
  177. CodeMirror.defineMIME('text/x-forth', 'forth')
  178. })