storable = (view_, keyOrOptions, options = {}) => {
validate(
view_ instanceof HTMLElement,
'View is not an HTMLElement. Did you forget to include the `viewof` keyword when passing to `storable`?'
)
const defaultKey = getNotebookKey()
const options_ = {
renderValue: inspect,
stringify: JSON.stringify,
parse: JSON.parse,
key: (typeof keyOrOptions === 'string') ? keyOrOptions : defaultKey,
...((typeof keyOrOptions === 'object') ? keyOrOptions : options),
}
;[
['renderValue', 'function'],
['stringify', 'function'],
['parse', 'function'],
['key', 'string']
].forEach(([key, type]) => validate(typeof options_[key] === type, `options.${key} must be a ${type}`))
const state = {
name: '',
currentList: [],
}
const syncToLocalStorage = () => {
localStorage.setItem(
options_.key,
JSON.stringify(state.currentList.map(([n, v], i, arr) => [n, options_.stringify(v, i, arr)]))
)
}
const syncFromLocalStorage = () => {
const stringList = localStorage.getItem(options_.key)
if (!stringList) {
state.currentList = []
return
}
state.currentList = JSON.parse(stringList).map(([n, vstring], i, arr) => [n, options_.parse(vstring, i, arr)])
}
const update = () => {
syncFromLocalStorage()
const tbody = saveInterface.querySelector('tbody[data-currentlist]')
tbody.replaceChildren(...renderCurrentList())
}
const handleNameInput = e => {
state.name = e.target.value
if (state.name !== '') {
saveInterface.querySelector('button[data-savebutton]').disabled = false
} else {
saveInterface.querySelector('button[data-savebutton]').disabled = true
}
}
const handleSave = e => {
e.preventDefault()
if (state.name == '') return
state.currentList.push([state.name, view_.value])
syncToLocalStorage()
update()
saveInterface.querySelector('input[data-nameinput]').value = ''
state.name = ''
}
const handleDeleteFactory = i => {
return e => {
state.currentList.splice(i, 1)
syncToLocalStorage()
update()
}
}
const handleApplyFactory = i => {
return e => {
e.preventDefault()
const [name, value] = state.currentList[i]
view_.value = value
view_.dispatchEvent(new Event('input'))
}
}
const handleViewInput = () => {
saveInterface.querySelector('td[data-currentvalue]').replaceChildren(
htl.html.fragment`${options_.renderValue(view_.value)}`
)
}
view_.addEventListener('input', handleViewInput)
const renderCurrentList = () => state.currentList.map(([name, value], i, arr) => htl.html.fragment`
<tr>
<td>
<a href="#" onclick=${handleApplyFactory(i)} style="font-weight: bold;">
${name}
</a>
</td>
<td>${options_.renderValue(value, i, arr)}</td>
<td>
<button onclick=${handleDeleteFactory(i)} type="button">Delete</button>
</td>
</tr>
`)
syncFromLocalStorage()
const saveInterface = htl.html`
<form data-saveform onsubmit=${handleSave} action="" style="max-width: 100%; overflow: auto">
<table>
<thead>
<tr>
<th>Name <span style="font-weight: 400">(click to restore)</span></th>
<th>Value</th>
<th></th>
</tr>
</thead>
<tbody data-currentlist>
${renderCurrentList()}
</tbody>
<tfoot>
<tr style="border-top: 1px solid #CCC; border-bottom: 1px solid #CCC;">
<td style="padding-top: 3px">
<label for=${DOM.uid(`name-${options_.key}`)} style="display: block; height: 0; width: 0; overflow: hidden;">Name</label>
<input id=${DOM.uid(`name-${options_.key}`)} data-nameinput oninput=${handleNameInput} type="text" placeholder="Name" style="max-width: 80px" />
</td>
<td style="padding-top: 3px" data-currentvalue>
${options_.renderValue(view_.value)}
</td>
<td style="padding-top: 3px">
<button data-savebutton disabled=${true} type="submit">Save</button>
</td>
</tr>
</tfoot>
</table>
</form>
`
return saveInterface
}