orator 0.3.2 → 0.3.3
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.
- data/lib/orator/base.rb +1 -1
- data/lib/orator/version.rb +1 -1
- data/lib/orator.rb +1 -1
- data/vendor/assets/javascripts/orator/chat_logger.coffee +166 -0
- data/vendor/assets/javascripts/orator/command_handler.coffee +85 -0
- data/vendor/assets/javascripts/orator/event_handler.coffee +45 -0
- data/vendor/assets/javascripts/orator/header.coffee +3 -0
- data/vendor/assets/javascripts/orator/middle_ground.coffee +19 -0
- data/vendor/assets/javascripts/orator/strbuf.coffee +39 -0
- data/vendor/assets/javascripts/orator/stringify.jquery.js +59 -0
- data/vendor/assets/javascripts/orator/swfobject.js +4 -0
- data/vendor/assets/javascripts/orator/web_socket.js +391 -0
- data/vendor/assets/javascripts/orator.coffee +29 -0
- data/vendor/assets/swfs/WebSocketMain.swf +0 -0
- metadata +12 -1
data/lib/orator/base.rb
CHANGED
data/lib/orator/version.rb
CHANGED
data/lib/orator.rb
CHANGED
@@ -0,0 +1,166 @@
|
|
1
|
+
###
|
2
|
+
# Handles creating logs for use in the element.
|
3
|
+
###
|
4
|
+
class Orator.Libraries.ChatLogger
|
5
|
+
|
6
|
+
@DEFAULT_OPTIONS:
|
7
|
+
format: """
|
8
|
+
<{{element}} class='{{classes}}'>
|
9
|
+
<span class='sender'>{{from}}</span>
|
10
|
+
<span class='body'>{{-formatted_body}}</span>
|
11
|
+
</{{element}}>
|
12
|
+
"""
|
13
|
+
format_maps: {}
|
14
|
+
format_field: 'event'
|
15
|
+
element: $("<ul/>")
|
16
|
+
|
17
|
+
###
|
18
|
+
# The constructor. This takes a signle options object. This is merged with
|
19
|
+
# the default options and set as the class's options.
|
20
|
+
###
|
21
|
+
constructor: (options)->
|
22
|
+
@options = $.extend(true, {}, Orator.Libraries.ChatLoger.DEFAULT_OPTIONS, options)
|
23
|
+
@format_maps = @options.format_maps || {}
|
24
|
+
|
25
|
+
format_maps: {}
|
26
|
+
|
27
|
+
###
|
28
|
+
# This creates a new chat log and returns it. The argument is an object
|
29
|
+
# that will be used as options for the ChatLog.
|
30
|
+
###
|
31
|
+
log: (options)->
|
32
|
+
new ChatLog(options, @options)
|
33
|
+
|
34
|
+
###
|
35
|
+
# This appends the log to the element specified in the options. The argument
|
36
|
+
# is an object that will be used as the options for ChatLog.
|
37
|
+
###
|
38
|
+
append: (options)->
|
39
|
+
@log(options).to(@options.element)
|
40
|
+
|
41
|
+
###
|
42
|
+
# Handles formatting the logs for appending to an element.
|
43
|
+
###
|
44
|
+
class ChatLog
|
45
|
+
|
46
|
+
###
|
47
|
+
# The default options to be used by the constructor to merge the given
|
48
|
+
# optiosn to.
|
49
|
+
###
|
50
|
+
@DEFAULT_OPTIONS:
|
51
|
+
# The tag that the element will use.
|
52
|
+
tag: 'li'
|
53
|
+
# Data to format the body with.
|
54
|
+
data: {}
|
55
|
+
# The body to put in the log.
|
56
|
+
body: null
|
57
|
+
# The sender of the log. Defaults to "(unknown)".
|
58
|
+
sender: null
|
59
|
+
# The classes the log should have.
|
60
|
+
class: ''
|
61
|
+
|
62
|
+
###
|
63
|
+
# Constructor for the ChatLog. The first argument is an object of user
|
64
|
+
# defined options, and the second one is the options from the ChatLogger.
|
65
|
+
###
|
66
|
+
constructor: (options, chat_logger_options)->
|
67
|
+
@options = $.extend({}, ChatLog.DEFAULT_OPTIONS, options)
|
68
|
+
@cl_opts = chat_logger_options
|
69
|
+
|
70
|
+
###
|
71
|
+
# Appends this ChatLog to the element passed by the first parameter.
|
72
|
+
# Formats the body string, and the format string.
|
73
|
+
#
|
74
|
+
# The format base (@see ChatLogger.DEFAULT_OPTIONS) can access any data
|
75
|
+
# from the options of this class using {{moustache tags}}. The body can
|
76
|
+
# acess any data from the options' data element; if the data is empty,
|
77
|
+
# the body is escaped and passed through.
|
78
|
+
#
|
79
|
+
# @see _format for more information on formatting.
|
80
|
+
###
|
81
|
+
to: (element)->
|
82
|
+
@options.formatted_body = _format(@_get_body(), @options.data)
|
83
|
+
@options.from = @options.sender || @options.data.name || "(unknown)"
|
84
|
+
@options.classes = @_define_classes()
|
85
|
+
_format(@cl_opts.format, @options)
|
86
|
+
|
87
|
+
###
|
88
|
+
# Defines the classes to be used by the element.
|
89
|
+
###
|
90
|
+
_define_classes: ->
|
91
|
+
str_buf = new Orator.Libraries.StrBuf
|
92
|
+
|
93
|
+
str_buf.append(@_escape(@options.class), ' ')
|
94
|
+
if @options.data.history then str_buf.append("history")
|
95
|
+
str_buf.to_s()
|
96
|
+
|
97
|
+
###
|
98
|
+
# Formats a given string `str` with the data from `data`, which are the
|
99
|
+
# first and second arguments, respectively. It replaces stuff from data
|
100
|
+
# that match the {{moustache}} format. It will check to see if `with`
|
101
|
+
# contains the element named by the {{moustache}}. If it does, it will
|
102
|
+
# escape the data and replace the {{moustache}} with it. If it doesn't,
|
103
|
+
# it will check to see if it starts with a `-` and if an element matches
|
104
|
+
# the rest of the name. If it does, it will replace the moustache with
|
105
|
+
# the data (unescaped). Otherwise, it will leave the {{moustache}} alone.
|
106
|
+
###
|
107
|
+
_format: (str, data)->
|
108
|
+
return @_escape(str) if $.isEmptyObject(data)
|
109
|
+
|
110
|
+
str.replace /\{\{(-?.+?)\}\}/g, (_, name)=>
|
111
|
+
if data[name] isnt undefined
|
112
|
+
@_escape(data[name])
|
113
|
+
else if name[0] == '-' and data[name.slice(1)] isnt undefined
|
114
|
+
data[name.slice(1)]
|
115
|
+
else
|
116
|
+
_
|
117
|
+
|
118
|
+
###
|
119
|
+
# Gets the body for use by the formatter. If the body was provided in the
|
120
|
+
# options, return that. Otherwise, try to look up the format in the
|
121
|
+
# format maps provided by the options from the ChatLogger. If it can't
|
122
|
+
# find it, returns an empty string.
|
123
|
+
###
|
124
|
+
_get_body: ->
|
125
|
+
return @options.body if @options.body
|
126
|
+
|
127
|
+
if @cl_opts.format_maps[@options.data[@cl_opts.format_field]]
|
128
|
+
@cl_opts.format_maps[@options.data[@cl_opts.format_field]]
|
129
|
+
else
|
130
|
+
""
|
131
|
+
|
132
|
+
# The map of escape characters for escaping stuff.
|
133
|
+
ESCAPE_CHARS:
|
134
|
+
"&": "&",
|
135
|
+
"<": "<",
|
136
|
+
">": ">",
|
137
|
+
'"': '"',
|
138
|
+
"'": ''',
|
139
|
+
"/": '/',
|
140
|
+
"{": '{',
|
141
|
+
"}": '}'
|
142
|
+
|
143
|
+
grab_keys = (hash)->
|
144
|
+
keys = []
|
145
|
+
|
146
|
+
for k of hash
|
147
|
+
keys.push(k)
|
148
|
+
|
149
|
+
keys
|
150
|
+
|
151
|
+
for_regexp = (array)->
|
152
|
+
array.join('').replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&')
|
153
|
+
|
154
|
+
# The regular expression to match against.
|
155
|
+
ESCAPE_REGEX: new RegExp("[#{for_regexp(grab_keys(@prototype.ESCAPE_CHARS))}]", "g")
|
156
|
+
|
157
|
+
###
|
158
|
+
# Escapes the data passed to it. Matches the data against a regular
|
159
|
+
# expression and if it finds anything, replaces the character with the map
|
160
|
+
# from ESCAPE_CHARS.
|
161
|
+
###
|
162
|
+
_escape: (data)->
|
163
|
+
(data || "").replace @ESCAPE_REGEX, (s)->
|
164
|
+
@ESCAPE_CHARS[s]
|
165
|
+
|
166
|
+
|
@@ -0,0 +1,85 @@
|
|
1
|
+
###
|
2
|
+
# Handles commands that need to be run.
|
3
|
+
###
|
4
|
+
class Orator.Libraries.CommandHandler
|
5
|
+
|
6
|
+
###
|
7
|
+
# Constructs the command handler. No arguments.
|
8
|
+
###
|
9
|
+
constructor:->
|
10
|
+
@commands = []
|
11
|
+
|
12
|
+
###
|
13
|
+
# Lists all of the commands that are registered.
|
14
|
+
###
|
15
|
+
all_commands:->
|
16
|
+
|
17
|
+
for k of @commands
|
18
|
+
k
|
19
|
+
|
20
|
+
###
|
21
|
+
# Sets a command with the given name and function.
|
22
|
+
###
|
23
|
+
command: (name, callback)->
|
24
|
+
@commands[name] = callback
|
25
|
+
|
26
|
+
###
|
27
|
+
# Runs a command in the context of inst with the arguments given. The first
|
28
|
+
# argument is the command, the second the context, and the rest of the
|
29
|
+
# arguments are passed to the command.
|
30
|
+
###
|
31
|
+
trigger: (command, inst, args...)->
|
32
|
+
return false unless @commands[command]
|
33
|
+
@commands[command].apply(inst, args)
|
34
|
+
|
35
|
+
###
|
36
|
+
# Parses the line and runs the command given from the line in the context of
|
37
|
+
# inst. First argument is a string containing the command and arguments,
|
38
|
+
# Second argument is the context to run under.
|
39
|
+
###
|
40
|
+
run: (line, inst)->
|
41
|
+
[command, args] = @parse(line)
|
42
|
+
|
43
|
+
@trigger.apply(@, [command, inst].concat(args))
|
44
|
+
|
45
|
+
###
|
46
|
+
# Parses a string containing the command and arguments. Handles quotes, but
|
47
|
+
# not escaping of them.
|
48
|
+
#
|
49
|
+
# @TODO: handle escaping quotes.
|
50
|
+
# @TODO: handle escaping spaces.
|
51
|
+
###
|
52
|
+
parse: (line)->
|
53
|
+
command = new Orator.Libraries.StrBuf
|
54
|
+
args = []
|
55
|
+
state = { in_quote: false, argument: -1 }
|
56
|
+
|
57
|
+
next_argument = ()->
|
58
|
+
state.argument++
|
59
|
+
args.push new Orator.Libraries.StrBuf
|
60
|
+
|
61
|
+
append = (c)->
|
62
|
+
if state.argument == -1
|
63
|
+
command.append(c)
|
64
|
+
else
|
65
|
+
args.slice(-1)[0].append(c)
|
66
|
+
|
67
|
+
|
68
|
+
for c in line
|
69
|
+
|
70
|
+
switch c
|
71
|
+
when ' '
|
72
|
+
if state.in_quote
|
73
|
+
append(c)
|
74
|
+
else
|
75
|
+
next_argument()
|
76
|
+
when '"', "'"
|
77
|
+
if state.in_quote == c
|
78
|
+
state.in_quote = false
|
79
|
+
else if state.in_quote isnt false
|
80
|
+
append(c)
|
81
|
+
else
|
82
|
+
state.in_quote = c
|
83
|
+
else append(c)
|
84
|
+
|
85
|
+
[command.toString(), args.map (arg)-> arg.toString()]
|
@@ -0,0 +1,45 @@
|
|
1
|
+
###
|
2
|
+
# Handles the events that the server passes.
|
3
|
+
###
|
4
|
+
class Orator.Libraries.EventHandler
|
5
|
+
|
6
|
+
###
|
7
|
+
# Constructor.
|
8
|
+
###
|
9
|
+
constructor: (@context)->
|
10
|
+
@events = []
|
11
|
+
|
12
|
+
###
|
13
|
+
# Sets up a callback for the event given. First argument is the event, the
|
14
|
+
# second is the callback.
|
15
|
+
###
|
16
|
+
on: (event, cb, times = Infinity)->
|
17
|
+
@events.push([event, cb, times])
|
18
|
+
|
19
|
+
###
|
20
|
+
# Runs the event with the context and the arguments. This runs all of the
|
21
|
+
# events that match, and returns an array of the results.
|
22
|
+
###
|
23
|
+
trigger: (event, args...)->
|
24
|
+
events = @_find_events(event)
|
25
|
+
|
26
|
+
return false if events.length == 0
|
27
|
+
|
28
|
+
for event in events
|
29
|
+
event[1].apply(@context, args)
|
30
|
+
|
31
|
+
if event[2]-- <= 0
|
32
|
+
@events.splice(@events.indexOf(event), 1)
|
33
|
+
|
34
|
+
###
|
35
|
+
# Finds the events that match the given event.
|
36
|
+
###
|
37
|
+
_find_events: (event)->
|
38
|
+
out = []
|
39
|
+
|
40
|
+
for r in @events
|
41
|
+
if r[0] == event
|
42
|
+
out.push(r)
|
43
|
+
|
44
|
+
out
|
45
|
+
|
@@ -0,0 +1,19 @@
|
|
1
|
+
###
|
2
|
+
# This is placed in between the socket and the developer to make sending events
|
3
|
+
# easier.
|
4
|
+
###
|
5
|
+
class Orator.Libraries.MiddleGround
|
6
|
+
|
7
|
+
###
|
8
|
+
# Constructor.
|
9
|
+
###
|
10
|
+
constructor: (@conn)->
|
11
|
+
|
12
|
+
###
|
13
|
+
# Sends an event to the server with the data provided. First argument is the
|
14
|
+
# event, second argument is the data.
|
15
|
+
###
|
16
|
+
send: (event, data)->
|
17
|
+
data.event = event
|
18
|
+
|
19
|
+
@conn.send $.stringify(data)
|
@@ -0,0 +1,39 @@
|
|
1
|
+
###
|
2
|
+
# Creates a buffer for a string.
|
3
|
+
###
|
4
|
+
class Orator.Libraries.StrBuf
|
5
|
+
|
6
|
+
###
|
7
|
+
# Creates the buffer. First argument is the string it starts with, the
|
8
|
+
# second one is what is used when joining the strings together.
|
9
|
+
###
|
10
|
+
constructor: (start = '', @join = '')->
|
11
|
+
@parts = [start]
|
12
|
+
@cache = null
|
13
|
+
|
14
|
+
###
|
15
|
+
# Appends to the buffer. Can take any number of arguments. If the argument
|
16
|
+
# is a StrBuf, concats its buffer to this buffer. If it's anything else, it
|
17
|
+
# tries its best to convert it to a string.
|
18
|
+
###
|
19
|
+
append:->
|
20
|
+
@cache = null
|
21
|
+
|
22
|
+
for arg in arguments
|
23
|
+
|
24
|
+
if arg._is_strbuf
|
25
|
+
@parts = @parts.concat(arg.parts)
|
26
|
+
else
|
27
|
+
@parts.push((arg || "").toString())
|
28
|
+
|
29
|
+
push: @prototype.append
|
30
|
+
|
31
|
+
###
|
32
|
+
# Converts the buffer to a string. Caches the result, if it can.
|
33
|
+
###
|
34
|
+
toString:->
|
35
|
+
@cache ||= @parts.join(@join)
|
36
|
+
|
37
|
+
_is_strbuf: true
|
38
|
+
|
39
|
+
to_s: @prototype.toString
|
@@ -0,0 +1,59 @@
|
|
1
|
+
if(!jQuery.stringify) {
|
2
|
+
jQuery.extend({
|
3
|
+
stringify: function stringify(obj) {
|
4
|
+
var t = typeof (obj);
|
5
|
+
if (t != "object" || obj === null) {
|
6
|
+
// simple data type
|
7
|
+
if (t == "string") obj = '"' + obj + '"';
|
8
|
+
return String(obj);
|
9
|
+
} else {
|
10
|
+
// recurse array or object
|
11
|
+
var n, v, json = [], arr = (obj && obj.constructor == Array);
|
12
|
+
|
13
|
+
for (n in obj) {
|
14
|
+
v = obj[n];
|
15
|
+
t = typeof(v);
|
16
|
+
if (obj.hasOwnProperty(n)) {
|
17
|
+
if (t == "string") v = '"' + v + '"'; else if (t == "object" && v !== null) v = jQuery.stringify(v);
|
18
|
+
json.push((arr ? "" : '"' + n + '":') + String(v));
|
19
|
+
}
|
20
|
+
}
|
21
|
+
return (arr ? "[" : "{") + String(json) + (arr ? "]" : "}");
|
22
|
+
}
|
23
|
+
}
|
24
|
+
});
|
25
|
+
}
|
26
|
+
|
27
|
+
// thank you, MDN!
|
28
|
+
if (!Array.prototype.indexOf) {
|
29
|
+
Array.prototype.indexOf = function (searchElement /*, fromIndex */ ) {
|
30
|
+
"use strict";
|
31
|
+
if (this == null) {
|
32
|
+
throw new TypeError();
|
33
|
+
}
|
34
|
+
var t = Object(this);
|
35
|
+
var len = t.length >>> 0;
|
36
|
+
if (len === 0) {
|
37
|
+
return -1;
|
38
|
+
}
|
39
|
+
var n = 0;
|
40
|
+
if (arguments.length > 1) {
|
41
|
+
n = Number(arguments[1]);
|
42
|
+
if (n != n) { // shortcut for verifying if it's NaN
|
43
|
+
n = 0;
|
44
|
+
} else if (n != 0 && n != Infinity && n != -Infinity) {
|
45
|
+
n = (n > 0 || -1) * Math.floor(Math.abs(n));
|
46
|
+
}
|
47
|
+
}
|
48
|
+
if (n >= len) {
|
49
|
+
return -1;
|
50
|
+
}
|
51
|
+
var k = n >= 0 ? n : Math.max(len - Math.abs(n), 0);
|
52
|
+
for (; k < len; k++) {
|
53
|
+
if (k in t && t[k] === searchElement) {
|
54
|
+
return k;
|
55
|
+
}
|
56
|
+
}
|
57
|
+
return -1;
|
58
|
+
}
|
59
|
+
}
|
@@ -0,0 +1,4 @@
|
|
1
|
+
/* SWFObject v2.2 <http://code.google.com/p/swfobject/>
|
2
|
+
is released under the MIT License <http://www.opensource.org/licenses/mit-license.php>
|
3
|
+
*/
|
4
|
+
var swfobject=function(){var D="undefined",r="object",S="Shockwave Flash",W="ShockwaveFlash.ShockwaveFlash",q="application/x-shockwave-flash",R="SWFObjectExprInst",x="onreadystatechange",O=window,j=document,t=navigator,T=false,U=[h],o=[],N=[],I=[],l,Q,E,B,J=false,a=false,n,G,m=true,M=function(){var aa=typeof j.getElementById!=D&&typeof j.getElementsByTagName!=D&&typeof j.createElement!=D,ah=t.userAgent.toLowerCase(),Y=t.platform.toLowerCase(),ae=Y?/win/.test(Y):/win/.test(ah),ac=Y?/mac/.test(Y):/mac/.test(ah),af=/webkit/.test(ah)?parseFloat(ah.replace(/^.*webkit\/(\d+(\.\d+)?).*$/,"$1")):false,X=!+"\v1",ag=[0,0,0],ab=null;if(typeof t.plugins!=D&&typeof t.plugins[S]==r){ab=t.plugins[S].description;if(ab&&!(typeof t.mimeTypes!=D&&t.mimeTypes[q]&&!t.mimeTypes[q].enabledPlugin)){T=true;X=false;ab=ab.replace(/^.*\s+(\S+\s+\S+$)/,"$1");ag[0]=parseInt(ab.replace(/^(.*)\..*$/,"$1"),10);ag[1]=parseInt(ab.replace(/^.*\.(.*)\s.*$/,"$1"),10);ag[2]=/[a-zA-Z]/.test(ab)?parseInt(ab.replace(/^.*[a-zA-Z]+(.*)$/,"$1"),10):0}}else{if(typeof O.ActiveXObject!=D){try{var ad=new ActiveXObject(W);if(ad){ab=ad.GetVariable("$version");if(ab){X=true;ab=ab.split(" ")[1].split(",");ag=[parseInt(ab[0],10),parseInt(ab[1],10),parseInt(ab[2],10)]}}}catch(Z){}}}return{w3:aa,pv:ag,wk:af,ie:X,win:ae,mac:ac}}(),k=function(){if(!M.w3){return}if((typeof j.readyState!=D&&j.readyState=="complete")||(typeof j.readyState==D&&(j.getElementsByTagName("body")[0]||j.body))){f()}if(!J){if(typeof j.addEventListener!=D){j.addEventListener("DOMContentLoaded",f,false)}if(M.ie&&M.win){j.attachEvent(x,function(){if(j.readyState=="complete"){j.detachEvent(x,arguments.callee);f()}});if(O==top){(function(){if(J){return}try{j.documentElement.doScroll("left")}catch(X){setTimeout(arguments.callee,0);return}f()})()}}if(M.wk){(function(){if(J){return}if(!/loaded|complete/.test(j.readyState)){setTimeout(arguments.callee,0);return}f()})()}s(f)}}();function f(){if(J){return}try{var Z=j.getElementsByTagName("body")[0].appendChild(C("span"));Z.parentNode.removeChild(Z)}catch(aa){return}J=true;var X=U.length;for(var Y=0;Y<X;Y++){U[Y]()}}function K(X){if(J){X()}else{U[U.length]=X}}function s(Y){if(typeof O.addEventListener!=D){O.addEventListener("load",Y,false)}else{if(typeof j.addEventListener!=D){j.addEventListener("load",Y,false)}else{if(typeof O.attachEvent!=D){i(O,"onload",Y)}else{if(typeof O.onload=="function"){var X=O.onload;O.onload=function(){X();Y()}}else{O.onload=Y}}}}}function h(){if(T){V()}else{H()}}function V(){var X=j.getElementsByTagName("body")[0];var aa=C(r);aa.setAttribute("type",q);var Z=X.appendChild(aa);if(Z){var Y=0;(function(){if(typeof Z.GetVariable!=D){var ab=Z.GetVariable("$version");if(ab){ab=ab.split(" ")[1].split(",");M.pv=[parseInt(ab[0],10),parseInt(ab[1],10),parseInt(ab[2],10)]}}else{if(Y<10){Y++;setTimeout(arguments.callee,10);return}}X.removeChild(aa);Z=null;H()})()}else{H()}}function H(){var ag=o.length;if(ag>0){for(var af=0;af<ag;af++){var Y=o[af].id;var ab=o[af].callbackFn;var aa={success:false,id:Y};if(M.pv[0]>0){var ae=c(Y);if(ae){if(F(o[af].swfVersion)&&!(M.wk&&M.wk<312)){w(Y,true);if(ab){aa.success=true;aa.ref=z(Y);ab(aa)}}else{if(o[af].expressInstall&&A()){var ai={};ai.data=o[af].expressInstall;ai.width=ae.getAttribute("width")||"0";ai.height=ae.getAttribute("height")||"0";if(ae.getAttribute("class")){ai.styleclass=ae.getAttribute("class")}if(ae.getAttribute("align")){ai.align=ae.getAttribute("align")}var ah={};var X=ae.getElementsByTagName("param");var ac=X.length;for(var ad=0;ad<ac;ad++){if(X[ad].getAttribute("name").toLowerCase()!="movie"){ah[X[ad].getAttribute("name")]=X[ad].getAttribute("value")}}P(ai,ah,Y,ab)}else{p(ae);if(ab){ab(aa)}}}}}else{w(Y,true);if(ab){var Z=z(Y);if(Z&&typeof Z.SetVariable!=D){aa.success=true;aa.ref=Z}ab(aa)}}}}}function z(aa){var X=null;var Y=c(aa);if(Y&&Y.nodeName=="OBJECT"){if(typeof Y.SetVariable!=D){X=Y}else{var Z=Y.getElementsByTagName(r)[0];if(Z){X=Z}}}return X}function A(){return !a&&F("6.0.65")&&(M.win||M.mac)&&!(M.wk&&M.wk<312)}function P(aa,ab,X,Z){a=true;E=Z||null;B={success:false,id:X};var ae=c(X);if(ae){if(ae.nodeName=="OBJECT"){l=g(ae);Q=null}else{l=ae;Q=X}aa.id=R;if(typeof aa.width==D||(!/%$/.test(aa.width)&&parseInt(aa.width,10)<310)){aa.width="310"}if(typeof aa.height==D||(!/%$/.test(aa.height)&&parseInt(aa.height,10)<137)){aa.height="137"}j.title=j.title.slice(0,47)+" - Flash Player Installation";var ad=M.ie&&M.win?"ActiveX":"PlugIn",ac="MMredirectURL="+O.location.toString().replace(/&/g,"%26")+"&MMplayerType="+ad+"&MMdoctitle="+j.title;if(typeof ab.flashvars!=D){ab.flashvars+="&"+ac}else{ab.flashvars=ac}if(M.ie&&M.win&&ae.readyState!=4){var Y=C("div");X+="SWFObjectNew";Y.setAttribute("id",X);ae.parentNode.insertBefore(Y,ae);ae.style.display="none";(function(){if(ae.readyState==4){ae.parentNode.removeChild(ae)}else{setTimeout(arguments.callee,10)}})()}u(aa,ab,X)}}function p(Y){if(M.ie&&M.win&&Y.readyState!=4){var X=C("div");Y.parentNode.insertBefore(X,Y);X.parentNode.replaceChild(g(Y),X);Y.style.display="none";(function(){if(Y.readyState==4){Y.parentNode.removeChild(Y)}else{setTimeout(arguments.callee,10)}})()}else{Y.parentNode.replaceChild(g(Y),Y)}}function g(ab){var aa=C("div");if(M.win&&M.ie){aa.innerHTML=ab.innerHTML}else{var Y=ab.getElementsByTagName(r)[0];if(Y){var ad=Y.childNodes;if(ad){var X=ad.length;for(var Z=0;Z<X;Z++){if(!(ad[Z].nodeType==1&&ad[Z].nodeName=="PARAM")&&!(ad[Z].nodeType==8)){aa.appendChild(ad[Z].cloneNode(true))}}}}}return aa}function u(ai,ag,Y){var X,aa=c(Y);if(M.wk&&M.wk<312){return X}if(aa){if(typeof ai.id==D){ai.id=Y}if(M.ie&&M.win){var ah="";for(var ae in ai){if(ai[ae]!=Object.prototype[ae]){if(ae.toLowerCase()=="data"){ag.movie=ai[ae]}else{if(ae.toLowerCase()=="styleclass"){ah+=' class="'+ai[ae]+'"'}else{if(ae.toLowerCase()!="classid"){ah+=" "+ae+'="'+ai[ae]+'"'}}}}}var af="";for(var ad in ag){if(ag[ad]!=Object.prototype[ad]){af+='<param name="'+ad+'" value="'+ag[ad]+'" />'}}aa.outerHTML='<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"'+ah+">"+af+"</object>";N[N.length]=ai.id;X=c(ai.id)}else{var Z=C(r);Z.setAttribute("type",q);for(var ac in ai){if(ai[ac]!=Object.prototype[ac]){if(ac.toLowerCase()=="styleclass"){Z.setAttribute("class",ai[ac])}else{if(ac.toLowerCase()!="classid"){Z.setAttribute(ac,ai[ac])}}}}for(var ab in ag){if(ag[ab]!=Object.prototype[ab]&&ab.toLowerCase()!="movie"){e(Z,ab,ag[ab])}}aa.parentNode.replaceChild(Z,aa);X=Z}}return X}function e(Z,X,Y){var aa=C("param");aa.setAttribute("name",X);aa.setAttribute("value",Y);Z.appendChild(aa)}function y(Y){var X=c(Y);if(X&&X.nodeName=="OBJECT"){if(M.ie&&M.win){X.style.display="none";(function(){if(X.readyState==4){b(Y)}else{setTimeout(arguments.callee,10)}})()}else{X.parentNode.removeChild(X)}}}function b(Z){var Y=c(Z);if(Y){for(var X in Y){if(typeof Y[X]=="function"){Y[X]=null}}Y.parentNode.removeChild(Y)}}function c(Z){var X=null;try{X=j.getElementById(Z)}catch(Y){}return X}function C(X){return j.createElement(X)}function i(Z,X,Y){Z.attachEvent(X,Y);I[I.length]=[Z,X,Y]}function F(Z){var Y=M.pv,X=Z.split(".");X[0]=parseInt(X[0],10);X[1]=parseInt(X[1],10)||0;X[2]=parseInt(X[2],10)||0;return(Y[0]>X[0]||(Y[0]==X[0]&&Y[1]>X[1])||(Y[0]==X[0]&&Y[1]==X[1]&&Y[2]>=X[2]))?true:false}function v(ac,Y,ad,ab){if(M.ie&&M.mac){return}var aa=j.getElementsByTagName("head")[0];if(!aa){return}var X=(ad&&typeof ad=="string")?ad:"screen";if(ab){n=null;G=null}if(!n||G!=X){var Z=C("style");Z.setAttribute("type","text/css");Z.setAttribute("media",X);n=aa.appendChild(Z);if(M.ie&&M.win&&typeof j.styleSheets!=D&&j.styleSheets.length>0){n=j.styleSheets[j.styleSheets.length-1]}G=X}if(M.ie&&M.win){if(n&&typeof n.addRule==r){n.addRule(ac,Y)}}else{if(n&&typeof j.createTextNode!=D){n.appendChild(j.createTextNode(ac+" {"+Y+"}"))}}}function w(Z,X){if(!m){return}var Y=X?"visible":"hidden";if(J&&c(Z)){c(Z).style.visibility=Y}else{v("#"+Z,"visibility:"+Y)}}function L(Y){var Z=/[\\\"<>\.;]/;var X=Z.exec(Y)!=null;return X&&typeof encodeURIComponent!=D?encodeURIComponent(Y):Y}var d=function(){if(M.ie&&M.win){window.attachEvent("onunload",function(){var ac=I.length;for(var ab=0;ab<ac;ab++){I[ab][0].detachEvent(I[ab][1],I[ab][2])}var Z=N.length;for(var aa=0;aa<Z;aa++){y(N[aa])}for(var Y in M){M[Y]=null}M=null;for(var X in swfobject){swfobject[X]=null}swfobject=null})}}();return{registerObject:function(ab,X,aa,Z){if(M.w3&&ab&&X){var Y={};Y.id=ab;Y.swfVersion=X;Y.expressInstall=aa;Y.callbackFn=Z;o[o.length]=Y;w(ab,false)}else{if(Z){Z({success:false,id:ab})}}},getObjectById:function(X){if(M.w3){return z(X)}},embedSWF:function(ab,ah,ae,ag,Y,aa,Z,ad,af,ac){var X={success:false,id:ah};if(M.w3&&!(M.wk&&M.wk<312)&&ab&&ah&&ae&&ag&&Y){w(ah,false);K(function(){ae+="";ag+="";var aj={};if(af&&typeof af===r){for(var al in af){aj[al]=af[al]}}aj.data=ab;aj.width=ae;aj.height=ag;var am={};if(ad&&typeof ad===r){for(var ak in ad){am[ak]=ad[ak]}}if(Z&&typeof Z===r){for(var ai in Z){if(typeof am.flashvars!=D){am.flashvars+="&"+ai+"="+Z[ai]}else{am.flashvars=ai+"="+Z[ai]}}}if(F(Y)){var an=u(aj,am,ah);if(aj.id==ah){w(ah,true)}X.success=true;X.ref=an}else{if(aa&&A()){aj.data=aa;P(aj,am,ah,ac);return}else{w(ah,true)}}if(ac){ac(X)}})}else{if(ac){ac(X)}}},switchOffAutoHideShow:function(){m=false},ua:M,getFlashPlayerVersion:function(){return{major:M.pv[0],minor:M.pv[1],release:M.pv[2]}},hasFlashPlayerVersion:F,createSWF:function(Z,Y,X){if(M.w3){return u(Z,Y,X)}else{return undefined}},showExpressInstall:function(Z,aa,X,Y){if(M.w3&&A()){P(Z,aa,X,Y)}},removeSWF:function(X){if(M.w3){y(X)}},createCSS:function(aa,Z,Y,X){if(M.w3){v(aa,Z,Y,X)}},addDomLoadEvent:K,addLoadEvent:s,getQueryParamValue:function(aa){var Z=j.location.search||j.location.hash;if(Z){if(/\?/.test(Z)){Z=Z.split("?")[1]}if(aa==null){return L(Z)}var Y=Z.split("&");for(var X=0;X<Y.length;X++){if(Y[X].substring(0,Y[X].indexOf("="))==aa){return L(Y[X].substring((Y[X].indexOf("=")+1)))}}}return""},expressInstallCallback:function(){if(a){var X=c(R);if(X&&l){X.parentNode.replaceChild(l,X);if(Q){w(Q,true);if(M.ie&&M.win){l.style.display="block"}}if(E){E(B)}}a=false}}}}();
|
@@ -0,0 +1,391 @@
|
|
1
|
+
// Copyright: Hiroshi Ichikawa <http://gimite.net/en/>
|
2
|
+
// License: New BSD License
|
3
|
+
// Reference: http://dev.w3.org/html5/websockets/
|
4
|
+
// Reference: http://tools.ietf.org/html/rfc6455
|
5
|
+
|
6
|
+
(function() {
|
7
|
+
|
8
|
+
if (window.WEB_SOCKET_FORCE_FLASH) {
|
9
|
+
// Keeps going.
|
10
|
+
} else if (window.WebSocket) {
|
11
|
+
return;
|
12
|
+
} else if (window.MozWebSocket) {
|
13
|
+
// Firefox.
|
14
|
+
window.WebSocket = MozWebSocket;
|
15
|
+
return;
|
16
|
+
}
|
17
|
+
|
18
|
+
var logger;
|
19
|
+
if (window.WEB_SOCKET_LOGGER) {
|
20
|
+
logger = WEB_SOCKET_LOGGER;
|
21
|
+
} else if (window.console && window.console.log && window.console.error) {
|
22
|
+
// In some environment, console is defined but console.log or console.error is missing.
|
23
|
+
logger = window.console;
|
24
|
+
} else {
|
25
|
+
logger = {log: function(){ }, error: function(){ }};
|
26
|
+
}
|
27
|
+
|
28
|
+
// swfobject.hasFlashPlayerVersion("10.0.0") doesn't work with Gnash.
|
29
|
+
if (swfobject.getFlashPlayerVersion().major < 10) {
|
30
|
+
logger.error("Flash Player >= 10.0.0 is required.");
|
31
|
+
return;
|
32
|
+
}
|
33
|
+
if (location.protocol == "file:") {
|
34
|
+
logger.error(
|
35
|
+
"WARNING: web-socket-js doesn't work in file:///... URL " +
|
36
|
+
"unless you set Flash Security Settings properly. " +
|
37
|
+
"Open the page via Web server i.e. http://...");
|
38
|
+
}
|
39
|
+
|
40
|
+
/**
|
41
|
+
* Our own implementation of WebSocket class using Flash.
|
42
|
+
* @param {string} url
|
43
|
+
* @param {array or string} protocols
|
44
|
+
* @param {string} proxyHost
|
45
|
+
* @param {int} proxyPort
|
46
|
+
* @param {string} headers
|
47
|
+
*/
|
48
|
+
window.WebSocket = function(url, protocols, proxyHost, proxyPort, headers) {
|
49
|
+
var self = this;
|
50
|
+
self.__id = WebSocket.__nextId++;
|
51
|
+
WebSocket.__instances[self.__id] = self;
|
52
|
+
self.readyState = WebSocket.CONNECTING;
|
53
|
+
self.bufferedAmount = 0;
|
54
|
+
self.__events = {};
|
55
|
+
if (!protocols) {
|
56
|
+
protocols = [];
|
57
|
+
} else if (typeof protocols == "string") {
|
58
|
+
protocols = [protocols];
|
59
|
+
}
|
60
|
+
// Uses setTimeout() to make sure __createFlash() runs after the caller sets ws.onopen etc.
|
61
|
+
// Otherwise, when onopen fires immediately, onopen is called before it is set.
|
62
|
+
self.__createTask = setTimeout(function() {
|
63
|
+
WebSocket.__addTask(function() {
|
64
|
+
self.__createTask = null;
|
65
|
+
WebSocket.__flash.create(
|
66
|
+
self.__id, url, protocols, proxyHost || null, proxyPort || 0, headers || null);
|
67
|
+
});
|
68
|
+
}, 0);
|
69
|
+
};
|
70
|
+
|
71
|
+
/**
|
72
|
+
* Send data to the web socket.
|
73
|
+
* @param {string} data The data to send to the socket.
|
74
|
+
* @return {boolean} True for success, false for failure.
|
75
|
+
*/
|
76
|
+
WebSocket.prototype.send = function(data) {
|
77
|
+
if (this.readyState == WebSocket.CONNECTING) {
|
78
|
+
throw "INVALID_STATE_ERR: Web Socket connection has not been established";
|
79
|
+
}
|
80
|
+
// We use encodeURIComponent() here, because FABridge doesn't work if
|
81
|
+
// the argument includes some characters. We don't use escape() here
|
82
|
+
// because of this:
|
83
|
+
// https://developer.mozilla.org/en/Core_JavaScript_1.5_Guide/Functions#escape_and_unescape_Functions
|
84
|
+
// But it looks decodeURIComponent(encodeURIComponent(s)) doesn't
|
85
|
+
// preserve all Unicode characters either e.g. "\uffff" in Firefox.
|
86
|
+
// Note by wtritch: Hopefully this will not be necessary using ExternalInterface. Will require
|
87
|
+
// additional testing.
|
88
|
+
var result = WebSocket.__flash.send(this.__id, encodeURIComponent(data));
|
89
|
+
if (result < 0) { // success
|
90
|
+
return true;
|
91
|
+
} else {
|
92
|
+
this.bufferedAmount += result;
|
93
|
+
return false;
|
94
|
+
}
|
95
|
+
};
|
96
|
+
|
97
|
+
/**
|
98
|
+
* Close this web socket gracefully.
|
99
|
+
*/
|
100
|
+
WebSocket.prototype.close = function() {
|
101
|
+
if (this.__createTask) {
|
102
|
+
clearTimeout(this.__createTask);
|
103
|
+
this.__createTask = null;
|
104
|
+
this.readyState = WebSocket.CLOSED;
|
105
|
+
return;
|
106
|
+
}
|
107
|
+
if (this.readyState == WebSocket.CLOSED || this.readyState == WebSocket.CLOSING) {
|
108
|
+
return;
|
109
|
+
}
|
110
|
+
this.readyState = WebSocket.CLOSING;
|
111
|
+
WebSocket.__flash.close(this.__id);
|
112
|
+
};
|
113
|
+
|
114
|
+
/**
|
115
|
+
* Implementation of {@link <a href="http://www.w3.org/TR/DOM-Level-2-Events/events.html#Events-registration">DOM 2 EventTarget Interface</a>}
|
116
|
+
*
|
117
|
+
* @param {string} type
|
118
|
+
* @param {function} listener
|
119
|
+
* @param {boolean} useCapture
|
120
|
+
* @return void
|
121
|
+
*/
|
122
|
+
WebSocket.prototype.addEventListener = function(type, listener, useCapture) {
|
123
|
+
if (!(type in this.__events)) {
|
124
|
+
this.__events[type] = [];
|
125
|
+
}
|
126
|
+
this.__events[type].push(listener);
|
127
|
+
};
|
128
|
+
|
129
|
+
/**
|
130
|
+
* Implementation of {@link <a href="http://www.w3.org/TR/DOM-Level-2-Events/events.html#Events-registration">DOM 2 EventTarget Interface</a>}
|
131
|
+
*
|
132
|
+
* @param {string} type
|
133
|
+
* @param {function} listener
|
134
|
+
* @param {boolean} useCapture
|
135
|
+
* @return void
|
136
|
+
*/
|
137
|
+
WebSocket.prototype.removeEventListener = function(type, listener, useCapture) {
|
138
|
+
if (!(type in this.__events)) return;
|
139
|
+
var events = this.__events[type];
|
140
|
+
for (var i = events.length - 1; i >= 0; --i) {
|
141
|
+
if (events[i] === listener) {
|
142
|
+
events.splice(i, 1);
|
143
|
+
break;
|
144
|
+
}
|
145
|
+
}
|
146
|
+
};
|
147
|
+
|
148
|
+
/**
|
149
|
+
* Implementation of {@link <a href="http://www.w3.org/TR/DOM-Level-2-Events/events.html#Events-registration">DOM 2 EventTarget Interface</a>}
|
150
|
+
*
|
151
|
+
* @param {Event} event
|
152
|
+
* @return void
|
153
|
+
*/
|
154
|
+
WebSocket.prototype.dispatchEvent = function(event) {
|
155
|
+
var events = this.__events[event.type] || [];
|
156
|
+
for (var i = 0; i < events.length; ++i) {
|
157
|
+
events[i](event);
|
158
|
+
}
|
159
|
+
var handler = this["on" + event.type];
|
160
|
+
if (handler) handler.apply(this, [event]);
|
161
|
+
};
|
162
|
+
|
163
|
+
/**
|
164
|
+
* Handles an event from Flash.
|
165
|
+
* @param {Object} flashEvent
|
166
|
+
*/
|
167
|
+
WebSocket.prototype.__handleEvent = function(flashEvent) {
|
168
|
+
|
169
|
+
if ("readyState" in flashEvent) {
|
170
|
+
this.readyState = flashEvent.readyState;
|
171
|
+
}
|
172
|
+
if ("protocol" in flashEvent) {
|
173
|
+
this.protocol = flashEvent.protocol;
|
174
|
+
}
|
175
|
+
|
176
|
+
var jsEvent;
|
177
|
+
if (flashEvent.type == "open" || flashEvent.type == "error") {
|
178
|
+
jsEvent = this.__createSimpleEvent(flashEvent.type);
|
179
|
+
} else if (flashEvent.type == "close") {
|
180
|
+
jsEvent = this.__createSimpleEvent("close");
|
181
|
+
jsEvent.wasClean = flashEvent.wasClean ? true : false;
|
182
|
+
jsEvent.code = flashEvent.code;
|
183
|
+
jsEvent.reason = flashEvent.reason;
|
184
|
+
} else if (flashEvent.type == "message") {
|
185
|
+
var data = decodeURIComponent(flashEvent.message);
|
186
|
+
jsEvent = this.__createMessageEvent("message", data);
|
187
|
+
} else {
|
188
|
+
throw "unknown event type: " + flashEvent.type;
|
189
|
+
}
|
190
|
+
|
191
|
+
this.dispatchEvent(jsEvent);
|
192
|
+
|
193
|
+
};
|
194
|
+
|
195
|
+
WebSocket.prototype.__createSimpleEvent = function(type) {
|
196
|
+
if (document.createEvent && window.Event) {
|
197
|
+
var event = document.createEvent("Event");
|
198
|
+
event.initEvent(type, false, false);
|
199
|
+
return event;
|
200
|
+
} else {
|
201
|
+
return {type: type, bubbles: false, cancelable: false};
|
202
|
+
}
|
203
|
+
};
|
204
|
+
|
205
|
+
WebSocket.prototype.__createMessageEvent = function(type, data) {
|
206
|
+
if (document.createEvent && window.MessageEvent && !window.opera) {
|
207
|
+
var event = document.createEvent("MessageEvent");
|
208
|
+
event.initMessageEvent("message", false, false, data, null, null, window, null);
|
209
|
+
return event;
|
210
|
+
} else {
|
211
|
+
// IE and Opera, the latter one truncates the data parameter after any 0x00 bytes.
|
212
|
+
return {type: type, data: data, bubbles: false, cancelable: false};
|
213
|
+
}
|
214
|
+
};
|
215
|
+
|
216
|
+
/**
|
217
|
+
* Define the WebSocket readyState enumeration.
|
218
|
+
*/
|
219
|
+
WebSocket.CONNECTING = 0;
|
220
|
+
WebSocket.OPEN = 1;
|
221
|
+
WebSocket.CLOSING = 2;
|
222
|
+
WebSocket.CLOSED = 3;
|
223
|
+
|
224
|
+
// Field to check implementation of WebSocket.
|
225
|
+
WebSocket.__isFlashImplementation = true;
|
226
|
+
WebSocket.__initialized = false;
|
227
|
+
WebSocket.__flash = null;
|
228
|
+
WebSocket.__instances = {};
|
229
|
+
WebSocket.__tasks = [];
|
230
|
+
WebSocket.__nextId = 0;
|
231
|
+
|
232
|
+
/**
|
233
|
+
* Load a new flash security policy file.
|
234
|
+
* @param {string} url
|
235
|
+
*/
|
236
|
+
WebSocket.loadFlashPolicyFile = function(url){
|
237
|
+
WebSocket.__addTask(function() {
|
238
|
+
WebSocket.__flash.loadManualPolicyFile(url);
|
239
|
+
});
|
240
|
+
};
|
241
|
+
|
242
|
+
/**
|
243
|
+
* Loads WebSocketMain.swf and creates WebSocketMain object in Flash.
|
244
|
+
*/
|
245
|
+
WebSocket.__initialize = function() {
|
246
|
+
|
247
|
+
if (WebSocket.__initialized) return;
|
248
|
+
WebSocket.__initialized = true;
|
249
|
+
|
250
|
+
if (WebSocket.__swfLocation) {
|
251
|
+
// For backword compatibility.
|
252
|
+
window.WEB_SOCKET_SWF_LOCATION = WebSocket.__swfLocation;
|
253
|
+
}
|
254
|
+
if (!window.WEB_SOCKET_SWF_LOCATION) {
|
255
|
+
logger.error("[WebSocket] set WEB_SOCKET_SWF_LOCATION to location of WebSocketMain.swf");
|
256
|
+
return;
|
257
|
+
}
|
258
|
+
if (!window.WEB_SOCKET_SUPPRESS_CROSS_DOMAIN_SWF_ERROR &&
|
259
|
+
!WEB_SOCKET_SWF_LOCATION.match(/(^|\/)WebSocketMainInsecure\.swf(\?.*)?$/) &&
|
260
|
+
WEB_SOCKET_SWF_LOCATION.match(/^\w+:\/\/([^\/]+)/)) {
|
261
|
+
var swfHost = RegExp.$1;
|
262
|
+
if (location.host != swfHost) {
|
263
|
+
logger.error(
|
264
|
+
"[WebSocket] You must host HTML and WebSocketMain.swf in the same host " +
|
265
|
+
"('" + location.host + "' != '" + swfHost + "'). " +
|
266
|
+
"See also 'How to host HTML file and SWF file in different domains' section " +
|
267
|
+
"in README.md. If you use WebSocketMainInsecure.swf, you can suppress this message " +
|
268
|
+
"by WEB_SOCKET_SUPPRESS_CROSS_DOMAIN_SWF_ERROR = true;");
|
269
|
+
}
|
270
|
+
}
|
271
|
+
var container = document.createElement("div");
|
272
|
+
container.id = "webSocketContainer";
|
273
|
+
// Hides Flash box. We cannot use display: none or visibility: hidden because it prevents
|
274
|
+
// Flash from loading at least in IE. So we move it out of the screen at (-100, -100).
|
275
|
+
// But this even doesn't work with Flash Lite (e.g. in Droid Incredible). So with Flash
|
276
|
+
// Lite, we put it at (0, 0). This shows 1x1 box visible at left-top corner but this is
|
277
|
+
// the best we can do as far as we know now.
|
278
|
+
container.style.position = "absolute";
|
279
|
+
if (WebSocket.__isFlashLite()) {
|
280
|
+
container.style.left = "0px";
|
281
|
+
container.style.top = "0px";
|
282
|
+
} else {
|
283
|
+
container.style.left = "-100px";
|
284
|
+
container.style.top = "-100px";
|
285
|
+
}
|
286
|
+
var holder = document.createElement("div");
|
287
|
+
holder.id = "webSocketFlash";
|
288
|
+
container.appendChild(holder);
|
289
|
+
document.body.appendChild(container);
|
290
|
+
// See this article for hasPriority:
|
291
|
+
// http://help.adobe.com/en_US/as3/mobile/WS4bebcd66a74275c36cfb8137124318eebc6-7ffd.html
|
292
|
+
swfobject.embedSWF(
|
293
|
+
WEB_SOCKET_SWF_LOCATION,
|
294
|
+
"webSocketFlash",
|
295
|
+
"1" /* width */,
|
296
|
+
"1" /* height */,
|
297
|
+
"10.0.0" /* SWF version */,
|
298
|
+
null,
|
299
|
+
null,
|
300
|
+
{hasPriority: true, swliveconnect : true, allowScriptAccess: "always"},
|
301
|
+
null,
|
302
|
+
function(e) {
|
303
|
+
if (!e.success) {
|
304
|
+
logger.error("[WebSocket] swfobject.embedSWF failed");
|
305
|
+
}
|
306
|
+
}
|
307
|
+
);
|
308
|
+
|
309
|
+
};
|
310
|
+
|
311
|
+
/**
|
312
|
+
* Called by Flash to notify JS that it's fully loaded and ready
|
313
|
+
* for communication.
|
314
|
+
*/
|
315
|
+
WebSocket.__onFlashInitialized = function() {
|
316
|
+
// We need to set a timeout here to avoid round-trip calls
|
317
|
+
// to flash during the initialization process.
|
318
|
+
setTimeout(function() {
|
319
|
+
WebSocket.__flash = document.getElementById("webSocketFlash");
|
320
|
+
WebSocket.__flash.setCallerUrl(location.href);
|
321
|
+
WebSocket.__flash.setDebug(!!window.WEB_SOCKET_DEBUG);
|
322
|
+
for (var i = 0; i < WebSocket.__tasks.length; ++i) {
|
323
|
+
WebSocket.__tasks[i]();
|
324
|
+
}
|
325
|
+
WebSocket.__tasks = [];
|
326
|
+
}, 0);
|
327
|
+
};
|
328
|
+
|
329
|
+
/**
|
330
|
+
* Called by Flash to notify WebSockets events are fired.
|
331
|
+
*/
|
332
|
+
WebSocket.__onFlashEvent = function() {
|
333
|
+
setTimeout(function() {
|
334
|
+
try {
|
335
|
+
// Gets events using receiveEvents() instead of getting it from event object
|
336
|
+
// of Flash event. This is to make sure to keep message order.
|
337
|
+
// It seems sometimes Flash events don't arrive in the same order as they are sent.
|
338
|
+
var events = WebSocket.__flash.receiveEvents();
|
339
|
+
for (var i = 0; i < events.length; ++i) {
|
340
|
+
WebSocket.__instances[events[i].webSocketId].__handleEvent(events[i]);
|
341
|
+
}
|
342
|
+
} catch (e) {
|
343
|
+
logger.error(e);
|
344
|
+
}
|
345
|
+
}, 0);
|
346
|
+
return true;
|
347
|
+
};
|
348
|
+
|
349
|
+
// Called by Flash.
|
350
|
+
WebSocket.__log = function(message) {
|
351
|
+
logger.log(decodeURIComponent(message));
|
352
|
+
};
|
353
|
+
|
354
|
+
// Called by Flash.
|
355
|
+
WebSocket.__error = function(message) {
|
356
|
+
logger.error(decodeURIComponent(message));
|
357
|
+
};
|
358
|
+
|
359
|
+
WebSocket.__addTask = function(task) {
|
360
|
+
if (WebSocket.__flash) {
|
361
|
+
task();
|
362
|
+
} else {
|
363
|
+
WebSocket.__tasks.push(task);
|
364
|
+
}
|
365
|
+
};
|
366
|
+
|
367
|
+
/**
|
368
|
+
* Test if the browser is running flash lite.
|
369
|
+
* @return {boolean} True if flash lite is running, false otherwise.
|
370
|
+
*/
|
371
|
+
WebSocket.__isFlashLite = function() {
|
372
|
+
if (!window.navigator || !window.navigator.mimeTypes) {
|
373
|
+
return false;
|
374
|
+
}
|
375
|
+
var mimeType = window.navigator.mimeTypes["application/x-shockwave-flash"];
|
376
|
+
if (!mimeType || !mimeType.enabledPlugin || !mimeType.enabledPlugin.filename) {
|
377
|
+
return false;
|
378
|
+
}
|
379
|
+
return mimeType.enabledPlugin.filename.match(/flashlite/i) ? true : false;
|
380
|
+
};
|
381
|
+
|
382
|
+
if (!window.WEB_SOCKET_DISABLE_AUTO_INITIALIZATION) {
|
383
|
+
// NOTE:
|
384
|
+
// This fires immediately if web_socket.js is dynamically loaded after
|
385
|
+
// the document is loaded.
|
386
|
+
swfobject.addDomLoadEvent(function() {
|
387
|
+
WebSocket.__initialize();
|
388
|
+
});
|
389
|
+
}
|
390
|
+
|
391
|
+
})();
|
@@ -0,0 +1,29 @@
|
|
1
|
+
#= require ./orator/header
|
2
|
+
#= require_self
|
3
|
+
#= require_tree ./orator/
|
4
|
+
|
5
|
+
window.WEB_SOCKET_SWF_LOCATION = "/assets/WebSocketMain.swf"
|
6
|
+
|
7
|
+
Orator.setup = (address, routes = ()->)->
|
8
|
+
libs = Orator.Libraries
|
9
|
+
|
10
|
+
event_handler = new libs.EventHandler({})
|
11
|
+
console.log(event_handler)
|
12
|
+
socket = new WebSocket(address)
|
13
|
+
event_handler.context.server = new libs.MiddleGround(socket)
|
14
|
+
routes.apply(event_handler, event_handler)
|
15
|
+
|
16
|
+
socket.onopen = (args...)->
|
17
|
+
event_handler.trigger('socket.open', args...)
|
18
|
+
|
19
|
+
socket.onmessage = (event)->
|
20
|
+
json_data = $.parseJSON(event.data)
|
21
|
+
event_handler.trigger(json_data.event, json_data, event)
|
22
|
+
|
23
|
+
socket.onclose = (args...)->
|
24
|
+
event_handler.trigger('socket.close', args...)
|
25
|
+
|
26
|
+
socket.onerror = (args...)->
|
27
|
+
event_handler.trigger('socket.error', args...)
|
28
|
+
|
29
|
+
Orator.event_handler = event_handler
|
Binary file
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: orator
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.3.
|
4
|
+
version: 0.3.3
|
5
5
|
prerelease:
|
6
6
|
platform: ruby
|
7
7
|
authors:
|
@@ -105,6 +105,17 @@ files:
|
|
105
105
|
- spec/client_spec.rb
|
106
106
|
- spec/base_orator_spec.rb
|
107
107
|
- spec/spec_helper.rb
|
108
|
+
- vendor/assets/swfs/WebSocketMain.swf
|
109
|
+
- vendor/assets/javascripts/orator.coffee
|
110
|
+
- vendor/assets/javascripts/orator/web_socket.js
|
111
|
+
- vendor/assets/javascripts/orator/stringify.jquery.js
|
112
|
+
- vendor/assets/javascripts/orator/header.coffee
|
113
|
+
- vendor/assets/javascripts/orator/event_handler.coffee
|
114
|
+
- vendor/assets/javascripts/orator/chat_logger.coffee
|
115
|
+
- vendor/assets/javascripts/orator/swfobject.js
|
116
|
+
- vendor/assets/javascripts/orator/command_handler.coffee
|
117
|
+
- vendor/assets/javascripts/orator/strbuf.coffee
|
118
|
+
- vendor/assets/javascripts/orator/middle_ground.coffee
|
108
119
|
- README.md
|
109
120
|
- Rakefile
|
110
121
|
- LICENSE
|