poly-cms 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (460) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +10 -0
  3. data/Gemfile +27 -0
  4. data/Gemfile.lock +95 -0
  5. data/README.md +3 -0
  6. data/Rakefile +45 -0
  7. data/bin/poly +88 -0
  8. data/config.ru +3 -0
  9. data/core/admin/assets/images/icons/cache-clear.svg +11 -0
  10. data/core/admin/assets/images/icons/delete-dark.svg +10 -0
  11. data/core/admin/assets/images/icons/delete.svg +10 -0
  12. data/core/admin/assets/images/icons/handle.svg +12 -0
  13. data/core/admin/assets/images/icons/logout.svg +13 -0
  14. data/core/admin/assets/images/icons/move.svg +10 -0
  15. data/core/admin/assets/images/icons/new.svg +9 -0
  16. data/core/admin/assets/images/icons/prompt.svg +11 -0
  17. data/core/admin/assets/images/icons/refresh.svg +40 -0
  18. data/core/admin/assets/images/icons/save.svg +7 -0
  19. data/core/admin/assets/images/poly-small-boxed.svg +51 -0
  20. data/core/admin/assets/images/poly-small.svg +52 -0
  21. data/core/admin/assets/images/poly.svg +36 -0
  22. data/core/admin/assets/js/app/file-ext.js +15 -0
  23. data/core/admin/assets/js/app/views/editor.js +300 -0
  24. data/core/admin/assets/js/app/views/git.js +65 -0
  25. data/core/admin/assets/js/app/views/main.js +59 -0
  26. data/core/admin/assets/js/main.js +44 -0
  27. data/core/admin/assets/js/routes.js +25 -0
  28. data/core/admin/assets/js/vendor/backbone.js +4 -0
  29. data/core/admin/assets/js/vendor/codemirror/.gitattributes +8 -0
  30. data/core/admin/assets/js/vendor/codemirror/.gitignore +6 -0
  31. data/core/admin/assets/js/vendor/codemirror/.travis.yml +3 -0
  32. data/core/admin/assets/js/vendor/codemirror/addon/comment/comment.js +145 -0
  33. data/core/admin/assets/js/vendor/codemirror/addon/comment/continuecomment.js +54 -0
  34. data/core/admin/assets/js/vendor/codemirror/addon/dialog/dialog.css +32 -0
  35. data/core/admin/assets/js/vendor/codemirror/addon/dialog/dialog.js +80 -0
  36. data/core/admin/assets/js/vendor/codemirror/addon/display/fullscreen.css +6 -0
  37. data/core/admin/assets/js/vendor/codemirror/addon/display/fullscreen.js +30 -0
  38. data/core/admin/assets/js/vendor/codemirror/addon/display/placeholder.js +54 -0
  39. data/core/admin/assets/js/vendor/codemirror/addon/edit/closebrackets.js +82 -0
  40. data/core/admin/assets/js/vendor/codemirror/addon/edit/closetag.js +87 -0
  41. data/core/admin/assets/js/vendor/codemirror/addon/edit/continuelist.js +25 -0
  42. data/core/admin/assets/js/vendor/codemirror/addon/edit/matchbrackets.js +86 -0
  43. data/core/admin/assets/js/vendor/codemirror/addon/edit/matchtags.js +56 -0
  44. data/core/admin/assets/js/vendor/codemirror/addon/edit/trailingspace.js +15 -0
  45. data/core/admin/assets/js/vendor/codemirror/addon/fold/brace-fold.js +93 -0
  46. data/core/admin/assets/js/vendor/codemirror/addon/fold/comment-fold.js +40 -0
  47. data/core/admin/assets/js/vendor/codemirror/addon/fold/foldcode.js +75 -0
  48. data/core/admin/assets/js/vendor/codemirror/addon/fold/foldgutter.css +21 -0
  49. data/core/admin/assets/js/vendor/codemirror/addon/fold/foldgutter.js +124 -0
  50. data/core/admin/assets/js/vendor/codemirror/addon/fold/indent-fold.js +26 -0
  51. data/core/admin/assets/js/vendor/codemirror/addon/fold/xml-fold.js +167 -0
  52. data/core/admin/assets/js/vendor/codemirror/addon/hint/anyword-hint.js +34 -0
  53. data/core/admin/assets/js/vendor/codemirror/addon/hint/css-hint.js +50 -0
  54. data/core/admin/assets/js/vendor/codemirror/addon/hint/html-hint.js +337 -0
  55. data/core/admin/assets/js/vendor/codemirror/addon/hint/javascript-hint.js +130 -0
  56. data/core/admin/assets/js/vendor/codemirror/addon/hint/pig-hint.js +121 -0
  57. data/core/admin/assets/js/vendor/codemirror/addon/hint/python-hint.js +95 -0
  58. data/core/admin/assets/js/vendor/codemirror/addon/hint/show-hint.css +38 -0
  59. data/core/admin/assets/js/vendor/codemirror/addon/hint/show-hint.js +274 -0
  60. data/core/admin/assets/js/vendor/codemirror/addon/hint/sql-hint.js +105 -0
  61. data/core/admin/assets/js/vendor/codemirror/addon/hint/xml-hint.js +69 -0
  62. data/core/admin/assets/js/vendor/codemirror/addon/lint/coffeescript-lint.js +27 -0
  63. data/core/admin/assets/js/vendor/codemirror/addon/lint/css-lint.js +19 -0
  64. data/core/admin/assets/js/vendor/codemirror/addon/lint/javascript-lint.js +126 -0
  65. data/core/admin/assets/js/vendor/codemirror/addon/lint/json-lint.js +17 -0
  66. data/core/admin/assets/js/vendor/codemirror/addon/lint/lint.css +73 -0
  67. data/core/admin/assets/js/vendor/codemirror/addon/lint/lint.js +203 -0
  68. data/core/admin/assets/js/vendor/codemirror/addon/merge/dep/diff_match_patch.js +50 -0
  69. data/core/admin/assets/js/vendor/codemirror/addon/merge/merge.css +92 -0
  70. data/core/admin/assets/js/vendor/codemirror/addon/merge/merge.js +474 -0
  71. data/core/admin/assets/js/vendor/codemirror/addon/mode/loadmode.js +51 -0
  72. data/core/admin/assets/js/vendor/codemirror/addon/mode/multiplex.js +101 -0
  73. data/core/admin/assets/js/vendor/codemirror/addon/mode/multiplex_test.js +30 -0
  74. data/core/admin/assets/js/vendor/codemirror/addon/mode/overlay.js +59 -0
  75. data/core/admin/assets/js/vendor/codemirror/addon/runmode/colorize.js +29 -0
  76. data/core/admin/assets/js/vendor/codemirror/addon/runmode/runmode-standalone.js +136 -0
  77. data/core/admin/assets/js/vendor/codemirror/addon/runmode/runmode.js +56 -0
  78. data/core/admin/assets/js/vendor/codemirror/addon/runmode/runmode.node.js +103 -0
  79. data/core/admin/assets/js/vendor/codemirror/addon/scroll/scrollpastend.js +34 -0
  80. data/core/admin/assets/js/vendor/codemirror/addon/search/match-highlighter.js +91 -0
  81. data/core/admin/assets/js/vendor/codemirror/addon/search/search.js +133 -0
  82. data/core/admin/assets/js/vendor/codemirror/addon/search/searchcursor.js +143 -0
  83. data/core/admin/assets/js/vendor/codemirror/addon/selection/active-line.js +39 -0
  84. data/core/admin/assets/js/vendor/codemirror/addon/selection/mark-selection.js +108 -0
  85. data/core/admin/assets/js/vendor/codemirror/addon/tern/tern.css +85 -0
  86. data/core/admin/assets/js/vendor/codemirror/addon/tern/tern.js +632 -0
  87. data/core/admin/assets/js/vendor/codemirror/addon/tern/worker.js +41 -0
  88. data/core/admin/assets/js/vendor/codemirror/addon/wrap/hardwrap.js +99 -0
  89. data/core/admin/assets/js/vendor/codemirror/bin/authors.sh +6 -0
  90. data/core/admin/assets/js/vendor/codemirror/bin/compress +92 -0
  91. data/core/admin/assets/js/vendor/codemirror/bin/lint +16 -0
  92. data/core/admin/assets/js/vendor/codemirror/bin/source-highlight +61 -0
  93. data/core/admin/assets/js/vendor/codemirror/bower.json +15 -0
  94. data/core/admin/assets/js/vendor/codemirror/demo/activeline.html +78 -0
  95. data/core/admin/assets/js/vendor/codemirror/demo/anywordhint.html +79 -0
  96. data/core/admin/assets/js/vendor/codemirror/demo/bidi.html +74 -0
  97. data/core/admin/assets/js/vendor/codemirror/demo/btree.html +86 -0
  98. data/core/admin/assets/js/vendor/codemirror/demo/buffers.html +109 -0
  99. data/core/admin/assets/js/vendor/codemirror/demo/changemode.html +59 -0
  100. data/core/admin/assets/js/vendor/codemirror/demo/closebrackets.html +63 -0
  101. data/core/admin/assets/js/vendor/codemirror/demo/closetag.html +40 -0
  102. data/core/admin/assets/js/vendor/codemirror/demo/complete.html +80 -0
  103. data/core/admin/assets/js/vendor/codemirror/demo/emacs.html +75 -0
  104. data/core/admin/assets/js/vendor/codemirror/demo/folding.html +75 -0
  105. data/core/admin/assets/js/vendor/codemirror/demo/fullscreen.html +130 -0
  106. data/core/admin/assets/js/vendor/codemirror/demo/hardwrap.html +69 -0
  107. data/core/admin/assets/js/vendor/codemirror/demo/html5complete.html +54 -0
  108. data/core/admin/assets/js/vendor/codemirror/demo/indentwrap.html +58 -0
  109. data/core/admin/assets/js/vendor/codemirror/demo/lint.html +171 -0
  110. data/core/admin/assets/js/vendor/codemirror/demo/loadmode.html +49 -0
  111. data/core/admin/assets/js/vendor/codemirror/demo/marker.html +52 -0
  112. data/core/admin/assets/js/vendor/codemirror/demo/markselection.html +45 -0
  113. data/core/admin/assets/js/vendor/codemirror/demo/matchhighlighter.html +47 -0
  114. data/core/admin/assets/js/vendor/codemirror/demo/matchtags.html +49 -0
  115. data/core/admin/assets/js/vendor/codemirror/demo/merge.html +82 -0
  116. data/core/admin/assets/js/vendor/codemirror/demo/multiplex.html +75 -0
  117. data/core/admin/assets/js/vendor/codemirror/demo/mustache.html +68 -0
  118. data/core/admin/assets/js/vendor/codemirror/demo/placeholder.html +45 -0
  119. data/core/admin/assets/js/vendor/codemirror/demo/preview.html +88 -0
  120. data/core/admin/assets/js/vendor/codemirror/demo/resize.html +58 -0
  121. data/core/admin/assets/js/vendor/codemirror/demo/runmode.html +62 -0
  122. data/core/admin/assets/js/vendor/codemirror/demo/search.html +94 -0
  123. data/core/admin/assets/js/vendor/codemirror/demo/spanaffectswrapping_shim.html +85 -0
  124. data/core/admin/assets/js/vendor/codemirror/demo/tern.html +128 -0
  125. data/core/admin/assets/js/vendor/codemirror/demo/theme.html +121 -0
  126. data/core/admin/assets/js/vendor/codemirror/demo/trailingspace.html +48 -0
  127. data/core/admin/assets/js/vendor/codemirror/demo/variableheight.html +61 -0
  128. data/core/admin/assets/js/vendor/codemirror/demo/vim.html +74 -0
  129. data/core/admin/assets/js/vendor/codemirror/demo/visibletabs.html +62 -0
  130. data/core/admin/assets/js/vendor/codemirror/demo/widget.html +85 -0
  131. data/core/admin/assets/js/vendor/codemirror/demo/xmlcomplete.html +116 -0
  132. data/core/admin/assets/js/vendor/codemirror/doc/activebookmark.js +42 -0
  133. data/core/admin/assets/js/vendor/codemirror/doc/compress.html +231 -0
  134. data/core/admin/assets/js/vendor/codemirror/doc/docs.css +226 -0
  135. data/core/admin/assets/js/vendor/codemirror/doc/internals.html +503 -0
  136. data/core/admin/assets/js/vendor/codemirror/doc/logo.png +0 -0
  137. data/core/admin/assets/js/vendor/codemirror/doc/logo.svg +147 -0
  138. data/core/admin/assets/js/vendor/codemirror/doc/manual.html +2512 -0
  139. data/core/admin/assets/js/vendor/codemirror/doc/realworld.html +134 -0
  140. data/core/admin/assets/js/vendor/codemirror/doc/releases.html +779 -0
  141. data/core/admin/assets/js/vendor/codemirror/doc/reporting.html +61 -0
  142. data/core/admin/assets/js/vendor/codemirror/doc/upgrade_v2.2.html +96 -0
  143. data/core/admin/assets/js/vendor/codemirror/doc/upgrade_v3.html +230 -0
  144. data/core/admin/assets/js/vendor/codemirror/index.html +192 -0
  145. data/core/admin/assets/js/vendor/codemirror/keymap/emacs.js +387 -0
  146. data/core/admin/assets/js/vendor/codemirror/keymap/extra.js +43 -0
  147. data/core/admin/assets/js/vendor/codemirror/keymap/vim.js +3704 -0
  148. data/core/admin/assets/js/vendor/codemirror/lib/codemirror.css +263 -0
  149. data/core/admin/assets/js/vendor/codemirror/lib/codemirror.js +5910 -0
  150. data/core/admin/assets/js/vendor/codemirror/mode/apl/apl.js +160 -0
  151. data/core/admin/assets/js/vendor/codemirror/mode/apl/index.html +72 -0
  152. data/core/admin/assets/js/vendor/codemirror/mode/asterisk/asterisk.js +183 -0
  153. data/core/admin/assets/js/vendor/codemirror/mode/asterisk/index.html +154 -0
  154. data/core/admin/assets/js/vendor/codemirror/mode/clike/clike.js +362 -0
  155. data/core/admin/assets/js/vendor/codemirror/mode/clike/index.html +195 -0
  156. data/core/admin/assets/js/vendor/codemirror/mode/clike/scala.html +767 -0
  157. data/core/admin/assets/js/vendor/codemirror/mode/clojure/clojure.js +224 -0
  158. data/core/admin/assets/js/vendor/codemirror/mode/clojure/index.html +88 -0
  159. data/core/admin/assets/js/vendor/codemirror/mode/cobol/cobol.js +240 -0
  160. data/core/admin/assets/js/vendor/codemirror/mode/cobol/index.html +210 -0
  161. data/core/admin/assets/js/vendor/codemirror/mode/coffeescript/coffeescript.js +349 -0
  162. data/core/admin/assets/js/vendor/codemirror/mode/coffeescript/index.html +740 -0
  163. data/core/admin/assets/js/vendor/codemirror/mode/commonlisp/commonlisp.js +105 -0
  164. data/core/admin/assets/js/vendor/codemirror/mode/commonlisp/index.html +177 -0
  165. data/core/admin/assets/js/vendor/codemirror/mode/css/css.js +639 -0
  166. data/core/admin/assets/js/vendor/codemirror/mode/css/index.html +70 -0
  167. data/core/admin/assets/js/vendor/codemirror/mode/css/scss.html +157 -0
  168. data/core/admin/assets/js/vendor/codemirror/mode/css/scss_test.js +84 -0
  169. data/core/admin/assets/js/vendor/codemirror/mode/css/test.js +130 -0
  170. data/core/admin/assets/js/vendor/codemirror/mode/d/d.js +205 -0
  171. data/core/admin/assets/js/vendor/codemirror/mode/d/index.html +273 -0
  172. data/core/admin/assets/js/vendor/codemirror/mode/diff/diff.js +32 -0
  173. data/core/admin/assets/js/vendor/codemirror/mode/diff/index.html +117 -0
  174. data/core/admin/assets/js/vendor/codemirror/mode/dtd/dtd.js +127 -0
  175. data/core/admin/assets/js/vendor/codemirror/mode/dtd/index.html +89 -0
  176. data/core/admin/assets/js/vendor/codemirror/mode/ecl/ecl.js +192 -0
  177. data/core/admin/assets/js/vendor/codemirror/mode/ecl/index.html +52 -0
  178. data/core/admin/assets/js/vendor/codemirror/mode/eiffel/eiffel.js +147 -0
  179. data/core/admin/assets/js/vendor/codemirror/mode/eiffel/index.html +430 -0
  180. data/core/admin/assets/js/vendor/codemirror/mode/erlang/erlang.js +484 -0
  181. data/core/admin/assets/js/vendor/codemirror/mode/erlang/index.html +75 -0
  182. data/core/admin/assets/js/vendor/codemirror/mode/fortran/fortran.js +173 -0
  183. data/core/admin/assets/js/vendor/codemirror/mode/fortran/index.html +81 -0
  184. data/core/admin/assets/js/vendor/codemirror/mode/gas/gas.js +330 -0
  185. data/core/admin/assets/js/vendor/codemirror/mode/gas/index.html +68 -0
  186. data/core/admin/assets/js/vendor/codemirror/mode/gfm/gfm.js +97 -0
  187. data/core/admin/assets/js/vendor/codemirror/mode/gfm/index.html +82 -0
  188. data/core/admin/assets/js/vendor/codemirror/mode/gfm/test.js +112 -0
  189. data/core/admin/assets/js/vendor/codemirror/mode/gherkin/gherkin.js +168 -0
  190. data/core/admin/assets/js/vendor/codemirror/mode/gherkin/index.html +48 -0
  191. data/core/admin/assets/js/vendor/codemirror/mode/go/go.js +168 -0
  192. data/core/admin/assets/js/vendor/codemirror/mode/go/index.html +85 -0
  193. data/core/admin/assets/js/vendor/codemirror/mode/groovy/groovy.js +211 -0
  194. data/core/admin/assets/js/vendor/codemirror/mode/groovy/index.html +84 -0
  195. data/core/admin/assets/js/vendor/codemirror/mode/haml/haml.js +153 -0
  196. data/core/admin/assets/js/vendor/codemirror/mode/haml/index.html +79 -0
  197. data/core/admin/assets/js/vendor/codemirror/mode/haml/test.js +94 -0
  198. data/core/admin/assets/js/vendor/codemirror/mode/haskell/haskell.js +250 -0
  199. data/core/admin/assets/js/vendor/codemirror/mode/haskell/index.html +73 -0
  200. data/core/admin/assets/js/vendor/codemirror/mode/haxe/haxe.js +429 -0
  201. data/core/admin/assets/js/vendor/codemirror/mode/haxe/index.html +103 -0
  202. data/core/admin/assets/js/vendor/codemirror/mode/htmlembedded/htmlembedded.js +73 -0
  203. data/core/admin/assets/js/vendor/codemirror/mode/htmlembedded/index.html +60 -0
  204. data/core/admin/assets/js/vendor/codemirror/mode/htmlmixed/htmlmixed.js +104 -0
  205. data/core/admin/assets/js/vendor/codemirror/mode/htmlmixed/index.html +85 -0
  206. data/core/admin/assets/js/vendor/codemirror/mode/http/http.js +98 -0
  207. data/core/admin/assets/js/vendor/codemirror/mode/http/index.html +45 -0
  208. data/core/admin/assets/js/vendor/codemirror/mode/index.html +112 -0
  209. data/core/admin/assets/js/vendor/codemirror/mode/jade/index.html +66 -0
  210. data/core/admin/assets/js/vendor/codemirror/mode/jade/jade.js +90 -0
  211. data/core/admin/assets/js/vendor/codemirror/mode/javascript/index.html +107 -0
  212. data/core/admin/assets/js/vendor/codemirror/mode/javascript/javascript.js +480 -0
  213. data/core/admin/assets/js/vendor/codemirror/mode/javascript/test.js +10 -0
  214. data/core/admin/assets/js/vendor/codemirror/mode/javascript/typescript.html +61 -0
  215. data/core/admin/assets/js/vendor/codemirror/mode/jinja2/index.html +50 -0
  216. data/core/admin/assets/js/vendor/codemirror/mode/jinja2/jinja2.js +42 -0
  217. data/core/admin/assets/js/vendor/codemirror/mode/less/index.html +753 -0
  218. data/core/admin/assets/js/vendor/codemirror/mode/less/less.js +344 -0
  219. data/core/admin/assets/js/vendor/codemirror/mode/livescript/index.html +459 -0
  220. data/core/admin/assets/js/vendor/codemirror/mode/livescript/livescript.js +267 -0
  221. data/core/admin/assets/js/vendor/codemirror/mode/livescript/livescript.ls +266 -0
  222. data/core/admin/assets/js/vendor/codemirror/mode/lua/index.html +86 -0
  223. data/core/admin/assets/js/vendor/codemirror/mode/lua/lua.js +144 -0
  224. data/core/admin/assets/js/vendor/codemirror/mode/markdown/index.html +359 -0
  225. data/core/admin/assets/js/vendor/codemirror/mode/markdown/markdown.js +551 -0
  226. data/core/admin/assets/js/vendor/codemirror/mode/markdown/test.js +656 -0
  227. data/core/admin/assets/js/vendor/codemirror/mode/meta.js +89 -0
  228. data/core/admin/assets/js/vendor/codemirror/mode/mirc/index.html +161 -0
  229. data/core/admin/assets/js/vendor/codemirror/mode/mirc/mirc.js +177 -0
  230. data/core/admin/assets/js/vendor/codemirror/mode/nginx/index.html +181 -0
  231. data/core/admin/assets/js/vendor/codemirror/mode/nginx/nginx.js +163 -0
  232. data/core/admin/assets/js/vendor/codemirror/mode/ntriples/index.html +45 -0
  233. data/core/admin/assets/js/vendor/codemirror/mode/ntriples/ntriples.js +170 -0
  234. data/core/admin/assets/js/vendor/codemirror/mode/ocaml/index.html +146 -0
  235. data/core/admin/assets/js/vendor/codemirror/mode/ocaml/ocaml.js +116 -0
  236. data/core/admin/assets/js/vendor/codemirror/mode/octave/index.html +95 -0
  237. data/core/admin/assets/js/vendor/codemirror/mode/octave/octave.js +118 -0
  238. data/core/admin/assets/js/vendor/codemirror/mode/pascal/index.html +61 -0
  239. data/core/admin/assets/js/vendor/codemirror/mode/pascal/pascal.js +94 -0
  240. data/core/admin/assets/js/vendor/codemirror/mode/perl/index.html +75 -0
  241. data/core/admin/assets/js/vendor/codemirror/mode/perl/perl.js +816 -0
  242. data/core/admin/assets/js/vendor/codemirror/mode/php/index.html +62 -0
  243. data/core/admin/assets/js/vendor/codemirror/mode/php/php.js +132 -0
  244. data/core/admin/assets/js/vendor/codemirror/mode/pig/index.html +55 -0
  245. data/core/admin/assets/js/vendor/codemirror/mode/pig/pig.js +171 -0
  246. data/core/admin/assets/js/vendor/codemirror/mode/properties/index.html +53 -0
  247. data/core/admin/assets/js/vendor/codemirror/mode/properties/properties.js +63 -0
  248. data/core/admin/assets/js/vendor/codemirror/mode/python/index.html +187 -0
  249. data/core/admin/assets/js/vendor/codemirror/mode/python/python.js +368 -0
  250. data/core/admin/assets/js/vendor/codemirror/mode/q/index.html +144 -0
  251. data/core/admin/assets/js/vendor/codemirror/mode/q/q.js +124 -0
  252. data/core/admin/assets/js/vendor/codemirror/mode/r/index.html +86 -0
  253. data/core/admin/assets/js/vendor/codemirror/mode/r/r.js +141 -0
  254. data/core/admin/assets/js/vendor/codemirror/mode/rpm/changes/changes.js +19 -0
  255. data/core/admin/assets/js/vendor/codemirror/mode/rpm/changes/index.html +67 -0
  256. data/core/admin/assets/js/vendor/codemirror/mode/rpm/spec/index.html +114 -0
  257. data/core/admin/assets/js/vendor/codemirror/mode/rpm/spec/spec.css +5 -0
  258. data/core/admin/assets/js/vendor/codemirror/mode/rpm/spec/spec.js +66 -0
  259. data/core/admin/assets/js/vendor/codemirror/mode/rst/index.html +534 -0
  260. data/core/admin/assets/js/vendor/codemirror/mode/rst/rst.js +560 -0
  261. data/core/admin/assets/js/vendor/codemirror/mode/ruby/index.html +185 -0
  262. data/core/admin/assets/js/vendor/codemirror/mode/ruby/ruby.js +247 -0
  263. data/core/admin/assets/js/vendor/codemirror/mode/rust/index.html +61 -0
  264. data/core/admin/assets/js/vendor/codemirror/mode/rust/rust.js +436 -0
  265. data/core/admin/assets/js/vendor/codemirror/mode/sass/index.html +66 -0
  266. data/core/admin/assets/js/vendor/codemirror/mode/sass/sass.js +330 -0
  267. data/core/admin/assets/js/vendor/codemirror/mode/scheme/index.html +77 -0
  268. data/core/admin/assets/js/vendor/codemirror/mode/scheme/scheme.js +232 -0
  269. data/core/admin/assets/js/vendor/codemirror/mode/shell/index.html +66 -0
  270. data/core/admin/assets/js/vendor/codemirror/mode/shell/shell.js +118 -0
  271. data/core/admin/assets/js/vendor/codemirror/mode/sieve/index.html +93 -0
  272. data/core/admin/assets/js/vendor/codemirror/mode/sieve/sieve.js +183 -0
  273. data/core/admin/assets/js/vendor/codemirror/mode/smalltalk/index.html +68 -0
  274. data/core/admin/assets/js/vendor/codemirror/mode/smalltalk/smalltalk.js +151 -0
  275. data/core/admin/assets/js/vendor/codemirror/mode/smarty/index.html +136 -0
  276. data/core/admin/assets/js/vendor/codemirror/mode/smarty/smarty.js +205 -0
  277. data/core/admin/assets/js/vendor/codemirror/mode/smartymixed/index.html +114 -0
  278. data/core/admin/assets/js/vendor/codemirror/mode/smartymixed/smartymixed.js +175 -0
  279. data/core/admin/assets/js/vendor/codemirror/mode/sparql/index.html +54 -0
  280. data/core/admin/assets/js/vendor/codemirror/mode/sparql/sparql.js +145 -0
  281. data/core/admin/assets/js/vendor/codemirror/mode/sql/index.html +75 -0
  282. data/core/admin/assets/js/vendor/codemirror/mode/sql/sql.js +365 -0
  283. data/core/admin/assets/js/vendor/codemirror/mode/stex/index.html +110 -0
  284. data/core/admin/assets/js/vendor/codemirror/mode/stex/stex.js +246 -0
  285. data/core/admin/assets/js/vendor/codemirror/mode/stex/test.js +120 -0
  286. data/core/admin/assets/js/vendor/codemirror/mode/tcl/index.html +143 -0
  287. data/core/admin/assets/js/vendor/codemirror/mode/tcl/tcl.js +131 -0
  288. data/core/admin/assets/js/vendor/codemirror/mode/tiddlywiki/index.html +155 -0
  289. data/core/admin/assets/js/vendor/codemirror/mode/tiddlywiki/tiddlywiki.css +14 -0
  290. data/core/admin/assets/js/vendor/codemirror/mode/tiddlywiki/tiddlywiki.js +353 -0
  291. data/core/admin/assets/js/vendor/codemirror/mode/tiki/index.html +95 -0
  292. data/core/admin/assets/js/vendor/codemirror/mode/tiki/tiki.css +26 -0
  293. data/core/admin/assets/js/vendor/codemirror/mode/tiki/tiki.js +308 -0
  294. data/core/admin/assets/js/vendor/codemirror/mode/toml/index.html +73 -0
  295. data/core/admin/assets/js/vendor/codemirror/mode/toml/toml.js +71 -0
  296. data/core/admin/assets/js/vendor/codemirror/mode/turtle/index.html +51 -0
  297. data/core/admin/assets/js/vendor/codemirror/mode/turtle/turtle.js +145 -0
  298. data/core/admin/assets/js/vendor/codemirror/mode/vb/index.html +103 -0
  299. data/core/admin/assets/js/vendor/codemirror/mode/vb/vb.js +259 -0
  300. data/core/admin/assets/js/vendor/codemirror/mode/vbscript/index.html +55 -0
  301. data/core/admin/assets/js/vendor/codemirror/mode/vbscript/vbscript.js +334 -0
  302. data/core/admin/assets/js/vendor/codemirror/mode/velocity/index.html +119 -0
  303. data/core/admin/assets/js/vendor/codemirror/mode/velocity/velocity.js +186 -0
  304. data/core/admin/assets/js/vendor/codemirror/mode/verilog/index.html +132 -0
  305. data/core/admin/assets/js/vendor/codemirror/mode/verilog/verilog.js +182 -0
  306. data/core/admin/assets/js/vendor/codemirror/mode/xml/index.html +57 -0
  307. data/core/admin/assets/js/vendor/codemirror/mode/xml/xml.js +345 -0
  308. data/core/admin/assets/js/vendor/codemirror/mode/xquery/index.html +210 -0
  309. data/core/admin/assets/js/vendor/codemirror/mode/xquery/test.js +64 -0
  310. data/core/admin/assets/js/vendor/codemirror/mode/xquery/xquery.js +432 -0
  311. data/core/admin/assets/js/vendor/codemirror/mode/yaml/index.html +80 -0
  312. data/core/admin/assets/js/vendor/codemirror/mode/yaml/yaml.js +97 -0
  313. data/core/admin/assets/js/vendor/codemirror/mode/z80/index.html +52 -0
  314. data/core/admin/assets/js/vendor/codemirror/mode/z80/z80.js +85 -0
  315. data/core/admin/assets/js/vendor/codemirror/package.json +19 -0
  316. data/core/admin/assets/js/vendor/codemirror/test/comment_test.js +51 -0
  317. data/core/admin/assets/js/vendor/codemirror/test/doc_test.js +329 -0
  318. data/core/admin/assets/js/vendor/codemirror/test/driver.js +139 -0
  319. data/core/admin/assets/js/vendor/codemirror/test/emacs_test.js +135 -0
  320. data/core/admin/assets/js/vendor/codemirror/test/index.html +209 -0
  321. data/core/admin/assets/js/vendor/codemirror/test/lint/acorn.js +1593 -0
  322. data/core/admin/assets/js/vendor/codemirror/test/lint/lint.js +139 -0
  323. data/core/admin/assets/js/vendor/codemirror/test/lint/walk.js +216 -0
  324. data/core/admin/assets/js/vendor/codemirror/test/mode_test.css +10 -0
  325. data/core/admin/assets/js/vendor/codemirror/test/mode_test.js +200 -0
  326. data/core/admin/assets/js/vendor/codemirror/test/phantom_driver.js +31 -0
  327. data/core/admin/assets/js/vendor/codemirror/test/run.js +34 -0
  328. data/core/admin/assets/js/vendor/codemirror/test/test.js +1562 -0
  329. data/core/admin/assets/js/vendor/codemirror/test/vim_test.js +2391 -0
  330. data/core/admin/assets/js/vendor/codemirror/theme/3024-day.css +34 -0
  331. data/core/admin/assets/js/vendor/codemirror/theme/3024-night.css +34 -0
  332. data/core/admin/assets/js/vendor/codemirror/theme/ambiance-mobile.css +5 -0
  333. data/core/admin/assets/js/vendor/codemirror/theme/ambiance.css +75 -0
  334. data/core/admin/assets/js/vendor/codemirror/theme/base16-dark.css +34 -0
  335. data/core/admin/assets/js/vendor/codemirror/theme/base16-light.css +34 -0
  336. data/core/admin/assets/js/vendor/codemirror/theme/blackboard.css +28 -0
  337. data/core/admin/assets/js/vendor/codemirror/theme/cobalt.css +21 -0
  338. data/core/admin/assets/js/vendor/codemirror/theme/eclipse.css +23 -0
  339. data/core/admin/assets/js/vendor/codemirror/theme/elegant.css +13 -0
  340. data/core/admin/assets/js/vendor/codemirror/theme/erlang-dark.css +30 -0
  341. data/core/admin/assets/js/vendor/codemirror/theme/lesser-dark.css +47 -0
  342. data/core/admin/assets/js/vendor/codemirror/theme/mbo.css +35 -0
  343. data/core/admin/assets/js/vendor/codemirror/theme/midnight.css +43 -0
  344. data/core/admin/assets/js/vendor/codemirror/theme/monokai.css +29 -0
  345. data/core/admin/assets/js/vendor/codemirror/theme/neat.css +12 -0
  346. data/core/admin/assets/js/vendor/codemirror/theme/night.css +24 -0
  347. data/core/admin/assets/js/vendor/codemirror/theme/paraiso-dark.css +34 -0
  348. data/core/admin/assets/js/vendor/codemirror/theme/paraiso-light.css +34 -0
  349. data/core/admin/assets/js/vendor/codemirror/theme/rubyblue.css +23 -0
  350. data/core/admin/assets/js/vendor/codemirror/theme/solarized-mod.css +181 -0
  351. data/core/admin/assets/js/vendor/codemirror/theme/solarized.css +180 -0
  352. data/core/admin/assets/js/vendor/codemirror/theme/the-matrix.css +26 -0
  353. data/core/admin/assets/js/vendor/codemirror/theme/tomorrow-night-eighties.css +34 -0
  354. data/core/admin/assets/js/vendor/codemirror/theme/twilight.css +28 -0
  355. data/core/admin/assets/js/vendor/codemirror/theme/vibrant-ink.css +30 -0
  356. data/core/admin/assets/js/vendor/codemirror/theme/xq-dark.css +49 -0
  357. data/core/admin/assets/js/vendor/codemirror/theme/xq-light.css +43 -0
  358. data/core/admin/assets/js/vendor/fastclick.js +777 -0
  359. data/core/admin/assets/js/vendor/jquery.js +6 -0
  360. data/core/admin/assets/js/vendor/lodash.underscore.js +38 -0
  361. data/core/admin/assets/js/vendor/require.js +36 -0
  362. data/core/admin/assets/scss/bootstrap.scss +18 -0
  363. data/core/admin/assets/scss/modules/_all.scss +7 -0
  364. data/core/admin/assets/scss/modules/_colors.scss +7 -0
  365. data/core/admin/assets/scss/modules/_extensions.scss +24 -0
  366. data/core/admin/assets/scss/modules/_mixins.scss +18 -0
  367. data/core/admin/assets/scss/modules/_susy.scss +7 -0
  368. data/core/admin/assets/scss/partials/_base.scss +33 -0
  369. data/core/admin/assets/scss/partials/_buttons.scss +9 -0
  370. data/core/admin/assets/scss/partials/_links.scss +15 -0
  371. data/core/admin/assets/scss/partials/_main.scss +14 -0
  372. data/core/admin/assets/scss/partials/_typography.scss +15 -0
  373. data/core/admin/assets/scss/sections/_editor.scss +274 -0
  374. data/core/admin/assets/scss/sections/_login.scss +34 -0
  375. data/core/admin/assets/scss/vendor/_normalize.scss +406 -0
  376. data/core/admin/layouts/admin.haml +67 -0
  377. data/core/admin/partials/_head.haml +23 -0
  378. data/core/admin/partials/_tree.haml +3 -0
  379. data/core/admin/partials/_treeitem.haml +4 -0
  380. data/core/admin/settings.yml +4 -0
  381. data/core/app.rb +21 -0
  382. data/core/classes/cached.rb +35 -0
  383. data/core/classes/checker.rb +37 -0
  384. data/core/classes/commit.rb +62 -0
  385. data/core/classes/error.rb +9 -0
  386. data/core/classes/page.rb +165 -0
  387. data/core/classes/partial.rb +13 -0
  388. data/core/classes/path.rb +30 -0
  389. data/core/classes/persona.rb +55 -0
  390. data/core/classes/sassfile.rb +95 -0
  391. data/core/classes/settings.rb +19 -0
  392. data/core/classes/theme.rb +55 -0
  393. data/core/helpers/base.rb +46 -0
  394. data/core/helpers/render.rb +26 -0
  395. data/core/helpers/truncate.rb +17 -0
  396. data/core/routes/admin.rb +139 -0
  397. data/core/routes/assetcompiler.rb +23 -0
  398. data/core/routes/base.rb +22 -0
  399. data/core/routes/bootstrap.rb +35 -0
  400. data/core/routes/cache.rb +52 -0
  401. data/core/routes/deploy.rb +16 -0
  402. data/core/routes/git.rb +23 -0
  403. data/poly.gemspec +29 -0
  404. data/sample/.cabi-data +0 -0
  405. data/sample/data/home/index.haml +22 -0
  406. data/sample/data/other/nav/links.yml +4 -0
  407. data/sample/data/page/about/index.haml +12 -0
  408. data/sample/data/page/contact-us/index.haml +11 -0
  409. data/sample/data/page/dynamic-page/index.haml +14 -0
  410. data/sample/data/page/error/index.haml +8 -0
  411. data/sample/data/page/html/index.html +4 -0
  412. data/sample/data/page/no-front-matter/index.haml +7 -0
  413. data/sample/data/page/not-found/index.haml +8 -0
  414. data/sample/data/page/static-page/index.haml +14 -0
  415. data/sample/data/post/about-life/index.md +12 -0
  416. data/sample/data/post/sample-post/index.haml +12 -0
  417. data/sample/data/post/technology/index.haml +12 -0
  418. data/sample/data/post/the-40-hour-workweek/index.haml +9 -0
  419. data/sample/data/post/things-are-changing/index.haml +9 -0
  420. data/sample/data/post/yet-another-post/index.haml +11 -0
  421. data/sample/plugins/my-plugin/plugin.rb +15 -0
  422. data/sample/settings.yml +21 -0
  423. data/sample/themes/base/assets/css/readme.txt +6 -0
  424. data/sample/themes/base/assets/images/poly.svg +36 -0
  425. data/sample/themes/base/assets/js/app/views/main.js +13 -0
  426. data/sample/themes/base/assets/js/main.js +31 -0
  427. data/sample/themes/base/assets/js/vendor/backbone.js +4 -0
  428. data/sample/themes/base/assets/js/vendor/jquery.js +6 -0
  429. data/sample/themes/base/assets/js/vendor/lodash.underscore.js +38 -0
  430. data/sample/themes/base/assets/js/vendor/require.js +36 -0
  431. data/sample/themes/base/assets/scss/bootstrap.scss +20 -0
  432. data/sample/themes/base/assets/scss/modules/_all.scss +7 -0
  433. data/sample/themes/base/assets/scss/modules/_colors.scss +7 -0
  434. data/sample/themes/base/assets/scss/modules/_extensions.scss +13 -0
  435. data/sample/themes/base/assets/scss/modules/_mixins.scss +11 -0
  436. data/sample/themes/base/assets/scss/modules/_susy.scss +7 -0
  437. data/sample/themes/base/assets/scss/partials/_base.scss +33 -0
  438. data/sample/themes/base/assets/scss/partials/_buttons.scss +0 -0
  439. data/sample/themes/base/assets/scss/partials/_links.scss +15 -0
  440. data/sample/themes/base/assets/scss/partials/_main.scss +39 -0
  441. data/sample/themes/base/assets/scss/partials/_typography.scss +15 -0
  442. data/sample/themes/base/assets/scss/sections/_footer.scss +22 -0
  443. data/sample/themes/base/assets/scss/sections/_header.scss +23 -0
  444. data/sample/themes/base/assets/scss/sections/_home.scss +13 -0
  445. data/sample/themes/base/assets/scss/sections/_pages.scss +11 -0
  446. data/sample/themes/base/assets/scss/vendor/_normalize.scss +406 -0
  447. data/sample/themes/base/layouts/default.haml +14 -0
  448. data/sample/themes/base/layouts/post.haml +15 -0
  449. data/sample/themes/base/partials/_footer.haml +9 -0
  450. data/sample/themes/base/partials/_ga.haml +10 -0
  451. data/sample/themes/base/partials/_head.haml +20 -0
  452. data/sample/themes/base/partials/_header.haml +5 -0
  453. data/sample/themes/base/settings.yml +4 -0
  454. data/test/data/post-receive-hook.json +146 -0
  455. data/test/test_basics.rb +50 -0
  456. data/test/test_data.rb +14 -0
  457. data/test/test_deploy.rb +23 -0
  458. data/test/test_includes.rb +43 -0
  459. data/test/test_page.rb +47 -0
  460. metadata +1049 -0
@@ -0,0 +1,226 @@
1
+ @font-face {
2
+ font-family: 'Source Sans Pro';
3
+ font-style: normal;
4
+ font-weight: 400;
5
+ src: local('Source Sans Pro'), local('SourceSansPro-Regular'), url(http://themes.googleusercontent.com/static/fonts/sourcesanspro/v5/ODelI1aHBYDBqgeIAH2zlBM0YzuT7MdOe03otPbuUS0.woff) format('woff');
6
+ }
7
+
8
+ body, html { margin: 0; padding: 0; height: 100%; }
9
+ section, article { display: block; padding: 0; }
10
+
11
+ body {
12
+ background: #f8f8f8;
13
+ font-family: 'Source Sans Pro', Helvetica, Arial, sans-serif;
14
+ line-height: 1.5;
15
+ }
16
+
17
+ p { margin-top: 0; }
18
+
19
+ h2, h3 {
20
+ font-weight: normal;
21
+ text-decoration: underline;
22
+ margin-bottom: .7em;
23
+ }
24
+ h2 { font-size: 120%; }
25
+ h3 { font-size: 110%; }
26
+ article > h2:first-child, section:first-child > h2 { margin-top: 0; }
27
+
28
+ a, a:visited, a:link, .quasilink {
29
+ color: #A21313;
30
+ text-decoration: none;
31
+ }
32
+
33
+ em {
34
+ padding-right: 2px;
35
+ }
36
+
37
+ .quasilink {
38
+ cursor: pointer;
39
+ }
40
+
41
+ article {
42
+ max-width: 700px;
43
+ margin: 0 auto;
44
+ border-left: 2px solid #E30808;
45
+ border-right: 1px solid #ddd;
46
+ padding: 30px 50px 100px 50px;
47
+ background: white;
48
+ z-index: 2;
49
+ position: relative;
50
+ min-height: 100%;
51
+ box-sizing: border-box;
52
+ -moz-box-sizing: border-box;
53
+ }
54
+
55
+ #nav {
56
+ position: fixed;
57
+ top: 30px;
58
+ right: 50%;
59
+ padding-right: 350px;
60
+ text-align: right;
61
+ z-index: 1;
62
+ }
63
+
64
+ @media screen and (max-width: 1000px) {
65
+ article {
66
+ margin: 0 0 0 160px;
67
+ }
68
+ #nav {
69
+ left: 0; right: none;
70
+ width: 160px;
71
+ }
72
+ }
73
+
74
+ #nav ul {
75
+ display: block;
76
+ margin: 0; padding: 0;
77
+ margin-bottom: 32px;
78
+ }
79
+
80
+ #nav li {
81
+ display: block;
82
+ margin-bottom: 4px;
83
+ }
84
+
85
+ #nav li ul {
86
+ font-size: 80%;
87
+ margin-bottom: 0;
88
+ display: none;
89
+ }
90
+
91
+ #nav li.active ul {
92
+ display: block;
93
+ }
94
+
95
+ #nav li li a {
96
+ padding-right: 20px;
97
+ }
98
+
99
+ #nav ul a {
100
+ color: black;
101
+ padding: 0 7px 1px 11px;
102
+ }
103
+
104
+ #nav ul a.active, #nav ul a:hover {
105
+ border-bottom: 1px solid #E30808;
106
+ color: #E30808;
107
+ }
108
+
109
+ #logo {
110
+ border: 0;
111
+ margin-right: 7px;
112
+ margin-bottom: 25px;
113
+ }
114
+
115
+ section {
116
+ border-top: 1px solid #E30808;
117
+ margin: 1.5em 0;
118
+ }
119
+
120
+ section.first {
121
+ border: none;
122
+ margin-top: 0;
123
+ }
124
+
125
+ #demo {
126
+ position: relative;
127
+ }
128
+
129
+ #demolist {
130
+ position: absolute;
131
+ right: 5px;
132
+ top: 5px;
133
+ z-index: 25;
134
+ }
135
+
136
+ #bankinfo {
137
+ text-align: left;
138
+ display: none;
139
+ padding: 0 .5em;
140
+ position: absolute;
141
+ border: 2px solid #aaa;
142
+ border-radius: 5px;
143
+ background: #eee;
144
+ top: 10px;
145
+ left: 30px;
146
+ }
147
+
148
+ #bankinfo_close {
149
+ position: absolute;
150
+ top: 0; right: 6px;
151
+ font-weight: bold;
152
+ cursor: pointer;
153
+ }
154
+
155
+ .bigbutton {
156
+ cursor: pointer;
157
+ text-align: center;
158
+ padding: 0 1em;
159
+ display: inline-block;
160
+ color: white;
161
+ position: relative;
162
+ line-height: 1.9;
163
+ color: white !important;
164
+ background: #A21313;
165
+ }
166
+
167
+ .bigbutton.right {
168
+ border-bottom-left-radius: 100px;
169
+ border-top-left-radius: 100px;
170
+ }
171
+
172
+ .bigbutton.left {
173
+ border-bottom-right-radius: 100px;
174
+ border-top-right-radius: 100px;
175
+ }
176
+
177
+ .bigbutton:hover {
178
+ background: #E30808;
179
+ }
180
+
181
+ th {
182
+ text-decoration: underline;
183
+ font-weight: normal;
184
+ text-align: left;
185
+ }
186
+
187
+ #features ul {
188
+ list-style: none;
189
+ margin: 0 0 1em;
190
+ padding: 0 0 0 1.2em;
191
+ }
192
+
193
+ #features li:before {
194
+ content: "-";
195
+ width: 1em;
196
+ display: inline-block;
197
+ padding: 0;
198
+ margin: 0;
199
+ margin-left: -1em;
200
+ }
201
+
202
+ .rel {
203
+ margin-bottom: 0;
204
+ }
205
+ .rel-note {
206
+ margin-top: 0;
207
+ color: #555;
208
+ }
209
+
210
+ pre {
211
+ padding-left: 15px;
212
+ border-left: 2px solid #ddd;
213
+ }
214
+
215
+ code {
216
+ padding: 0 2px;
217
+ }
218
+
219
+ strong {
220
+ text-decoration: underline;
221
+ font-weight: normal;
222
+ }
223
+
224
+ .field {
225
+ border: 1px solid #A21313;
226
+ }
@@ -0,0 +1,503 @@
1
+ <!doctype html>
2
+
3
+ <title>CodeMirror: Internals</title>
4
+ <meta charset="utf-8"/>
5
+ <link rel=stylesheet href="docs.css">
6
+ <style>dl dl {margin: 0;} .update {color: #d40 !important}</style>
7
+ <script src="activebookmark.js"></script>
8
+
9
+ <div id=nav>
10
+ <a href="http://codemirror.net"><img id=logo src="logo.png"></a>
11
+
12
+ <ul>
13
+ <li><a href="../index.html">Home</a>
14
+ <li><a href="manual.html">Manual</a>
15
+ <li><a href="https://github.com/marijnh/codemirror">Code</a>
16
+ </ul>
17
+ <ul>
18
+ <li><a href="#top">Introduction</a></li>
19
+ <li><a href="#approach">General Approach</a></li>
20
+ <li><a href="#input">Input</a></li>
21
+ <li><a href="#selection">Selection</a></li>
22
+ <li><a href="#update">Intelligent Updating</a></li>
23
+ <li><a href="#parse">Parsing</a></li>
24
+ <li><a href="#summary">What Gives?</a></li>
25
+ <li><a href="#btree">Content Representation</a></li>
26
+ <li><a href="#keymap">Key Maps</a></li>
27
+ </ul>
28
+ </div>
29
+
30
+ <article>
31
+
32
+ <h2 id=top>(Re-) Implementing A Syntax-Highlighting Editor in JavaScript</h2>
33
+
34
+ <p style="font-size: 85%" id="intro">
35
+ <strong>Topic:</strong> JavaScript, code editor implementation<br>
36
+ <strong>Author:</strong> Marijn Haverbeke<br>
37
+ <strong>Date:</strong> March 2nd 2011 (updated November 13th 2011)
38
+ </p>
39
+
40
+ <p style="padding: 0 3em 0 2em"><strong>Caution</strong>: this text was written briefly after
41
+ version 2 was initially written. It no longer (even including the
42
+ update at the bottom) fully represents the current implementation. I'm
43
+ leaving it here as a historic document. For more up-to-date
44
+ information, look at the entries
45
+ tagged <a href="http://marijnhaverbeke.nl/blog/#cm-internals">cm-internals</a>
46
+ on my blog.</p>
47
+
48
+ <p>This is a followup to
49
+ my <a href="http://codemirror.net/story.html">Brutal Odyssey to the
50
+ Dark Side of the DOM Tree</a> story. That one describes the
51
+ mind-bending process of implementing (what would become) CodeMirror 1.
52
+ This one describes the internals of CodeMirror 2, a complete rewrite
53
+ and rethink of the old code base. I wanted to give this piece another
54
+ Hunter Thompson copycat subtitle, but somehow that would be out of
55
+ place—the process this time around was one of straightforward
56
+ engineering, requiring no serious mind-bending whatsoever.</p>
57
+
58
+ <p>So, what is wrong with CodeMirror 1? I'd estimate, by mailing list
59
+ activity and general search-engine presence, that it has been
60
+ integrated into about a thousand systems by now. The most prominent
61
+ one, since a few weeks,
62
+ being <a href="http://googlecode.blogspot.com/2011/01/make-quick-fixes-quicker-on-google.html">Google
63
+ code's project hosting</a>. It works, and it's being used widely.</p>
64
+
65
+ <p>Still, I did not start replacing it because I was bored. CodeMirror
66
+ 1 was heavily reliant on <code>designMode</code>
67
+ or <code>contentEditable</code> (depending on the browser). Neither of
68
+ these are well specified (HTML5 tries
69
+ to <a href="http://www.w3.org/TR/html5/editing.html#contenteditable">specify</a>
70
+ their basics), and, more importantly, they tend to be one of the more
71
+ obscure and buggy areas of browser functionality—CodeMirror, by using
72
+ this functionality in a non-typical way, was constantly running up
73
+ against browser bugs. WebKit wouldn't show an empty line at the end of
74
+ the document, and in some releases would suddenly get unbearably slow.
75
+ Firefox would show the cursor in the wrong place. Internet Explorer
76
+ would insist on linkifying everything that looked like a URL or email
77
+ address, a behaviour that can't be turned off. Some bugs I managed to
78
+ work around (which was often a frustrating, painful process), others,
79
+ such as the Firefox cursor placement, I gave up on, and had to tell
80
+ user after user that they were known problems, but not something I
81
+ could help.</p>
82
+
83
+ <p>Also, there is the fact that <code>designMode</code> (which seemed
84
+ to be less buggy than <code>contentEditable</code> in Webkit and
85
+ Firefox, and was thus used by CodeMirror 1 in those browsers) requires
86
+ a frame. Frames are another tricky area. It takes some effort to
87
+ prevent getting tripped up by domain restrictions, they don't
88
+ initialize synchronously, behave strangely in response to the back
89
+ button, and, on several browsers, can't be moved around the DOM
90
+ without having them re-initialize. They did provide a very nice way to
91
+ namespace the library, though—CodeMirror 1 could freely pollute the
92
+ namespace inside the frame.</p>
93
+
94
+ <p>Finally, working with an editable document means working with
95
+ selection in arbitrary DOM structures. Internet Explorer (8 and
96
+ before) has an utterly different (and awkward) selection API than all
97
+ of the other browsers, and even among the different implementations of
98
+ <code>document.selection</code>, details about how exactly a selection
99
+ is represented vary quite a bit. Add to that the fact that Opera's
100
+ selection support tended to be very buggy until recently, and you can
101
+ imagine why CodeMirror 1 contains 700 lines of selection-handling
102
+ code.</p>
103
+
104
+ <p>And that brings us to the main issue with the CodeMirror 1
105
+ code base: The proportion of browser-bug-workarounds to real
106
+ application code was getting dangerously high. By building on top of a
107
+ few dodgy features, I put the system in a vulnerable position—any
108
+ incompatibility and bugginess in these features, I had to paper over
109
+ with my own code. Not only did I have to do some serious stunt-work to
110
+ get it to work on older browsers (as detailed in the
111
+ previous <a href="http://codemirror.net/story.html">story</a>), things
112
+ also kept breaking in newly released versions, requiring me to come up
113
+ with <em>new</em> scary hacks in order to keep up. This was starting
114
+ to lose its appeal.</p>
115
+
116
+ <section id=approach>
117
+ <h2>General Approach</h2>
118
+
119
+ <p>What CodeMirror 2 does is try to sidestep most of the hairy hacks
120
+ that came up in version 1. I owe a lot to the
121
+ <a href="http://ace.ajax.org">ACE</a> editor for inspiration on how to
122
+ approach this.</p>
123
+
124
+ <p>I absolutely did not want to be completely reliant on key events to
125
+ generate my input. Every JavaScript programmer knows that key event
126
+ information is horrible and incomplete. Some people (most awesomely
127
+ Mihai Bazon with <a href="http://ymacs.org">Ymacs</a>) have been able
128
+ to build more or less functioning editors by directly reading key
129
+ events, but it takes a lot of work (the kind of never-ending, fragile
130
+ work I described earlier), and will never be able to properly support
131
+ things like multi-keystoke international character
132
+ input. <a href="#keymap" class="update">[see below for caveat]</a></p>
133
+
134
+ <p>So what I do is focus a hidden textarea, and let the browser
135
+ believe that the user is typing into that. What we show to the user is
136
+ a DOM structure we built to represent his document. If this is updated
137
+ quickly enough, and shows some kind of believable cursor, it feels
138
+ like a real text-input control.</p>
139
+
140
+ <p>Another big win is that this DOM representation does not have to
141
+ span the whole document. Some CodeMirror 1 users insisted that they
142
+ needed to put a 30 thousand line XML document into CodeMirror. Putting
143
+ all that into the DOM takes a while, especially since, for some
144
+ reason, an editable DOM tree is slower than a normal one on most
145
+ browsers. If we have full control over what we show, we must only
146
+ ensure that the visible part of the document has been added, and can
147
+ do the rest only when needed. (Fortunately, the <code>onscroll</code>
148
+ event works almost the same on all browsers, and lends itself well to
149
+ displaying things only as they are scrolled into view.)</p>
150
+ </section>
151
+ <section id="input">
152
+ <h2>Input</h2>
153
+
154
+ <p>ACE uses its hidden textarea only as a text input shim, and does
155
+ all cursor movement and things like text deletion itself by directly
156
+ handling key events. CodeMirror's way is to let the browser do its
157
+ thing as much as possible, and not, for example, define its own set of
158
+ key bindings. One way to do this would have been to have the whole
159
+ document inside the hidden textarea, and after each key event update
160
+ the display DOM to reflect what's in that textarea.</p>
161
+
162
+ <p>That'd be simple, but it is not realistic. For even medium-sized
163
+ document the editor would be constantly munging huge strings, and get
164
+ terribly slow. What CodeMirror 2 does is put the current selection,
165
+ along with an extra line on the top and on the bottom, into the
166
+ textarea.</p>
167
+
168
+ <p>This means that the arrow keys (and their ctrl-variations), home,
169
+ end, etcetera, do not have to be handled specially. We just read the
170
+ cursor position in the textarea, and update our cursor to match it.
171
+ Also, copy and paste work pretty much for free, and people get their
172
+ native key bindings, without any special work on my part. For example,
173
+ I have emacs key bindings configured for Chrome and Firefox. There is
174
+ no way for a script to detect this. <a class="update"
175
+ href="#keymap">[no longer the case]</a></p>
176
+
177
+ <p>Of course, since only a small part of the document sits in the
178
+ textarea, keys like page up and ctrl-end won't do the right thing.
179
+ CodeMirror is catching those events and handling them itself.</p>
180
+ </section>
181
+ <section id="selection">
182
+ <h2>Selection</h2>
183
+
184
+ <p>Getting and setting the selection range of a textarea in modern
185
+ browsers is trivial—you just use the <code>selectionStart</code>
186
+ and <code>selectionEnd</code> properties. On IE you have to do some
187
+ insane stuff with temporary ranges and compensating for the fact that
188
+ moving the selection by a 'character' will treat \r\n as a single
189
+ character, but even there it is possible to build functions that
190
+ reliably set and get the selection range.</p>
191
+
192
+ <p>But consider this typical case: When I'm somewhere in my document,
193
+ press shift, and press the up arrow, something gets selected. Then, if
194
+ I, still holding shift, press the up arrow again, the top of my
195
+ selection is adjusted. The selection remembers where its <em>head</em>
196
+ and its <em>anchor</em> are, and moves the head when we shift-move.
197
+ This is a generally accepted property of selections, and done right by
198
+ every editing component built in the past twenty years.</p>
199
+
200
+ <p>But not something that the browser selection APIs expose.</p>
201
+
202
+ <p>Great. So when someone creates an 'upside-down' selection, the next
203
+ time CodeMirror has to update the textarea, it'll re-create the
204
+ selection as an 'upside-up' selection, with the anchor at the top, and
205
+ the next cursor motion will behave in an unexpected way—our second
206
+ up-arrow press in the example above will not do anything, since it is
207
+ interpreted in exactly the same way as the first.</p>
208
+
209
+ <p>No problem. We'll just, ehm, detect that the selection is
210
+ upside-down (you can tell by the way it was created), and then, when
211
+ an upside-down selection is present, and a cursor-moving key is
212
+ pressed in combination with shift, we quickly collapse the selection
213
+ in the textarea to its start, allow the key to take effect, and then
214
+ combine its new head with its old anchor to get the <em>real</em>
215
+ selection.</p>
216
+
217
+ <p>In short, scary hacks could not be avoided entirely in CodeMirror
218
+ 2.</p>
219
+
220
+ <p>And, the observant reader might ask, how do you even know that a
221
+ key combo is a cursor-moving combo, if you claim you support any
222
+ native key bindings? Well, we don't, but we can learn. The editor
223
+ keeps a set known cursor-movement combos (initialized to the
224
+ predictable defaults), and updates this set when it observes that
225
+ pressing a certain key had (only) the effect of moving the cursor.
226
+ This, of course, doesn't work if the first time the key is used was
227
+ for extending an inverted selection, but it works most of the
228
+ time.</p>
229
+ </section>
230
+ <section id="update">
231
+ <h2>Intelligent Updating</h2>
232
+
233
+ <p>One thing that always comes up when you have a complicated internal
234
+ state that's reflected in some user-visible external representation
235
+ (in this case, the displayed code and the textarea's content) is
236
+ keeping the two in sync. The naive way is to just update the display
237
+ every time you change your state, but this is not only error prone
238
+ (you'll forget), it also easily leads to duplicate work on big,
239
+ composite operations. Then you start passing around flags indicating
240
+ whether the display should be updated in an attempt to be efficient
241
+ again and, well, at that point you might as well give up completely.</p>
242
+
243
+ <p>I did go down that road, but then switched to a much simpler model:
244
+ simply keep track of all the things that have been changed during an
245
+ action, and then, only at the end, use this information to update the
246
+ user-visible display.</p>
247
+
248
+ <p>CodeMirror uses a concept of <em>operations</em>, which start by
249
+ calling a specific set-up function that clears the state and end by
250
+ calling another function that reads this state and does the required
251
+ updating. Most event handlers, and all the user-visible methods that
252
+ change state are wrapped like this. There's a method
253
+ called <code>operation</code> that accepts a function, and returns
254
+ another function that wraps the given function as an operation.</p>
255
+
256
+ <p>It's trivial to extend this (as CodeMirror does) to detect nesting,
257
+ and, when an operation is started inside an operation, simply
258
+ increment the nesting count, and only do the updating when this count
259
+ reaches zero again.</p>
260
+
261
+ <p>If we have a set of changed ranges and know the currently shown
262
+ range, we can (with some awkward code to deal with the fact that
263
+ changes can add and remove lines, so we're dealing with a changing
264
+ coordinate system) construct a map of the ranges that were left
265
+ intact. We can then compare this map with the part of the document
266
+ that's currently visible (based on scroll offset and editor height) to
267
+ determine whether something needs to be updated.</p>
268
+
269
+ <p>CodeMirror uses two update algorithms—a full refresh, where it just
270
+ discards the whole part of the DOM that contains the edited text and
271
+ rebuilds it, and a patch algorithm, where it uses the information
272
+ about changed and intact ranges to update only the out-of-date parts
273
+ of the DOM. When more than 30 percent (which is the current heuristic,
274
+ might change) of the lines need to be updated, the full refresh is
275
+ chosen (since it's faster to do than painstakingly finding and
276
+ updating all the changed lines), in the other case it does the
277
+ patching (so that, if you scroll a line or select another character,
278
+ the whole screen doesn't have to be
279
+ re-rendered). <span class="update">[the full-refresh
280
+ algorithm was dropped, it wasn't really faster than the patching
281
+ one]</span></p>
282
+
283
+ <p>All updating uses <code>innerHTML</code> rather than direct DOM
284
+ manipulation, since that still seems to be by far the fastest way to
285
+ build documents. There's a per-line function that combines the
286
+ highlighting, <a href="manual.html#markText">marking</a>, and
287
+ selection info for that line into a snippet of HTML. The patch updater
288
+ uses this to reset individual lines, the refresh updater builds an
289
+ HTML chunk for the whole visible document at once, and then uses a
290
+ single <code>innerHTML</code> update to do the refresh.</p>
291
+ </section>
292
+ <section id="parse">
293
+ <h2>Parsers can be Simple</h2>
294
+
295
+ <p>When I wrote CodeMirror 1, I
296
+ thought <a href="http://codemirror.net/story.html#parser">interruptable
297
+ parsers</a> were a hugely scary and complicated thing, and I used a
298
+ bunch of heavyweight abstractions to keep this supposed complexity
299
+ under control: parsers
300
+ were <a href="http://bob.pythonmac.org/archives/2005/07/06/iteration-in-javascript/">iterators</a>
301
+ that consumed input from another iterator, and used funny
302
+ closure-resetting tricks to copy and resume themselves.</p>
303
+
304
+ <p>This made for a rather nice system, in that parsers formed strictly
305
+ separate modules, and could be composed in predictable ways.
306
+ Unfortunately, it was quite slow (stacking three or four iterators on
307
+ top of each other), and extremely intimidating to people not used to a
308
+ functional programming style.</p>
309
+
310
+ <p>With a few small changes, however, we can keep all those
311
+ advantages, but simplify the API and make the whole thing less
312
+ indirect and inefficient. CodeMirror
313
+ 2's <a href="manual.html#modeapi">mode API</a> uses explicit state
314
+ objects, and makes the parser/tokenizer a function that simply takes a
315
+ state and a character stream abstraction, advances the stream one
316
+ token, and returns the way the token should be styled. This state may
317
+ be copied, optionally in a mode-defined way, in order to be able to
318
+ continue a parse at a given point. Even someone who's never touched a
319
+ lambda in his life can understand this approach. Additionally, far
320
+ fewer objects are allocated in the course of parsing now.</p>
321
+
322
+ <p>The biggest speedup comes from the fact that the parsing no longer
323
+ has to touch the DOM though. In CodeMirror 1, on an older browser, you
324
+ could <em>see</em> the parser work its way through the document,
325
+ managing some twenty lines in each 50-millisecond time slice it got. It
326
+ was reading its input from the DOM, and updating the DOM as it went
327
+ along, which any experienced JavaScript programmer will immediately
328
+ spot as a recipe for slowness. In CodeMirror 2, the parser usually
329
+ finishes the whole document in a single 100-millisecond time slice—it
330
+ manages some 1500 lines during that time on Chrome. All it has to do
331
+ is munge strings, so there is no real reason for it to be slow
332
+ anymore.</p>
333
+ </section>
334
+ <section id="summary">
335
+ <h2>What Gives?</h2>
336
+
337
+ <p>Given all this, what can you expect from CodeMirror 2?</p>
338
+
339
+ <ul>
340
+
341
+ <li><strong>Small.</strong> the base library is
342
+ some <span class="update">45k</span> when minified
343
+ now, <span class="update">17k</span> when gzipped. It's smaller than
344
+ its own logo.</li>
345
+
346
+ <li><strong>Lightweight.</strong> CodeMirror 2 initializes very
347
+ quickly, and does almost no work when it is not focused. This means
348
+ you can treat it almost like a textarea, have multiple instances on a
349
+ page without trouble.</li>
350
+
351
+ <li><strong>Huge document support.</strong> Since highlighting is
352
+ really fast, and no DOM structure is being built for non-visible
353
+ content, you don't have to worry about locking up your browser when a
354
+ user enters a megabyte-sized document.</li>
355
+
356
+ <li><strong>Extended API.</strong> Some things kept coming up in the
357
+ mailing list, such as marking pieces of text or lines, which were
358
+ extremely hard to do with CodeMirror 1. The new version has proper
359
+ support for these built in.</li>
360
+
361
+ <li><strong>Tab support.</strong> Tabs inside editable documents were,
362
+ for some reason, a no-go. At least six different people announced they
363
+ were going to add tab support to CodeMirror 1, none survived (I mean,
364
+ none delivered a working version). CodeMirror 2 no longer removes tabs
365
+ from your document.</li>
366
+
367
+ <li><strong>Sane styling.</strong> <code>iframe</code> nodes aren't
368
+ really known for respecting document flow. Now that an editor instance
369
+ is a plain <code>div</code> element, it is much easier to size it to
370
+ fit the surrounding elements. You don't even have to make it scroll if
371
+ you do not <a href="../demo/resize.html">want to</a>.</li>
372
+
373
+ </ul>
374
+
375
+ <p>On the downside, a CodeMirror 2 instance is <em>not</em> a native
376
+ editable component. Though it does its best to emulate such a
377
+ component as much as possible, there is functionality that browsers
378
+ just do not allow us to hook into. Doing select-all from the context
379
+ menu, for example, is not currently detected by CodeMirror.</p>
380
+
381
+ <p id="changes" style="margin-top: 2em;"><span style="font-weight:
382
+ bold">[Updates from November 13th 2011]</span> Recently, I've made
383
+ some changes to the codebase that cause some of the text above to no
384
+ longer be current. I've left the text intact, but added markers at the
385
+ passages that are now inaccurate. The new situation is described
386
+ below.</p>
387
+ </section>
388
+ <section id="btree">
389
+ <h2>Content Representation</h2>
390
+
391
+ <p>The original implementation of CodeMirror 2 represented the
392
+ document as a flat array of line objects. This worked well—splicing
393
+ arrays will require the part of the array after the splice to be
394
+ moved, but this is basically just a simple <code>memmove</code> of a
395
+ bunch of pointers, so it is cheap even for huge documents.</p>
396
+
397
+ <p>However, I recently added line wrapping and code folding (line
398
+ collapsing, basically). Once lines start taking up a non-constant
399
+ amount of vertical space, looking up a line by vertical position
400
+ (which is needed when someone clicks the document, and to determine
401
+ the visible part of the document during scrolling) can only be done
402
+ with a linear scan through the whole array, summing up line heights as
403
+ you go. Seeing how I've been going out of my way to make big documents
404
+ fast, this is not acceptable.</p>
405
+
406
+ <p>The new representation is based on a B-tree. The leaves of the tree
407
+ contain arrays of line objects, with a fixed minimum and maximum size,
408
+ and the non-leaf nodes simply hold arrays of child nodes. Each node
409
+ stores both the amount of lines that live below them and the vertical
410
+ space taken up by these lines. This allows the tree to be indexed both
411
+ by line number and by vertical position, and all access has
412
+ logarithmic complexity in relation to the document size.</p>
413
+
414
+ <p>I gave line objects and tree nodes parent pointers, to the node
415
+ above them. When a line has to update its height, it can simply walk
416
+ these pointers to the top of the tree, adding or subtracting the
417
+ difference in height from each node it encounters. The parent pointers
418
+ also make it cheaper (in complexity terms, the difference is probably
419
+ tiny in normal-sized documents) to find the current line number when
420
+ given a line object. In the old approach, the whole document array had
421
+ to be searched. Now, we can just walk up the tree and count the sizes
422
+ of the nodes coming before us at each level.</p>
423
+
424
+ <p>I chose B-trees, not regular binary trees, mostly because they
425
+ allow for very fast bulk insertions and deletions. When there is a big
426
+ change to a document, it typically involves adding, deleting, or
427
+ replacing a chunk of subsequent lines. In a regular balanced tree, all
428
+ these inserts or deletes would have to be done separately, which could
429
+ be really expensive. In a B-tree, to insert a chunk, you just walk
430
+ down the tree once to find where it should go, insert them all in one
431
+ shot, and then break up the node if needed. This breaking up might
432
+ involve breaking up nodes further up, but only requires a single pass
433
+ back up the tree. For deletion, I'm somewhat lax in keeping things
434
+ balanced—I just collapse nodes into a leaf when their child count goes
435
+ below a given number. This means that there are some weird editing
436
+ patterns that may result in a seriously unbalanced tree, but even such
437
+ an unbalanced tree will perform well, unless you spend a day making
438
+ strangely repeating edits to a really big document.</p>
439
+ </section>
440
+ <section id="keymap">
441
+ <h2>Keymaps</h2>
442
+
443
+ <p><a href="#approach">Above</a>, I claimed that directly catching key
444
+ events for things like cursor movement is impractical because it
445
+ requires some browser-specific kludges. I then proceeded to explain
446
+ some awful <a href="#selection">hacks</a> that were needed to make it
447
+ possible for the selection changes to be detected through the
448
+ textarea. In fact, the second hack is about as bad as the first.</p>
449
+
450
+ <p>On top of that, in the presence of user-configurable tab sizes and
451
+ collapsed and wrapped lines, lining up cursor movement in the textarea
452
+ with what's visible on the screen becomes a nightmare. Thus, I've
453
+ decided to move to a model where the textarea's selection is no longer
454
+ depended on.</p>
455
+
456
+ <p>So I moved to a model where all cursor movement is handled by my
457
+ own code. This adds support for a goal column, proper interaction of
458
+ cursor movement with collapsed lines, and makes it possible for
459
+ vertical movement to move through wrapped lines properly, instead of
460
+ just treating them like non-wrapped lines.</p>
461
+
462
+ <p>The key event handlers now translate the key event into a string,
463
+ something like <code>Ctrl-Home</code> or <code>Shift-Cmd-R</code>, and
464
+ use that string to look up an action to perform. To make keybinding
465
+ customizable, this lookup goes through
466
+ a <a href="manual.html#option_keyMap">table</a>, using a scheme that
467
+ allows such tables to be chained together (for example, the default
468
+ Mac bindings fall through to a table named 'emacsy', which defines
469
+ basic Emacs-style bindings like <code>Ctrl-F</code>, and which is also
470
+ used by the custom Emacs bindings).</p>
471
+
472
+ <p>A new
473
+ option <a href="manual.html#option_extraKeys"><code>extraKeys</code></a>
474
+ allows ad-hoc keybindings to be defined in a much nicer way than what
475
+ was possible with the
476
+ old <a href="manual.html#option_onKeyEvent"><code>onKeyEvent</code></a>
477
+ callback. You simply provide an object mapping key identifiers to
478
+ functions, instead of painstakingly looking at raw key events.</p>
479
+
480
+ <p>Built-in commands map to strings, rather than functions, for
481
+ example <code>"goLineUp"</code> is the default action bound to the up
482
+ arrow key. This allows new keymaps to refer to them without
483
+ duplicating any code. New commands can be defined by assigning to
484
+ the <code>CodeMirror.commands</code> object, which maps such commands
485
+ to functions.</p>
486
+
487
+ <p>The hidden textarea now only holds the current selection, with no
488
+ extra characters around it. This has a nice advantage: polling for
489
+ input becomes much, much faster. If there's a big selection, this text
490
+ does not have to be read from the textarea every time—when we poll,
491
+ just noticing that something is still selected is enough to tell us
492
+ that no new text was typed.</p>
493
+
494
+ <p>The reason that cheap polling is important is that many browsers do
495
+ not fire useful events on IME (input method engine) input, which is
496
+ the thing where people inputting a language like Japanese or Chinese
497
+ use multiple keystrokes to create a character or sequence of
498
+ characters. Most modern browsers fire <code>input</code> when the
499
+ composing is finished, but many don't fire anything when the character
500
+ is updated <em>during</em> composition. So we poll, whenever the
501
+ editor is focused, to provide immediate updates of the display.</p>
502
+
503
+ </article>