aboutsummaryrefslogtreecommitdiffstats
path: root/content_scripts/ui_component.coffee
blob: 12a024e4859792bb20e282c4b86ff0e09053c3c4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
class UIComponent
  iframeElement: null
  iframePort: null
  messageEventListeners: []
  showStyle: "display: block;"
  hideStyle: "display: none;"

  constructor: (iframeUrl, className, showStyle, hideStyle) ->
    @iframeElement = document.createElement "iframe"
    @iframeElement.className = className
    @iframeElement.seamless = "seamless"
    @iframeElement.src = chrome.runtime.getURL iframeUrl
    @iframeElement.addEventListener "load", => @openPort()
    document.documentElement.appendChild @iframeElement

    @setShowStyle showStyle if showStyle?
    @setHideStyle hideStyle if showStyle?
    @hide()

  # Open a port and pass it to the iframe via window.postMessage.
  openPort: ->
    messageChannel = new MessageChannel()
    @iframePort = messageChannel.port1
    @iframePort.onmessage = (event) => @handleMessage event

    # Get iframeMessageSecret so the iframe can determine that our message isn't the page impersonating us.
    chrome.storage.local.get "iframeMessageSecret", ({iframeMessageSecret: secret}) =>
      @iframeElement.contentWindow.postMessage secret, chrome.runtime.getURL(""), [messageChannel.port2]

  postMessage: (message) -> @iframePort.postMessage message

  # Execute each event listener on the current event until we get a non-null falsy return value.
  handleMessage: (event) ->
    for listener in @messageEventListeners
      retVal = listener.call this, event
      retVal ?= true
      return false unless retVal
    true

  addEventListener: (type, listener) ->
    if type == "message"
      @messageEventListeners.push listener
    undefined

  removeEventListener: (type, listener) ->
    if type == "message"
      @messageEventListeners = @messageEventListeners.filter (f) -> f != listener
    undefined

  setHideStyle: (@hideStyle) ->
    @hide() if @showing == false

  setShowStyle: (@showStyle) ->
    @show() if @showing == true

  setStyles: (@showStyle = @showStyle, @hideStyle = @hideStyle) ->
    if @showing
      @show()
    else
      @hide()

  show: (message) ->
    @postMessage message if message?
    @iframeElement.setAttribute "style", @showStyle
    @iframeElement.focus()
    @showing = true

  hide: ->
    @iframeElement.setAttribute "style", @hideStyle
    window.focus()
    @showing = false

handleHideMessage = (event) ->
  if event.data == "hide"
    @hide()
    false
  else
    true

root = exports ? window
root.UIComponent = UIComponent