Some test text!
Web / Guides / Annotate videos
WebViewer Video allows you to create annotations in a video file and attach them to a specific timeframe. Try it out in our Video Annotation Demo now. It leverages some of the familiar WebViewer annotation tools like Note, Free Text, Rectangle and Free Hand. Video annotations comes out of the box with zero configuration.
It is also possible to control the amount of time said annotation will be displayed on video playback. After the annotation is created, click on the desired annotation on top of the timeline reel and use the directional arrows to expand or contract the amount of time it will be displayed.
Annotations can also have comments attached to it, initially ordered by timeframe (bottom being the earliest to show on video). Those are searchable and can be further sorted by position, time, status, author, type and timecode.
A common use case is to save and retrieve the annotations. Just like on WebViewer, this data is stored in the format of a XFDF string that can be exported using the AnnotationsManager's exportAnnotations API.
One example is to have a custom save button added to the WebViewer's header using the setHeaderItems API that triggers a call to exportAnnotations and sending a POST request to your server to store it.
Assuming that there is a previously setup server running and a POST endpoint that accepts a body containing the XFDF string, you can achieve with the following code snippet:
// setHeaderItems is a function that can be extracted from a WebViewer instance
setHeaderItems((header) => {
header.push({
type: "actionButton",
img: "<svg>...</svg>",
onClick: async () => {
// Save annotations when button is clicked
// widgets and links will remain in the document without changing so it isn't necessary to export them
// Make a POST request with XFDF string
const saveXfdfString = (documentId, xfdfString) => {
return new Promise((resolve) => {
fetch(`/server/annotationHandler.js?documentId=${documentId}`, {
method: "POST",
body: xfdfString,
}).then((response) => {
if (response.status === 200) {
resolve();
}
});
});
};
// Step 1: Get a list of annotations
const annotations = docViewer.getAnnotationManager().getAnnotationsList();
// Step 2: Export to a string
var xfdfString = await annotManager.exportAnnotations({
links: false,
widgets: false,
annotList: annotations,
});
// Step 3: Call your endpoint and save it
await saveXfdfString(DOCUMENT_ID, xfdfString);
alert("Annotations saved successfully.");
},
});
});
To load the annotations, the process is quite similar. It involves making a GET request to your server to retrieve the XFDF string, use the AnnotationManager's importAnnotations and finally call video.updateAnnotationsToTime(0)
to apply the annotations to the specific timeframes that they were created.
These calls need to be done after the video has been loaded. You can safely do these steps inside docViewer's videoElementLoaded event like the following code snippet example:
docViewer.addEventListener("videoElementLoaded", () => {
const video = docViewer.getDocument().getVideo();
// Make a GET request to get XFDF string
const loadXfdfString = (documentId) => {
return new Promise((resolve) => {
fetch(`/server/annotationHandler.js?documentId=${documentId}`, {
method: "GET",
}).then((response) => {
if (response.status === 200) {
response.text().then((xfdfString) => {
resolve(xfdfString);
});
}
});
});
};
// Step 1: Make a request to your server to get the XFDF string
loadXfdfString(DOCUMENT_ID)
.then((xfdfString) => {
// Step 2: Call importAnnotations to save annotations to AnnotationManager
const annotManager = docViewer.getAnnotationManager();
return annotManager.importAnnotations(xfdfString);
})
.then(() => {
// Step 3: Apply the annotations to the video timeframes
video.updateAnnotationsToTime(0);
});
});
Annotations can also be added programmatically.
docViewer.on("videoElementLoaded", () => {
const annot = new instance.Core.Annotations.RectangleAnnotation({
Author: 'Test Author',
Subject: 'Rectangle',
ToolName: 'AnnotationCreateRectangle',
Width: 231.5,
Height: 182.5,
X: 321,
Y: 40,
Listable: true,
Color: new instance.Core.Annotations.Color(78, 125, 233, 1),
StrokeColor: new instance.Core.Annotations.Color(78, 125, 233, 1),
StrokeThickness: 5,
Opacity: 1,
});
// Annot time is in seconds
annot.setStartTime(0);
annot.setEndTime(100);
const video = getVideo();
annotManager.addAnnotations([ annot ]);
// Apply the annotations to the video timeframes
video.updateAnnotationsToTime(0);
});
For a complete working sample, you can check out the official WebViewer Video Sample.
Trial setup questions? Ask experts on Discord
Need other help? Contact Support
Pricing or product questions? Contact Sales