poly-cms 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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>