adapt the system with the new api

This commit is contained in:
lxsang 2020-05-10 21:39:42 +02:00
parent ae54c9a10c
commit f5e5d8501b
24 changed files with 1120 additions and 952 deletions

View File

@ -44,7 +44,7 @@ class BaseApplication extends this.OS.GUI.BaseModel
me.appmenu.set "items", [] me.appmenu.set "items", []
me.dialog.hide() if me.dialog me.dialog.hide() if me.dialog
@on "menuselect", (d) -> @on "menuselect", (d) ->
switch d.e.item.data.dataid switch d.data.item.get("data").dataid
when "#{me.name}-about" then me.openDialog "AboutDialog", ()-> when "#{me.name}-about" then me.openDialog "AboutDialog", ()->
when "#{me.name}-exit" then me.trigger "exit" when "#{me.name}-exit" then me.trigger "exit"
@on "apptitlechange", () -> me.sysdock.update() @on "apptitlechange", () -> me.sysdock.update()

View File

@ -50,227 +50,297 @@ class BaseDialog extends SubWindow
@parent.dialog = undefined if @parent @parent.dialog = undefined if @parent
this.OS.GUI.BaseDialog = BaseDialog this.OS.GUI.BaseDialog = BaseDialog
###
this dialog rende a tag as main content
and a list of buttons, the behaviour of
the button is specified by user. The conf
object is in the follow form
{
tag: <tag_name>,
buttons:[
{
label: 'buton label',
onclick: function(d){...}
}, ...
]
}
###
class BasicDialog extends BaseDialog class BasicDialog extends BaseDialog
constructor: ( name, @conf, @title) -> constructor: ( name, target) ->
super name super name
if typeof target is "string"
Ant.OS.GUI.htmlToScheme target, @, @host
else # a file handle
@render target.path
init: () -> init: () ->
@title = @name if not @title @scheme.set "apptitle", @data.title if @data and @data.title
html = "<afx-app-window data-id = '#{@name}' width='#{@conf.width}' height='#{@conf.height}'>"
html += "<afx-hbox><div data-width='7'></div><afx-vbox><div data-height='5'></div>"
html += "<#{v.tag} #{v.att} data-id = 'content#{k}'></#{v.tag}>" for k,v of @conf.tags
html += "<div data-height = '35' style=' text-align:right;padding-top:3px;'>"
html += "<afx-button data-id = 'bt#{k}' text = '#{v.label}' style='margin-left:5px;'></afx-button>" for k,v of @conf.buttons
html += "</div><div data-height='5'></div></afx-vbox><div data-width='7'></div></afx-hbox></afx-app-window>"
#render the html
Ant.OS.GUI.htmlToScheme html, @, @host
main: () ->
@scheme.set "apptitle", @title
@scheme.set "minimizable", false
@scheme.set "resizable", @conf.resizable if @conf.resizable isnt undefined
me = @
f = (_v) -> () -> _v.onclick me
# bind action to button
( (me.find "bt#{k}").set "onbtclick", f(v) ) for k, v of @conf.buttons
@conf.filldata @ if @conf.filldata
@conf.xtra @ if @conf.xtra
this.OS.GUI.BasicDialog = BasicDialog this.OS.GUI.BasicDialog = BasicDialog
class PromptDialog extends BasicDialog class PromptDialog extends BasicDialog
constructor: () -> constructor: () ->
super "PromptDialog", { super "PromptDialog", PromptDialog.scheme
tags: [
{ tag: "afx-label" },
{ tag: "input", att: "type = 'text' data-height='25'" }
],
width: 200,
height: 120,
resizable: false,
buttons: [
{
label: "__(Ok)",
onclick: (d) ->
txt = (d.find "content1").value
return d.quit() if txt is ""
d.handle txt if d.handle
d.quit()
},
{
label: "__(Cancel)",
onclick: (d) -> d.quit()
}
],
filldata: (d) ->
return unless d.data
(d.find "content0").set "text", d.data.label
(d.find "content1").value = d.data.value if d.data.value
$(d.find "content1").attr("type", d.data.type) if d.data.type
xtra: (d) ->
$( d.find "content1" ).keyup (e) ->
(d.find "bt0").trigger() if e.which is 13
}
init: () ->
super.init()
me = @
@find("lbl").set "text", @data.label if @data and @data.label
$(@find "txtInput").val @data.value if @data and @data.value
(@find "btnOk").set "onbtclick", (e) ->
me.handle($(me.find "txtInput").val()) if me.handle
me.quit()
(@find "btnCancel").set "onbtclick", (e) ->
me.quit()
PromptDialog.scheme = """
<afx-app-window width='200' height='150' apptitle = "Prompt">
<afx-vbox>
<afx-hbox>
<div data-width = "10" />
<afx-vbox>
<div data-height="10" />
<afx-label data-id = "lbl" />
<input type = "text" data-id= "txtInput" />
<div data-height="10" />
<afx-hbox data-height="30">
<div />
<afx-button data-id = "btnOk" text = "__(Ok)" data-width = "40" />
<afx-button data-id = "btnCancel" text = "__(Cancel)" data-width = "50" />
</afx-hbox>
</afx-vbox>
<div data-width = "10" />
</afx-hbox>
</afx-vbox>
</afx-app-window>
"""
this.OS.register "PromptDialog", PromptDialog this.OS.register "PromptDialog", PromptDialog
class CalendarDialog extends BasicDialog class CalendarDialog extends BasicDialog
constructor: () -> constructor: () ->
super "CalendarDialog", { super "CalendarDialog", CalendarDialog.scheme
tags: [{ tag: 'afx-calendar-view' }],
width: 300, init: () ->
height: 230, super.init()
resizable: false, me = @
buttons: [ (@find "btnOk").set "onbtclick", (e) ->
{ date = (me.find "cal").get "selectedDate"
label: "__(Ok)", return me.notify __("Please select a day") unless date
onclick: (d) -> me.handle(date) if me.handle
date = (d.find "content0").get "selectedDate" me.quit()
if date
d.handle date if d.handle (@find "btnCancel").set "onbtclick", (e) ->
d.quit() me.quit()
else
d.notify __("Please select a date") CalendarDialog.scheme = """
}, <afx-app-window width='300' height='230' apptitle = "Calendar" >
{ <afx-vbox>
label: "__(Cancel)", <afx-hbox>
onclick: (d) -> d.quit() <div data-width = "10" />
} <afx-vbox>
] <div data-height="10" />
} <afx-calendar-view data-id = "cal" />
<div data-height="10" />
<afx-hbox data-height="30">
<div />
<afx-button data-id = "btnOk" text = "__(Ok)" data-width = "40" />
<afx-button data-id = "btnCancel" text = "__(Cancel)" data-width = "50" />
</afx-hbox>
<div data-height="10" />
</afx-vbox>
<div data-width = "10" />
</afx-hbox>
</afx-vbox>
</afx-app-window>
"""
this.OS.register "CalendarDialog", CalendarDialog this.OS.register "CalendarDialog", CalendarDialog
class ColorPickerDialog extends BasicDialog class ColorPickerDialog extends BasicDialog
constructor: () -> constructor: () ->
super "ColorPickerDialog", { super "ColorPickerDialog", ColorPickerDialog.scheme
tags: [{ tag: 'afx-color-picker' }, {tag:'div', att: 'data-height="5"' }],
width: 313, init: () ->
height: 250, super.init()
resizable: false, me = @
buttons: [ (@find "btnOk").set "onbtclick", (e) ->
{ color = (me.find "cpicker").get "selectedColor"
label: "__(Ok)", return me.notify __("Please select color") unless color
onclick: (d) -> me.handle(color) if me.handle
c = (d.find "content0").get "selectedColor" me.quit()
if c
d.handle c if d.handle (@find "btnCancel").set "onbtclick", (e) ->
d.quit() me.quit()
else
d.notify "Please select a color" ColorPickerDialog.scheme = """
}, <afx-app-window width='320' height='250' apptitle = "Color picker" >
{ <afx-vbox>
label: "__(Cancel)", <afx-hbox>
onclick: (d) -> d.quit() <div data-width = "10" />
} <afx-vbox>
] <div data-height="10" />
} <afx-color-picker data-id = "cpicker" />
<div data-height="10" />
<afx-hbox data-height="30">
<div />
<afx-button data-id = "btnOk" text = "__(Ok)" data-width = "40" />
<afx-button data-id = "btnCancel" text = "__(Cancel)" data-width = "50" />
</afx-hbox>
<div data-height="10" />
</afx-vbox>
<div data-width = "10" />
</afx-hbox>
</afx-vbox>
</afx-app-window>
"""
this.OS.register "ColorPickerDialog", ColorPickerDialog this.OS.register "ColorPickerDialog", ColorPickerDialog
class InfoDialog extends BasicDialog class InfoDialog extends BasicDialog
constructor: () -> constructor: () ->
super "InfoDialog", { super "InfoDialog", InfoDialog.scheme
tags: [{ tag: 'afx-grid-view' }],
width: 250, init: () ->
height: 300, super.init()
resizable: true, me = @
buttons: [ { label: "__(Cancel)", onclick: (d) -> d.quit() } ], rows = []
filldata: (d) -> delete @data.title if @data and @data.title
return unless d.data rows.push [ { text: k }, { text: v } ] for k, v of @data
rows = [] (@find "grid").set "header", [ { text: __("Name"), width: 70 }, { text: __("Value") } ]
rows.push [ { value: k }, { value: v } ] for k, v of d.data (@find "grid").set "rows", rows
(d.find "content0").set "rows", rows (@find "btnCancel").set "onbtclick", (e) ->
} me.quit()
InfoDialog.scheme = """
<afx-app-window width='250' height='300' apptitle = "Info" >
<afx-vbox>
<afx-hbox>
<div data-width = "10" />
<afx-vbox>
<div data-height="10" />
<afx-grid-view data-id = "grid" />
<div data-height="10" />
<afx-hbox data-height="30">
<div />
<afx-button data-id = "btnCancel" text = "__(Cancel)" data-width = "50" />
</afx-hbox>
<div data-height="10" />
</afx-vbox>
<div data-width = "10" />
</afx-hbox>
</afx-vbox>
</afx-app-window>
"""
this.OS.register "InfoDialog", InfoDialog this.OS.register "InfoDialog", InfoDialog
class YesNoDialog extends BasicDialog class YesNoDialog extends BasicDialog
constructor: () -> constructor: () ->
super "YesNoDialog", { super "YesNoDialog", YesNoDialog.scheme
tags: [{ tag: "afx-label" }],
width: 300, init: () ->
height: 100, super.init()
resizable: true, me = @
buttons: [ @find("lbl").set "text", @data.label if @data and @data.label
{ (@find "btnYes").set "onbtclick", (e) ->
label: "__(Yes)", onclick: (d) -> me.handle(true) if me.handle
d.handle true if d.handle me.quit()
d.quit() (@find "btnNo").set "onbtclick", (e) ->
}, me.handle(false) if me.handle
{ me.quit()
label: "__(No)", onclick: (d) ->
d.handle false if d.handle YesNoDialog.scheme = """
d.quit() <afx-app-window width='200' height='150' apptitle = "Prompt">
} <afx-vbox>
], <afx-hbox>
filldata: (d) -> <div data-width = "10" />
return unless d.data <afx-vbox>
l = d.find "content0" <div data-height="10" />
for k, v of d.data <afx-label data-id = "lbl" />
l.set k, v <div data-height="10" />
} <afx-hbox data-height="30">
<div />
<afx-button data-id = "btnYes" text = "__(Yes)" data-width = "40" />
<afx-button data-id = "btnNo" text = "__(No)" data-width = "40" />
</afx-hbox>
</afx-vbox>
<div data-width = "10" />
</afx-hbox>
</afx-vbox>
</afx-app-window>
"""
this.OS.register "YesNoDialog", YesNoDialog this.OS.register "YesNoDialog", YesNoDialog
class SelectionDialog extends BasicDialog class SelectionDialog extends BasicDialog
constructor: () -> constructor: () ->
super "SelectionDialog", { super "SelectionDialog", SelectionDialog.scheme
tags: [{ tag: "afx-list-view" }],
width: 250,
height: 300,
resizable: false,
buttons: [
{
label: "__(Ok)", onclick: (d) ->
el = d.find "content0"
it = el.get "selected"
return unless it
d.handle it if d.handle
d.quit()
},
{ label: "__(Cancel)", onclick: (d) -> d.quit() }
],
filldata: (d) ->
return unless d.data
(d.find "content0").set "items", d.data
xtra: (d) ->
( d.find "content0" ).set "onlistdbclick", (e) ->
(d.find "bt0").trigger()
}
this.OS.register "SelectionDialog", SelectionDialog
class AboutDialog extends BaseDialog
constructor: () ->
super "AboutDialog"
init: () -> init: () ->
@render "os://resources/schemes/about.html" super.init()
me = @
(@find "list").set "data", @data.data if @data and @data.data
(@find "btnOk").set "onbtclick", (e) ->
data = (me.find "list").get "selectedItem"
return me.notify __("Please select an item") unless data
me.handle(data) if me.handle
me.quit()
main: () -> (@find "btnCancel").set "onbtclick", (e) ->
me.quit()
SelectionDialog.scheme = """
<afx-app-window width='250' height='300' apptitle = "Selection">
<afx-vbox>
<afx-hbox>
<div data-width = "10" />
<afx-vbox>
<div data-height="10" />
<afx-list-view data-id = "list" />
<div data-height="10" />
<afx-hbox data-height="30">
<div />
<afx-button data-id = "btnOk" text = "__(Ok)" data-width = "40" />
<afx-button data-id = "btnCancel" text = "__(Cancels)" data-width = "50" />
</afx-hbox>
</afx-vbox>
<div data-width = "10" />
</afx-hbox>
</afx-vbox>
</afx-app-window>
"""
this.OS.register "SelectionDialog", SelectionDialog
class AboutDialog extends BasicDialog
constructor: () ->
super "AboutDialog", AboutDialog.scheme
init: () ->
mt = @meta() mt = @meta()
@scheme.set "apptitle", __("About: {0}",mt.name) me = @
(@find "mylabel").set "*", {icon:mt.icon, iconclass:mt.iconclass, text:"#{mt.name}(v#{mt.version})"} @scheme.set "apptitle", __("About: {0}", mt.name)
(@find "mylabel").set "*", {
icon: mt.icon,
iconclass: mt.iconclass,
text: "#{mt.name}(v#{mt.version})"
}
($ @find "mydesc").html mt.description ($ @find "mydesc").html mt.description
# grid data for author info # grid data for author info
return unless mt.info return unless mt.info
rows = [] rows = []
rows.push [ { value: k }, { value: v } ] for k, v of mt.info rows.push [ { text: k }, { text: v } ] for k, v of mt.info
(@find "mygrid").set "header", [ { text: "", width: 100 }, { text: "" } ]
(@find "mygrid").set "rows", rows (@find "mygrid").set "rows", rows
(@find "btnCancel").set "onbtclick", (e) ->
me.quit()
AboutDialog.scheme = """
<afx-app-window data-id = 'about-window' width='300' height='200'>
<afx-vbox>
<div style="text-align:center; margin-top:10px;" data-height="50">
<h3 style = "margin:0;padding:0;">
<afx-label data-id = 'mylabel'></afx-label>
</h3>
<i><p style = "margin:0; padding:0" data-id = 'mydesc'></p></i>
</div>
<afx-grid-view data-id = 'mygrid'></afx-grid-view>
<afx-hbox data-height="30">
<div />
<afx-button data-id = "btnCancel" text = "__(Cancel)" data-width = "60" />
</afx-hbox>
<div data-height = "10"/>
</afx-vbox>
</afx-app-window>
"""
this.OS.register "AboutDialog", AboutDialog this.OS.register "AboutDialog", AboutDialog
class FileDiaLog extends BaseDialog class FileDiaLog extends BaseDialog

View File

@ -67,24 +67,27 @@ class BaseModel
subscribe: (e, f) -> subscribe: (e, f) ->
Ant.OS.announcer.on e, f, @ Ant.OS.announcer.on e, f, @
openDialog: (d, f, title, data) -> openDialog: (d, data) ->
if @dialog me = @
@dialog.show() new Promise (resolve, reject) ->
return if me.dialog
if typeof d is "string" me.dialog.show()
if not Ant.OS.GUI.subwindows[d]
@error __("Dialog {0} not found", d)
return return
@dialog = new Ant.OS.GUI.subwindows[d]() if typeof d is "string"
else if not Ant.OS.GUI.subwindows[d]
@dialog = d me.error __("Dialog {0} not found", d)
#@dialog.observable = riot.observable() unless @dialog return
@dialog.parent = @ me.dialog = new Ant.OS.GUI.subwindows[d]()
@dialog.handle = f else
@dialog.pid = @pid me.dialog = d
@dialog.data = data #@dialog.observable = riot.observable() unless @dialog
@dialog.title = title me.dialog.parent = me
@dialog.init() me.dialog.handle = resolve
me.dialog.reject = reject
me.dialog.pid = me.pid
me.dialog.data = data
me.dialog.title = data.title if data and data.title
me.dialog.init()
ask: (t, m, f) -> ask: (t, m, f) ->
@._gui.openDialog "YesNoDialog", (d) -> @._gui.openDialog "YesNoDialog", (d) ->

View File

@ -32,6 +32,10 @@ class BaseService extends this.OS.GUI.BaseModel
#implement by user #implement by user
# event registe, etc # event registe, etc
# scheme loader # scheme loader
update: () ->
@domel.set "data", @
meta: () -> meta: () ->
Ant.OS.APP[@name].meta Ant.OS.APP[@name].meta
attach: (h) -> attach: (h) ->

View File

@ -172,71 +172,69 @@ Ant.OS.API =
#request a user data #request a user data
mid: () -> mid: () ->
return Ant.OS.announcer.getMID() return Ant.OS.announcer.getMID()
post: (p, d, c, f) -> post: (p, d) ->
q = Ant.OS.announcer.getMID() new Promise (resolve, reject) ->
Ant.OS.API.loading q, p q = Ant.OS.announcer.getMID()
$.ajax {
type: 'POST',
url: p,
contentType: 'application/json',
data: JSON.stringify d,
dataType: 'json',
success: null
}
#$.getJSON p, d
.done (data) ->
Ant.OS.API.loaded q, p, "OK"
c(data)
.fail (e, s) ->
Ant.OS.API.loaded q, p, "FAIL"
f(e, s)
blob: (p, c, f) ->
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
c @response
Ant.OS.API.loaded q, p, "OK"
else
f e, @
Ant.OS.API.loaded q, p, "FAIL"
Ant.OS.API.loading q, p
r.send()
upload: (p, d, c, f) ->
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 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 { $.ajax {
url: p,
data: formd,
type: 'POST', type: 'POST',
contentType: false, url: p,
processData: false, contentType: 'application/json',
data: JSON.stringify d,
dataType: 'json',
success: null
} }
.done (data) -> .done (data) ->
Ant.OS.API.loaded q, p, "OK" Ant.OS.API.loaded q, p, "OK"
c(data) resolve(data)
o.remove()
.fail (e, s) -> .fail (e, s) ->
Ant.OS.API.loaded q, p, "FAIL" Ant.OS.API.loaded q, p, "FAIL"
f(e, s) reject(e, s)
o.remove()
o.click() 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
resolve @response
Ant.OS.API.loaded q, p, "OK"
else
reject e, @
Ant.OS.API.loaded q, p, "FAIL"
Ant.OS.API.loading q, p
r.send()
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()
.fail (e, s) ->
Ant.OS.API.loaded q, p, "FAIL"
reject(e, s)
o.remove()
o.click()
saveblob: (name, b) -> saveblob: (name, b) ->
url = window.URL.createObjectURL b url = window.URL.createObjectURL b
@ -249,41 +247,44 @@ Ant.OS.API =
window.URL.revokeObjectURL(url) window.URL.revokeObjectURL(url)
o.remove() o.remove()
systemConfig: ->
Ant.OS.API.request 'config', (result) ->
console.log result
loading: (q, p) -> loading: (q, p) ->
Ant.OS.announcer.trigger "loading", { id: q, data: { m: "#{p}", s: true }, name: "OS" } Ant.OS.announcer.trigger "loading", { id: q, data: { m: "#{p}", s: true }, name: "OS" }
loaded: (q, p, m ) ->
Ant.OS.announcer.trigger "loaded", { id: q, data: { m: "#{m}: #{p}", s: false }, name: "OS" }
get: (p, c, f, t) ->
conf =
type: 'GET',
url: p,
conf.dataType = t if t
q = Ant.OS.announcer.getMID() loaded: (q, p, m ) ->
Ant.OS.API.loading q, p Ant.OS.announcer.trigger "loaded", {
$.ajax conf id: q, data: { m: "#{m}: #{p}", s: false }, name: "OS" }
.done (data) ->
Ant.OS.API.loaded q, p, "OK" get: (p, t) ->
c(data) new Promise (resolve, reject) ->
.fail (e, s) -> conf =
Ant.OS.API.loaded q, p, "FAIL" type: 'GET',
f(e, s) url: p
script: (p, c, f) -> conf.dataType = t if t
q = Ant.OS.announcer.getMID() q = Ant.OS.announcer.getMID()
Ant.OS.API.loading q, p Ant.OS.API.loading q, p
$.getScript p $.ajax conf
.done (data) -> .done (data) ->
Ant.OS.API.loaded q, p, "OK" Ant.OS.API.loaded q, p, "OK"
c(data) resolve(data)
.fail (e, s) -> .fail (e, s) ->
Ant.OS.API.loaded q, p, "FAIL" Ant.OS.API.loaded q, p, "FAIL"
f(e, s) reject(e, s)
resource: (r, c, f) ->
script: (p) ->
new Promise (resolve, reject) ->
q = Ant.OS.announcer.getMID()
Ant.OS.API.loading q, p
$.getScript p
.done (data) ->
Ant.OS.API.loaded q, p, "OK"
resolve(data)
.fail (e, s) ->
Ant.OS.API.loaded q, p, "FAIL"
reject(e, s)
resource: (r) ->
path = "resources/#{r}" path = "resources/#{r}"
Ant.OS.API.get path, c, f Ant.OS.API.get path
libready: (l) -> libready: (l) ->
return Ant.OS.API.shared[l] || false return Ant.OS.API.shared[l] || false
@ -324,19 +325,24 @@ Ant.OS.API =
libs.splice 0, 1 libs.splice 0, 1
Ant.OS.API.requires libs, f Ant.OS.API.requires libs, f
Ant.OS.API.require libs[0], null Ant.OS.API.require libs[0], null
packages: packages:
fetch: (f) -> fetch: () ->
Ant.OS.API.handle.packages { Ant.OS.API.handle.packages {
command: "list", args: { paths: (v for k, v of Ant.OS.setting.system.pkgpaths) } command: "list", args: { paths: (v for k, v of Ant.OS.setting.system.pkgpaths) }
}, f }
cache: (f) ->
cache: () ->
Ant.OS.API.handle.packages { Ant.OS.API.handle.packages {
command: "cache", args: { paths: (v for k, v of Ant.OS.setting.system.pkgpaths) } command: "cache", args: { paths: (v for k, v of Ant.OS.setting.system.pkgpaths) }
}, f }
setting: (f) -> setting: (f) ->
Ant.OS.API.handle.setting f Ant.OS.API.handle.setting f
apigateway: (d, ws, c) -> apigateway: (d, ws, c) ->
return Ant.OS.API.handle.apigateway d, ws, c return Ant.OS.API.handle.apigateway d, ws, c
search: (text) -> search: (text) ->
r = [] r = []
@ -350,17 +356,17 @@ Ant.OS.API =
onsearch: (name, fn) -> onsearch: (name, fn) ->
Ant.OS.API.searchHandle[name] = fn unless Ant.OS.API.searchHandle[name] Ant.OS.API.searchHandle[name] = fn unless Ant.OS.API.searchHandle[name]
setLocale: (name, f) -> setLocale: (name) ->
path = "resources/languages/#{name}.json" new Promise (resolve, reject) ->
Ant.OS.API.get path, (d) -> path = "resources/languages/#{name}.json"
Ant.OS.setting.system.locale = name Ant.OS.API.get(path, "json")
Ant.OS.API.lang = d .then (d) ->
if f then f() else Ant.OS.announcer.trigger "systemlocalechange", name Ant.OS.setting.system.locale = name
, (e, s) -> Ant.OS.API.lang = d
#Ant.OS.setting.system.locale = "en_GB" Ant.OS.announcer.trigger "systemlocalechange", name
Ant.OS.announcer.oserror __("Language file {0} not found", path), e, s resolve d
f() if f .catch (e) ->
, "json" reject e
throwe: (n) -> throwe: (n) ->
err = undefined err = undefined

View File

@ -107,21 +107,29 @@ Ant.OS or=
boot: -> boot: ->
#first login #first login
console.log "Booting sytem" console.log "Booting sytem"
Ant.OS.API.handle.auth (d) -> Ant.OS.API.handle.auth()
# in case someone call it more than once :) .then (d) ->
if d.error # in case someone call it more than once :)
# show login screen if d.error
Ant.OS.GUI.login() # show login screen
else Ant.OS.GUI.login()
# startX :) else
Ant.OS.GUI.startAntOS d.result # startX :)
Ant.OS.GUI.startAntOS d.result
.catch (e) ->
console.error e
cleanupHandles: {} cleanupHandles: {}
exit: -> exit: ->
#do clean up first #do clean up first
f() for n, f of Ant.OS.cleanupHandles f() for n, f of Ant.OS.cleanupHandles
Ant.OS.API.handle.setting (r) -> Ant.OS.API.handle.setting()
Ant.OS.cleanup() .then (r) ->
Ant.OS.API.handle.logout() Ant.OS.cleanup()
Ant.OS.API.handle.logout()
.catch (e) ->
console.error e
onexit: (n, f) -> onexit: (n, f) ->
Ant.OS.cleanupHandles[n] = f unless Ant.OS.cleanupHandles[n] Ant.OS.cleanupHandles[n] = f unless Ant.OS.cleanupHandles[n]

View File

@ -19,20 +19,23 @@
class DB class DB
constructor: (@table) -> constructor: (@table) ->
save: (d, f) -> save: (d) ->
Ant.OS.API.handle.dbquery "save", { table: @table, data: d }, f Ant.OS.API.handle.dbquery "save", { table: @table, data: d }
delete: (c, f) ->
delete: (c) ->
rq = { table: @table } rq = { table: @table }
return ( Ant.OS.announcer.oserror __("VDB Unknown condition for delete command"), return new Promise (resolve, reject) ->
(Ant.OS.API.throwe "OS.DB"), c ) unless c and c isnt "" reject(Ant.OS.API.throwe("OS.DB: unkown condition")) unless c and c isnt ""
if isNaN c if isNaN c
rq.cond = c rq.cond = c
else else
rq.id = c rq.id = c
Ant.OS.API.handle.dbquery "delete", rq, f Ant.OS.API.handle.dbquery "delete", rq
get: (id, f) ->
Ant.OS.API.handle.dbquery "get", { table: @table, id: id }, f get: (id) ->
find: (cond, f) -> Ant.OS.API.handle.dbquery "get", { table: @table, id: id }
Ant.OS.API.handle.dbquery "select", { table: @table, cond: cond }, f
find: (cond) ->
Ant.OS.API.handle.dbquery "select", { table: @table, cond: cond }
Ant.OS.API.DB = DB Ant.OS.API.DB = DB

View File

@ -37,14 +37,14 @@ Ant.OS.GUI =
children: [], children: [],
dataid: "sys-apps" dataid: "sys-apps"
iconclass: "fa fa-adn", iconclass: "fa fa-adn",
onmenuselect: (d) -> onchildselect: (d) ->
Ant.OS.GUI.launch d.item.data.app Ant.OS.GUI.launch d.data.item.get("data").name
} }
], ],
onmenuselect: (d) -> onchildselect: (d) ->
return Ant.OS.exit() if d.item.data.dataid is "sys-logout" id = d.data.item.get("data").dataid
return Ant.OS.GUI.toggleFullscreen() if d.item.data.dataid is "os-fullsize" return Ant.OS.exit() if id is "sys-logout"
Ant.OS.GUI.launch d.item.data.app unless d.item.data.dataid return Ant.OS.GUI.toggleFullscreen() if id is "os-fullsize"
} }
] ]
htmlToScheme: (html, app, parent) -> htmlToScheme: (html, app, parent) ->
@ -57,7 +57,7 @@ Ant.OS.GUI =
app.main() app.main()
app.show() app.show()
loadScheme: (path, app, parent) -> loadScheme: (path, app, parent) ->
path.asFileHandle().read (x) -> path.asFileHandle().read().then (x) ->
return null unless x return null unless x
Ant.OS.GUI.htmlToScheme x, app, parent Ant.OS.GUI.htmlToScheme x, app, parent
#, (e, s) -> #, (e, s) ->
@ -100,10 +100,10 @@ Ant.OS.GUI =
srv = arr[1] srv = arr[1]
app = arr[0] app = arr[0]
return Ant.OS.PM.createProcess srv, Ant.OS.APP[srv] if Ant.OS.APP[srv] return Ant.OS.PM.createProcess srv, Ant.OS.APP[srv] if Ant.OS.APP[srv]
Ant.OS.GUI.loadApp app, Ant.OS.GUI.loadApp app
(a) -> .then (a) ->
return Ant.OS.PM.createProcess srv, Ant.OS.APP[srv] if Ant.OS.APP[srv] return Ant.OS.PM.createProcess srv, Ant.OS.APP[srv] if Ant.OS.APP[srv]
(e, s) -> .catch (e, s) ->
Ant.OS.announcer.trigger "srvroutineready", srv Ant.OS.announcer.trigger "srvroutineready", srv
Ant.OS.announcer.osfail __("Cannot read service script: {0}", srv), e, s Ant.OS.announcer.osfail __("Cannot read service script: {0}", srv), e, s
@ -153,38 +153,37 @@ Ant.OS.GUI =
($ Ant.OS.APP[app].style).remove() if Ant.OS.APP[app] and Ant.OS.APP[app].style ($ Ant.OS.APP[app].style).remove() if Ant.OS.APP[app] and Ant.OS.APP[app].style
delete Ant.OS.APP[app] delete Ant.OS.APP[app]
loadApp: (app, ok, err) -> loadApp: (app) ->
path = "os://packages/#{app}" new Promise (resolve, reject) ->
path = Ant.OS.setting.system.packages[app].path if Ant.OS.setting.system.packages[app].path path = Ant.OS.setting.system.packages[app].path if Ant.OS.setting.system.packages[app].path
js = path + "/main.js" js = path + "/main.js"
js.asFileHandle().read (d) -> js.asFileHandle().read("script")
# load app meta data .then (d) ->
"#{path}/package.json".asFileHandle().read (data) -> # load app meta data
data.path = path "#{path}/package.json".asFileHandle().read("json")
Ant.OS.APP[app].meta = data if Ant.OS.APP[app] .then (data) ->
Ant.OS.APP[v].meta = data for v in data.services if data.services data.path = path
#load css file Ant.OS.APP[app].meta = data if Ant.OS.APP[app]
css = "#{path}/main.css" Ant.OS.APP[v].meta = data for v in data.services if data.services
css.asFileHandle().onready (d) -> #load css file
stamp = (new Date).timestamp() css = "#{path}/main.css"
el = $ '<link>', { rel: 'stylesheet', type: 'text/css', 'href': "#{Ant.OS.API.handle.get}/#{css}?stamp=#{stamp}" } css.asFileHandle().onready()
.appendTo 'head' .then (d) ->
Ant.OS.APP[app].style = el[0] if Ant.OS.APP[app] stamp = (new Date).timestamp()
ok app el = $ '<link>', { rel: 'stylesheet', type: 'text/css', 'href': "#{Ant.OS.API.handle.get}/#{css}?stamp=#{stamp}" }
, () -> .appendTo 'head'
#launch Ant.OS.APP[app].style = el[0] if Ant.OS.APP[app]
ok app resolve app
, "json" .catch (e) -> resolve app
#ok app .catch (e) -> reject e
, "script" #ok app
.catch (e) -> reject e
launch: (app, args) -> launch: (app, args) ->
if not Ant.OS.APP[app] if not Ant.OS.APP[app]
# first load it # first load it
Ant.OS.GUI.loadApp app, Ant.OS.GUI.loadApp(app).then (a) ->
(a)-> Ant.OS.PM.createProcess a, Ant.OS.APP[a], args
Ant.OS.PM.createProcess a, Ant.OS.APP[a], args
, (e, s) ->
else else
# now launch it # now launch it
if Ant.OS.APP[app] if Ant.OS.APP[app]
@ -289,135 +288,136 @@ Ant.OS.GUI =
initDM: -> initDM: ->
# check login first # check login first
Ant.OS.API.resource "schemes/dm.html", (x) -> Ant.OS.API.resource("schemes/dm.html")
return null unless x .then (x) ->
scheme = $.parseHTML x return null unless x
($ "#wrapper").append scheme scheme = $.parseHTML x
($ "#wrapper").append scheme
Ant.OS.announcer.observable.one "sysdockloaded", () -> Ant.OS.announcer.observable.one "sysdockloaded", () ->
($ window).bind 'keydown', (event) -> ($ window).bind 'keydown', (event) ->
dock = ($ "#sysdock")[0] dock = ($ "#sysdock")[0]
return unless dock return unless dock
app = dock.get "selectedApp" app = dock.get "selectedApp"
#return true unless app #return true unless app
c = String.fromCharCode(event.which).toUpperCase() c = String.fromCharCode(event.which).toUpperCase()
fnk = undefined fnk = undefined
if event.ctrlKey if event.ctrlKey
fnk = "CTRL" fnk = "CTRL"
else if event.metaKey else if event.metaKey
fnk = "META" fnk = "META"
else if event.shiftKey else if event.shiftKey
fnk = "SHIFT" fnk = "SHIFT"
else if event.altKey else if event.altKey
fnk = "ALT" fnk = "ALT"
return unless fnk return unless fnk
r = if app then app.shortcut fnk, c, event else true r = if app then app.shortcut fnk, c, event else true
return event.preventDefault() if not r return event.preventDefault() if not r
return unless Ant.OS.GUI.shortcut[fnk] return unless Ant.OS.GUI.shortcut[fnk]
return unless Ant.OS.GUI.shortcut[fnk][c] return unless Ant.OS.GUI.shortcut[fnk][c]
Ant.OS.GUI.shortcut[fnk][c](event) Ant.OS.GUI.shortcut[fnk][c](event)
event.preventDefault() event.preventDefault()
# system menu and dock # system menu and dock
$("#syspanel")[0].uify() $("#syspanel")[0].uify()
$("#sysdock")[0].uify() $("#sysdock")[0].uify()
$("#systooltip")[0].uify() $("#systooltip")[0].uify()
$("#contextmenu")[0].uify() $("#contextmenu")[0].uify()
#riot.mount ($ "#syspanel", $ "#wrapper") #riot.mount ($ "#syspanel", $ "#wrapper")
#riot.mount ($ "#sysdock", $ "#workspace"), { items: [] } #riot.mount ($ "#sysdock", $ "#workspace"), { items: [] }
#riot.mount ($ "#systooltip", $ "#wrapper") #riot.mount ($ "#systooltip", $ "#wrapper")
# context menu # context menu
#riot.mount ($ "#contextmenu", $ "#wrapper") #riot.mount ($ "#contextmenu", $ "#wrapper")
($ "#workspace").contextmenu (e) -> Ant.OS.GUI.bindContextMenu e ($ "#workspace").contextmenu (e) -> Ant.OS.GUI.bindContextMenu e
# tooltip # tooltip
($ "#workspace").mouseover (e) -> ($ "#workspace").mouseover (e) ->
el = $(e.target).closest "[tooltip]" el = $(e.target).closest "[tooltip]"
return unless el.length > 0 return unless el.length > 0
Ant.OS.GUI.showTooltip el, ($(el).attr "tooltip"), e Ant.OS.GUI.showTooltip el, ($(el).attr "tooltip"), e
# desktop default file manager # desktop default file manager
desktop = $ Ant.OS.GUI.workspace desktop = $ Ant.OS.GUI.workspace
fp = Ant.OS.setting.desktop.path.asFileHandle() desktop[0].fetch = () ->
desktop[0].fetch = () ->
fn = () ->
fp = Ant.OS.setting.desktop.path.asFileHandle() fp = Ant.OS.setting.desktop.path.asFileHandle()
fp.read (d) -> fn = () ->
return Ant.OS.announcer.osfail d.error, (Ant.OS.API.throwe "OS.VFS"), d.error if d.error fp.read().then (d) ->
items = [] return Ant.OS.announcer.osfail d.error, (Ant.OS.API.throwe "OS.VFS"), d.error if d.error
$.each d.result, (i, v) -> items = []
return if v.filename[0] is '.' and not Ant.OS.setting.desktop.showhidden $.each d.result, (i, v) ->
v.text = v.filename return if v.filename[0] is '.' and not Ant.OS.setting.desktop.showhidden
#v.text = v.text.substring(0,9) + "..." ifv.text.length > 10 v.text = v.filename
v.iconclass = v.type #v.text = v.text.substring(0,9) + "..." ifv.text.length > 10
items.push(v) v.iconclass = v.type
desktop[0].set "items", items items.push(v)
desktop[0].refresh() desktop[0].set "data", items
desktop[0].refresh()
fp.onready () -> fp.onready()
fn() .then () -> fn()
, ( e ) -> # try to create the path .catch ( e ) -> # try to create the path
console.log "#{fp.path} not found" console.log "#{fp.path} not found"
name = fp.basename name = fp.basename
fp.parent().asFileHandle().mk name, (r) -> fp.parent().asFileHandle().mk(name).then (r) ->
ex = Ant.OS.API.throwe "OS.VFS" ex = Ant.OS.API.throwe "OS.VFS"
if r.error then Ant.OS.announcer.osfail d.error, ex, d.error else fn() if r.error then Ant.OS.announcer.osfail d.error, ex, d.error else fn()
desktop[0].ready = (e) -> desktop[0].ready = (e) ->
e.observable = Ant.OS.announcer e.observable = Ant.OS.announcer
window.onresize = () -> window.onresize = () ->
Ant.OS.announcer.trigger "desktopresize" Ant.OS.announcer.trigger "desktopresize"
e.refresh() e.refresh()
desktop[0].set "onlistselect", (d) -> desktop[0].set "onlistselect", (d) ->
($ "#sysdock").get(0).set "selectedApp", null ($ "#sysdock").get(0).set "selectedApp", null
desktop[0].set "onlistdbclick", ( d ) -> desktop[0].set "onlistdbclick", ( d ) ->
($ "#sysdock").get(0).set "selectedApp", null ($ "#sysdock").get(0).set "selectedApp", null
it = desktop[0].get "selected" it = desktop[0].get "selectedItem"
Ant.OS.GUI.openWith it Ant.OS.GUI.openWith it.get("data")
#($ "#workingenv").on "click", (e) -> #($ "#workingenv").on "click", (e) ->
# desktop[0].set "selected", -1 # desktop[0].set "selected", -1
desktop.on "click", (e) -> desktop.on "click", (e) ->
return unless e.target is desktop[0] return unless e.target.tagName.toUpperCase() is "UL"
desktop[0].set "selected", -1 desktop[0].unselect()
($ "#sysdock").get(0).set "selectedApp", null ($ "#sysdock").get(0).set "selectedApp", null
#console.log "desktop clicked"
desktop[0].contextmenuHandle = (e, m) -> desktop[0].contextmenuHandle = (e, m) ->
desktop[0].set "selected", -1 if e.target is desktop[0] desktop[0].unselect() if e.target.tagName.toUpperCase() is "UL"
($ "#sysdock").get(0).set "selectedApp", null ($ "#sysdock").get(0).set "selectedApp", null
menu = [ menu = [
{ text: __("Open"), dataid: "desktop-open" }, { text: __("Open"), dataid: "desktop-open" },
{ text: __("Refresh"), dataid: "desktop-refresh" } { text: __("Refresh"), dataid: "desktop-refresh" }
] ]
menu = menu.concat ( v for k, v of Ant.OS.setting.desktop.menu) menu = menu.concat ( v for k, v of Ant.OS.setting.desktop.menu)
m.set "items", menu m.set "items", menu
m.set "onmenuselect", (evt) -> m.set "onmenuselect", (evt) ->
switch evt.item.data.dataid item = evt.data.item.get("data")
when "desktop-open" switch item.dataid
it = desktop[0].get "selected" when "desktop-open"
return Ant.OS.GUI.openWith it if it it = desktop[0].get "selectedItem"
it = Ant.OS.setting.desktop.path.asFileHandle() return Ant.OS.GUI.openWith it.get("data") if it
it.mime = "dir" it = Ant.OS.setting.desktop.path.asFileHandle()
Ant.OS.GUI.openWith it it.mime = "dir"
when "desktop-refresh" Ant.OS.GUI.openWith it
desktop[0].fetch() when "desktop-refresh"
else desktop[0].fetch()
Ant.OS.GUI.launch evt.item.data.app, evt.item.data.args if evt.item.data.app else
m.show(e) Ant.OS.GUI.launch item.app, item.args if item.app
m.show(e)
desktop[0].fetch()
Ant.OS.announcer.observable.on "VFS", (d) ->
desktop[0].fetch() if d.data.file.hash() is fp.hash() or d.data.file.parent().hash() is fp.hash()
Ant.OS.announcer.ostrigger "desktoploaded"
# mount it
desktop[0].uify()
# riot.mount desktop
.catch (e) ->
console.log e
alert __("System fail: Cannot init desktop manager")
desktop[0].fetch()
Ant.OS.announcer.observable.on "VFS", (d) ->
desktop[0].fetch() if d.data.file.hash() is fp.hash() or d.data.file.parent().hash() is fp.hash()
Ant.OS.announcer.ostrigger "desktoploaded"
# mount it
desktop[0].uify()
# riot.mount desktop
, (e, s) ->
alert __("System fail: Cannot init desktop manager")
console.log s, e
refreshDesktop: () -> refreshDesktop: () ->
($ Ant.OS.GUI.workspace)[0].fetch() ($ Ant.OS.GUI.workspace)[0].fetch()
@ -444,20 +444,24 @@ Ant.OS.GUI =
return new Ant.OS.GUI.BasicDialog conf.name, conf.layout return new Ant.OS.GUI.BasicDialog conf.name, conf.layout
login: () -> login: () ->
Ant.OS.API.resource "schemes/login.html", (x) -> Ant.OS.API.resource "schemes/login.html"
return null unless x .then (x) ->
scheme = $.parseHTML x return null unless x
($ "#wrapper").append scheme scheme = $.parseHTML x
($ "#btlogin").click () -> ($ "#wrapper").append scheme
data = ($ "#btlogin").click () ->
username: ($ "#txtuser").val(), data =
password: ($ "#txtpass").val() username: ($ "#txtuser").val(),
Ant.OS.API.handle.login data, (d) -> password: ($ "#txtpass").val()
if d.error then ($ "#login_error").html d.error else Ant.OS.GUI.startAntOS d.result Ant.OS.API.handle.login data
($ "#txtpass").keyup (e) -> .then (d) ->
($ "#btlogin").click() if e.which is 13 if d.error then ($ "#login_error").html d.error else Ant.OS.GUI.startAntOS d.result
, (e, s) -> .catch (e) ->
alert __("System fail: Cannot init login screen") ($ "#login_error").html "Login: server error"
($ "#txtpass").keyup (e) ->
($ "#btlogin").click() if e.which is 13
.catch (e) ->
alert __("System fail: Cannot init login screen")
startAntOS: (conf) -> startAntOS: (conf) ->
# clean up things # clean up things
@ -473,9 +477,9 @@ Ant.OS.GUI =
Ant.OS.announcer.observable.on "systemlocalechange", (name) -> Ant.OS.announcer.observable.on "systemlocalechange", (name) ->
($ "#syspanel")[0].update() ($ "#syspanel")[0].update()
Ant.OS.API.packages.cache (ret) -> Ant.OS.API.packages.cache().then (ret) ->
if ret.result if ret.result
Ant.OS.API.packages.fetch (r) -> Ant.OS.API.packages.fetch().then (r) ->
if r.result if r.result
for k, v of r.result for k, v of r.result
v.text = v.name v.text = v.name
@ -493,5 +497,6 @@ Ant.OS.GUI =
(Ant.OS.GUI.launch a) for a in Ant.OS.setting.system.startup.apps (Ant.OS.GUI.launch a) for a in Ant.OS.setting.system.startup.apps
#Ant.OS.GUI.launch "DummyApp" #Ant.OS.GUI.launch "DummyApp"
# initDM # initDM
Ant.OS.API.setLocale Ant.OS.setting.system.locale, () -> Ant.OS.API.setLocale Ant.OS.setting.system.locale
Ant.OS.GUI.initDM() .then () ->
Ant.OS.GUI.initDM()

View File

@ -15,7 +15,7 @@
# You should have received a copy of the GNU General Public License # You should have received a copy of the GNU General Public License
#along with this program. If not, see https://www.gnu.org/licenses/. #along with this program. If not, see https://www.gnu.org/licenses/.
Ant.OS.API.HOST = Ant.location.hostname+ (if Ant.location.port then":#{Ant.location.port}" else "") Ant.OS.API.HOST = Ant.location.hostname + (if Ant.location.port then":#{Ant.location.port}" else "")
Ant.OS.API.REST = "#{Ant.location.protocol}//#{Ant.OS.API.HOST}" Ant.OS.API.REST = "#{Ant.location.protocol}//#{Ant.OS.API.HOST}"
Ant.OS.API.handle = Ant.OS.API.handle =
@ -23,101 +23,80 @@ Ant.OS.API.handle =
get: "#{Ant.OS.API.REST}/VFS/get" get: "#{Ant.OS.API.REST}/VFS/get"
# get shared file with publish # get shared file with publish
shared: "#{Ant.OS.API.REST}/VFS/shared" shared: "#{Ant.OS.API.REST}/VFS/shared"
scandir: (p, c ) -> scandir: (p) ->
path = "#{Ant.OS.API.REST}/VFS/scandir" path = "#{Ant.OS.API.REST}/VFS/scandir"
Ant.OS.API.post path, { path: p }, c, (e, s) -> Ant.OS.API.post path, { path: p }
Ant.OS.announcer.osfail __("Fail to scan directory: {0}", p), e, s
mkdir: (p, c ) -> mkdir: (p) ->
path = "#{Ant.OS.API.REST}/VFS/mkdir" path = "#{Ant.OS.API.REST}/VFS/mkdir"
Ant.OS.API.post path, { path: p }, c, (e, s) -> Ant.OS.API.post path, { path: p }
Ant.OS.announcer.osfail __("Fail to create directory: {0}", p), e, s
sharefile: (p, pub , c) -> sharefile: (p, pub) ->
path = "#{Ant.OS.API.REST}/VFS/publish" path = "#{Ant.OS.API.REST}/VFS/publish"
Ant.OS.API.post path, { path: p , publish: pub }, c, (e, s) -> Ant.OS.API.post path, { path: p , publish: pub }
Ant.OS.announcer.osfail __("Fail to publish file: {0}", p), e, s
fileinfo: (p, c) -> fileinfo: (p) ->
path = "#{Ant.OS.API.REST}/VFS/fileinfo" path = "#{Ant.OS.API.REST}/VFS/fileinfo"
Ant.OS.API.post path, { path: p }, c, (e, s) -> Ant.OS.API.post path, { path: p }
Ant.OS.announcer.osfail __("Fail to get file meta data: {0}", p), e, s
readfile: (p, c, t) -> readfile: (p, t) ->
path = "#{Ant.OS.API.REST}/VFS/get/" path = "#{Ant.OS.API.REST}/VFS/get/"
Ant.OS.API.get path + p, c, (e, s) -> Ant.OS.API.get path + p, t
Ant.OS.announcer.osfail __("Fail to read file: {0}", p), e, s
, t
move: (s, d, c) -> move: (s, d) ->
path = "#{Ant.OS.API.REST}/VFS/move" path = "#{Ant.OS.API.REST}/VFS/move"
Ant.OS.API.post path, { src: s, dest: d }, c, (e, s) -> Ant.OS.API.post path, { src: s, dest: d }
Ant.OS.announcer.osfail __("Fail to move file: {0} -> {1}", s, d), e, s
delete: (p , c) -> delete: (p) ->
path = "#{Ant.OS.API.REST}/VFS/delete" path = "#{Ant.OS.API.REST}/VFS/delete"
Ant.OS.API.post path, { path: p }, c, (e, s) -> Ant.OS.API.post path, { path: p }
Ant.OS.announcer.osfail __("Fail to delete: {0}", p), e, s
fileblob: (p, c) -> fileblob: (p) ->
path = "#{Ant.OS.API.REST}/VFS/get/" path = "#{Ant.OS.API.REST}/VFS/get/"
Ant.OS.API.blob path + p, c, (e, s) -> Ant.OS.API.blob path + p
Ant.OS.announcer.osfail "Fail to read file: #{p}", e, s
packages: (d, c) -> packages: (d) ->
path = "#{Ant.OS.API.REST}/system/packages" path = "#{Ant.OS.API.REST}/system/packages"
Ant.OS.API.post path, d, c, (e, s) -> Ant.OS.API.post path, d
Ant.OS.announcer.osfail __("Fail to {0} package", d.command), e, s
upload: (d, c) -> upload: (d) ->
path = "#{Ant.OS.API.REST}/VFS/upload" path = "#{Ant.OS.API.REST}/VFS/upload"
Ant.OS.API.upload path, d, c, (e, s) -> Ant.OS.API.upload path, d
Ant.OS.announcer.osfail __("Fail to upload file to: {0}", d), e, s
write: (p, d , c) -> write: (p, d) ->
path = "#{Ant.OS.API.REST}/VFS/write" path = "#{Ant.OS.API.REST}/VFS/write"
Ant.OS.API.post path, { path: p, data: d }, c, (e, s) -> Ant.OS.API.post path, { path: p, data: d }
Ant.OS.announcer.osfail __("Fail to write to file: {0}", p), e, s
scanapp: (p, c ) -> scanapp: (p, c ) ->
path = "#{Ant.OS.API.REST}/system/application" path = "#{Ant.OS.API.REST}/system/application"
apigateway: (d, ws, c) -> apigateway: (d, ws) ->
if ws if ws
path = "#{Ant.OS.API.HOST}/system/apigateway?ws=1" new Promise (resolve, reject) ->
proto = if window.location.protocol is "https:" then "wss://" else "ws://" path = "#{Ant.OS.API.HOST}/system/apigateway?ws=1"
socket = new WebSocket proto + path proto = if window.location.protocol is "https:" then "wss://" else "ws://"
if c then c(socket) socket = new WebSocket proto + path
return socket resolve(socket)
else else
path = "#{Ant.OS.API.REST}/system/apigateway?ws=0" path = "#{Ant.OS.API.REST}/system/apigateway?ws=0"
Ant.OS.API.post path, d, c, (e, s) -> Ant.OS.API.post path, d
Ant.OS.announcer.osfail __("Fail to invoke gateway api"), e, s
auth: (c) -> auth: () ->
p = "#{Ant.OS.API.REST}/user/auth" p = "#{Ant.OS.API.REST}/user/auth"
Ant.OS.API.post p, {}, c, (e, s) -> Ant.OS.API.post p, {}
console.log e, s
alert __("Resource not found: {0}", p) login: (d) ->
login: (d, c) ->
p = "#{Ant.OS.API.REST}/user/login" p = "#{Ant.OS.API.REST}/user/login"
Ant.OS.API.post p, d, c, () -> Ant.OS.API.post p, d
alert __("Resource not found: {0}", p)
logout: () -> logout: () ->
p = "#{Ant.OS.API.REST}/user/logout" p = "#{Ant.OS.API.REST}/user/logout"
Ant.OS.API.post p, {}, (d) -> Ant.OS.API.post p, {}
Ant.OS.boot()
, () ->
alert __("Resource not found: {0}", p)
setting: (f) ->
p = "#{Ant.OS.API.REST}/system/settings"
Ant.OS.API.post p, Ant.OS.setting, (d) ->
Ant.OS.announcer.oserror __("Cannot save system setting"), d.error if d.error
f(d) if f
, (e, s) ->
m = __("Fail to make request: {0}", p)
Ant.OS.announcer.osfail m , e, s
f({ error: m }) if f
dbquery: (cmd, d, c) -> setting: () ->
p = "#{Ant.OS.API.REST}/system/settings"
Ant.OS.API.post p, Ant.OS.setting
dbquery: (cmd, d) ->
path = "#{Ant.OS.API.REST}/VDB/#{cmd}" path = "#{Ant.OS.API.REST}/VDB/#{cmd}"
Ant.OS.API.post path, d, c, (e, s) -> Ant.OS.API.post path, d
Ant.OS.announcer.osfail __("Fail to query data from database: {0}", path), e, s

View File

@ -3,7 +3,7 @@
<!--div class = "afx-clear"></div--> <!--div class = "afx-clear"></div-->
<div id = "workspace"> <div id = "workspace">
<afx-apps-dock id="sysdock"></afx-apps-dock> <afx-apps-dock id="sysdock"></afx-apps-dock>
<afx-float-list id = "desktop" ></afx-float-list> <afx-float-list id = "desktop" dir="vertical" ></afx-float-list>
<!--div id = "workingenv"></div--> <!--div id = "workingenv"></div-->
</div> </div>
<afx-menu id="contextmenu" data-id="contextmenu" context="true" style="display:none;"></afx-menu> <afx-menu id="contextmenu" data-id="contextmenu" context="true" style="display:none;"></afx-menu>

View File

@ -1,60 +1,67 @@
class AppDockTag extends Ant.OS.GUI.BaseTag class AppDockTag extends Ant.OS.GUI.BaseTag
constructor: (r, o) -> constructor: (r, o) ->
super r, o super r, o
@setopt "selectedApp"
@setopt "onappselect", (e) -> @setopt "onappselect", (e) ->
@setopt "items", [] @setopt "items", []
@setopt "selectedApp", undefined
me = @ me = @
@root.newapp = (a) -> me.addApp a @root.newapp = (a) -> me.addApp a
@root.removeapp = (a) -> me.removeApp a
__selectedApp__: (v) -> __selectedApp__: (v) ->
el = undefined
for it in @get("items")
it.app.blur()
$(it.domel).removeClass()
el = it.domel if v and v is it.app
return unless el
$(el).addClass "selected"
$(Ant.OS.GUI.workspace)[0].unselect()
addApp: (item) -> addApp: (item) ->
@items.push item @get("items").push item
@set "selectedApp", item.app
el = $("<afx-button>") el = $("<afx-button>")
el.appendTo @root el.appendTo @root
el[0].uify @observable el[0].uify @observable
el[0].set "*", item el[0].set "*", item
el.attr "tooltip", "cr:#{item.app.title()}" el.attr "tooltip", "cr:#{item.app.title()}"
item.domel = el[0]
me = @ me = @
el[0].set "onbtclick", (e) -> el[0].set "onbtclick", (e) ->
e.id = me.aid() e.id = me.aid()
e.data.app = item e.data.app = item
me.get("onappselect") e item.app.show()
for v in @items @set "selectedApp", item.app
v.app.blur()
removeApp: (a) -> removeApp: (a) ->
i = -1 i = -1
for v, k in self.items for v, k in @get "items"
if v.app.pid == a.pid if v.app.pid == a.pid
i = k i = k
break break
if i != -1 if i != -1
delete @items[i].app items = @get("items")
@items.splice(i, 1) delete items[i].app
items.splice(i, 1)
$($(@root).children()[i]).remove() $($(@root).children()[i]).remove()
mount: () -> mount: () ->
Ant.OS.announcer.trigger "sysdockloaded"
me = @ me = @
@root.contextmenuHandle = (e, m) -> @root.contextmenuHandle = (e, m) ->
return if e.target is me.root return if e.target is me.root
bt = $(e.target).closest "afx-button" bt = $(e.target).closest "afx-button"
appidx = $(@root).children().indexOf bt app = bt[0].get "app"
app = self.items[appidx].app
m.set "items", [ m.set "items", [
{ text: "__(Show)", dataid: "show" }, { text: "__(Show)", dataid: "show" },
{ text: "__(Hide)", dataid: "hide" }, { text: "__(Hide)", dataid: "hide" },
{ text: "__(Close)", dataid: "quit" } { text: "__(Close)", dataid: "quit" }
] ]
m.set "onmenuselect", (evt) -> m.set "onmenuselect", (evt) ->
console.log evt item = evt.data.item.get("data")
### if(app[evt.item.data.dataid]) if(app[item.dataid])
app[evt.item.data.dataid]() ### app[item.dataid]()
m.show(e) m.show(e)
Ant.OS.announcer.trigger "sysdockloaded"
Ant.OS.GUI.define "afx-apps-dock", AppDockTag Ant.OS.GUI.define "afx-apps-dock", AppDockTag

View File

@ -12,7 +12,6 @@ class ButtonTag extends Ant.OS.GUI.BaseTag
__color__: (v) -> __color__: (v) ->
console.log @refs
@refs.label.set "color", v @refs.label.set "color", v
__icon__: (v) -> __icon__: (v) ->

View File

@ -18,6 +18,7 @@ class FloatListTag extends ListViewTag
@calibrate() @calibrate()
__dir__: (v) -> __dir__: (v) ->
@calibrate() @calibrate()
mount: () -> mount: () ->
me = @ me = @
$(@refs.container) $(@refs.container)
@ -28,6 +29,8 @@ class FloatListTag extends ListViewTag
.css "display", "block" .css "display", "block"
.css "width", "100%" .css "width", "100%"
@observable.on "resize", (e) -> me.calibrate() @observable.on "resize", (e) -> me.calibrate()
@root.ready(@root) if @root.ready
push: (v) -> push: (v) ->
el = super.push(v) el = super.push(v)
@enable_drag el @enable_drag el

View File

@ -83,6 +83,7 @@ class ListViewTag extends Ant.OS.GUI.BaseTag
@root.push = (e) -> me.push e @root.push = (e) -> me.push e
@root.remove = (e) -> me.remove e @root.remove = (e) -> me.remove e
@root.unshift = (e) -> me.unshift e @root.unshift = (e) -> me.unshift e
@root.unselect = () -> me.unselect()
multiselect: () -> multiselect: () ->
return false if @get "dropdown" return false if @get "dropdown"
@ -137,6 +138,12 @@ class ListViewTag extends Ant.OS.GUI.BaseTag
for item in data for item in data
@push item, false @push item, false
unselect: () ->
v.set "selected", false for v in @get("selectedItems")
@set "selectedItems", []
@set "selectedItem", undefined
iclick: (e, flag) -> iclick: (e, flag) ->
return if not e.item return if not e.item
list = @get("selectedItems") list = @get("selectedItems")

View File

@ -28,6 +28,6 @@ class OverlayTag extends Ant.OS.GUI.BaseTag
} }
layout: () -> layout: () ->
@refs.yield = @root @refs.yield = @roots
[] []
Ant.OS.GUI.define "afx-overlay", OverlayTag Ant.OS.GUI.define "afx-overlay", OverlayTag

View File

@ -68,91 +68,148 @@ class BaseFileHandle
return -1 unless @path return -1 unless @path
return @path.hash() return @path.hash()
sendB64: (p, f) -> b64: (t) ->
# t is object or mime type
me = @ me = @
m = if p is "object" then "text/plain" else p new Promise (resolve, reject) ->
return f "" unless @cache m = if t is "object" then "text/plain" else t
if p is "object" or typeof @cache is "string" return resolve "" unless me.cache
b64 = if p is "object" then (JSON.stringify @cache).asBase64() else @cache.asBase64() if t is "object" or typeof me.cache is "string"
b64 = "data:#{m};base64,#{b64}" if t is "object"
f(b64) b64 = (JSON.stringify me.cache).asBase64()
else else
reader = new FileReader() b64 = me.cache.asBase64()
reader.readAsDataURL(@cache) b64 = "data:#{m};base64,#{b64}"
reader.onload = () -> resolve(b64)
f reader.result else
reader.onerror = (e) -> reader = new FileReader()
return Ant.OS.announcer.osfail __("VFS Cannot encode file: {0}", me.path), (Ant.OS.API.throwe "OS.VFS"), e reader.readAsDataURL(me.cache)
reader.onload = () ->
resolve reader.result
reader.onerror = (e) ->
reject e
parent: () -> parent: () ->
return @ if @isRoot() return @ if @isRoot()
return (@protocol + "://" + (@genealogy.slice 0 , @genealogy.length - 1).join "/") return (@protocol + "://" + (@genealogy.slice 0 , @genealogy.length - 1).join "/")
onready: (f, err) -> onready: () ->
# read meta data # read meta data
return f() if @ready
me = @ me = @
me.meta (d) -> new Promise (resolve, reject) ->
if d.error return resolve(me.info) if me.ready
return if err then err d else Ant.OS.announcer.osfail "#{me.path}: #{d.error}", (Ant.OS.API.throwe "OS.VFS"), d.error me.meta()
me.info = d.result .then (d) ->
me.ready = true return reject d if d.errors
f() me.info = d.result
me.ready = true
resolve(d.result)
.catch (e) -> resolve e
read: (f, t) -> read: (t) ->
me = @ me = @
@onready (() -> me.action "read", t, f) new Promise (resolve, reject) ->
me.onready()
.then (r) ->
me._rd(t)
.then (d) -> resolve d
.catch (e) -> reject e
.catch (e) -> reject e
write: (d, f) -> write: (d, t) ->
me = @ me = @
@action "write", d, (r) -> new Promise (resolve, reject) ->
Ant.OS.announcer.ostrigger "VFS", { m: "write", file: me } if r.result me._wr(d, t)
f r .then (r) -> resolve r
.catch (e) -> reject e
mk: (d, f) -> mk: (d) ->
me = @ me = @
@onready (() -> me.action "mk", d, (r) -> new Promise (resolve, reject) ->
Ant.OS.announcer.ostrigger "VFS", { m: "mk", file: me } if r.result me.onready()
f r) .then (r) ->
me._mk(d)
.then (d) -> resolve d
.catch (e) -> reject e
.catch (e) -> reject e
remove: (f) -> remove: () ->
me = @ me = @
@onready (() -> me.action "remove", null, (r) -> new Promise (resolve, reject) ->
Ant.OS.announcer.ostrigger "VFS", { m: "remove", file: me } if r.result me.onready()
f r) .then (r) ->
me._rm()
.then (d) -> resolve d
.catch (e) -> reject e
.catch (e) -> reject e
upload: (f) -> upload: () ->
me = @ me = @
@onready (() -> me.action "upload", null, (r) -> new Promise (resolve, reject) ->
Ant.OS.announcer.ostrigger "VFS", { m: "upload", file: me } if r.result me.onready()
f r) .then (r) ->
publish: (f) -> me._up()
me = @ .then (d) -> resolve d
@onready (() -> me.action "publish", null, (r) -> .catch (e) -> reject e
Ant.OS.announcer.ostrigger "VFS", { m: "publish", file: me } if r.result .catch (e) -> reject e
f r)
download: (f) ->
me = @
@onready (() -> me.action "download", null, f)
move: (d, f) -> publish: () ->
me = @ me = @
@onready (() -> me.action "move", d, (r) -> new Promise (resolve, reject) ->
Ant.OS.announcer.ostrigger "VFS", { m: "move", file: d.asFileHandle() } if r.result me.onready()
f r) .then (r) ->
me._pub()
.then (d) -> resolve d
.catch (e) -> reject e
.catch (e) -> reject e
execute: (f) -> download: () ->
me = @ me = @
@onready (() -> me.action "execute", null, f) new Promise (resolve, reject) ->
me.onready()
.then (r) ->
me._down()
.then (d) -> resolve d
.catch (e) -> reject e
.catch (e) -> reject e
#mk: (f) -> move: (d) ->
me = @
new Promise (resolve, reject) ->
me.onready()
.then (r) ->
me._mv()
.then (d) -> resolve d
.catch (e) -> reject e
.catch (e) -> reject e
meta: (f) -> execute: () ->
me = @
new Promise (resolve, reject) ->
me.onready()
.then (r) ->
me._exec()
.then (d) -> resolve d
.catch (e) -> reject e
.catch (e) -> reject e
getlink: () -> @path getlink: () -> @path
# for main action read, write, remove, execute
# must be implemented by subclasses unsupported: (t) ->
action: (n, p, f) -> me = @
return Ant.OS.announcer.osfail __("VFS unknown action: {0}", n), (Ant.OS.API.throwe "OS.VFS"), n new Promise (resolve, reject) ->
reject { error: __("Action {0} is unsupported on: {1}", t, me.path) }
# actions must be implemented by subclasses
_rd: (t) -> @unsupported "read"
_wr: (d, t) -> @unsupported "write"
_mk: (d) -> @unsupported "mk"
_rm: () -> @unsupported "remove"
_mv: (d) -> @unsupported "move"
_up: () -> @unsupported "upload"
_down: () -> @unsupported "download"
_exec: () -> @unsupported "execute"
_pub: () -> @unsupported "publish"
# now export the class # now export the class
Ant.OS.API.VFS.BaseFileHandle = BaseFileHandle Ant.OS.API.VFS.BaseFileHandle = BaseFileHandle
@ -162,43 +219,66 @@ class RemoteFileHandle extends Ant.OS.API.VFS.BaseFileHandle
constructor: (path) -> constructor: (path) ->
super path super path
meta: (f) -> meta: () ->
Ant.OS.API.handle.fileinfo @path, f Ant.OS.API.handle.fileinfo @path
getlink: () -> getlink: () ->
Ant.OS.API.handle.get + "/" + @path Ant.OS.API.handle.get + "/" + @path
action: (n, p, f) -> _rd: (t) ->
# t: binary, text, any type
return Ant.OS.API.handle.scandir @path if @info.type is "dir"
#read the file
return Ant.OS.API.handle.fileblob @path if t is "binary"
Ant.OS.API.handle.readfile @path, if t then t else "text"
_wr: (d, t) ->
# t is base64 or undefined
return Ant.OS.API.handle.write me.path, me.cache if p is "base64"
me = @ me = @
switch n new Promise (resolve, reject) ->
when "read" me.b64(t)
return Ant.OS.API.handle.scandir @path, f if @info.type is "dir" .then (r) ->
#read the file Ant.OS.API.handle.write me.path, d
return Ant.OS.API.handle.fileblob @path, f if p is "binary" resolve r
Ant.OS.API.handle.readfile @path, f, if p then p else "text" .catch (e) -> reject e
when "mk"
return f { error: __("{0} is not a directory", @path) } if @info.type is "file" _mk: (d) ->
Ant.OS.API.handle.mkdir "#{@path}/#{p}", f me = @
when "write" if @info.type is "file"
return Ant.OS.API.handle.write me.path, me.cache, f if p is "base64" return new Promise (resolve, reject) ->
@sendB64 p, (data) -> reject { error: __("{0} is not a directory", me.path) }
Ant.OS.API.handle.write me.path, data, f Ant.OS.API.handle.mkdir "#{@path}/#{d}"
when "upload"
return if @info.type is "file" _rm: () ->
Ant.OS.API.handle.upload @path, f Ant.OS.API.handle.delete @path
when "remove"
Ant.OS.API.handle.delete @path, f _mv: (d) ->
when "publish" Ant.OS.API.handle.move @path, d
Ant.OS.API.handle.sharefile @path, true , f
when "download"
return if @info.type is "dir" _up: () ->
Ant.OS.API.handle.fileblob @path, (d) -> me = @
if @info.type isnt "file"
return new Promise (resolve, reject) ->
reject { error: __("{0} is not a file", me.path) }
Ant.OS.API.handle.upload @path
_down: () ->
me = @
new Promise (resolve, reject) ->
if me.info.type is "dir"
return reject { error: __("{0} is not a file", me.path) }
Ant.OS.API.handle.fileblob(@path)
.then (d) ->
blob = new Blob [d], { type: "octet/stream" } blob = new Blob [d], { type: "octet/stream" }
Ant.OS.API.saveblob me.basename, blob Ant.OS.API.saveblob me.basename, blob
when "move" resolve()
Ant.OS.API.handle.move @path, p, f .catch (e) ->
else reject e
return Ant.OS.announcer.osfail __("VFS unknown action: {0}", n), (Ant.OS.API.throwe "OS.VFS"), n
_pub: () ->
Ant.OS.API.handle.sharefile @path, true
Ant.OS.API.VFS.register "^(home|desktop|os|Untitled)$", RemoteFileHandle Ant.OS.API.VFS.register "^(home|desktop|os|Untitled)$", RemoteFileHandle
@ -209,39 +289,13 @@ class ApplicationHandle extends Ant.OS.API.VFS.BaseFileHandle
@info = Ant.OS.setting.system.packages[@basename] if @basename @info = Ant.OS.setting.system.packages[@basename] if @basename
@ready = true @ready = true
meta: (f) -> _rd: (t) ->
f()
action: (n, p, f) ->
me = @ me = @
switch n new Promise (resolve, reject) ->
when "read" return resolve { result: me.info } if me.info
return f { result: @info } if @info return reject { error: __("Application meta data isnt found") } unless me.isRoot()
return unless @isRoot() resolve { result: ( v for k, v of Ant.OS.setting.system.packages ) }
f { result: ( v for k, v of Ant.OS.setting.system.packages ) }
when "mk"
return
when "write"
return
when "upload"
# install
return
when "remove"
#uninstall
return
when "publish"
return
when "download"
return
when "move"
return
else
return Ant.OS.announcer.osfail __("VFS unknown action: {0}", n), (Ant.OS.API.throwe "OS.VFS"), n
Ant.OS.API.VFS.register "^app$", ApplicationHandle Ant.OS.API.VFS.register "^app$", ApplicationHandle
@ -255,103 +309,73 @@ class BufferFileHandle extends Ant.OS.API.VFS.BaseFileHandle
size: if data then data.length else 0 size: if data then data.length else 0
name: @basename name: @basename
type: "file" type: "file"
meta: (f) ->
f() _rd: (t) ->
me = @
new Promise (resolve, reject) ->
resolve { result: me.cache }
_wr: (d, t) ->
@cache = d
@onchange @ if @onchange
new Promise (resolve, reject) ->
resolve { result: true }
_down: () ->
me = @
new Promise (resolve, reject) ->
blob = new Blob [me.cache], { type: "octet/stream" }
Ant.OS.API.saveblob me.basename, blob
resolve()
onchange: (f) -> onchange: (f) ->
@onchange = f @onchange = f
action: (n, p, f) ->
me = @
switch n
when "read"
return f { result: @cache }
when "mk"
return
when "write"
@cache = p
@onchange @ if @onchange
f { result: true }
when "upload"
# install
return
when "remove"
#uninstall
return
when "publish"
return
when "download"
blob = new Blob [@cache], { type: "octet/stream" }
Ant.OS.API.saveblob me.basename, blob
when "move"
return
else
return Ant.OS.announcer.osfail __("VFS unknown action: {0}", n), (Ant.OS.API.throwe "OS.VFS"), n
Ant.OS.API.VFS.register "^mem$", BufferFileHandle Ant.OS.API.VFS.register "^mem$", BufferFileHandle
class URLFileHandle extends Ant.OS.API.VFS.BaseFileHandle class URLFileHandle extends Ant.OS.API.VFS.BaseFileHandle
constructor: (path) -> constructor: (path) ->
super path super path
@ready = true @ready = true
meta: (f) ->
f { result: true } _rd: (t) ->
action: (n, p, f) -> Ant.OS.API.get @path, if t then t else "text"
me = @
switch n
when "read"
Ant.OS.API.get @path, (d) ->
f(d)
, (e, s) ->
Ant.OS.announcer.oserror __("VFS cannot read : {0}", me.path), e, s
, if p then p else "text"
else
return Ant.OS.announcer.oserror __("VFS unknown action: {0}", n), (Ant.OS.API.throwe "OS.VFS"), n
Ant.OS.API.VFS.register "^(http|https)$", URLFileHandle Ant.OS.API.VFS.register "^(http|https)$", URLFileHandle
class SharedFileHandle extends Ant.OS.API.VFS.BaseFileHandle class SharedFileHandle extends Ant.OS.API.VFS.BaseFileHandle
constructor: (path) -> constructor: (path) ->
super path super path
@ready = true if @isRoot() @ready = true if @isRoot()
meta: (f) ->
Ant.OS.API.handle.fileinfo @path, f
action: (n, p, f) -> meta: () ->
Ant.OS.API.handle.fileinfo @path
_rd: (t) ->
return Ant.OS.API.get "#{Ant.OS.API.handle.shared}/all", t if @isRoot()
#read the file
return Ant.OS.API.handle.fileblob @path if t is "binary"
Ant.OS.API.handle.readfile @path, if t then t else "text"
_wr: (d, t) ->
Ant.OS.API.handle.write @path, d
_rm: () ->
Ant.OS.API.handle.sharefile @basename, false
_down: () ->
me = @ me = @
switch n new Promise (resolve, reject) ->
when "read" if me.info.type is "dir"
return Ant.OS.API.get "#{Ant.OS.API.handle.shared}/all", f, ((e, s)->) if @isRoot() return reject { error: __("{0} is not a file", me.path) }
#read the file Ant.OS.API.handle.fileblob me.path, (d) ->
return Ant.OS.API.handle.fileblob @path, f if p is "binary" blob = new Blob [d], { type: "octet/stream" }
Ant.OS.API.handle.readfile @path, f, if p then p else "text" Ant.OS.API.saveblob me.basename, blob
when "mk" resolve()
return
when "write" _pub: () ->
Ant.OS.API.handle.write @path, p, f me = @
return new Promise (resolve, reject) -> resolve { result: me.basename }
when "remove"
Ant.OS.API.handle.sharefile @basename, false, f
when "upload"
return
when "publish"
return f { result: @basename }
when "download"
return if @info.type is "dir"
Ant.OS.API.handle.fileblob @path, (d) ->
blob = new Blob [d], { type: "octet/stream" }
Ant.OS.API.saveblob me.basename, blob
when "move"
return
else
return Ant.OS.announcer.osfail __("VFS unknown action: {0}", n), (Ant.OS.API.throwe "OS.VFS"), n
Ant.OS.API.VFS.register "^shared$", SharedFileHandle Ant.OS.API.VFS.register "^shared$", SharedFileHandle

View File

@ -31,9 +31,9 @@ class Calendar extends this.OS.GUI.BaseService
awake: (e) -> awake: (e) ->
@.openDialog "CalendarDialog", (d) -> @.openDialog("CalendarDialog" )
.then (d) ->
console.log d console.log d
, "Calendar"
# do nothing # do nothing
cleanup: (evt) -> cleanup: (evt) ->
console.log "cleanup for quit" console.log "cleanup for quit"

View File

@ -1,5 +1,55 @@
<afx-app-window apptitle="Preview" width="100" height="100"> <afx-app-window data-id="example-show-case" apptitle="AntOS GUI widgets" width="650" height="500">
<afx-hbox> <afx-vbox>
<afx-button data-height="30" toggle = "true" text="__(Read more)" iconclass="fa fa-camera-retro fa-lg" id="button"></afx-button> <afx-menu data-height="30" data-id="menu" />
</afx-hbox> <afx-tab-container data-id="tabctn" tabbarheight= "30">
<afx-hbox tabname="Widgets">
<afx-vbox data-width="150">
<afx-tree-view data-id="tree" />
<afx-slider data-id="slider" data-height="30" value="50"/>
</afx-vbox>
<afx-resizer data-width="5" />
<afx-vbox data-width="grow">
<afx-hbox min-height="50">
<afx-switch data-id="switch" />
<afx-button text="__(This is the label)"
data-id="bttest"
iconclass="fa fa-camera-retro fa-lg"
icon="os://packages/DummyApp/icon.png"/>
<afx-nspinner data-id="spin" value="10" step="2" />
</afx-hbox>
<afx-resizer data-height="5" />
<afx-hbox>
<afx-list-view data-id="list" dropdown="false" multiselect="true" />
</afx-hbox>
<afx-hbox data-height="150">
<afx-grid-view data-id="grid" multiselect="false" />
</afx-hbox>
</afx-vbox>
</afx-hbox>
<afx-hbox tabname="Virtual desktop">
<afx-float-list data-id = "flist"/>
</afx-hbox>
<afx-hbox tabname="Calendar">
<afx-calendar-view data-id = "cal"/>
</afx-hbox>
<afx-hbox tabname="Color picker">
<afx-color-picker data-id = "cpk"/>
</afx-hbox>
<afx-hbox tabname="File view">
<afx-vbox>
<afx-file-view data-id = "fileview"/>
<afx-list-view data-id = "viewoption" data-height="30" dropdown="true" />
</afx-vbox>
</afx-hbox>
<afx-hbox tabname="Dialogs">
<afx-vbox>
<afx-list-view data-id = "dialoglist"/>
<afx-button data-id = "btrundia" text="Run dialog" data-height="30"/>
</afx-vbox>
</afx-hbox>
</afx-tab-container>
</afx-vbox>
</afx-app-window> </afx-app-window>

View File

@ -5,107 +5,20 @@ class ShowCase extends this.OS.GUI.BaseApplication
main: () -> main: () ->
me = @ me = @
@announcer = new Ant.OS.API.Announcer()
@announcer.on "evt1", (data) ->
console.log "evt1", data
@announcer.on "evt1", (data) ->
console.log "evt1 fn1: ", data
fn = (data) ->
console.log "evt1 fn2: ", data
@announcer.on "evt1", fn
@announcer.on "evt2", (data) ->
console.log "evt2", data
@announcer.one "evt1", (data) ->
console.log "evt1 only one", data
@announcer.one "*", (data) ->
console.log "evt1 only one*", data
@announcer.on "*", fn
console.log me.announcer bt = @find 'bttest'
@on "btclick", (e) -> bt.set "onbtclick", (e) ->
me.openwin()
###
me.announcer.trigger("evt1", "Hello 1")
me.announcer.off("*")
me.announcer.trigger("evt2", "Hello 2")
console.log me.announcer
me.notify "Hello"
console.log tag
###
openwin: () ->
me = @
scheme = $.parseHTML """
<afx-app-window data-id="example-show-case" apptitle="Preview" width="650" height="500">
<afx-vbox>
<afx-menu data-height="30" data-id="menu" />
<afx-tab-container data-id="tabctn" tabbarheight= "30">
<afx-hbox tabname="Widgets">
<afx-vbox data-width="150">
<afx-tree-view data-id="tree" />
<afx-slider data-id="slider" data-height="30" value="50"/>
</afx-vbox>
<afx-resizer data-width="5" />
<afx-vbox data-width="grow">
<afx-hbox min-height="50">
<afx-switch data-id="switch" />
<afx-button text="__(This is the label)"
data-id="bttest"
iconclass="fa fa-camera-retro fa-lg"
icon="os://packages/DummyApp/icon.png"/>
<afx-nspinner data-id="spin" value="10" step="2" />
</afx-hbox>
<afx-resizer data-height="5" />
<afx-hbox>
<afx-list-view data-id="list" dropdown="false" multiselect="true" />
</afx-hbox>
<afx-hbox data-height="150">
<afx-grid-view data-id="grid" multiselect="false" />
</afx-hbox>
</afx-vbox>
</afx-hbox>
<afx-hbox tabname="Virtual desktop">
<afx-float-list data-id = "flist"/>
</afx-hbox>
<afx-hbox tabname="Calendar">
<afx-calendar-view data-id = "cal"/>
</afx-hbox>
<afx-hbox tabname="Color picker">
<afx-color-picker data-id = "cpk"/>
</afx-hbox>
<afx-hbox tabname="File view">
<afx-vbox>
<afx-file-view data-id = "fileview"/>
<afx-list-view data-id = "viewoption" data-height="30" dropdown="true" />
</afx-vbox>
</afx-hbox>
</afx-tab-container>
</afx-vbox>
</afx-app-window>
"""
ctmenu = $.parseHTML """<afx-menu data-id="mn-context" context="true" style="display:none;" /></div>"""
($ "#desktop").append scheme[0]
($ "#wrapper").append ctmenu[0]
me.subwin = scheme[0].uify()
bt = $ "[data-id='bttest']", scheme[0]
bt[0].set "onbtclick", (e) ->
console.log "btclicked" console.log "btclicked"
me.subwin.set "resizable", true
me.subwin.set "minimizable", false
me.subwin.observable.on "exit", () ->
me.subwin.observable.off "*"
$(me.subwin).remove()
me.quit()
me.subwin.observable.on "btclick", (e) -> @observable.on "btclick", (e) ->
console.log "button clicked" console.log "button clicked"
me.subwin.observable.on "menuselect", (e) -> @observable.on "menuselect", (e) ->
console.log e.id console.log e.id
list = $ "[data-id='list']", scheme[0] list = @find 'list'
list[0].set "data", [ list.set "data", [
{ text: "some thing with avery long text" }, { text: "some thing with avery long text" },
{ text: "some thing 1", closable: true }, { text: "some thing 1", closable: true },
{ text: "some thing 2", iconclass: "fa fa-camera-retro fa-lg" }, { text: "some thing 2", iconclass: "fa fa-camera-retro fa-lg" },
@ -113,42 +26,39 @@ class ShowCase extends this.OS.GUI.BaseApplication
{ text: "some thing 4" }, { text: "some thing 4" },
{ text: "some thing 5" } { text: "some thing 5" }
] ]
list[0].unshift { text: "shifted el" } list.unshift { text: "shifted el" }
console.log "after shift", list[0].get("data") list.set "onlistselect", (e) -> console.log(e.data.items)
list[0].set "onlistselect", (e) -> @observable.on "itemclose", (e) ->
console.log(e.data.items) console.log "remove", list.get("data")
me.subwin.observable.on "itemclose", (e) ->
console.log "remove", list[0].get("data")
console.log list[0].get "selectedItem" console.log list[0].get "selectedItem"
console.log list[0].get "selectedItems" console.log list[0].get "selectedItems"
sw = $ "[data-id='switch']", scheme[0] sw = @find 'switch'
sw[0].set "onchange", (e) -> sw.set "onchange", (e) ->
console.log e.data console.log e.data
spin = $ "[data-id='spin']", scheme[0] spin = @find 'spin'
spin[0].set "onchange", (e) -> spin.set "onchange", (e) ->
console.log e.data console.log e.data
menu = $ "[data-id='menu']", scheme[0] menu = @find 'menu'
menu[0].set "items", @menu() menu.set "items", @menu()
ctmenu = ctmenu[0].uify(me.subwin.observable)
ctmenu.set "items", @menu()
ctmenu.set "onmenuselect", (e) ->
console.log "root event", e
list[0].contextmenuHandle = (e) ->
console.log e
ctmenu.show e
grid = $ "[data-id='grid']", scheme[0] list.contextmenuHandle = (e, m) ->
grid[0].set "oncelldbclick", (e) -> m.set "items", me.menu()
m.show e
grid = @find 'grid'
grid.set "oncelldbclick", (e) ->
console.log "on dbclick", e console.log "on dbclick", e
grid[0].set "onrowselect", (e) -> grid.set "onrowselect", (e) ->
console.log "on rowselect", e.data.items console.log "on rowselect", e.data.items
me.subwin.observable.on "cellselect", (e) ->
@observable.on "cellselect", (e) ->
console.log "observable", e console.log "observable", e
grid[0].set "header", [{ text: "header1", width: 80 }, { text: "header2" }, { text: "header3" }]
grid[0].set "rows", [ grid.set "header", [{ text: "header1", width: 80 }, { text: "header2" }, { text: "header3" }]
grid.set "rows", [
[{ text: "text 1" }, { text: "text 2" }, { text: "text 3" }], [{ text: "text 1" }, { text: "text 2" }, { text: "text 3" }],
[{ text: "text 4" }, { text: "text 5" }, { text: "text 6" }], [{ text: "text 4" }, { text: "text 5" }, { text: "text 6" }],
[{ text: "text 7" }, { text: "text 8" }, { text: "text 9" }], [{ text: "text 7" }, { text: "text 8" }, { text: "text 9" }],
@ -191,42 +101,42 @@ class ShowCase extends this.OS.GUI.BaseApplication
] ]
} }
tree = $ "[data-id='tree']", scheme[0] tree = @find 'tree'
tree[0].set "data", tdata tree.set "data", tdata
tree[0].set "ontreeselect", (e) -> tree.set "ontreeselect", (e) ->
console.log e.data.item.get "treepath" console.log e.data.item.get "treepath"
tree[0].set "ontreedbclick", (e) -> tree.set "ontreedbclick", (e) ->
console.log "treedbclick", e console.log "treedbclick", e
me.subwin.observable.on "treedbclick", (e) -> @observable.on "treedbclick", (e) ->
console.log "observable treedbclick", e console.log "observable treedbclick", e
slider = $ "[data-id='slider']", scheme[0] slider = @find 'slider'
slider[0].set "onchanging", (v) -> slider.set "onchanging", (v) ->
console.log v console.log v
list = $ "[data-id='flist']", scheme[0] flist = @find 'flist'
list[0].set "data", [ flist.set "data", [
{ text: "File.txt" }, { text: "File.txt" },
{ text: "FileB.doc" }, { text: "FileB.doc" },
{ text: "Data.doc", iconclass: "fa fa-camera-retro fa-lg" } { text: "Data.doc", iconclass: "fa fa-camera-retro fa-lg" }
] ]
cal = $ "[data-id='cal']", scheme[0] cal = @find 'cal'
cal[0].set "ondateselect", (e) -> cal.set "ondateselect", (e) ->
console.log e console.log e
pk = $ "[data-id='cpk']", scheme[0] pk = @find 'cpk'
pk[0].set "oncolorselect", (e) -> pk.set "oncolorselect", (e) ->
console.log e console.log e
pk[0].set "oncolorselect", (e) -> pk.set "oncolorselect", (e) ->
console.log e console.log e
fileview = $("[data-id='fileview']", scheme[0])[0] fileview = @find 'fileview'
fileview.set "fetch", (path) -> fileview.set "fetch", (path) ->
new Promise (resolve, reject) -> new Promise (resolve, reject) ->
dir = path.asFileHandle() dir = path.asFileHandle()
dir.read (d) -> dir.read().then (d) ->
p = dir.parent().asFileHandle() p = dir.parent().asFileHandle()
p.filename = "[..]" p.filename = "[..]"
p.type = "dir" p.type = "dir"
@ -235,7 +145,7 @@ class ShowCase extends this.OS.GUI.BaseApplication
resolve d.result resolve d.result
fileview.set "path", "home:///" fileview.set "path", "home:///"
viewoption = $("[data-id='viewoption']", scheme[0])[0] viewoption = @find 'viewoption'
viewoption.set "data", [ viewoption.set "data", [
{ text: "icon" }, { text: "icon" },
{ text: "list" }, { text: "list" },
@ -245,6 +155,74 @@ class ShowCase extends this.OS.GUI.BaseApplication
console.log e.data.item.get("data").text console.log e.data.item.get("data").text
fileview.set "view", e.data.item.get("data").text fileview.set "view", e.data.item.get("data").text
dllist = @find "dialoglist"
btrun = @find "btrundia"
dllist.set "data", [
{ text: "Prompt dialog", id: "prompt" },
{ text: "Calendar dialog", id: "calendar" },
{ text: "Color picker dialog", id: "colorpicker" },
{ text: "Info dialog", id: "info" },
{ text: "YesNo dialog", id: "yesno" },
{ text: "Selection dialog", id: "selection" },
{ text: "About dialog", id: "about" },
{ text: "File dialog", id: "file" }
]
btrun.set "onbtclick", (e) ->
item = dllist.get "selectedItem"
return unless item
switch item.get("data").id
when "prompt"
me.openDialog("PromptDialog", {
title: "Prompt review",
value: "txt data",
label: "enter value"
})
.then (d) ->
console.log d
when "calendar"
me.openDialog("CalendarDialog", {
title: "Calendar"
})
.then (d) ->
console.log d
when "colorpicker"
me.openDialog("ColorPickerDialog")
.then (d) ->
console.log d
when "info"
me.openDialog("InfoDialog", {
title: "Info application",
name: "Show case",
date: "10/12/2014",
description: "the brown fox jumps over the lazy dog"
})
.then (d) ->
when "yesno"
me.openDialog("YesNoDialog", {
title: "Question ?",
label: "Do you realy want to delete file ?"
})
.then (d) ->
console.log d
when "selection"
me.openDialog("SelectionDialog", {
title: "Select data ?",
data: [
{ text: "Option 1" },
{ text: "Option 2" },
{ text: "Option 3", iconclass: "fa fa-camera-retro fa-lg" }
]
})
.then (d) ->
console.log d
when "about"
me.openDialog("AboutDialog" )
.then (d) ->
else return
mnFile: () -> mnFile: () ->
#console.log file #console.log file
@ -274,9 +252,7 @@ class ShowCase extends this.OS.GUI.BaseApplication
{ text: "__(Paste)", dataid: "#{@name}-paste", shortcut: 'C-P' } { text: "__(Paste)", dataid: "#{@name}-paste", shortcut: 'C-P' }
], onchildselect: (e) -> console.log "child", e ], onchildselect: (e) -> console.log "child", e
} }
cleanup: () ->
return unless @subwin
$(@subwin).remove()
menu: () -> menu: () ->
me = @ me = @
menu = [ menu = [

View File

@ -41,12 +41,19 @@ afx-calendar-view i.nextmonth:before{
right:5px;*/ right:5px;*/
} }
/*
afx-calendar-view afx-grid-view afx-grid-row.selected{ afx-calendar-view afx-grid-view afx-grid-row.selected{
background-color: white; background-color: white;
color:#414339; color:#414339;
} }*/
afx-calendar-view afx-grid-view afx-grid-row .cellselected{ afx-calendar-view afx-grid-view div.afx-grid-row-selected .afx-grid-cell
{
background-color: transparent;
color:black;
}
afx-calendar-view afx-grid-view div.afx-grid-row-selected .afx-grid-cell-selected
{
background-color: #116cd6; background-color: #116cd6;
color:white; color:white;
border-radius: 6px; border-radius: 6px;

View File

@ -7,21 +7,28 @@ afx-grid-view afx-grid-row div{
cursor:default; cursor:default;
} }
afx-grid-view afx-grid-row:nth-child(even){
afx-grid-view div.afx-grid-row:nth-child(even) .afx-grid-cell
{
background-color: #f5F5F5; background-color: #f5F5F5;
} }
afx-grid-view afx-grid-row.grid_row_header div{ afx-grid-view div.afx-grid-row-selected .afx-grid-cell
{
background-color: #116cd6;
color:white;
}
afx-grid-view div.afx-grid-row-selected .afx-grid-cell-selected
{
font-weight: bold;
}
afx-grid-view div.grid_row_header{
border-right: 2px solid #e5e5e5; border-right: 2px solid #e5e5e5;
user-select:none; user-select:none;
-webkit-user-select:none; -webkit-user-select:none;
cursor:default; cursor:default;
}
afx-grid-view afx-grid-row.selected {
background-color: #116cd6;
color:white;
}
afx-grid-view afx-grid-row.grid_row_header {
font-weight: bold; font-weight: bold;
border: 1px solid #e5e5e5; border: 1px solid #e5e5e5;
border-right:0; border-right:0;

View File

@ -37,8 +37,7 @@ afx-list-view > div.list-container > ul li{
-webkit-user-select:none; -webkit-user-select:none;
cursor:default; cursor:default;
} }
afx-list-view > div.list-container > ul afx-list-item:nth-child(even) li, afx-list-view > div.list-container > ul afx-list-item:nth-child(even) li{
afx-list-view > div.list-container > ul li:nth-child(even){
background-color:#f5F5F5; background-color:#f5F5F5;
} }
afx-list-view i.closable{ afx-list-view i.closable{
@ -63,8 +62,7 @@ afx-list-view > div.list-container > ul li > i {
margin-right: 3px; margin-right: 3px;
} }
afx-list-view > div.list-container > ul > afx-list-item > li.selected, afx-list-view > div.list-container > ul > afx-list-item > li.selected{
afx-list-view > div.list-container > ul li.selected {
background-color: #116cd6; background-color: #116cd6;
color:white; color:white;
} }

View File

@ -7,6 +7,17 @@ afx-tab-bar afx-list-view {
padding:0; padding:0;
margin:0; margin:0;
} }
afx-tab-bar afx-list-view > div.list-container > ul afx-list-item:nth-child(even) li
{
background-color: transparent;
}
afx-tab-bar afx-list-view > div.list-container > ul > afx-list-item > li.selected
{
background-color: #116cd6;
color:white;
}
afx-tab-bar afx-list-view > div.list-container > ul li{ afx-tab-bar afx-list-view > div.list-container > ul li{
float:left; float:left;
border-top-left-radius: 5px; border-top-left-radius: 5px;

View File

@ -39,7 +39,7 @@ html,body{
cursor: default; cursor: default;
padding:0px; padding:0px;
} }
#desktop > div > .float_list_item { #desktop > div > ul afx-list-item {
display:block; display:block;
background-color:transparent; background-color:transparent;
text-align: center; text-align: center;
@ -48,7 +48,7 @@ html,body{
padding:3px; padding:3px;
} }
#desktop > div > div.float_list_item_selected { #desktop > div > ul afx-list-item li.selected {
display:block; display:block;
background-color: #116cd6; background-color: #116cd6;
color:white; color:white;
@ -59,7 +59,7 @@ html,body{
padding:3px; padding:3px;
} }
#desktop > div > div.float_list_item i.file:before{ #desktop > div > ul afx-list-item i.file:before{
content: "\f15b\a"; content: "\f15b\a";
font-family: "FontAwesome"; font-family: "FontAwesome";
font-size: 32px; font-size: 32px;
@ -68,11 +68,11 @@ html,body{
font-style: normal; font-style: normal;
font-weight: normal; font-weight: normal;
} }
#desktop > div > div.float_list_item span{ #desktop > div > ul afx-list-item span{
width: 100%; width: 100%;
word-wrap: break-word; word-wrap: break-word;
} }
#desktop > div > div.float_list_item i.dir:before{ #desktop > div > ul afx-list-item i.dir:before{
display: block; display: block;
content: "\f07b\a"; content: "\f07b\a";
font-family: "FontAwesome"; font-family: "FontAwesome";
@ -81,6 +81,7 @@ html,body{
font-weight: normal; font-weight: normal;
font-style: normal; font-style: normal;
} }
#systooltip { #systooltip {
border:1px solid #a6a6a6; border:1px solid #a6a6a6;
border-radius: 3px; border-radius: 3px;