2023-01-19 19:19:06 +02:00
|
|
|
import * as React from 'react';
|
|
|
|
import Folder from '@joplin/lib/models/Folder';
|
|
|
|
import { themeStyle } from '@joplin/lib/theme';
|
|
|
|
import { _ } from '@joplin/lib/locale';
|
|
|
|
import { filename, basename } from '@joplin/lib/path-utils';
|
|
|
|
import importEnex from '@joplin/lib/import-enex';
|
|
|
|
import { AppState } from '../app.reducer';
|
2017-11-11 19:36:47 +02:00
|
|
|
const { connect } = require('react-redux');
|
|
|
|
|
2023-01-19 19:19:06 +02:00
|
|
|
interface Props {
|
|
|
|
filePath: string;
|
|
|
|
themeId: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface Message {
|
|
|
|
key: string;
|
|
|
|
text: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface State {
|
|
|
|
filePath: string;
|
|
|
|
doImport: boolean;
|
|
|
|
messages: Message[];
|
|
|
|
}
|
|
|
|
|
|
|
|
class ImportScreenComponent extends React.Component<Props, State> {
|
2019-12-13 03:16:34 +02:00
|
|
|
UNSAFE_componentWillMount() {
|
2017-11-11 19:36:47 +02:00
|
|
|
this.setState({
|
|
|
|
doImport: true,
|
|
|
|
filePath: this.props.filePath,
|
|
|
|
messages: [],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-01-19 19:19:06 +02:00
|
|
|
UNSAFE_componentWillReceiveProps(newProps: Props) {
|
2017-11-11 19:36:47 +02:00
|
|
|
if (newProps.filePath) {
|
2019-07-29 14:13:23 +02:00
|
|
|
this.setState(
|
|
|
|
{
|
|
|
|
doImport: true,
|
|
|
|
filePath: newProps.filePath,
|
|
|
|
messages: [],
|
|
|
|
},
|
|
|
|
() => {
|
2023-01-19 19:19:06 +02:00
|
|
|
void this.doImport();
|
2019-07-29 14:13:23 +02:00
|
|
|
}
|
|
|
|
);
|
2017-11-11 19:36:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
if (this.state.filePath && this.state.doImport) {
|
2023-01-19 19:19:06 +02:00
|
|
|
void this.doImport();
|
2017-11-11 19:36:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-19 19:19:06 +02:00
|
|
|
addMessage(key: string, text: string) {
|
2017-11-11 19:36:47 +02:00
|
|
|
const messages = this.state.messages.slice();
|
|
|
|
|
2017-12-07 02:57:36 +02:00
|
|
|
messages.push({ key: key, text: text });
|
2017-11-11 19:36:47 +02:00
|
|
|
|
|
|
|
this.setState({ messages: messages });
|
|
|
|
}
|
|
|
|
|
2017-12-07 02:57:36 +02:00
|
|
|
uniqueMessages() {
|
2020-03-14 01:46:14 +02:00
|
|
|
const output = [];
|
2017-12-07 02:57:36 +02:00
|
|
|
const messages = this.state.messages.slice();
|
2020-03-14 01:46:14 +02:00
|
|
|
const foundKeys = [];
|
2017-12-07 02:57:36 +02:00
|
|
|
for (let i = messages.length - 1; i >= 0; i--) {
|
|
|
|
const msg = messages[i];
|
|
|
|
if (foundKeys.indexOf(msg.key) >= 0) continue;
|
|
|
|
foundKeys.push(msg.key);
|
|
|
|
output.unshift(msg);
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2017-11-11 19:36:47 +02:00
|
|
|
async doImport() {
|
|
|
|
const filePath = this.props.filePath;
|
2018-06-27 22:45:31 +02:00
|
|
|
const folderTitle = await Folder.findUniqueItemTitle(filename(filePath));
|
2017-11-11 19:36:47 +02:00
|
|
|
|
|
|
|
this.addMessage('start', _('New notebook "%s" will be created and file "%s" will be imported into it', folderTitle, basename(filePath)));
|
|
|
|
|
|
|
|
let lastProgress = '';
|
|
|
|
|
|
|
|
const options = {
|
2023-01-19 19:19:06 +02:00
|
|
|
onProgress: (progressState: any) => {
|
2020-03-14 01:46:14 +02:00
|
|
|
const line = [];
|
2017-11-11 19:36:47 +02:00
|
|
|
line.push(_('Found: %d.', progressState.loaded));
|
|
|
|
line.push(_('Created: %d.', progressState.created));
|
|
|
|
if (progressState.updated) line.push(_('Updated: %d.', progressState.updated));
|
|
|
|
if (progressState.skipped) line.push(_('Skipped: %d.', progressState.skipped));
|
|
|
|
if (progressState.resourcesCreated) line.push(_('Resources: %d.', progressState.resourcesCreated));
|
|
|
|
if (progressState.notesTagged) line.push(_('Tagged: %d.', progressState.notesTagged));
|
|
|
|
lastProgress = line.join(' ');
|
|
|
|
this.addMessage('progress', lastProgress);
|
|
|
|
},
|
2023-01-19 19:19:06 +02:00
|
|
|
onError: (error: any) => {
|
2017-12-07 02:57:36 +02:00
|
|
|
// Don't display the error directly because most of the time it doesn't matter
|
|
|
|
// (eg. for weird broken HTML, but the note is still imported)
|
|
|
|
console.warn('When importing ENEX file', error);
|
2017-11-11 19:36:47 +02:00
|
|
|
},
|
2019-07-29 14:13:23 +02:00
|
|
|
};
|
2017-11-11 19:36:47 +02:00
|
|
|
|
2017-11-12 01:13:14 +02:00
|
|
|
const folder = await Folder.save({ title: folderTitle });
|
2019-07-29 14:13:23 +02:00
|
|
|
|
2017-11-12 01:13:14 +02:00
|
|
|
await importEnex(folder.id, filePath, options);
|
2017-11-11 19:36:47 +02:00
|
|
|
|
|
|
|
this.addMessage('done', _('The notes have been imported: %s', lastProgress));
|
|
|
|
this.setState({ doImport: false });
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
2020-09-15 15:01:07 +02:00
|
|
|
const theme = themeStyle(this.props.themeId);
|
2017-12-07 02:57:36 +02:00
|
|
|
const messages = this.uniqueMessages();
|
2017-11-11 19:36:47 +02:00
|
|
|
|
|
|
|
const messagesStyle = {
|
|
|
|
padding: 10,
|
|
|
|
fontSize: theme.fontSize,
|
|
|
|
fontFamily: theme.fontFamily,
|
|
|
|
backgroundColor: theme.backgroundColor,
|
|
|
|
};
|
|
|
|
|
|
|
|
const messageComps = [];
|
|
|
|
for (let i = 0; i < messages.length; i++) {
|
|
|
|
messageComps.push(<div key={messages[i].key}>{messages[i].text}</div>);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div style={{}}>
|
2019-07-29 14:13:23 +02:00
|
|
|
<div style={messagesStyle}>{messageComps}</div>
|
2017-11-11 19:36:47 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-19 19:19:06 +02:00
|
|
|
const mapStateToProps = (state: AppState) => {
|
2017-11-11 19:36:47 +02:00
|
|
|
return {
|
2020-09-15 15:01:07 +02:00
|
|
|
themeId: state.settings.theme,
|
2017-11-11 19:36:47 +02:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
const ImportScreen = connect(mapStateToProps)(ImportScreenComponent);
|
|
|
|
|
2023-01-19 19:19:06 +02:00
|
|
|
export default ImportScreen;
|
|
|
|
|