mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-05 12:50:29 +02:00
177 lines
5.8 KiB
JavaScript
177 lines
5.8 KiB
JavaScript
const React = require('react');
|
|
const { connect } = require('react-redux');
|
|
const { themeStyle } = require('../theme.js');
|
|
const { _ } = require('lib/locale.js');
|
|
const NoteTextViewer = require('./NoteTextViewer.min');
|
|
const HelpButton = require('./HelpButton.min');
|
|
const BaseModel = require('lib/BaseModel');
|
|
const Revision = require('lib/models/Revision');
|
|
const Setting = require('lib/models/Setting');
|
|
const RevisionService = require('lib/services/RevisionService');
|
|
const shared = require('lib/components/shared/note-screen-shared.js');
|
|
const MdToHtml = require('lib/MdToHtml');
|
|
const { time } = require('lib/time-utils.js');
|
|
const ReactTooltip = require('react-tooltip');
|
|
const { substrWithEllipsis } = require('lib/string-utils');
|
|
|
|
class NoteRevisionViewerComponent extends React.PureComponent {
|
|
|
|
constructor() {
|
|
super();
|
|
|
|
this.state = {
|
|
revisions: [],
|
|
currentRevId: '',
|
|
note: null,
|
|
restoring: false,
|
|
};
|
|
|
|
this.viewerRef_ = React.createRef();
|
|
|
|
this.viewer_domReady = this.viewer_domReady.bind(this);
|
|
this.revisionList_onChange = this.revisionList_onChange.bind(this);
|
|
this.importButton_onClick = this.importButton_onClick.bind(this);
|
|
this.backButton_click = this.backButton_click.bind(this);
|
|
}
|
|
|
|
style() {
|
|
const theme = themeStyle(this.props.theme);
|
|
|
|
let style = {
|
|
root: {
|
|
backgroundColor: theme.backgroundColor,
|
|
display: 'flex',
|
|
flex: 1,
|
|
flexDirection: 'column',
|
|
},
|
|
titleInput: Object.assign({}, theme.inputStyle, { flex: 1 }),
|
|
revisionList: Object.assign({}, theme.dropdownList, { marginLeft: 10, flex: 0.5 }),
|
|
};
|
|
|
|
return style;
|
|
}
|
|
|
|
async viewer_domReady() {
|
|
// this.viewerRef_.current.wrappedInstance.openDevTools();
|
|
|
|
const revisions = await Revision.allByType(BaseModel.TYPE_NOTE, this.props.noteId);
|
|
|
|
this.setState({
|
|
revisions: revisions,
|
|
currentRevId: revisions.length ? revisions[revisions.length - 1].id : '',
|
|
}, () => {
|
|
this.reloadNote();
|
|
});
|
|
}
|
|
|
|
async importButton_onClick() {
|
|
if (!this.state.note) return;
|
|
this.setState({ restoring: true });
|
|
await RevisionService.instance().importRevisionNote(this.state.note);
|
|
this.setState({ restoring: false });
|
|
alert(_('The note "%s" has been successfully restored to the notebook "%s".', substrWithEllipsis(this.state.note.title, 0, 32), RevisionService.instance().restoreFolderTitle()));
|
|
}
|
|
|
|
backButton_click() {
|
|
if (this.props.onBack) this.props.onBack();
|
|
}
|
|
|
|
revisionList_onChange(event) {
|
|
const value = event.target.value;
|
|
|
|
if (!value) {
|
|
if (this.props.onBack) this.props.onBack();
|
|
} else {
|
|
this.setState({
|
|
currentRevId: value,
|
|
}, () => {
|
|
this.reloadNote();
|
|
});
|
|
}
|
|
}
|
|
|
|
async reloadNote() {
|
|
let noteBody = '';
|
|
if (!this.state.revisions.length || !this.state.currentRevId) {
|
|
noteBody = _('This note has no history');
|
|
this.setState({ note: null });
|
|
} else {
|
|
const revIndex = BaseModel.modelIndexById(this.state.revisions, this.state.currentRevId);
|
|
const note = await RevisionService.instance().revisionNote(this.state.revisions, revIndex);
|
|
if (!note) return;
|
|
noteBody = note.body;
|
|
this.setState({ note: note });
|
|
}
|
|
|
|
const theme = themeStyle(this.props.theme);
|
|
|
|
const mdToHtml = new MdToHtml({
|
|
resourceBaseUrl: 'file://' + Setting.value('resourceDir') + '/',
|
|
});
|
|
|
|
const result = mdToHtml.render(noteBody, theme, {
|
|
codeTheme: theme.codeThemeCss,
|
|
userCss: this.props.customCss ? this.props.customCss : '',
|
|
resources: await shared.attachedResources(noteBody),
|
|
});
|
|
|
|
this.viewerRef_.current.wrappedInstance.send('setHtml', result.html, { cssFiles: result.cssFiles });
|
|
}
|
|
|
|
render() {
|
|
const theme = themeStyle(this.props.theme);
|
|
const style = this.style();
|
|
|
|
const revisionListItems = [];
|
|
const revs = this.state.revisions.slice().reverse();
|
|
for (let i = 0; i < revs.length; i++) {
|
|
const rev = revs[i];
|
|
const stats = Revision.revisionPatchStatsText(rev);
|
|
|
|
revisionListItems.push(<option
|
|
key={rev.id}
|
|
value={rev.id}
|
|
>{time.formatMsToLocal(rev.item_updated_time) + ' (' + stats + ')'}</option>);
|
|
}
|
|
|
|
const restoreButtonTitle = _('Restore');
|
|
const helpMessage = _('Click "%s" to restore the note. It will be copied in the notebook named "%s". The current version of the note will not be replaced or modified.', restoreButtonTitle, RevisionService.instance().restoreFolderTitle());
|
|
|
|
const titleInput = (
|
|
<div style={{display:'flex', flexDirection: 'row', alignItems:'center', marginBottom: 10, borderWidth: 1, borderBottomStyle: 'solid', borderColor: theme.dividerColor, paddingBottom:10}}>
|
|
<button onClick={this.backButton_click} style={Object.assign({}, theme.buttonStyle, { marginRight: 10, height: theme.inputStyle.height })}>{'⬅ ' + _('Back')}</button>
|
|
<input readOnly type="text" style={style.titleInput} value={this.state.note ? this.state.note.title : ''}/>
|
|
<select disabled={!this.state.revisions.length} value={this.state.currentRevId} style={style.revisionList} onChange={this.revisionList_onChange}>
|
|
{revisionListItems}
|
|
</select>
|
|
<button disabled={!this.state.revisions.length || this.state.restoring} onClick={this.importButton_onClick} style={Object.assign({}, theme.buttonStyle, { marginLeft: 10, height: theme.inputStyle.height })}>{restoreButtonTitle}</button>
|
|
<HelpButton tip={helpMessage} id="noteRevisionHelpButton" onClick={this.helpButton_onClick}/>
|
|
</div>
|
|
);
|
|
|
|
const viewer = <NoteTextViewer
|
|
viewerStyle={{display:'flex', flex:1}}
|
|
ref={this.viewerRef_}
|
|
onDomReady={this.viewer_domReady}
|
|
/>
|
|
|
|
return (
|
|
<div style={style.root}>
|
|
{titleInput}
|
|
{viewer}
|
|
<ReactTooltip place="bottom" delayShow={300} className="help-tooltip"/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
}
|
|
|
|
const mapStateToProps = (state) => {
|
|
return {
|
|
theme: state.settings.theme,
|
|
};
|
|
};
|
|
|
|
const NoteRevisionViewer = connect(mapStateToProps)(NoteRevisionViewerComponent);
|
|
|
|
module.exports = NoteRevisionViewer; |