File viewer for React.
npm install @cyntler/react-doc-viewer

> [!WARNING]
**Due to lack of free time to develop this library, I am stopping work on this library.
> It will not be developed in the near future.**
File viewer for React v17+.
> This is a fork of https://github.com/Alcumus/react-doc-viewer (inactivity for a long time).
> [!IMPORTANT]
> This library uses the official MS Office online document viewing service. This means it works on an iframe basis and only supports public file URLs! Therefore, it may not be compatible with all projects. Currently, there is no way to natively render MS Office documents in the browser.
- Supported file types
- Storybook Demo
- Installation
- Usage
- Required styles
- Basic
- Initial Active Document
- Control over the displayed document
- Displaying blob/uploaded documents
- Included Renderers
- Custom Renderer
- Custom File Loader
- Theme
- Custom pre-fetch HTTP Verb
- Custom Request Headers
- Internationalization (i18n)
- Styling
- CSS Class
- CSS Class Default Override
- React Inline
- Styled Components
- Using DocViewerRef
- Config
- Overriding Header Component
- Overriding Loading Renderer
- Overriding No Renderer (Error)
| Extension | MIME Type | Comments |
| --------- | ------------------------------------------------------------------------- | ------------- |
| bmp | image/bmp | |
| csv | text/csv | |
| odt | application/vnd.oasis.opendocument.text | |
| doc | application/msword | Public URLs only! |
| docx | application/vnd.openxmlformats-officedocument.wordprocessingml.document | Public URLs only! |
| gif | image/gif | |
| htm | text/htm | |
| html | text/html | |
| jpg | image/jpg | |
| jpeg | image/jpeg | |
| pdf | application/pdf | |
| png | image/png | |
| ppt | application/vnd.ms-powerpoint | Public URLs only! |
| pptx | application/vnd.openxmlformats-officedocument.presentationml.presentation | Public URLs only! |
| tiff | image/tiff | |
| txt | text/plain | |
| xls | application/vnd.ms-excel | Public URLs only! |
| xlsx | application/vnd.openxmlformats-officedocument.spreadsheetml.sheet | Public URLs only! |
| mp4 | video/mp4 | |
| webp | image/webp | |
https://cyntler.github.io/react-doc-viewer
Use one of the package managers for Node.js.
``bash`
npm i @cyntler/react-doc-viewer
# or
yarn add @cyntler/react-doc-viewer
> Warning: _By default the component height will expand and contract to the current loaded file. The width will expand to fill the parent._
The library exports a CSS file containing classes needed for correct rendering of e.g. PDF files. It is best to include it at the beginning of the application or in the place where you use this library.
`tsx`
import "@cyntler/react-doc-viewer/dist/index.css";
DocViewer requires at least an array of document objects to function.
Each document object must have a uri to a file, either a url that returns a file or a local file.
`tsx
import DocViewer, { DocViewerRenderers } from "@cyntler/react-doc-viewer";
import "@cyntler/react-doc-viewer/dist/index.css";
function App() {
const docs = [
{ uri: "https://url-to-my-pdf.pdf" }, // Remote file
{ uri: require("./example-files/pdf.pdf") }, // Local File
];
return
}
`
By default, the first item in your documents array will be displayed after the component is rendered. However, there is a prop initialActiveDocument that you can point to the initial document that should be displayed.
`tsx
import DocViewer, { DocViewerRenderers } from "@cyntler/react-doc-viewer";
import "@cyntler/react-doc-viewer/dist/index.css";
const App = () => {
const docs = [
{ uri: "https://url-to-my-pdf.pdf" }, // Remote file
{ uri: require("./example-files/pdf.pdf") }, // Local File
];
return (
initialActiveDocument={docs[1]}
pluginRenderers={DocViewerRenderers}
/>
);
};
`
From version 1.11.0 you can control the displayed document through two props: activeDocument and onDocumentChange.
`jsx
const DocViewerControlOverDisplayedDocument = () => {
const docs = [
{ uri: "https://url-to-my-pdf.pdf" }, // Remote file
{ uri: require("./example-files/pdf.pdf") }, // Local File
];
const [activeDocument, setActiveDocument] = useState(docs[0]);
const handleDocumentChange = (document) => {
setActiveDocument(document);
};
return (
<>
activeDocument={activeDocument}
onDocumentChange={handleDocumentChange}
/>
>
);
};
`
Since v1.6.2 you can use documents in the form of blobs, which allows you to e.g. display uploaded files.
`jsx
const DocViewerWithInputApp = () => {
const [selectedDocs, setSelectedDocs] = useState
return (
<>
type="file"
accept=".pdf"
multiple
onChange={(el) =>
el.target.files?.length &&
setSelectedDocs(Array.from(el.target.files))
}
/>
uri: window.URL.createObjectURL(file),
fileName: file.name,
}))}
pluginRenderers={DocViewerRenderers}
/>
>
);
};
`
To use the included renderers.
DocViewerRenderers is an Array of all the included renderers.
`tsx
import DocViewer, { DocViewerRenderers } from "@cyntler/react-doc-viewer";
import "@cyntler/react-doc-viewer/dist/index.css";
{/ ... /}
/>;
`
Or you can import individual renderers.
`tsx
import DocViewer, { PDFRenderer, PNGRenderer } from "@cyntler/react-doc-viewer";
import "@cyntler/react-doc-viewer/dist/index.css";
{/ ... /}
/>;
`
To create a custom renderer, that will just exist for your project.
`tsx
import React from "react";
import DocViewer from "@cyntler/react-doc-viewer";
const MyCustomPNGRenderer: DocRenderer = ({
mainState: { currentDocument },
}) => {
if (!currentDocument) return null;
return (
MyCustomPNGRenderer.fileTypes = ["png", "image/png"];
MyCustomPNGRenderer.weight = 1;
`
And supply it to pluginRenderers inside an Array.
`tsx
import DocViewer, { DocViewerRenderers } from "@cyntler/react-doc-viewer";
import "@cyntler/react-doc-viewer/dist/index.css";
documents={
[
// ...
]
}
/>;
`
If you need to prevent the actual loading of the file by @cyntler/react-doc-viewer.
You can decorate your custom renderer with a callback to do as you wish. e.g. Load the file yourself in an iFrame.
`tsx`
MyCustomPNGRenderer.fileLoader = ({
documentURI,
signal,
fileLoaderComplete,
}) => {
myCustomFileLoaderCode().then(() => {
// Whenever you have finished you must call fileLoaderComplete() to remove the loading animation
fileLoaderComplete();
});
};
You can provide a theme object with one or all of the available properties.
`xml`
theme={{
primary: "#5296d8",
secondary: "#ffffff",
tertiary: "#5296d899",
textPrimary: "#ffffff",
textSecondary: "#5296d8",
textTertiary: "#00000099",
disableThemeScrollbar: false,
}}
/>
Some services (such as AWS) provide URLs that works only for one pre-configured verb.
By default, @cyntler/react-doc-viewer fetches document metadata through a HEAD request in order to guess its Content-Type.prefetchMethod
If you need to have a specific verb for the pre-fetching, use the option on the DocViewer:
`tsx
import DocViewer, { DocViewerRenderers } from "@cyntler/react-doc-viewer";
`
Provide request headers, i.e. for authenticating with an API etc.
`tsx
const headers = {
"X-Access-Token": "1234567890",
"My-Custom-Header": "my-custom-value",
};
`
From v1.6.0 you can pass the language prop to the DocViewer component to get translated sentences and words that can be displayed by this library.
`xml`
The translations are based on the .json files that can be found in the src/locales directory.
Any styling applied to the component, is directly applied to the main div container.
`xml`
Each component / div already has a DOM id that can be used to style any part of the document viewer.
`css`
#react-doc-viewer #header-bar {
background-color: #faf;
}
`xml`
`tsx
import styled from "styled-components";
// ...
// ...
const MyDocViewer = styled(DocViewer)
border-radius: 10px;;`
Since v1.13.0 you can control the display of the document with reference.
`tsx
import DocViewer, { DocViewerRef } from "@cyntler/react-doc-viewer";
export const UsingRef = () => {
const docViewerRef = useRef
return (
<>
Config
You can provide a config object, which configures parts of the component as required.
`tsx
documents={docs}
config={{
header: {
disableHeader: false,
disableFileName: false,
retainURLParams: false,
},
csvDelimiter: ",", // "," as default,
pdfZoom: {
defaultZoom: 1.1, // 1 as default,
zoomJump: 0.2, // 0.1 as default,
},
pdfVerticalScrollByDefault: true, // false as default
}}
/>
`$3
You can pass a callback function to
config.header.overrideComponent that returns a React Element. The function's parameters will be populated and usable, this function will also be re-called whenever the mainState updates.
Parameters include the state object from the main component, and document navigation functions for previousDocument and nextDocument.Example:
`tsx
const MyHeader: IHeaderOverride = (state, previousDocument, nextDocument) => {
if (!state.currentDocument || state.config?.header?.disableFileName) {
return null;
} return (
<>
{state.currentDocument.uri || ""}
onClick={nextDocument}
disabled={state.currentFileNo >= state.documents.length - 1}
>
Next Document
>
);
}; pluginRenderers={DocViewerRenderers}
documents={
{
// ...
}
}
config={{
header: {
overrideComponent: MyHeader,
},
}}
/>;
`$3
You can pass a callback function to
config.loadingRenderer.overrideComponent that returns a React Element.Example:
`tsx
const MyLoadingRenderer = ({ document, fileName }) => {
const fileText = fileName || document?.fileType || ""; if (fileText) {
return
Loading Renderer ({fileText})...;
} return
Loading Renderer...;
}; pluginRenderers={DocViewerRenderers}
documents={
{
// ...
}
}
config={{
loadingRenderer: {
overrideComponent: MyLoadingRenderer,
},
}}
/>;
`By default, the loading component is rendered if document loading process takes more than 500 ms.
You can change this time value or disable this feature to make the component display immediately:
`tsx
const MyLoadingRenderer = ({ document, fileName }) => {
...
}; pluginRenderers={DocViewerRenderers}
documents={
{
// ...
}
}
config={{
loadingRenderer: {
overrideComponent: MyLoadingRenderer,
showLoadingTimeout: false, // false if you want to disable or number to provide your own value (ms)
},
}}
/>;
`$3
You can pass a callback function to
config.noRenderer.overrideComponent that returns a React Element.Example:
`tsx
const MyNoRenderer = ({ document, fileName }) => {
const fileText = fileName || document?.fileType || ""; if (fileText) {
return
No Renderer Error! ({fileText});
} return
No Renderer Error!;
}; pluginRenderers={DocViewerRenderers}
documents={
{
// ...
}
}
config={{
noRenderer: {
overrideComponent: MyNoRenderer,
},
}}
/>;
``