2019-12-13 03:16:34 +02:00
|
|
|
import * as React from 'react';
|
|
|
|
import { useState, useEffect } from 'react';
|
2020-11-07 17:59:37 +02:00
|
|
|
import JoplinServerApi from '@joplin/lib/JoplinServerApi';
|
|
|
|
import { _, _n } from '@joplin/lib/locale';
|
2021-01-22 19:41:11 +02:00
|
|
|
import Note from '@joplin/lib/models/Note';
|
|
|
|
import Setting from '@joplin/lib/models/Setting';
|
2021-05-13 18:57:37 +02:00
|
|
|
import DialogButtonRow from './DialogButtonRow';
|
|
|
|
import { themeStyle, buildStyle } from '@joplin/lib/theme';
|
2021-01-29 20:45:11 +02:00
|
|
|
import { reg } from '@joplin/lib/registry';
|
2021-05-13 18:57:37 +02:00
|
|
|
import Dialog from './Dialog';
|
|
|
|
import DialogTitle from './DialogTitle';
|
|
|
|
import ShareService from '@joplin/lib/services/share/ShareService';
|
2021-05-16 17:28:49 +02:00
|
|
|
import { StateShare } from '@joplin/lib/services/share/reducer';
|
|
|
|
import { NoteEntity } from '@joplin/lib/services/database/types';
|
|
|
|
import Button from './Button/Button';
|
|
|
|
import { connect } from 'react-redux';
|
|
|
|
import { AppState } from '../app';
|
2019-12-13 03:16:34 +02:00
|
|
|
const { clipboard } = require('electron');
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
interface Props {
|
2020-11-12 21:29:22 +02:00
|
|
|
themeId: number;
|
|
|
|
noteIds: Array<string>;
|
|
|
|
onClose: Function;
|
2021-05-16 17:28:49 +02:00
|
|
|
shares: StateShare[];
|
2019-12-13 03:16:34 +02:00
|
|
|
}
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
function styles_(props: Props) {
|
2020-11-12 21:13:28 +02:00
|
|
|
return buildStyle('ShareNoteDialog', props.themeId, (theme: any) => {
|
2019-12-13 03:16:34 +02: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 14:45:57 +02:00
|
|
|
...theme.textStyle,
|
2019-12-13 03:16:34 +02: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,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
export function ShareNoteDialog(props: Props) {
|
2019-12-13 03:16:34 +02:00
|
|
|
console.info('Render ShareNoteDialog');
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
const [notes, setNotes] = useState<NoteEntity[]>([]);
|
2019-12-13 03:16:34 +02:00
|
|
|
const [sharesState, setSharesState] = useState<string>('unknown');
|
2021-05-16 17:28:49 +02:00
|
|
|
// const [shares, setShares] = useState<SharesMap>({});
|
2019-12-13 03:16:34 +02:00
|
|
|
|
|
|
|
const noteCount = notes.length;
|
2020-09-15 15:01:07 +02:00
|
|
|
const theme = themeStyle(props.themeId);
|
2019-12-13 03:16:34 +02:00
|
|
|
const styles = styles_(props);
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
useEffect(() => {
|
|
|
|
void ShareService.instance().refreshShares();
|
|
|
|
}, []);
|
|
|
|
|
2019-12-13 03:16:34 +02:00
|
|
|
useEffect(() => {
|
|
|
|
async function fetchNotes() {
|
|
|
|
const result = [];
|
2020-03-14 01:46:14 +02:00
|
|
|
for (const noteId of props.noteIds) {
|
2019-12-13 03:16:34 +02:00
|
|
|
result.push(await Note.load(noteId));
|
|
|
|
}
|
|
|
|
setNotes(result);
|
|
|
|
}
|
|
|
|
|
2020-11-25 16:40:25 +02:00
|
|
|
void fetchNotes();
|
2019-12-13 03:16:34 +02:00
|
|
|
}, [props.noteIds]);
|
|
|
|
|
|
|
|
const buttonRow_click = () => {
|
|
|
|
props.onClose();
|
|
|
|
};
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
const copyLinksToClipboard = (shares: StateShare[]) => {
|
2019-12-13 03:16:34 +02:00
|
|
|
const links = [];
|
2021-05-16 17:28:49 +02:00
|
|
|
for (const share of shares) links.push(ShareService.instance().shareUrl(share));
|
2019-12-13 03:16:34 +02:00
|
|
|
clipboard.writeText(links.join('\n'));
|
|
|
|
};
|
|
|
|
|
|
|
|
const shareLinkButton_click = async () => {
|
2021-05-13 18:57:37 +02:00
|
|
|
const service = ShareService.instance();
|
|
|
|
|
2019-12-13 03:16:34 +02:00
|
|
|
let hasSynced = false;
|
|
|
|
let tryToSync = false;
|
|
|
|
while (true) {
|
|
|
|
try {
|
|
|
|
if (tryToSync) {
|
2019-12-17 14:45:57 +02:00
|
|
|
setSharesState('synchronizing');
|
2020-03-13 19:42:50 +02:00
|
|
|
await reg.waitForSyncFinishedThenSync();
|
2019-12-13 03:16:34 +02:00
|
|
|
tryToSync = false;
|
|
|
|
hasSynced = true;
|
|
|
|
}
|
|
|
|
|
2019-12-17 14:45:57 +02:00
|
|
|
setSharesState('creating');
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
const newShares: StateShare[] = [];
|
2019-12-13 03:16:34 +02:00
|
|
|
|
|
|
|
for (const note of notes) {
|
2021-05-13 18:57:37 +02:00
|
|
|
const share = await service.shareNote(note.id);
|
2021-05-16 17:28:49 +02:00
|
|
|
newShares.push(share);
|
2019-12-13 03:16:34 +02:00
|
|
|
}
|
|
|
|
|
2021-05-13 18:57:37 +02:00
|
|
|
setSharesState('synchronizing');
|
|
|
|
await reg.waitForSyncFinishedThenSync();
|
|
|
|
setSharesState('creating');
|
2019-12-17 14:45:57 +02:00
|
|
|
|
2021-05-13 18:57:37 +02:00
|
|
|
copyLinksToClipboard(newShares);
|
2019-12-13 03:16:34 +02:00
|
|
|
|
|
|
|
setSharesState('created');
|
2021-05-16 17:28:49 +02:00
|
|
|
|
|
|
|
await ShareService.instance().refreshShares();
|
2019-12-13 03:16:34 +02:00
|
|
|
} 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;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
// 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 unshareNoteButton_click = async (event: any) => {
|
|
|
|
await ShareService.instance().unshareNote(event.noteId);
|
|
|
|
await ShareService.instance().refreshShares();
|
2019-12-13 03:16:34 +02:00
|
|
|
};
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
const renderNote = (note: NoteEntity) => {
|
|
|
|
const unshareButton = !props.shares.find(s => s.note_id === note.id) ? null : (
|
|
|
|
<Button tooltip={_('Unshare note')} iconName="fas fa-share-alt" onClick={() => unshareNoteButton_click({ noteId: note.id })}/>
|
2019-12-13 03:16:34 +02:00
|
|
|
);
|
|
|
|
|
2021-05-16 17:28:49 +02:00
|
|
|
// const removeButton = notes.length <= 1 ? null : (
|
|
|
|
// <Button iconName="fa fa-times" onClick={() => removeNoteButton_click({ noteId: note.id })}/>
|
|
|
|
// );
|
|
|
|
|
|
|
|
// const unshareButton = !shares[note.id] ? null : (
|
|
|
|
// <button onClick={() => unshareNoteButton_click({ noteId: note.id })} style={styles.noteRemoveButton}>
|
|
|
|
// <i style={styles.noteRemoveButtonIcon} className={'fas fa-share-alt'}></i>
|
|
|
|
// </button>
|
|
|
|
// );
|
|
|
|
|
|
|
|
// 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>
|
|
|
|
// );
|
|
|
|
|
2019-12-13 03:16:34 +02:00
|
|
|
return (
|
|
|
|
<div key={note.id} style={styles.note}>
|
2021-05-16 17:28:49 +02:00
|
|
|
<span style={styles.noteTitle}>{note.title}</span>{unshareButton}
|
2019-12-13 03:16:34 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2020-11-12 21:13:28 +02:00
|
|
|
const renderNoteList = (notes: any) => {
|
2019-12-13 03:16:34 +02:00
|
|
|
const noteComps = [];
|
2021-05-16 17:28:49 +02:00
|
|
|
for (const note of notes) {
|
|
|
|
noteComps.push(renderNote(note));
|
2019-12-13 03:16:34 +02:00
|
|
|
}
|
|
|
|
return <div style={styles.noteList}>{noteComps}</div>;
|
|
|
|
};
|
|
|
|
|
2020-11-12 21:13:28 +02:00
|
|
|
const statusMessage = (sharesState: string): string => {
|
2019-12-17 14:45:57 +02:00
|
|
|
if (sharesState === 'synchronizing') return _('Synchronising...');
|
2019-12-13 03:16:34 +02: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 20:45:11 +02: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>;
|
|
|
|
}
|
|
|
|
|
2021-05-13 18:57:37 +02:00
|
|
|
function renderContent() {
|
|
|
|
return (
|
|
|
|
<div>
|
|
|
|
<DialogTitle title={_('Share Notes')}/>
|
2019-12-13 03:16:34 +02:00
|
|
|
{renderNoteList(notes)}
|
2019-12-17 14:45:57 +02: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 03:16:34 +02:00
|
|
|
<div style={theme.textStyle}>{statusMessage(sharesState)}</div>
|
2021-01-29 20:45:11 +02:00
|
|
|
{renderEncryptionWarningMessage()}
|
2021-05-16 17:28:49 +02:00
|
|
|
<DialogButtonRow
|
|
|
|
themeId={props.themeId}
|
|
|
|
onClick={buttonRow_click}
|
|
|
|
okButtonShow={false}
|
|
|
|
cancelButtonLabel={_('Close')}
|
|
|
|
/>
|
2019-12-13 03:16:34 +02:00
|
|
|
</div>
|
2021-05-13 18:57:37 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Dialog renderContent={renderContent}/>
|
2019-12-13 03:16:34 +02:00
|
|
|
);
|
|
|
|
}
|
2021-05-16 17:28:49 +02:00
|
|
|
|
|
|
|
const mapStateToProps = (state: AppState) => {
|
|
|
|
return {
|
|
|
|
shares: state.shareService.shares.filter(s => !!s.note_id),
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export default connect(mapStateToProps)(ShareNoteDialog as any);
|