require 'hyperstack/hotloader/add_error_boundry'
require 'hyperstack/hotloader/stack-trace.js'
require 'hyperstack/hotloader/css_reloader'
require 'opal-parser' # gives me 'eval', for hot-loading code

require 'json'
require 'hyperstack/hotloader/short_cut.js'

# Opal client to support hot reloading
$eval_proc = proc do |file_name, s|
  $_hyperstack_reloader_file_name = file_name
  eval s
end

module Hyperstack

  class Hotloader
    def self.callbacks
      @@callbacks ||= Hash.new { |h, k| h[k] = Array.new }
    end

    STACKDIRS = ['hyperstack', 'corelib']

    def self.file_name(frame)
      return unless frame
      file_name = `#{frame}.fileName`
      match = %r{^(.+\/assets)\/}.match(file_name)
      return unless match
      file_name = file_name.gsub(match[0], '').split('/')
      n = (file_name.length - 1) / 2
      return unless file_name[0..n - 1] == file_name[n..2 * n - 1]
      return if STACKDIRS.include?(file_name[0])
      "#{match[0]}#{file_name[0..n - 1].join('/')}/#{file_name[-1]}"
    end

    def self.when_file_updates(&block)
      return if `Hyperstack.hotloader.ignore_callback_mapping`
      return callbacks[$_hyperstack_reloader_file_name] << block if $_hyperstack_reloader_file_name
      callback = lambda do |frames|
        # StackTrace.get maps the stack frames back to the source maps
        # we then associate any files in the call stack that match our application file pattern
        # with the block, which will remove any callbacks defined when those files were executing.
        # Note that StackTrace.getSync gives us the raw .js file names so it will not work
        frames.collect(&method(:file_name)).compact.uniq.each { |name| callbacks[name] << block if name }
      end
      error = lambda do |e|
        `console.error(#{"hyperstack hot loader could not find source file for callback: #{e}"})`
      end
      `StackTrace.get().then(#{callback}).catch(#{error})`
    end

    def self.remove(file_name)
      callbacks[file_name].each(&:call)
      callbacks[file_name] = []
    end

    def connect_to_websocket(port)
      host = `window.location.host`.sub(/:\d+/, '')
      host = '127.0.0.1' if host == ''
      protocol = `window.location.protocol` == 'https:' ? 'wss:' : 'ws:'
      ws_url = "#{host}:#{port}"
      puts "Hot-Reloader connecting to #{ws_url}"
      ws = `new WebSocket(#{protocol} + '//' + #{ws_url})`
      `#{ws}.onmessage = #{lambda { |e| reload(e) }}`
      `setInterval(function() { #{ws}.send('') }, #{@ping * 1000})` if @ping
    end

    def notify_error(reload_request)
      msg = "Hotloader #{reload_request[:filename]} RELOAD ERROR:\n\n#{$!}"
      puts msg
      alert msg if use_alert?
    end

    @@USE_ALERT = true
    def self.alerts_on!
      @@USE_ALERT = true
    end

    def self.alerts_off!
      @@USE_ALERT = false
    end

    def use_alert?
      @@USE_ALERT
    end

    def reload(e)
      reload_request = JSON.parse(`e.data`)
      if reload_request[:type] == "ruby"
        puts "Reloading #{reload_request[:filename]} (asset_path: #{reload_request[:asset_path]})"
        begin
          #Hyperstack::Context.reset! false
          file_name = reload_request[:asset_path].gsub(/.+hyperstack\//, '')  # this gsub we need sometimes????
          Hotloader.remove(file_name)
          $eval_proc.call file_name, reload_request[:source_code]
        rescue
          notify_error(reload_request)
        end
        if @reload_post_callback
          @reload_post_callback.call
        else
          puts "no reloading callback to call"
        end
      end
      if reload_request[:type] == "css"
        @css_reloader.reload(reload_request, `document`)
      end
    end

    # @param port [Integer] opal hot reloader port to connect to
    # @param reload_post_callback [Proc] optional callback to be called after re evaluating a file for example in react.rb files we want to do a React::Component.force_update!
    def initialize(port=25222, ping=nil, &reload_post_callback)
      @port = port
      @reload_post_callback  = reload_post_callback
      @css_reloader = CssReloader.new
      @ping = ping
    end
    # Opens a websocket connection that evaluates new files and runs the optional @reload_post_callback
    def listen
      connect_to_websocket(@port)
    end

    def self.listen(port=25222, ping=nil)
      ::Hyperstack::Internal::Component::TopLevelRailsComponent.include AddErrorBoundry
      @server = Hotloader.new(port, ping) do
        # TODO: check this out when Operations are integrated
        # if defined?(Hyperloop::Internal::Operation::ClientDrivers) &&
        #    Hyperloop::ClientDrivers.respond_to?(:initialize_client_drivers_on_boot)
        #   Hyperloop::ClientDrivers.initialize_client_drivers_on_boot
        # end
        Hyperstack::Component.force_update!
      end
      @server.listen
    end

  end
end