mirror of
https://github.com/laurent22/joplin.git
synced 2025-04-04 21:35:03 +02:00
Desktop: Allow unsharing a note
This commit is contained in:
parent
6f2f24171d
commit
f7d164be6e
@ -18,6 +18,6 @@ export const runtime = (comp: any): CommandRuntime => {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
enabledCondition: 'joplinServerConnected && oneNoteSelected',
|
enabledCondition: 'joplinServerConnected && someNotesSelected',
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -110,6 +110,7 @@ function NoteListItem(props: NoteListItemProps, ref: any) {
|
|||||||
|
|
||||||
let listItemTitleStyle = Object.assign({}, props.style.listItemTitle);
|
let listItemTitleStyle = Object.assign({}, props.style.listItemTitle);
|
||||||
listItemTitleStyle.paddingLeft = !item.is_todo ? hPadding : 4;
|
listItemTitleStyle.paddingLeft = !item.is_todo ? hPadding : 4;
|
||||||
|
if (item.is_shared) listItemTitleStyle.color = theme.colorWarn;
|
||||||
if (item.is_todo && !!item.todo_completed) listItemTitleStyle = Object.assign(listItemTitleStyle, props.style.listItemTitleCompleted);
|
if (item.is_todo && !!item.todo_completed) listItemTitleStyle = Object.assign(listItemTitleStyle, props.style.listItemTitleCompleted);
|
||||||
|
|
||||||
const displayTitle = Note.displayTitle(item);
|
const displayTitle = Note.displayTitle(item);
|
||||||
|
@ -10,19 +10,21 @@ import { reg } from '@joplin/lib/registry';
|
|||||||
import Dialog from './Dialog';
|
import Dialog from './Dialog';
|
||||||
import DialogTitle from './DialogTitle';
|
import DialogTitle from './DialogTitle';
|
||||||
import ShareService from '@joplin/lib/services/share/ShareService';
|
import ShareService from '@joplin/lib/services/share/ShareService';
|
||||||
|
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';
|
||||||
const { clipboard } = require('electron');
|
const { clipboard } = require('electron');
|
||||||
|
|
||||||
interface ShareNoteDialogProps {
|
interface Props {
|
||||||
themeId: number;
|
themeId: number;
|
||||||
noteIds: Array<string>;
|
noteIds: Array<string>;
|
||||||
onClose: Function;
|
onClose: Function;
|
||||||
|
shares: StateShare[];
|
||||||
}
|
}
|
||||||
|
|
||||||
interface SharesMap {
|
function styles_(props: Props) {
|
||||||
[key: string]: any;
|
|
||||||
}
|
|
||||||
|
|
||||||
function styles_(props: ShareNoteDialogProps) {
|
|
||||||
return buildStyle('ShareNoteDialog', props.themeId, (theme: any) => {
|
return buildStyle('ShareNoteDialog', props.themeId, (theme: any) => {
|
||||||
return {
|
return {
|
||||||
noteList: {
|
noteList: {
|
||||||
@ -60,17 +62,21 @@ function styles_(props: ShareNoteDialogProps) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function ShareNoteDialog(props: ShareNoteDialogProps) {
|
export function ShareNoteDialog(props: Props) {
|
||||||
console.info('Render ShareNoteDialog');
|
console.info('Render ShareNoteDialog');
|
||||||
|
|
||||||
const [notes, setNotes] = useState<any[]>([]);
|
const [notes, setNotes] = useState<NoteEntity[]>([]);
|
||||||
const [sharesState, setSharesState] = useState<string>('unknown');
|
const [sharesState, setSharesState] = useState<string>('unknown');
|
||||||
const [shares, setShares] = useState<SharesMap>({});
|
// const [shares, setShares] = useState<SharesMap>({});
|
||||||
|
|
||||||
const noteCount = notes.length;
|
const noteCount = notes.length;
|
||||||
const theme = themeStyle(props.themeId);
|
const theme = themeStyle(props.themeId);
|
||||||
const styles = styles_(props);
|
const styles = styles_(props);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
void ShareService.instance().refreshShares();
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
async function fetchNotes() {
|
async function fetchNotes() {
|
||||||
const result = [];
|
const result = [];
|
||||||
@ -87,9 +93,9 @@ export default function ShareNoteDialog(props: ShareNoteDialogProps) {
|
|||||||
props.onClose();
|
props.onClose();
|
||||||
};
|
};
|
||||||
|
|
||||||
const copyLinksToClipboard = (shares: SharesMap) => {
|
const copyLinksToClipboard = (shares: StateShare[]) => {
|
||||||
const links = [];
|
const links = [];
|
||||||
for (const n in shares) links.push(ShareService.instance().shareUrl(shares[n]));
|
for (const share of shares) links.push(ShareService.instance().shareUrl(share));
|
||||||
clipboard.writeText(links.join('\n'));
|
clipboard.writeText(links.join('\n'));
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -109,15 +115,13 @@ export default function ShareNoteDialog(props: ShareNoteDialogProps) {
|
|||||||
|
|
||||||
setSharesState('creating');
|
setSharesState('creating');
|
||||||
|
|
||||||
const newShares = Object.assign({}, shares);
|
const newShares: StateShare[] = [];
|
||||||
|
|
||||||
for (const note of notes) {
|
for (const note of notes) {
|
||||||
const share = await service.shareNote(note.id);
|
const share = await service.shareNote(note.id);
|
||||||
newShares[note.id] = share;
|
newShares.push(share);
|
||||||
}
|
}
|
||||||
|
|
||||||
setShares(newShares);
|
|
||||||
|
|
||||||
setSharesState('synchronizing');
|
setSharesState('synchronizing');
|
||||||
await reg.waitForSyncFinishedThenSync();
|
await reg.waitForSyncFinishedThenSync();
|
||||||
setSharesState('creating');
|
setSharesState('creating');
|
||||||
@ -125,6 +129,8 @@ export default function ShareNoteDialog(props: ShareNoteDialogProps) {
|
|||||||
copyLinksToClipboard(newShares);
|
copyLinksToClipboard(newShares);
|
||||||
|
|
||||||
setSharesState('created');
|
setSharesState('created');
|
||||||
|
|
||||||
|
await ShareService.instance().refreshShares();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error.code === 404 && !hasSynced) {
|
if (error.code === 404 && !hasSynced) {
|
||||||
reg.logger().info('ShareNoteDialog: Note does not exist on server - trying to sync it.', error);
|
reg.logger().info('ShareNoteDialog: Note does not exist on server - trying to sync it.', error);
|
||||||
@ -142,34 +148,53 @@ export default function ShareNoteDialog(props: ShareNoteDialogProps) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const removeNoteButton_click = (event: any) => {
|
// const removeNoteButton_click = (event: any) => {
|
||||||
const newNotes = [];
|
// const newNotes = [];
|
||||||
for (let i = 0; i < notes.length; i++) {
|
// for (let i = 0; i < notes.length; i++) {
|
||||||
const n = notes[i];
|
// const n = notes[i];
|
||||||
if (n.id === event.noteId) continue;
|
// if (n.id === event.noteId) continue;
|
||||||
newNotes.push(n);
|
// newNotes.push(n);
|
||||||
}
|
// }
|
||||||
setNotes(newNotes);
|
// setNotes(newNotes);
|
||||||
|
// };
|
||||||
|
|
||||||
|
const unshareNoteButton_click = async (event: any) => {
|
||||||
|
await ShareService.instance().unshareNote(event.noteId);
|
||||||
|
await ShareService.instance().refreshShares();
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderNote = (note: any) => {
|
const renderNote = (note: NoteEntity) => {
|
||||||
const removeButton = notes.length <= 1 ? null : (
|
const unshareButton = !props.shares.find(s => s.note_id === note.id) ? null : (
|
||||||
<button onClick={() => removeNoteButton_click({ noteId: note.id })} style={styles.noteRemoveButton}>
|
<Button tooltip={_('Unshare note')} iconName="fas fa-share-alt" onClick={() => unshareNoteButton_click({ noteId: note.id })}/>
|
||||||
<i style={styles.noteRemoveButtonIcon} className={'fa fa-times'}></i>
|
|
||||||
</button>
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// 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>
|
||||||
|
// );
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div key={note.id} style={styles.note}>
|
<div key={note.id} style={styles.note}>
|
||||||
<span style={styles.noteTitle}>{note.title}</span>{removeButton}
|
<span style={styles.noteTitle}>{note.title}</span>{unshareButton}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderNoteList = (notes: any) => {
|
const renderNoteList = (notes: any) => {
|
||||||
const noteComps = [];
|
const noteComps = [];
|
||||||
for (const noteId of Object.keys(notes)) {
|
for (const note of notes) {
|
||||||
noteComps.push(renderNote(notes[noteId]));
|
noteComps.push(renderNote(note));
|
||||||
}
|
}
|
||||||
return <div style={styles.noteList}>{noteComps}</div>;
|
return <div style={styles.noteList}>{noteComps}</div>;
|
||||||
};
|
};
|
||||||
@ -194,7 +219,12 @@ export default function ShareNoteDialog(props: ShareNoteDialogProps) {
|
|||||||
<button disabled={['creating', 'synchronizing'].indexOf(sharesState) >= 0} style={styles.copyShareLinkButton} onClick={shareLinkButton_click}>{_n('Copy Shareable Link', 'Copy Shareable Links', noteCount)}</button>
|
<button disabled={['creating', 'synchronizing'].indexOf(sharesState) >= 0} style={styles.copyShareLinkButton} onClick={shareLinkButton_click}>{_n('Copy Shareable Link', 'Copy Shareable Links', noteCount)}</button>
|
||||||
<div style={theme.textStyle}>{statusMessage(sharesState)}</div>
|
<div style={theme.textStyle}>{statusMessage(sharesState)}</div>
|
||||||
{renderEncryptionWarningMessage()}
|
{renderEncryptionWarningMessage()}
|
||||||
<DialogButtonRow themeId={props.themeId} onClick={buttonRow_click} okButtonShow={false} cancelButtonLabel={_('Close')}/>
|
<DialogButtonRow
|
||||||
|
themeId={props.themeId}
|
||||||
|
onClick={buttonRow_click}
|
||||||
|
okButtonShow={false}
|
||||||
|
cancelButtonLabel={_('Close')}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -203,3 +233,11 @@ export default function ShareNoteDialog(props: ShareNoteDialogProps) {
|
|||||||
<Dialog renderContent={renderContent}/>
|
<Dialog renderContent={renderContent}/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const mapStateToProps = (state: AppState) => {
|
||||||
|
return {
|
||||||
|
shares: state.shareService.shares.filter(s => !!s.note_id),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default connect(mapStateToProps)(ShareNoteDialog as any);
|
||||||
|
@ -69,7 +69,6 @@ function listItemTextColor(props: any) {
|
|||||||
|
|
||||||
export const StyledListItemAnchor = styled.a`
|
export const StyledListItemAnchor = styled.a`
|
||||||
font-size: ${(props: any) => Math.round(props.theme.fontSize * 1.0833333)}px;
|
font-size: ${(props: any) => Math.round(props.theme.fontSize * 1.0833333)}px;
|
||||||
// font-weight: 500;
|
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
color: ${(props: any) => listItemTextColor(props)};
|
color: ${(props: any) => listItemTextColor(props)};
|
||||||
cursor: default;
|
cursor: default;
|
||||||
|
@ -35,6 +35,7 @@ if [ "$1" == "1" ]; then
|
|||||||
echo 'mkbook "other"' >> "$CMD_FILE"
|
echo 'mkbook "other"' >> "$CMD_FILE"
|
||||||
echo 'use "shared"' >> "$CMD_FILE"
|
echo 'use "shared"' >> "$CMD_FILE"
|
||||||
echo 'mknote "note 1"' >> "$CMD_FILE"
|
echo 'mknote "note 1"' >> "$CMD_FILE"
|
||||||
|
echo 'mknote "note 2"' >> "$CMD_FILE"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
cd "$ROOT_DIR/packages/app-cli"
|
cd "$ROOT_DIR/packages/app-cli"
|
||||||
|
@ -347,7 +347,8 @@ export default class Folder extends BaseItem {
|
|||||||
public static async updateResourceShareIds() {
|
public static async updateResourceShareIds() {
|
||||||
// Find all resources where share_id is different from parent note
|
// Find all resources where share_id is different from parent note
|
||||||
// share_id. Then update share_id on all these resources. Essentially it
|
// share_id. Then update share_id on all these resources. Essentially it
|
||||||
// makes it match the resource share_id to the note share_id.
|
// makes it match the resource share_id to the note share_id. At the
|
||||||
|
// same time we also process the is_shared property.
|
||||||
const rows = await this.db().selectAll(`
|
const rows = await this.db().selectAll(`
|
||||||
SELECT r.id, n.share_id, n.is_shared
|
SELECT r.id, n.share_id, n.is_shared
|
||||||
FROM note_resources nr
|
FROM note_resources nr
|
||||||
|
@ -307,7 +307,7 @@ export default class Note extends BaseItem {
|
|||||||
includeTimestamps: true,
|
includeTimestamps: true,
|
||||||
}, options);
|
}, options);
|
||||||
|
|
||||||
const output = ['id', 'title', 'is_todo', 'todo_completed', 'todo_due', 'parent_id', 'encryption_applied', 'order', 'markup_language', 'is_conflict'];
|
const output = ['id', 'title', 'is_todo', 'todo_completed', 'todo_due', 'parent_id', 'encryption_applied', 'order', 'markup_language', 'is_conflict', 'is_shared'];
|
||||||
|
|
||||||
if (options.includeTimestamps) {
|
if (options.includeTimestamps) {
|
||||||
output.push('updated_time');
|
output.push('updated_time');
|
||||||
|
@ -104,7 +104,7 @@ export default class ShareService {
|
|||||||
await Folder.updateAllShareIds();
|
await Folder.updateAllShareIds();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async shareNote(noteId: string) {
|
public async shareNote(noteId: string): Promise<StateShare> {
|
||||||
const note = await Note.load(noteId);
|
const note = await Note.load(noteId);
|
||||||
if (!note) throw new Error(`No such note: ${noteId}`);
|
if (!note) throw new Error(`No such note: ${noteId}`);
|
||||||
|
|
||||||
@ -115,6 +115,24 @@ export default class ShareService {
|
|||||||
return share;
|
return share;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async unshareNote(noteId: string) {
|
||||||
|
const note = await Note.load(noteId);
|
||||||
|
if (!note) throw new Error(`No such note: ${noteId}`);
|
||||||
|
|
||||||
|
const shares = await this.refreshShares();
|
||||||
|
const noteShares = shares.filter(s => s.note_id === noteId);
|
||||||
|
|
||||||
|
const promises: Promise<void>[] = [];
|
||||||
|
|
||||||
|
for (const share of noteShares) {
|
||||||
|
promises.push(this.deleteShare(share.id));
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all(promises);
|
||||||
|
|
||||||
|
await Note.save({ id: note.id, is_shared: 0 });
|
||||||
|
}
|
||||||
|
|
||||||
public shareUrl(share: StateShare): string {
|
public shareUrl(share: StateShare): string {
|
||||||
return `${this.api().baseUrl()}/shares/${share.id}`;
|
return `${this.api().baseUrl()}/shares/${share.id}`;
|
||||||
}
|
}
|
||||||
@ -170,13 +188,15 @@ export default class ShareService {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public async refreshShares() {
|
public async refreshShares(): Promise<StateShare[]> {
|
||||||
const result = await this.loadShares();
|
const result = await this.loadShares();
|
||||||
|
|
||||||
this.store.dispatch({
|
this.store.dispatch({
|
||||||
type: 'SHARE_SET',
|
type: 'SHARE_SET',
|
||||||
shares: result.items,
|
shares: result.items,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
return result.items;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async refreshShareUsers(shareId: string) {
|
public async refreshShareUsers(shareId: string) {
|
||||||
|
@ -11,7 +11,7 @@ const theme: Theme = {
|
|||||||
oddBackgroundColor: '#eeeeee',
|
oddBackgroundColor: '#eeeeee',
|
||||||
color: '#32373F', // For regular text
|
color: '#32373F', // For regular text
|
||||||
colorError: 'red',
|
colorError: 'red',
|
||||||
colorWarn: '#9A5B00',
|
colorWarn: 'rgb(228 86 0)',
|
||||||
colorFaded: '#7C8B9E', // For less important text
|
colorFaded: '#7C8B9E', // For less important text
|
||||||
colorBright: '#000000', // For important text
|
colorBright: '#000000', // For important text
|
||||||
dividerColor: '#dddddd',
|
dividerColor: '#dddddd',
|
||||||
|
Loading…
x
Reference in New Issue
Block a user