2022-08-21 19:34:24 +02:00
|
|
|
namespace OS {
|
|
|
|
|
|
|
|
export namespace application {
|
|
|
|
const BUTTON_ICONS = {
|
|
|
|
opendoc: "",
|
|
|
|
newdoc: ""
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @class LibreOffice
|
|
|
|
* @extends {BaseApplication}
|
|
|
|
*/
|
|
|
|
export class LibreOffice extends BaseApplication {
|
|
|
|
private access_token: string;
|
|
|
|
private curr_file: API.VFS.BaseFileHandle;
|
|
|
|
private placeholder: HTMLDivElement;
|
|
|
|
private eid: string;
|
|
|
|
private iframe: HTMLIFrameElement;
|
2022-08-24 13:34:57 +02:00
|
|
|
private editor_meta: GenericObject<GenericObject<string>>;
|
2022-08-21 19:34:24 +02:00
|
|
|
private post_msg_handle: (e:any) => void;
|
2022-08-24 13:34:57 +02:00
|
|
|
private mimes: string[];
|
|
|
|
private current_mode: string;
|
2022-08-21 19:34:24 +02:00
|
|
|
|
|
|
|
static discovery_uri: string;
|
|
|
|
constructor(args: AppArgumentsType[]) {
|
|
|
|
super("LibreOffice", args);
|
|
|
|
this.access_token = undefined;
|
|
|
|
this.curr_file = undefined;
|
|
|
|
this.eid = `id${Math.random().toString(36).replace(".","")}`;
|
|
|
|
this.iframe = undefined;
|
2022-08-24 13:34:57 +02:00
|
|
|
this.mimes = this.meta().mimes.map(e=>e);
|
|
|
|
this.current_mode = undefined;
|
2022-08-21 19:34:24 +02:00
|
|
|
this.post_msg_handle = (e) =>
|
|
|
|
{
|
|
|
|
this.process_iframe_msg(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
main(): void {
|
|
|
|
if(this.args && this.args.length > 0)
|
|
|
|
{
|
|
|
|
this.curr_file = this.args[0].path.asFileHandle();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.placeholder = this.find("editor-area") as HTMLDivElement;
|
|
|
|
this.placeholder.id = this.eid;
|
|
|
|
|
|
|
|
(this.find("btn-open-file") as GUI.tag.ButtonTag).onbtclick = (e) =>
|
|
|
|
{
|
|
|
|
this.openFile();
|
|
|
|
}
|
|
|
|
(this.find("btn-new-doc") as GUI.tag.ButtonTag).onbtclick = (e) =>
|
|
|
|
{
|
|
|
|
this.create("word");
|
|
|
|
}
|
|
|
|
|
|
|
|
(this.find("btn-new-cell") as GUI.tag.ButtonTag).onbtclick = (e) =>
|
|
|
|
{
|
|
|
|
this.create("sheet");
|
|
|
|
}
|
|
|
|
|
|
|
|
(this.find("btn-new-slide") as GUI.tag.ButtonTag).onbtclick = (e) =>
|
|
|
|
{
|
|
|
|
this.create("slide");
|
|
|
|
}
|
|
|
|
$(window).on("message", this.post_msg_handle);
|
|
|
|
this.discover()
|
|
|
|
.then((data) =>
|
|
|
|
{
|
|
|
|
this.editor_meta = data;
|
|
|
|
if(this.curr_file)
|
|
|
|
this.open()
|
|
|
|
})
|
|
|
|
.catch((e) =>{
|
|
|
|
this.error(__("Unable to discover LibreOffice service: {0}", e.toString()), e);
|
|
|
|
this.quit(true);
|
|
|
|
});
|
|
|
|
}
|
2022-08-24 13:34:57 +02:00
|
|
|
menu(): OS.GUI.BasicItemType[]{
|
|
|
|
const nodes = [
|
|
|
|
{ text: "__(New)", dataid :"new" },
|
|
|
|
{ text: "__(Open)", dataid :"open" }
|
|
|
|
]
|
|
|
|
if(this.current_mode == "edit")
|
|
|
|
{
|
|
|
|
nodes.push({ text: "__(Save)", dataid :"save"});
|
|
|
|
nodes.push({ text: "__(Save As)", dataid :"saveas"});
|
|
|
|
}
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
text: "__(File)",
|
|
|
|
nodes: nodes,
|
|
|
|
onchildselect: (e) =>
|
|
|
|
{
|
|
|
|
switch(e.data.item.data.dataid)
|
|
|
|
{
|
|
|
|
case "new":
|
|
|
|
this.check_dirty().then((_)=>this.new_document());
|
|
|
|
break;
|
|
|
|
case "open":
|
|
|
|
this.check_dirty().then((_)=>this.openFile());
|
|
|
|
break;
|
|
|
|
case "save":
|
|
|
|
this.post_message("Action_Save", {
|
|
|
|
DontTerminateEdit: true,
|
|
|
|
DontSaveIfUnmodified: true,
|
|
|
|
Notify: true
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case "saveas":
|
|
|
|
this.check_dirty().then((_)=>this.save_as());
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
2022-08-21 19:34:24 +02:00
|
|
|
private update_title()
|
|
|
|
{
|
|
|
|
let title = this.curr_file.path;
|
|
|
|
if(this.curr_file.dirty)
|
|
|
|
{
|
|
|
|
title += ` ${__("(modified)")}`;
|
|
|
|
}
|
|
|
|
(this.scheme as GUI.tag.WindowTag).apptitle = title;
|
|
|
|
}
|
|
|
|
private post_message(id:string, values?: GenericObject<any>)
|
|
|
|
{
|
|
|
|
let msg:GenericObject<any> = {MessageId: id,SendTime: Date.now()};
|
|
|
|
if(values)
|
|
|
|
msg.Values = values;
|
|
|
|
this.iframe.contentWindow.postMessage(JSON.stringify(msg),"*");
|
|
|
|
}
|
|
|
|
private process_iframe_msg(e:any)
|
|
|
|
{
|
|
|
|
if(e.originalEvent.source != this.iframe.contentWindow)
|
|
|
|
return;
|
|
|
|
let orgevt = e.originalEvent;
|
|
|
|
let data = JSON.parse(orgevt.data);
|
|
|
|
switch(data.MessageId)
|
|
|
|
{
|
|
|
|
case 'Action_Load_Resp':
|
|
|
|
if(!data.Values.success)
|
|
|
|
{
|
|
|
|
this.error(data.Values.errorMsg);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'App_LoadingStatus':
|
|
|
|
if(data.Values.Status == "Document_Loaded")
|
|
|
|
{
|
|
|
|
this.post_message("Host_PostmessageReady");
|
|
|
|
this.trigger("document_file_loaded");
|
2022-08-24 13:34:57 +02:00
|
|
|
if(this.current_mode == "edit")
|
|
|
|
{
|
|
|
|
this.post_message("Insert_Button",
|
2022-08-21 19:34:24 +02:00
|
|
|
{
|
|
|
|
id:'lool_new_file',
|
|
|
|
imgurl:BUTTON_ICONS.newdoc,
|
|
|
|
label: __("New file").__(),
|
|
|
|
hint: __("Create new document").__(),
|
|
|
|
insertBefore: 'save'
|
|
|
|
}
|
2022-08-24 13:34:57 +02:00
|
|
|
);
|
|
|
|
this.post_message("Insert_Button",
|
|
|
|
{
|
|
|
|
id:'lool_open_file',
|
|
|
|
imgurl:BUTTON_ICONS.opendoc,
|
|
|
|
label: __("Open file").__(),
|
|
|
|
hint: __("Open document").__(),
|
|
|
|
insertBefore: 'lool_new_file'
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2022-08-21 19:34:24 +02:00
|
|
|
}
|
|
|
|
if(data.Values.Status == "Frame_Ready")
|
|
|
|
{
|
|
|
|
$(this.iframe).css("visibility","visible");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'Doc_ModifiedStatus':
|
|
|
|
this.curr_file.dirty = data.Values.Modified;
|
|
|
|
this.update_title();
|
|
|
|
break;
|
|
|
|
case "Clicked_Button":
|
|
|
|
switch(data.Values.Id)
|
|
|
|
{
|
|
|
|
case 'lool_open_file':
|
|
|
|
this.check_dirty().then((_)=>this.openFile());
|
|
|
|
break;
|
|
|
|
case 'lool_new_file':
|
|
|
|
this.check_dirty().then((_)=>this.new_document());
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
break;
|
2022-08-24 13:34:57 +02:00
|
|
|
case "UI_SaveAs":
|
|
|
|
this.check_dirty().then((_)=>this.save_as());
|
|
|
|
break;
|
2022-08-21 19:34:24 +02:00
|
|
|
default:
|
|
|
|
console.log(data);
|
|
|
|
}
|
|
|
|
//console.log(this.eid, e);
|
|
|
|
|
|
|
|
}
|
2022-08-24 13:34:57 +02:00
|
|
|
private save_as()
|
|
|
|
{
|
|
|
|
this.openDialog("FileDialog", {
|
|
|
|
title: __("Save file as"),
|
|
|
|
type: "dir",
|
|
|
|
file: this.curr_file.asFileHandle()
|
|
|
|
})
|
|
|
|
.then(async (d) => {
|
|
|
|
const file = `${d.file.path}/${d.name}`.asFileHandle();
|
|
|
|
try
|
|
|
|
{
|
|
|
|
|
|
|
|
const r = await this.exec({
|
|
|
|
action: 'duplicate',
|
|
|
|
args:{src: this.curr_file.path, dest: file.path}
|
|
|
|
});
|
|
|
|
if(r.error)
|
|
|
|
{
|
|
|
|
throw r.error;
|
|
|
|
}
|
|
|
|
this.curr_file = file;
|
|
|
|
this.open();
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
this.error(__("Unable to save file as {0}: {1}", file.path, e.toString()),e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2022-08-21 19:34:24 +02:00
|
|
|
private new_document()
|
|
|
|
{
|
|
|
|
this.openDialog("SelectionDialog", {
|
|
|
|
title: __("Create new"),
|
|
|
|
data:[
|
|
|
|
{
|
|
|
|
text: __("Document"),
|
|
|
|
iconclass: "fa fa-file-word-o",
|
|
|
|
type: "word"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
text: __("Spreadsheet"),
|
|
|
|
iconclass: "fa fa-file-excel-o",
|
|
|
|
type: "sheet"
|
|
|
|
},
|
|
|
|
{
|
|
|
|
text: __("Presentation"),
|
|
|
|
iconclass: "fa fa-file-powerpoint-o",
|
|
|
|
type: "slide"
|
|
|
|
},
|
|
|
|
]
|
|
|
|
})
|
|
|
|
.then((d) =>
|
|
|
|
{
|
|
|
|
this.create(d.type);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
private discover(): Promise<GenericObject<any>>
|
|
|
|
{
|
|
|
|
return new Promise(async (resolve, reject) => {
|
|
|
|
try{
|
|
|
|
let xml_text = await LibreOffice.discovery_uri.asFileHandle().read();
|
|
|
|
let parser = new DOMParser();
|
|
|
|
let xml_doc = parser.parseFromString(xml_text,"text/xml");
|
|
|
|
let apps = xml_doc.getElementsByTagName("app");
|
|
|
|
let meta = {};
|
|
|
|
if(apps)
|
|
|
|
{
|
|
|
|
for(let app of apps){
|
2022-08-24 13:34:57 +02:00
|
|
|
let name = app.getAttribute("name")
|
|
|
|
if(name.match(/^[^\/]*\/[^\/]*$/g))
|
|
|
|
{
|
|
|
|
if(!(this.mimes as any).includes(name))
|
|
|
|
{
|
|
|
|
this.mimes.push(name);
|
|
|
|
}
|
|
|
|
}
|
2022-08-21 19:34:24 +02:00
|
|
|
let actions = app.getElementsByTagName("action");
|
|
|
|
if(actions)
|
|
|
|
{
|
|
|
|
for(let action of actions)
|
|
|
|
{
|
|
|
|
let ext = action.getAttribute("ext");
|
2022-08-24 13:34:57 +02:00
|
|
|
let mode = action.getAttribute("name");
|
2022-08-21 19:34:24 +02:00
|
|
|
let urlsrc = action.getAttribute("urlsrc");
|
|
|
|
if(ext && ext != "" && urlsrc)
|
|
|
|
{
|
2022-08-24 13:34:57 +02:00
|
|
|
meta[ext] =
|
|
|
|
{
|
|
|
|
url: urlsrc,
|
|
|
|
mode: mode
|
|
|
|
}
|
2022-08-21 19:34:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
resolve(meta);
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
reject(__e(e));
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
}
|
|
|
|
private openFile(): void {
|
|
|
|
this.openDialog("FileDialog", {
|
|
|
|
title: __("Open file"),
|
|
|
|
type: "file",
|
2022-08-24 13:34:57 +02:00
|
|
|
mimes: this.mimes
|
2022-08-21 19:34:24 +02:00
|
|
|
})
|
|
|
|
.then((d) =>
|
|
|
|
{
|
|
|
|
this.curr_file = d.file.path.asFileHandle();
|
|
|
|
this.open();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
private create(type: string): void
|
|
|
|
{
|
|
|
|
let ext = undefined
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case 'word':
|
|
|
|
ext = "docx";
|
|
|
|
break;
|
|
|
|
case 'sheet':
|
|
|
|
ext = "xlsx";
|
|
|
|
break;
|
|
|
|
case "slide":
|
|
|
|
ext = "pptx";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
this.error(__("Unknown doc type {0}", type));
|
|
|
|
}
|
|
|
|
this.openDialog("FileDialog", {
|
|
|
|
title: __("Save file as"),
|
|
|
|
type: "dir",
|
|
|
|
file: `home://Untitled.${ext}`.asFileHandle()
|
|
|
|
})
|
|
|
|
.then(async (d) =>
|
|
|
|
{
|
|
|
|
try{
|
|
|
|
let file = `${d.file.path}/${d.name}`.asFileHandle();
|
|
|
|
let model = `${this.path()}/templates/model.${ext}`.asFileHandle();
|
|
|
|
let data = await model.read("binary");
|
|
|
|
let blob = new Blob([data], {
|
|
|
|
type: model.info.mime,
|
|
|
|
});
|
|
|
|
file.cache = blob;
|
|
|
|
await file.write(model.info.mime);
|
|
|
|
file.cache = undefined;
|
|
|
|
this.curr_file = file;
|
|
|
|
this.open();
|
|
|
|
}
|
|
|
|
catch(e)
|
|
|
|
{
|
|
|
|
this.error(__("Unable to create document {0}", e.toString()),e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
private open(): void
|
|
|
|
{
|
|
|
|
(this.scheme as GUI.tag.WindowTag).apptitle = __("Libre Office Online");
|
|
|
|
if(!this.curr_file)
|
|
|
|
return;
|
|
|
|
this.exec({
|
|
|
|
action: 'token',
|
|
|
|
args:{file: this.curr_file.path}
|
|
|
|
})
|
|
|
|
.then((r) =>{
|
|
|
|
if(r.error)
|
|
|
|
{
|
|
|
|
this.error(r.error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.access_token = r.result.sid;
|
2022-08-24 13:34:57 +02:00
|
|
|
let mt = this.editor_meta[this.curr_file.ext];
|
|
|
|
if(!mt || !mt.url)
|
2022-08-21 19:34:24 +02:00
|
|
|
{
|
|
|
|
return this.error(__("Unknown editor for extension {0}", this.curr_file.ext));
|
|
|
|
}
|
2022-08-24 13:34:57 +02:00
|
|
|
this.current_mode = mt.mode;
|
|
|
|
// refresh the file menu
|
|
|
|
this.appmenu.items = this.baseMenu() || [];
|
2022-08-21 19:34:24 +02:00
|
|
|
$(this.placeholder).empty();
|
|
|
|
let el = $('<iframe>', {
|
2022-08-24 13:34:57 +02:00
|
|
|
src: `${mt.url}?WOPISrc=${this.uapi()}`,
|
2022-08-21 19:34:24 +02:00
|
|
|
frameborder: 0
|
|
|
|
});
|
|
|
|
this.iframe = el[0] as HTMLIFrameElement;
|
|
|
|
el
|
|
|
|
.css("width", "100%")
|
|
|
|
.css("height", "100%")
|
|
|
|
.css("display", "block")
|
|
|
|
.css("visibility","hidden")
|
|
|
|
.appendTo(this.placeholder);
|
|
|
|
this.load(new Promise((ok,r)=>{
|
|
|
|
this.one("document_file_loaded",(_)=> ok(true))
|
|
|
|
}));
|
|
|
|
|
|
|
|
})
|
|
|
|
.catch((e) => {
|
|
|
|
this.error(e.toString(), e);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
private exec(request: GenericObject<any>): Promise<any> {
|
|
|
|
let cmd = {
|
|
|
|
path: this.meta().path + "/api/api.lua",
|
|
|
|
parameters: request
|
|
|
|
};
|
|
|
|
return this.call(cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
private uapi(): string
|
|
|
|
{
|
|
|
|
let cmd = {
|
|
|
|
path: this.meta().path + "/api/api.lua",
|
|
|
|
parameters: {
|
|
|
|
action: 'file',
|
|
|
|
args: {file: this.curr_file.path}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// encode data to URL safe base64 encoding
|
|
|
|
let encoded =
|
|
|
|
btoa(JSON.stringify(cmd))
|
|
|
|
.trimBy("=")
|
|
|
|
.replace(/\+/g, '-')
|
|
|
|
.replace(/\//g, '_');
|
|
|
|
return `${this._api.REST}/system/apigateway/${encoded}/wopi/files/${this.curr_file.basename}&${this.access_token}`;
|
|
|
|
}
|
|
|
|
private check_dirty():Promise<any>
|
|
|
|
{
|
|
|
|
return new Promise((ok,_)=>{
|
|
|
|
if(this.curr_file && this.curr_file.dirty)
|
|
|
|
{
|
|
|
|
|
|
|
|
this.ask({ title: "__(Document unsaved)", text: "__(Continue action without saving?)" })
|
|
|
|
.then((d) =>
|
|
|
|
{
|
|
|
|
if(!d) return;
|
|
|
|
ok(true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ok(true);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
cleanup(e):void
|
|
|
|
{
|
|
|
|
if(this.curr_file && this.curr_file.dirty)
|
|
|
|
{
|
|
|
|
e.preventDefault();
|
2022-08-24 13:34:57 +02:00
|
|
|
this.check_dirty().then((_)=>{
|
|
|
|
this.curr_file.dirty = false;
|
|
|
|
this.quit(true);
|
|
|
|
}
|
|
|
|
);
|
2022-08-21 19:34:24 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
$(window).off("message",this.post_msg_handle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LibreOffice.discovery_uri = "https://loo.iohub.dev/hosting/discovery";
|
|
|
|
}
|
|
|
|
}
|