Clipboard API and events

Editor’s Draft,

More details about this document
This version:
https://w3c.github.io/clipboard-apis/
Latest published version:
https://www.w3.org/TR/clipboard-apis/
Previous Versions:
Feedback:
GitHub
Inline In Spec
Editors:
(Google)
(Microsoft)
Former Editors:
Hallvord R. M. Steen (Mozilla)
(Microsoft)
Explainer:
Async Clipboard API Explainer

Abstract

This document describes APIs for accessing data on the system clipboard. It provides operations for overriding the default clipboard actions (cut, copy and paste), and for directly accessing the clipboard contents.

Status of this document

This section describes the status of this document at the time of its publication. A list of current W3C publications and the latest revision of this technical report can be found in the W3C technical reports index at https://www.w3.org/TR/.

This document was published by the Web Editing Working Group as an Editors Draft. This document is intended to become a W3C Recommendation.

Changes to this document may be tracked at https://github.com/w3c/clipboard-apis.

This document was published by the Web Editing Working Group as a Working Draft. Feedback and comments on this specification are welcome. Please use GitHub issues Historical discussions can be found in the public-editing-tf@w3.org archives.

Publication as an Editors Draft does not imply endorsement by W3C and its Members. 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 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 03 November 2023 W3C Process Document.

1. Introduction

This section is non-normative.

This specification defines how the system clipboard is exposed to web applications.

There are two general APIs described in this specification:

2. Use Cases

This section is non-normative.

2.1. Changing Default Clipboard Operations

There are many scenarios where it is desireable to change the default clipboard operations (cut/copy/paste). Here are a few examples:

2.2. Remote Clipboard Synchronization

For web applications that communicate with remote devices (e.g., remote access or remote shell applications), there is often a need for the clipboard data to be kept in sync between the two devices.

One important aspect of this use case is that it requires access to the clipboard in the absense of a user gesture or interaction.

To copy the clipboard data from a remote device to the local clipboard, a web application would take the remote clipboard data and then write() the data to the local clipboard.
To copy the local clipboard data to a remote device, a web application would listen for clipboardchange events, read() from the clipboard whenever it is updated, and then send the new clipboard data to the remote device.

2.3. Trigger Clipboard Actions

Applications that provide an alternate interface to a user agent sometimes need to be able to trigger clipboard actions in the user agent.

As an example, consider a screen reader application that provides a more accessible interface to a standard web browser. While the reader can display content and allow the user to interact with it, actions like clipboard copy need to occur in the underlying browser to ensure that the clipboard content is set correctly (with any metadata added by the browser during copy).

3. Terminology

The term editable context means any element that is either an editing host, a textarea element, or an input element with its type attribute set to any of "text", "search", "tel", "url", "email", "password" or "number".

4. Model

The platform provides a system clipboard.

The system clipboard has a list of system clipboard items that are collectively called the system clipboard data.

Each system clipboard item has a list of system clipboard representations.

Each system clipboard representation has a name, which is a string, and data, which is a sequence of bytes.

5. Clipboard Events

5.1. Clipboard event interfaces

The ClipboardEvent interface extends the Event interface.

dictionary ClipboardEventInit : EventInit {
  DataTransfer? clipboardData = null;
};
clipboardData

A DataTransfer object to hold data and meta data related to the event.

[Exposed=Window]
interface ClipboardEvent : Event {
  constructor(DOMString type, optional ClipboardEventInit eventInitDict = {});
  readonly attribute DataTransfer? clipboardData;
};
clipboardData

The clipboardData attribute is an instance of the DataTransfer interface which lets a script read and manipulate values on the system clipboard during user-initiated copy, cut and paste operations. The associated drag data store is a live but filtered view of the system clipboard, exposing mandatory data types the implementation knows the script can safely access. For synthetic events, the drag data store contains the data added by the script that created the event.

The clipboardData object’s items and files properties enable processing of multi-part or non-textual data from the clipboard.

The interface can be used to construct events. An example is given below:

var pasteEvent = new ClipboardEvent('paste');
pasteEvent.clipboardData.items.add('My string', 'text/plain');
document.dispatchEvent(pasteEvent);

Note: Synthetic clipboard events will not actually modify the clipboard or the document. In other words, while the script above will fire a paste event, the data will not be pasted into the document.

5.2. Clipboard events

5.2.1. The clipboardchange event

The clipboardchange event fires whenever the contents of the system clipboard are changed. These changes could be due to any of the following (non-exhaustive):

If the clipboard contents are changed outside the user agent, then the clipboardchange event MUST fire when the user agent regains focus.

Since synthetic cut and copy events do not update the system clipboard, they will not trigger a "clipboardchange" event.

5.2.2. The copy event

When the user initiates a copy action, the user agent fires a clipboard event named copy.

If the event is not canceled, the currently selected data will be copied to the system clipboard. The current document selection is not affected.

The copy event bubbles, is cancelable, and is composed.

See § 8.1 The copy action for a detailed description of the processing model for this event.

A synthetic copy event can be manually constructed and dispatched, but it will not affect the contents of the system clipboard.

5.2.3. The cut event

When the user initiates a cut action, the user agent fires a clipboard event named cut.

In an editable context, if the event is not canceled the action will place the currently selected data on the system clipboard and remove the selection from the document. The cut event fires before the selected data is removed. When the cut operation is completed, the selection is collapsed.

In a non-editable context, the clipboardData will be an empty list. Note that the cut event will still be fired in this case.

The cut event bubbles, is cancelable, and is composed.

See § 8.2 The cut action for a detailed description of the processing model for this event.

A synthetic cut event can be manually constructed and dispatched, but it will not affect the contents of the document or of the system clipboard.

5.2.4. The paste event

When a user initiates a paste action, the user agent fires a clipboard event named paste. The event fires before any clipboard data is inserted into the document.

If the cursor is in an editable context, the paste action will insert clipboard data in the most suitable format (if any) supported for the given context.

The paste action has no effect in a non-editable context, but the paste event fires regardless.

The paste event bubbles, is cancelable, and is composed.

See § 8.3 The paste action for a detailed description of the processing model for this event.

A synthetic paste event can be manually constructed and dispatched, but it will not affect the contents of the document.

5.3. Integration with other scripts and events

5.3.1. Event handlers that are allowed to modify the clipboard

Event handlers may write to the clipboard if any of the following is true:

The implementation may allow other trusted event types to modify the clipboard if the implementation authors believe that those event types are likely to express user intention. The implementation may also support configuration that trusts specific sites or apps to modify the clipboard regardless of the origin of the scripting thread.

Synthetic cut and copy events must not modify data on the system clipboard.

5.3.2. Event handlers that are allowed to read from clipboard

Event handlers may read data from the system clipboard if either of the following is true

Synthetic paste events must not give a script access to data on the real system clipboard.

5.3.3. Integration with rich text editing APIs

If an implementation supports ways to execute clipboard commands through scripting, for example by calling the document.execCommand() method with the commands "cut", "copy" and "paste", the implementation must trigger the corresponding action, which again will dispatch the associated clipboard event.

These are the steps to follow when triggering copy, cut or paste actions through a scripting API:

  1. Execute the corresponding action synchronously.

  2. Use the action’s return value as the return value for the API call.

Note: Copy and cut commands triggered through a scripting API will only affect the contents of the real clipboard if the event is dispatched from an event that is trusted and triggered by the user, or if the implementation is configured to allow this. Paste commands triggered through a scripting API will only fire paste events and give access to clipboard contents if the implementation is configured to allow this. How implementations can be configured to allow read or write access to the clipboard is outside the scope of this specification.

5.3.4. Interaction with other events

If the clipboard operation is triggered by keyboard input, the implementation must fire the corresponding event that initiates the clipboard operation. The event is asynchronous but must be dispatched before keyup events for the relevant keys.

The cut and paste actions may cause the implementation to dispatch other supported events, such as textInput, input, change, validation events, DOMCharacterDataModified and DOMNodeRemoved / DOMNodeInserted. Any such events are queued up to fire after processing of the cut/paste event is finished.

The implementation must not dispatch other input-related events like textInput, input, change, and validation events in response to the copy operation.

5.3.5. Event listeners that modify selection or focus

If the event listener modifies the selection or focusable area, the clipboard action must be completed on the modified selection.

6. Clipboard Event API

The Clipboard Event API allows you to override the default cut, copy and paste behavior of the user agent.

Access to the clipboard is performed using the standard DataTransfer methods to mutate the items on a ClipboardEvent's clipboardData attribute. One consequence of this is that these clipboard APIs can only access clipboard data in the context of a ClipboardEvent handler.

Note: If you need to access the clipboard outside of a clipboard event handler, see § 7 Asynchronous Clipboard API.

Note: The Clipboard Event APIs are synchronous, so they are limited in what they can do. Actions which are potentially blocking (like asking for permission or transcoding a image) are not supported by these APIs. See § 7 Asynchronous Clipboard API for a more powerful API that can support blocking or other time-consuming actions.

6.1. Overriding the copy event

To override the default copy event behavior, a copy event handler must be added and this event handler must call preventDefault() to cancel the event.

Canceling the event is required in order for the system clipboard to be updated with the data in clipboardData. If the ClipboardEvent is not canceled, then the data from the current document selection will be copied instead.

// Overwrite what is being copied to the clipboard.
document.addEventListener('copy', function(e) {
  // e.clipboardData is initially empty, but we can set it to the
  // data that we want copied onto the clipboard.
  e.clipboardData.setData('text/plain', 'Hello, world!');
  e.clipboardData.setData('text/html', '<b>Hello, world!</b>');

  // This is necessary to prevent the current document selection from
  // being written to the clipboard.
  e.preventDefault();
});

6.2. Overriding the cut event

To override the default cut event behavior, a cut event handler must be added and this event handler must call preventDefault() to cancel the event.

Canceling the event is required in order for the system clipboard to be updated with the data in clipboardData. If the ClipboardEvent is not canceled, then the data from the current document selection will be copied instead.

Note that canceling the cut event will also prevent the document from being updated (i.e., the current selection will not be removed). The event handler will need to manually update the document to remove the currently selected text.

// Overwrite what is copied to the clipboard.
document.addEventListener('cut', function(e) {
  // e.clipboardData is initially empty, but we can set it to the
  // data that we want copied onto the clipboard as part of the cut.
  // Write the data that we want copied onto the clipboard.
  e.clipboardData.setData('text/plain', 'Hello, world!');
  e.clipboardData.setData('text/html', '<b>Hello, world!</b>');

  // Since we will be canceling the cut operation, we need to manually
  // update the document to remove the currently selected text.
  deleteCurrentDocumentSelection();

  // This is necessary to prevent the document selection from being
  // written to the clipboard.
  e.preventDefault();
});

6.3. Overriding the paste event

To override the default paste event behavior, a paste event handler must be added and this event handler must call preventDefault() to cancel the event.

Canceling the event is required so that the user agent does not update the document with data from the system clipboard.

Note that canceling the paste event will also prevent the document from being updated (i.e., nothing will be pasted into the document). The event handler will need to manually paste the data into the document.

Also note that, when pasting, the drag data store mode flag is read-only, hence calling setData() from a paste event handler will not modify the data that is inserted, and not modify the data on the clipboard.

// Overwrite what is being pasted onto the clipboard.
document.addEventListener('paste', function(e) {
  // e.clipboardData contains the data that is about to be pasted.
  if (e.clipboardData.types.indexOf('text/html') > -1) {
    var oldData = e.clipboardData.getData('text/html');
    var newData = '<b>Ha Ha!</b> ' + oldData;

    // Since we are canceling the paste operation, we need to manually
    // paste the data into the document.
    pasteClipboardData(newData);

    // This is necessary to prevent the default paste action.
    e.preventDefault();
  }
});

6.4. Mandatory data types

The implementation must recognize the native OS clipboard format description for the following data types, to be able to populate the DataTransferItemList and ClipboardItem with the correct description for paste events, and set the correct data format on the OS clipboard in response to copy and cut events.

6.4.1. Reading from the clipboard

These data types must be exposed by paste events if a corresponding native type exists on the clipboard:

6.4.2. Writing to the clipboard

These data types must be placed on the clipboard with a corresponding native type description if added to a DataTransfer object during copy and cut events.

Warning! The data types that untrusted scripts are allowed to write to the clipboard are limited as a security precaution. Untrusted scripts can attempt to exploit security vulnerabilities in local software by placing data known to trigger those vulnerabilities on the clipboard.

6.5. Optional data types

The implementation MAY recognize the native OS clipboard format description for the following data types, to be able to populate the ClipboardItem with the correct description for paste events, and set the correct data format on the OS clipboard in response to copy and cut events.

These data types MAY be exposed by UAs if a corresponding native type exists on the clipboard:

6.6. Unsanitized data types

This section is non-normative.

These data types MUST NOT be sanitized by UAs:

These data types MAY NOT be sanitized by UAs:

Optional unsanitized data types are mime types specified by the web authors that MAY NOT be sanitized by the user agent. The valid optional unsanitized data types are listed below:

optional unsanitized data types may not be supported by a user agent due to their privacy requirements.

7. Asynchronous Clipboard API

partial interface Navigator {
  [SecureContext, SameObject] readonly attribute Clipboard clipboard;
};

7.2. ClipboardItem Interface

typedef Promise<(DOMString or Blob)> ClipboardItemData;

[SecureContext, Exposed=Window]
interface ClipboardItem {
  constructor(record<DOMString, ClipboardItemData> items,
              optional ClipboardItemOptions options = {});

  readonly attribute PresentationStyle presentationStyle;
  readonly attribute FrozenArray<DOMString> types;

  Promise<Blob> getType(DOMString type);

  static boolean supports(DOMString type);
};

enum PresentationStyle { "unspecified", "inline", "attachment" };

dictionary ClipboardItemOptions {
  PresentationStyle presentationStyle = "unspecified";
};
clipboardItem = new ClipboardItem([items, options])
Creates a new ClipboardItem object. items denote list of representations, each representation has a mime type and a Promise to Blob or DOMString corresponding to the mime type, options can be used to fill its ClipboardItemOptions, as per the example below.
const format1 = 'text/plain';
const promise_text_blob = Promise.resolve(new Blob(['hello'], {type: format1}));
const clipboardItemInput = new ClipboardItem(
  {[format1]: promise_text_blob},
  {presentationStyle: "unspecified"});
clipboardItem.getType(type)
Returns a Promise to the Blob corresponding to the mime type type.
clipboardItem.types
Returns the list of mime types contained in the clipboard item object.
ClipboardItem.supports(type)
Returns true if type is in mandatory data types or optional data types, else, returns false.

A clipboard item is conceptually data that the user has expressed a desire to make shareable by invoking a "cut" or "copy" command. A clipboard item serves two purposes. First, it allows a website to read data copied by a user to the system clipboard. Second, it allows a website to write data to the system clipboard.

For example, if a user copies a range of cells from a spreadsheet of a native application, it will result in one clipboard item. If a user copies a set of files from their desktop, that list of files will be represented by multiple clipboard items.

Some platforms may support having more than one clipboard item at a time on the clipboard, while other platforms replace the previous clipboard item with the new one.

A clipboard item has a list of representations, each representation with an associated mime type (a MIME type), an isCustom flag, initially false, that indicates if this representation should be treated as a web custom format (as opposed to a well-known format of the system clipboard), and data (a ClipboardItemData).

A web custom format has isCustom set to true.

In the example where the user copies a range of cells from a spreadsheet, it may be represented as an image (image/png), an HTML table (text/html), or plain text (text/plain), or a web custom format (web text/csv).

Each of these MIME types describe a different representation of the same clipboard item at different levels of fidelity and make the clipboard item more consumable by target applications during paste.

Making the range of cells available as an image will allow the user to paste the cells into a photo editing app, while the text/plain format can be used by text editor apps.

A clipboard item has a presentation style (a PresentationStyle). It helps distinguish whether apps "pasting" a clipboard item should insert the contents of an appropriate representation inline at the point of paste or if it should be treated as an attachment.

Web apps that support pasting only a single clipboard item should use the first clipboard item.

write() chooses the last clipboard item.

Web apps that support pasting more than one clipboard item could, for example, provide a user interface that previews the contents of each clipboard item and allow the user to choose which one to paste. Further, apps are expected to enumerate the MIME types of the clipboard item they are pasting and select the one best-suited for the app according to some app-specific algorithm. Alternatively, an app can present the user with options on how to paste a clipboard item, e.g. "paste as image" or "paste formatted text", etc.

A ClipboardItem object has an associated clipboard item, which is a clipboard item.

A ClipboardItem object has an associated types array, which is a FrozenArray<DOMString>.

To create a ClipboardItem object, given a clipboard item clipboardItem and a Realm realm, run these steps:

  1. Let clipboardItemObject be a new ClipboardItem with realm.

  2. Set clipboardItemObject’s clipboard item to clipboardItem.

The new ClipboardItem(items, options) constructor steps are:

  1. If items is empty, then throw a TypeError.

  2. If options is empty, then set options["presentationStyle"] = "unspecified".

  3. Set this's clipboard item to a new clipboard item.

  4. Set this's clipboard item's presentation style to options["presentationStyle"].

  5. Let types be a list of DOMString.

  6. For each (key, value) in items:

    1. Let representation be a new representation.

    2. Let isCustom be false.

    3. If key starts with `"web "` prefix, then

      1. Remove `"web "` prefix and assign the remaining string to key.

      2. Set isCustom true

    4. Set representation’s isCustom flag to isCustom.

    5. Let mimeType be the result of parsing a MIME type given key.

    6. If mimeType is failure, then throw a TypeError.

    7. If this's clipboard item's list of representations contains a representation whose MIME type is mimeType and whose [representation/isCustom] is isCustom, then throw a TypeError.

    The step above prevents collision between mime-types well-known by the user agent and those that are intended by the author to be treated as a custom type. For example, it would be possible for the author’s list of items to contain representations for "text/html" as well as "web text/html".

    1. Set representation’s MIME type to mimeType.

    2. Set representation’s data to value.

    3. Append representation to this's clipboard item's list of representations.

    4. Let mimeTypeString be the result of serializing a MIME type with mimeType.

    5. If isCustom is true, prefix mimeTypeString with `"web "`.

    6. Add mimeTypeString to types.

  7. Set this's types array to the result of running create a frozen array from types.

7.2.1. presentationStyle

The presentationStyle getter steps are to return this's clipboard item's presentation style.

7.2.2. types

The types getter steps are to return this's types array.

7.2.3. getType(type)

This method must run the below steps:

  1. Let realm be this's relevant realm.

  2. Let isCustom be false.

  3. If type starts with `"web "` prefix, then:

    1. Remove `"web "` prefix and assign the remaining string to type.

    2. Set isCustom to true.

  4. Let mimeType be the result of parsing a MIME type given type.

  5. If mimeType is failure, then throw a TypeError.

  6. Let itemTypeList be this's clipboard item's list of representations.

  7. Let p be a new promise in realm.

  8. For each representation in itemTypeList:

    1. If representation’s MIME type is mimeType and representation’s isCustom is isCustom, then:

      1. Let representationDataPromise be the representation’s data.

      2. React to representationDataPromise:

        1. If representationDataPromise was fulfilled with value v, then:

          1. If v is a DOMString, then follow the below steps:

            1. Let dataAsBytes be the result of UTF-8 encoding v.

            2. Let blobData be a Blob created using dataAsBytes with its type set to mimeType, serialized.

            3. Resolve p with blobData.

          2. If v is a Blob, then follow the below steps:

            1. Resolve p with v.

        2. If representationDataPromise was rejected, then:

          1. Reject p with "NotFoundError" DOMException in realm.

            Web developers might be interested in the underlying rejection reason.

      3. Return p.

  9. Reject p with "NotFoundError" DOMException in realm.

  10. Return p.

7.2.4. supports(type)

This method must run the below steps:

  1. If type is in mandatory data types or optional data types, then return true.

  2. If not, then return false.

7.3. Clipboard Interface

typedef sequence<ClipboardItem> ClipboardItems;

[SecureContext, Exposed=Window]
interface Clipboard : EventTarget {
  Promise<ClipboardItems> read(optional ClipboardUnsanitizedFormats formats = {});
  Promise<DOMString> readText();
  Promise<undefined> write(ClipboardItems data);
  Promise<undefined> writeText(DOMString data);
};

dictionary ClipboardUnsanitizedFormats {
  sequence<DOMString> unsanitized;
};

Some methods of the Clipboard interface take or return multiple ClipboardItem objects. However, not all platforms support more than one clipboard item; on such platforms, the algorithms below will ignore any ClipboardItem objects beyond the first one that are passed to write(), and read() and readText() only get one clipboard item from the OS.

A clipboard items object is a sequence of clipboard items.

A web author needs to create a data which is an array of ClipboardItems in order to write content to system clipboard using the write(data) method. read() returns a Promise to clipboard items object that represents contents of system clipboard data.

unsanitized is a sequence of DOMStrings corresponding to the mime type that the author wants to be treated as optional unsanitized data types.

The unsanitized option MAY NOT be supported by the user agent. The web authors shouldn’t assume that the content for the MIME types listed in unsanitized would be unsanitized as there could be privacy modes where this option may not be allowed.

The clipboard task source is triggered in response to reading or writing of system clipboard data.

7.3.1. read(formats)

The read(formats) method must run these steps:
  1. Let realm be this's relevant realm.

  2. Let p be a new promise in realm.

  3. If formats is not empty, then:

    1. For each format in formats["unsanitized"]:

      1. If format is not in optional unsanitized data types, then reject p with format "NotAllowedError" DOMException in realm.

  4. Run the following steps in parallel:

    1. Let r be the result of running check clipboard read permission.

    2. If r is false, then:

      1. Queue a global task on the permission task source, given realm’s global object, to reject p with "NotAllowedError" DOMException in realm.

      2. Abort these steps.

    3. Let data be a copy of the system clipboard data.

    4. Let items be a sequence<clipboard item>.

    5. For each systemClipboardItem in data:

      1. Let item be a new clipboard item.

      2. For each systemClipboardRepresentation in systemClipboardItem:

        1. Let mimeType be the result of running the well-known mime type from os specific format algorithm given systemClipboardRepresentation’s name.

        2. If mimeType is null, continue this loop.

        3. Let representation be a new representation.

        4. Set representation’s MIME type to mimeType.

        5. Let isUnsanitized be false.

        6. If formats is not empty, then:

          1. For each format in formats["unsanitized"]:

            1. If format is equal to MIME type, set isUnsanitized to true.

        7. Resolve representation’s data with systemClipboardRepresentation’s data.

          It should be possible to read the data asynchronously from the system clipboard after the author calls getType, however, this set of steps implies that data will be provided at the time of read.

        8. The user agent, MAY sanitize representation’s data, unless representation’s MIME type's essence is "image/png", which should remain unsanitized to preserve meta data, or if it satisfies the below conditions:

          1. representation’s MIME type is in optional unsanitized data types list.

          2. isUnsanitized is true.

        9. Append representation to item’s list of representations.

        10. Set isUnsanitized to false.

      3. If item’s list of representations size is greater than 0, append item to items.

    6. If items has a size > 0, then:

      1. Let firstItem be items[0]

      2. Run the read web custom format algorithm given firstItem.

    7. Else:

      1. Let customItem be a new clipboard item.

      2. Run the read web custom format algorithm given customItem.

      3. If customItem’s list of representations size is greater than 0, append customItem to items.

    8. Queue a global task on the clipboard task source, given realm’s global object, to perform the below steps:

      1. Let clipboardItems be a sequence<ClipboardItem>.

      2. For each clipboard item underlyingItem of items:

        1. Let clipboardItem be the result of running the steps of create a ClipboardItem object given underlyingItem and realm.

        2. Append clipboardItem to clipboardItems.

      3. Resolve p with clipboardItems.

  5. Return p.

const items = await navigator.clipboard.read();
const textBlob = await items[0].getType("text/plain");
const text = await (new Response(textBlob)).text();

7.3.2. readText()

The readText() method must run these steps:
  1. Let realm be this's relevant realm.

  2. Let p be a new promise in realm.

  3. Run the following steps in parallel:

    1. Let r be the result of running check clipboard read permission.

    2. If r is false, then:

      1. Queue a global task on the permission task source, given realm’s global object, to reject p with "NotAllowedError" DOMException in realm.

      2. Abort these steps.

    3. Let data be a copy of the system clipboard data.

      Some OSs contain multiple clipboard (e.g. Linux, "primary", "secondary", "selection"). Define from which of those data is read.

      Add definition of sanitized copy.

    4. Queue a global task on the clipboard task source, given realm’s global object, to perform the below steps:

      1. For each systemClipboardItem in data:

        1. For each systemClipboardRepresentation in systemClipboardItem:

          1. Let mimeType be the result of running the well-known mime type from os specific format algorithm given systemClipboardRepresentation’s name.

          2. If mimeType is null, continue this loop.

          3. Let representation be a new representation.

          4. If representation’s MIME type essence is "text/plain", then:

            1. Set representation’s MIME type to mimeType.

            2. Let representationDataPromise be the representation’s data.

            3. React to representationDataPromise:

              1. If representationDataPromise was fulfilled with value v, then:

                1. If v is a DOMString, then follow the below steps:

                  1. Resolve p with v.

                  2. Return p.

                2. If v is a Blob, then follow the below steps:

                  1. Let string be the result of UTF-8 decoding v’s underlying byte sequence.

                  2. Resolve p with string.

                  3. Return p.

              2. If representationDataPromise was rejected, then:

                1. Reject p with "NotFoundError" DOMException in realm.

                2. Return p.

      2. Reject p with "NotFoundError" DOMException in realm.

      3. Return p.

navigator.clipboard.readText().then(function(data) {
  console.log("Your string: ", data);
});

7.3.3. write(data)

The write(data) method must run these steps:
  1. Let realm be this's relevant realm.

  2. Let p be a new promise in realm.

  3. Run the following steps in parallel:

    1. Let r be the result of running check clipboard write permission.

      clipboard-write was removed in https://github.com/w3c/clipboard-apis/pull/164.

    2. If r is false, then:

      1. Queue a global task on the permission task source, given realm’s global object, to reject p with "NotAllowedError" DOMException in realm.

      2. Abort these steps.

    3. Queue a global task on the clipboard task source, given realm’s global object, to perform the below steps:

      1. Let itemList and cleanItemList be an empty sequence<Blob>.

      2. Let dataList be a sequence<ClipboardItem>.

      3. If data’s size is greater than 1, and the current operating system does not support multiple native clipboard items on the system clipboard, then add data[0] to dataList, else, set dataList to data.

        when data contains multiple items and the operating system supports multiple native clipboard items, the current algorithm writes the items in sequence to the system clipboard instead of writing them collectively.

      4. For each clipboardItem in dataList:

        1. For each representation in clipboardItem’s clipboard item's list of representations:

          1. Let representationDataPromise be the representation’s data.

          2. React to representationDataPromise:

            1. If representationDataPromise was fulfilled with value v, then:

              1. If v is a DOMString, then follow the below steps:

                1. Let dataAsBytes be the result of UTF-8 encoding v.

                2. Let blobData be a Blob created using dataAsBytes with its type set to representation’s MIME type.

                3. Add blobData to itemList.

              2. If v is a Blob, then add v to itemList.

            2. If representationDataPromise was rejected, then:

              1. Reject p with "NotAllowedError" DOMException in realm.

              2. Abort these steps.

        2. For each blob in itemList:

          1. Let type be the blob’s type.

          2. If type is not in the mandatory data types or optional data types list, then reject p with "NotAllowedError" DOMException in realm and abort these steps.

          3. Let cleanItem be an optionally sanitized copy of blob.

            Add definition of sanitized copy.

          4. If sanitization was attempted and was not successfully completed, then follow the below steps:

            1. Reject p with "NotAllowedError" DOMException in realm.

            2. Abort these steps.

          5. Append cleanItem to cleanItemList.

        3. Let option be clipboardItem’s clipboard item's presentation style.

        4. Write blobs and option to the clipboard with cleanItemList and option.

      5. Resolve p.

  4. Return p.

var data = [new ClipboardItem({ "text/plain": Promise.resolve(new Blob(["Text data"], { type: "text/plain" })) })];
navigator.clipboard.write(data).then(function() {
  console.log("Copied to clipboard successfully!");
}, function() {
  console.error("Unable to write to clipboard. :-(");
});

7.3.4. writeText(data)

The writeText(data) method must run these steps:
  1. Let realm be this's relevant realm.

  2. Let p be a new promise in realm.

  3. Run the following steps in parallel:

    1. Let r be the result of running check clipboard write permission.

      clipboard-write was removed in https://github.com/w3c/clipboard-apis/pull/164.

    2. If r is false, then:

      1. Queue a global task on the permission task source, given realm’s global object, to reject p with "NotAllowedError" DOMException in realm.

      2. Abort these steps.

    3. Queue a global task on the clipboard task source, given realm’s global object, to perform the below steps:

      1. Let itemList be an empty sequence<Blob>.

      2. Let textBlob be a new Blob created with: type attribute set to "text/plain;charset=utf-8", and its underlying byte sequence set to the UTF-8 encoding of data.

        Note: On Windows replace `\n` characters with `\r\n` in data before creating textBlob.

      3. Add textBlob to itemList.

      4. Let option be set to "unspecified".

      5. Write blobs and option to the clipboard with itemList and option.

      6. Resolve p.

  4. Return p.

await navigator.clipboard.writeText("Howdy, partner!");

8. Clipboard Actions

This section defines clipboard actions and the processing model for event dispatch.

Each clipboard action has two flags called script-triggered and script-may-access-clipboard.

The script-triggered flag is set if the action runs because of a script, for example a document.execCommand() call. Future scripting APIs that interact with the clipboard should also use these actions, and the script-triggered flag must be set accordingly.

The script-may-access-clipboard flag is set as follows:

  1. If action is copy or cut and the script thread is allowed to modify the clipboard, then

    1. Set the action’s script-may-access-clipboard flag

  2. If action is paste and the script thread is allowed to read from clipboard, then

    1. Set the action’s script-may-access-clipboard flag.

8.1. The copy action

The copy action consists of the following steps:

  1. If the script-triggered flag is set, then

    1. If the script-may-access-clipboard flag is unset, then

      1. Return false from the copy action, terminate this algorithm

  2. Fire a clipboard event named copy

  3. If the event was not canceled, then

    1. Copy the selected contents, if any, to the clipboard. Implementations should create alternate text/html and text/plain clipboard formats when content in a web page is selected.

    2. Fire a clipboard event named clipboardchange

  4. Else, if the event was canceled, then

    1. Call the write content to the clipboard algorithm, passing on the DataTransferItemList list items, a clear-was-called flag and a types-to-clear list.

  5. Return true from the copy action

8.2. The cut action

The cut action consists of the following steps:

  1. If the script-triggered flag is set, then

    1. If the script-may-access-clipboard flag is unset, then

      1. Return false from the cut action, terminate this algorithm

  2. Fire a clipboard event named cut

  3. If the event was not canceled, then

    1. If there is a selection in an editable context where cutting is enabled, then

      1. Copy the selected contents, if any, to the clipboard. Implementations should create alternate text/html and text/plain clipboard formats when content in a web page is selected.

      2. Remove the contents of the selection from the document and collapse the selection.

      3. Fire a clipboard event named clipboardchange

      4. Queue tasks to fire any events that should fire due to the modification, see § 5.3 Integration with other scripts and events for details.

    2. Else, if there is no selection or the context is not editable, then

      1. Return false

  4. Else, if the event was canceled, then

    1. Call the write content to the clipboard algorithm, passing on the DataTransferItemList list items, a clear-was-called flag and a types-to-clear list.

    2. Fire a clipboard event named clipboardchange

  5. Return true from the cut action

8.3. The paste action

For the paste action, the script-may-access-clipboard flag depends on an implementation-specific permission mechanism for determining what sites or apps may read from the clipboard. When a paste action is triggered by a script, the implementation must not make clipboard contents available without the user’s permission. If the permission has not already been granted, the permission prompt must include the hostname of the document associated with the script thread.

The paste action consists of the following steps:

  1. If the script-triggered flag is set, then

    1. If script-may-access-clipboard is unset, then

      1. Return false from the paste action, terminate this algorithm

  2. Fire a clipboard event named paste

  3. If the event was not canceled, then

    1. If there is a selection or cursor in an editable context where pasting is enabled, then

      1. Insert the most suitable content found on the clipboard, if any, into the context.

      2. Queue tasks to fire any events that should fire due to the modification, see § 5.3 Integration with other scripts and events for details.

    2. Else

      1. Return false

  4. Else, if the event was canceled

    1. Return false

  5. Return true from the action

9. Permissions API Integration

The [permissions] API provides a uniform way for websites to access powerful features like the clipboard. It allows websites to request permissions from users and query which permissions they have.

For the clipboard, one permission is defined: "clipboard-write"

Note: Clipboard permissions currently only apply to the Async Clipboard API. Future versions of this specification may be updated to apply this permission to other Clipboard interactions.

These clipboard permissions are powerful features permission-related algorithms and types are defined as follows:

permission descriptor type
dictionary ClipboardPermissionDescriptor : PermissionDescriptor {
  boolean allowWithoutGesture = false;
};

There are 4 clipboard permissions:

With the following relationships:

While user agents MUST support the ClipboardPermissionDescriptor as described in this specification, they, of course, retain complete control over the default settings and how (or if) they are exposed to the user.

A user agent that wants to have separate user-settable write controls over the clipboard and always require a user gesture would handle each descriptor as follows:

9.1. Clipboard read permission

9.1.1. check clipboard read permission

  1. Let hasGesture be true if the relevant global object of this has transient activation, false otherwise.

  2. If hasGesture then,

    1. Return true if the current script is running as a result of user interaction with a "Paste" element created by the user agent or operating system.

  3. Return false.

9.2. Clipboard write permission

9.2.1. check clipboard write permission

  1. Let writeWithoutGesture be the permission state of the { name: "clipboard-write", allowWithoutGesture: true } permission.

  2. If writeWithoutGesture is granted, then return true.

  3. Let hasGesture be true if the relevant global object of this has transient activation, false otherwise.

  4. If hasGesture then,

    1. Let systemCopy be true if the current script is running as a result of user interaction with a "cut" or "copy" element created by the user agent or operating system.

    2. If systemCopy is true, then return true.

    3. Return the result of request permission to use the { name: "clipboard-write", allowWithoutGesture: false } permission.

      Note: User agents may choose to request a stronger permission that will implicitly update this permission.

  5. Return the result of request permission to use the { name: "clipboard-write", allowWithoutGesture: true } permission.

10. Security Considerations

Enabling authors to change what is copied by a user, or to make an automated copy of something that was never selected and allowing unrestricted calls to paste information can raise various security concerns.

Some example scenarios include:

10.1. Pasting HTML and multi-part data

This section is non-normative.

There are certain security risks associated with pasting formatted or multi-part data.

To determine what policies to use, the factors we consider are:

This is an overview of the scenarios and the possible security policies:

Origin of data Origin of script Rules
Originates from online source Same as data Do not sanitize HTML. Do not access any local files.
Different origin Optionally sanitize content. Do not access any local files.
Originates from local application Any Do not sanitize HTML. Grant access to local files

Some implementations mitigate the risks associated with pasting rich text by stripping potentially malicious content such as SCRIPT elements and javascript: links by default when pasting rich text, but allow a paste event handler to retrieve and process the original, un-sanitized data.

10.2. General security policies

The implementation must not download referenced online resources, or expose their contents in the files list or DataTransferItemList.

If the data on the clipboard is not from a local application, the implementation must not give access to any referenced local files. For example, if the data contains <img src="file://localhost/example.jpg"> but the data’s origin is an online resource, the implementation must not add an entry for example.jpg to the clipboardData.items list.

10.3. Transcoding images

To prevent malicious image data from being placed on the clipboard, the image data may be transcoded to produce a safe version of the image. This prevents websites from trying to exploit security vulnerabilities in other applications.

Implementations should not transcode images being read from the clipboard. Transcoding images can lose important metadata (such as the physical resolution of an image). This is also consistent with other methods in which images can be shared with a website (such as `<input type=file>`).

10.4. Nuisance considerations

Scripts may use the DataTransfer API to annoy and confuse users by altering the data on the system clipboard from copy and cut events. This specification does not attempt to prevent such nuisances, though implementations may add additional restrictions.

Implementations must handle scripts that try to place excessive amounts of data on the clipboard gracefully.

11. Privacy Considerations

Because these APIs provide access to the user’s clipboard data, there are serious privacy concerns due to the possibility that the clipboard will contain personally-identifiable information (PII) like names, addresses, or passwords.

In general, user agents must ensure that untrusted scripts do not get uncontrolled access to a user’s clipboard data through these APIs.

11.1. Privacy and the Clipboard Event API

The Clipboard Event API allows scripts running in the context of a clipboard event handler to access a copy of the system clipboard and potentially modify the data being written to the clipboard.

User agents should be aware of the following requirements with regards to securing the data accessed by the Clipboard Event API:

Even though the Clipboard Event API is not covered by the Clipboard permission, user agents may choose to provide a method for the user to disable this API or to configure which sites are allowed to access it.

11.2. Privacy and the Asychronous Clipboard API

The Asynchronous Clipboard API is a powerful feature because it can allow access to the clipboard data from any script (access is not restricted to Clipboard Event handlers) and data can be accessed in the absence of a user provided gesture.

To help prevent abuse, this API must not be available unless the script is executing in the context of a document that has focus.

11.2.1. Privacy and Clipboard Permission

The Clipboard Permission gates access to this API, but user agents get to choose the permission defaults and which of these permission settings can be set by the user. For example, a user agent may choose to only allow access to the Asynchronous Clipboard API when a user gesture is present and always deny script requests for access without a gesture.

User agents may opt to have this permission automatically expire sometime after the user grants permission, for example, by having the permission expire:

11.3. Other Privacy Concerns

If the user agent allows clipboard data to be read using document.execCommand("paste") then the user agent must ensure that the user has explicitly allowed that access to take place.

12. Acknowledgements

This section is non-normative

The editors would like to acknowledge the contributions of the former editors who helped shepherd this specification through various meetings and mailing-list discussions to bring it to its current state.

The editors would also like to acknowledge their intellectual debt to the documentation of Data Transfer functionalities from Microsoft [MICROSOFT-CLIP-OP] and earlier drafts of the [HTML5] specification. We are also grateful for the draft "safe copy and paste" from Paul Libbrecht (this draft is no longer available on the Web).

And finally, we would like to acknowledge the contributions made by the following:

Adam Barth, Shawn Carnell, Daniel Cheng, Daniel Dardailler, Domenic Denicola, Al Gilman, James Graham, James Greene, Ian Hickson, Darwin Huang, Lachlan Hunt, Philip Jägenstedt, Anne van Kesteren, Marijn Kruisselbrink, Aaron Leventhal, Jim Ley, Paul Libbrecht, "Martijn", Glenn Maynard, Chris Mills, ms2ger, Ryosuke Niwa, Robert O’Callahan, Dave Poehlman, "ROBO Design", Janina Sajka, Rich Schwerdtfeger, Jonas Sicking, Maciej Stachowiak, Mihai Sucan, Dmitry Titov, Ojan Vafai, Tarquin Wilton-Jones, Tom Wlodkowski, Bo Cupp, mbrodesser and Boris Zbarsky.

Appendix A: Algorithms

write content to the clipboard

Input

items, a DataTransferItemList list of items to write

clear-was-called, a boolean

types-to-clear, a list

Output

None

  1. If the items list is not empty, then

    1. Clear the clipboard

    2. For each part in the list,

      1. If data type is text/plain, then

        1. Ensure encoding is correct per OS and locale conventions

        2. Normalize line endings according to platform conventions

        3. Place text on clipboard with the appropriate OS clipboard format description

      2. Else, if data is of a type listed in the mandatory data types list, then

        1. Place part on clipboard with the appropriate OS clipboard format description

      3. Else

        1. This is left to the implementation...

          It’s not good to leave things up to the implementation. What should happen here?

          Note: Due to limitations in the implementation of operating system clipboards, scripts should not assume that custom formats will be available to other applications on the system. For example, there is a limit to how many custom clipboard formats can be registered in Microsoft Windows. While it is possible to use any string for setData()’s type argument, sticking to the mandatory data types is strongly recommended.

  2. Else, the items list is empty. Follow these steps to determine whether to clear the clipboard:

    1. If the list of items is empty and the clear-was-called flag is true, then

      1. If the types-to-clear list is empty, then

        1. Clear the clipboard

      2. Else

        1. Remove types in the types-to-clear list from the clipboard in an operating system and implementation-specific way

          The "remove specific types from clipboard" feature is at risk. It doesn’t seem all that important, and it’s unclear if it can be implemented easily across relevant platforms.

write blobs and option to the clipboard

Input

items, a sequence<Blob>

presentationStyle, a clipboard item's presentation style

Output

None

  1. Let webCustomFormats be a sequence<Blob>.

  2. For each item in items:

    1. Let formatString be the result of running os specific well-known format given item’s type.

    2. If formatString is empty then follow the below steps:

      1. Let webCustomFormatString be the item’s type.

      2. Let webCustomFormat be an empty type.

      3. If webCustomFormatString starts with `"web "` prefix, then remove the `"web "` prefix and store the remaining string in webMimeTypeString.

      4. Let webMimeType be the result of parsing a MIME type given webMimeTypeString.

      5. If webMimeType is failure, then abort all steps.

      6. Let webCustomFormat’s type's essence equal to webMimeType.

      7. Set item’s type to webCustomFormat.

      8. Append webCustomFormat to webCustomFormats.

    3. Let payload be the result of UTF-8 decoding item’s underlying byte sequence.

    4. Insert payload and presentationStyle into the system clipboard using formatString as the native clipboard format.

    some OSs contain multiple clipboard (e.g. Linux, "primary", "secondary", "selection"). Define to which of those data is written.

  3. Write web custom formats given webCustomFormats.

os specific well-known format

Input

mimeType, a type

Output

wellKnownFormat, a platform specific string type. On Mac it’s NSPasteboardType, on Windows it’s LPCWSTR and Linux it’s a const char*.

For Windows see https://docs.microsoft.com/en-us/windows/win32/dataxchg/standard-clipboard-formats and https://docs.microsoft.com/en-us/windows/win32/dataxchg/about-atom-tables?redirectedfrom=MSDN For Mac see https://developer.apple.com/documentation/appkit/nspasteboardtype

  1. Let wellKnownFormat be an empty string.

  2. If mimeType’s essence is "text/plain", then

    On Windows, follow the convention described below:

    1. Assign CF_UNICODETEXT to wellKnownFormat.

    On MacOS, follow the convention described below:

    1. Assign NSPasteboardTypeString to wellKnownFormat.

    On Linux, ChromeOS, and Android, follow the convention described below:

    1. Assign "text/plain" to wellKnownFormat.

  3. Else, if mimeType’s essence is "text/html", then

    On Windows, follow the convention described below:

    1. Assign CF_HTML to wellKnownFormat.

    On MacOS, follow the convention described below:

    1. Assign NSPasteboardTypeHTML to wellKnownFormat.

    On Linux, ChromeOS, and Android, follow the convention described below:

    1. Assign "text/html" to wellKnownFormat.

  4. Else, if mimeType’s essence is "image/png", then

    On Windows, follow the convention described below:

    1. Assign "PNG" to wellKnownFormat.

    On MacOS, follow the convention described below:

    1. Assign NSPasteboardTypePNG to wellKnownFormat.

    On Linux, ChromeOS, and Android, follow the convention described below:

    1. Assign "image/png" to wellKnownFormat.

  5. Else, if mimeType’s essence is "image/svg+xml", then

    On Windows, follow the convention described below:

    1. Assign CFSTR_MIME_SVG_XML to wellKnownFormat.

    On MacOS, follow the convention described below:

    1. Assign UTTypeSVG to wellKnownFormat.

    On Linux, ChromeOS, and Android, follow the convention described below:

    1. Assign "image/svg+xml" to wellKnownFormat.

  6. Return wellKnownFormat.

well-known mime type from os specific format

Input

osFormatName, a platform specific string type. On Mac it’s NSPasteboardType, on Windows it’s LPCWSTR and Linux it’s a const char*.

Output

mimeType, a MIME type

For Windows see https://docs.microsoft.com/en-us/windows/win32/dataxchg/standard-clipboard-formats and https://docs.microsoft.com/en-us/windows/win32/dataxchg/about-atom-tables?redirectedfrom=MSDN For Mac see https://developer.apple.com/documentation/appkit/nspasteboardtype

On Windows, follow the convention described below:

  1. If osFormatName "UnicodeText", then set mimeTypeString to "text/plain".

  2. Else, if osFormatName "HTML Format", then set mimeTypeString to "text/html".

  3. Else, if osFormatName "PNG", then set mimeTypeString to "image/png".

  4. Else, if osFormatName CFSTR_MIME_SVG_XML, then set mimeTypeString to "image/svg+xml".

On MacOS, follow the convention described below:

  1. If osFormatName is NSPasteboardTypeString, then set mimeTypeString to "text/plain".

  2. Else, if osFormatName is NSPasteboardTypeHTML, then set mimeTypeString to "text/html".

  3. Else, if osFormatName NSPasteboardTypePNG, then set mimeTypeString to "image/png".

  4. Else, if osFormatName UTTypeSVG, then set mimeTypeString to "image/svg+xml".

On Linux, ChromeOS, and Android, follow the convention described below:

  1. If osFormatName is "text/plain", then set mimeTypeString to "text/plain".

  2. Else, if osFormatName "text/html", then set mimeTypeString to "text/html".

  3. Else, if osFormatName "image/png", then set mimeTypeString to "image/png".

  4. Else, if osFormatName "image/svg+xml", then set mimeTypeString to "image/svg+xml".

  1. Let mimeType be the result of parsing a MIME type given mimeTypeString.

  2. Return mimeType.

read web custom format

Input

item, a clipboard item

  1. Let webCustomFormatMap be the os specific custom map name.

  2. Read webCustomFormatMap from the system clipboard.

    specify in more detail the process of reading a webCustomFormatMap.

  3. If webCustomFormatMap is empty, then return item.

  4. Let webCustomFormatMapString be the JSON string deserialized from webCustomFormatMap.

    Note: Need a JSON reader to deserialize the content from the webCustomFormatMap.

  5. For each (key, value) in webCustomFormatMapString:

    1. Let mimeType be the result of parsing a MIME type given key.

    2. If mimeType is failure, then continue the loop.

    3. Let representation be a new representation.

    4. Set representation’s MIME type to mimeType.

    5. Set representation’s isCustom flag to true.

    6. Read webCustomFormat from the system clipboard given value.

    7. If the read is successful, then set representation’s data to the resulting data from the system clipboard, else continue the loop.

    8. Append representation to item’s list of representations.

write web custom formats

Input

items, a sequence<Blob>

  1. Let idx be a number initialized to 0.

  2. Let webCustomFormatMap be the os specific custom map name.

  3. Let webCustomFormatMapString be an empty JSON string.

  4. For each item in items:

    1. Let webCustomFormat be the os specific custom name.

    2. Let webCustomFormatIdx be the result of appending idx to webCustomFormat.

    3. Insert item’s type as key and webCustomFormatIdx as value into the webCustomFormatMapString using a JSON serializer.

      Note: Need a JSON writer to serialize the content into the webCustomFormatMapString.

    4. Insert the item into the system clipboard using webCustomFormatIdx as the format.

    5. Increment idx.

    6. If idx is greater than 100, then break from this loop.

  5. Insert the webCustomFormatMapString into the system clipboard using webCustomFormatMap as the format.

os specific custom map name

Output

webCustomFormatMap, a string

On Windows, follow the convention described below:

  1. Assign "Web Custom Format Map" to webCustomFormatMap.

  2. Return webCustomFormatMap.

On MacOS, follow the convention described below:

  1. Assign "org.w3.web-custom-format.map" to webCustomFormatMap.

  2. Return webCustomFormatMap.

On Linux, ChromeOS, and Android, follow the convention described below:

  1. Assign "application/web;type=\"custom/formatmap\"" to webCustomFormatMap.

  2. Return webCustomFormatMap.

os specific custom name

Output

webCustomFormat, a string

On Windows, follow the convention described below:

  1. Assign "Web Custom Format" to webCustomFormat.

  2. Return webCustomFormat.

On MacOS, follow the convention described below:

  1. Assign "org.w3.web-custom-format.type-" to webCustomFormat.

  2. Return webCustomFormat.

On Linux, ChromeOS, and Android, follow the convention described below:

  1. Assign "application/web;type="custom/format" to webCustomFormat.

  2. Return webCustomFormat.

fire a clipboard event

Input

e, the ClipboardEvent to fire

Output

None

  1. Let clear-was-called be false

  2. Let types-to-clear be an empty list

  3. Let clipboard-event-data be a new DataTransfer object with an empty items list.

  4. Let clipboard-entry be the sequence number of the current clipboard content, or null if the OS clipboard does not support sequence numbers

  5. Let trusted be true if the event is generated by the user agent, false otherwise

  6. Let target be set as follows:

    1. If the context is editable, then

      1. Set target to be the element that contains the start of the visible selection or cursor in document order, or the body element if there is no visible selection or cursor.

    2. Else, if the context is not editable, then

      1. Set target to the focused node, or the body element if no node has focus.

  7. Process the event as follows:

    1. If e is "paste", then

      1. Set the clipboard-event-data’s internal drag data store mode flag to read-only.

      2. If trusted is true, or the implementation is configured to give script-generated events read access to the OS clipboard

        1. For each clipboard-part on the OS clipboard, carry out these steps:

          1. If clipboard-part contains plain text, then

            1. Ensure the text is in the encoding the scripting engine uses internally

            2. Let new-data be a new DataTransferItem with drag data item kind set to string and drag data item type string set to text/plain.

            3. Set new-data’s data to the plain text.

            4. Add new-data to clipboard-event-data’s items

          2. If clipboard-part represents file references, then for each file reference:

            1. Determine MIME type of referenced file

            2. Let new-data be a new DataTransferItem with drag data item kind set to file and drag data item type string set to the corresponding MIME type, or application/octet-stream if the file’s type is unknown.

            3. Set new-data’s data to the file reference data.

            4. Add new-data to clipboard-event-data’s items

          3. If clipboard-part contains HTML- or XHTML-formatted text, according to the operating system’s convention for describing such clipboard formats, then

            1. If the implementation supports pasting HTML, then the implementation must process the markup by calling process an html paste event with clipboard-part and clipboard-event-data.

          4. If clipboard-part contains data in another supported binary or text-based format (see mandatory data types), then

            1. Determine the MIME type of the data

            2. Let new-data be a new DataTransferItem with drag data item kind set to file, drag data item type string set to the corresponding MIME type

            3. Set new-data’s data to be the binary or text-based data.

            4. Add new-data to clipboard-event-data’s items

      3. Update clipboard-event-data’s files property to match entries in clipboard-event-data’s items.

      4. Update clipboard-event-data’s types property to match entries in clipboard-event-data’s items.

    2. If e is "copy" or "cut", then

      1. Set the associated DataTransfer object’s internal drag data store mode flag to read/write

  8. Set e’s clipboardData to clipboard-event-data.

  9. Set e’s isTrusted to trusted.

  10. Set e’s composed to true.

  11. Dispatch the event e which bubbles and is cancelable, and which uses the ClipboardEvent interface, at target.

    Implementation requirements for access to data during event dispatch are defined in [HTML]. Some additional clipboard event-specific processing rules are given below:

    Why here? Why not in the HTML spec?

    1. If a script calls clearData() or clear() and the DataTransfer object’s internal drag data store mode flag is read/write, then

      1. Set the clear-was-called flag to true. If an argument is given, add the argument to the types-to-clear list.

    2. If a script calls setData() or modifies items and the clear-was-called flag is true, then

      1. If the types-to-clear list is empty, then

        1. Set the clear-was-called flag to false, then

      2. Else, if setData()’s type argument or the new item’s drag data item type string is found in the types-to-clear list, then

        1. Remove it from the list. If the list is now empty, set the clear-was-called flag to false.

    3. If a script calls getData() or accesses items in the DataTransferItemList and clipboard-entry is set, then

      1. Check that the clipboard data’s sequence number matches clipboard-entry. If the clipboard no longer contains the same entry, set the DataTransferItemList object’s internal drag data store mode to protected.

    Warning! A malicious script listening to a paste event may set up a never-ending loop in order to read what the user places on the clipboard in the future. On platforms where a clipboard sequence number is not available, other limitations should be implemented.

process an HTML paste event

Input

clipboard-part, the clipboard part to process

clipboard-event-data, the DataTransfer object for this event

Output

None

  1. Let new-data be a new DataTransferItem with drag data item kind set to Plain Unicode string, drag data item type string set to text/html or application/xhtml+xml accordingly.

  2. Extract the markup from clipboard-part and use the relevant parser to construct a DOM tree.

  3. If the markup’s source URL is known, then resolve all relative URLs in HREF and SRC attributes using the source URL as base URL, and set the respective attributes to the resolved absolute URL.

  4. If the markup’s origin is from a local application, then check whether there are references to local files and/or other parts of the OS clipboard’s contents. If such references are found, references to sub-parts must be replaced by content-id references using the cid: URL scheme [RFC2392]. To do so, process each attribute referencing a local file or clipboard part according to the following steps:

    Are these steps necessary? Do we know about native (platform) clipboard implementations that support multiple parts with internal references?

    This feature is at risk because it’s unclear whether it is required, and because it’s hard to test in a cross-platform way.

    1. If clipboard-event-data’s items already contains an entry for the referenced file or clipboard part, then

      1. Set itemNumber to the index of the existing entry.

    2. Else,

      1. Let new-file-data be a new DataTransferItem with drag data item kind set to "file", and drag data item type string set to the MIME type of the file or clipboard part if known, or application/octet-stream if the file’s type is unknown.

      2. Let file-info be a new File object with name set to the name part of the HTML attribute contents, and lastModified set to the timestamp of the referenced file or 0 if the entry references a clipboard part.

      3. Set new-file-data’s internal File object to file-info.

      4. Add new-file-data to clipboard-event-data’s items and let itemNumber be the index of this entry in the DataTransferItemList.

    3. Update the DOM attribute that referenced the local file or clipboard part to contain the string 'cid:' followed by itemNumber.

  5. Serialize the processed DOM and update new-data with the resulting HTML code.

  6. Add new-data to clipboard-event-data’s items.

Conformance

Document conventions

Conformance requirements are expressed with a combination of descriptive assertions and RFC 2119 terminology. The key words “MUST”, “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”, “SHOULD NOT”, “RECOMMENDED”, “MAY”, and “OPTIONAL” in the normative parts of this document are to be interpreted as described in RFC 2119. However, for readability, these words do not appear in all uppercase letters in this specification.

All of the text of this specification is normative except sections explicitly marked as non-normative, examples, and notes. [RFC2119]

Examples in this specification are introduced with the words “for example” or are set apart from the normative text with class="example", like this:

This is an example of an informative example.

Informative notes begin with the word “Note” and are set apart from the normative text with class="note", like this:

Note, this is an informative note.

Conformant Algorithms

Requirements phrased in the imperative as part of algorithms (such as "strip any leading space characters" or "return false and abort these steps") are to be interpreted with the meaning of the key word ("must", "should", "may", etc) used in introducing the algorithm.

Conformance requirements phrased as algorithms or specific steps can be implemented in any manner, so long as the end result is equivalent. In particular, the algorithms defined in this specification are intended to be easy to understand and are not intended to be performant. Implementers are encouraged to optimize.

Index

Terms defined by this specification

Terms defined by reference

References

Normative References

[DOM]
Anne van Kesteren. DOM Standard. Living Standard. URL: https://dom.spec.whatwg.org/
[ENCODING]
Anne van Kesteren. Encoding Standard. Living Standard. URL: https://encoding.spec.whatwg.org/
[FileAPI]
Marijn Kruisselbrink. File API. URL: https://w3c.github.io/FileAPI/
[HTML]
Anne van Kesteren; et al. HTML Standard. Living Standard. URL: https://html.spec.whatwg.org/multipage/
[INFRA]
Anne van Kesteren; Domenic Denicola. Infra Standard. Living Standard. URL: https://infra.spec.whatwg.org/
[MIMESNIFF]
Gordon P. Hemsley. MIME Sniffing Standard. Living Standard. URL: https://mimesniff.spec.whatwg.org/
[PERMISSIONS]
Marcos Caceres; Mike Taylor. Permissions. URL: https://w3c.github.io/permissions/
[RFC2119]
S. Bradner. Key words for use in RFCs to Indicate Requirement Levels. March 1997. Best Current Practice. URL: https://datatracker.ietf.org/doc/html/rfc2119
[RFC2392]
E. Levinson. Content-ID and Message-ID Uniform Resource Locators. August 1998. Proposed Standard. URL: https://www.rfc-editor.org/rfc/rfc2392
[WEBIDL]
Edgar Chen; Timothy Gu. Web IDL Standard. Living Standard. URL: https://webidl.spec.whatwg.org/

Informative References

[HTML5]
Ian Hickson; et al. HTML5. URL: https://www.w3.org/html/wg/drafts/html/master/
[MICROSOFT-CLIP-OP]
About DHTML Data Transfer. Microsoft Developer Network.. URL: https://msdn.microsoft.com/en-us/ie/ms537658(v=vs.94)
[SVG11]
Erik Dahlström; et al. Scalable Vector Graphics (SVG) 1.1 (Second Edition). 16 August 2011. REC. URL: https://www.w3.org/TR/SVG11/

IDL Index

dictionary ClipboardEventInit : EventInit {
  DataTransfer? clipboardData = null;
};

[Exposed=Window]
interface ClipboardEvent : Event {
  constructor(DOMString type, optional ClipboardEventInit eventInitDict = {});
  readonly attribute DataTransfer? clipboardData;
};

partial interface Navigator {
  [SecureContext, SameObject] readonly attribute Clipboard clipboard;
};

typedef Promise<(DOMString or Blob)> ClipboardItemData;

[SecureContext, Exposed=Window]
interface ClipboardItem {
  constructor(record<DOMString, ClipboardItemData> items,
              optional ClipboardItemOptions options = {});

  readonly attribute PresentationStyle presentationStyle;
  readonly attribute FrozenArray<DOMString> types;

  Promise<Blob> getType(DOMString type);

  static boolean supports(DOMString type);
};

enum PresentationStyle { "unspecified", "inline", "attachment" };

dictionary ClipboardItemOptions {
  PresentationStyle presentationStyle = "unspecified";
};

typedef sequence<ClipboardItem> ClipboardItems;

[SecureContext, Exposed=Window]
interface Clipboard : EventTarget {
  Promise<ClipboardItems> read(optional ClipboardUnsanitizedFormats formats = {});
  Promise<DOMString> readText();
  Promise<undefined> write(ClipboardItems data);
  Promise<undefined> writeText(DOMString data);
};

dictionary ClipboardUnsanitizedFormats {
  sequence<DOMString> unsanitized;
};

dictionary ClipboardPermissionDescriptor : PermissionDescriptor {
  boolean allowWithoutGesture = false;
};

Issues Index

Web developers might be interested in the underlying rejection reason.
It should be possible to read the data asynchronously from the system clipboard after the author calls getType, however, this set of steps implies that data will be provided at the time of read.
Some OSs contain multiple clipboard (e.g. Linux, "primary", "secondary", "selection"). Define from which of those data is read.
Add definition of sanitized copy.
clipboard-write was removed in https://github.com/w3c/clipboard-apis/pull/164.
when data contains multiple items and the operating system supports multiple native clipboard items, the current algorithm writes the items in sequence to the system clipboard instead of writing them collectively.
Add definition of sanitized copy.
clipboard-write was removed in https://github.com/w3c/clipboard-apis/pull/164.
It’s not good to leave things up to the implementation. What should happen here?
The "remove specific types from clipboard" feature is at risk. It doesn’t seem all that important, and it’s unclear if it can be implemented easily across relevant platforms.
some OSs contain multiple clipboard (e.g. Linux, "primary", "secondary", "selection"). Define to which of those data is written.
specify in more detail the process of reading a webCustomFormatMap.
Why here? Why not in the HTML spec?
Are these steps necessary? Do we know about native (platform) clipboard implementations that support multiple parts with internal references?
This feature is at risk because it’s unclear whether it is required, and because it’s hard to test in a cross-platform way.
MDN

Clipboard/read

Firefox🔰 90+Safari13.1+Chrome?
Opera63+Edge?
Edge (Legacy)NoneIENone
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet12.0+Opera Mobile54+
MDN

Clipboard/readText

FirefoxNoneSafari13.1+Chrome66+
Opera?Edge79+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?
MDN

Clipboard/write

In all current engines.

Firefox87+Safari13.1+Chrome66+
Opera63+Edge79+
Edge (Legacy)NoneIENone
Firefox for Android?iOS Safari?Chrome for Android66+Android WebView?Samsung Internet12.0+Opera Mobile54+
MDN

Clipboard/writeText

In all current engines.

Firefox63+Safari13.1+Chrome66+
Opera?Edge79+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?
MDN

Clipboard

In all current engines.

Firefox63+Safari13.1+Chrome66+
Opera?Edge79+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardEvent/ClipboardEvent

In all current engines.

Firefox22+Safari10.1+Chrome58+
Opera?Edge79+
Edge (Legacy)17+IENone
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardEvent/clipboardData

In all current engines.

Firefox22+Safari10.1+Chrome41+
Opera?Edge79+
Edge (Legacy)12+IE5+
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardEvent

In all current engines.

Firefox22+Safari10.1+Chrome41+
Opera?Edge79+
Edge (Legacy)12+IE4+
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardItem/ClipboardItem

In all current engines.

Firefox🔰 87+Safari13.1+Chrome98+
Opera?Edge98+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android98+Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardItem/getType

In all current engines.

Firefox🔰 87+Safari13.1+Chrome76+
Opera?Edge79+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android84+Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardItem/presentationStyle

Firefox🔰 87+Safari13.1+ChromeNone
Opera?EdgeNone
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardItem/types

In all current engines.

Firefox🔰 87+Safari13.1+Chrome76+
Opera?Edge79+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android84+Android WebView?Samsung Internet?Opera Mobile?
MDN

ClipboardItem

In all current engines.

Firefox🔰 87+Safari13.1+Chrome76+
Opera?Edge79+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android84+Android WebView?Samsung Internet?Opera Mobile?
MDN

Element/copy_event

In all current engines.

Firefox22+Safari3+Chrome1+
Opera12.1+Edge79+
Edge (Legacy)12+IE9+
Firefox for Android?iOS Safari3+Chrome for Android?Android WebView?Samsung Internet?Opera Mobile12.1+
MDN

Element/cut_event

In all current engines.

Firefox22+Safari3+Chrome1+
Opera12.1+Edge79+
Edge (Legacy)12+IE9+
Firefox for Android?iOS Safari3+Chrome for Android?Android WebView?Samsung Internet?Opera Mobile12.1+
MDN

Element/paste_event

In all current engines.

Firefox22+Safari3+Chrome1+
Opera12.1+Edge79+
Edge (Legacy)12+IE9+
Firefox for Android?iOS Safari3+Chrome for Android?Android WebView37+Samsung Internet?Opera Mobile12.1+
MDN

Navigator/clipboard

In all current engines.

Firefox63+Safari13.1+Chrome66+
Opera?Edge79+
Edge (Legacy)?IENone
Firefox for Android?iOS Safari?Chrome for Android?Android WebView?Samsung Internet?Opera Mobile?