2019-12-13 01:16:34 +00:00
import * as React from 'react' ;
import { useState , useEffect } from 'react' ;
2020-11-07 15:59:37 +00:00
import JoplinServerApi from '@joplin/lib/JoplinServerApi' ;
import { _ , _n } from '@joplin/lib/locale' ;
2021-01-22 17:41:11 +00:00
import Note from '@joplin/lib/models/Note' ;
import Setting from '@joplin/lib/models/Setting' ;
import BaseItem from '@joplin/lib/models/BaseItem' ;
2021-01-29 18:45:11 +00:00
import SyncTargetJoplinServer from '@joplin/lib/SyncTargetJoplinServer' ;
const { themeStyle , buildStyle } = require ( '@joplin/lib/theme' ) ;
const DialogButtonRow = require ( './DialogButtonRow.min' ) ;
import { reg } from '@joplin/lib/registry' ;
2019-12-13 01:16:34 +00:00
const { clipboard } = require ( 'electron' ) ;
interface ShareNoteDialogProps {
2020-11-12 19:29:22 +00:00
themeId : number ;
noteIds : Array < string > ;
onClose : Function ;
2019-12-13 01:16:34 +00:00
}
interface SharesMap {
[ key : string ] : any ;
}
2020-11-12 19:13:28 +00:00
function styles_ ( props : ShareNoteDialogProps ) {
return buildStyle ( 'ShareNoteDialog' , props . themeId , ( theme : any ) = > {
2019-12-13 01:16:34 +00:00
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 : {
2019-12-17 12:45:57 +00:00
. . . theme . textStyle ,
2019-12-13 01:16:34 +00:00
flex : 1 ,
display : 'flex' ,
color : theme.color ,
} ,
noteRemoveButton : {
background : 'none' ,
border : 'none' ,
} ,
noteRemoveButtonIcon : {
color : theme.color ,
fontSize : '1.4em' ,
} ,
copyShareLinkButton : {
. . . theme . buttonStyle ,
marginBottom : 10 ,
} ,
} ;
} ) ;
}
2020-11-12 19:13:28 +00:00
export default function ShareNoteDialog ( props : ShareNoteDialogProps ) {
2019-12-13 01:16:34 +00:00
console . info ( 'Render ShareNoteDialog' ) ;
const [ notes , setNotes ] = useState < any [ ] > ( [ ] ) ;
const [ sharesState , setSharesState ] = useState < string > ( 'unknown' ) ;
const [ shares , setShares ] = useState < SharesMap > ( { } ) ;
const noteCount = notes . length ;
2020-09-15 14:01:07 +01:00
const theme = themeStyle ( props . themeId ) ;
2019-12-13 01:16:34 +00:00
const styles = styles_ ( props ) ;
useEffect ( ( ) = > {
async function fetchNotes() {
const result = [ ] ;
2020-03-13 23:46:14 +00:00
for ( const noteId of props . noteIds ) {
2019-12-13 01:16:34 +00:00
result . push ( await Note . load ( noteId ) ) ;
}
setNotes ( result ) ;
}
2020-11-25 14:40:25 +00:00
void fetchNotes ( ) ;
2019-12-13 01:16:34 +00:00
} , [ props . noteIds ] ) ;
2021-01-29 18:45:11 +00:00
const fileApi = async ( ) = > {
const syncTarget = reg . syncTarget ( ) as SyncTargetJoplinServer ;
return syncTarget . fileApi ( ) ;
} ;
const joplinServerApi = async ( ) : Promise < JoplinServerApi > = > {
return ( await fileApi ( ) ) . driver ( ) . api ( ) ;
2019-12-13 01:16:34 +00:00
} ;
const buttonRow_click = ( ) = > {
props . onClose ( ) ;
} ;
2021-01-29 18:45:11 +00:00
const copyLinksToClipboard = ( api : JoplinServerApi , shares : SharesMap ) = > {
2019-12-13 01:16:34 +00:00
const links = [ ] ;
2021-01-29 18:45:11 +00:00
for ( const n in shares ) links . push ( api . shareUrl ( shares [ n ] ) ) ;
2019-12-13 01:16:34 +00:00
clipboard . writeText ( links . join ( '\n' ) ) ;
} ;
const shareLinkButton_click = async ( ) = > {
let hasSynced = false ;
let tryToSync = false ;
while ( true ) {
try {
if ( tryToSync ) {
2019-12-17 12:45:57 +00:00
setSharesState ( 'synchronizing' ) ;
2020-03-13 17:42:50 +00:00
await reg . waitForSyncFinishedThenSync ( ) ;
2019-12-13 01:16:34 +00:00
tryToSync = false ;
hasSynced = true ;
}
2019-12-17 12:45:57 +00:00
setSharesState ( 'creating' ) ;
2021-01-29 18:45:11 +00:00
const api = await joplinServerApi ( ) ;
2019-12-13 01:16:34 +00:00
const newShares = Object . assign ( { } , shares ) ;
2019-12-17 12:45:57 +00:00
let sharedStatusChanged = false ;
2019-12-13 01:16:34 +00:00
for ( const note of notes ) {
2021-01-29 18:45:11 +00:00
const fullPath = ( await fileApi ( ) ) . fullPath ( BaseItem . systemPath ( note . id ) ) ;
const share = await api . shareFile ( fullPath ) ;
newShares [ note . id ] = share ;
2019-12-17 12:45:57 +00:00
const changed = await BaseItem . updateShareStatus ( note , true ) ;
if ( changed ) sharedStatusChanged = true ;
2019-12-13 01:16:34 +00:00
}
setShares ( newShares ) ;
2019-12-17 12:45:57 +00:00
if ( sharedStatusChanged ) {
setSharesState ( 'synchronizing' ) ;
2020-03-13 17:42:50 +00:00
await reg . waitForSyncFinishedThenSync ( ) ;
2019-12-17 12:45:57 +00:00
setSharesState ( 'creating' ) ;
}
2021-01-29 18:45:11 +00:00
copyLinksToClipboard ( api , newShares ) ;
2019-12-13 01:16:34 +00:00
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 ;
}
} ;
2020-11-12 19:13:28 +00:00
const removeNoteButton_click = ( event : any ) = > {
2019-12-13 01:16:34 +00:00
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 ) ;
} ;
2020-11-12 19:13:28 +00:00
const renderNote = ( note : any ) = > {
2019-12-13 01:16:34 +00:00
const removeButton = notes . length <= 1 ? null : (
< button onClick = { ( ) = > removeNoteButton_click ( { noteId : note.id } ) } style = { styles . noteRemoveButton } >
< i style = { styles . noteRemoveButtonIcon } className = { 'fa fa-times' } > < / i >
< / button >
) ;
return (
< div key = { note . id } style = { styles . note } >
< span style = { styles . noteTitle } > { note . title } < / span > { removeButton }
< / div >
) ;
} ;
2020-11-12 19:13:28 +00:00
const renderNoteList = ( notes : any ) = > {
2019-12-13 01:16:34 +00:00
const noteComps = [ ] ;
2020-03-13 23:46:14 +00:00
for ( const noteId of Object . keys ( notes ) ) {
2019-12-13 01:16:34 +00:00
noteComps . push ( renderNote ( notes [ noteId ] ) ) ;
}
return < div style = { styles . noteList } > { noteComps } < / div > ;
} ;
2020-11-12 19:13:28 +00:00
const statusMessage = ( sharesState : string ) : string = > {
2019-12-17 12:45:57 +00:00
if ( sharesState === 'synchronizing' ) return _ ( 'Synchronising...' ) ;
2019-12-13 01:16:34 +00:00
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 '' ;
} ;
2021-01-29 18:45:11 +00:00
function renderEncryptionWarningMessage() {
if ( ! Setting . value ( 'encryption.enabled' ) ) return null ;
return < div style = { theme . textStyle } > { _ ( 'Note: When a note is shared, it will no longer be encrypted on the server.' ) } < hr / > < / div > ;
}
function renderBetaWarningMessage() {
return < div style = { theme . textStyle } > { 'Sharing notes via Joplin Server is a Beta feature and the API might change later on. What it means is that if you share a note, the link might become invalid after an upgrade, and you will have to share it again.' } < / div > ;
}
2019-12-17 12:45:57 +00:00
2019-12-13 01:16:34 +00:00
const rootStyle = Object . assign ( { } , theme . dialogBox ) ;
rootStyle . width = '50%' ;
return (
< div style = { theme . dialogModalLayer } >
< div style = { rootStyle } >
< div style = { theme . dialogTitle } > { _ ( 'Share Notes' ) } < / div >
{ renderNoteList ( notes ) }
2019-12-17 12:45:57 +00:00
< button disabled = { [ 'creating' , 'synchronizing' ] . indexOf ( sharesState ) >= 0 } style = { styles . copyShareLinkButton } onClick = { shareLinkButton_click } > { _n ( 'Copy Shareable Link' , 'Copy Shareable Links' , noteCount ) } < / button >
2019-12-13 01:16:34 +00:00
< div style = { theme . textStyle } > { statusMessage ( sharesState ) } < / div >
2021-01-29 18:45:11 +00:00
{ renderEncryptionWarningMessage ( ) }
{ renderBetaWarningMessage ( ) }
2020-09-15 14:01:07 +01:00
< DialogButtonRow themeId = { props . themeId } onClick = { buttonRow_click } okButtonShow = { false } cancelButtonLabel = { _ ( 'Close' ) } / >
2019-12-13 01:16:34 +00:00
< / div >
< / div >
) ;
}