UNPKG

solid-panes

Version:

Solid-compatible Panes: applets and views for the mashlib and databrowser

431 lines (351 loc) • 16.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0; var _solidUi = require("solid-ui"); var _rdflib = require("rdflib"); /* pad Pane ** */ var paneDef = { // icon: (module.__dirname || __dirname) + 'images/ColourOn.png', icon: _solidUi.icons.iconBase + 'noun_79217.svg', name: 'pad', audience: [_solidUi.ns.solid('PowerUser')], // Does the subject deserve an pad pane? label: function label(subject, context) { var t = context.session.store.findTypeURIs(subject); if (t['http://www.w3.org/ns/pim/pad#Notepad']) { return 'pad'; } return null; // No under other circumstances }, mintClass: _solidUi.ns.pad('Notepad'), mintNew: function mintNew(context, newPaneOptions) { var store = context.session.store; var updater = store.updater; if (newPaneOptions.me && !newPaneOptions.me.uri) { throw new Error('notepad mintNew: Invalid userid'); } var newInstance = newPaneOptions.newInstance = newPaneOptions.newInstance || store.sym(newPaneOptions.newBase + 'index.ttl#this'); // var newInstance = kb.sym(newBase + 'pad.ttl#thisPad'); var newPadDoc = newInstance.doc(); store.add(newInstance, _solidUi.ns.rdf('type'), _solidUi.ns.pad('Notepad'), newPadDoc); store.add(newInstance, _solidUi.ns.dc('title'), 'Shared Notes', newPadDoc); store.add(newInstance, _solidUi.ns.dc('created'), new Date(), newPadDoc); // @@ TODO Remove casting if (newPaneOptions.me) { store.add(newInstance, _solidUi.ns.dc('author'), newPaneOptions.me, newPadDoc); } // kb.add(newInstance, ns.pad('next'), newInstance, newPadDoc); // linked list empty @@ var chunk = store.sym(newInstance.uri + '_line0'); store.add(newInstance, _solidUi.ns.pad('next'), chunk, newPadDoc); // Linked list has one entry store.add(chunk, _solidUi.ns.pad('next'), newInstance, newPadDoc); store.add(chunk, _solidUi.ns.dc('author'), newPaneOptions.me, newPadDoc); store.add(chunk, _solidUi.ns.sioc('content'), '', newPadDoc); return new Promise(function (resolve, reject) { updater.put(newPadDoc, store.statementsMatching(undefined, undefined, undefined, newPadDoc), 'text/turtle', function (uri2, ok, message) { if (ok) { resolve(newPaneOptions); } else { reject(new Error('FAILED to save new tool at: ' + uri2 + ' : ' + message)); } }); }); }, // and follow instructions there // @@ TODO Set better type for paneOptions render: function render(subject, context, paneOptions) { var dom = context.dom; var store = context.session.store; // Utility functions var complainIfBad = function complainIfBad(ok, message) { if (!ok) { div.appendChild(_solidUi.widgets.errorMessageBlock(dom, message, 'pink')); } }; var clearElement = function clearElement(ele) { while (ele.firstChild) { ele.removeChild(ele.firstChild); } return ele; }; // Access control // Two variations of ACL for this app, public read and public read/write // In all cases owner has read write control var genACLtext = function genACLtext(docURI, aclURI, allWrite) { var g = (0, _rdflib.graph)(); var auth = (0, _rdflib.Namespace)('http://www.w3.org/ns/auth/acl#'); var a = g.sym(aclURI + '#a1'); var acl = g.sym(aclURI); var doc = g.sym(docURI); g.add(a, _solidUi.ns.rdf('type'), auth('Authorization'), acl); g.add(a, auth('accessTo'), doc, acl); g.add(a, auth('agent'), me, acl); g.add(a, auth('mode'), auth('Read'), acl); g.add(a, auth('mode'), auth('Write'), acl); g.add(a, auth('mode'), auth('Control'), acl); a = g.sym(aclURI + '#a2'); g.add(a, _solidUi.ns.rdf('type'), auth('Authorization'), acl); g.add(a, auth('accessTo'), doc, acl); g.add(a, auth('agentClass'), _solidUi.ns.foaf('Agent'), acl); g.add(a, auth('mode'), auth('Read'), acl); if (allWrite) { g.add(a, auth('mode'), auth('Write'), acl); } // TODO: Figure out why `serialize` isn't on the type definition according to TypeScript: return (0, _rdflib.serialize)(acl, g, aclURI, 'text/turtle'); }; /** * @param docURI * @param allWrite * @param callbackFunction * * @returns {Promise<Response>} */ var setACL = function setACL(docURI, allWrite, callbackFunction) { var aclDoc = store.any((0, _rdflib.sym)(docURI), (0, _rdflib.sym)('http://www.iana.org/assignments/link-relations/acl')); // @@ check that this get set by web.js if (aclDoc) { // Great we already know where it is var aclText = genACLtext(docURI, aclDoc.uri, allWrite); return fetcher.webOperation('PUT', aclDoc.uri, { data: aclText, contentType: 'text/turtle' }).then(function () { return callbackFunction(true); })["catch"](function (err) { callbackFunction(false, err.message); }); } else { return fetcher.load(docURI)["catch"](function (err) { callbackFunction(false, 'Getting headers for ACL: ' + err); }).then(function () { var aclDoc = store.any((0, _rdflib.sym)(docURI), (0, _rdflib.sym)('http://www.iana.org/assignments/link-relations/acl')); if (!aclDoc) { // complainIfBad(false, "No Link rel=ACL header for " + docURI); throw new Error('No Link rel=ACL header for ' + docURI); } var aclText = genACLtext(docURI, aclDoc.uri, allWrite); return fetcher.webOperation('PUT', aclDoc.uri, { data: aclText, contentType: 'text/turtle' }); }).then(function () { return callbackFunction(true); })["catch"](function (err) { callbackFunction(false, err.message); }); } }; // Reproduction: spawn a new instance // // Viral growth path: user of app decides to make another instance var newInstanceButton = function newInstanceButton() { var button = div.appendChild(dom.createElement('button')); button.textContent = 'Start another pad'; button.addEventListener('click', function () { return showBootstrap(subject, spawnArea, 'pad'); }); return button; }; // Option of either using the workspace system or just typing in a URI var showBootstrap = function showBootstrap(thisInstance, container, noun) { var div = clearElement(container); var appDetails = { noun: 'notepad' }; div.appendChild(_solidUi.authn.newAppInstance(dom, appDetails, initializeNewInstanceInWorkspace)); div.appendChild(dom.createElement('hr')); // @@ var p = div.appendChild(dom.createElement('p')); p.textContent = 'Where would you like to store the data for the ' + noun + '? ' + 'Give the URL of the directory where you would like the data stored.'; var baseField = div.appendChild(dom.createElement('input')); baseField.setAttribute('type', 'text'); baseField.size = 80 // really a string ; baseField.label = 'base URL'; baseField.autocomplete = 'on'; div.appendChild(dom.createElement('br')); // @@ var button = div.appendChild(dom.createElement('button')); button.textContent = 'Start new ' + noun + ' at this URI'; button.addEventListener('click', function (_e) { var newBase = baseField.value; if (newBase.slice(-1) !== '/') { newBase += '/'; } initializeNewInstanceAtBase(thisInstance, newBase); }); }; // Create new document files for new instance of app var initializeNewInstanceInWorkspace = function initializeNewInstanceInWorkspace(ws) { // @@ TODO Clean up type for newBase var newBase = store.any(ws, _solidUi.ns.space('uriPrefix')); if (!newBase) { newBase = ws.uri.split('#')[0]; } else { newBase = newBase.value; } if (newBase.slice(-1) !== '/') { _rdflib.log.error(appPathSegment + ': No / at end of uriPrefix ' + newBase); // @@ paramater? newBase = newBase + '/'; } var now = new Date(); newBase += appPathSegment + '/id' + now.getTime() + '/'; // unique id initializeNewInstanceAtBase(thisInstance, newBase); }; var initializeNewInstanceAtBase = function initializeNewInstanceAtBase(thisInstance, newBase) { var here = (0, _rdflib.sym)(thisInstance.uri.split('#')[0]); var base = here; // @@ ??? var newPadDoc = store.sym(newBase + 'pad.ttl'); var newIndexDoc = store.sym(newBase + 'index.html'); var toBeCopied = [{ local: 'index.html', contentType: 'text/html' }]; var newInstance = store.sym(newPadDoc.uri + '#thisPad'); // log.debug("\n Ready to put " + kb.statementsMatching(undefined, undefined, undefined, there)); //@@ var agenda = []; var f; // @@ This needs some form of visible progress bar for (f = 0; f < toBeCopied.length; f++) { var item = toBeCopied[f]; var fun = function copyItem(item) { agenda.push(function () { var newURI = newBase + item.local; console.log('Copying ' + base + item.local + ' to ' + newURI); var setThatACL = function setThatACL() { setACL(newURI, false, function (ok, message) { if (!ok) { complainIfBad(ok, 'FAILED to set ACL ' + newURI + ' : ' + message); console.log('FAILED to set ACL ' + newURI + ' : ' + message); } else { agenda.shift()(); // beware too much nesting } }); }; store.fetcher.webCopy(base + item.local, newBase + item.local, item.contentType).then(function () { return _solidUi.authn.checkUser(); }).then(function (webId) { me = webId; setThatACL(); })["catch"](function (err) { console.log('FAILED to copy ' + base + item.local + ' : ' + err.message); complainIfBad(false, 'FAILED to copy ' + base + item.local + ' : ' + err.message); }); }); }; fun(item); } agenda.push(function createNewPadDataFile() { store.add(newInstance, _solidUi.ns.rdf('type'), PAD('Notepad'), newPadDoc); // TODO @@ Remove casting of add store.add(newInstance, _solidUi.ns.dc('created'), new Date(), // @@ TODO Remove casting newPadDoc); if (me) { store.add(newInstance, _solidUi.ns.dc('author'), me, newPadDoc); } store.add(newInstance, PAD('next'), newInstance, newPadDoc); // linked list empty // Keep a paper trail @@ Revisit when we have non-public ones @@ Privacy store.add(newInstance, _solidUi.ns.space('inspiration'), thisInstance, padDoc); store.add(newInstance, _solidUi.ns.space('inspiration'), thisInstance, newPadDoc); updater.put(newPadDoc, store.statementsMatching(undefined, undefined, undefined, newPadDoc), 'text/turtle', function (_uri2, ok, message) { if (ok) { agenda.shift()(); } else { complainIfBad(ok, 'FAILED to save new notepad at: ' + newPadDoc.uri + ' : ' + message); console.log('FAILED to save new notepad at: ' + newPadDoc.uri + ' : ' + message); } }); }); agenda.push(function () { setACL(newPadDoc.uri, true, function (ok, body) { complainIfBad(ok, 'Failed to set Read-Write ACL on pad data file: ' + body); if (ok) agenda.shift()(); }); }); agenda.push(function () { // give the user links to the new app var p = div.appendChild(dom.createElement('p')); p.setAttribute('style', 'font-size: 140%;'); p.innerHTML = "Your <a href='" + newIndexDoc.uri + "'><b>new notepad</b></a> is ready. " + "<br/><br/><a href='" + newIndexDoc.uri + "'>Go to new pad</a>"; }); agenda.shift()(); // Created new data files. }; // Update on incoming changes var showResults = function showResults(exists) { console.log('showResults()'); me = _solidUi.authn.currentUser(); _solidUi.authn.checkUser().then(function (webId) { me = webId; }); var title = store.any(subject, _solidUi.ns.dc('title')) || store.any(subject, _solidUi.ns.vcard('fn')); if (paneOptions.solo && typeof window !== 'undefined' && title) { window.document.title = title.value; } options.exists = exists; padEle = _solidUi.pad.notepad(dom, padDoc, subject, me, options); naviMain.appendChild(padEle); var partipationTarget = store.any(subject, _solidUi.ns.meeting('parentMeeting')) || subject; _solidUi.pad.manageParticipation(dom, naviMiddle2, padDoc, partipationTarget, me, options); store.updater.setRefreshHandler(padDoc, padEle.reloadAndSync); // initiated = }; // Read or create empty data file var loadPadData = function loadPadData() { fetcher.nowOrWhenFetched(padDoc.uri, undefined, function (ok, body, response) { if (!ok) { if (response.status === 404) { // / Check explicitly for 404 error console.log('Initializing results file ' + padDoc); updater.put(padDoc, [], 'text/turtle', function (_uri2, ok, message) { if (ok) { clearElement(naviMain); showResults(false); } else { complainIfBad(ok, 'FAILED to create results file at: ' + padDoc.uri + ' : ' + message); console.log('FAILED to craete results file at: ' + padDoc.uri + ' : ' + message); } }); } else { // Other error, not 404 -- do not try to overwite the file complainIfBad(ok, 'FAILED to read results file: ' + body); } } else { // Happy read clearElement(naviMain); if (store.holds(subject, _solidUi.ns.rdf('type'), _solidUi.ns.wf('TemplateInstance'))) { showBootstrap(subject, naviMain, 'pad'); } showResults(true); naviMiddle3.appendChild(newInstanceButton()); } }); }; // Body of Pane var appPathSegment = 'app-pad.timbl.com'; // how to allocate this string and connect to var fetcher = store.fetcher; var updater = store.updater; var me; var PAD = (0, _rdflib.Namespace)('http://www.w3.org/ns/pim/pad#'); var thisInstance = subject; var padDoc = subject.doc(); var padEle; var div = dom.createElement('div'); // Build the DOM var structure = div.appendChild(dom.createElement('table')); // @@ make responsive style structure.setAttribute('style', 'background-color: white; min-width: 94%; margin-right:3% margin-left: 3%; min-height: 13em;'); var naviLoginoutTR = structure.appendChild(dom.createElement('tr')); naviLoginoutTR.appendChild(dom.createElement('td')); // naviLoginout1 naviLoginoutTR.appendChild(dom.createElement('td')); naviLoginoutTR.appendChild(dom.createElement('td')); var naviTop = structure.appendChild(dom.createElement('tr')); // stuff var naviMain = naviTop.appendChild(dom.createElement('td')); naviMain.setAttribute('colspan', '3'); var naviMiddle = structure.appendChild(dom.createElement('tr')); // controls var naviMiddle1 = naviMiddle.appendChild(dom.createElement('td')); var naviMiddle2 = naviMiddle.appendChild(dom.createElement('td')); var naviMiddle3 = naviMiddle.appendChild(dom.createElement('td')); var naviStatus = structure.appendChild(dom.createElement('tr')); // status etc var statusArea = naviStatus.appendChild(dom.createElement('div')); var naviSpawn = structure.appendChild(dom.createElement('tr')); // create new var spawnArea = naviSpawn.appendChild(dom.createElement('div')); var naviMenu = structure.appendChild(dom.createElement('tr')); naviMenu.setAttribute('class', 'naviMenu'); // naviMenu.setAttribute('style', 'margin-top: 3em;'); naviMenu.appendChild(dom.createElement('td')); // naviLeft naviMenu.appendChild(dom.createElement('td')); naviMenu.appendChild(dom.createElement('td')); var options = { statusArea: statusArea, timingArea: naviMiddle1 }; loadPadData(); return div; } }; // ends var _default = paneDef; exports["default"] = _default; //# sourceMappingURL=padPane.js.map