UNPKG

@epubjs-react-native/core

Version:

A digital book reader in .opf .epub format for react native using epub.js library inside a webview.

228 lines 7.41 kB
function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); } import React, { useContext, useEffect, useState } from 'react'; import { Platform } from 'react-native'; import { LoadingFile } from './utils/LoadingFile'; import { View } from './View'; import { useInjectWebViewVariables } from './hooks/useInjectWebviewVariables'; import { ReaderContext, defaultTheme as initialTheme } from './context'; import { isURL } from './utils/isURL'; import { getSourceType } from './utils/getSourceType'; import { getSourceName } from './utils/getPathname'; import { SourceType } from './utils/enums/source-type.enum'; import { isFsUri } from './utils/isFsUri'; import jszip from './jszip'; import epubjs from './epubjs'; export function Reader({ src, width = '100%', height = '100%', defaultTheme = initialTheme, initialLocations, allowScriptedContent = Platform.OS === 'ios', onPressExternalLink, renderLoadingFileComponent = props => /*#__PURE__*/React.createElement(LoadingFile, _extends({}, props, { width: width, height: height })), fileSystem: useFileSystem, menuItems, manager = 'default', flow = 'auto', snap, spread, fullsize, charactersPerLocation, ...rest }) { const { downloadFile, size: fileSize, progress: downloadProgress, success: downloadSuccess, error: downloadError, documentDirectory, writeAsStringAsync } = useFileSystem(); const enableSelection = menuItems ? true : rest.enableSelection || false; const allowPopups = onPressExternalLink ? true : rest.allowPopups || false; const { setIsLoading, isLoading } = useContext(ReaderContext); const { injectWebViewVariables } = useInjectWebViewVariables(); const [template, setTemplate] = useState(null); const [templateUrl, setTemplateUrl] = useState(null); const [allowedUris, setAllowedUris] = useState(null); useEffect(() => { (async () => { setIsLoading(true); const jszipFileUri = `${documentDirectory}/jszip.min.js`; const epubjsFileUri = `${documentDirectory}/epub.min.js`; try { await writeAsStringAsync(jszipFileUri, jszip); } catch (e) { throw new Error('failed to write jszip js file'); } try { await writeAsStringAsync(epubjsFileUri, epubjs); } catch (e) { throw new Error('failed to write epubjs js file'); } setAllowedUris(`${jszipFileUri},${epubjsFileUri}`); if (src) { const sourceType = getSourceType(src); const isExternalSource = isURL(src); const isSrcInFs = isFsUri(src); if (!sourceType) { throw new Error(`Invalid source type: ${src}`); } if (!isExternalSource) { if (isSrcInFs) { setAllowedUris(`${src}${jszipFileUri},${epubjsFileUri}`); } if (sourceType === SourceType.BASE64) { setTemplate(injectWebViewVariables({ jszip: jszipFileUri, epubjs: epubjsFileUri, type: SourceType.BASE64, book: src, theme: defaultTheme, locations: initialLocations, enableSelection, allowScriptedContent, allowPopups, manager, flow, snap, spread, fullsize, charactersPerLocation })); setIsLoading(false); } else { setTemplate(injectWebViewVariables({ jszip: jszipFileUri, epubjs: epubjsFileUri, type: SourceType.BINARY, book: src, theme: defaultTheme, locations: initialLocations, enableSelection, allowScriptedContent, allowPopups, manager, flow, snap, spread, fullsize, charactersPerLocation })); setIsLoading(false); } } if (isExternalSource) { const sourceName = getSourceName(src); if (!sourceName) { throw new Error(`Invalid source name: ${src}`); } if (sourceType === SourceType.OPF || sourceType === SourceType.EPUB) { setTemplate(injectWebViewVariables({ jszip: jszipFileUri, epubjs: epubjsFileUri, type: sourceType, book: src, theme: defaultTheme, locations: initialLocations, enableSelection, allowScriptedContent, allowPopups, manager, flow, snap, spread, fullsize, charactersPerLocation })); setIsLoading(false); } else { const { uri: bookFileUri } = await downloadFile(src, sourceName); if (!bookFileUri) throw new Error("Couldn't download book"); setAllowedUris(`${bookFileUri},${jszipFileUri},${epubjsFileUri}`); setTemplate(injectWebViewVariables({ jszip: jszipFileUri, epubjs: epubjsFileUri, type: sourceType, book: bookFileUri, theme: defaultTheme, locations: initialLocations, enableSelection, allowScriptedContent, allowPopups, manager, flow, snap, spread, fullsize, charactersPerLocation })); setIsLoading(false); } } } })(); // eslint-disable-next-line react-hooks/exhaustive-deps }, [allowPopups, allowScriptedContent, defaultTheme, documentDirectory, downloadFile, enableSelection, initialLocations, injectWebViewVariables, setIsLoading, src // ! Causing unknown loop // writeAsStringAsync, ]); useEffect(() => { const saveTemplateFileToDoc = async () => { try { if (template) { const content = template; const fileUri = `${documentDirectory}/index.html`; await writeAsStringAsync(fileUri, content); setTemplateUrl(fileUri); } } catch (error) { throw new Error('Error saving index.html file:'); } }; if (template) { saveTemplateFileToDoc(); } }, [documentDirectory, template, writeAsStringAsync]); if (isLoading) { return renderLoadingFileComponent({ fileSize, downloadProgress, downloadSuccess, downloadError }); } if (!templateUrl || !allowedUris) { return renderLoadingFileComponent({ fileSize, downloadProgress, downloadSuccess, downloadError }); } return /*#__PURE__*/React.createElement(View, _extends({ templateUri: templateUrl, allowedUris: allowedUris, width: width, height: height, defaultTheme: defaultTheme || initialTheme, onPressExternalLink: onPressExternalLink, enableSelection: enableSelection, menuItems: menuItems, manager: manager, flow: flow, snap: snap }, rest)); }