2018-03-15 11:00:24 +01:00
|
|
|
# Copyright 2017-2018 Xuan Sang LE <xsang.le AT gmail DOT com>
|
|
|
|
|
|
|
|
# AnTOS Web desktop is is licensed under the GNU General Public
|
|
|
|
# License v3.0, see the LICENCE file for more information
|
|
|
|
|
|
|
|
# This program is free software: you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public License as
|
|
|
|
# published by the Free Software Foundation, either version 3 of
|
|
|
|
# the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
# General Public License for more details.
|
|
|
|
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
#along with this program. If not, see https://www.gnu.org/licenses/.
|
2018-03-10 20:42:09 +01:00
|
|
|
class FormatedString
|
|
|
|
constructor: (@fs, args) ->
|
|
|
|
@values = []
|
|
|
|
return unless args
|
|
|
|
@values[i] = args[i] for i in [0..args.length - 1]
|
2018-03-10 22:46:28 +01:00
|
|
|
toString: () ->
|
|
|
|
@__()
|
2018-03-10 20:42:09 +01:00
|
|
|
__: () ->
|
2020-05-17 17:05:12 +02:00
|
|
|
return @fs.l().replace /{(\d+)}/g, (match, number) =>
|
|
|
|
return if typeof @values[number] != 'undefined' then @values[number].__() else match
|
2018-03-10 20:42:09 +01:00
|
|
|
hash: () ->
|
|
|
|
@__().hash()
|
|
|
|
|
2020-05-14 23:08:57 +02:00
|
|
|
match: (t) ->
|
|
|
|
@__().match t
|
|
|
|
|
2018-03-10 20:42:09 +01:00
|
|
|
asBase64: () ->
|
|
|
|
@__().asBase64()
|
|
|
|
|
|
|
|
unescape: () ->
|
|
|
|
@__().unescape()
|
|
|
|
|
|
|
|
asUint8Array: () ->
|
|
|
|
@__().asUint8Array()
|
|
|
|
|
|
|
|
format: () ->
|
|
|
|
args = arguments
|
|
|
|
@values[i] = args[i] for i in [0..args.length - 1]
|
2018-03-10 22:46:28 +01:00
|
|
|
|
2018-03-21 11:21:33 +01:00
|
|
|
class Version
|
|
|
|
constructor:(@string) ->
|
|
|
|
arr = @string.split "-"
|
|
|
|
br =
|
|
|
|
"r": 3,
|
|
|
|
"b": 2,
|
|
|
|
"a": 1
|
|
|
|
@branch = 3
|
|
|
|
@branch = br[arr[1]] if arr.length is 2 and br[arr[1]]
|
|
|
|
mt = arr[0].match /\d+/g
|
|
|
|
throw new Error __("Version string is in invalid format: {0}", @string) if not mt
|
|
|
|
@major = 0
|
|
|
|
@minor = 0
|
|
|
|
@patch = 0
|
|
|
|
@major = Number mt[0] if mt.length >= 1
|
|
|
|
@minor = Number mt[1] if mt.length >= 2
|
|
|
|
@patch = Number mt[2] if mt.length >= 3
|
|
|
|
|
|
|
|
# this function return
|
|
|
|
# 0 if the version is unchanged
|
|
|
|
# 1 if the current version is newer
|
|
|
|
# -1 if the current version is older
|
|
|
|
compare: (o) ->
|
|
|
|
other = o.__v()
|
|
|
|
return 1 if @branch > other.branch
|
|
|
|
return -1 if @branch < other.branch
|
|
|
|
return 0 if @major is other.major and @minor is other.minor and @patch is other.patch
|
|
|
|
return 1 if @major > other.major
|
|
|
|
return -1 if @major < other.major
|
|
|
|
return 1 if @minor > other.minor
|
|
|
|
return -1 if @minor < other.minor
|
|
|
|
return 1 if @patch > other.patch
|
|
|
|
return -1
|
|
|
|
nt: (o) ->
|
|
|
|
return (@compare o) is 1
|
|
|
|
ot: (o) ->
|
|
|
|
return (@compare o) is -1
|
|
|
|
__v: () -> @
|
|
|
|
toString: () -> @string
|
|
|
|
|
2018-03-10 22:46:28 +01:00
|
|
|
Object.defineProperty Object.prototype, '__',
|
|
|
|
value: () ->
|
|
|
|
return @toString()
|
|
|
|
enumerable: false
|
|
|
|
writable: true
|
2018-03-10 20:42:09 +01:00
|
|
|
|
2018-02-18 21:11:49 +01:00
|
|
|
String.prototype.hash = () ->
|
|
|
|
hash = 5381
|
|
|
|
i = this.length
|
|
|
|
hash = (hash * 33) ^ this.charCodeAt(--i) while i
|
|
|
|
hash >>> 0
|
2018-03-21 11:21:33 +01:00
|
|
|
String.prototype.__v = () ->
|
|
|
|
return new Version @
|
2018-02-18 21:11:49 +01:00
|
|
|
String.prototype.asBase64 = () ->
|
|
|
|
tmp = encodeURIComponent this
|
|
|
|
return btoa ( tmp.replace /%([0-9A-F]{2})/g, (match, p1) ->
|
|
|
|
return String.fromCharCode (parseInt p1, 16)
|
|
|
|
)
|
|
|
|
String.prototype.unescape = () ->
|
|
|
|
d = @
|
|
|
|
d = d.replace /\\\\/g, "\\"
|
|
|
|
d = d.replace /\\"/g, '"'
|
|
|
|
d = d.replace /\\n/g, "\n"
|
|
|
|
d = d.replace /\\t/g, "\t"
|
|
|
|
d = d.replace /\\b/g, "\b"
|
|
|
|
d = d.replace /\\f/g, "\f"
|
|
|
|
d = d.replace /\\r/g, "\r"
|
|
|
|
d
|
2018-03-10 20:42:09 +01:00
|
|
|
String.prototype.asUint8Array = () ->
|
2018-03-05 00:20:25 +01:00
|
|
|
bytes = []
|
|
|
|
for i in [0..(@length - 1)]
|
|
|
|
bytes.push @charCodeAt i
|
|
|
|
bytes = new Uint8Array(bytes)
|
|
|
|
return bytes
|
2018-02-18 21:11:49 +01:00
|
|
|
|
2018-03-09 19:54:33 +01:00
|
|
|
if not String.prototype.format
|
|
|
|
String.prototype.format = () ->
|
|
|
|
args = arguments
|
2018-03-10 23:24:53 +01:00
|
|
|
return @replace /{(\d+)}/g, (match, number) ->
|
2018-03-20 13:07:23 +01:00
|
|
|
return if typeof args[number] != 'undefined' then args[number].__() else match
|
2018-03-10 20:42:09 +01:00
|
|
|
|
|
|
|
String.prototype.f = () ->
|
|
|
|
args = arguments
|
|
|
|
return new FormatedString(@, args)
|
2018-03-10 12:22:01 +01:00
|
|
|
|
|
|
|
String.prototype.__ = () ->
|
|
|
|
match = @match(/^__\((.*)\)$/)
|
2018-03-10 22:46:28 +01:00
|
|
|
return match[1].l() if match
|
2018-03-10 12:22:01 +01:00
|
|
|
return @
|
2018-03-10 22:46:28 +01:00
|
|
|
String.prototype.l = () ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.lang[@] = @ unless Ant.OS.API.lang[@]
|
|
|
|
return Ant.OS.API.lang[@]
|
2018-03-09 19:54:33 +01:00
|
|
|
# language directive
|
2018-03-10 20:42:09 +01:00
|
|
|
|
2018-03-09 19:54:33 +01:00
|
|
|
this.__ = () ->
|
|
|
|
args = arguments
|
|
|
|
return "Undefined" unless args.length > 0
|
|
|
|
d = args[0]
|
2018-03-10 22:46:28 +01:00
|
|
|
d.l()
|
|
|
|
return new FormatedString d, (args[i] for i in [1 .. args.length - 1])
|
2018-03-09 19:54:33 +01:00
|
|
|
|
2018-02-18 21:11:49 +01:00
|
|
|
Date.prototype.toString = () ->
|
|
|
|
dd = @getDate()
|
|
|
|
mm = @getMonth() + 1
|
|
|
|
yyyy = @getFullYear()
|
|
|
|
hh = @getHours()
|
|
|
|
mi = @getMinutes()
|
|
|
|
se = @getSeconds()
|
|
|
|
|
|
|
|
dd = "0#{dd}" if dd < 10
|
|
|
|
mm = "0#{mm}" if mm < 10
|
|
|
|
hh = "0#{hh}" if hh < 10
|
|
|
|
mi = "0#{mi}" if mi < 10
|
|
|
|
se = "0#{se}" if se < 10
|
|
|
|
return "#{dd}/#{mm}/#{yyyy} #{hh}:#{mi}:#{se}"
|
|
|
|
|
|
|
|
Date.prototype.timestamp = () ->
|
|
|
|
return @getTime() / 1000 | 0
|
|
|
|
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API =
|
|
|
|
# the handle object could be a any remote or local handle to
|
2017-08-07 00:49:24 +02:00
|
|
|
# fetch user data, used by the API to make requests
|
2020-02-20 18:28:03 +01:00
|
|
|
# handles are defined in /src/handles
|
|
|
|
handle: {}
|
2018-02-16 18:38:14 +01:00
|
|
|
shared: {} # shared libraries
|
2020-05-18 18:53:59 +02:00
|
|
|
searchHandle: {}
|
|
|
|
lang: {}
|
2017-08-07 00:49:24 +02:00
|
|
|
#request a user data
|
2018-03-01 13:56:42 +01:00
|
|
|
mid: () ->
|
2020-02-20 18:28:03 +01:00
|
|
|
return Ant.OS.announcer.getMID()
|
2020-05-10 21:39:42 +02:00
|
|
|
post: (p, d) ->
|
|
|
|
new Promise (resolve, reject) ->
|
|
|
|
q = Ant.OS.announcer.getMID()
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.loading q, p
|
2018-01-25 19:15:41 +01:00
|
|
|
$.ajax {
|
|
|
|
type: 'POST',
|
2020-05-10 21:39:42 +02:00
|
|
|
url: p,
|
|
|
|
contentType: 'application/json',
|
2020-05-18 18:53:59 +02:00
|
|
|
data: JSON.stringify(d,
|
|
|
|
(k, v) ->
|
|
|
|
return undefined if k is "domel"
|
|
|
|
return v
|
|
|
|
, 4),
|
2020-05-10 21:39:42 +02:00
|
|
|
dataType: 'json',
|
|
|
|
success: null
|
2018-01-25 19:15:41 +01:00
|
|
|
}
|
|
|
|
.done (data) ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.loaded q, p, "OK"
|
2020-05-10 21:39:42 +02:00
|
|
|
resolve(data)
|
2020-05-18 18:53:59 +02:00
|
|
|
.fail (j, s, e) ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.loaded q, p, "FAIL"
|
2020-05-18 18:53:59 +02:00
|
|
|
if e
|
|
|
|
reject e
|
|
|
|
else
|
|
|
|
reject(Ant.OS.API.throwe s)
|
2020-05-10 21:39:42 +02:00
|
|
|
|
|
|
|
blob: (p) ->
|
|
|
|
new Promise (resolve, reject) ->
|
|
|
|
q = Ant.OS.announcer.getMID()
|
|
|
|
r = new XMLHttpRequest()
|
|
|
|
r.open "GET", p, true
|
|
|
|
r.responseType = "arraybuffer"
|
|
|
|
r.onload = (e) ->
|
|
|
|
if @status is 200 and @readyState is 4
|
|
|
|
Ant.OS.API.loaded q, p, "OK"
|
2020-05-19 01:03:49 +02:00
|
|
|
resolve @response
|
2020-05-10 21:39:42 +02:00
|
|
|
else
|
|
|
|
Ant.OS.API.loaded q, p, "FAIL"
|
2020-05-19 01:03:49 +02:00
|
|
|
reject Ant.OS.API.throwe __("Unable to get blob: {0}", p)
|
2020-05-10 21:39:42 +02:00
|
|
|
Ant.OS.API.loading q, p
|
|
|
|
r.send()
|
2020-05-19 01:03:49 +02:00
|
|
|
|
2020-05-10 21:39:42 +02:00
|
|
|
upload: (p, d) ->
|
|
|
|
new Promise (resolve, reject) ->
|
|
|
|
q = Ant.OS.announcer.getMID()
|
|
|
|
#insert a temporal file selector
|
|
|
|
o = ($ '<input>').attr('type', 'file').css("display", "none")
|
|
|
|
o.change () ->
|
|
|
|
Ant.OS.API.loading q, p
|
|
|
|
formd = new FormData()
|
|
|
|
formd.append 'path', d
|
|
|
|
# TODO: only one file is selected at this time
|
|
|
|
formd.append 'upload', o[0].files[0]
|
|
|
|
|
|
|
|
$.ajax {
|
|
|
|
url: p,
|
|
|
|
data: formd,
|
|
|
|
type: 'POST',
|
|
|
|
contentType: false,
|
|
|
|
processData: false,
|
|
|
|
}
|
|
|
|
.done (data) ->
|
|
|
|
Ant.OS.API.loaded q, p, "OK"
|
|
|
|
resolve(data)
|
|
|
|
o.remove()
|
2020-05-18 18:53:59 +02:00
|
|
|
.fail (j, s, e) ->
|
2020-05-10 21:39:42 +02:00
|
|
|
Ant.OS.API.loaded q, p, "FAIL"
|
2020-05-18 18:53:59 +02:00
|
|
|
if e
|
|
|
|
reject e
|
|
|
|
else
|
|
|
|
reject(Ant.OS.API.throwe s)
|
2020-05-10 21:39:42 +02:00
|
|
|
o.remove()
|
|
|
|
o.click()
|
2018-01-25 19:15:41 +01:00
|
|
|
|
|
|
|
saveblob: (name, b) ->
|
|
|
|
url = window.URL.createObjectURL b
|
|
|
|
o = ($ '<a>')
|
|
|
|
.attr("href", url)
|
|
|
|
.attr("download", name)
|
|
|
|
.css("display", "none")
|
|
|
|
.appendTo("body")
|
|
|
|
o[0].click()
|
|
|
|
window.URL.revokeObjectURL(url)
|
|
|
|
o.remove()
|
2017-08-07 00:49:24 +02:00
|
|
|
|
2017-08-26 16:50:13 +02:00
|
|
|
loading: (q, p) ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.announcer.trigger "loading", { id: q, data: { m: "#{p}", s: true }, name: "OS" }
|
2020-05-10 21:39:42 +02:00
|
|
|
|
2017-08-26 16:50:13 +02:00
|
|
|
loaded: (q, p, m ) ->
|
2020-05-10 21:39:42 +02:00
|
|
|
Ant.OS.announcer.trigger "loaded", {
|
|
|
|
id: q, data: { m: "#{m}: #{p}", s: false }, name: "OS" }
|
|
|
|
|
|
|
|
get: (p, t) ->
|
|
|
|
new Promise (resolve, reject) ->
|
|
|
|
conf =
|
|
|
|
type: 'GET',
|
|
|
|
url: p
|
|
|
|
conf.dataType = t if t
|
|
|
|
q = Ant.OS.announcer.getMID()
|
|
|
|
Ant.OS.API.loading q, p
|
|
|
|
$.ajax conf
|
|
|
|
.done (data) ->
|
|
|
|
Ant.OS.API.loaded q, p, "OK"
|
|
|
|
resolve(data)
|
2020-05-18 18:53:59 +02:00
|
|
|
.fail (j, s, e) ->
|
2020-05-10 21:39:42 +02:00
|
|
|
Ant.OS.API.loaded q, p, "FAIL"
|
2020-05-18 18:53:59 +02:00
|
|
|
if e
|
|
|
|
reject e
|
|
|
|
else
|
|
|
|
reject(Ant.OS.API.throwe s)
|
2020-05-10 21:39:42 +02:00
|
|
|
|
|
|
|
script: (p) ->
|
2020-05-18 18:53:59 +02:00
|
|
|
Ant.OS.API.get p, "script"
|
2020-05-10 21:39:42 +02:00
|
|
|
|
|
|
|
resource: (r) ->
|
2017-08-27 23:40:02 +02:00
|
|
|
path = "resources/#{r}"
|
2020-05-10 21:39:42 +02:00
|
|
|
Ant.OS.API.get path
|
2018-01-29 19:16:29 +01:00
|
|
|
|
2018-02-27 16:40:36 +01:00
|
|
|
libready: (l) ->
|
2020-02-20 18:28:03 +01:00
|
|
|
return Ant.OS.API.shared[l] || false
|
2018-02-27 16:40:36 +01:00
|
|
|
|
2020-05-12 21:08:39 +02:00
|
|
|
requires: (l) ->
|
|
|
|
new Promise (resolve, reject) ->
|
|
|
|
if not Ant.OS.API.shared[l]
|
2020-05-15 20:55:13 +02:00
|
|
|
libfp = l.asFileHandle()
|
|
|
|
switch libfp.ext
|
|
|
|
when "css"
|
|
|
|
libfp.onready()
|
|
|
|
.then () ->
|
|
|
|
$('<link>', {
|
|
|
|
rel: 'stylesheet',
|
|
|
|
type: 'text/css',
|
|
|
|
'href': "#{libfp.getlink()}"
|
|
|
|
})
|
|
|
|
.appendTo 'head'
|
|
|
|
Ant.OS.API.shared[l] = true
|
2020-05-17 01:40:18 +02:00
|
|
|
console.log "Loaded :", l
|
2020-05-15 20:55:13 +02:00
|
|
|
Ant.OS.announcer.trigger "sharedlibraryloaded", l
|
2020-05-17 01:40:18 +02:00
|
|
|
resolve undefined
|
2020-05-15 20:55:13 +02:00
|
|
|
.catch (e) -> reject e
|
|
|
|
when "js"
|
|
|
|
Ant.OS.API.script libfp.getlink()
|
2020-05-17 01:40:18 +02:00
|
|
|
.then (data) ->
|
2020-05-15 20:55:13 +02:00
|
|
|
Ant.OS.API.shared[l] = true
|
2020-05-17 01:40:18 +02:00
|
|
|
console.log "Loaded :", l
|
2020-05-15 20:55:13 +02:00
|
|
|
Ant.OS.announcer.trigger "sharedlibraryloaded", l
|
2020-05-17 01:40:18 +02:00
|
|
|
resolve(data)
|
2020-05-12 21:08:39 +02:00
|
|
|
.catch (e) ->
|
2020-05-15 20:55:13 +02:00
|
|
|
reject e
|
|
|
|
else
|
2020-05-17 17:05:12 +02:00
|
|
|
reject Ant.OS.API.throwe __("Invalid library: {0}", l)
|
2020-05-12 21:08:39 +02:00
|
|
|
else
|
|
|
|
console.log l, "Library exist, no need to load"
|
|
|
|
Ant.OS.announcer.trigger "sharedlibraryloaded", l
|
|
|
|
resolve()
|
2018-02-16 18:38:14 +01:00
|
|
|
|
2020-05-12 21:08:39 +02:00
|
|
|
require: (libs) ->
|
|
|
|
new Promise (resolve, reject) ->
|
|
|
|
return resolve() unless libs.length > 0
|
|
|
|
Ant.OS.announcer.observable.one "sharedlibraryloaded", (l) ->
|
|
|
|
libs.splice 0, 1
|
|
|
|
Ant.OS.API.require libs
|
|
|
|
.catch (e) -> reject e
|
|
|
|
.then (r) -> resolve(r)
|
|
|
|
Ant.OS.API.requires libs[0]
|
|
|
|
.catch (e) -> reject e
|
2020-05-10 21:39:42 +02:00
|
|
|
|
2018-01-29 19:16:29 +01:00
|
|
|
packages:
|
2020-05-10 21:39:42 +02:00
|
|
|
fetch: () ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.handle.packages {
|
|
|
|
command: "list", args: { paths: (v for k, v of Ant.OS.setting.system.pkgpaths) }
|
2020-05-10 21:39:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
cache: () ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.handle.packages {
|
|
|
|
command: "cache", args: { paths: (v for k, v of Ant.OS.setting.system.pkgpaths) }
|
2020-05-10 21:39:42 +02:00
|
|
|
}
|
|
|
|
|
2018-03-12 18:55:40 +01:00
|
|
|
setting: (f) ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.handle.setting f
|
2020-05-10 21:39:42 +02:00
|
|
|
|
2019-08-30 13:27:17 +02:00
|
|
|
apigateway: (d, ws, c) ->
|
2020-02-20 18:28:03 +01:00
|
|
|
return Ant.OS.API.handle.apigateway d, ws, c
|
2020-05-10 21:39:42 +02:00
|
|
|
|
2018-03-03 13:44:52 +01:00
|
|
|
search: (text) ->
|
|
|
|
r = []
|
|
|
|
|
2020-02-20 18:28:03 +01:00
|
|
|
for k, v of Ant.OS.API.searchHandle
|
|
|
|
ret = Ant.OS.API.searchHandle[k](text)
|
2018-03-03 13:44:52 +01:00
|
|
|
if ret.length > 0
|
|
|
|
ret.unshift { text: k, class: "search-header", dataid: "header" }
|
|
|
|
r = r.concat ret
|
|
|
|
return r
|
|
|
|
|
|
|
|
onsearch: (name, fn) ->
|
2020-02-20 18:28:03 +01:00
|
|
|
Ant.OS.API.searchHandle[name] = fn unless Ant.OS.API.searchHandle[name]
|
2018-03-03 13:44:52 +01:00
|
|
|
|
2020-05-10 21:39:42 +02:00
|
|
|
setLocale: (name) ->
|
|
|
|
new Promise (resolve, reject) ->
|
|
|
|
path = "resources/languages/#{name}.json"
|
|
|
|
Ant.OS.API.get(path, "json")
|
|
|
|
.then (d) ->
|
|
|
|
Ant.OS.setting.system.locale = name
|
|
|
|
Ant.OS.API.lang = d
|
|
|
|
Ant.OS.announcer.trigger "systemlocalechange", name
|
|
|
|
resolve d
|
|
|
|
.catch (e) ->
|
|
|
|
reject e
|
2018-03-09 19:54:33 +01:00
|
|
|
|
2018-01-24 01:03:14 +01:00
|
|
|
throwe: (n) ->
|
|
|
|
err = undefined
|
|
|
|
try
|
|
|
|
throw new Error(n)
|
|
|
|
catch e
|
|
|
|
err = e
|
|
|
|
return "" if not err
|
2018-03-12 22:47:23 +01:00
|
|
|
return err
|
|
|
|
# utilities functioncs
|
|
|
|
switcher: () ->
|
|
|
|
o = {}
|
|
|
|
p = {}
|
|
|
|
p[arguments[i]] = false for i in [0..arguments.length - 1 ]
|
|
|
|
Object.defineProperty o, "__p", {
|
|
|
|
enumerable: false,
|
|
|
|
value: p
|
|
|
|
}
|
|
|
|
fn = (o, v) ->
|
|
|
|
Object.defineProperty o, v, {
|
|
|
|
enumerable: true,
|
|
|
|
set: (value) ->
|
2020-05-18 18:53:59 +02:00
|
|
|
for k, l of @__p
|
2018-03-12 22:47:23 +01:00
|
|
|
@__p[k] = false
|
|
|
|
o.__p[v] = value
|
|
|
|
, get: () ->
|
|
|
|
return o.__p[v]
|
|
|
|
}
|
|
|
|
for k, v of o.__p
|
|
|
|
fn o, k
|
|
|
|
Object.defineProperty o, "selected", {
|
|
|
|
configurable: true,
|
|
|
|
enumerable: false,
|
|
|
|
get: () ->
|
2020-05-18 18:53:59 +02:00
|
|
|
for k, v of o.__p
|
2018-03-12 22:47:23 +01:00
|
|
|
return k if v
|
|
|
|
}
|
|
|
|
return o
|