// Copyright (c) 2022-2024 Niko Bonnieure, Par le Peuple, NextGraph.org developers // All rights reserved. // Licensed under the Apache License, Version 2.0 // // or the MIT license , // at your option. All files in the project carrying such // notice may not be copied, modified, or distributed except // according to those terms. import { writable, readable, readonly, derived, get, type Writable, } from "svelte/store"; import { official_classes } from "./classes"; import { official_apps, official_services } from "./zeras"; import { format } from "svelte-i18n"; let loaded_external_apps = {}; export const load_app = async (appName: string) => { let app = await get_app(appName); return await import(`./apps/${app["ng:b"]}.svelte`); }; export const load_official_app = async (app) => { //console.log(app); //console.log(app["ng:b"]); let component = await import(`./apps/${app["ng:b"]}.svelte`); //console.log(component.default); return component.default; }; export const get_app = (appName: string) => { if (appName.startsWith("n:g:z")) { let app = official_apps[appName]; if (!app) throw new Error("Unknown official app"); return app; } else { //TODO: load external app from its repo // TODO: keep it in cache in loaded_external_apps } }; export const invoke_service = async (serviceName: string, nuri: string, args: object) => { if (serviceName.startsWith("n:g:z")) { let service = official_services[serviceName]; if (!service) throw new Error("Unknown official service"); // TODO: do this in WebWorker // TODO: if on native app or CLI: use deno //return await ng.app_invoke(serviceName[6..], nuri, args); } else { // TODO: if on webapp: only allow those invocations from IFrame of external app or from n:g:z:external_service_invoke (which runs in an IFrame) and run it from webworker // TODO: if on native app or CLI: use deno // TODO: load external service from its repo } }; export const get_class = (class_name) => { if (class_name.startsWith("app:") && class_name !== "app:z") { //TODO: load external app from its repo // cache it in loaded_external_apps // return the class } else { return official_classes[class_name]; } }; const find_viewers_for_class = (class_name: string) => { let found = []; for (const zera of Object.entries(official_apps)) { if (zera[0].includes("dump") || zera[0].includes("source")) continue; let viewers = zera[1]["ng:o"]; if (viewers && viewers.includes(class_name)) { found.push(zera[0]); } } return found; } const find_editors_for_class = (class_name: string) => { let found = []; for (const zera of Object.entries(official_apps)) { let viewers = zera[1]["ng:w"]; if (viewers && viewers.includes(class_name)) { found.push(zera[0]); } } return found; } const find_source_viewer_for_class = (class_def, class_name) => { switch (class_def["ng:crdt"]) { case 'Graph': return "n:g:z:crdt_source_viewer:rdf"; case 'YMap': return "n:g:z:crdt_source_viewer:ymap"; case 'YArray': return "n:g:z:crdt_source_viewer:yarray"; case 'Automerge': case 'Elmer': return "n:g:z:crdt_source_viewer:json"; case 'YXml': return "n:g:z:crdt_source_viewer:xml"; case 'YText': if (class_name === "post:text" || class_name.startsWith("code")) return false; return "n:g:z:crdt_source_viewer:text"; } } const class_to_viewers_editors = (class_name: string) => { let class_def = get_class(class_name); let has_discrete = class_def["ng:crdt"] !== "Graph"; let discrete_viewer = has_discrete ? class_def["ng:o"] : undefined; let discrete_editor = has_discrete ? class_def["ng:w"] : undefined; let graph_viewers = []; let graph_editors = []; if (!has_discrete) { if (class_def["ng:o"]) graph_viewers.push(class_def["ng:o"]); if (class_def["ng:w"]) graph_editors.push(class_def["ng:w"]); } for (const additional_g_v of find_viewers_for_class("data:graph")){ if (!graph_viewers.includes(additional_g_v)) graph_viewers.push(additional_g_v); } for (const additional_g_e of find_editors_for_class("data:graph")){ if (!graph_editors.includes(additional_g_e)) graph_editors.push(additional_g_e); } let graph_viewer = graph_viewers[0]; let graph_editor = graph_editors[0]; let discrete_viewers = []; let discrete_editors = []; if (has_discrete) { if (discrete_viewer) discrete_viewers.push(discrete_viewer); if (discrete_editor) discrete_editors.push(discrete_editor); for (const v of find_viewers_for_class(class_name)) { if (v!==discrete_viewer) discrete_viewers.push(v); } for (const e of find_editors_for_class(class_name)) { if (e!==discrete_editor) discrete_editors.push(e); } let source_viewer = find_source_viewer_for_class(class_def, class_name); if (source_viewer) discrete_viewers.push(source_viewer); if (!discrete_viewer) discrete_viewer = discrete_viewers[0]; } return { graph_viewers, graph_editors, discrete_viewers, discrete_editors, graph_viewer, graph_editor, discrete_viewer, discrete_editor } } export const open_branch = (nuri: string, in_tab: boolean) => { if (!get(all_tabs)[nuri]) { all_tabs.update((old) => { if (!old[nuri]) { //console.log("creating tab for ",nuri) old[nuri] = JSON.parse(JSON.stringify(old[""])); old[nuri].branch.nuri = nuri; } return old; }); } if (in_tab) { cur_branch.set(nuri); let store_type = get(all_tabs)[nuri].store.store_type; if (store_type) change_nav_bar(`nav:${store_type}`,get(format)(`doc.${store_type}_store`), undefined); } } export const update_class = (cur_tab, class_name) => { cur_tab.branch.class = class_name; cur_tab.graph_or_discrete = get_class(class_name)["ng:crdt"] === "Graph"; cur_tab.branch.has_discrete = !cur_tab.graph_or_discrete; return {...cur_tab, ...class_to_viewers_editors(class_name)}; } export const update_branch_display = (cur_tab) => { if (cur_tab.doc.nuri == cur_tab.branch.nuri) { cur_tab.branch.type = "main"; cur_tab.branch.display = "main"; } } export const show_modal_menu = writable(false); export const show_modal_create = writable(false); export const in_memory_graph = writable(""); export const in_memory_discrete = writable(""); //TODO: call it also when switching branches inside same repo export const reset_in_memory = () => { //console.log("reset_in_memory"); in_memory_graph.update((d)=> {return "";}); in_memory_discrete.update((d)=> {return "";}); //console.log(get(in_memory_graph)); //console.log(get(in_memory_discrete)); }; export const all_tabs = writable({ "":{ store: { repo: false, // a StoreRepo serialization overlay: "", // "v:" has_outer: "", // "l:" store_type: "", //"public" "protected", "private", "group", "dialog", readcap: "", // "r:" readcap of main is_member: "", // "r:" readcap of store root branch can_edit: true, inner: "", // "w:l:" stream: { // only if not a Dialog notif: 0, last: "", nuri: "", }, // comes from main branch of store title: "", icon: "", description: "", }, plato: { nuri: "", can_edit: false, }, branch: { nuri: "", // :o:v or :o:v:b readcap: "", // "r:" comment_branch: "", // nuri class: "", has_discrete: false, id: "", //"b:xxx" branch id (can be null if not of type "branch") c: "", //"c:xxx" commit(s) id type: "", // "main", "stream", "detached", "branch", "in_memory" (does not save) display: "", // or main or stream or a:xx or branch:X or c:X (only 7 chars) attachments: 0, files: 0, comments: 0, title: "", icon: "", description: "", app: "", // current app being used }, view_or_edit: true, // true=> view, false=> edit graph_viewer: "", // selected viewer graph_editor: "", // selected editor discrete_viewer: "", // selected viewer discrete_editor: "", // selected editor graph_viewers: [], // list of available viewers graph_editors: [], // list of available editors discrete_viewers: [], // list of available viewers discrete_editors: [], // list of available editors find: "",//or string to find graph_or_discrete: false, // set to branch.crdt === "data/graph" doc: { nuri: "",// :o is_store: false, is_member: "", // ":r" readcap of root branch authors: [], inbox: "", can_edit: false, live_edit: false, title: "", icon: "", description: "", live_editors: { }, branches : {}, }, folders_pane: false, toc_pane: false, messenger_pane: false, right_pane: "", // "branches", "files", "history", "comments", "info", "chat", "mc" action: false, // "repost", "dm", "react", "author", "copy", "forward", "link", "qr", "download", "embed", "new_block", "notifs", "schema", "signature", "permissions", "settings", "print", "console", "source", "services", "dev", show_menu: false, persistent_error: false, header_in_view: true, } }); export let in_memory_save = []; let in_memory_save_callback = async (updates) => {}; export const cur_tab_register_on_save = (f:(updates) => Promise) => { in_memory_save_callback = f; in_memory_save = []; } export const cur_tab_deregister_on_save = async () => { await save(); in_memory_save_callback = async (updates) => {}; } export const save = async () => { // saving the doc // TODO fetch updates from local storage nav_bar.update((o) => { if (o.save === true) o.save = false; return o; }); if (in_memory_save.length > 0) { await in_memory_save_callback(in_memory_save); } in_memory_save = []; } export const set_header_in_view = function(val) { cur_tab_update((old) => { old.header_in_view = val; return old;}); } export const cur_branch = writable(""); export const cur_tab = derived([cur_branch, all_tabs], ([cb, all]) => {return all[cb];}); export const can_have_header = derived(cur_tab, ($cur_tab) => { return !($cur_tab.doc.is_store && ( $cur_tab.store.store_type === "private" || $cur_tab.store.store_type === "dialog")); }); export const cur_tab_branch_nuri = derived(cur_tab, ($cur_tab) => { return $cur_tab.branch.nuri; }); export const cur_tab_doc_can_edit = derived(cur_tab, ($cur_tab) => { return $cur_tab.doc.can_edit; }); export const cur_tab_doc_is_store = derived(cur_tab, ($cur_tab) => { return $cur_tab.doc.is_store; }); export const cur_tab_doc_is_member = derived(cur_tab, ($cur_tab) => { return $cur_tab.doc.is_member; }); export const cur_tab_store_type = derived(cur_tab, ($cur_tab) => { return $cur_tab.store.store_type; }); export const cur_tab_persistent_error = derived(cur_tab, ($cur_tab) => { return $cur_tab.persistent_error; }); export const cur_tab_header_in_view = derived(cur_tab, ($cur_tab) => { return $cur_tab.header_in_view; }); export const cur_tab_right_pane = derived(cur_tab, ($cur_tab) => { return $cur_tab.right_pane; }); export const cur_tab_folders_pane = derived(cur_tab, ($cur_tab) => { return $cur_tab.folders_pane; }); export const cur_tab_toc_pane = derived(cur_tab, ($cur_tab) => { return $cur_tab.toc_pane; }); export const cur_tab_show_menu = derived(cur_tab, ($cur_tab) => { return $cur_tab.show_menu; }); export const cur_tab_branch_class = derived(cur_tab, ($cur_tab) => { return $cur_tab.branch.class; }); export const cur_tab_branch_has_discrete = derived(cur_tab, ($cur_tab) => { return $cur_tab.branch.has_discrete; }); export const cur_tab_graph_or_discrete = derived(cur_tab, ($cur_tab) => { return $cur_tab.graph_or_discrete; }); export const cur_tab_view_or_edit = derived(cur_tab, ($cur_tab) => { return $cur_tab.view_or_edit; }); export const edit_header_button = derived(cur_tab, ($cur_tab) => { return ($cur_tab.doc.is_store && ( $cur_tab.store.store_type === "public" || $cur_tab.store.store_type === "protected"))? "doc.header.buttons.edit_profile" : "doc.header.buttons.edit"; }); export const header_title = derived(cur_tab, ($cur_tab) => { if ($cur_tab.doc.is_store) { if ($cur_tab.store.store_type !== "private" && $cur_tab.store.store_type !== "dialog") { return $cur_tab.store.title; } } else { let title = $cur_tab.branch.title || $cur_tab.doc.title; if (title) return title; let app = get_app($cur_tab.branch.class); if (app) return app["ng:n"]; } return false; }); /* to be used with */ export const header_icon = derived(cur_tab, ($cur_tab) => { if ($cur_tab.doc.is_store) { if ($cur_tab.store.store_type !== "private") { return $cur_tab.store.icon;// TODO: fetch image and return blob: } } else { let icon = $cur_tab.branch.icon || $cur_tab.doc.icon; if (icon) return icon;// TODO: fetch image and return blob: let class_type = get_class($cur_tab.branch.class); if (class_type) return "class:" + $cur_tab.branch.class; } return false; }); export const header_description = derived(cur_tab, ($cur_tab) => { if ($cur_tab.doc.is_store) { if ($cur_tab.store.store_type !== "private") { return $cur_tab.store.description; } } else { let description = $cur_tab.branch.description || $cur_tab.doc.description; if (description) return description; } return false; }); export const cur_tab_store_name_override = derived(cur_tab, ($cur_tab) => { if ($cur_tab.doc.is_store && $cur_tab.store.store_type !== "private" && $cur_tab.store.title && !$cur_tab.header_in_view) return $cur_tab.store.title; else return false; }); export const cur_tab_store_icon_override = derived(cur_tab, ($cur_tab) => { if ($cur_tab.doc.is_store && $cur_tab.store.store_type !== "private" && $cur_tab.store.icon && !$cur_tab.header_in_view) return $cur_tab.store.icon; else return false; }); export const tab_update = function( tab, fn ) { all_tabs.update((all) => { all[tab] = fn(all[tab]); return all; }); }; export const cur_tab_update = function( fn ) { all_tabs.update((all) => { all[get(cur_branch)] = fn(all[get(cur_branch)]); return all; }); }; export const live_editing = writable(false); export const showMenu = () => { show_modal_menu.set(true); cur_tab_update(ct => { ct.show_menu = true; return ct; }); } export const hideMenu = () => { show_modal_menu.set(false); cur_tab_update(ct => { ct.show_menu = false; return ct; }); } export const nav_bar = writable({ //icon: "class:post:rich", icon: "", //icon: "blob:http://localhost:1421/be6f968f-ff51-4e8f-bd32-36c60b7af49a", title: "", back: false, newest: 0, save: undefined, }); live_editing.subscribe((val) => { cur_tab_update((old)=> { old.doc.live_edit = val; nav_bar.update((o) => { o.save = old.doc.live_edit ? undefined : ( in_memory_save.length > 0 ? true : false ) return o; }); return old; }); }); export const nav_bar_newest = derived(nav_bar, ($nav_bar) => { return $nav_bar.newest; }); export const nav_bar_save = derived(nav_bar, ($nav_bar) => { return $nav_bar.save; }); export const nav_bar_back = derived(nav_bar, ($nav_bar) => { return $nav_bar.back; }); export const nav_bar_title = derived(nav_bar, ($nav_bar) => { return $nav_bar.title; }); export const nav_bar_icon = derived(nav_bar, ($nav_bar) => { return $nav_bar.icon; }); export const nav_bar_reset_newest = () => { nav_bar.update((old) => { old.newest = 0; return old; }); } export const change_nav_bar = (icon, title, back) => { nav_bar.update((old) => { if (icon !== undefined) { old.icon = icon; } if (title !== undefined) { old.title = title; } if (back !== undefined) { old.back = back; } return old; }); live_editing.set(get(cur_tab).doc.live_edit); }; export const persistent_error = (nuri, pe) => { tab_update(nuri, tab => { tab.persistent_error = pe; return tab; }); } export const all_files_count = derived(cur_tab, ($cur_tab) => { let total = $cur_tab.branch.files; return total ? `(${total})` : ""; }); export const all_comments_count = derived(cur_tab, ($cur_tab) => { let total = $cur_tab.branch.comments; return total ? `(${total})` : ""; }); export const has_editor_chat = derived(cur_tab, ($cur_tab) => { return $cur_tab.doc.can_edit && $cur_tab.store.store_type !== "private" && $cur_tab.store.store_type !== "dialog"; }); export const toggle_live_edit = async () => { let is_live; cur_tab_update(ct => { ct.doc.live_edit = !ct.doc.live_edit; is_live = ct.doc.live_edit; live_editing.set(ct.doc.live_edit); return ct; }); if (is_live) { //send all the updates with live_discrete_update await save(); } } export const set_viewer = (app_name: string) => { if (get(cur_tab).graph_or_discrete) { cur_tab_update(ct => {ct.graph_viewer = app_name; ct.branch.app = app_name; return ct;}); } else { cur_tab_update(ct => {ct.discrete_viewer = app_name; ct.branch.app = app_name; return ct;}); } } export const set_editor = (app_name: string) => { if (get(cur_tab).graph_or_discrete) { cur_tab_update(ct => {ct.graph_editor = app_name; ct.branch.app = app_name; return ct;}); } else { cur_tab_update(ct => {ct.discrete_editor = app_name; ct.branch.app = app_name; return ct;}); } } export const toggle_graph_discrete = () => { cur_tab_update(ct => { ct.graph_or_discrete = !ct.graph_or_discrete; return ct; }); } export const set_graph_discrete = (val:boolean) => { cur_tab_update(ct => { ct.graph_or_discrete = val; return ct; }); } export const set_view_or_edit = (val:boolean) => { cur_tab_update(ct => { ct.view_or_edit = val; return ct; }); } export const open_viewer = () => { set_view_or_edit(true); } export const cur_viewer = derived(cur_tab, ($cur_tab) => { let app_name = $cur_tab.graph_or_discrete ? $cur_tab.graph_viewer : $cur_tab.discrete_viewer; if (app_name) { let app = get_app(app_name); return app; } }); export const cur_editor = derived(cur_tab, ($cur_tab) => { let app_name = $cur_tab.graph_or_discrete ? $cur_tab.graph_editor : $cur_tab.discrete_editor; if (app_name) { let app = get_app(app_name); return app; } }); export const cur_app = derived(cur_tab, ($cur_tab) => { let app_name = $cur_tab.view_or_edit ? $cur_tab.graph_or_discrete ? $cur_tab.graph_viewer : $cur_tab.discrete_viewer : $cur_tab.graph_or_discrete ? $cur_tab.graph_editor : $cur_tab.discrete_editor; if (app_name) { let app = get_app(app_name); return app; } }); export const available_viewers = derived(cur_tab, ($cur_tab) => { let list = $cur_tab.graph_or_discrete ? $cur_tab.graph_viewers : $cur_tab.discrete_viewers; return list.map((viewer) => { let app = { ...get_app(viewer) }; if (!app["ng:u"]) app["ng:u"] = "view"; return app }); }); export const available_editors = derived(cur_tab, ($cur_tab) => { let list = $cur_tab.graph_or_discrete ? $cur_tab.graph_editors : $cur_tab.discrete_editors; return list.map((editor) => { let app = { ...get_app(editor) }; if (!app["ng:u"]) app["ng:u"] = "edit"; return app }); });