module JSON %x{ var $parse = JSON.parse, $hasOwn = Opal.hasOwnProperty; function to_opal(value, options) { switch (typeof value) { case 'string': return value; case 'number': return value; case 'boolean': return !!value; case 'null': return nil; case 'object': if (!value) return nil; if (value.$$is_array) { var arr = #{`options.array_class`.new}; for (var i = 0, ii = value.length; i < ii; i++) { #{`arr`.push(`to_opal(value[i], options)`)}; } return arr; } else { var hash = #{`options.object_class`.new}; for (var k in value) { if ($hasOwn.call(value, k)) { #{`hash`[`k`] = `to_opal(value[k], options)`}; } } var klass; if ((klass = #{`hash`[JSON.create_id]}) != nil) { klass = Opal.get(klass); return #{`klass`.json_create(`hash`)}; } else { return hash; } } } }; } class << self attr_accessor :create_id end self.create_id = :json_class def self.[](value, options = {}) if String === value parse(value, options) else generate(value, options) end end def self.parse(source, options = {}) from_object(`$parse(source)`, options) end def self.parse!(source, options = {}) parse(source, options) end # Raw js object => opal object def self.from_object(js_object, options = {}) options[:object_class] ||= Hash options[:array_class] ||= Array `to_opal(js_object, options.smap)` end def self.generate(obj, options = {}) obj.to_json(options) end def self.dump(obj, io = nil, limit = nil) string = generate(obj) if io io = io.to_io if io.responds_to? :to_io io.write string io else string end end end class Object def to_json to_s.to_json end end # BUG: Enumerable must come before Array, otherwise it overrides #to_json # this is due to how modules are implemented. module Enumerable def to_json to_a.to_json end end class Array def to_json %x{ var result = []; for (var i = 0, length = #{self}.length; i < length; i++) { result.push(#{`self[i]`.to_json}); } return '[' + result.join(', ') + ']'; } end end class Boolean def to_json `(self == true) ? 'true' : 'false'` end end class Hash def to_json %x{ var result = []; for (var i = 0, keys = self.keys, length = keys.length, key, value; i < length; i++) { key = keys[i]; if (key.$$is_string) { value = self.smap[key]; } else { value = key.value; key = key.key; } result.push(#{`key`.to_s.to_json} + ':' + #{`value`.to_json}); } return '{' + result.join(', ') + '}'; } end end class NilClass def to_json 'null' end end class Numeric def to_json `self.toString()` end end class String alias to_json inspect end class Time def to_json strftime("%FT%T%z").to_json end end class Date def to_json to_s.to_json end def as_json to_s end end