2021-03-26 11:02:16 +02:00
|
|
|
import { useRef, useCallback } from 'react';
|
2020-10-16 17:26:19 +02:00
|
|
|
|
|
|
|
import useSource from './hooks/useSource';
|
|
|
|
import useOnMessage from './hooks/useOnMessage';
|
|
|
|
import useOnResourceLongPress from './hooks/useOnResourceLongPress';
|
|
|
|
|
|
|
|
const React = require('react');
|
2022-09-05 13:46:13 +02:00
|
|
|
import { View } from 'react-native';
|
2021-01-22 19:41:11 +02:00
|
|
|
import BackButtonDialogBox from '../BackButtonDialogBox';
|
2021-01-29 20:45:11 +02:00
|
|
|
import { reg } from '@joplin/lib/registry';
|
2022-09-05 13:46:13 +02:00
|
|
|
import ExtendedWebView from '../ExtendedWebView';
|
2020-10-16 17:26:19 +02:00
|
|
|
|
|
|
|
interface Props {
|
2020-11-12 21:29:22 +02:00
|
|
|
themeId: number;
|
|
|
|
style: any;
|
|
|
|
noteBody: string;
|
|
|
|
noteMarkupLanguage: number;
|
|
|
|
highlightedKeywords: string[];
|
|
|
|
noteResources: any;
|
|
|
|
paddingBottom: number;
|
|
|
|
noteHash: string;
|
2023-06-30 11:30:29 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2020-11-12 21:29:22 +02:00
|
|
|
onJoplinLinkClick: Function;
|
2023-06-30 11:30:29 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2020-11-12 21:29:22 +02:00
|
|
|
onCheckboxChange?: Function;
|
2023-06-30 11:30:29 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2020-11-12 21:29:22 +02:00
|
|
|
onMarkForDownload?: Function;
|
2023-06-30 11:30:29 +02:00
|
|
|
// eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied
|
2020-11-12 21:29:22 +02:00
|
|
|
onLoadEnd?: Function;
|
2020-10-16 17:26:19 +02:00
|
|
|
}
|
|
|
|
|
2021-03-26 11:02:16 +02:00
|
|
|
const webViewStyle = {
|
|
|
|
backgroundColor: 'transparent',
|
|
|
|
};
|
|
|
|
|
2020-11-12 21:13:28 +02:00
|
|
|
export default function NoteBodyViewer(props: Props) {
|
2020-10-16 17:26:19 +02:00
|
|
|
const dialogBoxRef = useRef(null);
|
|
|
|
|
2022-09-05 13:46:13 +02:00
|
|
|
const { html, injectedJs } = useSource(
|
2020-10-16 17:26:19 +02:00
|
|
|
props.noteBody,
|
|
|
|
props.noteMarkupLanguage,
|
|
|
|
props.themeId,
|
|
|
|
props.highlightedKeywords,
|
|
|
|
props.noteResources,
|
|
|
|
props.paddingBottom,
|
2023-08-22 12:58:53 +02:00
|
|
|
props.noteHash,
|
2020-10-16 17:26:19 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
const onResourceLongPress = useOnResourceLongPress(
|
|
|
|
props.onJoplinLinkClick,
|
2023-08-22 12:58:53 +02:00
|
|
|
dialogBoxRef,
|
2020-10-16 17:26:19 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
const onMessage = useOnMessage(
|
|
|
|
props.onCheckboxChange,
|
|
|
|
props.noteBody,
|
|
|
|
props.onMarkForDownload,
|
|
|
|
props.onJoplinLinkClick,
|
2023-08-22 12:58:53 +02:00
|
|
|
onResourceLongPress,
|
2020-10-16 17:26:19 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
const onLoadEnd = useCallback(() => {
|
|
|
|
if (props.onLoadEnd) props.onLoadEnd();
|
|
|
|
}, [props.onLoadEnd]);
|
|
|
|
|
|
|
|
function onError() {
|
|
|
|
reg.logger().error('WebView error');
|
|
|
|
}
|
|
|
|
|
2022-09-05 13:46:13 +02:00
|
|
|
const BackButtonDialogBox_ = BackButtonDialogBox as any;
|
|
|
|
|
2020-10-16 17:26:19 +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.
|
|
|
|
//
|
|
|
|
// 2020-10-15: As we've now fully switched to WebKit for iOS (useWebKit=true) and
|
|
|
|
// since the WebView package went through many versions it's possible that
|
|
|
|
// the above no longer applies.
|
|
|
|
return (
|
|
|
|
<View style={props.style}>
|
2022-09-05 13:46:13 +02:00
|
|
|
<ExtendedWebView
|
2022-09-12 11:46:12 +02:00
|
|
|
webviewInstanceId='NoteBodyViewer'
|
2022-09-05 13:46:13 +02:00
|
|
|
themeId={props.themeId}
|
2020-10-16 17:26:19 +02:00
|
|
|
style={webViewStyle}
|
2022-09-05 13:46:13 +02:00
|
|
|
html={html}
|
2020-10-16 17:26:19 +02:00
|
|
|
injectedJavaScript={injectedJs.join('\n')}
|
|
|
|
mixedContentMode="always"
|
|
|
|
onLoadEnd={onLoadEnd}
|
|
|
|
onError={onError}
|
|
|
|
onMessage={onMessage}
|
|
|
|
/>
|
2021-01-22 19:41:11 +02:00
|
|
|
<BackButtonDialogBox_ ref={dialogBoxRef}/>
|
2020-10-16 17:26:19 +02:00
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|