- @iframe.onload = =>
- @idoc = @iframe.contentDocument
-
- ignore_key_codes =
- '18': true # alt
- '20': true # capslock
- '17': true # ctrl
- '144': true # numlock
- '16': true # shift
- '91': true # windows "start" key
- control_key_codes = # we react to these, but they aren't typing
- '37': KEY_LEFT
- '38': KEY_UP
- '39': KEY_RIGHT
- '40': KEY_DOWN
- '35': KEY_END
- '8': KEY_BACKSPACE
- '46': KEY_DELETE
- '13': KEY_ENTER
- '27': KEY_ESCAPE
- '36': KEY_HOME
- '45': KEY_INSERT
- '33': KEY_PAGE_UP
- '34': KEY_PAGE_DOWN
- '9': KEY_TAB
-
- @idoc.body.onkeyup = (e) =>
- return if e.ctrlKey
- return false if ignore_key_codes[e.keyCode]?
- #return false if control_key_codes[e.keyCode]?
- @idoc.body.onkeydown = (e) =>
- return if e.ctrlKey
- return false if ignore_key_codes[e.keyCode]?
- #return false if control_key_codes[e.keyCode]?
- switch e.keyCode
- when KEY_LEFT
- if @cursor?
- new_cursor = find_prev_cursor_position @tree, @cursor...
- if new_cursor?
- @move_cursor new_cursor
- else
- for c in @tree
- new_cursor = find_next_cursor_position @tree, c, -1
- if new_cursor?
- @move_cursor new_cursor
- break
- return false
- when KEY_UP
- return false
- when KEY_RIGHT
- if @cursor?
- new_cursor = find_next_cursor_position @tree, @cursor...
- if new_cursor?
- @move_cursor new_cursor
- else
- for c in @tree
- new_cursor = find_prev_cursor_position @tree, c, -1
- if new_cursor?
- @move_cursor new_cursor
- break
- return false
- when KEY_DOWN
- return false
- when KEY_END
- return false
- when KEY_BACKSPACE
- return false
- when KEY_DELETE
- return false
- when KEY_ENTER
- return false
- when KEY_ESCAPE
- return false
- when KEY_HOME
- return false
- when KEY_INSERT
- return false
- when KEY_PAGE_UP
- return false
- when KEY_PAGE_DOWN
- return false
- when KEY_TAB
- return false
- @idoc.body.onkeypress = (e) =>
- return if e.ctrlKey
- return false if ignore_key_codes[e.keyCode]?
- return false if control_key_codes[e.keyCode]? # handled in keydown
- char = e.charCode ? e.keyCode
- if char and @cursor?
- char = String.fromCharCode char
- if @cursor[1] is 0
- @cursor[0].text = char + @cursor[0].text
- else if @cursor[1] is @cursor[0].text.length - 1
- @cursor[0].text += char
- else
- @cursor[0].text =
- @cursor[0].text.substr(0, @cursor[1]) +
- char +
- @cursor[0].text.substr(@cursor[1])
- @cursor[0].el.nodeValue = @cursor[0].text
- @move_cursor [@cursor[0], @cursor[1] + 1]
+ @outer_iframe = domify document, iframe: {}
+ outer_iframe_style = 'border: none !important; margin: 0 !important; padding: 0 !important; height: 100% !important; width: 100% !important;'
+ if @options.editor_id?
+ @outer_iframe.setAttribute 'id', @options.editor_id
+ @outer_iframe.onload = =>
+ @outer_idoc = @outer_iframe.contentDocument
+ icss = domify @outer_idoc, style: children: [
+ domify @outer_idoc, text: css
+ ]
+ @outer_idoc.head.appendChild icss
+ @iframe = domify @outer_idoc, iframe: sandbox: 'allow-same-origin allow-scripts'
+ @iframe.onload = =>
+ @init_1()
+ timeout 200, => # firefox never fires this onload
+ @init_1() unless @init_1_called
+ @outer_idoc.body.appendChild(
+ domify @outer_idoc, div: id: 'wrap1', children: [
+ domify @outer_idoc, div: style: "position: absolute; top: 0; left: 1px; font-size: 10px", children: [ domify @outer_idoc, text: "Peach HTML5 Editor" ]
+ @wrap2 = domify @outer_idoc, div: id: 'wrap2', children: [
+ domify @outer_idoc, div: id: 'wrap3', children: [
+ @iframe
+ @overlay = domify @outer_idoc, div: id: 'overlay'
+ ]
+ ]
+ ]
+ )
+ outer_wrap = domify document, div: class: 'peach_html5_editor'
+ @in_el.parentNode.appendChild outer_wrap
+ outer_bounds = get_el_bounds outer_wrap
+ if outer_bounds.w < 300
+ outer_bounds.w = 300
+ if outer_bounds.h < 300
+ outer_bounds.h = 300
+ outer_iframe_style += "width: #{outer_bounds.w}px; height: #{outer_bounds.h}px;"
+ @outer_iframe.setAttribute 'style', outer_iframe_style
+ css = @generate_outer_css w: outer_bounds.w, h: outer_bounds.h
+ outer_wrap.appendChild @outer_iframe
+ init_1: -> # @iframe has loaded (but not it's css)
+ @idoc = @iframe.contentDocument
+ @init_1_called = true
+ # chromium doesn't resolve relative urls as though they were at the same domain
+ # so add a <base> tag
+ @idoc.head.appendChild domify @idoc, base: href: this_url_sans_path()
+ # don't let @iframe have scrollbars
+ @idoc.head.appendChild domify @idoc, style: children: [domify @idoc, text: "body { overflow: hidden; }"]
+ # load css file
+ if @options.css_file
+ istyle = domify @idoc, link: rel: 'stylesheet', href: @options.css_file
+ istyle.onload = =>
+ @init_2()
+ @idoc.head.appendChild istyle
+ else
+ @init_2()
+ init_2: -> # @iframe and it's css file(s) are ready
+ @overlay.onclick = (e) =>
+ @have_focus()
+ return event_return e, @onclick e
+ @overlay.ondoubleclick = (e) =>
+ @have_focus()
+ return event_return e, @ondoubleclick e
+ @outer_idoc.body.onkeyup = (e) =>
+ @have_focus()
+ return event_return e, @onkeyup e
+ @outer_idoc.body.onkeydown = (e) =>
+ @have_focus()
+ return event_return e, @onkeydown e
+ @outer_idoc.body.onkeypress = (e) =>
+ @have_focus()
+ return event_return e, @onkeypress e
+ @load_html @in_el.value
+ if @options.on_init?
+ @options.on_init()
+ generate_outer_css: (args) ->
+ w = args.w ? 300
+ h = args.h ? 300
+ inner_padding = args.inner_padding ? overlay_padding
+ frame_width = args.frame_width ? inner_padding
+ occupy = (left, top = left, right = left, bottom = top) ->
+ w -= left + right
+ h -= top + bottom
+ return Math.max(left, top, right, bottom)
+ ret = ''
+ ret += 'body {'
+ ret += 'margin: 0;'
+ ret += 'padding: 0;'
+ ret += 'color: black;'
+ ret += 'background: white;'
+ ret += '}'
+ ret += '#wrap1 {'
+ ret += "border: #{occupy 1}px solid black;"
+ ret += "padding: #{occupy frame_width}px;"
+ ret += '}'
+ ret += '#wrap2 {'
+ ret += "border: #{occupy 1}px solid black;"
+ @wrap2_height = h # including padding because padding scrolls
+ ret += "padding: #{occupy inner_padding}px;"
+ ret += "padding-right: #{inner_padding + occupy 0, 0, 15, 0}px;" # for scroll bar
+ ret += "width: #{w}px;"
+ ret += "height: #{h}px;"
+ ret += 'overflow-x: hidden;'
+ ret += 'overflow-y: scroll;'
+ ret += '}'
+ ret += '#wrap3 {'
+ ret += 'position: relative;'
+ ret += "width: #{w}px;"
+ ret += "min-height: #{h}px;"
+ ret += '}'
+ ret += 'iframe {'
+ ret += 'box-sizing: border-box;'
+ ret += 'margin: 0;'
+ ret += 'border: none;'
+ ret += 'padding: 0;'
+ ret += "width: #{w}px;"
+ #ret += "height: #{h}px;" # height auto-set when content set/changed
+ ret += '-ms-user-select: none;'
+ ret += '-webkit-user-select: none;'
+ ret += '-moz-user-select: none;'
+ ret += 'user-select: none;'
+ ret += '}'
+ ret += '#overlay {'
+ ret += 'position: absolute;'
+ ret += "left: -#{inner_padding}px;"
+ ret += "top: -#{inner_padding}px;"
+ ret += "right: -#{inner_padding}px;"
+ ret += "bottom: -#{inner_padding}px;"
+ ret += 'overflow: hidden;'
+ ret += '}'
+ ret += '.lightbox {'
+ ret += 'position: absolute;'
+ ret += 'background: rgba(100,100,100,0.2);'
+ ret += '}'
+ ret += '#cursor {'
+ ret += 'position: absolute;'
+ ret += 'width: 2px;'
+ ret += 'background: linear-gradient(0deg, rgba(0,0,0,1), rgba(255,255,255,1), rgba(0,0,0,1), rgba(255,255,255,1), rgba(0,0,0,1), rgba(255,255,255,1), rgba(0,0,0,1), rgba(255,255,255,1), rgba(0,0,0,1));'
+ ret += 'background-size: 200% 200%;'
+ ret += '-webkit-animation: blink 1s linear normal infinite;'
+ ret += 'animation: blink 1s linear normal infinite;'
+ ret += '}'
+ ret += '@-webkit-keyframes blink {'
+ ret += '0%{background-position:0% 0%}'
+ ret += '100%{background-position:0% -100%}'
+ ret += '}'
+ ret += '@keyframes blink { '
+ ret += '0%{background-position:0% 0%}'
+ ret += '100%{background-position:0% -100%}'
+ ret += '}'
+ ret += '.ann_box {'
+ ret += 'z-index: 5;'
+ ret += 'position: absolute;'
+ ret += 'border: 1px solid rgba(0,0,0,0.1);'
+ ret += 'outline: 1px solid rgba(255,255,255,0.1);' # in case there's a black background
+ ret += '}'
+ ret += '.ann_tag {'
+ ret += 'z-index: 10;'
+ ret += 'position: absolute;'
+ ret += 'font-size: 8px;'
+ ret += 'white-space: pre;'
+ ret += 'background: rgba(255,255,255,0.4);'
+ ret += '-ms-user-select: none;'
+ ret += '-webkit-user-select: none;'
+ ret += '-moz-user-select: none;'
+ ret += 'user-select: none;'
+ ret += '}'
+ return ret
+ overlay_event_to_inner_xy: (e) ->
+ unless @wrap2_offset?
+ @wrap2_offset = get_el_bounds @wrap2
+ x = e.pageX - overlay_padding
+ y = e.pageY - overlay_padding + @wrap2.scrollTop
+ return x: x - @wrap2_offset.x, y: y - @wrap2_offset.y
+ onclick: (e) ->
+ xy = @overlay_event_to_inner_xy e
+ new_cursor = xy_to_cursor @tree, xy
+ if new_cursor?
+ @move_cursor new_cursor
+ else
+ @kill_cursor()
+ return false
+ ondoubleclick: (e) ->
+ return false
+ onkeyup: (e) ->
+ return if e.ctrlKey
+ return false if ignore_key_codes[e.keyCode]?
+ #return false if control_key_codes[e.keyCode]?
+ onkeydown: (e) ->
+ return if e.ctrlKey
+ return false if ignore_key_codes[e.keyCode]?
+ #return false if control_key_codes[e.keyCode]?
+ switch e.keyCode
+ when KEY_LEFT
+ if @cursor?
+ new_cursor = find_prev_cursor_position @tree, @cursor
+ else
+ new_cursor = first_cursor_position @tree
+ if new_cursor?
+ @move_cursor new_cursor