2018-03-09 22:59:12 +02:00
|
|
|
const React = require('react'); const Component = React.Component;
|
2018-05-02 16:13:20 +02:00
|
|
|
const { Platform, WebView, View } = require('react-native');
|
2019-03-08 19:14:17 +02:00
|
|
|
const { themeStyle } = require('lib/components/global-style.js');
|
2018-03-09 22:59:12 +02:00
|
|
|
const Resource = require('lib/models/Resource.js');
|
|
|
|
const Setting = require('lib/models/Setting.js');
|
|
|
|
const { reg } = require('lib/registry.js');
|
2019-03-08 19:14:17 +02:00
|
|
|
const { shim } = require('lib/shim');
|
2018-03-09 22:59:12 +02:00
|
|
|
const MdToHtml = require('lib/MdToHtml.js');
|
2019-03-08 19:14:17 +02:00
|
|
|
const shared = require('lib/components/shared/note-screen-shared.js');
|
2017-07-30 21:51:18 +02:00
|
|
|
|
|
|
|
class NoteBodyViewer extends Component {
|
2018-03-09 22:59:12 +02:00
|
|
|
|
2017-07-30 21:51:18 +02:00
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.state = {
|
|
|
|
resources: {},
|
2017-08-21 22:46:31 +02:00
|
|
|
webViewLoaded: false,
|
2018-03-09 22:59:12 +02:00
|
|
|
}
|
2017-08-21 22:46:31 +02:00
|
|
|
|
|
|
|
this.isMounted_ = false;
|
2017-07-30 21:51:18 +02:00
|
|
|
}
|
|
|
|
|
2018-04-30 18:38:19 +02:00
|
|
|
UNSAFE_componentWillMount() {
|
2018-05-02 16:13:20 +02:00
|
|
|
this.mdToHtml_ = new MdToHtml();
|
2017-08-21 22:46:31 +02:00
|
|
|
this.isMounted_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
2017-11-05 17:35:38 +02:00
|
|
|
this.mdToHtml_ = null;
|
2017-08-21 22:46:31 +02:00
|
|
|
this.isMounted_ = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
onLoadEnd() {
|
2018-12-29 04:12:23 +02:00
|
|
|
setTimeout(() => {
|
|
|
|
if (this.props.onLoadEnd) this.props.onLoadEnd();
|
|
|
|
}, 100);
|
|
|
|
|
2017-08-21 22:46:31 +02:00
|
|
|
if (this.state.webViewLoaded) return;
|
|
|
|
|
|
|
|
// Need to display after a delay to avoid a white flash before
|
|
|
|
// the content is displayed.
|
|
|
|
setTimeout(() => {
|
|
|
|
if (!this.isMounted_) return;
|
|
|
|
this.setState({ webViewLoaded: true });
|
2017-09-10 18:57:06 +02:00
|
|
|
}, 100);
|
2017-08-21 22:46:31 +02:00
|
|
|
}
|
|
|
|
|
2018-05-09 19:04:48 +02:00
|
|
|
shouldComponentUpdate(nextProps, nextState) {
|
2018-10-05 20:49:36 +02:00
|
|
|
|
|
|
|
const safeGetNoteProp = (props, propName) => {
|
|
|
|
if (!props) return null;
|
|
|
|
if (!props.note) return null;
|
|
|
|
return props.note[propName];
|
|
|
|
}
|
|
|
|
|
2018-05-09 19:04:48 +02:00
|
|
|
// To address https://github.com/laurent22/joplin/issues/433
|
|
|
|
// If a checkbox in a note is ticked, the body changes, which normally would trigger a re-render
|
|
|
|
// of this component, which has the unfortunate side effect of making the view scroll back to the top.
|
|
|
|
// This re-rendering however is uncessary since the component is already visually updated via JS.
|
|
|
|
// So here, if the note has not changed, we prevent the component from updating.
|
|
|
|
// This fixes the above issue. A drawback of this is if the note is updated via sync, this change
|
|
|
|
// will not be displayed immediately.
|
2018-10-05 20:49:36 +02:00
|
|
|
const currentNoteId = safeGetNoteProp(this.props, 'id');
|
|
|
|
const nextNoteId = safeGetNoteProp(nextProps, 'id');
|
|
|
|
|
|
|
|
if (currentNoteId !== nextNoteId || nextState.webViewLoaded !== this.state.webViewLoaded) return true;
|
|
|
|
|
|
|
|
// If the length of the body has changed, then it's something other than a checkbox that has changed,
|
|
|
|
// for example a resource that has been attached to the note while in View mode. In that case, update.
|
|
|
|
return (safeGetNoteProp(this.props, 'body') + '').length !== (safeGetNoteProp(nextProps, 'body') + '').length;
|
2018-05-09 19:04:48 +02:00
|
|
|
}
|
|
|
|
|
2018-10-13 00:25:11 +02:00
|
|
|
rebuildMd() {
|
2019-03-08 19:14:17 +02:00
|
|
|
// this.mdToHtml_.clearCache();
|
2018-10-13 00:25:11 +02:00
|
|
|
this.forceUpdate();
|
|
|
|
}
|
|
|
|
|
2017-07-30 21:51:18 +02:00
|
|
|
render() {
|
|
|
|
const note = this.props.note;
|
|
|
|
const style = this.props.style;
|
|
|
|
const onCheckboxChange = this.props.onCheckboxChange;
|
2019-03-08 19:14:17 +02:00
|
|
|
const theme = themeStyle(this.props.theme);
|
|
|
|
|
|
|
|
const bodyToRender = note ? note.body : '';
|
2017-11-05 17:35:38 +02:00
|
|
|
|
|
|
|
const mdOptions = {
|
2017-11-06 20:05:12 +02:00
|
|
|
onResourceLoaded: () => {
|
2018-12-15 02:42:19 +02:00
|
|
|
if (this.resourceLoadedTimeoutId_) {
|
|
|
|
clearTimeout(this.resourceLoadedTimeoutId_);
|
|
|
|
this.resourceLoadedTimeoutId_ = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.resourceLoadedTimeoutId_ = setTimeout(() => {
|
|
|
|
this.resourceLoadedTimeoutId_ = null;
|
|
|
|
this.forceUpdate();
|
|
|
|
}, 100);
|
2017-11-05 17:35:38 +02:00
|
|
|
},
|
2018-03-09 22:59:12 +02:00
|
|
|
paddingBottom: '3.8em', // Extra bottom padding to make it possible to scroll past the action button (so that it doesn't overlap the text)
|
2018-12-16 19:32:42 +02:00
|
|
|
highlightedKeywords: this.props.highlightedKeywords,
|
2019-03-08 19:14:17 +02:00
|
|
|
resources: this.props.noteResources,//await shared.attachedResources(bodyToRender),
|
|
|
|
codeTheme: theme.codeThemeCss,
|
2017-11-05 17:35:38 +02:00
|
|
|
};
|
|
|
|
|
2019-03-08 19:14:17 +02:00
|
|
|
let result = this.mdToHtml_.render(bodyToRender, this.props.webViewStyle, mdOptions);
|
|
|
|
let html = result.html;
|
|
|
|
|
|
|
|
const injectedJs = [this.mdToHtml_.injectedJavaScript()];
|
|
|
|
injectedJs.push(shim.injectedJs('webviewLib'));
|
|
|
|
injectedJs.push('webviewLib.initialize({ postMessage: postMessage });');
|
2018-02-04 19:12:24 +02:00
|
|
|
|
2019-03-08 19:14:17 +02:00
|
|
|
console.info(injectedJs);
|
2019-02-28 01:38:50 +02:00
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
html = `
|
2018-02-04 19:12:24 +02:00
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
|
|
|
|
</head>
|
|
|
|
<body>
|
2018-03-09 22:59:12 +02:00
|
|
|
` + html + `
|
2018-02-04 19:12:24 +02:00
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
`;
|
2017-08-21 22:46:31 +02:00
|
|
|
|
2019-02-24 13:07:49 +02:00
|
|
|
let webViewStyle = {'backgroundColor': this.props.webViewStyle.backgroundColor}
|
2017-11-19 17:19:36 +02:00
|
|
|
// On iOS, the onLoadEnd() event is never fired so always
|
|
|
|
// display the webview (don't do the little trick
|
|
|
|
// to avoid the white flash).
|
2018-03-09 22:59:12 +02:00
|
|
|
if (Platform.OS !== 'ios') {
|
2017-11-19 17:19:36 +02:00
|
|
|
webViewStyle.opacity = this.state.webViewLoaded ? 1 : 0.01;
|
|
|
|
}
|
2017-07-30 21:51:18 +02:00
|
|
|
|
2017-11-20 02:21:40 +02:00
|
|
|
// On iOS scalesPageToFit work like this:
|
|
|
|
//
|
|
|
|
// Find the widest image, resize it *and everything else* by x% so that
|
|
|
|
// the image fits within the viewport. The problem is that it means if there's
|
|
|
|
// a large image, everything is going to be scaled to a very small size, making
|
|
|
|
// the text unreadable.
|
|
|
|
//
|
|
|
|
// On Android:
|
|
|
|
//
|
|
|
|
// Find the widest elements and scale them (and them only) to fit within the viewport
|
|
|
|
// It means it's going to scale large images, but the text will remain at the normal
|
|
|
|
// size.
|
|
|
|
//
|
|
|
|
// That means we can use scalesPageToFix on Android but not on iOS.
|
|
|
|
// The weird thing is that on iOS, scalesPageToFix=false along with a CSS
|
|
|
|
// rule "img { max-width: 100% }", works like scalesPageToFix=true on Android.
|
|
|
|
// So we use scalesPageToFix=false on iOS along with that CSS rule.
|
|
|
|
|
2017-11-20 21:01:19 +02:00
|
|
|
// `baseUrl` is where the images will be loaded from. So images must use a path relative to resourceDir.
|
2017-11-21 21:47:29 +02:00
|
|
|
const source = {
|
|
|
|
html: html,
|
2018-03-09 22:59:12 +02:00
|
|
|
baseUrl: 'file://' + Setting.value('resourceDir') + '/',
|
2017-11-21 21:47:29 +02:00
|
|
|
};
|
2017-11-20 21:01:19 +02:00
|
|
|
|
2017-07-30 21:51:18 +02:00
|
|
|
return (
|
|
|
|
<View style={style}>
|
|
|
|
<WebView
|
2018-03-09 22:59:12 +02:00
|
|
|
scalesPageToFit={Platform.OS !== 'ios'}
|
2017-08-21 22:46:31 +02:00
|
|
|
style={webViewStyle}
|
2017-11-20 21:01:19 +02:00
|
|
|
source={source}
|
2019-03-08 19:14:17 +02:00
|
|
|
injectedJavaScript={injectedJs.join('\n')}
|
2018-10-02 21:02:38 +02:00
|
|
|
originWhitelist={['file://*', './*', 'http://*', 'https://*']}
|
|
|
|
mixedContentMode="always"
|
|
|
|
allowFileAccess={true}
|
2017-08-21 22:46:31 +02:00
|
|
|
onLoadEnd={() => this.onLoadEnd()}
|
2018-03-15 01:17:02 +02:00
|
|
|
onError={() => reg.logger().error('WebView error') }
|
2018-03-09 22:59:12 +02:00
|
|
|
onMessage={(event) => {
|
2017-07-30 21:51:18 +02:00
|
|
|
let msg = event.nativeEvent.data;
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
if (msg.indexOf('checkboxclick:') === 0) {
|
2019-03-08 19:14:17 +02:00
|
|
|
const newBody = shared.toggleCheckbox(msg, this.props.note.body);
|
|
|
|
if (this.props.onCheckboxChange) this.props.onCheckboxChange(newBody);
|
2018-03-09 22:59:12 +02:00
|
|
|
} else if (msg.indexOf('bodyscroll:') === 0) {
|
2017-11-05 17:35:38 +02:00
|
|
|
//msg = msg.split(':');
|
|
|
|
//this.bodyScrollTop_ = Number(msg[1]);
|
2017-07-30 21:51:18 +02:00
|
|
|
} else {
|
2018-05-02 16:13:20 +02:00
|
|
|
this.props.onJoplinLinkClick(msg);
|
2017-07-30 21:51:18 +02:00
|
|
|
}
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
2018-03-09 22:59:12 +02:00
|
|
|
|
2017-07-30 21:51:18 +02:00
|
|
|
}
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
module.exports = { NoteBodyViewer };
|