Small improvements:
- Switch to CodeMirror - Switch to Parcel Bundler - Fix synchronisation bug - Update dependencies
							
								
								
									
										9
									
								
								Caddyfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,9 @@ | |||||||
|  | https://notes.dev.hibas123.de:443 { | ||||||
|  |    proxy / localhost:1234/ | ||||||
|  |    tls D:\Certs\_wildcard.dev.hibas123.de.pem D:\Certs\_wildcard.dev.hibas123.de-key.pem | ||||||
|  | } | ||||||
|  |  | ||||||
|  | https://sf.dev.hibas123.de:443 { | ||||||
|  |    proxy / localhost:3004/ | ||||||
|  |    tls D:\Certs\_wildcard.dev.hibas123.de.pem D:\Certs\_wildcard.dev.hibas123.de-key.pem | ||||||
|  | } | ||||||
							
								
								
									
										10080
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
							
								
								
									
										48
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @ -4,48 +4,34 @@ | |||||||
|    "description": "", |    "description": "", | ||||||
|    "main": "index.js", |    "main": "index.js", | ||||||
|    "scripts": { |    "scripts": { | ||||||
|       "build": "webpack --mode=production", |       "build": "parcel build src/index.html", | ||||||
|       "watch": "webpack --mode=development --watch", |       "dev": "parcel src/index.html" | ||||||
|       "start": "webpack-dev-server --mode=production", |  | ||||||
|       "start:prod": "webpack-dev-server --mode=production", |  | ||||||
|       "start:dev": "webpack-dev-server --mode=development" |  | ||||||
|    }, |    }, | ||||||
|    "author": "Fabian Stamm <dev@fabianstamm.de>", |    "author": "Fabian Stamm <dev@fabianstamm.de>", | ||||||
|    "license": "MIT", |    "license": "MIT", | ||||||
|  |    "browserslist": [ | ||||||
|  |       "last 2 Chrome versions" | ||||||
|  |    ], | ||||||
|    "dependencies": { |    "dependencies": { | ||||||
|       "@hibas123/secure-file-wrapper": "^2.5.0", |       "@hibas123/secure-file-wrapper": "^2.5.1", | ||||||
|       "@hibas123/theme": "^1.0.5", |       "@hibas123/theme": "^1.0.5", | ||||||
|       "@hibas123/utils": "^2.1.0", |       "@hibas123/utils": "^2.2.17", | ||||||
|       "aes-js": "^3.1.2", |       "aes-js": "^3.1.2", | ||||||
|       "feather-icons": "^4.22.1", |       "codemirror": "^5.58.3", | ||||||
|       "idb": "^4.0.3", |       "feather-icons": "^4.28.0", | ||||||
|  |       "idb": "^5.0.7", | ||||||
|       "js-sha256": "^0.9.0", |       "js-sha256": "^0.9.0", | ||||||
|       "lodash.clonedeep": "^4.5.0", |       "lodash.clonedeep": "^4.5.0", | ||||||
|  |       "parcel": "^1.12.4", | ||||||
|  |       "preact": "^10.5.7", | ||||||
|  |       "preact-feather": "^4.1.0", | ||||||
|       "secure-file-wrapper": "git+https://git.stamm.me/OpenServer/OSSecureFileWrapper.git", |       "secure-file-wrapper": "git+https://git.stamm.me/OpenServer/OSSecureFileWrapper.git", | ||||||
|       "uuid": "^3.3.2" |       "uuid": "^8.3.1" | ||||||
|    }, |    }, | ||||||
|    "devDependencies": { |    "devDependencies": { | ||||||
|  |       "@types/codemirror": "0.0.99", | ||||||
|       "@types/lodash.clonedeep": "^4.5.6", |       "@types/lodash.clonedeep": "^4.5.6", | ||||||
|       "@types/uikit": "^2.27.7", |       "@types/uuid": "^8.3.0", | ||||||
|       "@types/uuid": "^3.4.5", |       "typescript": "^4.1.2" | ||||||
|       "copy-webpack-plugin": "^5.0.3", |  | ||||||
|       "css-loader": "^3.0.0", |  | ||||||
|       "file-loader": "^4.0.0", |  | ||||||
|       "html-webpack-plugin": "^3.2.0", |  | ||||||
|       "mini-css-extract-plugin": "^0.7.0", |  | ||||||
|       "node-sass": "^4.12.0", |  | ||||||
|       "preact": "^8.3.1", |  | ||||||
|       "preact-svg-loader": "^0.2.1", |  | ||||||
|       "raw-loader": "^3.0.0", |  | ||||||
|       "sass-loader": "^7.1.0", |  | ||||||
|       "style-loader": "^0.23.1", |  | ||||||
|       "ts-loader": "^6.0.4", |  | ||||||
|       "typescript": "^3.5.2", |  | ||||||
|       "webpack": "^4.35.2", |  | ||||||
|       "webpack-bundle-analyzer": "^3.3.2", |  | ||||||
|       "webpack-cli": "^3.3.5", |  | ||||||
|       "webpack-dev-server": "^3.7.2", |  | ||||||
|       "webpack-visualizer-plugin": "^0.1.11", |  | ||||||
|       "worker-loader": "^2.0.0" |  | ||||||
|    } |    } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,55 +0,0 @@ | |||||||
| { |  | ||||||
|     "short_name": "Secure Notes", |  | ||||||
|     "name": "Secure Notes", |  | ||||||
|     "decription": "A place to store your notes securly", |  | ||||||
|     "share_target": { |  | ||||||
|         "action": "/share", |  | ||||||
|         "method": "GET", |  | ||||||
|         "enctype": "application/x-www-form-urlencoded", |  | ||||||
|         "params": { |  | ||||||
|             "title": "title", |  | ||||||
|             "text": "text", |  | ||||||
|             "url": "url" |  | ||||||
|         } |  | ||||||
|     }, |  | ||||||
|     "icons": [{ |  | ||||||
|             "src": "notepad16.png", |  | ||||||
|             "sizes": "16x16", |  | ||||||
|             "type": "image/png" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "src": "notepad24.png", |  | ||||||
|             "sizes": "24x24", |  | ||||||
|             "type": "image/png" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "src": "notepad32.png", |  | ||||||
|             "sizes": "32x32", |  | ||||||
|             "type": "image/png" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "src": "notepad64.png", |  | ||||||
|             "sizes": "64x64", |  | ||||||
|             "type": "image/png" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "src": "notepad128.png", |  | ||||||
|             "sizes": "128x128", |  | ||||||
|             "type": "image/png" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "src": "notepad256.png", |  | ||||||
|             "sizes": "256x256", |  | ||||||
|             "type": "image/png" |  | ||||||
|         }, |  | ||||||
|         { |  | ||||||
|             "src": "notepad512.png", |  | ||||||
|             "sizes": "512x512", |  | ||||||
|             "type": "image/png" |  | ||||||
|         } |  | ||||||
|     ], |  | ||||||
|     "start_url": "/", |  | ||||||
|     "display": "standalone", |  | ||||||
|     "theme_color": "#1E88E5", |  | ||||||
|     "background_color": "#ffffff" |  | ||||||
| } |  | ||||||
| @ -1,121 +0,0 @@ | |||||||
| function log(...params) { |  | ||||||
|     console.log.apply(this, [...["%c[SW]: %c", "color: #f4b942;", "color:unset;"], ...params]) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const CACHE = "offline"; |  | ||||||
|  |  | ||||||
| let precacheFiles = [ |  | ||||||
|     "/", |  | ||||||
|     "/index.html", |  | ||||||
|     "/main.js", |  | ||||||
|     "/main.css", |  | ||||||
|     "/serviceworker.js" |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| //Install stage sets up the cache-array to configure pre-cache content |  | ||||||
| self.addEventListener('install', (evt) => { |  | ||||||
|     log('The service worker is being installed.'); |  | ||||||
|     evt.waitUntil(precache().then(() => { |  | ||||||
|         log('Skip waiting on install'); |  | ||||||
|     }).catch(log).then(() => self.skipWaiting())); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| //allow sw to control of current page |  | ||||||
| self.addEventListener('activate', (event) => { |  | ||||||
|     log('Claiming clients for current page'); |  | ||||||
|     return self.clients.claim(); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| self.addEventListener('message', (event) => { |  | ||||||
|     log("Clearing cache"); |  | ||||||
|     caches.delete(CACHE); |  | ||||||
|     event.waitUntil(precache()); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| var Types; |  | ||||||
| (function (Types) { |  | ||||||
|     Types[Types["CACHE"] = 0] = "CACHE"; |  | ||||||
|     Types[Types["NOCACHE"] = 1] = "NOCACHE"; |  | ||||||
|     Types[Types["REFRESH"] = 2] = "REFRESH"; |  | ||||||
|     Types[Types["INDEX"] = 3] = "INDEX"; |  | ||||||
| })(Types || (Types = {})); |  | ||||||
|  |  | ||||||
| let rules = [{ |  | ||||||
|         match: (url) => { |  | ||||||
|             return url.indexOf("/api/") >= 0; |  | ||||||
|         }, |  | ||||||
|         type: Types.NOCACHE |  | ||||||
|     }, { |  | ||||||
|         match: (url) => { |  | ||||||
|             return url.indexOf("/share") >= 0; |  | ||||||
|         }, |  | ||||||
|         type: Types.INDEX |  | ||||||
|     }, |  | ||||||
|     { |  | ||||||
|         match: () => { |  | ||||||
|             return true; |  | ||||||
|         }, |  | ||||||
|         type: Types.REFRESH |  | ||||||
|     } |  | ||||||
| ] |  | ||||||
|  |  | ||||||
| self.addEventListener('fetch', (evt) => { |  | ||||||
|     if (evt.request.method != 'GET') return; // Dont care about POST requests |  | ||||||
|     let rule = rules.find(rule => rule.match(evt.request.url)); |  | ||||||
|     evt.respondWith((async () => { |  | ||||||
|         log("Cache:", Types[rule.type]); |  | ||||||
|         switch (rule.type) { |  | ||||||
|             case Types.CACHE: |  | ||||||
|                 return fromCache(evt.request); |  | ||||||
|             case Types.REFRESH: |  | ||||||
|                 return refresh(evt.request).then(r => { |  | ||||||
|                     evt.waitUntil(r.refresh.catch(_ => {})); |  | ||||||
|                     return r.result; |  | ||||||
|                 }); |  | ||||||
|             case Types.NOCACHE: |  | ||||||
|                 return fetch(evt.request); |  | ||||||
|             case Types.INDEX: |  | ||||||
|                 return refresh(new Request("/")).then(r => { |  | ||||||
|                     evt.waitUntil(r.refresh.catch(_ => {})); |  | ||||||
|                     return r.result; |  | ||||||
|                 }) |  | ||||||
|         } |  | ||||||
|     })()); |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| async function fromCache(request) { |  | ||||||
|     let cache = await caches.open(CACHE); |  | ||||||
|     let matching = await cache.match(request); |  | ||||||
|     if (matching) |  | ||||||
|         return matching |  | ||||||
|  |  | ||||||
|     let res = await fetch(request.clone()); |  | ||||||
|     await cache.put(request, { |  | ||||||
|         match: (url) => { |  | ||||||
|             return url.indexOf("/version_hash") >= 0; |  | ||||||
|         }, |  | ||||||
|         type: Types.NOCACHE |  | ||||||
|     }, res); |  | ||||||
|     return await cache.match(request); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function refresh(request) { |  | ||||||
|     let cache = await caches.open(CACHE); |  | ||||||
|     let web = fetch(request.clone()).then(res => { |  | ||||||
|         return cache.put(request, res).then(() => { |  | ||||||
|             return cache.match(request); |  | ||||||
|         }) |  | ||||||
|     }) |  | ||||||
|     let matching = await cache.match(request); |  | ||||||
|  |  | ||||||
|     return { |  | ||||||
|         result: matching ? matching : web, |  | ||||||
|         refresh: web |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function precache() { |  | ||||||
|     return caches.open(CACHE).then(function (cache) { |  | ||||||
|         return cache.addAll(precacheFiles); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
| @ -1,12 +1,14 @@ | |||||||
| import { h } from "preact"; | import { h } from "preact"; | ||||||
| import "./add_button.scss"; | import "./add_button.scss"; | ||||||
| import Plus from "feather-icons/dist/icons/plus.svg"; | import { Plus } from "preact-feather"; | ||||||
| export default function AddButton({ onClick }: { onClick: () => void }) { | export default function AddButton({ onClick }: { onClick: () => void }) { | ||||||
|    return <button |    return ( | ||||||
|       title="add button" |       <button | ||||||
|       class="fab btn-primary" |          title="add button" | ||||||
|       onClick={() => onClick()} |          class="fab btn-primary" | ||||||
|    > |          onClick={() => onClick()} | ||||||
|       <Plus width={undefined} height={undefined} /> |       > | ||||||
|    </button> |          <Plus width={undefined} height={undefined} /> | ||||||
|  |       </button> | ||||||
|  |    ); | ||||||
| } | } | ||||||
							
								
								
									
										3
									
								
								src/components/CodeMirror.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,3 @@ | |||||||
|  | .CodeMirror { | ||||||
|  |    height: auto; | ||||||
|  | } | ||||||
							
								
								
									
										44
									
								
								src/components/CodeMirror.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,44 @@ | |||||||
|  | import { h } from "preact"; | ||||||
|  | import { useEffect, useRef } from "preact/hooks"; | ||||||
|  |  | ||||||
|  | import * as CM from "codemirror"; | ||||||
|  | import "codemirror/lib/codemirror.css"; | ||||||
|  | import "codemirror/theme/base16-dark.css"; | ||||||
|  |  | ||||||
|  | import "./CodeMirror.scss"; | ||||||
|  | import Theme from "../theme"; | ||||||
|  |  | ||||||
|  | interface ICodeMirrorProps { | ||||||
|  |    value: string; | ||||||
|  |    onChange: (value: string) => void; | ||||||
|  |    onSave?: (value: string) => void; | ||||||
|  |    onClose?: () => void; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export default function CodeMirror(props: ICodeMirrorProps) { | ||||||
|  |    const ref = useRef<HTMLTextAreaElement>(); | ||||||
|  |    useEffect(() => { | ||||||
|  |       const instance = CM.fromTextArea(ref.current, { | ||||||
|  |          value: props.value, | ||||||
|  |          mode: "markdown", | ||||||
|  |          lineNumbers: true, | ||||||
|  |          lineWrapping: true, | ||||||
|  |          theme: Theme.isDark.value ? "base16-dark" : "default", | ||||||
|  |          viewportMargin: Infinity, | ||||||
|  |          // extraKeys: { | ||||||
|  |          //    "Ctrl-S": (cm) => { | ||||||
|  |          //       const val = cm.getValue(); | ||||||
|  |          //       props?.onSave(val); | ||||||
|  |          //    }, | ||||||
|  |          //    Esc: props.onClose, | ||||||
|  |          // }, | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       if (props.value) instance.setValue(props.value); | ||||||
|  |  | ||||||
|  |       instance.on("change", () => props?.onChange(instance.getValue())); | ||||||
|  |  | ||||||
|  |       return () => {}; | ||||||
|  |    }, [ref]); | ||||||
|  |    return <textarea ref={ref}></textarea>; | ||||||
|  | } | ||||||
| @ -1,6 +1,6 @@ | |||||||
| import { h, Component } from "preact"; | import { h, Component } from "preact"; | ||||||
| import Notes from "../notes"; | import Notes from "../notes"; | ||||||
| import Refresh from "feather-icons/dist/icons/refresh-cw.svg"; | import { RefreshCw as Refresh } from "preact-feather"; | ||||||
| import "./footer.scss"; | import "./footer.scss"; | ||||||
| import Notifications from "../notifications"; | import Notifications from "../notifications"; | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,13 +1,15 @@ | |||||||
| import { Observable } from "@hibas123/utils"; | import { Observable } from "@hibas123/utils"; | ||||||
| import { h, Component } from "preact"; | import { h, Component } from "preact"; | ||||||
| import CloseIcon from "feather-icons/dist/icons/x.svg"; | import { X } from "preact-feather"; | ||||||
|  |  | ||||||
| export default abstract class Modal<T> { | export default abstract class Modal<T> { | ||||||
|    // Static |    // Static | ||||||
|    private static modalObservableServer = new Observable<{ modal: Modal<any>, close: boolean }>(); |    private static modalObservableServer = new Observable<{ | ||||||
|  |       modal: Modal<any>; | ||||||
|  |       close: boolean; | ||||||
|  |    }>(); | ||||||
|    public static modalObservable = Modal.modalObservableServer.getPublicApi(); |    public static modalObservable = Modal.modalObservableServer.getPublicApi(); | ||||||
|  |  | ||||||
|  |  | ||||||
|    protected abstract title: string; |    protected abstract title: string; | ||||||
|  |  | ||||||
|    // Private |    // Private | ||||||
| @ -17,10 +19,8 @@ export default abstract class Modal<T> { | |||||||
|  |  | ||||||
|    // Protected |    // Protected | ||||||
|    protected result(value: T | null) { |    protected result(value: T | null) { | ||||||
|       if (this.closeOnResult) |       if (this.closeOnResult) this.close(); | ||||||
|          this.close() |       if (this.onResult) this.onResult(value); | ||||||
|       if (this.onResult) |  | ||||||
|          this.onResult(value); |  | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    //Public |    //Public | ||||||
| @ -41,7 +41,7 @@ export default abstract class Modal<T> { | |||||||
|    public async getResult(close = true) { |    public async getResult(close = true) { | ||||||
|       this.closeOnResult = close; |       this.closeOnResult = close; | ||||||
|       this.show(false); |       this.show(false); | ||||||
|       return new Promise<T | null>((yes) => this.onResult = yes); |       return new Promise<T | null>((yes) => (this.onResult = yes)); | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    public close() { |    public close() { | ||||||
| @ -50,52 +50,68 @@ export default abstract class Modal<T> { | |||||||
|  |  | ||||||
|    public abstract getComponent(): JSX.Element; |    public abstract getComponent(): JSX.Element; | ||||||
|  |  | ||||||
|    public static BaseModal = class BaseModal<T> extends Component<{ modal: Modal<T> }, {}> { |    public static BaseModal = class BaseModal<T> extends Component< | ||||||
|  |       { modal: Modal<T> }, | ||||||
|  |       {} | ||||||
|  |    > { | ||||||
|       render() { |       render() { | ||||||
|          return <div class="modal-container" onClick={(evt) => { |          return ( | ||||||
|             let path = evt.composedPath(); |             <div | ||||||
|             if (!path.find(e => { |                class="modal-container" | ||||||
|                let s = (e as Element); |                onClick={(evt) => { | ||||||
|                return s.id === "ModalContent"; |                   let path = evt.composedPath(); | ||||||
|             })) { |                   if ( | ||||||
|                this.props.modal.result(null) |                      !path.find((e) => { | ||||||
|             } |                         let s = e as Element; | ||||||
|          }} onKeyDown={evt => { |                         return s.id === "ModalContent"; | ||||||
|             if (evt.keyCode === 27) { |                      }) | ||||||
|                this.props.modal.result(null) |                   ) { | ||||||
|             } |                      this.props.modal.result(null); | ||||||
|          }}> |  | ||||||
|             <div id="ModalContent" class="modal" > |  | ||||||
|                <div class="modal-title" style=""> |  | ||||||
|                   <h3>{this.props.modal.title}</h3> |  | ||||||
|                   {/* <div> */} |  | ||||||
|                   {!this.props.modal.noClose ? |  | ||||||
|                      <CloseIcon onClick={() => this.props.modal.result(null)} width={undefined} height={undefined} /> |  | ||||||
|                      : undefined |  | ||||||
|                   } |                   } | ||||||
|                   {/* </div> */} |                }} | ||||||
|  |                onKeyDown={(evt) => { | ||||||
|  |                   if (evt.keyCode === 27) { | ||||||
|  |                      this.props.modal.result(null); | ||||||
|  |                   } | ||||||
|  |                }} | ||||||
|  |             > | ||||||
|  |                <div id="ModalContent" class="modal"> | ||||||
|  |                   <div class="modal-title" style=""> | ||||||
|  |                      <h3>{this.props.modal.title}</h3> | ||||||
|  |                      {/* <div> */} | ||||||
|  |                      {!this.props.modal.noClose ? ( | ||||||
|  |                         <X | ||||||
|  |                            onClick={() => this.props.modal.result(null)} | ||||||
|  |                            width={undefined} | ||||||
|  |                            height={undefined} | ||||||
|  |                         /> | ||||||
|  |                      ) : undefined} | ||||||
|  |                      {/* </div> */} | ||||||
|  |                   </div> | ||||||
|  |                   {this.props.children} | ||||||
|                </div> |                </div> | ||||||
|                {this.props.children} |  | ||||||
|             </div> |             </div> | ||||||
|          </div> |          ); | ||||||
|       } |       } | ||||||
|    } |    }; | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| export class ModalComponent extends Component<{}, { modal: Modal<any> | undefined, component: JSX.Element | undefined }>{ | export class ModalComponent extends Component< | ||||||
|  |    {}, | ||||||
|  |    { modal: Modal<any> | undefined; component: JSX.Element | undefined } | ||||||
|  | > { | ||||||
|    constructor(props) { |    constructor(props) { | ||||||
|       super(props); |       super(props); | ||||||
|       this.onModal = this.onModal.bind(this); |       this.onModal = this.onModal.bind(this); | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    onModal({ modal, close }: { modal: Modal<any>, close: boolean }) { |    onModal({ modal, close }: { modal: Modal<any>; close: boolean }) { | ||||||
|       if (!close && this.state.modal !== modal) { |       if (!close && this.state.modal !== modal) { | ||||||
|          this.setState({ modal: modal, component: modal.getComponent() }) |          this.setState({ modal: modal, component: modal.getComponent() }); | ||||||
|       } |       } else { | ||||||
|       else { |          if (this.state.modal === modal && close) | ||||||
|          if (this.state.modal === modal && close) // Only close if the same |             // Only close if the same | ||||||
|             this.setState({ modal: undefined, component: undefined }) |             this.setState({ modal: undefined, component: undefined }); | ||||||
|       } |       } | ||||||
|    } |    } | ||||||
|  |  | ||||||
| @ -108,8 +124,6 @@ export class ModalComponent extends Component<{}, { modal: Modal<any> | undefine | |||||||
|    } |    } | ||||||
|  |  | ||||||
|    render() { |    render() { | ||||||
|       return <div> |       return <div>{this.state.component}</div>; | ||||||
|          {this.state.component} |  | ||||||
|       </div> |  | ||||||
|    } |    } | ||||||
| } | } | ||||||
| @ -2,36 +2,71 @@ import { h } from "preact"; | |||||||
| import { Page } from "../../../page"; | import { Page } from "../../../page"; | ||||||
| import Theme, { ThemeStates } from "../../../theme"; | import Theme, { ThemeStates } from "../../../theme"; | ||||||
| import Navigation from "../../../navigation"; | import Navigation from "../../../navigation"; | ||||||
| import ArrowLeft from "feather-icons/dist/icons/arrow-left.svg"; | import { ArrowLeft } from "preact-feather"; | ||||||
|  |  | ||||||
| export default class SettingsPage extends Page<{ state: any }, { vault: string }> { | export default class SettingsPage extends Page< | ||||||
|     componentWillMount() { |    { state: any }, | ||||||
|  |    { vault: string } | ||||||
|  | > { | ||||||
|  |    componentWillMount() {} | ||||||
|  |  | ||||||
|     } |    render() { | ||||||
|  |       let active = Theme.active(); | ||||||
|     render() { |       return ( | ||||||
|         let active = Theme.active(); |          <div> | ||||||
|         return <div> |  | ||||||
|             <header class="header"> |             <header class="header"> | ||||||
|                 <a class="header-icon-button" onClick={() => history.back()}><ArrowLeft height={undefined} width={undefined} /></a> |                <a class="header-icon-button" onClick={() => history.back()}> | ||||||
|                 <h3 style="display:inline" class="header-title" onClick={() => Navigation.setPage("/")}>Settings</h3> |                   <ArrowLeft height={undefined} width={undefined} /> | ||||||
|                 <span></span> |                </a> | ||||||
|  |                <h3 | ||||||
|  |                   style="display:inline" | ||||||
|  |                   class="header-title" | ||||||
|  |                   onClick={() => Navigation.setPage("/")} | ||||||
|  |                > | ||||||
|  |                   Settings | ||||||
|  |                </h3> | ||||||
|  |                <span></span> | ||||||
|             </header> |             </header> | ||||||
|             <div class="container"> |             <div class="container"> | ||||||
|                 <div className="input-group"> |                <div className="input-group"> | ||||||
|                     <label>Select Theme: </label> |                   <label>Select Theme: </label> | ||||||
|                     <select class="inp" onChange={(ev) => Theme.change(Number((ev.target as HTMLSelectElement).value))}> |                   <select | ||||||
|                         {Object.keys(ThemeStates) |                      class="inp" | ||||||
|                             .filter(e => Number.isNaN(Number(e))) |                      onChange={(ev) => | ||||||
|                             .map(e => <option selected={ThemeStates[e] === active} value={ThemeStates[e]}>{e.charAt(0).toUpperCase() + e.slice(1).toLowerCase()}</option>)} |                         Theme.change( | ||||||
|                         {/* <option value={ThemeStates.AUTO}>Auto</option> |                            Number((ev.target as HTMLSelectElement).value) | ||||||
|  |                         ) | ||||||
|  |                      } | ||||||
|  |                   > | ||||||
|  |                      {Object.keys(ThemeStates) | ||||||
|  |                         .filter((e) => Number.isNaN(Number(e))) | ||||||
|  |                         .map((e) => ( | ||||||
|  |                            <option | ||||||
|  |                               selected={ThemeStates[e] === active} | ||||||
|  |                               value={ThemeStates[e]} | ||||||
|  |                            > | ||||||
|  |                               {e.charAt(0).toUpperCase() + | ||||||
|  |                                  e.slice(1).toLowerCase()} | ||||||
|  |                            </option> | ||||||
|  |                         ))} | ||||||
|  |                      {/* <option value={ThemeStates.AUTO}>Auto</option> | ||||||
|                         <option value={ThemeStates.LIGHT}>Light</option> |                         <option value={ThemeStates.LIGHT}>Light</option> | ||||||
|                         <option value={ThemeStates.DARK}>Dark</option> */} |                         <option value={ThemeStates.DARK}>Dark</option> */} | ||||||
|                     </select> |                   </select> | ||||||
|                 </div> |                </div> | ||||||
|                 {/* <button class="btn" onClick={() => Theme.toggle()}>Toggle Dark Mode</button> */} |                {/* <button class="btn" onClick={() => Theme.toggle()}>Toggle Dark Mode</button> */} | ||||||
|                 <button class="btn" onClick={() => window.navigator.serviceWorker.controller.postMessage("message")}>Clear cache</button> |                <button | ||||||
|  |                   class="btn" | ||||||
|  |                   onClick={() => | ||||||
|  |                      window.navigator.serviceWorker.controller.postMessage( | ||||||
|  |                         "message" | ||||||
|  |                      ) | ||||||
|  |                   } | ||||||
|  |                > | ||||||
|  |                   Clear cache | ||||||
|  |                </button> | ||||||
|             </div> |             </div> | ||||||
|         </div >; |          </div> | ||||||
|     } |       ); | ||||||
|  |    } | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										209
									
								
								src/components/routes/vault/Entry copy.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,209 @@ | |||||||
|  | import { h, Component } from "preact"; | ||||||
|  | import { IVault, ViewNote } from "../../../notes"; | ||||||
|  | import { Trash2 as Trash, X, Save } from "preact-feather"; | ||||||
|  |  | ||||||
|  | import Navigation from "../../../navigation"; | ||||||
|  | import { YesNoModal } from "../../modals/YesNoModal"; | ||||||
|  | import Notifications, { MessageType } from "../../../notifications"; | ||||||
|  |  | ||||||
|  | const minRows = 3; | ||||||
|  | export default class EntryComponent extends Component< | ||||||
|  |    { vault: Promise<IVault>; id: string | undefined; note: string | undefined }, | ||||||
|  |    { title: string; changed: boolean } | ||||||
|  | > { | ||||||
|  |    private text: string = ""; | ||||||
|  |    private vault: IVault; | ||||||
|  |    // private lineHeight: number = 24; | ||||||
|  |    private note: ViewNote; | ||||||
|  |  | ||||||
|  |    private rows: number = minRows; | ||||||
|  |  | ||||||
|  |    private skip_save: boolean = false; | ||||||
|  |  | ||||||
|  |    private inputElm: HTMLInputElement; | ||||||
|  |  | ||||||
|  |    constructor(props) { | ||||||
|  |       super(props); | ||||||
|  |       this.state = { changed: false, title: "" }; | ||||||
|  |       this.textAreaChange = this.textAreaChange.bind(this); | ||||||
|  |       this.textAreaKeyPress = this.textAreaKeyPress.bind(this); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    private toVault() { | ||||||
|  |       history.back(); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    componentDidMount() { | ||||||
|  |       this.inputElm.focus(); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    async componentWillMount() { | ||||||
|  |       this.text = ""; | ||||||
|  |       this.vault = undefined; | ||||||
|  |       this.note = undefined; | ||||||
|  |       this.rows = minRows; | ||||||
|  |       this.skip_save = false; | ||||||
|  |       this.setState({ changed: false, title: "" }); | ||||||
|  |       try { | ||||||
|  |          this.skip_save = false; | ||||||
|  |  | ||||||
|  |          this.vault = await this.props.vault; | ||||||
|  |          let note: ViewNote; | ||||||
|  |          let changed = false; | ||||||
|  |          if (this.props.id) note = await this.vault.getNote(this.props.id); | ||||||
|  |          else { | ||||||
|  |             note = this.vault.newNote(); | ||||||
|  |             if (this.props.note) { | ||||||
|  |                note.__value = this.props.note; | ||||||
|  |                changed = true; | ||||||
|  |             } | ||||||
|  |          } | ||||||
|  |  | ||||||
|  |          if (!note) { | ||||||
|  |             Notifications.sendNotification( | ||||||
|  |                "Note not found!", | ||||||
|  |                MessageType.ERROR | ||||||
|  |             ); | ||||||
|  |          } else { | ||||||
|  |             this.note = note; | ||||||
|  |             this.text = note.__value; | ||||||
|  |             let rows = this.getRows(this.text); | ||||||
|  |             if (rows !== this.rows) { | ||||||
|  |                this.rows = rows; | ||||||
|  |             } | ||||||
|  |             let [title] = this.text.split("\n", 1); | ||||||
|  |             this.setState({ title, changed }); | ||||||
|  |          } | ||||||
|  |       } catch (err) { | ||||||
|  |          Notifications.sendError(err); | ||||||
|  |       } | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    private async save() { | ||||||
|  |       try { | ||||||
|  |          if (this.state.changed) { | ||||||
|  |             this.note.__value = this.text; | ||||||
|  |             await this.vault.saveNote(this.note); | ||||||
|  |             this.setState({ changed: false }); | ||||||
|  |          } | ||||||
|  |       } catch (err) { | ||||||
|  |          Notifications.sendError(err); | ||||||
|  |       } | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    componentWillUnmount() { | ||||||
|  |       if (!this.skip_save) this.save(); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    strToNr(value: string) { | ||||||
|  |       let match = value.match(/\d/g); | ||||||
|  |       return Number(match.join("")); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    getRows(value: string) { | ||||||
|  |       const lines = (value.match(/\r?\n/g) || "").length + 1; | ||||||
|  |       return Math.max(lines + 1, minRows); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    textAreaChange(evt: KeyboardEvent) { | ||||||
|  |       if (evt.keyCode === 17 || evt.keyCode === 27) return; //No character, so not relevant for this function | ||||||
|  |       if (!this.state.changed && this.textAreaKeyPress(evt)) | ||||||
|  |          this.setState({ changed: true }); | ||||||
|  |  | ||||||
|  |       let target = evt.target as HTMLTextAreaElement; | ||||||
|  |       let value = target.value; | ||||||
|  |  | ||||||
|  |       this.text = value; | ||||||
|  |  | ||||||
|  |       let [title] = value.split("\n", 1); | ||||||
|  |       if (title !== this.state.title) this.setState({ title }); | ||||||
|  |  | ||||||
|  |       let rows = this.getRows(value); | ||||||
|  |       if (rows !== this.rows) { | ||||||
|  |          target.rows = rows; | ||||||
|  |          this.rows = rows; | ||||||
|  |       } | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    async exitHandler() { | ||||||
|  |       if (this.state.changed) { | ||||||
|  |          let modal = new YesNoModal("Really want to quit?"); | ||||||
|  |          let res = await modal.getResult(); | ||||||
|  |          if (res === true) { | ||||||
|  |             this.skip_save = true; | ||||||
|  |             this.toVault(); | ||||||
|  |          } | ||||||
|  |       } else this.toVault(); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    textAreaKeyPress(evt: KeyboardEvent) { | ||||||
|  |       console.log(evt); | ||||||
|  |       if ((evt.keyCode === 83 || evt.keyCode === 13) && evt.ctrlKey) { | ||||||
|  |          evt.preventDefault(); | ||||||
|  |          this.save(); | ||||||
|  |          return false; | ||||||
|  |       } else if (evt.keyCode === 27) { | ||||||
|  |          evt.preventDefault(); | ||||||
|  |          this.exitHandler(); | ||||||
|  |          return false; | ||||||
|  |       } | ||||||
|  |       return true; | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    render() { | ||||||
|  |       const save_handler = async () => { | ||||||
|  |          await this.save(); | ||||||
|  |          Navigation.setPage( | ||||||
|  |             "/vault", | ||||||
|  |             { id: this.vault.id }, | ||||||
|  |             { entry: "false" }, | ||||||
|  |             true | ||||||
|  |          ); | ||||||
|  |          this.toVault(); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       const delete_handler = async () => { | ||||||
|  |          await this.vault.deleteNote(this.props.id); | ||||||
|  |          this.toVault(); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       console.log("Rerender"); | ||||||
|  |  | ||||||
|  |       return ( | ||||||
|  |          <div> | ||||||
|  |             <header class="header"> | ||||||
|  |                <a class="header-icon-button" onClick={() => this.exitHandler()}> | ||||||
|  |                   <X height={undefined} width={undefined} /> | ||||||
|  |                </a> | ||||||
|  |                {this.state.changed ? ( | ||||||
|  |                   <a | ||||||
|  |                      class="header-icon-button" | ||||||
|  |                      style="margin-left: 0.5em;" | ||||||
|  |                      onClick={() => save_handler()} | ||||||
|  |                   > | ||||||
|  |                      <Save height={undefined} width={undefined} /> | ||||||
|  |                   </a> | ||||||
|  |                ) : undefined} | ||||||
|  |                <h3 style="display:inline" class="button header-title"> | ||||||
|  |                   {this.state.title} | ||||||
|  |                </h3> | ||||||
|  |                <a class="header-icon-button" onClick={() => delete_handler()}> | ||||||
|  |                   <Trash height={undefined} width={undefined} /> | ||||||
|  |                </a> | ||||||
|  |             </header> | ||||||
|  |             <div class="container"> | ||||||
|  |                <textarea | ||||||
|  |                   value={this.text} | ||||||
|  |                   rows={this.rows} | ||||||
|  |                   class="inp" | ||||||
|  |                   style="width:100%;" | ||||||
|  |                   onInput={this.textAreaChange} | ||||||
|  |                   onKeyDown={this.textAreaKeyPress} | ||||||
|  |                   onChange={this.textAreaChange} | ||||||
|  |                   ref={(elm) => (this.inputElm = elm)} | ||||||
|  |                /> | ||||||
|  |             </div> | ||||||
|  |          </div> | ||||||
|  |       ); | ||||||
|  |    } | ||||||
|  | } | ||||||
| @ -1,188 +1,132 @@ | |||||||
| import { h, Component } from "preact" | import { h, Component } from "preact"; | ||||||
| import { IVault, ViewNote } from "../../../notes"; | import { IVault, ViewNote } from "../../../notes"; | ||||||
| import Trash from "feather-icons/dist/icons/trash-2.svg" | import { Trash2 as Trash, X, Save } from "preact-feather"; | ||||||
| import X from "feather-icons/dist/icons/x.svg" |  | ||||||
| import Save from "feather-icons/dist/icons/save.svg" |  | ||||||
|  |  | ||||||
| import Navigation from "../../../navigation"; | import Navigation from "../../../navigation"; | ||||||
| import { YesNoModal } from "../../modals/YesNoModal"; | import { YesNoModal } from "../../modals/YesNoModal"; | ||||||
| import Notifications, { MessageType } from "../../../notifications"; | import Notifications, { MessageType } from "../../../notifications"; | ||||||
|  | import CodeMirror from "../../CodeMirror"; | ||||||
|  | import { useEffect, useMemo, useState } from "preact/hooks"; | ||||||
|  | import { usePromise } from "../../../hooks"; | ||||||
|  |  | ||||||
| const minRows = 3; | interface IEntryProps { | ||||||
| export default class EntryComponent extends Component<{ vault: Promise<IVault>, id: string | undefined, note: string | undefined }, { title: string, changed: boolean }> { |    vault: IVault; | ||||||
|    private text: string = ""; |    id?: string; | ||||||
|    private vault: IVault; |    note?: string; | ||||||
|    // private lineHeight: number = 24; | } | ||||||
|    private note: ViewNote; |  | ||||||
|  |  | ||||||
|    private rows: number = minRows; | export default function Entry(props: IEntryProps) { | ||||||
|  |    const [changed, setChanged] = useState(false); | ||||||
|  |  | ||||||
|    private skip_save: boolean = false; |    const [text, setText] = useState(""); | ||||||
|  |    const title = useMemo(() => text?.split("\n", 1)[0], [text]); | ||||||
|  |  | ||||||
|    private inputElm: HTMLInputElement; |    const [loading, error, note] = usePromise(async () => { | ||||||
|  |       let note: ViewNote; | ||||||
|  |       if (props.id) { | ||||||
|  |          note = await props.vault.getNote(props.id); | ||||||
|  |       } else { | ||||||
|  |          note = props.vault.newNote(); | ||||||
|  |          if (props.note) { | ||||||
|  |             note.__value = props.note; | ||||||
|  |             setChanged(true); | ||||||
|  |          } | ||||||
|  |       } | ||||||
|  |  | ||||||
|    constructor(props) { |       if (!note) { | ||||||
|       super(props); |          Notifications.sendNotification("Note not found!", MessageType.ERROR); | ||||||
|       this.state = { changed: false, title: "" }; |          history.back(); | ||||||
|       this.textAreaChange = this.textAreaChange.bind(this); |       } else { | ||||||
|       this.textAreaKeyPress = this.textAreaKeyPress.bind(this); |          setText(note.__value); | ||||||
|    } |       } | ||||||
|  |  | ||||||
|    private toVault() { |       return note; | ||||||
|       history.back(); |    }, [props.vault, props.id]); | ||||||
|    } |  | ||||||
|  |  | ||||||
|    componentDidMount() { |    if (loading) { | ||||||
|       this.inputElm.focus(); |       return <div>Loading entry</div>; | ||||||
|    } |    } else { | ||||||
|  |       const save = async () => { | ||||||
|    async componentWillMount() { |          try { | ||||||
|       this.text = ""; |             if (changed) { | ||||||
|       this.vault = undefined; |                note.__value = text; | ||||||
|       this.note = undefined; |                await props.vault.saveNote(note); | ||||||
|       this.rows = minRows; |                setChanged(false); | ||||||
|       this.skip_save = false; |  | ||||||
|       this.setState({ changed: false, title: "" }); |  | ||||||
|       try { |  | ||||||
|          this.skip_save = false; |  | ||||||
|  |  | ||||||
|          this.vault = await this.props.vault; |  | ||||||
|          let note: ViewNote; |  | ||||||
|          let changed = false; |  | ||||||
|          if (this.props.id) |  | ||||||
|             note = await this.vault.getNote(this.props.id) |  | ||||||
|          else { |  | ||||||
|             note = this.vault.newNote(); |  | ||||||
|             if (this.props.note) { |  | ||||||
|                note.__value = this.props.note; |  | ||||||
|                changed = true; |  | ||||||
|             } |             } | ||||||
|  |          } catch (err) { | ||||||
|  |             Notifications.sendError(err); | ||||||
|          } |          } | ||||||
|  |       }; | ||||||
|  |  | ||||||
|          if (!note) { |       const del = async () => { | ||||||
|             Notifications.sendNotification("Note not found!", MessageType.ERROR); |          await props.vault.deleteNote(props.id); | ||||||
|          } else { |          history.back(); | ||||||
|             this.note = note; |       }; | ||||||
|             this.text = note.__value; |  | ||||||
|             let rows = this.getRows(this.text); |       const close = async () => { | ||||||
|             if (rows !== this.rows) { |          if (changed) { | ||||||
|                this.rows = rows; |             let modal = new YesNoModal("Really want to quit?"); | ||||||
|  |             let res = await modal.getResult(); | ||||||
|  |             if (!res) return; | ||||||
|  |          } | ||||||
|  |          history.back(); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       // TODO: Add warning on possibly unwanted exit | ||||||
|  |       useEffect(() => { | ||||||
|  |          const keyevent = (evt: KeyboardEvent) => { | ||||||
|  |             if (evt.key === "s" && evt.ctrlKey) { | ||||||
|  |                evt.preventDefault(); | ||||||
|  |                save(); | ||||||
|  |                return false; | ||||||
|  |             } else if (evt.key === "Escape") { | ||||||
|  |                evt.preventDefault(); | ||||||
|  |                close(); | ||||||
|  |                return false; | ||||||
|             } |             } | ||||||
|             let [title] = this.text.split("\n", 1); |             return true; | ||||||
|             this.setState({ title, changed }) |          }; | ||||||
|          } |  | ||||||
|       } catch (err) { |  | ||||||
|          Notifications.sendError(err); |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private async save() { |          document.addEventListener("keydown", keyevent); | ||||||
|       try { |          return () => { | ||||||
|          if (this.state.changed) { |             document.removeEventListener("keydown", keyevent); | ||||||
|             this.note.__value = this.text; |          }; | ||||||
|             await this.vault.saveNote(this.note); |       }); | ||||||
|             this.setState({ changed: false }) |  | ||||||
|          } |  | ||||||
|       } catch (err) { |  | ||||||
|          Notifications.sendError(err); |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    componentWillUnmount() { |       return ( | ||||||
|       if (!this.skip_save) |          <div> | ||||||
|          this.save() |             <header class="header" style="margin-bottom: 0;"> | ||||||
|    } |                <a class="header-icon-button" onClick={close}> | ||||||
|  |                   <X height={undefined} width={undefined} /> | ||||||
|    strToNr(value: string) { |                </a> | ||||||
|       let match = value.match(/\d/g) |                {changed && ( | ||||||
|       return Number(match.join("")) |                   <a | ||||||
|    } |                      class="header-icon-button" | ||||||
|  |                      style="margin-left: 0.5em;" | ||||||
|    getRows(value: string) { |                      onClick={save} | ||||||
|       const lines = (value.match(/\r?\n/g) || '').length + 1 |                   > | ||||||
|       return Math.max(lines + 1, minRows); |                      <Save height={undefined} width={undefined} /> | ||||||
|    } |                   </a> | ||||||
|  |                )} | ||||||
|    textAreaChange(evt: KeyboardEvent) { |                <h3 style="display:inline" class="button header-title"> | ||||||
|       if (evt.keyCode === 17 || evt.keyCode === 27) return; //No character, so not relevant for this function |                   {title} | ||||||
|       if (!this.state.changed && this.textAreaKeyPress(evt)) this.setState({ changed: true }) |                </h3> | ||||||
|  |                <a class="header-icon-button" onClick={del}> | ||||||
|       let target = (evt.target as HTMLTextAreaElement) |                   <Trash height={undefined} width={undefined} /> | ||||||
|       let value = target.value; |                </a> | ||||||
|  |             </header> | ||||||
|       this.text = value; |             <div class="container" style="padding: 0"> | ||||||
|  |                <CodeMirror | ||||||
|       let [title] = value.split("\n", 1); |                   value={text} | ||||||
|       if (title !== this.state.title) |                   onChange={(value) => { | ||||||
|          this.setState({ title }); |                      setChanged(true); | ||||||
|  |                      setText(value); | ||||||
|       let rows = this.getRows(value); |                   }} | ||||||
|       if (rows !== this.rows) { |                   // onSave={save} | ||||||
|          target.rows = rows; |                   // onClose={close} | ||||||
|          this.rows = rows; |                /> | ||||||
|       } |             </div> | ||||||
|    } |  | ||||||
|  |  | ||||||
|    async exitHandler() { |  | ||||||
|       if (this.state.changed) { |  | ||||||
|          let modal = new YesNoModal("Really want to quit?"); |  | ||||||
|          let res = await modal.getResult(); |  | ||||||
|          if (res === true) { |  | ||||||
|             this.skip_save = true; |  | ||||||
|             this.toVault(); |  | ||||||
|          } |  | ||||||
|       } else |  | ||||||
|          this.toVault() |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    textAreaKeyPress(evt: KeyboardEvent) { |  | ||||||
|       console.log(evt); |  | ||||||
|       if ((evt.keyCode === 83 || evt.keyCode === 13) && evt.ctrlKey) { |  | ||||||
|          evt.preventDefault() |  | ||||||
|          this.save(); |  | ||||||
|          return false; |  | ||||||
|       } |  | ||||||
|       else if (evt.keyCode === 27) { |  | ||||||
|          evt.preventDefault(); |  | ||||||
|          this.exitHandler(); |  | ||||||
|          return false; |  | ||||||
|       } |  | ||||||
|       return true; |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    render() { |  | ||||||
|       const save_handler = async () => { |  | ||||||
|          await this.save() |  | ||||||
|          Navigation.setPage("/vault", { id: this.vault.id }, { entry: "false" }, true); |  | ||||||
|          this.toVault() |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       const delete_handler = async () => { |  | ||||||
|          await this.vault.deleteNote(this.props.id); |  | ||||||
|          this.toVault() |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       console.log("Rerender") |  | ||||||
|  |  | ||||||
|       return <div> |  | ||||||
|          <header class="header"> |  | ||||||
|             <a class="header-icon-button" onClick={() => this.exitHandler()}><X height={undefined} width={undefined} /></a> |  | ||||||
|             {this.state.changed ? <a class="header-icon-button" style="margin-left: 0.5em;" onClick={() => save_handler()}><Save height={undefined} width={undefined} /></a> : undefined} |  | ||||||
|             <h3 style="display:inline" class="button header-title">{this.state.title}</h3> |  | ||||||
|             <a class="header-icon-button" onClick={() => delete_handler()}><Trash height={undefined} width={undefined} /></a> |  | ||||||
|          </header> |  | ||||||
|          <div class="container"> |  | ||||||
|             <textarea |  | ||||||
|                value={this.text} |  | ||||||
|                rows={this.rows} |  | ||||||
|                class="inp" |  | ||||||
|                style="width:100%;" |  | ||||||
|                onInput={this.textAreaChange} |  | ||||||
|                onKeyDown={this.textAreaKeyPress} |  | ||||||
|                onChange={this.textAreaChange} |  | ||||||
|                ref={elm => this.inputElm = elm} |  | ||||||
|             /> |  | ||||||
|          </div> |          </div> | ||||||
|       </div> |       ); | ||||||
|    } |    } | ||||||
| } | } | ||||||
| @ -1,24 +1,25 @@ | |||||||
| import { h, Component } from "preact" | import { h, Component } from "preact"; | ||||||
| import Notes, { IVault, BaseNote } from "../../../notes"; | import Notes, { IVault, BaseNote } from "../../../notes"; | ||||||
| import AddButton from "../../AddButton"; | import AddButton from "../../AddButton"; | ||||||
| import Navigation from "../../../navigation"; | import Navigation from "../../../navigation"; | ||||||
| import ArrowLeft from "feather-icons/dist/icons/arrow-left.svg" | import { ArrowLeft, Search } from "preact-feather"; | ||||||
| import Search from "feather-icons/dist/icons/search.svg" |  | ||||||
| import ContextMenu from "../../context"; | import ContextMenu from "../../context"; | ||||||
| import Notifications from "../../../notifications"; | import Notifications from "../../../notifications"; | ||||||
| import { Observable, Lock } from "@hibas123/utils"; | import { Observable, Lock } from "@hibas123/utils"; | ||||||
|  |  | ||||||
|  | export default class EntryList extends Component< | ||||||
| export default class EntryList extends Component<{ vault: Promise<IVault> }, { notes: BaseNote[], context: JSX.Element | undefined }> { |    { vault: IVault }, | ||||||
|  |    { notes: BaseNote[]; context: h.JSX.Element | undefined } | ||||||
|  | > { | ||||||
|    rawNotes: BaseNote[]; |    rawNotes: BaseNote[]; | ||||||
|  |  | ||||||
|    private searchObservableServer = new Observable<void>(1000); |    private searchObservableServer = new Observable<void>(1000); | ||||||
|    private searchObservable = this.searchObservableServer.getPublicApi(); |    private searchObservable = this.searchObservableServer.getPublicApi(); | ||||||
|  |  | ||||||
|    constructor(props) { |    constructor(props) { | ||||||
|       super(props) |       super(props); | ||||||
|       console.log("Creating new Instance of EntryList") |       console.log("Creating new Instance of EntryList"); | ||||||
|       this.state = { notes: [], context: undefined } |       this.state = { notes: [], context: undefined }; | ||||||
|       this.onDragOver = this.onDragOver.bind(this); |       this.onDragOver = this.onDragOver.bind(this); | ||||||
|       this.onDrop = this.onDrop.bind(this); |       this.onDrop = this.onDrop.bind(this); | ||||||
|       this.reloadNotes = this.reloadNotes.bind(this); |       this.reloadNotes = this.reloadNotes.bind(this); | ||||||
| @ -27,14 +28,13 @@ export default class EntryList extends Component<{ vault: Promise<IVault> }, { n | |||||||
|    vault: IVault; |    vault: IVault; | ||||||
|  |  | ||||||
|    async reloadNotes(s?: boolean) { |    async reloadNotes(s?: boolean) { | ||||||
|       if (s) |       if (s) return; | ||||||
|          return; |  | ||||||
|       this.rawNotes = await this.vault.getAllNotes(); |       this.rawNotes = await this.vault.getAllNotes(); | ||||||
|       await this.applySearch(true); |       await this.applySearch(true); | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    async componentWillMount() { |    async componentWillMount() { | ||||||
|       this.vault = await this.props.vault; |       this.vault = this.props.vault; | ||||||
|       this.reloadNotes(); |       this.reloadNotes(); | ||||||
|       document.body.addEventListener("dragover", this.onDragOver); |       document.body.addEventListener("dragover", this.onDragOver); | ||||||
|       document.body.addEventListener("drop", this.onDrop); |       document.body.addEventListener("drop", this.onDrop); | ||||||
| @ -64,59 +64,74 @@ export default class EntryList extends Component<{ vault: Promise<IVault> }, { n | |||||||
|       const close = () => { |       const close = () => { | ||||||
|          document.documentElement.removeEventListener("click", close); |          document.documentElement.removeEventListener("click", close); | ||||||
|          this.setState({ context: undefined }); |          this.setState({ context: undefined }); | ||||||
|       } |       }; | ||||||
|       document.documentElement.addEventListener("click", close); |       document.documentElement.addEventListener("click", close); | ||||||
|  |  | ||||||
|  |  | ||||||
|       const shareNote = async () => { |       const shareNote = async () => { | ||||||
|          let nav = window.navigator as any; |          let nav = window.navigator as any; | ||||||
|          if (nav.share !== undefined) { |          if (nav.share !== undefined) { | ||||||
|             let vnote = await this.vault.getNote(note._id); |             let vnote = await this.vault.getNote(note._id); | ||||||
|             nav.share({ title: vnote.preview.split("\n")[0], text: vnote.__value }) |             nav.share({ | ||||||
|                .then(() => console.log('Successful share')) |                title: vnote.preview.split("\n")[0], | ||||||
|                .catch(error => { |                text: vnote.__value, | ||||||
|                   console.error('Error sharing:', error) |             }) | ||||||
|                   Notifications.sendError(error) |                .then(() => console.log("Successful share")) | ||||||
|  |                .catch((error) => { | ||||||
|  |                   console.error("Error sharing:", error); | ||||||
|  |                   Notifications.sendError(error); | ||||||
|                }); |                }); | ||||||
|          } else { |          } else { | ||||||
|             Notifications.sendError("Sharing not possible on this device") |             Notifications.sendError("Sharing not possible on this device"); | ||||||
|          } |          } | ||||||
|       } |       }; | ||||||
|  |  | ||||||
|       const deleteNote = async () => { |       const deleteNote = async () => { | ||||||
|          this.vault.deleteNote(note._id).then(() => { |          this.vault | ||||||
|             Notifications.sendSuccess("Deleted") |             .deleteNote(note._id) | ||||||
|             this.rawNotes = this.rawNotes.filter(e => e._id !== note._id); |             .then(() => { | ||||||
|             this.setState({ notes: this.state.notes.filter(e => e._id !== note._id) }); |                Notifications.sendSuccess("Deleted"); | ||||||
|          }).catch(err => { |                this.rawNotes = this.rawNotes.filter((e) => e._id !== note._id); | ||||||
|             Notifications.sendError(err); |                this.setState({ | ||||||
|          }) |                   notes: this.state.notes.filter((e) => e._id !== note._id), | ||||||
|       } |                }); | ||||||
|  |             }) | ||||||
|  |             .catch((err) => { | ||||||
|  |                Notifications.sendError(err); | ||||||
|  |             }); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|       const copyNote = async () => { |       const copyNote = async () => { | ||||||
|          this.vault.getNote(note._id).then(note => { |          this.vault.getNote(note._id).then((note) => { | ||||||
|             const el = document.createElement('textarea'); |             const el = document.createElement("textarea"); | ||||||
|             el.value = note.__value; |             el.value = note.__value; | ||||||
|             document.body.appendChild(el); |             document.body.appendChild(el); | ||||||
|             el.select(); |             el.select(); | ||||||
|             document.execCommand('copy'); |             document.execCommand("copy"); | ||||||
|             document.body.removeChild(el); |             document.body.removeChild(el); | ||||||
|             Notifications.sendSuccess("Copied"); |             Notifications.sendSuccess("Copied"); | ||||||
|          }) |          }); | ||||||
|       } |       }; | ||||||
|  |  | ||||||
|       let share; |       let share; | ||||||
|       if ((window.navigator as any).share) { |       if ((window.navigator as any).share) { | ||||||
|          share = <button class="btn" onClick={shareNote}> |          share = ( | ||||||
|             share |             <button class="btn" onClick={shareNote}> | ||||||
|          </button> |                share | ||||||
|  |             </button> | ||||||
|  |          ); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       let context = <ContextMenu event={evt} > |       let context = ( | ||||||
|          {share} |          <ContextMenu event={evt}> | ||||||
|          <button class="btn" onClick={deleteNote}>delete</button> |             {share} | ||||||
|          <button class="btn" onClick={copyNote}>copy</button> |             <button class="btn" onClick={deleteNote}> | ||||||
|       </ContextMenu> |                delete | ||||||
|  |             </button> | ||||||
|  |             <button class="btn" onClick={copyNote}> | ||||||
|  |                copy | ||||||
|  |             </button> | ||||||
|  |          </ContextMenu> | ||||||
|  |       ); | ||||||
|  |  | ||||||
|       this.setState({ context }); |       this.setState({ context }); | ||||||
|       return false; |       return false; | ||||||
| @ -131,57 +146,69 @@ export default class EntryList extends Component<{ vault: Promise<IVault> }, { n | |||||||
|             if (item.kind === "file") { |             if (item.kind === "file") { | ||||||
|                let file = item.getAsFile(); |                let file = item.getAsFile(); | ||||||
|                if (file.type !== "application/json") { |                if (file.type !== "application/json") { | ||||||
|                   Notifications.sendError("Invalid File Type!!!") |                   Notifications.sendError("Invalid File Type!!!"); | ||||||
|                } else { |                } else { | ||||||
|                   try { |                   try { | ||||||
|                      let data = await new Promise<string>((yes, no) => { |                      let data = await new Promise<string>((yes, no) => { | ||||||
|                         let fr = new FileReader() |                         let fr = new FileReader(); | ||||||
|                         fr.onload = (ev) => { |                         fr.onload = (ev) => { | ||||||
|                            yes((ev.target as any).result); |                            yes((ev.target as any).result); | ||||||
|                         } |                         }; | ||||||
|                         fr.onerror = no; |                         fr.onerror = no; | ||||||
|                         fr.readAsText(file); |                         fr.readAsText(file); | ||||||
|                      }) |                      }); | ||||||
|                      let parsed = JSON.parse(data); |                      let parsed = JSON.parse(data); | ||||||
|                      let c = new Error("Could not parse!"); |                      let c = new Error("Could not parse!"); | ||||||
|                      let notes: { content: string, time: Date }[] = null; |                      let notes: { content: string; time: Date }[] = null; | ||||||
|                      if (Array.isArray(parsed)) { // Could be from SecureNotes 1 |                      if (Array.isArray(parsed)) { | ||||||
|                         notes = parsed.map(elm => { |                         // Could be from SecureNotes 1 | ||||||
|                            if (typeof elm.message !== "string" || typeof elm.time !== "string") { |                         notes = parsed.map((elm) => { | ||||||
|  |                            if ( | ||||||
|  |                               typeof elm.message !== "string" || | ||||||
|  |                               typeof elm.time !== "string" | ||||||
|  |                            ) { | ||||||
|                               throw c; |                               throw c; | ||||||
|                            } |                            } | ||||||
|  |  | ||||||
|                            return { |                            return { | ||||||
|                               content: elm.message, |                               content: elm.message, | ||||||
|                               time: new Date(elm.time) |                               time: new Date(elm.time), | ||||||
|                            } |                            }; | ||||||
|                         }) |                         }); | ||||||
|                      } else if (parsed.version) { // Could be from SecureNotes 2 |                      } else if (parsed.version) { | ||||||
|                         if (parsed.version === 1) { //Could be from SecureNotes 2 version 1 |                         // Could be from SecureNotes 2 | ||||||
|                            notes = (parsed.notes as any[]).map(elm => { |                         if (parsed.version === 1) { | ||||||
|                               if (typeof elm.content !== "string" || typeof elm.time !== "string") { |                            //Could be from SecureNotes 2 version 1 | ||||||
|  |                            notes = (parsed.notes as any[]).map((elm) => { | ||||||
|  |                               if ( | ||||||
|  |                                  typeof elm.content !== "string" || | ||||||
|  |                                  typeof elm.time !== "string" | ||||||
|  |                               ) { | ||||||
|                                  throw c; |                                  throw c; | ||||||
|                               } |                               } | ||||||
|                               return { |                               return { | ||||||
|                                  content: elm.content, |                                  content: elm.content, | ||||||
|                                  time: new Date(elm.time) |                                  time: new Date(elm.time), | ||||||
|                               } |                               }; | ||||||
|                            }) |                            }); | ||||||
|                         } else { |                         } else { | ||||||
|                            throw c; |                            throw c; | ||||||
|                         } |                         } | ||||||
|                      } else |                      } else throw c; | ||||||
|                         throw c; |  | ||||||
|  |  | ||||||
|                      await Promise.all(notes.map(n => { |                      await Promise.all( | ||||||
|                         let note = this.vault.newNote(); |                         notes.map((n) => { | ||||||
|                         note.__value = n.content; |                            let note = this.vault.newNote(); | ||||||
|                         return this.vault.saveNote(note, n.time); |                            note.__value = n.content; | ||||||
|                      })); |                            return this.vault.saveNote(note, n.time); | ||||||
|  |                         }) | ||||||
|  |                      ); | ||||||
|                      await this.reloadNotes(); |                      await this.reloadNotes(); | ||||||
|                      Notifications.sendSuccess(`Imported ${notes.length} notes!`); |                      Notifications.sendSuccess( | ||||||
|  |                         `Imported ${notes.length} notes!` | ||||||
|  |                      ); | ||||||
|                   } catch (err) { |                   } catch (err) { | ||||||
|                      Notifications.sendError("Cannot read File!") |                      Notifications.sendError("Cannot read File!"); | ||||||
|                      console.error(err); |                      console.error(err); | ||||||
|                   } |                   } | ||||||
|                } |                } | ||||||
| @ -198,8 +225,7 @@ export default class EntryList extends Component<{ vault: Promise<IVault> }, { n | |||||||
|          return; |          return; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       if (this.searchLock.locked) |       if (this.searchLock.locked) return; | ||||||
|          return; |  | ||||||
|       const lock = await this.searchLock.getLock(); |       const lock = await this.searchLock.getLock(); | ||||||
|       console.time("SearchOP"); |       console.time("SearchOP"); | ||||||
|  |  | ||||||
| @ -213,7 +239,7 @@ export default class EntryList extends Component<{ vault: Promise<IVault> }, { n | |||||||
|             return parts.every(function (el) { |             return parts.every(function (el) { | ||||||
|                return note.preview.toLowerCase().indexOf(el) > -1; |                return note.preview.toLowerCase().indexOf(el) > -1; | ||||||
|             }); |             }); | ||||||
|          } |          }; | ||||||
|  |  | ||||||
|          let elements: BaseNote[]; |          let elements: BaseNote[]; | ||||||
|          if (!force && this.oldSearch && search.startsWith(this.oldSearch)) { |          if (!force && this.oldSearch && search.startsWith(this.oldSearch)) { | ||||||
| @ -222,29 +248,32 @@ export default class EntryList extends Component<{ vault: Promise<IVault> }, { n | |||||||
|             elements = [...this.rawNotes]; |             elements = [...this.rawNotes]; | ||||||
|          } |          } | ||||||
|  |  | ||||||
|  |          await new Promise((yes) => { | ||||||
|          await new Promise(yes => { |  | ||||||
|             const idle = () => { |             const idle = () => { | ||||||
|                window.requestIdleCallback(deadline => { |                window.requestIdleCallback( | ||||||
|                   let invTR = deadline.timeRemaining() <= 0; |                   (deadline) => { | ||||||
|                   while ((deadline.timeRemaining() > 0 || invTR) && elements.length > 0) { |                      let invTR = deadline.timeRemaining() <= 0; | ||||||
|                      let element = elements.shift(); |                      while ( | ||||||
|                      if (match(element)) { |                         (deadline.timeRemaining() > 0 || invTR) && | ||||||
|                         notes.push(element); |                         elements.length > 0 | ||||||
|  |                      ) { | ||||||
|  |                         let element = elements.shift(); | ||||||
|  |                         if (match(element)) { | ||||||
|  |                            notes.push(element); | ||||||
|  |                         } | ||||||
|                      } |                      } | ||||||
|                   } |                      if (elements.length > 0) idle(); | ||||||
|                   if (elements.length > 0) |                      else yes(); | ||||||
|                      idle(); |                   }, | ||||||
|                   else |                   { timeout: 100 } | ||||||
|                      yes(); |                ); | ||||||
|                }, { timeout: 100 }); |             }; | ||||||
|             } |  | ||||||
|             idle(); |             idle(); | ||||||
|          }) |          }); | ||||||
|          // notes = elements.filter(note => match(note)); |          // notes = elements.filter(note => match(note)); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       await new Promise(yes => this.setState({ notes }, yes)); |       await new Promise((yes) => this.setState({ notes }, yes)); | ||||||
|       this.oldSearch = search; |       this.oldSearch = search; | ||||||
|       lock.release(); |       lock.release(); | ||||||
|       console.timeEnd("SearchOP"); |       console.timeEnd("SearchOP"); | ||||||
| @ -260,38 +289,67 @@ export default class EntryList extends Component<{ vault: Promise<IVault> }, { n | |||||||
|    searchInput: HTMLInputElement; |    searchInput: HTMLInputElement; | ||||||
|    render() { |    render() { | ||||||
|       const open_entry = (id: string | null) => { |       const open_entry = (id: string | null) => { | ||||||
|          Navigation.setPage("/vault", { id: this.vault.id }, { id, entry: "true" }) |          Navigation.setPage( | ||||||
|       } |             "/vault", | ||||||
|  |             { id: this.vault.id }, | ||||||
|  |             { id, entry: "true" } | ||||||
|  |          ); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|       return <div> |       return ( | ||||||
|          {this.state.context} |          <div> | ||||||
|          <header class="header"> |             {this.state.context} | ||||||
|             <a class="header-icon-button" onClick={() => history.back()}><ArrowLeft height={undefined} width={undefined} /></a> |             <header class="header"> | ||||||
|             <h3 style="display:inline" class="header-title" onClick={() => Navigation.setPage("/")}>{this.vault ? this.vault.name : ""}</h3> |                <a class="header-icon-button" onClick={() => history.back()}> | ||||||
|             <span></span> |                   <ArrowLeft height={undefined} width={undefined} /> | ||||||
|             {/* <a class="button header_icon_button"><MoreVertival height={undefined} width={undefined} /></a> */} |                </a> | ||||||
|          </header> |                <h3 | ||||||
|          <AddButton onClick={() => open_entry(null)} /> |                   style="display:inline" | ||||||
|          <div class="container"> |                   class="header-title" | ||||||
|             <div style="display:flex;"> |                   onClick={() => Navigation.setPage("/")} | ||||||
|                <input class="inp" type="text" style="width: 100%; height: 40px;" onKeyUp={this.searchChanged} ref={elm => this.searchInput = elm} /> |                > | ||||||
|                <button class="btn btn-primary" style="padding: 5px 10px; margin: 0; height: 40px; width: 40px;"> |                   {this.vault ? this.vault.name : ""} | ||||||
|                   <Search /> |                </h3> | ||||||
|                </button> |                <span></span> | ||||||
|             </div> |                {/* <a class="button header_icon_button"><MoreVertival height={undefined} width={undefined} /></a> */} | ||||||
|  |             </header> | ||||||
|  |             <AddButton onClick={() => open_entry(null)} /> | ||||||
|  |             <div class="container"> | ||||||
|  |                <div style="display:flex;"> | ||||||
|  |                   <input | ||||||
|  |                      class="inp" | ||||||
|  |                      type="text" | ||||||
|  |                      style="width: 100%; height: 40px;" | ||||||
|  |                      onKeyUp={this.searchChanged} | ||||||
|  |                      ref={(elm) => (this.searchInput = elm)} | ||||||
|  |                   /> | ||||||
|  |                   <button | ||||||
|  |                      class="btn btn-primary" | ||||||
|  |                      style="padding: 5px 10px; margin: 0; height: 40px; width: 40px;" | ||||||
|  |                   > | ||||||
|  |                      <Search /> | ||||||
|  |                   </button> | ||||||
|  |                </div> | ||||||
|  |  | ||||||
|             <div class="vault-list" style="margin-top: 1rem;"> |                <div class="vault-list" style="margin-top: 1rem;"> | ||||||
|                {this.state.notes.map(note => { |                   {this.state.notes.map((note) => { | ||||||
|                   let [first, second] = note.preview.split("\n", 2); |                      let [first, second] = note.preview.split("\n", 2); | ||||||
|                   return <div class="card vault-vault" onContextMenu={evt => this.onContext(evt, note)} onClick={() => { |                      return ( | ||||||
|                      open_entry(note._id) |                         <div | ||||||
|                   }}> |                            class="card vault-vault" | ||||||
|                      <div>{first}</div> |                            onContextMenu={(evt) => this.onContext(evt, note)} | ||||||
|                      <div>{second}</div> |                            onClick={() => { | ||||||
|                   </div> |                               open_entry(note._id); | ||||||
|                })} |                            }} | ||||||
|  |                         > | ||||||
|  |                            <div>{first}</div> | ||||||
|  |                            <div>{second}</div> | ||||||
|  |                         </div> | ||||||
|  |                      ); | ||||||
|  |                   })} | ||||||
|  |                </div> | ||||||
|             </div> |             </div> | ||||||
|          </div> |          </div> | ||||||
|       </div>; |       ); | ||||||
|    } |    } | ||||||
| } | } | ||||||
| @ -1,47 +1,55 @@ | |||||||
| import { h } from "preact" | import { h } from "preact"; | ||||||
| import { Page } from "../../../page"; | import { Page } from "../../../page"; | ||||||
| import Notes, { IVault, BaseNote } from "../../../notes"; | import Notes, { IVault, BaseNote } from "../../../notes"; | ||||||
| import Navigation from "../../../navigation"; | import Navigation from "../../../navigation"; | ||||||
| import EntryComponent from "./Entry"; | import EntryComponent from "./Entry"; | ||||||
| import EntryList from "./EntryList"; | import EntryList from "./EntryList"; | ||||||
|  |  | ||||||
| import "./vault.scss" | import "./vault.scss"; | ||||||
|  | import { useState } from "preact/hooks"; | ||||||
|  | import { usePromise } from "../../../hooks"; | ||||||
|  |  | ||||||
| export interface VaultProps { | export interface VaultProps { | ||||||
|    state: { id: string }; |    state: { id: string }; | ||||||
|    hidden: { entry: string, id: string, note: string }; |    hidden: { entry: string; id: string; note: string }; | ||||||
| } | } | ||||||
|  |  | ||||||
| export default class VaultPage extends Page<VaultProps, { entries: BaseNote[] }> { | export default function VaultPage(props: VaultProps) { | ||||||
|    vault: Promise<IVault> |    console.log("Vault page"); | ||||||
|    constructor(props: VaultProps) { |    const [loading, error, vault] = usePromise( | ||||||
|       super(props); |       () => | ||||||
|       this.state = { entries: [] }; |          Notes.getVault( | ||||||
|    } |             this.props.state.id, | ||||||
|  |             Notes.getVaultKey(this.props.state.id) | ||||||
|  |          ), | ||||||
|  |       [props.state.id] | ||||||
|  |    ); | ||||||
|  |  | ||||||
|    async componentWillMount() { |    if (vault) { | ||||||
|       this.vault = Notes.getVault(this.props.state.id, Notes.getVaultKey(this.props.state.id)) |       window.debug.activeVault = vault; | ||||||
|       this.vault.then(vlt => { |       window.debug.createNotes = (cnt = 10) => { | ||||||
|          window.debug.activeVault = vlt; |          for (let i = 0; i < cnt; i++) { | ||||||
|          window.debug.createNotes = (cnt = 10) => { |             let nt = vault.newNote(); | ||||||
|             for (let i = 0; i < cnt; i++) { |             nt.__value = `Random Note ${i}\ With some Content ${i}`; | ||||||
|                let nt = vlt.newNote(); |             vault.saveNote(nt); | ||||||
|                nt.__value = `Random Note ${i}\ With some Content ${i}`; |  | ||||||
|                vlt.saveNote(nt); |  | ||||||
|             } |  | ||||||
|          } |          } | ||||||
|       }) |       }; | ||||||
|       this.vault.catch(err => { |  | ||||||
|          Navigation.setPage("/") |  | ||||||
|       }); |  | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    render() { |    if (loading) return <div>Loading Vault</div>; | ||||||
|       if (this.props.hidden && this.props.hidden.entry === "true") { |    // Maybe return loading animation or so | ||||||
|          return <EntryComponent vault={this.vault} id={this.props.hidden.id} note={this.props.hidden.note} /> |    else if (error) return <div>{error.message}</div>; | ||||||
|  |    else { | ||||||
|  |       if (props.hidden?.entry) { | ||||||
|  |          return ( | ||||||
|  |             <EntryComponent | ||||||
|  |                vault={vault} | ||||||
|  |                id={props.hidden.id} | ||||||
|  |                note={props.hidden.note} | ||||||
|  |             /> | ||||||
|  |          ); | ||||||
|       } else { |       } else { | ||||||
|          return <EntryList vault={this.vault} /> |          return <EntryList vault={vault} />; | ||||||
|       } |       } | ||||||
|    } |    } | ||||||
| } | } | ||||||
| @ -1,10 +1,8 @@ | |||||||
| import { h } from "preact" | import { h } from "preact"; | ||||||
| import { Page } from "../../../page"; | import { Page } from "../../../page"; | ||||||
| import Notes, { VaultList } from "../../../notes"; | import Notes, { VaultList } from "../../../notes"; | ||||||
| import "./vaults.scss" | import "./vaults.scss"; | ||||||
| import Lock from "feather-icons/dist/icons/lock.svg"; | import { Lock, Unlock, Settings } from "preact-feather"; | ||||||
| import Unlock from "feather-icons/dist/icons/unlock.svg"; |  | ||||||
| import Settings from "feather-icons/dist/icons/settings.svg" |  | ||||||
| import Navigation from "../../../navigation"; | import Navigation from "../../../navigation"; | ||||||
| import { InputModal } from "../../modals/InputModal"; | import { InputModal } from "../../modals/InputModal"; | ||||||
| import { YesNoModal } from "../../modals/YesNoModal"; | import { YesNoModal } from "../../modals/YesNoModal"; | ||||||
| @ -18,7 +16,14 @@ export interface VaultsProps { | |||||||
|    onSelected?: (vaultid: string) => void; |    onSelected?: (vaultid: string) => void; | ||||||
| } | } | ||||||
|  |  | ||||||
| export default class VaultsPage extends Page<VaultsProps, { vaults: VaultList, modal: JSX.Element | undefined, context: JSX.Element | undefined }> { | export default class VaultsPage extends Page< | ||||||
|  |    VaultsProps, | ||||||
|  |    { | ||||||
|  |       vaults: VaultList; | ||||||
|  |       modal: h.JSX.Element | undefined; | ||||||
|  |       context: h.JSX.Element | undefined; | ||||||
|  |    } | ||||||
|  | > { | ||||||
|    constructor(props: VaultsProps) { |    constructor(props: VaultsProps) { | ||||||
|       super(props); |       super(props); | ||||||
|       this.state = { vaults: [], modal: undefined, context: undefined }; |       this.state = { vaults: [], modal: undefined, context: undefined }; | ||||||
| @ -26,15 +31,14 @@ export default class VaultsPage extends Page<VaultsProps, { vaults: VaultList, m | |||||||
|    } |    } | ||||||
|  |  | ||||||
|    updateVaults(s?: boolean) { |    updateVaults(s?: boolean) { | ||||||
|       if (s) |       if (s) return; | ||||||
|          return; |       return new Promise((yes) => { | ||||||
|       return new Promise(yes => { |          Notes.getVaults().then((vaults) => this.setState({ vaults }, yes)); | ||||||
|          Notes.getVaults().then(vaults => this.setState({ vaults }, yes)) |       }); | ||||||
|       }) |  | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    componentWillMount() { |    componentWillMount() { | ||||||
|       this.updateVaults() |       this.updateVaults(); | ||||||
|       Notes.syncObservable.subscribe(this.updateVaults); |       Notes.syncObservable.subscribe(this.updateVaults); | ||||||
|    } |    } | ||||||
|  |  | ||||||
| @ -42,15 +46,19 @@ export default class VaultsPage extends Page<VaultsProps, { vaults: VaultList, m | |||||||
|       Notes.syncObservable.unsubscribe(this.updateVaults); |       Notes.syncObservable.unsubscribe(this.updateVaults); | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    async getKey(vault: { name: string, id: string }, permanent = true) { |    async getKey(vault: { name: string; id: string }, permanent = true) { | ||||||
|       let inp_mod = new InputModal("Enter password for " + vault.name, "Password", "password"); |       let inp_mod = new InputModal( | ||||||
|  |          "Enter password for " + vault.name, | ||||||
|  |          "Password", | ||||||
|  |          "password" | ||||||
|  |       ); | ||||||
|       let key = undefined; |       let key = undefined; | ||||||
|       while (true) { |       while (true) { | ||||||
|          // inp_mod.show(); |          // inp_mod.show(); | ||||||
|          let value = await inp_mod.getResult(false); |          let value = await inp_mod.getResult(false); | ||||||
|          if (value === null) { |          if (value === null) { | ||||||
|             console.log("Value is null") |             console.log("Value is null"); | ||||||
|             inp_mod.close() |             inp_mod.close(); | ||||||
|             return false; |             return false; | ||||||
|          } else { |          } else { | ||||||
|             key = Notes.passwordToKey(value); |             key = Notes.passwordToKey(value); | ||||||
| @ -58,11 +66,11 @@ export default class VaultsPage extends Page<VaultsProps, { vaults: VaultList, m | |||||||
|                await Notes.getVault(vault.id, key); |                await Notes.getVault(vault.id, key); | ||||||
|                break; |                break; | ||||||
|             } catch (err) { |             } catch (err) { | ||||||
|                Notifications.sendError("Invalid password!") |                Notifications.sendError("Invalid password!"); | ||||||
|             } |             } | ||||||
|          } |          } | ||||||
|       } |       } | ||||||
|       inp_mod.close() |       inp_mod.close(); | ||||||
|  |  | ||||||
|       let perm = false; |       let perm = false; | ||||||
|       if (permanent) { |       if (permanent) { | ||||||
| @ -79,27 +87,24 @@ export default class VaultsPage extends Page<VaultsProps, { vaults: VaultList, m | |||||||
|       return true; |       return true; | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    async openVault(vault: { name: string, encrypted: boolean, id: string }) { |    async openVault(vault: { name: string; encrypted: boolean; id: string }) { | ||||||
|       const action = () => { |       const action = () => { | ||||||
|          if (this.props.selectVault) { |          if (this.props.selectVault) { | ||||||
|             this.props.onSelected(vault.id); |             this.props.onSelected(vault.id); | ||||||
|          } else { |          } else { | ||||||
|             Navigation.setPage("/vault", { id: vault.id }) |             Navigation.setPage("/vault", { id: vault.id }); | ||||||
|          } |          } | ||||||
|       } |       }; | ||||||
|  |  | ||||||
|       if (vault.encrypted) { |       if (vault.encrypted) { | ||||||
|          let key = Notes.getVaultKey(vault.id); |          let key = Notes.getVaultKey(vault.id); | ||||||
|          if (key) |          if (key) action(); | ||||||
|             action() |  | ||||||
|          else { |          else { | ||||||
|             if (await this.getKey(vault)) |             if (await this.getKey(vault)) action(); | ||||||
|                action(); |  | ||||||
|          } |          } | ||||||
|       } else { |       } else { | ||||||
|          action() |          action(); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    async addButtonClick() { |    async addButtonClick() { | ||||||
| @ -113,122 +118,176 @@ export default class VaultsPage extends Page<VaultsProps, { vaults: VaultList, m | |||||||
|  |  | ||||||
|       let password; |       let password; | ||||||
|       if (encrypted) { |       if (encrypted) { | ||||||
|          let password_modal = new InputModal("Enter new password", "Password", "password"); |          let password_modal = new InputModal( | ||||||
|  |             "Enter new password", | ||||||
|  |             "Password", | ||||||
|  |             "password" | ||||||
|  |          ); | ||||||
|          password = await password_modal.getResult(); |          password = await password_modal.getResult(); | ||||||
|          if (password === null) return; |          if (password === null) return; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       let key; |       let key; | ||||||
|       if (password) { |       if (password) { | ||||||
|          key = Notes.passwordToKey(password) |          key = Notes.passwordToKey(password); | ||||||
|       } |       } | ||||||
|       await Notes.createVault(name, key) |       await Notes.createVault(name, key); | ||||||
|       this.updateVaults(); |       this.updateVaults(); | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    onContext(evt: MouseEvent, vault: { name: string, encrypted: boolean, id: string }) { |    onContext( | ||||||
|  |       evt: MouseEvent, | ||||||
|  |       vault: { name: string; encrypted: boolean; id: string } | ||||||
|  |    ) { | ||||||
|       evt.preventDefault(); |       evt.preventDefault(); | ||||||
|       evt.stopPropagation(); |       evt.stopPropagation(); | ||||||
|  |  | ||||||
|       const close = () => { |       const close = () => { | ||||||
|          document.documentElement.removeEventListener("click", close); |          document.documentElement.removeEventListener("click", close); | ||||||
|          this.setState({ context: undefined }); |          this.setState({ context: undefined }); | ||||||
|       } |       }; | ||||||
|       document.documentElement.addEventListener("click", close); |       document.documentElement.addEventListener("click", close); | ||||||
|  |  | ||||||
|       let deleteb = <button class="btn" onClick={async () => { |       let deleteb = ( | ||||||
|          let delete_modal = new YesNoModal("Delete Vault? Cannot be undone!"); |          <button | ||||||
|          let result = await delete_modal.getResult(); |             class="btn" | ||||||
|          if (result) { |             onClick={async () => { | ||||||
|             Notes.deleteVault(vault.id).then(() => { |                let delete_modal = new YesNoModal( | ||||||
|                this.updateVaults(); |                   "Delete Vault? Cannot be undone!" | ||||||
|             }).catch(err => { |                ); | ||||||
|                Notifications.sendError("Error deleting vault!") |                let result = await delete_modal.getResult(); | ||||||
|                console.error(err); |                if (result) { | ||||||
|             }) |                   Notes.deleteVault(vault.id) | ||||||
|          } |                      .then(() => { | ||||||
|       }}> |                         this.updateVaults(); | ||||||
|          delete |                      }) | ||||||
|       </button>; |                      .catch((err) => { | ||||||
|  |                         Notifications.sendError("Error deleting vault!"); | ||||||
|  |                         console.error(err); | ||||||
|  |                      }); | ||||||
|  |                } | ||||||
|  |             }} | ||||||
|  |          > | ||||||
|  |             delete | ||||||
|  |          </button> | ||||||
|  |       ); | ||||||
|  |  | ||||||
|       let delete_key; |       let delete_key; | ||||||
|       if (Notes.getVaultKey(vault.id)) { |       if (Notes.getVaultKey(vault.id)) { | ||||||
|          delete_key = <button class="btn" onClick={() => { |          delete_key = ( | ||||||
|             Notes.forgetVaultKey(vault.id); |             <button | ||||||
|             Notifications.sendSuccess("Forgot password!") |                class="btn" | ||||||
|          }}> |                onClick={() => { | ||||||
|             forget password |                   Notes.forgetVaultKey(vault.id); | ||||||
|          </button>; |                   Notifications.sendSuccess("Forgot password!"); | ||||||
|  |                }} | ||||||
|  |             > | ||||||
|  |                forget password | ||||||
|  |             </button> | ||||||
|  |          ); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       let exportb = <button class="btn" onClick={async () => { |       let exportb = ( | ||||||
|          let key: Uint8Array; |          <button | ||||||
|          if (vault.encrypted) { |             class="btn" | ||||||
|             await this.getKey(vault, false) |             onClick={async () => { | ||||||
|             key = Notes.getVaultKey(vault.id); |                let key: Uint8Array; | ||||||
|          } |                if (vault.encrypted) { | ||||||
|          let note_vault = await Notes.getVault(vault.id, key); |                   await this.getKey(vault, false); | ||||||
|          let base_notes = await note_vault.getAllNotes(); |                   key = Notes.getVaultKey(vault.id); | ||||||
|          let notes = await Promise.all(base_notes.map(e => { |  | ||||||
|             return note_vault.getNote(e._id); |  | ||||||
|          })); |  | ||||||
|  |  | ||||||
|          let result = |  | ||||||
|          { |  | ||||||
|             version: 1, |  | ||||||
|             notes: notes.map(e => { |  | ||||||
|                return { |  | ||||||
|                   content: e.__value, |  | ||||||
|                   time: e.time |  | ||||||
|                } |                } | ||||||
|             }) |                let note_vault = await Notes.getVault(vault.id, key); | ||||||
|          } |                let base_notes = await note_vault.getAllNotes(); | ||||||
|  |                let notes = await Promise.all( | ||||||
|  |                   base_notes.map((e) => { | ||||||
|  |                      return note_vault.getNote(e._id); | ||||||
|  |                   }) | ||||||
|  |                ); | ||||||
|  |  | ||||||
|          var dataStr = "data:text/json;charset=utf-8," + encodeURIComponent(JSON.stringify(result, undefined, 3)); |                let result = { | ||||||
|          var downloadAnchorNode = document.createElement('a'); |                   version: 1, | ||||||
|          downloadAnchorNode.setAttribute("href", dataStr); |                   notes: notes.map((e) => { | ||||||
|          downloadAnchorNode.setAttribute("download", "notes_export_" + vault.name + ".json"); |                      return { | ||||||
|          document.body.appendChild(downloadAnchorNode); // required for firefox |                         content: e.__value, | ||||||
|          downloadAnchorNode.click(); |                         time: e.time, | ||||||
|          downloadAnchorNode.remove(); |                      }; | ||||||
|       }}> |                   }), | ||||||
|          export |                }; | ||||||
|       </button>; |  | ||||||
|  |  | ||||||
|       let context = <ContextMenu event={evt} > |                var dataStr = | ||||||
|          {deleteb} |                   "data:text/json;charset=utf-8," + | ||||||
|          {delete_key} |                   encodeURIComponent(JSON.stringify(result, undefined, 3)); | ||||||
|          {exportb} |                var downloadAnchorNode = document.createElement("a"); | ||||||
|       </ContextMenu> |                downloadAnchorNode.setAttribute("href", dataStr); | ||||||
|  |                downloadAnchorNode.setAttribute( | ||||||
|  |                   "download", | ||||||
|  |                   "notes_export_" + vault.name + ".json" | ||||||
|  |                ); | ||||||
|  |                document.body.appendChild(downloadAnchorNode); // required for firefox | ||||||
|  |                downloadAnchorNode.click(); | ||||||
|  |                downloadAnchorNode.remove(); | ||||||
|  |             }} | ||||||
|  |          > | ||||||
|  |             export | ||||||
|  |          </button> | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       let context = ( | ||||||
|  |          <ContextMenu event={evt}> | ||||||
|  |             {deleteb} | ||||||
|  |             {delete_key} | ||||||
|  |             {exportb} | ||||||
|  |          </ContextMenu> | ||||||
|  |       ); | ||||||
|  |  | ||||||
|       this.setState({ context }); |       this.setState({ context }); | ||||||
|       return false; |       return false; | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    render() { |    render() { | ||||||
|       let elms = this.state.vaults.map(vault => { |       let elms = this.state.vaults.map((vault) => { | ||||||
|          return <li class="vaults_vault" onClick={() => this.openVault(vault)} onContextMenu={(evt) => this.onContext(evt, vault)}> |          return ( | ||||||
|             {vault.encrypted ? <Lock height={undefined} width={undefined} /> : <Unlock height={undefined} width={undefined} />} |             <li | ||||||
|             <span> |                class="vaults_vault" | ||||||
|                {vault.name} |                onClick={() => this.openVault(vault)} | ||||||
|             </span> |                onContextMenu={(evt) => this.onContext(evt, vault)} | ||||||
|          </li> |             > | ||||||
|       }) |                {vault.encrypted ? ( | ||||||
|  |                   <Lock height={undefined} width={undefined} /> | ||||||
|  |                ) : ( | ||||||
|  |                   <Unlock height={undefined} width={undefined} /> | ||||||
|  |                )} | ||||||
|  |                <span>{vault.name}</span> | ||||||
|  |             </li> | ||||||
|  |          ); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|       return <div style={{ marginTop: "-12px", paddingTop: "12px" }} > |       return ( | ||||||
|          {/* {this.state.modal} */} |          <div style={{ marginTop: "-12px", paddingTop: "12px" }}> | ||||||
|          {this.state.context} |             {/* {this.state.modal} */} | ||||||
|          <header class="header"> |             {this.state.context} | ||||||
|             <span></span> |             <header class="header"> | ||||||
|             <h3 style="display:inline" onClick={() => Navigation.setPage("/")}>{this.props.selectVault ? "Select Vault for share" : "Your vaults:"}</h3> |                <span></span> | ||||||
|             <a class="header-icon-button" onClick={() => Navigation.setPage("/settings")}><Settings height={undefined} width={undefined} /></a> |                <h3 | ||||||
|          </header> |                   style="display:inline" | ||||||
|          <AddButton onClick={() => this.addButtonClick()} /> |                   onClick={() => Navigation.setPage("/")} | ||||||
|          <div class="container"> |                > | ||||||
|             <ul class="list list-divider list-clickable"> |                   {this.props.selectVault | ||||||
|                {elms} |                      ? "Select Vault for share" | ||||||
|             </ul> |                      : "Your vaults:"} | ||||||
|  |                </h3> | ||||||
|  |                <a | ||||||
|  |                   class="header-icon-button" | ||||||
|  |                   onClick={() => Navigation.setPage("/settings")} | ||||||
|  |                > | ||||||
|  |                   <Settings height={undefined} width={undefined} /> | ||||||
|  |                </a> | ||||||
|  |             </header> | ||||||
|  |             <AddButton onClick={() => this.addButtonClick()} /> | ||||||
|  |             <div class="container"> | ||||||
|  |                <ul class="list list-divider list-clickable">{elms}</ul> | ||||||
|  |             </div> | ||||||
|          </div> |          </div> | ||||||
|       </div> |       ); | ||||||
|    } |    } | ||||||
| } | } | ||||||
							
								
								
									
										23
									
								
								src/hooks.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,23 @@ | |||||||
|  | import { useEffect, useMemo, useState } from "preact/hooks"; | ||||||
|  |  | ||||||
|  | export function usePromise<T>(promiseFnc: () => Promise<T>, params: any[]) { | ||||||
|  |    const promise = useMemo(promiseFnc, params); | ||||||
|  |    const [loading, setLoading] = useState(true); | ||||||
|  |    const [error, setError] = useState(undefined); | ||||||
|  |    const [value, setValue] = useState(undefined); | ||||||
|  |  | ||||||
|  |    useEffect(() => { | ||||||
|  |       let canceled = false; | ||||||
|  |  | ||||||
|  |       promise | ||||||
|  |          .then((res) => setValue(res)) | ||||||
|  |          .catch((err) => setError(err)) | ||||||
|  |          .finally(() => setLoading(false)); | ||||||
|  |  | ||||||
|  |       return () => { | ||||||
|  |          canceled = true; | ||||||
|  |       }; | ||||||
|  |    }, [promise]); | ||||||
|  |  | ||||||
|  |    return [loading, error, value]; | ||||||
|  | } | ||||||
							
								
								
									
										113
									
								
								src/index.html
									
									
									
									
									
								
							
							
						
						| @ -1,61 +1,70 @@ | |||||||
| <!DOCTYPE html> | <!DOCTYPE html> | ||||||
| <html lang="en"> | <html lang="en"> | ||||||
|  |    <head> | ||||||
|  |       <title>SecureNotes</title> | ||||||
|  |       <meta charset="utf8" /> | ||||||
|  |       <meta name="Description" content="Notes app" /> | ||||||
|  |       <meta name="viewport" content="width=device-width,initial-scale=1" /> | ||||||
|  |       <link rel="manifest" href="./manifest.webmanifest" /> | ||||||
|  |       <!-- <link rel="shortcut icon" href="/public/icon-72x72.png"> --> | ||||||
|  |  | ||||||
| <head> |       <!-- Add to home screen for Safari on iOS --> | ||||||
|    <title>SecureNotes</title> |       <meta name="apple-mobile-web-app-capable" content="yes" /> | ||||||
|    <meta charset="utf8" /> |       <meta name="apple-mobile-web-app-status-bar-style" content="black" /> | ||||||
|    <meta name="Description" content="Notes app"> |       <meta name="apple-mobile-web-app-title" content="Secure Notes" /> | ||||||
|    <meta name="viewport" content="width=device-width,initial-scale=1" /> |  | ||||||
|    <link rel="manifest" href="/manifest.json"> |  | ||||||
|    <!-- <link rel="shortcut icon" href="/public/icon-72x72.png"> --> |  | ||||||
|  |  | ||||||
|    <!-- Add to home screen for Safari on iOS --> |       <!-- sizes="180x180" --> | ||||||
|    <meta name="apple-mobile-web-app-capable" content="yes"> |       <!-- href="/public/apple-touch-icon.png" --> | ||||||
|    <meta name="apple-mobile-web-app-status-bar-style" content="black"> |  | ||||||
|    <meta name="apple-mobile-web-app-title" content="Secure Notes"> |  | ||||||
|  |  | ||||||
|  |       <link | ||||||
|  |          rel="apple-touch-icon" | ||||||
|  |          sizes="256x256" | ||||||
|  |          href="/public/notepad256.png" | ||||||
|  |       /> | ||||||
|  |       <!-- <link | ||||||
|  |          rel="mask-icon" | ||||||
|  |          href="/public/safari-pinned-tab.svg" | ||||||
|  |          color="#1E88E5" | ||||||
|  |       /> --> | ||||||
|  |       <meta name="msapplication-TileColor" content="#1E88E5" /> | ||||||
|  |       <meta name="theme-color" content="#1E88E5" /> | ||||||
|  |    </head> | ||||||
|  |  | ||||||
|    <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png"> |    <body> | ||||||
|    <link rel="mask-icon" href="/safari-pinned-tab.svg" color="#1E88E5"> |       <noscript> You have to enable JavaScript to use this site! </noscript> | ||||||
|    <script src="encoding.js"></script> |       <div id="app"></div> | ||||||
|    <meta name="msapplication-TileColor" content="#1E88E5"> |       <script> | ||||||
|    <meta name="theme-color" content="#1E88E5"> |          if (navigator.serviceWorker.controller) { | ||||||
| </head> |             if (localStorage.getItem("debug")) { | ||||||
|  |                console.warn( | ||||||
| <body> |                   "Debuggung and service worker found, make shure to clear cache!" | ||||||
|    <noscript> |                ); | ||||||
|       You have to enable JavaScript to use this site! |             } | ||||||
|    </noscript> |             console.log("active service worker found, no need to register"); | ||||||
|    <div id="app"></div> |  | ||||||
|    <script> |  | ||||||
|       // // Check that service workers are registered |  | ||||||
|       // if ('serviceWorker' in navigator) { |  | ||||||
|       //    // Use the window load event to keep the page load performant |  | ||||||
|       //    window.addEventListener('load', () => { |  | ||||||
|       //       navigator.serviceWorker.register('/public/serviceworker.js'); |  | ||||||
|       //    }); |  | ||||||
|       // } |  | ||||||
|       if (navigator.serviceWorker.controller) { |  | ||||||
|          if (localStorage.getItem("debug")) { |  | ||||||
|             console.warn("Debuggung and service worker found, make shure to clear cache!"); |  | ||||||
|          } |  | ||||||
|          console.log('active service worker found, no need to register') |  | ||||||
|       } else { |  | ||||||
|          if (localStorage.getItem("debug")) { |  | ||||||
|             console.warn("Disabling Service Worker in debug mode!") |  | ||||||
|          } else { |          } else { | ||||||
|             // Register the ServiceWorker |             if (localStorage.getItem("debug")) { | ||||||
|             navigator.serviceWorker.register('/serviceworker.js', { |                console.warn("Disabling Service Worker in debug mode!"); | ||||||
|                scope: '/' |             } else { | ||||||
|             }).then(function (reg) { |                // Register the ServiceWorker | ||||||
|                console.log('Service worker has been registered for scope:' + reg.scope); |                navigator.serviceWorker | ||||||
|                navigator.serviceWorker.controller.addEventListener("cleared_cache", evt => { |                   .register("serviceworker.js", { | ||||||
|                   console.log(evt); |                      scope: "/", | ||||||
|                }) |                   }) | ||||||
|             }); |                   .then(function (reg) { | ||||||
|  |                      console.log( | ||||||
|  |                         "Service worker has been registered for scope:" + | ||||||
|  |                            reg.scope | ||||||
|  |                      ); | ||||||
|  |                      navigator.serviceWorker.controller.addEventListener( | ||||||
|  |                         "cleared_cache", | ||||||
|  |                         (evt) => { | ||||||
|  |                            console.log(evt); | ||||||
|  |                         } | ||||||
|  |                      ); | ||||||
|  |                   }); | ||||||
|  |             } | ||||||
|          } |          } | ||||||
|       } |       </script> | ||||||
|    </script> |       <script src="index.tsx"></script> | ||||||
| </body> |    </body> | ||||||
|  |  | ||||||
| </html> | </html> | ||||||
							
								
								
									
										56
									
								
								src/manifest.webmanifest
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,56 @@ | |||||||
|  | { | ||||||
|  |    "short_name": "Secure Notes", | ||||||
|  |    "name": "Secure Notes", | ||||||
|  |    "decription": "A place to store your notes securly", | ||||||
|  |    "share_target": { | ||||||
|  |       "action": "/share", | ||||||
|  |       "method": "GET", | ||||||
|  |       "enctype": "application/x-www-form-urlencoded", | ||||||
|  |       "params": { | ||||||
|  |          "title": "title", | ||||||
|  |          "text": "text", | ||||||
|  |          "url": "url" | ||||||
|  |       } | ||||||
|  |    }, | ||||||
|  |    "icons": [ | ||||||
|  |       { | ||||||
|  |          "src": "public/notepad16.png", | ||||||
|  |          "sizes": "16x16", | ||||||
|  |          "type": "image/png" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |          "src": "public/notepad24.png", | ||||||
|  |          "sizes": "24x24", | ||||||
|  |          "type": "image/png" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |          "src": "public/notepad32.png", | ||||||
|  |          "sizes": "32x32", | ||||||
|  |          "type": "image/png" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |          "src": "public/notepad64.png", | ||||||
|  |          "sizes": "64x64", | ||||||
|  |          "type": "image/png" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |          "src": "public/notepad128.png", | ||||||
|  |          "sizes": "128x128", | ||||||
|  |          "type": "image/png" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |          "src": "public/notepad256.png", | ||||||
|  |          "sizes": "256x256", | ||||||
|  |          "type": "image/png" | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |          "src": "public/notepad512.png", | ||||||
|  |          "sizes": "512x512", | ||||||
|  |          "type": "image/png" | ||||||
|  |       } | ||||||
|  |    ], | ||||||
|  |    "start_url": "/", | ||||||
|  |    "display": "standalone", | ||||||
|  |    "theme_color": "#1E88E5", | ||||||
|  |    "background_color": "#ffffff" | ||||||
|  | } | ||||||
| @ -366,7 +366,6 @@ class NotesProvider { | |||||||
|             }; |             }; | ||||||
|  |  | ||||||
|             try { |             try { | ||||||
|                // log(id, "LRO: ", !!local, !!remote, !!oplog) |  | ||||||
|                if (remote && !oplog) { |                if (remote && !oplog) { | ||||||
|                   if (local) { |                   if (local) { | ||||||
|                      let old = |                      let old = | ||||||
|  | |||||||
							
								
								
									
										0
									
								
								public/favicon.ico → src/public/favicon.ico
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 1.1 KiB | 
							
								
								
									
										0
									
								
								public/notepad128.png → src/public/notepad128.png
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 1.3 KiB | 
							
								
								
									
										0
									
								
								public/notepad16.png → src/public/notepad16.png
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 562 B After Width: | Height: | Size: 562 B | 
							
								
								
									
										0
									
								
								public/notepad24.png → src/public/notepad24.png
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 554 B After Width: | Height: | Size: 554 B | 
							
								
								
									
										0
									
								
								public/notepad256.png → src/public/notepad256.png
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 2.0 KiB After Width: | Height: | Size: 2.0 KiB | 
							
								
								
									
										0
									
								
								public/notepad32.png → src/public/notepad32.png
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 720 B After Width: | Height: | Size: 720 B | 
							
								
								
									
										0
									
								
								public/notepad512.png → src/public/notepad512.png
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 4.1 KiB After Width: | Height: | Size: 4.1 KiB | 
							
								
								
									
										0
									
								
								public/notepad64.png → src/public/notepad64.png
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						| Before Width: | Height: | Size: 915 B After Width: | Height: | Size: 915 B | 
							
								
								
									
										130
									
								
								src/serviceworker.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @ -0,0 +1,130 @@ | |||||||
|  | function log(...params) { | ||||||
|  |    console.log.apply(this, [ | ||||||
|  |       ...["%c[SW]: %c", "color: #f4b942;", "color:unset;"], | ||||||
|  |       ...params, | ||||||
|  |    ]); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | const CACHE = "offline"; | ||||||
|  |  | ||||||
|  | let precacheFiles = ["/", "/index.html"]; | ||||||
|  |  | ||||||
|  | //Install stage sets up the cache-array to configure pre-cache content | ||||||
|  | self.addEventListener("install", (evt) => { | ||||||
|  |    log("The service worker is being installed."); | ||||||
|  |    evt.waitUntil( | ||||||
|  |       precache() | ||||||
|  |          .then(() => { | ||||||
|  |             log("Skip waiting on install"); | ||||||
|  |          }) | ||||||
|  |          .catch(log) | ||||||
|  |          .then(() => self.skipWaiting()) | ||||||
|  |    ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | //allow sw to control of current page | ||||||
|  | self.addEventListener("activate", (event) => { | ||||||
|  |    log("Claiming clients for current page"); | ||||||
|  |    return self.clients.claim(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | self.addEventListener("message", (event) => { | ||||||
|  |    log("Clearing cache"); | ||||||
|  |    caches.delete(CACHE); | ||||||
|  |    event.waitUntil(precache()); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | var Types; | ||||||
|  | (function (Types) { | ||||||
|  |    Types[(Types["CACHE"] = 0)] = "CACHE"; | ||||||
|  |    Types[(Types["NOCACHE"] = 1)] = "NOCACHE"; | ||||||
|  |    Types[(Types["REFRESH"] = 2)] = "REFRESH"; | ||||||
|  |    Types[(Types["INDEX"] = 3)] = "INDEX"; | ||||||
|  | })(Types || (Types = {})); | ||||||
|  |  | ||||||
|  | let rules = [ | ||||||
|  |    { | ||||||
|  |       match: (url) => { | ||||||
|  |          return url.indexOf("/api/") >= 0; | ||||||
|  |       }, | ||||||
|  |       type: Types.NOCACHE, | ||||||
|  |    }, | ||||||
|  |    { | ||||||
|  |       match: (url) => { | ||||||
|  |          return url.indexOf("/share") >= 0; | ||||||
|  |       }, | ||||||
|  |       type: Types.INDEX, | ||||||
|  |    }, | ||||||
|  |    { | ||||||
|  |       match: () => { | ||||||
|  |          return true; | ||||||
|  |       }, | ||||||
|  |       type: Types.REFRESH, | ||||||
|  |    }, | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | self.addEventListener("fetch", (evt) => { | ||||||
|  |    if (evt.request.method != "GET") return; // Dont care about POST requests | ||||||
|  |    let rule = rules.find((rule) => rule.match(evt.request.url)); | ||||||
|  |    evt.respondWith( | ||||||
|  |       (async () => { | ||||||
|  |          log("Cache:", Types[rule.type]); | ||||||
|  |          switch (rule.type) { | ||||||
|  |             case Types.CACHE: | ||||||
|  |                return fromCache(evt.request); | ||||||
|  |             case Types.REFRESH: | ||||||
|  |                return refresh(evt.request).then((r) => { | ||||||
|  |                   evt.waitUntil(r.refresh.catch((_) => {})); | ||||||
|  |                   return r.result; | ||||||
|  |                }); | ||||||
|  |             case Types.NOCACHE: | ||||||
|  |                return fetch(evt.request); | ||||||
|  |             case Types.INDEX: | ||||||
|  |                return refresh(new Request("/")).then((r) => { | ||||||
|  |                   evt.waitUntil(r.refresh.catch((_) => {})); | ||||||
|  |                   return r.result; | ||||||
|  |                }); | ||||||
|  |          } | ||||||
|  |       })() | ||||||
|  |    ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | async function fromCache(request) { | ||||||
|  |    let cache = await caches.open(CACHE); | ||||||
|  |    let matching = await cache.match(request); | ||||||
|  |    if (matching) return matching; | ||||||
|  |  | ||||||
|  |    let res = await fetch(request.clone()); | ||||||
|  |    await cache.put( | ||||||
|  |       request, | ||||||
|  |       { | ||||||
|  |          match: (url) => { | ||||||
|  |             return url.indexOf("/version_hash") >= 0; | ||||||
|  |          }, | ||||||
|  |          type: Types.NOCACHE, | ||||||
|  |       }, | ||||||
|  |       res | ||||||
|  |    ); | ||||||
|  |    return await cache.match(request); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function refresh(request) { | ||||||
|  |    let cache = await caches.open(CACHE); | ||||||
|  |    let web = fetch(request.clone()).then((res) => { | ||||||
|  |       return cache.put(request, res).then(() => { | ||||||
|  |          return cache.match(request); | ||||||
|  |       }); | ||||||
|  |    }); | ||||||
|  |    let matching = await cache.match(request); | ||||||
|  |  | ||||||
|  |    return { | ||||||
|  |       result: matching ? matching : web, | ||||||
|  |       refresh: web, | ||||||
|  |    }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function precache() { | ||||||
|  |    return caches.open(CACHE).then(function (cache) { | ||||||
|  |       return cache.addAll(precacheFiles); | ||||||
|  |    }); | ||||||
|  | } | ||||||
							
								
								
									
										69
									
								
								src/theme.ts
									
									
									
									
									
								
							
							
						
						| @ -1,52 +1,61 @@ | |||||||
| const light = require("!!raw-loader!@hibas123/theme/out/light.css").default; | import { AwaitStore, Observable } from "@hibas123/utils"; | ||||||
| const dark = require("!!raw-loader!@hibas123/theme/out/dark.css").default; | import { readFileSync } from "fs"; | ||||||
|  |  | ||||||
|  | const light = readFileSync( | ||||||
|  |    "node_modules/@hibas123/theme/out/light.css", | ||||||
|  |    "utf-8" | ||||||
|  | ); | ||||||
|  | const dark = readFileSync("node_modules/@hibas123/theme/out/dark.css", "utf-8"); | ||||||
|  |  | ||||||
| export enum ThemeStates { | export enum ThemeStates { | ||||||
|     AUTO, |    AUTO, | ||||||
|     LIGHT, |    LIGHT, | ||||||
|     DARK |    DARK, | ||||||
| } | } | ||||||
|  |  | ||||||
| let themeConfig: ThemeStates = Number(localStorage.getItem("theme")); | let themeConfig: ThemeStates = Number(localStorage.getItem("theme")); | ||||||
| if (Number.isNaN(themeConfig)) | if (Number.isNaN(themeConfig)) themeConfig = ThemeStates.AUTO; | ||||||
|     themeConfig = ThemeStates.AUTO; |  | ||||||
|  |  | ||||||
| let isDark = false; | let isDark = false; | ||||||
| let mediaIsDark = false; | let mediaIsDark = false; | ||||||
|  |  | ||||||
| if (window.matchMedia) { | if (window.matchMedia) { | ||||||
|     const mediaq = matchMedia("(prefers-color-scheme: dark)"); |    const mediaq = matchMedia("(prefers-color-scheme: dark)"); | ||||||
|     mediaIsDark = mediaq.matches; |    mediaIsDark = mediaq.matches; | ||||||
|     mediaq.onchange = ev => { |    mediaq.onchange = (ev) => { | ||||||
|         mediaIsDark = ev.matches; |       mediaIsDark = ev.matches; | ||||||
|         apply(); |       apply(); | ||||||
|     } |    }; | ||||||
|     console.log(mediaq); |    console.log(mediaq); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | const DarkModeStore = new AwaitStore<boolean>(isDark); | ||||||
|  |  | ||||||
| let styleElm: HTMLStyleElement; | let styleElm: HTMLStyleElement; | ||||||
| function apply(force?: boolean) { | function apply(force?: boolean) { | ||||||
|     let shouldDark = themeConfig === ThemeStates.AUTO ? mediaIsDark : themeConfig === ThemeStates.DARK; |    let shouldDark = | ||||||
|     if (force || shouldDark !== isDark) { |       themeConfig === ThemeStates.AUTO | ||||||
|         if (styleElm) styleElm.remove(); |          ? mediaIsDark | ||||||
|         styleElm = document.createElement("style"); |          : themeConfig === ThemeStates.DARK; | ||||||
|         document.head.appendChild(styleElm); |    if (force || shouldDark !== isDark) { | ||||||
|         styleElm.innerHTML = shouldDark ? dark : light; |       if (styleElm) styleElm.remove(); | ||||||
|         isDark = shouldDark; |       styleElm = document.createElement("style"); | ||||||
|     } |       document.head.appendChild(styleElm); | ||||||
|  |       styleElm.innerHTML = shouldDark ? dark : light; | ||||||
|  |       isDark = shouldDark; | ||||||
|  |       DarkModeStore.send(isDark); | ||||||
|  |    } | ||||||
| } | } | ||||||
| apply(true); | apply(true); | ||||||
|  |  | ||||||
|  |  | ||||||
| function change(state: ThemeStates) { | function change(state: ThemeStates) { | ||||||
|     themeConfig = state; |    themeConfig = state; | ||||||
|     localStorage.setItem("theme", String(themeConfig)); |    localStorage.setItem("theme", String(themeConfig)); | ||||||
|     apply(); |    apply(); | ||||||
| } | } | ||||||
|  |  | ||||||
| export default { | export default { | ||||||
|     active: () => themeConfig, |    active: () => themeConfig, | ||||||
|     change: (state: ThemeStates) => change(state) |    change: (state: ThemeStates) => change(state), | ||||||
| } |    isDark: DarkModeStore, | ||||||
|  | }; | ||||||
|  | |||||||
							
								
								
									
										12
									
								
								src/types.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						| @ -1,13 +1,3 @@ | |||||||
| declare module "*.svg" { |  | ||||||
|     const SVG: (props: { |  | ||||||
|         width?: number | undefined |  | ||||||
|         height?: number | undefined |  | ||||||
|         onClick?: (evt: MouseEvent) => void; |  | ||||||
|         style?: string; |  | ||||||
|     }) => JSX.Element; |  | ||||||
|     export default SVG; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| interface Window { | interface Window { | ||||||
|     debug: any; |    debug: any; | ||||||
| } | } | ||||||
 Fabian Stamm
					Fabian Stamm