Copyright © 2016 W3C® (MIT, ERCIM, Keio, Beihang). W3C liability, trademark and document use rules apply.
Annotations are typically used to convey information about a resource or associations between resources. Simple examples include a comment or tag on a single web page or image, or a blog post about a news article.
The Web Annotation Data Model specification describes a structured model and format to enable annotations to be shared and reused across different hardware and software platforms. Common use cases can be modeled in a manner that is simple and convenient, while at the same time enabling more complex requirements, including linking arbitrary content to a particular data point or to segments of timed multimedia resources.
The specification provides a specific JSON format for ease of creation and consumption of annotations based on the conceptual model that accommodates these use cases, and the vocabulary of terms that represents it.
This section describes the status of this document at the time of its publication. Other documents may supersede this document. A list of current W3C publications and the latest revision of this technical report can be found in the W3C technical reports index at http://www.w3.org/TR/.
This is a work in progress. No section should be considered final, and the absence of any content does not imply that such content is out of scope, or may not appear in the future. If you feel something should be covered, please tell us!
This specification was derived from the Open Annotation Community Group's outcomes, and details of the differences between the two are maintained in the Acknowledgement appendix.
This document was published by the Web Annotation Working Group as a Working Draft. This document is intended to become a W3C Recommendation. If you wish to make comments regarding this document, please send them to [email protected] (subscribe, archives). All comments are welcome.
Publication as a Working Draft does not imply endorsement by the W3C Membership. This is a draft document and may be updated, replaced or obsoleted by other documents at any time. It is inappropriate to cite this document as other than work in progress.
This document was produced by a group operating under the 5 February 2004 W3C Patent Policy. W3C maintains a public list of any patent disclosures made in connection with the deliverables of the group; that page also includes instructions for disclosing a patent. An individual who has actual knowledge of a patent which the individual believes contains Essential Claim(s) must disclose the information in accordance with section 6 of the W3C Patent Policy.
This document is governed by the 1 September 2015 W3C Process Document.
This section is non-normative.
Annotating, the act of creating associations between distinct pieces of information, is a pervasive activity online in many guises. Web citizens make comments about online resources using either tools built in to the hosting website, external web services, or the functionality of an annotation client. Comments about shared photos or videos, reviews of products, or even social network mentions of web resources could all be considered as annotations. In addition, there are a plethora of "sticky note" systems and stand-alone multimedia annotation systems. This specification describes a common approach to expressing these annotations, and more.
The Web Annotation Data Model provides an extensible, interoperable framework for expressing annotations such that they can easily be shared between platforms, with sufficient richness of expression to satisfy complex requirements while remaining simple enough to also allow for the most common use cases, such as attaching a piece of text to a single web resource.
An annotation is considered to be a set of connected resources, typically including a body and target, and conveys that the body is related to the target. The exact nature of this relationship changes according to the intention of the annotation, but the body is most frequently somehow "about" the target. This perspective results in a basic model with three parts, depicted below. The full model supports additional functionality, enabling content to be embedded within the annotation, selecting arbitrary segments of resources, choosing the appropriate representation of a resource and providing styling hints to help clients render the annotation appropriately. Annotations created by or intended for machines are also possible, ensuring that the Data Web is not ignored in favor of only considering the human-oriented Document Web.
The Web Annotation Data Model does not prescribe a transport protocol for creating, managing and retrieving annotations. Instead it describes a resource oriented structure and serialization of that structure that could be carried over many different protocols. The related [annotation-protocol] specification describes a recommended transport layer, which may be adopted separately.
The primary aim of the Web Annotation Data Model is to provide a standard description model and format to enable annotations to be shared between systems. This interoperability may be either for sharing with others, or the migration of private annotations between devices or platforms. The shared annotations must be able to be integrated into existing collections and reused without loss of significant information. The model should cover as many annotation use cases as possible, while keeping the simple annotations easy and expanding from that baseline to make complex uses possible.
The Web Annotation Data Model is a single, consistent model that can be used by all interested parties. All efforts have been made to keep the implementation costs for both producers and consumers to a minimum. A single method of fulfilling a use case is strongly preferred over multiple methods, unless there are existing standards that need to be accommodated or there is a significant cost associated with using only a single method. While the Data Model is built using Linked Data fundamentals, the design is intended to allow rich and performant non-graph-based implementations. As such, inferencing and other graph-based queries are explicitly not a priority for optimization in the design of the model.
The examples throughout the document are serialized as [JSON-LD] using the Context given in Appendix A of the Annotation Vocabulary [annotation-vocab], which is the preferred serialization format.
When the only information that is recorded in the annotation is the IRI of a resource, then that IRI is used as the value of the relationship, as in Example 1. When there is more information about the resource, the IRI is the value of the id
property of the object which is the value of the relationship, as in Example 2.
As well as sections marked as non-normative, all authoring guidelines, diagrams, examples, and notes in this specification are non-normative. Everything else in this specification is normative.
The key words MAY, MUST, MUST NOT, NOT RECOMMENDED, RECOMMENDED, SHOULD, and SHOULD NOT are to be interpreted as described in [RFC2119].
The Web Annotation Data Model is defined using the following basic principles:
The following principles describe additional distinctions regarding the exact nature of Target and Body:
The following principles describe additional semantics regarding multiple resources:
An Annotation is a web resource. Typically, an Annotation has a single Body, which is a comment or other descriptive resource, and a single Target that the Body is somehow "about". The Annotation likely also has additional descriptive properties.
Example Use Case: Alice has written a post that makes a comment about a particular web page. Her client creates an Annotation with the post as the body resource, and the web page as the target resource.
Term | Type | Description |
---|---|---|
@context | Property | The context that determines the meaning of the JSON as an Annotation
The Annotation MUST have 1 or more @context values and http://www.w3.org/ns/anno.jsonld MUST be one of them. If there is only one value, then it MUST be provided as a string. |
id | Property | The identity of the Annotation
An Annotation SHOULD have exactly 1 IRI that identifies it. |
type | Relationship | The type of the Annotation.
An Annotation MUST have 1 or more types, and the Annotation class MUST be one of them. |
Annotation | Class | The class for Web Annotations
The Annotation class MUST be associated with an Annotation using type . |
body | Relationship | The relationship between an Annotation and its Body
There SHOULD be 1 or more body relationships associated with an Annotation but there MAY be 0.
|
target | Relationship | The relationship between an Annotation and its Target
There MUST be 1 or more target relationships associated with an Annotation.
|
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno1", "type": "Annotation", "body": "http://example.org/post1", "target": "http://example.com/page1" }
Web Resources are identified with a IRI and have various properties, often including a format or language for the resource's content. This information may be recorded as part of the Annotation, even if the representation of the resource must be retrieved from the Web.
Example Use Case: Beatrice records a long analysis of a patent, and publishes the audio on her website as an mp3. She then creates an Annotation with the mp3 as the body, and the PDF of the patent as the target.
Term | Type | Description |
---|---|---|
id | Property | The IRI that identifies the Body or Target resource.
Bodies or Targets which are External Web Resources MUST have exactly 1 id with the value of the resource's IRI.
|
language | Property | The language of the Web Resource's content
The Body or Target SHOULD have exactly 1 language associated with it, but MAY have 0 or more. The value of the property SHOULD be a language code following the [rfc5646] specifiction. |
format | Property | The format of the Web Resource's content
The Body or Target SHOULD have exactly 1 format associated with it, but MAY have 0 or more. The value of the property SHOULD be the media-type of the format, following the [rfc6838] specification. |
format
property. The [w3c-language-tags] article provides a good overview of the values that implementers can expect to encounter in the language
property.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno2", "type": "Annotation", "body": { "id": "http://example.org/analysis1.mp3", "format": "audio/mpeg", "language": "fr" }, "target": { "id": "http://example.gov/patent1.pdf", "format": "application/pdf", "language": "en" } }
It is useful for clients to know the general type of a Web Resource in advance. If the client cannot render videos, then knowing that the Body is a video will allow it to avoid needlessly downloading a potentially large content stream. For resources that do not have obvious media types, such as many data formats, it is also useful for a client to know that a resource with the format text/csv
should not simply be rendered as plain text, despite the first part of the media type, whereas application/pdf
may be able to be rendered by the user agent despite the main type being 'application'.
Example Use Case: Corina shoots a video of her comment about a website on her phone and uploads it. She associates the video with the website via an Annotation, and her client adds types as a hint to consuming systems.
Term | Type | Description |
---|---|---|
type | Relationship | The type of the Body resource.
The Body or Target MAY have 1 or more type s, and if so, the value SHOULD be drawn from the list of classes below, but MAY come from other vocabularies. |
Dataset | Class | The class for a resource which encodes data in a defined structure |
Image | Class | The class for image resources, primarily intended to be seen |
Video | Class | The class for video resources, with or without audio |
Sound | Class | The class for a resource primarily intended to be heard |
Text | Class | The class for a resource primarily intended to be read |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno3", "type": "Annotation", "body": { "id": "http://example.org/video1", "type": "Video" }, "target": { "id": "http://example.org/website1", "type": "Text" } }
Many Annotations involve part of an External Web Resource, rather than its entirety. In the Web [webarch], segments of resources are identified using IRIs with a fragment component that at the same time both describes how to extract the segment of interest from the resource, and identifies the extracted content. For simple Annotations, it is valuable to be able to use these IRIs with a fragment component as the identifier for either Body or Target.
Example Use Case: Dawn wants to describe a particular region of an image. She highlights that area in her client and types in the description. Her client then constructs an IRI with an appropriate fragment component as the target.
Term | Type | Description |
---|---|---|
id | Property | The IRI that identifies the Body or Target resource.
Bodies or Targets which are External Web Resources MUST have exactly 1 id with the value of the resource's IRI, and that IRI MAY have a fragment component.
|
type
, format
and language
, plus those described in the Other Properties section below, can be applied to the segment of the resource, just like for the full resource.It is important to be aware of the consequences of using an IRI with a fragment component, and the restrictions that using them places on implementations.
http://example.com/image.jpg#xywh=1,1,1,1
would not be discovered in a simple search for
http://example.com/image.jpg
, even though it is part of it.{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno4", "type": "Annotation", "body": "http://example.org/description1", "target": { "id": "http://example.com/image1#xywh=100,100,300,300", "type": "Image", "format": "image/jpeg" } }
In many situations, the body of the Annotation will be in a text format, and created at the same time as the Annotation and not have a separate IRI. In these cases, the Body's text can be included as part of the Annotation to avoid having to interact with multiple systems. The Body may also have the features of External Web Resources, including especially the language of the text and the format that it is conveyed in.
Example Use Case: Emily writes a comment about how much she likes an image on a photo sharing website. Her client creates an Annotation with the comment embedded within it, and adds that it is in French and formatted using HTML.
Term | Type | Description |
---|---|---|
id | Property | The IRI that identifies the Textual Body
The Body MAY have exactly 1 IRI that identifies it. |
type | Relationship | The type of the Textual Body resource.
The Body SHOULD have the TextualBody class, and MAY have other classes. |
TextualBody | Class | A class assigned to the Body for embedding textual resources within the Annotation.
The Body SHOULD have the TextualBody class. |
text | Property | The character sequence of the content of the Textual Body. There MUST be exactly 1 text property associated with the TextualBody. |
Systems SHOULD assume that Textual Bodies have the Text
class, described in Classes above, even if it is not explicitly included in the type
property.
The properties of External Web Resources, such as language
and format
also apply to embedded Textual Body resources.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno5", "type":"Annotation", "body": { "type" : "TextualBody", "text" : "<p>Comment text</p>", "format" : "text/html", "language" : "en" }, "target": "http://example.org/photo1" }
The simplest type of Body is a plain text string, without additional information or properties. This type of body is useful for the simplest of Annotations only, and is NOT RECOMMENDED for uses where the body may need to be referred to from outside of the Annotation.
Example Use Case Franceska wants to create a quick Annotation from a simple, command line client. She creates the JSON serialization in a text file and sends it to her Annotation server to maintain.
Term | Type | Description |
---|---|---|
bodyText | Property | The string value of the body of the Annotation.
There MAY be exactly 1 bodyText for an Annotation, and the value MUST conform to the requirements below. If the bodyText property is present, then the body relationship MUST NOT also be present. |
There are several restrictions on when this form may be used and how it is to be interpreted.
The string body:
xsd:string
and the data type MUST NOT be expressed in the serialization.text
property of a Textual Body.format
property with the value text/plain
.purpose
property with the value of commenting
.TextualBody
construction, rather than maintaining the bodyText
form.{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno6", "type":"Annotation", "bodyText": "Comment text", "target": "http://example.org/target1" }
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno7", "type":"Annotation", "body": { "type": "TextualBody", "text": "Comment text", "format": "text/plain", "purpose": "commenting" }, "target": "http://example.org/target1" }
Some Annotations may not have a Body at all, such as a simple highlight or bookmark without any accompanying text. It is also possible for an Annotation to have multiple Bodies and/or Targets. In this case, each Body is considered to be equally related to each Target individually, rather than to the set of all of the Targets.
Example Use Case: Gretchen highlights a particular region of her ebook in green and, knowing what such a highlight means, she does not give a comment. Her client associates a stylesheet with the Annotation, and does not create a body at all.
Example Use Case: Hannah associates a tag and a description with two images using a single annotation.
The body
relationship is omitted when there is no body for the Annotation.
The body
and/or target
relationships of the Annotation may be arrays rather than a single object. The values may be either strings containing the IRI of the resource or objects.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno8", "type": "Annotation", "target": "http://example.org/ebook1" }
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno9", "type": "Annotation", "body": [ "http://example.org/description1", { "type": "TextualBody", "text": "tag1" } ], "target": [ "http://example.org/image1", "http://example.org/image2" ] }
A Choice has an ordered list of resources from which an application should select only one to process or display. The order is given from the most preferable to least preferable, according to the Annotation's creator or publisher.
Example Use Case: Irina writes up her discussion of a particular website in both French and English. As the two posts are equivalent, there is no need to display both, and instead she wants French speakers to see the French comment, and everyone else to see the English version. Her client creates as Choice with the English comment listed first.
Term | Type | Description |
---|---|---|
Choice | Class | A construction that conveys to a consuming application that it should select one of the listed resources to display to the user, and not render all of them. |
items | Relationship | A list of resources to choose from, with the default option listed first. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno10", "type": "Annotation", "body": { "type": "Choice", "items": [ { "id": "http://example.org/note1", "language": "en" }, { "id": "http://example.org/note2", "language": "fr" } ] }, "target": "http://example.org/website1" }
The person, organization or machine responsible for the Annotation or referenced resource deserves credit for their contribution, and the time at which those resources are created is useful for display ordering and filtering out old, potentially irrelevant content. The creator of the Annotation is also useful for determining the trustworthiness of the Annotation. The software used to create and serialize the Annotation, along with when that activity occurred, is useful for both advertising and debugging issues.
Example Use Case: Jane writes a review of a restaurant online, and wishes to be associated with that review so that her friends know that it was her review and can trust it. Her client adds her account's identity, and its own identity, to the Annotation and the appropriate timestamps for when the resources were created.
Term | Type | Description |
---|---|---|
creator | Relationship | The agent responsible for creating the Annotation, Body or Target. This may be either a human, an organization or a software agent.
There SHOULD be exactly 1 creator relationship per resource, but MAY be 0 or more than 1, as the resource's creator may wish to remain anonymous, or multiple agents may have worked together on it. |
created | Property | The time at which the Annotation, Body or Target was created.
There SHOULD be exactly 1 created property per resource, and MUST NOT be more than 1.
The datetime MUST be expressed according to the W3C Datetime Format [datetime], and SHOULD be a complete xsd:dateTime with a timezone. |
generator | Relationship | The agent responsible for generating the serialization of the Annotation.
There MAY be 0 or more generator relationships per Annotation |
generated | Property | The time at which the Annotation serialization was generated.
There MAY be exactly 1 generated property per Annotation, and MUST NOT be more than 1.
The datetime MUST be expressed according to the W3C Datetime Format [datetime], and SHOULD be a complete xsd:dateTime with a timezone. |
modified | Property | The time at which the Annotation, Body or Target was modified, after creation.
There MAY be exactly 1 modified property per resource, and MUST NOT be more than 1.
The datetime datetime MUST be expressed according to the W3C Datetime Format [datetime], and SHOULD be a complete xsd:dateTime with a timezone. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno11", "type": "Annotation", "creator": "http://example.org/user1", "created": "2015-01-28T12:00:00Z", "modified": "2015-01-29T09:00:00Z", "generator": "http://example.org/client1", "generated": "2015-02-04T12:00:00Z", "body": { "id": "http://example.net/review1", "creator": "http://example.net/user2", "created": "2014-06-02T17:00:00Z" }, "target": "http://example.com/restaurant1" }
More information about the agents involved in the creation of an Annotation is normally required beyond an IRI that identifies them. This includes whether they are an individual, a group or a piece of software and properties such as real name, account name, and email address.
Example Use Case: Kelly wants to submit an Annotation to a system that does not manage her identity, and would like a pseudonym to be displayed. Her client adds this information to the Annotation to send to the service.
Term | Type | Description |
---|---|---|
id | Property | The IRI that identifies the agent
An Agent SHOULD have exactly 1 IRI that identifies it, and MUST NOT have more than 1. |
type | Relationship | The type of the Agent.
An Agent SHOULD have 1 or more classes, from those listed below. |
Person | Class | The class for a human agent |
Organization | Class | The class for an organization, as opposed to an individual. |
Software | Class | The class for a software agent, such as a user's client or a machine learning system that creates Annotations. |
name | Property | The name of the agent.
Each agent SHOULD have exactly 1 name property, and MAY have 0 or more. |
account | Property | The account name of the agent.
Each agent SHOULD have exactly 1 account property, and MAY have 0. |
Relationship | The email address associated with the agent, using the mailto: IRI scheme [rfc6086].
Each agent MAY have 1 or more email addresses | |
email_sha1 | Property | The text representation of the result of applying the sha1 algorithm to the email IRI of the agent, including the 'mailto:' prefix and no whitespace. This allows the mail address to be used as an identifier without publishing the address publicly. Each agent MAY have 1 or more values in the email_sha1 property. |
homepage | Relationship | The home page for the agent. Each agent MAY have 1 or more home pages. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno12", "type": "Annotation", "creator": { "id": "http://example.org/user1", "type": "Person", "name": "My Pseudonym", "nick": "pseudo", "email_sha1": "58bad08927902ff9307b621c54716dcc5083e339" }, "generator": { "id": "http://example.org/client1", "type": "SoftwareAgent", "name": "Code v2.1", "homepage": "http://example.org/client1/homepage1" }, "body": "http://example.net/review1", "target": "http://example.com/restaurant1" }
Beyond the agents that are associated with the creation and management of the Annotation and other resources, it is also useful to know the audience or class of consuming agent that the resource is intended for. This allows for roles (such as teacher versus student) or properties of the class (such as a suggested age range).
Example Use Case: Lynda writes some notes about using a particular textbook to teach a class. She adds that the intended audience of the Annotation is teachers (who are using the textbook), to distinguish from other Annotations that might have an audience of the students (who are also using the textbook, but to learn from).
Term | Type | Description |
---|---|---|
id | Property | The IRI that identifies the Audience.
There MAY be exactly 1 IRI given that identifies the Audience. |
type | Relationship | The type of the Audience, from the schema.org class structure. The Audience SHOULD have 1 or more type s and they SHOULD come from the schema.org class structure. |
audience | Relationship | The relationship between an Annotation and its intended Audience.
There MAY be 0 or more Audiences for each Annotation. |
Further properties that describe the audience are used from schema.org's Audience classes. The properties and class names MUST be prefixed in the JSON with schema:
to ensure that they are uniquely distinguished from any other properties or classes.
The use of audience
does not imply nor enable any access restriction to prevent the annotation from being seen. Systems SHOULD use the information for filtering the display of Annotations based on their knowledge of the user, and not assume that the Annotation or other resources will require authentication and authorization.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno13", "type": "Annotation", "audience": { "id": "http://example.edu/roles/teacher", "type": "schema:EducationalAudience", "schema:educationalRole": "teacher" }, "body": "http://example.net/classnotes1", "target": "http://example.com/textbook1" }
In many cases it is important to understand the reasons why the Annotation was created, or why the Textual Body was included in the Annotation, not just the times and agents involved. These reasons are provided by declaring the motivation for the Annotation's creation or the purpose for the inclusion of the Body in the Annotation; the "why" rather than the "who" and "when" described in the previous sections.
Example Use Case: Megan annotates a resource intending to bookmark it for future reference, and provides a description and a tag to make it easier to find again. Her client adds the right motivations to the Annotation and the Body resources to capture this.
Term | Type | Description |
---|---|---|
motivation | Relationship | The relationship between an Annotation and a Motivation.
There SHOULD be exactly 1 motivation for each Annotation, and MAY be 0 or more than 1. |
purpose | Relationship | The reason for the inclusion of the Textual Body within the Annotation.
There MAY be 0 or more purpose s associated with a TextualBody . |
Motivation | Class | The Motivation for an Annotation is a reason for its creation, and might include things like Replying to another annotation, Commenting on a resource, or Linking to a related resource. |
Motivations | ||
bookmarking | Instance | The motivation for when the user intends to create a bookmark to the Target or part thereof. For example an Annotation that bookmarks the point in a text where the reader finished reading. |
classifying | Instance | The motivation for when the user intends to that classify the Target as something. For example to classify an image as a portrait. |
commenting | Instance | The motivation for when the user intends to comment about the Target. For example to provide a commentary about a particular PDF document. |
describing | Instance | The motivation for when the user intends to describe the Target, as opposed to a comment about them. For example describing the above PDF's contents, rather than commenting on their accuracy. |
editing | Instance | The motivation for when the user intends to request a change or edit to the Target resource. For example an Annotation that requests a typo to be corrected. |
highlighting | Instance | The motivation for when the user intends to highlight the Target resource or segment of it. For example to draw attention to the selected text that the annotator disagrees with. |
identifying | Instance | The motivation for when the user intends to assign an identity to the Target. For example to associate the IRI that identifies a city with a mention of the city in a web page. |
linking | Instance | The motivation for when the user intends to link to a resource related to the Target. |
moderating | Instance | The motivation for when the user intends to assign some value or quality to the Target. For example annotating an Annotation to moderate it up in a trust network or threaded discussion. |
questioning | Instance | The motivation for when the user intends to ask a question about the Target. For example to ask for assistance with a particular section of text, or question its veracity. |
replying | Instance | The motivation for when the user intends to reply to a previous statement, either an Annotation or another resource. For example providing the assistance requested in the above. |
reviewing | Instance | The motivation for when the user intends to review the Target in some assessing fashion, rather than simply make a comment about it. For example to write a review of a Book. |
tagging | Instance | The motivation for when the user intends to associate a tag with the Target. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno14", "type": "Annotation", "motivation": "bookmarking", "body": [ { "text": "readme", "purpose": "tagging" }, { "text": "A good description of the topic that bears further investigation", "purpose": "describing" } ], "target": "http://example.com/page1" }
It is common practice to associate a licence or rights statement with a resource, in order to describe the conditions under which it may be used. This allows the user to make appropriate use of the resource, as well as allowing some automated systems to confirm that the usage is permitted.
Example Use Case: Noelle writes a review of a product and wishes to be known as the author of the review, however does not mind how the Annotation that relates the review and the product together is used. She asserts these two separate rights statements with the Annotation and Body individually.
Term | Type | Description |
---|---|---|
rights | Relationship | The relationship between an Annotation, Body or Target and a statement about the rights or licence under which the resource may be used.
There MAY be at exactly 1 rights statement or license linked from each resource. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno15", "type": "Annotation", "body": { "id": "http://example.net/review1", "rights": "http://creativecommons.org/licenses/by/4.0/" }, "target": "http://example.com/product1" }
In a massively distributed system such as the Web, information is often copied. In order to track the provenance of the Annotation and other related resources, it is possible to record additional IRIs that also identify the resource. These may be dereferencable "permalinks", identities assigned by a client offline without any knowledge of the web, or simply the location where the current harvesting system discovered the resource.
Example Use Case: Ophelia creates an Annotation and sends it to multiple systems to maintain, one personal and one public. She wants to ensure that the copies can be aligned, and so she sets a UUID as the canonical IRI, allowing the service to assign an HTTP IRI for it. A subsequent system then harvests the public copy, maintaining the canonical UUID and moves the assigned HTTP IRI to via
, replacing the id
field with its own IRI.
Term | Type | Description |
---|---|---|
canonical | Relationship | The relationship between an Annotation, Body or Target and the IRI that SHOULD be used to track its identity, regardless of where it is made accessible. If this property is set, then systems MUST NOT change or delete it. Systems SHOULD NOT assign a canonical IRI without prior agreement if one is not present, as the Annotation could already have a canonical IRI elsewhere.
There MAY be exactly 1 canonical IRI for each resource. |
via | Relationship | The relationship between an Annotation, Body or Target and the IRI of where that resource was obtained from by the system that is making it available.
There MAY be 0 or more IRIs provided in via for each resource. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno16", "type": "Annotation", "canonical": "urn:uuid:dbfb1861-0ecf-41ad-be94-a584e5c4f1df", "via": "http://other.example.org/anno1", "body": { "id": "http://example.net/review1", "rights": "http://creativecommons.org/licenses/by/4.0/" }, "target": "http://example.com/product1" }
While it is possible using only the constructions described above to create Annotations that reference parts of resources by using IRIs with a fragment component, there are many situations when this is not sufficient. For example, even a simple circular region of an image, or a diagonal line across it, are not possible. Selecting an arbitrary span of text in an HTML page, perhaps the simplest annotation concept, is also not supported by fragments. Furthermore, there are non-segment use cases that require a client to retrieve a specific state or representation of the resource, to style it in a particular way, to associate a role with the resource that is specific to the Annotation's use of it, or for the Annotation to only apply when the resource is used in a particular context.
The Web Annotation Data Model uses a new type of resource to capture these Annotation-specific requirements: a SpecificResource. The SpecificResource is used in between the Annotation and the Body or Target, as appropriate, to capture additional information about how it is used in the Annotation. The descriptions are typically referenced from the SpecificResource as separate entities and can be of various types to capture the different requirements. For example, if the Target of the Annotation is a circular region of an image, then the SpecificResource is the circular region, it is described by a Selector, and is also associated with the source Image resource.
Specific Resources and Specifiers MAY be External Web Resources with their own IRIs, such as in the example for the Selector construction, however it is RECOMMENDED that they be included in the Annotation's representation to avoid requiring unnecessary network interactions to retrieve all of the information needed to process the Annotation.
The types of additional specificity that are defined by this document:
Term | Type | Description |
---|---|---|
id | Property | The identity of the Specific Resource
A Specific Resource MAY have exactly 1 IRI that identifies it. |
type | Relationship | The class of the Specific Resource
The Specific Resource SHOULD have the SpecificResource class. |
SpecificResource | Class | The class for Specific Resources
The SpecificResource class SHOULD be associated with a Specific Resource to be clear as to its role as a more specific region or state of another resource. |
source | Relationship | The relationship between a Specific Resource and the resource that it is a more specific representation of.
There MUST be exactly 1 source relationship associated with a Specific Resource. The source resource MAY be described in detail as in the core data model or be just the resource's IRI.
|
The same Specific Resource and Specifier classes are used for both Target and Body. The examples in this section only use one of these, however the same model applies for both.
As well as Textual Bodies, External Web Resources may also be given a Motivation as to their inclusion within the Annotation. This is done using the Specific Resource pattern, as the purpose specifies the way in which the resource is used in the context of the Annotation in the same way as a Selector describes the segment or a State describes the representation.
Example Use Case: Petra wants to tag a photo with an identifier for a city, rather than just type the city's name which could be ambiguous. Her client uses a well-known IRI for the city having done a search for it, and creates a Specific Resource to manage that purpose assignment.
Term | Type | Description |
---|---|---|
purpose | Relationship | The reason for including the Web Resource in the Annotation.
There MAY be 0 or more Motivations associated with the SpecificResource using purpose . |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno17", "type": "Annotation", "body": { "type": "SpecificResource", "purpose": "tagging", "source": "http://example.org/city1" }, "target": { "id": "http://example.org/photo1", "type": "Image" } }
Many Annotations refer to part of a resource, rather than all of it, as the Target. We call that part of the resource a Segment (of Interest). A Selector is used to describe how to determine the Segment from within the Source resource. The nature of the Selector will be dependent on the type of resource, as the methods to describe Segments from various media-types will differ.
Example Use Case: Qitara wants to associate a selection of text in a web page, with a slice of a dataset. She selects both using her client, and creates the Annotation with a SpecificResource that has a Selector for each of the Body and the Target.
Term | Type | Description |
---|---|---|
selector | Relationship | The relationship between a Specific Resource and a Selector.
There MUST be exactly 0 or 1 selector relationships associated with a Specific Resource.
|
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno18", "type": "Annotation", "body": { "source": "http://example.org/page1", "selector": "http://example.org/paraselector1" }, "target": { "source": "http://example.com/dataset1", "selector": "http://example.org/dataselector1" } }
As the most well understood mechanism for selecting a Segment is to use the fragment part of an IRI defined by the representation's media type, it is useful to allow this as a description mechanism via a Selector. This allows existing and future fragment specifications to be used with Specific Resources in a consistent way. To be clear about which fragment type is being used, the Selector may refer to the specification that defines it.
Example Use Case: Ramona wants to associate part of a video as the description of an image. She selects the time range within the video and clicks that it is describing the target. Her client then creates the Annotation using a SpecificResource with a FragmentSelector and the describing
Motivation.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. FragmentSelectors MUST have a type and the value MUST be FragmentSelector . |
FragmentSelector | Class | A resource which describes the Segment through the use of the fragment component of an IRI. |
value | Property | The contents of the fragment component of an IRI that describes the Segment. The FragmentSelector MUST have exactly 1 value property. |
conformsTo | Relationship | The relationship between the FragmentSelector and the specification that defines the syntax of the IRI fragment in the value property.
The Fragment Selector SHOULD have exactly 1 conformsTo link to the specification that defines the syntax of the fragment and MUST NOT have more than 1. |
It is RECOMMENDED to use FragmentSelector
as a consistent method compatible with other means of describing SpecificResources, rather than using the IRI with a fragment directly. Consuming applications SHOULD be aware of both.
The following IRIs are some of the specifications that define the semantics of fragments, and hence may be used with the conformsTo
relationship. Other IRIs MAY also be used.
Name | Fragment Specification | Description |
---|---|---|
HTML | http://tools.ietf.org/rfc/rfc3236 | [rfc3236] Example: namedSection |
http://tools.ietf.org/rfc/rfc3778 | [rfc3778] Example: page=10&viewrect=50,50,640,480 | |
Plain Text | http://tools.ietf.org/rfc/rfc5147 | [rfc5147] Example: char=0,10 |
XML | http://tools.ietf.org/rfc/rfc3023 | [rfc3023] Example: xpointer(/a/b/c) |
RDF/XML | http://tools.ietf.org/rfc/rfc3870 | [rfc3870] Example: namedResource |
CSV | http://tools.ietf.org/rfc/rfc7111 | [rfc7111] Example: row=5-7 |
Media | http://www.w3.org/TR/media-frags/ | [media-frags] Example: xywh=50,50,640,480 |
SVG | http://www.w3.org/TR/SVG/ | [SVG] Example: svgView(viewBox(50,50,640,480)) |
EPUB3 | http://www.idpf.org/epub/linking/cfi/epub-cfi.html | [CFI] Example: epubcfi(/6/4[chap01ref]!/4[body01]/10[para05]/3:10) |
source
, a #
, and the value
. For example, the IRI from the example below would be http://example.org/video1#t=30,60
.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno19", "type": "Annotation", "body": { "source": "http://example.org/video1", "role": "describing", "selector": { "type": "FragmentSelector", "conformsTo": "http://www.w3.org/TR/media-frags/", "value": "t=30,60" } }, "target": "http://example.org/image1" }
One of the most common ways to select elements in the HTML Document Object Model is to use CSS Selectors [CSS3-selectors]. CSS Selectors allow for a wide variety of well supported ways to describe the path to an element in a web page, and thus cover many of the basic use cases for Web Annotation. Results are not defined for when a CSS Selector is applied to a representation that does not conform to the Document Object Model.
Note that CSS may also be used for styling a resource within an annotation. This class is specifically to re-use the CSS Selector mechanism to select a segment of a resource that conforms to the Document Object Model.
Example Use Case: Sally selects a paragraph in a web page that she wishes to write a note about. Her client calculates a CSS path that cleanly identifies that element and adds it to the annotation.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. CssSelectors MUST have a type and the value MUST be CssSelector . |
CssSelector | Class | The type of the CSS Selector resource.
CSS Selectors MUST have this class associated with them. |
value | Property | The CSS selection path to the Segment.
There MUST be exactly 1 value associated with a CSS Selector. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno20", "type": "Annotation", "body": "http://example.org/note1", "target": { "source": "http://example.org/page1.html", "selector": { "type": "CssSelector", "value": "#elemid > .elemclass + p" } } }
Another common method of selecting elements and content within a resource that supports the Document Object Model (DOM), such as documents in XML or HTML, is to use an XPath selection [DOM-Level-3-XPath]. XPath allows a great deal of flexibility when describing the path through the structure to the selected content. Results are not defined for when an XPath Selector is applied to a representation that does not conform to the DOM.
Example Use Case: Teynika selects a span within a table in an HTML page and writes a note about the content. To refer explicitly to this element, her client carefully constructs an XPath to identify it as the target of the Annotation.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. XPath Selectors MUST have a type and the value MUST be XPathSelector . |
XPathSelector | Class | The type of the XPath Selector resource.
XPath Selectors MUST have this class associated with them. |
value | Property | The xpath to the selected segment.
There MUST be exactly 1 value associated with an XPath Selector. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno21", "type": "Annotation", "body": "http://example.org/note1", "target": { "source": "http://example.org/page1.html", "selector": { "type": "XPathSelector", "value": "/html/body/p[2]/table/tr[2]/td[3]/span" } } }
This Selector describes a range of text by copying it, and including some of the text immediately before (a prefix) and after (a suffix) it to distinguish between multiple copies of the same sequence of characters.
For example, if the document were again "abcdefghijklmnopqrstuvwxyz", one could select "efg" by a prefix of "abcd", the match of "efg" and a suffix of "hijk".
Example Use Case: Ulrika selects a typo ('anotation') in a web page and adds a comment that it should be replaced with the correct spelling ('annotation').
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. Text Quote Selectors MUST have a type and the value MUST be TextQuoteSelector . |
TextQuoteSelector | Class | The class for a Selector that describes a textual segment by means of quoting it, plus passages before or after it.
The TextQuoteSelector MUST have this class associated with it. |
exact | Property | A copy of the text which is being selected, after normalization.
Each TextQuoteSelector MUST have exactly 1 exact property. |
prefix | Property | A snippet of text that occurs immediately before the text which is being selected.
Each TextQuoteSelector SHOULD have exactly 1 prefix property, and MUST NOT have more than 1. |
suffix | Property | The snippet of text that occurs immediately after the text which is being selected.
Each TextQuoteSelector SHOULD have exactly 1 suffix property, and MUST NOT have more than 1. |
The text MUST be normalized before recording. Thus HTML/XML tags should be removed, character entities should be replaced with the character that they encode, unnecessary whitespace should be normalized, character encoding should be turned into UTF-8, and so forth. The normalization routine may be performed automatically by a browser, and other applications should implement the DOM String Comparisons method. This allows the Selector to be used with different encodings and user agents and still have the same semantics and utility.
When generating the Text Position Selector values, it is recommended that implementations built for an environment that supports DOM Level 3 APIs use the textContent
property of top level, user visible Node object [DOM-Level-3-Core] of the resource. This content would be located at document.body.textContent
in an HTML5 document being accessed via a JavaScript DOM implementation.
If, after processing the prefix, exact, and suffix, the user agent discovers multiple matching text sequences, then the selection SHOULD be treated as matching all of the matches.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno22", "type": "Annotation", "body": "http://example.org/comment1", "target": { "source": "http://example.org/page1", "selector": { "type": "TextQuoteSelector", "exact": "anotation", "prefix": "this is an ", "suffix": " that has some" } } }
This Selector describes a range of text by recording the start and end positions of the selection in the stream. Position 0 would be immediately before the first character, position 1 would be immediately before the second character, and so on. The start character is thus included in the list, but the end character is not.
For example, if the document was "abcdefghijklmnopqrstuvwxyz", the start was 4, and the end was 7, then the selection would be "efg".
Example Use Case: Valeria writes a review of an ebook that does not allow its content to be extracted and copied. Her client describes the selection using its start and end position in the content.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. Text Position Selectors MUST have a type and the value MUST be TextPositionSelector . |
TextPositionSelector | Class | The class for a Selector which describes a range of text based on its start and end positions.
The TextPositionSelector MUST have this class associated with it. |
start | Property | The starting position of the segment of text. The first character in the full text is character position 0, and the character is included within the segment.
Each TextPositionSelector MUST have exactly 1 start property, and the value MUST be a non-negative integer. |
end | Property | The end position of the segment of text. The character is not included within the segment.
Each TextPositionSelector MUST have exactly 1 end property, and the value MUST be a non-negative integer. |
The text MUST be normalized before counting the characters, in the same way as for Text Quote Selector.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno23", "type": "Annotation", "body": "http://example.org/review1", "target": { "source": "http://example.org/ebook1", "selector": { "type": "TextPositionSelector", "start": 412, "end": 795 } } }
Similar to the Text Position Selector, the Data Position Selector uses the same properties but works at the byte in bitstream level rather than the character in text level.
Example Use Case: Wendy writes comments about regions of online disk images for forensic purposes and describing emulation requirements. Her client generates the start and end positions from the binary stream, rather than the more human readable display she is using.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. Data Position Selectors MUST have a type and the value MUST be DataPositionSelector . |
DataPositionSelector | Class | The class for a Selector which describes a range of data based on its start and end positions within the byte stream.
The DataPositionSelector MUST have this class associated with it. |
start | Property | The starting position of the segment of data. The first byte is character position 0.
Each DataPositionSelector MUST have exactly 1 start property. |
end | Property | The end position of the segment of data. The last character is not included within the segment.
Each DataPositionSelector MUST have exactly 1 end property. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno24", "type": "Annotation", "body": "http://example.org/note1", "target": { "source": "http://example.org/diskimg1", "selector": { "type": "DataPositionSelector", "start": 4096, "end": 4104 } } }
An SvgSelector defines an area through the use of the Scalable Vector Graphics [SVG] standard. This allows the user to select a non-rectangular area of the content, such as a circle or polygon by describing the region using SVG. The SVG may be either embedded within the Annotation or referenced as an External Web Resource.
Note that the SvgSelector uses SVG to select an area of a resource. Segments of an SVG representation may also be selected using selectors, including the FragmentSelector or even an SvgSelector.
Example Use Case: Xena is tagging an old map online with a diagonal region for a historical road. Her client creates SVG polygon to describe the region, relative to the image content.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. SVG Selectors MUST have a type and the value MUST include SVGSelector . It MUST also include the Content class if the SVG representation is embedded within the Annotation document. |
SvgSelector | Class | The class for a Selector which defines a shape for the selected area using the SVG standard.
The Selector MUST have this class associated with it and MAY also have the Content class associated with it.
|
Content | Class | The class for resources embedded in the Annotation, other than Textual Bodies.
If the content of the SVG description is embedded, then the Selector MUST have this class associated with it as well as the SvgSelector class.
|
text | Property | The character sequence of the SVG content. There MAY be exactly 1 text property associated with the Selector, and if so the value of the property MUST be well-formed SVG XML. |
format | Property | The format of the content.
The Selector MAY have exactly 1 format associated with it. If present, the value of the property MUST be the SVG media-type of image/svg+xml . |
The dimensions of the SVG shape or canvas MUST be relative to the dimensions of the Source resource, such that scaling the shape's size to the full size of the image correctly describes the desired area.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno25", "type": "Annotation", "body": "http://example.org/road1", "target": { "source": "http://example.org/map1", "selector": { "id": "http://example.org/svg1", "type": "SvgSelector" } } }
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno26", "type": "Annotation", "body": "http://example.org/road1", "target": { "source": "http://example.org/map1", "selector": { "type": ["SvgSelector", "Content"], "text": "<svg:svg> ... </svg:svg>", "format": "image/svg+xml" } } }
Selections made by users may be extensive and/or cross over internal boundaries in the representation, making it difficult to construct a single selector that robustly describes the correct content. A Range Selector can be used to identify the beginning and the end of the selection by using other Selectors. In this way, two points can be accurately identified using the most appropriate selection mechanisms, and then linked together to form the selection. The selection consists of everything from the beginning of the starting selector through to the beginning of the ending selector, but not including it.
Example Use Case: Yadira wants to comment on two adjacent cells in a table that is part of a web page. She selects the two cells and her client constructs XPaths to the the first cell, and the cell that immediately follows the second. Her client then creates a Range Selector with the first XPath Selector as the start, and the second XPath selector as the end.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Selector. Range Selectors MUST have a type and the value MUST be RangeSelector . |
RangeSelector | Class | The type of the Range Selector resource.
Range Selectors MUST have this class associated with them. |
startSelector | Relationship | The Selector which describes the inclusive starting point of the range.
There MUST be exactly 1 startSelector associated with a Range Selector. |
endSelector | Relationship | The Selector which describes the exclusive ending point of the range.
There MUST be exactly 1 endSelector associated with a Range Selector. Both startSelector and endSelector SHOULD be of the same class. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno27", "type": "Annotation", "body": "http://example.org/comment1", "target": { "source": "http://example.org/page1.html", "selector": { "type": "RangeSelector", "startSelector": { "type": "XPathSelector", "value": "//table[1]/tr[1]/td[2]" }, "endSelector": { "type": "XPathSelector", "value": "//table[1]/tr[1]/td[4]" } } } }
It may be easier, more reliable or more accurate to specify the segment of interest of a resource as a selection of a selection, rather than as a selection of the complete resource. Particularly for resources that contain other resources, such as various packaging formats, this also allows decomposition of the selection mechanisms when the components do not have unique identifiers. This is accomplished by having selectors chained together, where each refines the results of the previous one.
Example Use Case: Zara selects a paragraph of text and then a short phrase within it to comment on. Her client records the phrase as a TextQuoteSelector that further modifies a FragmentSelector used to identify the paragraph that the phrase is part of.
Term | Type | Description |
---|---|---|
refinedBy | Relationship | The relationship between a broader selector and the more specific selector that should be applied to the results of the first.
A Selector MAY be refinedBy 1 or more other Selectors. If more than 1 is given, then they are considered to be alternatives that will result in the same selection. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno28", "type": "Annotation", "body": "http://example.org/comment1", "target": { "source": "http://example.org/page1", "selector": { "type": "FragmentSelector", "value": "para5", "refinedBy": { "type": "TextQuoteSelector", "exact": "Selected Text", "prefix": "text before the ", "suffix": " and text after it" } } } }
A State describes the intended state of a resource as applied to the particular Annotation, and thus provides the information needed to retrieve the correct representation of that resource. Web resources change over time, and a State might be used to describe how to recover the intended previous version. Web resources also have multiple formats, and a State might equally be used to describe how to retrieve that particular format.
Example Use Case: Alexandra makes a comment about a web page that changes frequently. Her client records information to allow other clients to hopefully reconstruct the original target of the annotation.
Term | Type | Description |
---|---|---|
state | Relationship | The relationship between the SpecificResource and the State.
There MAY be 0 or more state relationships for each SpecificResource. |
States MUST be processed before processing Selector or Style information.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno29", "type": "Annotation", "body": "http://example.org/note1", "target": { "source": "http://example.org/page1", "state": { "id": "http://example.org/state1" } } }
A Time State resource records the time at which the resource is appropriate for the Annotation, typically the time that the Annotation was created and/or a link to a persistent copy of the current version. The timestamp for the resource could be resolved via the Memento protocol, described in RFC 7089 [rfc7089].
Example Use Case: Britney makes a note about the current state of the front page of a news website, and flags that the page is likely to change often. Her client adds in a State with the current time to describe the version of the page being annotated.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the State. Time States MUST have a type and the value MUST be TimeState . |
TimeState | Class | A description of how to retrieve a representation
of the Source resource that is temporally appropriate for the Annotation.
The State MUST have this class associated with it. |
sourceDate | Property | The timestamp at which the Source resource should be interpreted for the Annotation.
There MAY be 0 or more sourceDate properties per TimeState. If there is more than 1, each gives an alternative timestamp at which the Source may be interpreted. The timestamp MUST be expressed in the xsd:dateTime format, and SHOULD have a timezone specified. If sourceDate is provided, then sourceDateStart and sourceDateEnd MUST NOT be provided. |
sourceDateStart | Property | The timestamp that begins the interval over which the Source resource should be interpreted for the Annotation.
There MAY be exactly 1 sourceDateStart property per TimeState. The timestamp MUST be expressed in the xsd:dateTime format, and SHOULD have a timezone specified. If sourceDateStart is provided then sourceDateEnd MUST also be provided. |
sourceDateEnd | Property | The timestamp that ends the interval over which the Source resource should be interpreted for the Annotation.
There MAY be exactly 1 sourceDateEnd property per TimeState. The timestamp MUST be expressed in the xsd:dateTime format, and SHOULD have a timezone specified. If sourceDateEnd is provided then sourceDateStart MUST also be provided. |
cached | Relationship | A link to a copy of the Source resource's
representation, appropriate for the Annotation.
There MAY be 0 or more cached relationships per TimeState. If there is more than 1, each gives an alternative copy of the representation.
|
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno30", "type": "Annotation", "body": "http://example.org/note1", "target": { "source": "http://example.org/page1", "state": { "type": "TimeState", "cached": "http://archive.example.org/copy1", "sourceDate": "2015-07-20T13:30:00Z" } } }
As there are potentially many representations that can be delivered from a resource with a single IRI, and a Specific Resource may only apply to one of them, it is important to be able to record the HTTP Request headers that need to be sent to retrieve the correct representation. The HttpRequestState resource maintains a copy of the headers to be replayed when obtaining the representation.
Example Use Case: Carla retrieves a PDF representation of a web resource that can deliver HTML, PDF or plain text and then writes a description about it. She signals that her description is only about the PDF representation. Her client then includes a State to describe how to retrieve the target representation.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the State. Request Header States MUST have a type and the value MUST be HttpReqeustState . |
HttpRequestState | Class | A description of how to retrieve an appropriate representation of the Source resource for the Annotation, based on the HTTP Request headers to send on the request.
The State MUST have this class associated with it. |
value | Property | The HTTP request headers to send as a single, complete string.
An HttpRequestState MUST have exactly 1 value property. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno31", "type": "Annotation", "body": "http://example.org/description1", "target": { "source": "http://example.org/resource1", "state": { "type": "HttpRequestState", "value": "Accept: application/pdf" } } }
Similar to the refinement of selection, it may be easier, more reliable or more accurate to specify the appropriate state of the resource as a hierarchy of atomic State resources. This is particularly appropriate for representing the combination of a State that reflects an internal transformation along with the results of a State that describes an external request. This decomposition is accomplished by having the states chained together in the same way as Selectors.
Example Use Case: Devina writes a comment about a travel e-book which has many versions available over time, and is available in different formats. She is particularly commenting on a specific version and format, so her client adds both a TimeState to capture the time and an HttpRequestState to capture the format.
Term | Type | Description |
---|---|---|
refinedBy | Relationship | The relationship between a broader State and the more specific State that should be applied to the results of the first.
Each State MAY be refinedBy 1 or more other States. If more than 1 is given, then they are considered to be alternatives that will result in the same representation. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno32", "type": "Annotation", "body": "http://example.org/comment1", "target": { "source": "http://example.org/ebook1", "state": { "type": "TimeState", "sourceDate": "2016-02-01T12:05:23Z", "refinedBy": { "type": "HttpRequestState", "value": "Accept: application/epub+zip" } } } }
The interpretation of a particular Annotation, or the Annotation's body, may rely on the rendering style of the Annotation being consistent across implementations. For Annotations on binary content such as Images or Video, the background color of the target may not be accessible to the annotation client, and the default coloring may be difficult to perceive, such as a black rectangle rendered as the target area on an image of the night sky. Rendering information is recorded using CSS stylesheets and references to classes defined in those stylesheets.
Example Use Case: Erin highlights two paragraphs in a document, and selects in her client that one should be highlighted in red and the other in yellow. She then makes a comment that the yellow part contradicts the red part. Her client records that she selected the red and yellow coloration of the targets.
Term | Type | Description |
---|---|---|
type | Relationship | The class of the Style. CSS Stylesheets MAY have a type and if included the value MUST be CssStylesheet . |
CssStylesheet | Class | A resource which describes styles for resources participating in the Annotation using CSS.
The class MAY be associated with the stylesheet resource. |
Content | Class | The class for resources embedded in the Annotation, other than Textual Bodies.
If the content of the CSS description is embedded, then the State MUST have this class associated with it as well as the CssStylesheet class.
|
stylesheet | Relationship | The relationship between an Annotation and the Style.
There MAY be 0 or 1 stylesheet relationships for each Annotation. |
styleClass | Property | The name of the class used in the CSS description that should be applied to the Specific Resource. There MAY be 0 or more styleClass properties on a Specific Resource. |
The CSS Stylesheet is associated with the Annotation itself, and the content provides the rendering hints about the Annotation's constituent resources. It MAY have its own dereferenceable IRI that provides the information, or it may be embedded within the Annotation. This is to avoid having single line stylesheets each associated with different resources, and instead to allow reference to a single IRI that governs the full set of styles for a particular implementation.
Publishing systems MUST NOT assume that they will be processed; they are only provided as hints rather than requirements.
When rendering a Specific Resource, consuming applications SHOULD check to see if it has a styleClass
property. If it does, then the application SHOULD attempt to locate the appropriate selector in the CSS document, and then apply the css-value block. If a Specific Resource has a styleClass
value, but no such class is described by a stylesheet
attached to the Annotation, then the styleClass
MUST be ignored.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno33", "type": "Annotation", "stylesheet": "http://example.org/style1", "body": "http://example.org/comment1", "target": { "source": "http://example.org/document1", "styleClass": "red" } }
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno34", "type": "Annotation", "stylesheet": { "type": ["CssStylesheet", "Content"], "value": ".red { color: red }", "format": "text/css" }, "body": "http://example.org/body1", "target": { "source": "http://example.org/target1", "styleClass": "red" } }
It may be valuable to know the software that was used to process and/or render the target resource when the annotation was created. This information can be used by later systems to potentially recreate the environment to ensure that the annotation can be more easily and more accurately reconnected with the appropriate part of the target's representation. This life cycle information is associated with the Specific Resource, as it is very likely to change between Annotations for the same target, and thus cannot be associated with the target resource directly.
Example Use Case: Felicity is using a browser based client to render a PDF of a scholarly article. Her browser uses a particular library to render the PDF as HTML. She annotates a paragraph in the view that she sees, the HTML rendering, and her client records that the library that was used for rendering in the annotation, along with her comment and the target PDF.
Term | Type | Description |
---|---|---|
renderedVia | Relationship |
The relationship between the Specific Resource that represents the target in the annotation, and the piece of software or other system that was used to render the target when the annotation was created.
There MAY be 0 or more renderedVia relationships for each Specific Resource.
|
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno35", "type": "Annotation", "body": "http://example.org/comment1", "target": { "source": "http://example.edu/article.pdf", "selector": "http://example.org/selectors/html-selector1", "renderedVia": { "id": "http://example.com/pdf-to-html-library", "type": "Software", "schema:softwareVersion": "2.5" } } }
It is sometimes important for an Annotation to capture the context in which it was made, in terms of the resources that the annotator was viewing or using at the time. This does not imply an assertion that the annotation is only valid for the image in the context of that page, it just records that the page was being viewed.
Example Use Case: Gabrielle makes a comment about an image in a particular web page to say that it is not the right organization's logo. Her client includes the page that the image is being rendered in, however the annotation is associated with the image resource itself.
Term | Type | Description |
---|---|---|
scope | Relationship | The relationship between a Specific Resource and the resource that provides the scope or context for it in this Annotation.
There MAY be 0 or more scope relationships for each Specific Resource.
|
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno36", "type": "Annotation", "body": "http://example.org/note1", "target": { "source": "http://example.org/image1", "scope": "http://example.org/page1" } }
It is often useful to be able to collect Annotations together into a list, called an Annotation Collection. This list, which is always ordered, serves as a means to refer to the Annotations that are contained within it, and to maintain any information about the Collection itself.
The Collection model is divided into two sections: the Annotation Collection that manages the identity of the list and its description, and Annotation Pages that list the Annotations which are members of the Collection.
Example Use Case: Heather works for a publishing house and has transformed the author's commentary on their steampunk novel into a set of annotations for sale. The company wishes to have them available as an add-on for customers that have already bought the novel, and also in a bundle for new sales.
As Annotation Collections might get very large, the model distinguishes between the Collection itself and sequence of component pages that in turn list the Annotations. The Collection maintains information about itself, including creation or descriptive information to aid with discovery and understanding of the Collection, and also references to at least the first Page of Annotations. By starting with the first Annotation in the first Page, and traversing the Pages to the last Annotation of the last Page, all Annotations in the Collection will have been discovered.
Annotations MAY be within multiple Collections at the same time, and the Collection MAY be created or maintained by agents other than those that create or maintain the included Annotations.
Term | Type | Description |
---|---|---|
@context | Property | The context that determines the meaning of the JSON as an Annotation Collection.
The Collection MUST have 1 or more @context values and http://www.w3.org/ns/anno.jsonld MUST be one of them. |
id | Property | The identity of the Collection. The Collection MUST have exactly 1 IRI that identifies it. |
type | Property | The type of the Collection. The Collection MUST have 1 or more types, and the AnnotationCollection class MUST be one of them. |
AnnotationCollection | Class | The class for ordered Collections of Annotations. This class MUST be associated with the Collection using type |
label | Property | A human readable label intended as the name of the Collection. Collections SHOULD have 1 or more label . |
total | Property | The total number of Annotations in the Collection. Collections SHOULD have a total , and if present it MUST be an xsd:integer greater than or equal to 0. |
first | Relationship | The first page of Annotations that are included within the Collection. A Collection that has a total number of Annotations greater than 0 MUST have a first page of Annotations. The first Page MAY be embedded within the representation of the Collection. |
last | Relationship | The last page of Annotations that are included within the Collection. A Collection SHOULD have a reference to the IRI of the last page of Annotations. |
Other properties MAY be added to the Collection to describe its use, intellectual property rights, provenance and any other features that are considered useful. These properties SHOULD come from those described in this specification if possible, but MAY come from any appropriate vocabulary.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/collection1", "type": "AnnotationCollection", "label": "Steampunk Annotations", "creator": "http://example.com/publisher", "total": 42023, "first": "http://example.org/page1", "last": "http://example.org/page42" }
An Annotation Page is part of an Annotation Collection, and has an ordered list of some or all of the Annotations that are within the Collection. Each Collection may have multiple pages, and these are traversed by following the next
and prev
links between the pages.
Term | Type | Description |
---|---|---|
@context | Property | The context that determines the meaning of the JSON as an Annotation Collection Page.
If the Page is NOT embedded within a Collection, it MUST have 1 or more @context values and http://www.w3.org/ns/anno.jsonld MUST be one of them. If it is embedded, then it SHOULD NOT have an @context property. |
id | Property | The identity of the Page. The Page MUST have exactly 1 IRI that provides its identity. |
type | Property | The type of the Page. The Page MUST have 1 or more types, and the AnnotationPage class MUST be one of them. |
AnnotationPage | Class | The class of Annotation Pages. This class MUST be associated with the Page using type . |
partOf | Relationship | The relationship between the Page and the Annotation Collection that it is part of. Each Page SHOULD have a exactly 1 partOf relationship to the Collection. Properties of the Collection MAY be included in the representation of the Page. |
items | Relationship | The list of Annotations that are the members of the Page. Each Page MUST have an array of 1 or more Annotations as the value of items . |
next | Relationship | A reference to the next Page in the sequence of pages that make up the Collection. If the current page is not the last page in the Collection, it MUST have a reference to the page that follows it. |
prev | Relationship | A reference to the previous Page in the sequence of pages that make up the Collection. If the current page is not the first page in the Collection, it SHOULD have a reference to the page that it follows. |
startIndex | Property | The relative position of the first Annotation in the items list, relative to the Annotation Collection. The first entry in the first page is considered to be entry 0.Each Page SHOULD have exactly 1 startIndex , and MUST NOT have more than 1. The value MUST be a non-negative integer. |
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/page1", "type": "AnnotationPage", "partOf": "http://example.org/collection1", "next": "http://example.org/page2", "startIndex": 0, "items": [ { "id": "http://example.org/anno1", "type": "Annotation", "body": "http://example.net/comment1", "target": "http://example.com/book/chapter1" }, { "id": "http://example.org/anno2", "type": "Annotation", "body": "http://example.net/comment2", "target": "http://example.com/book/chapter2" } ] }
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/collection1", "type": "AnnotationCollection", "label": "Two Annotations", "total": 2, "first": { "id": "http://example.org/page1", "type": "AnnotationPage", "startIndex": 0, "items": [ { "id": "http://example.org/anno1", "type": "Annotation", "body": "http://example.net/comment1", "target": "http://example.com/book/chapter1" }, { "id": "http://example.org/anno2", "type": "Annotation", "body": "http://example.net/comment2", "target": "http://example.com/book/chapter2" } ] } }
Entirely Contrived Example Use Case: Ingeborg wants to associate a comment that she wrote in English within the annotation or a external mp3 of the same content in German by someone else, plus a tag, with a range of characters from a particular element in an XML representation of a document as it was at a certain point in time, and for it to be displayed in a particular way.
{ "@context": "http://www.w3.org/ns/anno.jsonld", "id": "http://example.org/anno37", "type": "Annotation", "motivation": "commenting", "creator": { "id": "http://example.org/user1", "type": "Person", "name": "A. Person", "nick": "user1" }, "created": "2015-10-13T13:00:00Z", "generator": { "id": "http://example.org/client1", "type": "SoftwareAgent", "name": "Code v2.1", "homepage": "http://example.org/homepage1" }, "generated": "2015-10-14T15:13:28Z", "stylesheet": { "id": "http://example.org/stylesheet1", "type": "CssStylesheet" }, "body": [ { "type": "TextualBody", "role": "tagging", "text": "love" }, { "type": "Choice", "members": [ { "type": "TextualBody", "role": "describing", "text": "I really love this particular bit of text in this XML. No really.", "format": "text/plain", "language": "en", "creator": "http://example.org/user1" }, { "type": "SpecificResource", "role": "describing", "source": { "id": "http://example.org/comment1", "type": "Audio", "format": "audio/mpeg", "language": "de", "creator": { "id": "http://example.org/user2", "type": "Person" } } } ] } ], "target": { "type": "SpecificResource", "styleClass": "mystyle", "source": "http://example.com/document1", "state": [ { "type": "HttpRequestState", "value": "Accept: application/xml" }, { "type": "TimeState", "sourceDate": "2015-09-25T12:00:00Z" } ], "selector": { "type": "List", "members": [ { "type": "FragmentSelector", "value": "xpointer(/doc/body/section[2]/para[1])" }, { "type": "TextPositionSelector", "start": 6, "end": 27 } ] } } }
The Web Annotation Working Group gratefully acknowledges the contributions of the Open Annotation Community Group. The output of the Community Group was fundamental to the current data model. In particular the editors would like to thank Herbert Van de Sompel of Los Alamos National Laboratory for his editorial contributions throughout the Community Group process.
The following people have been instrumental in providing thoughts, feedback, reviews, content, criticism and input in the creation of this specification:
Significant technical changes in this specification from the Open Annotation Community Group's draft are: