2023-06-18 20:13:35 +02:00
|
|
|
import { h, Component, createRef } from 'preact'
|
2023-06-17 09:11:14 +02:00
|
|
|
import htm from 'htm'
|
|
|
|
import { signal } from 'preact/signals'
|
2023-07-01 20:33:26 +02:00
|
|
|
import { Keys, Key } from 'key'
|
2023-07-12 22:35:38 +02:00
|
|
|
import Crypto from 'crypto'
|
2023-06-17 09:11:14 +02:00
|
|
|
const html = htm.bind(h)
|
|
|
|
|
|
|
|
export class NodeUI extends Component {
|
2023-07-12 22:35:38 +02:00
|
|
|
constructor(props) {//{{{
|
|
|
|
super(props)
|
2023-06-18 22:05:10 +02:00
|
|
|
this.menu = signal(false)
|
2023-06-17 09:11:14 +02:00
|
|
|
this.node = signal(null)
|
2023-06-18 20:13:35 +02:00
|
|
|
this.nodeContent = createRef()
|
2023-07-01 20:33:26 +02:00
|
|
|
this.keys = signal([])
|
2023-06-27 14:44:36 +02:00
|
|
|
|
2023-07-01 20:33:26 +02:00
|
|
|
this.page = signal('node')
|
2023-06-17 09:11:14 +02:00
|
|
|
window.addEventListener('popstate', evt=>{
|
|
|
|
if(evt.state && evt.state.hasOwnProperty('nodeID'))
|
|
|
|
this.goToNode(evt.state.nodeID, true)
|
|
|
|
else
|
|
|
|
this.goToNode(0, true)
|
|
|
|
})
|
2023-06-27 14:44:36 +02:00
|
|
|
|
2023-06-20 07:59:54 +02:00
|
|
|
window.addEventListener('keydown', evt=>this.keyHandler(evt))
|
2023-06-17 09:11:14 +02:00
|
|
|
}//}}}
|
|
|
|
render() {//{{{
|
|
|
|
if(this.node.value === null)
|
|
|
|
return
|
|
|
|
|
|
|
|
let node = this.node.value
|
2023-07-18 06:27:50 +02:00
|
|
|
document.title = `N: ${node.Name}`
|
2023-06-17 09:11:14 +02:00
|
|
|
|
|
|
|
let crumbs = [
|
|
|
|
html`<div class="crumb" onclick=${()=>this.goToNode(0)}>Start</div>`
|
|
|
|
]
|
|
|
|
|
2023-06-18 20:13:35 +02:00
|
|
|
crumbs = crumbs.concat(node.Crumbs.slice(0).map(node=>
|
2023-06-17 09:11:14 +02:00
|
|
|
html`<div class="crumb" onclick=${()=>this.goToNode(node.ID)}>${node.Name}</div>`
|
|
|
|
).reverse())
|
|
|
|
|
|
|
|
let children = node.Children.sort((a,b)=>{
|
2023-07-01 20:33:26 +02:00
|
|
|
if(a.Name.toLowerCase() > b.Name.toLowerCase()) return 1
|
|
|
|
if(a.Name.toLowerCase() < b.Name.toLowerCase()) return -1
|
2023-06-17 09:11:14 +02:00
|
|
|
return 0
|
|
|
|
}).map(child=>html`
|
|
|
|
<div class="child-node" onclick=${()=>this.goToNode(child.ID)}>${child.Name}</div>
|
|
|
|
`)
|
|
|
|
|
2023-06-18 20:13:35 +02:00
|
|
|
let modified = ''
|
|
|
|
if(this.props.app.nodeModified.value)
|
2023-07-01 20:33:26 +02:00
|
|
|
modified = 'modified'
|
|
|
|
|
|
|
|
|
|
|
|
// Page to display
|
|
|
|
let page = ''
|
|
|
|
switch(this.page.value) {
|
|
|
|
case 'node':
|
2023-07-14 16:34:52 +02:00
|
|
|
if(node.ID == 0) {
|
|
|
|
page = html`
|
|
|
|
${children.length > 0 ? html`<div class="child-nodes">${children}</div>` : html``}
|
|
|
|
`
|
|
|
|
} else {
|
2023-07-12 22:35:38 +02:00
|
|
|
let padlock = ''
|
|
|
|
if(node.CryptoKeyID > 0)
|
|
|
|
padlock = html`<img src="/images/${window._VERSION}/padlock-black.svg" style="height: 24px;" />`
|
2023-07-01 20:33:26 +02:00
|
|
|
page = html`
|
|
|
|
${children.length > 0 ? html`<div class="child-nodes">${children}</div>` : html``}
|
2023-07-12 22:35:38 +02:00
|
|
|
<div class="node-name">
|
|
|
|
${node.Name} ${padlock}
|
|
|
|
</div>
|
|
|
|
<${NodeContent} key=${node.ID} node=${node} ref=${this.nodeContent} />
|
2023-07-01 20:33:26 +02:00
|
|
|
<${NodeFiles} node=${this.node.value} />
|
|
|
|
`
|
2023-07-12 22:35:38 +02:00
|
|
|
}
|
2023-07-01 20:33:26 +02:00
|
|
|
break
|
2023-06-18 20:13:35 +02:00
|
|
|
|
2023-07-01 20:33:26 +02:00
|
|
|
case 'upload':
|
|
|
|
page = html`<${UploadUI} nodeui=${this} />`
|
|
|
|
break
|
2023-06-21 23:52:21 +02:00
|
|
|
|
2023-07-01 20:33:26 +02:00
|
|
|
case 'node-properties':
|
|
|
|
page = html`<${NodeProperties} nodeui=${this} />`
|
|
|
|
break
|
|
|
|
|
|
|
|
case 'keys':
|
|
|
|
page = html`<${Keys} nodeui=${this} />`
|
|
|
|
break
|
2023-07-19 10:00:36 +02:00
|
|
|
|
|
|
|
case 'search':
|
|
|
|
page = html`<${Search} nodeui=${this} />`
|
|
|
|
break
|
2023-07-01 20:33:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
let menu = ''
|
2023-06-21 23:52:21 +02:00
|
|
|
if(this.menu.value)
|
2023-07-01 20:33:26 +02:00
|
|
|
menu = html`<${Menu} nodeui=${this} />`
|
2023-06-18 22:05:10 +02:00
|
|
|
|
2023-06-21 23:52:21 +02:00
|
|
|
return html`
|
|
|
|
${menu}
|
2023-06-18 20:13:35 +02:00
|
|
|
<header class="${modified}" onclick=${()=>this.saveNode()}>
|
2023-06-27 15:08:48 +02:00
|
|
|
<div class="tree"><img src="/images/${window._VERSION}/tree.svg" onclick=${()=>document.getElementById('app').classList.toggle('toggle-tree')} /></div>
|
2023-06-18 20:13:35 +02:00
|
|
|
<div class="name">Notes</div>
|
2023-07-12 22:35:38 +02:00
|
|
|
<div class="add" onclick=${evt=>this.createNode(evt)}><img src="/images/${window._VERSION}/add.svg" /></div>
|
|
|
|
<div class="keys" onclick=${evt=>{ evt.stopPropagation(); this.showPage('keys')}}><img src="/images/${window._VERSION}/padlock.svg" /></div>
|
2023-06-20 08:13:32 +02:00
|
|
|
<div class="menu" onclick=${evt=>this.showMenu(evt)}>☰</div>
|
2023-06-18 20:13:35 +02:00
|
|
|
</header>
|
2023-06-18 22:05:10 +02:00
|
|
|
|
2023-06-27 14:44:36 +02:00
|
|
|
<div id="crumbs">
|
|
|
|
<div class="crumbs">${crumbs}</crumbs>
|
|
|
|
</div>
|
2023-06-18 22:05:10 +02:00
|
|
|
|
2023-07-01 20:33:26 +02:00
|
|
|
${page}
|
2023-06-17 09:11:14 +02:00
|
|
|
`
|
|
|
|
}//}}}
|
2023-07-12 22:35:38 +02:00
|
|
|
async componentDidMount() {//{{{
|
|
|
|
// When rendered and fetching the node, keys could be needed in order to
|
|
|
|
// decrypt the content.
|
|
|
|
await this.retrieveKeys()
|
|
|
|
|
2023-06-27 14:44:36 +02:00
|
|
|
this.props.app.startNode.retrieve(node=>{
|
2023-06-17 09:11:14 +02:00
|
|
|
this.node.value = node
|
2023-06-27 14:44:36 +02:00
|
|
|
|
|
|
|
// The tree isn't guaranteed to have loaded yet. This is also run from
|
|
|
|
// the tree code, in case the node hasn't loaded.
|
|
|
|
this.props.app.tree.crumbsUpdateNodes(node)
|
2023-06-17 09:11:14 +02:00
|
|
|
})
|
|
|
|
}//}}}
|
2023-06-18 22:05:10 +02:00
|
|
|
|
2023-06-20 07:59:54 +02:00
|
|
|
keyHandler(evt) {//{{{
|
|
|
|
let handled = true
|
2023-07-19 10:00:36 +02:00
|
|
|
|
|
|
|
// All keybindings is Alt+Shift, since the popular browsers at the time (2023) allows to override thees.
|
|
|
|
// Ctrl+S is the exception to using Alt+Shift, since it is overridable and in such widespread use for saving.
|
|
|
|
// Thus, the exception is acceptable to consequent use of alt+shift.
|
|
|
|
if(!(evt.shiftKey && evt.altKey) && !(evt.key.toUpperCase() == 'S' && evt.ctrlKey))
|
|
|
|
return
|
|
|
|
|
2023-06-20 07:59:54 +02:00
|
|
|
switch(evt.key.toUpperCase()) {
|
2023-07-01 20:33:26 +02:00
|
|
|
case 'E':
|
2023-07-19 10:00:36 +02:00
|
|
|
this.showPage('keys')
|
2023-06-20 07:59:54 +02:00
|
|
|
break
|
|
|
|
|
|
|
|
case 'N':
|
2023-07-19 10:00:36 +02:00
|
|
|
this.createNode()
|
2023-06-20 07:59:54 +02:00
|
|
|
break
|
|
|
|
|
2023-07-01 20:33:26 +02:00
|
|
|
case 'P':
|
2023-07-19 10:00:36 +02:00
|
|
|
this.showPage('node-properties')
|
2023-07-01 20:33:26 +02:00
|
|
|
break
|
|
|
|
|
|
|
|
case 'S':
|
2023-07-19 10:00:36 +02:00
|
|
|
this.saveNode()
|
2023-07-01 20:33:26 +02:00
|
|
|
break
|
|
|
|
|
2023-06-21 23:52:21 +02:00
|
|
|
case 'U':
|
2023-07-19 10:00:36 +02:00
|
|
|
this.showPage('upload')
|
|
|
|
break
|
|
|
|
|
|
|
|
case 'F':
|
|
|
|
this.showPage('search')
|
|
|
|
break
|
2023-06-21 23:52:21 +02:00
|
|
|
|
2023-06-20 07:59:54 +02:00
|
|
|
default:
|
|
|
|
handled = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if(handled) {
|
|
|
|
evt.preventDefault()
|
|
|
|
evt.stopPropagation()
|
|
|
|
}
|
|
|
|
}//}}}
|
2023-06-20 08:13:32 +02:00
|
|
|
showMenu(evt) {//{{{
|
|
|
|
evt.stopPropagation()
|
2023-06-18 22:05:10 +02:00
|
|
|
this.menu.value = true
|
|
|
|
}//}}}
|
2023-06-20 08:13:32 +02:00
|
|
|
logout() {//{{{
|
|
|
|
window.localStorage.removeItem('session.UUID')
|
|
|
|
location.href = '/'
|
|
|
|
}//}}}
|
2023-06-18 22:05:10 +02:00
|
|
|
|
2023-06-17 09:11:14 +02:00
|
|
|
goToNode(nodeID, dontPush) {//{{{
|
2023-06-18 20:13:35 +02:00
|
|
|
if(this.props.app.nodeModified.value) {
|
|
|
|
if(!confirm("Changes not saved. Do you want to discard changes?"))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-06-17 09:11:14 +02:00
|
|
|
if(!dontPush)
|
2023-06-18 20:13:35 +02:00
|
|
|
history.pushState({ nodeID }, '', `/?node=${nodeID}`)
|
2023-06-27 14:44:36 +02:00
|
|
|
|
|
|
|
// New node is fetched in order to retrieve content and files.
|
|
|
|
// Such data is unnecessary to transfer for tree/navigational purposes.
|
2023-06-17 09:11:14 +02:00
|
|
|
let node = new Node(this.props.app, nodeID)
|
|
|
|
node.retrieve(node=>{
|
2023-06-18 20:13:35 +02:00
|
|
|
this.props.app.nodeModified.value = false
|
2023-06-17 09:11:14 +02:00
|
|
|
this.node.value = node
|
2023-07-12 22:35:38 +02:00
|
|
|
this.showPage('node')
|
2023-06-27 14:44:36 +02:00
|
|
|
|
|
|
|
// Tree needs to know another node is selected, in order to render any
|
|
|
|
// previously selected node not selected.
|
|
|
|
this.props.app.tree.setSelected(node)
|
2023-06-27 15:08:48 +02:00
|
|
|
|
|
|
|
// Hide tree toggle, as this would be the next natural action to do manually anyway.
|
|
|
|
// At least in mobile mode.
|
2023-07-01 20:33:26 +02:00
|
|
|
document.getElementById('app').classList.remove('toggle-tree')
|
2023-06-17 09:11:14 +02:00
|
|
|
})
|
|
|
|
}//}}}
|
2023-06-20 08:13:32 +02:00
|
|
|
createNode(evt) {//{{{
|
2023-06-21 23:52:21 +02:00
|
|
|
if(evt)
|
|
|
|
evt.stopPropagation()
|
2023-06-18 20:13:35 +02:00
|
|
|
let name = prompt("Name")
|
|
|
|
if(!name)
|
|
|
|
return
|
2023-06-22 07:10:26 +02:00
|
|
|
this.node.value.create(name, nodeID=>this.goToNode(nodeID))
|
2023-06-18 20:13:35 +02:00
|
|
|
}//}}}
|
2023-06-18 22:05:10 +02:00
|
|
|
saveNode() {//{{{
|
2023-06-20 07:35:34 +02:00
|
|
|
let content = this.nodeContent.current.contentDiv.current.value
|
2023-07-14 16:17:37 +02:00
|
|
|
this.node.value.setContent(content)
|
|
|
|
this.node.value.save(()=>this.props.app.nodeModified.value = false)
|
2023-06-18 22:05:10 +02:00
|
|
|
}//}}}
|
|
|
|
renameNode() {//{{{
|
|
|
|
let name = prompt("New name")
|
|
|
|
if(!name)
|
|
|
|
return
|
|
|
|
|
2023-06-22 07:10:26 +02:00
|
|
|
this.node.value.rename(name, ()=>{
|
2023-06-18 22:05:10 +02:00
|
|
|
this.goToNode(this.node.value.ID)
|
|
|
|
this.menu.value = false
|
|
|
|
})
|
|
|
|
}//}}}
|
|
|
|
deleteNode() {//{{{
|
|
|
|
if(!confirm("Do you want to delete this note and all sub-notes?"))
|
|
|
|
return
|
2023-06-22 07:10:26 +02:00
|
|
|
this.node.value.delete(()=>{
|
2023-06-18 22:05:10 +02:00
|
|
|
this.goToNode(this.node.value.ParentID)
|
|
|
|
this.menu.value = false
|
|
|
|
})
|
|
|
|
}//}}}
|
2023-07-01 20:33:26 +02:00
|
|
|
|
2023-07-12 22:35:38 +02:00
|
|
|
async retrieveKeys() {//{{{
|
|
|
|
return new Promise((resolve, reject)=>{
|
|
|
|
this.props.app.request('/key/retrieve', {})
|
|
|
|
.then(res=>{
|
|
|
|
this.keys.value = res.Keys.map(keyData=>new Key(keyData, this.keyCounter))
|
|
|
|
resolve(this.keys.value)
|
|
|
|
})
|
|
|
|
.catch(reject)
|
|
|
|
})
|
|
|
|
}//}}}
|
|
|
|
keyCounter() {//{{{
|
|
|
|
return window._app.current.request('/key/counter', {})
|
|
|
|
.then(res=>BigInt(res.Counter))
|
|
|
|
.catch(window._app.current.responseError)
|
|
|
|
}//}}}
|
|
|
|
getKey(id) {//{{{
|
|
|
|
let keys = this.keys.value
|
|
|
|
for(let i = 0; i < keys.length; i++)
|
|
|
|
if(keys[i].ID == id)
|
|
|
|
return keys[i]
|
|
|
|
return null
|
|
|
|
}//}}}
|
|
|
|
|
2023-07-01 20:33:26 +02:00
|
|
|
showPage(pg) {//{{{
|
|
|
|
this.page.value = pg
|
|
|
|
}//}}}
|
2023-06-18 20:13:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class NodeContent extends Component {
|
|
|
|
constructor(props) {//{{{
|
|
|
|
super(props)
|
|
|
|
this.contentDiv = createRef()
|
|
|
|
this.state = {
|
|
|
|
modified: false,
|
|
|
|
}
|
|
|
|
}//}}}
|
2023-07-12 22:35:38 +02:00
|
|
|
render({ node }) {//{{{
|
|
|
|
let content = ''
|
|
|
|
try {
|
|
|
|
content = node.content()
|
|
|
|
} catch(err) {
|
|
|
|
return html`
|
|
|
|
<div id="node-content" class="node-content encrypted">${err.message}</div>
|
|
|
|
`
|
|
|
|
}
|
|
|
|
|
2023-06-20 07:35:34 +02:00
|
|
|
return html`
|
2023-06-27 14:44:36 +02:00
|
|
|
<div class="grow-wrap">
|
|
|
|
<textarea id="node-content" class="node-content" ref=${this.contentDiv} oninput=${()=>this.contentChanged()} required rows=1>${content}</textarea>
|
|
|
|
</div>
|
2023-06-20 07:35:34 +02:00
|
|
|
`
|
|
|
|
}//}}}
|
|
|
|
componentDidMount() {//{{{
|
|
|
|
this.resize()
|
2023-06-27 14:44:36 +02:00
|
|
|
window.addEventListener('resize', ()=>this.resize())
|
2023-06-20 07:35:34 +02:00
|
|
|
}//}}}
|
|
|
|
componentDidUpdate() {//{{{
|
|
|
|
this.resize()
|
|
|
|
}//}}}
|
|
|
|
contentChanged() {//{{{
|
|
|
|
window._app.current.nodeModified.value = true
|
|
|
|
this.resize()
|
|
|
|
}//}}}
|
|
|
|
resize() {//{{{
|
2023-06-27 14:44:36 +02:00
|
|
|
let textarea = document.getElementById('node-content')
|
|
|
|
if(textarea)
|
|
|
|
textarea.parentNode.dataset.replicatedValue = textarea.value
|
2023-06-18 20:13:35 +02:00
|
|
|
}//}}}
|
2023-07-12 22:35:38 +02:00
|
|
|
unlock() {//{{{
|
2023-07-14 16:34:52 +02:00
|
|
|
let pass = prompt(`Password for "${this.props.model.description}"`)
|
2023-07-12 22:35:38 +02:00
|
|
|
if(!pass)
|
|
|
|
return
|
|
|
|
|
|
|
|
try {
|
|
|
|
this.props.model.unlock(pass)
|
|
|
|
this.forceUpdate()
|
|
|
|
} catch(err) {
|
|
|
|
alert(err)
|
|
|
|
}
|
|
|
|
}//}}}
|
2023-06-17 09:11:14 +02:00
|
|
|
}
|
|
|
|
|
2023-06-22 06:52:27 +02:00
|
|
|
class NodeFiles extends Component {
|
2023-06-22 08:28:51 +02:00
|
|
|
render({ node }) {//{{{
|
|
|
|
if(node.Files === null || node.Files.length == 0)
|
|
|
|
return
|
|
|
|
|
|
|
|
let files = node.Files
|
|
|
|
.sort((a, b)=>{
|
|
|
|
if(a.Filename.toUpperCase() < b.Filename.toUpperCase()) return -1
|
|
|
|
if(a.Filename.toUpperCase() > b.Filename.toUpperCase()) return 1
|
|
|
|
return 0
|
|
|
|
})
|
|
|
|
.map(file=>
|
|
|
|
html`
|
2023-06-22 16:48:31 +02:00
|
|
|
<div class="filename" onclick=${()=>node.download(file.ID)}>${file.Filename}</div>
|
2023-06-22 08:28:51 +02:00
|
|
|
<div class="size">${this.formatSize(file.Size)}</div>
|
|
|
|
`
|
|
|
|
)
|
|
|
|
|
|
|
|
return html`
|
|
|
|
<div id="file-section">
|
|
|
|
<div class="header">Files</div>
|
|
|
|
<div class="files">
|
|
|
|
${files}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`
|
|
|
|
}//}}}
|
|
|
|
formatSize(size) {//{{{
|
|
|
|
if(size < 1048576) {
|
|
|
|
return `${Math.round(size / 1024)} KiB`
|
|
|
|
} else {
|
|
|
|
return `${Math.round(size / 1048576)} MiB`
|
|
|
|
}
|
|
|
|
}//}}}
|
2023-06-22 06:52:27 +02:00
|
|
|
}
|
|
|
|
|
2023-06-27 14:44:36 +02:00
|
|
|
export class Node {
|
2023-06-17 09:11:14 +02:00
|
|
|
constructor(app, nodeID) {//{{{
|
2023-06-27 14:44:36 +02:00
|
|
|
this.app = app
|
|
|
|
this.ID = nodeID
|
|
|
|
this.ParentID = 0
|
|
|
|
this.UserID = 0
|
2023-07-12 22:35:38 +02:00
|
|
|
this.CryptoKeyID = 0
|
2023-06-27 14:44:36 +02:00
|
|
|
this.Name = ''
|
2023-07-12 22:35:38 +02:00
|
|
|
this._content = ''
|
2023-06-27 14:44:36 +02:00
|
|
|
this.Children = []
|
|
|
|
this.Crumbs = []
|
|
|
|
this.Files = []
|
2023-07-12 22:35:38 +02:00
|
|
|
this._decrypted = false
|
2023-06-27 14:44:36 +02:00
|
|
|
this._expanded = false // start value for the TreeNode component,
|
|
|
|
// it doesn't control it afterwards.
|
|
|
|
// Used to expand the crumbs upon site loading.
|
2023-06-17 09:11:14 +02:00
|
|
|
}//}}}
|
|
|
|
retrieve(callback) {//{{{
|
|
|
|
this.app.request('/node/retrieve', { ID: this.ID })
|
|
|
|
.then(res=>{
|
2023-07-12 22:35:38 +02:00
|
|
|
this.ParentID = res.Node.ParentID
|
|
|
|
this.UserID = res.Node.UserID
|
|
|
|
this.CryptoKeyID = res.Node.CryptoKeyID
|
|
|
|
this.Name = res.Node.Name
|
|
|
|
this._content = res.Node.Content
|
|
|
|
this.Children = res.Node.Children
|
|
|
|
this.Crumbs = res.Node.Crumbs
|
|
|
|
this.Files = res.Node.Files
|
2023-06-17 09:11:14 +02:00
|
|
|
callback(this)
|
|
|
|
})
|
|
|
|
.catch(this.app.responseError)
|
|
|
|
}//}}}
|
2023-06-22 07:10:26 +02:00
|
|
|
delete(callback) {//{{{
|
|
|
|
this.app.request('/node/delete', {
|
|
|
|
NodeID: this.ID,
|
|
|
|
})
|
|
|
|
.then(callback)
|
|
|
|
.catch(this.app.responseError)
|
|
|
|
}//}}}
|
|
|
|
create(name, callback) {//{{{
|
|
|
|
this.app.request('/node/create', {
|
|
|
|
Name: name.trim(),
|
|
|
|
ParentID: this.ID,
|
|
|
|
})
|
|
|
|
.then(res=>{
|
|
|
|
callback(res.Node.ID)
|
|
|
|
})
|
|
|
|
.catch(this.app.responseError)
|
|
|
|
}//}}}
|
2023-07-14 16:17:37 +02:00
|
|
|
async save(callback) {//{{{
|
|
|
|
try {
|
|
|
|
await this.#encrypt()
|
2023-07-12 22:35:38 +02:00
|
|
|
|
2023-07-14 16:17:37 +02:00
|
|
|
let req = {
|
|
|
|
NodeID: this.ID,
|
|
|
|
Content: this._content,
|
|
|
|
CryptoKeyID: this.CryptoKeyID,
|
|
|
|
}
|
|
|
|
this.app.request('/node/update', req)
|
|
|
|
.then(callback)
|
|
|
|
.catch(this.app.responseError)
|
|
|
|
} catch (err) {
|
|
|
|
this.app.responseError(err)
|
|
|
|
}
|
2023-06-22 07:10:26 +02:00
|
|
|
}//}}}
|
|
|
|
rename(name, callback) {//{{{
|
|
|
|
this.app.request('/node/rename', {
|
|
|
|
Name: name.trim(),
|
|
|
|
NodeID: this.ID,
|
|
|
|
})
|
|
|
|
.then(callback)
|
|
|
|
.catch(this.app.responseError)
|
|
|
|
}//}}}
|
2023-06-22 16:48:31 +02:00
|
|
|
download(fileID) {//{{{
|
|
|
|
let headers = {
|
|
|
|
'Content-Type': 'application/json',
|
|
|
|
}
|
|
|
|
|
|
|
|
if(this.app.session.UUID !== '')
|
|
|
|
headers['X-Session-Id'] = this.app.session.UUID
|
|
|
|
|
|
|
|
let fname = ""
|
|
|
|
fetch("/node/download", {
|
|
|
|
method: 'POST',
|
|
|
|
headers,
|
|
|
|
body: JSON.stringify({
|
|
|
|
NodeID: this.ID,
|
|
|
|
FileID: fileID,
|
|
|
|
}),
|
|
|
|
})
|
|
|
|
.then(response=>{
|
2023-06-22 17:42:34 +02:00
|
|
|
let match = response.headers.get('content-disposition').match(/filename="([^"]*)"/)
|
|
|
|
fname = match[1]
|
2023-06-22 16:48:31 +02:00
|
|
|
return response.blob()
|
|
|
|
})
|
|
|
|
.then(blob=>{
|
|
|
|
let url = window.URL.createObjectURL(blob)
|
2023-07-01 20:33:26 +02:00
|
|
|
let a = document.createElement('a')
|
|
|
|
a.href = url
|
|
|
|
a.download = fname
|
|
|
|
document.body.appendChild(a) // we need to append the element to the dom -> otherwise it will not work in firefox
|
|
|
|
a.click()
|
|
|
|
a.remove() //afterwards we remove the element again
|
2023-06-22 16:48:31 +02:00
|
|
|
})
|
|
|
|
}//}}}
|
2023-07-12 22:35:38 +02:00
|
|
|
content() {//{{{
|
2023-07-14 16:17:37 +02:00
|
|
|
if(this.CryptoKeyID != 0 && !this._decrypted)
|
2023-07-12 22:35:38 +02:00
|
|
|
this.#decrypt()
|
|
|
|
return this._content
|
|
|
|
}//}}}
|
2023-07-14 16:17:37 +02:00
|
|
|
setContent(new_content) {//{{{
|
|
|
|
this._content = new_content
|
|
|
|
if(this.CryptoKeyID == 0)
|
|
|
|
// Logic behind plaintext not being decrypted is that
|
|
|
|
// only encrypted values can be in a decrypted state.
|
|
|
|
this._decrypted = false
|
|
|
|
else
|
|
|
|
this._decrypted = true
|
|
|
|
}//}}}
|
|
|
|
async setCryptoKey(new_key) {//{{{
|
|
|
|
return this.#encrypt(true, new_key)
|
2023-07-12 22:35:38 +02:00
|
|
|
}//}}}
|
|
|
|
#decrypt() {//{{{
|
2023-07-14 16:17:37 +02:00
|
|
|
if(this.CryptoKeyID == 0 || this._decrypted)
|
|
|
|
return
|
|
|
|
|
2023-07-12 22:35:38 +02:00
|
|
|
let obj_key = this.app.nodeUI.current.getKey(this.CryptoKeyID)
|
|
|
|
if(obj_key === null || obj_key.ID != this.CryptoKeyID)
|
|
|
|
throw('Invalid key')
|
|
|
|
|
|
|
|
// Ask user to unlock key first
|
|
|
|
var pass = null
|
|
|
|
while(pass || obj_key.status() == 'locked') {
|
2023-07-14 16:34:52 +02:00
|
|
|
pass = prompt(`Password for "${obj_key.description}"`)
|
2023-07-12 22:35:38 +02:00
|
|
|
if(!pass)
|
|
|
|
throw new Error(`Key "${obj_key.description}" is locked`)
|
|
|
|
|
|
|
|
try {
|
|
|
|
obj_key.unlock(pass)
|
|
|
|
} catch(err) {
|
|
|
|
alert(err)
|
|
|
|
}
|
|
|
|
pass = null
|
|
|
|
}
|
|
|
|
|
|
|
|
if(obj_key.status() == 'locked')
|
|
|
|
throw new Error(`Key "${obj_key.description}" is locked`)
|
|
|
|
|
|
|
|
let crypto = new Crypto(obj_key.key)
|
|
|
|
this._decrypted = true
|
|
|
|
this._content = sjcl.codec.utf8String.fromBits(
|
|
|
|
crypto.decrypt(this._content)
|
|
|
|
)
|
|
|
|
}//}}}
|
2023-07-14 16:17:37 +02:00
|
|
|
async #encrypt(change_key = false, new_key = null) {//{{{
|
|
|
|
// Nothing to do if not changing key and already encrypted.
|
|
|
|
if(!change_key && this.CryptoKeyID != 0 && !this._decrypted)
|
|
|
|
return this._content
|
|
|
|
|
|
|
|
let content = this.content()
|
|
|
|
|
|
|
|
// Changing key to no encryption or already at no encryption -
|
|
|
|
// set to not decrypted (only encrypted values can be
|
|
|
|
// decrypted) and return plain value.
|
|
|
|
if((change_key && new_key === null) || (!change_key && this.CryptoKeyID == 0)) {
|
|
|
|
this._decrypted = false
|
|
|
|
this.CryptoKeyID = 0
|
|
|
|
return content
|
|
|
|
}
|
|
|
|
|
|
|
|
let key_id = change_key ? new_key.ID : this.CryptoKeyID
|
|
|
|
let obj_key = this.app.nodeUI.current.getKey(key_id)
|
|
|
|
if(obj_key === null || obj_key.ID != key_id)
|
2023-07-12 22:35:38 +02:00
|
|
|
throw('Invalid key')
|
|
|
|
|
|
|
|
if(obj_key.status() == 'locked')
|
|
|
|
throw new Error(`Key "${obj_key.description}" is locked`)
|
|
|
|
|
|
|
|
let crypto = new Crypto(obj_key.key)
|
|
|
|
let content_bits = sjcl.codec.utf8String.toBits(content)
|
|
|
|
let counter = await this.app.nodeUI.current.keyCounter()
|
2023-07-14 16:17:37 +02:00
|
|
|
this.CryptoKeyID = obj_key.ID
|
|
|
|
this._content = crypto.encrypt(content_bits, counter, true)
|
|
|
|
this._decrypted = false
|
|
|
|
return this._content
|
2023-07-12 22:35:38 +02:00
|
|
|
}//}}}
|
2023-06-17 09:11:14 +02:00
|
|
|
}
|
|
|
|
|
2023-06-21 23:52:21 +02:00
|
|
|
class Menu extends Component {
|
2023-06-22 06:52:27 +02:00
|
|
|
render({ nodeui }) {//{{{
|
2023-06-21 23:52:21 +02:00
|
|
|
return html`
|
2023-06-22 06:52:27 +02:00
|
|
|
<div id="blackout" onclick=${()=>nodeui.menu.value = false}></div>
|
|
|
|
<div id="menu">
|
|
|
|
<div class="item" onclick=${()=>{ nodeui.renameNode(); nodeui.menu.value = false }}>Rename</div>
|
2023-07-01 20:33:26 +02:00
|
|
|
<div class="item" onclick=${()=>{ nodeui.deleteNode(); nodeui.menu.value = false }}>Delete</div>
|
|
|
|
<div class="item separator" onclick=${()=>{ nodeui.showPage('properties'); nodeui.menu.value = false }}>Properties</div>
|
2023-06-22 06:52:27 +02:00
|
|
|
<div class="item separator" onclick=${()=>{ nodeui.upload.value = true; nodeui.menu.value = false }}>Upload</div>
|
|
|
|
<div class="item" onclick=${()=>{ nodeui.logout(); nodeui.menu.value = false }}>Log out</div>
|
2023-06-21 23:52:21 +02:00
|
|
|
</div>
|
|
|
|
`
|
|
|
|
}//}}}
|
|
|
|
}
|
|
|
|
class UploadUI extends Component {
|
2023-06-22 08:28:51 +02:00
|
|
|
constructor(props) {//{{{
|
|
|
|
super(props)
|
2023-06-21 23:52:21 +02:00
|
|
|
this.file = createRef()
|
|
|
|
this.filelist = signal([])
|
|
|
|
this.fileRefs = []
|
|
|
|
this.progressRefs = []
|
|
|
|
}//}}}
|
|
|
|
render({ nodeui }) {//{{{
|
|
|
|
let filelist = this.filelist.value
|
|
|
|
let files = []
|
|
|
|
for(let i = 0; i < filelist.length; i++) {
|
|
|
|
files.push(html`<div key=file_${i} ref=${this.fileRefs[i]} class="file">${filelist.item(i).name}</div><div class="progress" ref=${this.progressRefs[i]}></div>`)
|
|
|
|
}
|
|
|
|
|
|
|
|
return html`
|
|
|
|
<div id="blackout" onclick=${()=>nodeui.upload.value = false}></div>
|
|
|
|
<div id="upload">
|
|
|
|
<input type="file" ref=${this.file} onchange=${()=>this.upload()} multiple />
|
|
|
|
<div class="files">
|
|
|
|
${files}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
`
|
|
|
|
}//}}}
|
|
|
|
componentDidMount() {//{{{
|
|
|
|
this.file.current.focus()
|
|
|
|
}//}}}
|
|
|
|
|
|
|
|
upload() {//{{{
|
2023-06-22 06:52:27 +02:00
|
|
|
let nodeID = this.props.nodeui.node.value.ID
|
2023-06-21 23:52:21 +02:00
|
|
|
this.fileRefs = []
|
|
|
|
this.progressRefs = []
|
|
|
|
|
|
|
|
let input = this.file.current
|
|
|
|
this.filelist.value = input.files
|
|
|
|
for(let i = 0; i < input.files.length; i++) {
|
|
|
|
this.fileRefs.push(createRef())
|
|
|
|
this.progressRefs.push(createRef())
|
2023-06-22 06:52:27 +02:00
|
|
|
|
|
|
|
this.postFile(
|
|
|
|
input.files[i],
|
|
|
|
nodeID,
|
|
|
|
progress=>{
|
|
|
|
this.progressRefs[i].current.innerHTML = `${progress}%`
|
|
|
|
},
|
2023-06-22 08:28:51 +02:00
|
|
|
res=>{
|
|
|
|
this.props.nodeui.node.value.Files.push(res.File)
|
|
|
|
this.props.nodeui.forceUpdate()
|
|
|
|
|
2023-06-22 06:52:27 +02:00
|
|
|
this.fileRefs[i].current.classList.add("done")
|
|
|
|
this.progressRefs[i].current.classList.add("done")
|
2023-06-22 08:28:51 +02:00
|
|
|
|
|
|
|
this.props.nodeui.upload.value = false
|
2023-06-22 06:52:27 +02:00
|
|
|
})
|
2023-06-21 23:52:21 +02:00
|
|
|
}
|
|
|
|
}//}}}
|
2023-06-22 06:52:27 +02:00
|
|
|
postFile(file, nodeID, progressCallback, doneCallback) {//{{{
|
2023-06-21 23:52:21 +02:00
|
|
|
var formdata = new FormData()
|
|
|
|
formdata.append('file', file)
|
2023-06-22 06:52:27 +02:00
|
|
|
formdata.append('NodeID', nodeID)
|
2023-06-21 23:52:21 +02:00
|
|
|
|
|
|
|
var request = new XMLHttpRequest()
|
|
|
|
|
|
|
|
request.addEventListener("error", ()=>{
|
|
|
|
window._app.current.responseError({ upload: "An unknown error occured" })
|
|
|
|
})
|
|
|
|
|
|
|
|
request.addEventListener("loadend", ()=>{
|
|
|
|
if(request.status != 200) {
|
|
|
|
window._app.current.responseError({ upload: request.statusText })
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let response = JSON.parse(request.response)
|
|
|
|
if(!response.OK) {
|
|
|
|
window._app.current.responseError({ upload: response.Error })
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-06-22 08:28:51 +02:00
|
|
|
doneCallback(response)
|
2023-06-21 23:52:21 +02:00
|
|
|
})
|
|
|
|
|
|
|
|
request.upload.addEventListener('progress', evt=>{
|
|
|
|
var fileSize = file.size
|
|
|
|
|
|
|
|
if(evt.loaded <= fileSize)
|
|
|
|
progressCallback(Math.round(evt.loaded / fileSize * 100))
|
|
|
|
if(evt.loaded == evt.total)
|
|
|
|
progressCallback(100)
|
|
|
|
})
|
|
|
|
|
|
|
|
request.open('post', '/node/upload')
|
|
|
|
request.setRequestHeader("X-Session-Id", window._app.current.session.UUID)
|
|
|
|
//request.timeout = 45000
|
|
|
|
request.send(formdata)
|
|
|
|
}//}}}
|
|
|
|
}
|
2023-07-01 20:33:26 +02:00
|
|
|
class NodeProperties extends Component {
|
|
|
|
constructor(props) {//{{{
|
|
|
|
super(props)
|
2023-07-12 22:35:38 +02:00
|
|
|
this.props.nodeui.retrieveKeys()
|
|
|
|
this.selected_key_id = 0
|
2023-07-01 20:33:26 +02:00
|
|
|
}//}}}
|
|
|
|
render({ nodeui }) {//{{{
|
2023-07-12 22:35:38 +02:00
|
|
|
let save = true
|
|
|
|
let keys = nodeui.keys.value
|
|
|
|
.sort((a, b)=>{
|
|
|
|
if(a.description < b.description) return -1
|
|
|
|
if(a.description > b.description) return 1
|
|
|
|
return 0
|
|
|
|
})
|
|
|
|
.map(key=>{
|
|
|
|
this.props.nodeui.keys.value.some(uikey=>{
|
|
|
|
if(uikey.ID == nodeui.node.value.ID) {
|
|
|
|
save = (uikey.status() == 'unlocked')
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return html`
|
|
|
|
<div class="key ${key.status()}">
|
|
|
|
<input type="radio" name="key" id="key-${key.ID}" checked=${nodeui.node.value.CryptoKeyID == key.ID} disabled=${key.status() == 'locked'} oninput=${()=>this.selected_key_id = key.ID} />
|
|
|
|
<label for="key-${key.ID}">${key.description}</label>
|
|
|
|
</div>`
|
|
|
|
})
|
|
|
|
|
2023-07-01 20:33:26 +02:00
|
|
|
return html`
|
|
|
|
<div id="properties">
|
2023-07-12 22:35:38 +02:00
|
|
|
<h1>Note properties</h1>
|
|
|
|
|
|
|
|
These properties are only for this note.
|
|
|
|
|
|
|
|
<h2>Encryption</h2>
|
|
|
|
<div class="key">
|
|
|
|
<input type="radio" id="key-none" name="key" checked=${nodeui.node.value.CryptoKeyID == 0} oninput=${()=>this.selected_key_id = 0} />
|
|
|
|
<label for="key-none">None</label>
|
|
|
|
</div>
|
|
|
|
${keys}
|
|
|
|
|
|
|
|
${save ? html`<button style="margin-top: 32px" onclick=${()=>this.save()}>Save</button>` : ''}
|
2023-07-01 20:33:26 +02:00
|
|
|
</div>
|
|
|
|
`
|
|
|
|
}//}}}
|
2023-07-14 16:17:37 +02:00
|
|
|
async save() {//{{{
|
2023-07-12 22:35:38 +02:00
|
|
|
let nodeui = this.props.nodeui
|
|
|
|
let node = nodeui.node.value
|
|
|
|
|
|
|
|
// Find the actual key object used for encryption
|
2023-07-14 16:17:37 +02:00
|
|
|
let new_key = nodeui.getKey(this.selected_key_id)
|
|
|
|
let current_key = nodeui.getKey(node.CryptoKeyID)
|
2023-07-12 22:35:38 +02:00
|
|
|
|
2023-07-14 16:17:37 +02:00
|
|
|
if(current_key && current_key.status() == 'locked') {
|
2023-07-12 22:35:38 +02:00
|
|
|
alert("Decryption key is locked and can not be used.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-07-14 16:17:37 +02:00
|
|
|
if(new_key && new_key.status() == 'locked') {
|
2023-07-12 22:35:38 +02:00
|
|
|
alert("Key is locked and can not be used.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-07-14 16:17:37 +02:00
|
|
|
await node.setCryptoKey(new_key)
|
|
|
|
node.save(()=>this.props.nodeui.showPage('node'))
|
2023-07-12 22:35:38 +02:00
|
|
|
}//}}}
|
2023-07-01 20:33:26 +02:00
|
|
|
}
|
2023-06-21 23:52:21 +02:00
|
|
|
|
2023-07-19 10:00:36 +02:00
|
|
|
class Search extends Component {
|
|
|
|
constructor() {//{{{
|
|
|
|
super()
|
|
|
|
this.state = {
|
|
|
|
matches: [],
|
|
|
|
results_returned: false,
|
|
|
|
}
|
|
|
|
}//}}}
|
|
|
|
render({ nodeui }, { matches, results_returned }) {//{{{
|
|
|
|
let match_elements = [
|
|
|
|
html`<h2>Results</h2>`,
|
|
|
|
]
|
|
|
|
let matched_nodes = matches.map(node=>html`
|
|
|
|
<div class="matched-node" onclick=${()=>nodeui.goToNode(node.ID)}>
|
|
|
|
${node.Name}
|
|
|
|
</div>
|
|
|
|
`)
|
|
|
|
match_elements.push(html`<div class="matches">${matched_nodes}</div>`)
|
|
|
|
|
|
|
|
return html`
|
|
|
|
<div id="search">
|
|
|
|
<h1>Search</h1>
|
|
|
|
|
|
|
|
<input type="text" id="search-for" placeholder="Search for" onkeydown=${evt=>this.keyHandler(evt)} />
|
|
|
|
<button onclick=${()=>this.search()}>Search</button>
|
|
|
|
|
|
|
|
${results_returned ? match_elements : ''}
|
|
|
|
</div>`
|
|
|
|
}//}}}
|
|
|
|
componentDidMount() {//{{{
|
|
|
|
document.getElementById('search-for').focus()
|
|
|
|
}//}}}
|
|
|
|
|
|
|
|
keyHandler(evt) {//{{{
|
|
|
|
let handled = true
|
|
|
|
|
|
|
|
switch(evt.key.toUpperCase()) {
|
|
|
|
case 'ENTER':
|
|
|
|
this.search()
|
|
|
|
break
|
|
|
|
|
|
|
|
default:
|
|
|
|
handled = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if(handled) {
|
|
|
|
evt.preventDefault()
|
|
|
|
evt.stopPropagation()
|
|
|
|
}
|
|
|
|
}//}}}
|
|
|
|
search() {//{{{
|
|
|
|
let Search = document.getElementById('search-for').value
|
|
|
|
|
|
|
|
window._app.current.request('/node/search', { Search })
|
|
|
|
.then(res=>{
|
|
|
|
this.setState({
|
|
|
|
matches: res.Nodes,
|
|
|
|
results_returned: true,
|
|
|
|
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.catch(window._app.current.responseError)
|
|
|
|
}//}}}
|
|
|
|
}
|
|
|
|
|
2023-06-17 09:11:14 +02:00
|
|
|
// vim: foldmethod=marker
|