Import/export annotations

There are a few ways to import or export annotations such as from a file, a database, or a document. There are also more advanced loading options to help with finer control of the data.

Importing and exporting annotations using files

One of the options is to use XFDF files to save and load annotations. You can use AJAX requests to save and load the XFDF string from the server, and setup the server to write and read XFDF files. For example,

Import XFDF

Importing annotations require at least the document to be loaded. When using the setDocumentXFDFRetriever API, the XFDF is imported at the earliest point possible.

1WebViewer(...)
2 .then(instance => {
3 const { documentViewer, annotationManager } = instance.Core;
4
5 // Import annotations as soon as we are able to
6 documentViewer.setDocumentXFDFRetriever(async () => {
7 // load the annotation data
8 const response = await fetch('path/to/annotation/server');
9 const xfdfString = await response.text();
10
11 // <xfdf>
12 // <annots>
13 // <text subject="Comment" page="0" color="#FFE6A2" ... />
14 // </annots>
15 // </xfdf>
16 return xfdfString;
17 });
18 });

Alternatively, importing annotations can be done with the importAnnotations API as well. The earliest point in which XFDF can be imported is documentLoaded.

1documentViewer.addEventListener('documentLoaded', async () => {
2 const response = await fetch('path/to/annotation/server');
3 const xfdfString = await response.text();
4
5 await annotationManager.importAnnotations(xfdfString);
6});

Export XFDF

1WebViewer(...)
2 .then(instance => {
3 const { documentViewer, annotationManager } = instance.Core;
4
5 documentViewer.addEventListener('annotationsLoaded', () => {
6 // widgets and links will remain in the document without changing so it isn't necessary to export them
7 annotationManager.exportAnnotations({ links: false, widgets: false }).then(xfdfString => {
8 fetch('path/to/annotation/server', {
9 method: 'POST',
10 body: xfdfString // written into an XFDF file in server
11 });
12 // Full samples are available at the end of this section.
13 });
14 });
15 });

Overview

Apryse Docs Image

In the POST and GET requests you can pass an ID to the server to uniquely identify the XFDF file that should be saved/loaded. You have full flexibility to choose this ID but here are some simple examples:

  1. Use the filename as a unique identifier to have one XFDF file per document.
  2. Use a combination of the filename and username as a unique identifier to have one XFDF per user per document.

For samples about saving annotations into XFDF files in different backends, see Github repos below:

documentXFDFRetriever vs importAnnotations

Using the documentViewer.setXFDFRetriever or documentXFDFRetriever WebViewer constructor option is the preferred way of importing annotations from your server if you are adding them all at once initially.

  • documentXFDFRetriever merges your annotations at the proper time automatically and prevents conflicts and potential flashing of annotations on the page.
  • WebViewer will process the internal annotations from the document asynchronously. If you try to import your annotations with importAnnotations and your server has modifications to an annotation inside the document, for example on page 10 and WebViewer hasn't loaded the annotations for page 10 yet then there may be a conflict when merging these changes together.
  • Using the annotationsLoaded event is one way to work around this so that there aren't any conflicts, however this event doesn't fire until annotations on every page have been loaded, so for documents with many pages this may take some time. It may also cause annotations to jump if the internal annotation has loaded on a page, then much later your server annotations are imported the annotation may change positions or even be deleted from the document.

importAnnotations is fine for importing annotations later, after the document's internal annotations have loaded, however for the initial import of your server annotations documentXFDFRetriever is recommended instead.

Did you find this helpful?

Trial setup questions?

Ask experts on Discord

Need other help?

Contact Support

Pricing or product questions?

Contact Sales