import * as React from 'react'; import { useState, useEffect } from 'react'; import JoplinServerApi from '@joplin/lib/JoplinServerApi'; import { _, _n } from '@joplin/lib/locale'; const { themeStyle, buildStyle } = require('@joplin/lib/theme'); const DialogButtonRow = require('./DialogButtonRow.min'); const Note = require('@joplin/lib/models/Note'); const Setting = require('@joplin/lib/models/Setting').default; const BaseItem = require('@joplin/lib/models/BaseItem'); const { reg } = require('@joplin/lib/registry.js'); const { clipboard } = require('electron'); interface ShareNoteDialogProps { themeId: number, noteIds: Array, onClose: Function, } interface SharesMap { [key: string]: any; } function styles_(props:ShareNoteDialogProps) { return buildStyle('ShareNoteDialog', props.themeId, (theme:any) => { return { noteList: { marginBottom: 10, }, note: { flex: 1, flexDirection: 'row', display: 'flex', alignItems: 'center', border: '1px solid', borderColor: theme.dividerColor, padding: '0.5em', marginBottom: 5, }, noteTitle: { ...theme.textStyle, flex: 1, display: 'flex', color: theme.color, }, noteRemoveButton: { background: 'none', border: 'none', }, noteRemoveButtonIcon: { color: theme.color, fontSize: '1.4em', }, copyShareLinkButton: { ...theme.buttonStyle, marginBottom: 10, }, }; }); } export default function ShareNoteDialog(props:ShareNoteDialogProps) { console.info('Render ShareNoteDialog'); const [notes, setNotes] = useState([]); const [sharesState, setSharesState] = useState('unknown'); const [shares, setShares] = useState({}); const noteCount = notes.length; const theme = themeStyle(props.themeId); const styles = styles_(props); useEffect(() => { async function fetchNotes() { const result = []; for (const noteId of props.noteIds) { result.push(await Note.load(noteId)); } setNotes(result); } fetchNotes(); }, [props.noteIds]); const appApi = async () => { return reg.syncTargetNextcloud().appApi(); }; const buttonRow_click = () => { props.onClose(); }; const copyLinksToClipboard = (shares:SharesMap) => { const links = []; for (const n in shares) links.push(shares[n]._url); clipboard.writeText(links.join('\n')); }; const shareLinkButton_click = async () => { let hasSynced = false; let tryToSync = false; while (true) { try { if (tryToSync) { setSharesState('synchronizing'); await reg.waitForSyncFinishedThenSync(); tryToSync = false; hasSynced = true; } setSharesState('creating'); const api = await appApi(); const syncTargetId = api.syncTargetId(Setting.toPlainObject()); const newShares = Object.assign({}, shares); let sharedStatusChanged = false; for (const note of notes) { const result = await api.exec('POST', 'shares', { syncTargetId: syncTargetId, noteId: note.id, }); newShares[note.id] = result; const changed = await BaseItem.updateShareStatus(note, true); if (changed) sharedStatusChanged = true; } setShares(newShares); if (sharedStatusChanged) { setSharesState('synchronizing'); await reg.waitForSyncFinishedThenSync(); setSharesState('creating'); } copyLinksToClipboard(newShares); setSharesState('created'); } catch (error) { if (error.code === 404 && !hasSynced) { reg.logger().info('ShareNoteDialog: Note does not exist on server - trying to sync it.', error); tryToSync = true; continue; } reg.logger().error('ShareNoteDialog: Cannot share note:', error); setSharesState('idle'); alert(JoplinServerApi.connectionErrorMessage(error)); } break; } }; const removeNoteButton_click = (event:any) => { const newNotes = []; for (let i = 0; i < notes.length; i++) { const n = notes[i]; if (n.id === event.noteId) continue; newNotes.push(n); } setNotes(newNotes); }; const renderNote = (note:any) => { const removeButton = notes.length <= 1 ? null : ( ); return (
{note.title}{removeButton}
); }; const renderNoteList = (notes:any) => { const noteComps = []; for (const noteId of Object.keys(notes)) { noteComps.push(renderNote(notes[noteId])); } return
{noteComps}
; }; const statusMessage = (sharesState:string):string => { if (sharesState === 'synchronizing') return _('Synchronising...'); if (sharesState === 'creating') return _n('Generating link...', 'Generating links...', noteCount); if (sharesState === 'created') return _n('Link has been copied to clipboard!', 'Links have been copied to clipboard!', noteCount); return ''; }; const encryptionWarningMessage = !Setting.value('encryption.enabled') ? null :
{_('Note: When a note is shared, it will no longer be encrypted on the server.')}
; const rootStyle = Object.assign({}, theme.dialogBox); rootStyle.width = '50%'; return (
{_('Share Notes')}
{renderNoteList(notes)}
{statusMessage(sharesState)}
{encryptionWarningMessage}
); }