Skip to content

Extra Tab Contents API

YUKI "Piro" Hiroshi edited this page Feb 9, 2024 · 21 revisions

(generated by Table of Contents Generator for GitHub Wiki)

Abstract

Important Notes:

This feature is available on TST 3.9.0 and later.

TST provides ability to embed arbitrary contents inside tabs and the new tab button via its API. You can provide custom UI elements on TST's tabs - icons, buttons, thumbnails, and more.

(screenshot)

There are some example usecases made by me:

How to insert extra contents

You can set extra contents for a tab with a message with the type set-extra-contents. For example:

const TST_ID = 'treestyletab@piro.sakura.ne.jp';

function insertContents(tabId) {
  browser.runtime.sendMessage(TST_ID, {
    type:     'set-extra-contents',
    place:    'tab-front',
    tab:      tabId,
    contents: `<button id="foo-button" part="foo-button">foo</button>`
  });
}

Parameters are:

  • place: String, tye place to insert. Possible values are:
    • tab-front: Front of tab contents.
    • tab-behind: Background of tab contents.
    • tab-above: Above tab contents. (TST 4.0 and later)
    • tab-below: Below tab contents. (TST 4.0 and later)
    • tab-indent: Indent area of each tab.
    • new-tab-button: In the "New Tab" button.
    • tabbar-top: Above all tabs.
    • tabbar-bottom: Below all tabs (and the "New Tab" button), above the subpanel area.
  • tab (optional, tabs, tabId and tabIds are aliases): Integer (tab ID), array of Integer(tab ID)s, or String (a query to specify tabs).
  • window (optional, windowId is an alias): Integer, the ID of a browser window.
  • contents (optional): String, HTML source of extra contents. Dangerous contents are automatically sanitized. If you specify null contents, previous contents will be cleared.
  • style (optional): String, CSS style definitions for inserted contents.

If you set different contents to a same tab, existing contents are updated to new one. For more better performance and experience, there are some hints:

  • TST tries to apply minimum changes to existing DOM based on a diff-based DOM updater, so you can apply animation effects to contents with CSS transitions easily.
  • If you give an identifier for each element via the id attribute, it will help TST to apply changes with less steps.
    • Currently the namespace of id is shared with other helper addons which uses Tab Extra Contents API. So Firefox may report error to the browser console from duplicated ids. To avoid this concern, an alternative attribute anonid (named from anonymous-id) is available.
  • It is recommended to keep DOM structure, change only attributes and text contents for more performance.
  • On TST 4.0 and later, inserted extra contents will be cleared when the tab goes away from the viewport. You'll need to listen the tabs-rendered notification message type to restore extra contents to tabs re-appear in the viewport.
  • Contents inserted as tab-above and tab-below (available on TST 4.0 and later) will increase the height of the target tab. To consistency of height of each tab, TST automatically adjusts height of all other tabs same to the tab you most recently inserted extra contents to.

Insert extra contents to all tabs automatically

// For all existing tabs in currently shown sidebars
browser.tabs.query({}).then(tabs => {
  for (const tab of tabs) {
    insertContents(tab.id);
  }
});

// For new tabs opened after initialized
browser.tabs.onCreated.addListener(tab => {
  insertContents(tab.id);
});

// For existing tabs, after the sidebar is shown
async function registerToTST() {
  try {
    await browser.runtime.sendMessage(TST_ID, {
      type: 'register-self',
      name: browser.i18n.getMessage('extensionName'),
      icons: browser.runtime.getManifest().icons,
      listeningTypes: ['sidebar-show']
    });
  }
  catch(e) {
    // TST is not available
  }
}
registerToTST();

browser.runtime.onMessageExternal.addListener((message, sender) => {
  if (sender.id != TST_ID)
    return;

  switch (message.type) {
    case 'sidebar-show':
      browser.tabs.query({ windowId: message.windowId }).then(tabs => {
        for (const tab of tabs) {
          insertContents(tab.id);
        }
      });
      break;
  }
});

Updating specific property of extra contents dynamically

A message with the type set-extra-contents-properties allows you to update property values of DOM nodes of extra contents dynamically, without rebuilding extra contents.

function updateTooltip(tabId) {
  browser.runtime.sendMessage(TST_ID, {
    type:  'set-extra-contents-properties',
    place: 'tab-front',
    tab:   tabId,
    part:  'foo-button',
    properties: {
      title: 'Click me!',
    },
  });
}

function clearInputField() {
  browser.runtime.sendMessage(TST_ID, {
    type:  'set-extra-contents-properties',
    place: 'tabbar-bottom',
    part:  'input-field',
    properties: {
      value: '',
    },
  });
}

Parameters are:

  • place: same to set-extra-contents.
  • part: String, the part of the target element.
  • properties: Object, a hash of updated properties. Keys are treated as property names.
  • tab (optional, tabs, tabId and tabIds are aliases): Integer (tab ID), array of Integer(tab ID)s, or String (a query to specify tabs).
  • window (optional, windowId is an alias): Integer, the ID of a browser window.

Focusing to an extra contents dynamically

A message with the type focus-to-extra-contents allows you to focus to an extra contents dynamically.

function FocusToInputField() {
  browser.runtime.sendMessage(TST_ID, {
    type:  'focus-to-extra-contents',
    place: 'tabbar-bottom',
    part:  'input-field',
  });
}

Parameters are:

  • place: same to set-extra-contents.
  • part: String, the part of the target element.
  • tab (optional, tabs, tabId and tabIds are aliases): Integer (tab ID), array of Integer(tab ID)s, or String (a query to specify tabs).
  • window (optional, windowId is an alias): Integer, the ID of a browser window.

Styling extra contents

Extra contents are inserted under a shadow root. The container element generated by TST always has a fixed part name container, and TST appends a common part name extra-contents-by-(sanitized addon id) to all inserted elements originally with their own part attribute. For example, if you set <button id="foo-button" part="foo-button">foo</button> as the contents, it will become:

<span part="extra-contents-by-(sanitized addon id) container">
  <button id="foo-button" part="foo-button extra-contents-by-(sanitized addon id)">
    foo
  </button>
</span>

Such shadow DOM elements won't be styled with regular CSS applied to the document, thus there are two methods to style extra tab contents.

Styling with CSS Shadow Parts

CSS Shadow Parts, available on Firefox 72 and later, is the recommended way. You can use ::part() pseudo element to specify elements under shadow root, for example:

async function registerToTST() {
  try {
    await browser.runtime.sendMessage(TST_ID, {
      type: 'register-self',
      ...
      style: `
        ::part(%EXTRA_CONTENTS_PART% container) {
          background: ThreeDFace;
          border: 1px solid ThreeDDarkShadow;
          color: ButtonText;
        }

        ::part(%EXTRA_CONTENTS_PART% foo-button) {
          background: transparent;
          border: none;
        }

        tab-item.active ::part(%EXTRA_CONTENTS_PART% foo-button) {
          background: InactiveCaption;
          color: InactiveCaptionText;
        }

        tab-item.active ::part(%EXTRA_CONTENTS_PART% foo-button):hover {
          background: ActiveCaption;
          color: CaptionText;
        }
      `
    });
  }
  catch(e) {
  }
}

A placeholder %EXTRA_CONTENTS_PART% is available: it will be replaced to the common part name (extra-contents-by-(sanitized addon id)) automatically.

Styling with <style> element

On old versions of Firefox without CSS Shadow Parts support, you need to use a different way: style parameter for each contents. For exmaple:

browser.runtime.sendMessage(TST_ID, {
  type:  'set-extra-contents',
  ...
  style: `
    [part~="%EXTRA_CONTENTS_PART%"][part~="container"] {
      background: ThreeDFace;
      border: 1px solid ThreeDDarkShadow;
      color: ButtonText;
    }

    [part~="%EXTRA_CONTENTS_PART%"][part~="foo-button"] {
      background: transparent;
      border: none;
    }
  `
});

It will generates custom <style> element for each shadow root, so it may decrease system performance.

Styling with custom user styles

With specifying the calculated common part name with the format extra-contents-by-(sanitized addon id), you can customize styling of shadow DOM contents. Sanitizing is done with a rule: replacing all unsafe characters except alphabets, numbers, hyphen and underscore with _. For example, if the id is tst-active-tab-in-collapsed-tree@piro.sakura.ne.jp, unsafe characters @ and . are replaced then the result becomes extra-contents-by-tst-active-tab-in-collapsed-tree_piro_sakura_ne_jp.

::part(extra-contents-by-tst-active-tab-in-collapsed-tree_piro_sakura_ne_jp tab-container) {
  left: 2em;
}

Handle events on extra contents

Notification messages with following types are are available to listen events on extra contents:

  • Mouse events
    • extra-contents-clicked
    • extra-contents-dblclicked
    • extra-contents-mousedown
    • extra-contents-mouseup
  • Keyboard events
    • extra-contents-keydown
    • extra-contents-keyup
  • UI events
    • extra-contents-input
    • extra-contents-change
    • extra-contents-focus
    • extra-contents-blur
  • Text composition events
    • extra-contents-compositionstart
    • extra-contents-compositionupdate
    • extra-contents-compositionend

They are similar to general event-like notifications (tab-mousedown, tab-mouseup, tab-clicked and tab-dblclicked) but they have some extra properties. Here is the full list of properties:

  • targetType: String, the type of the owner element. Possible values are: tab, newtabbutton, selector (TST's menu UI), tabbar-top, tabbar-bottom, blank (the blank area of the tab bar) or outside (non-TST-native UI, ex. confirmation dialog).
  • originalTarget: String, the source of the actual element which the event fired on.
  • originalTargetPart: String, the value of the part attribute of the closest element which the event fired on. (TST 4.0 and later)
  • windowId (and window): Integer, the ID of the browser window.
  • tab: A tree item corresponding to the operated tab. This will be blank if the event is fied on non-tab area.
  • fieldValue: Corresponding to the value of the input field (input, textarea or select) which the event is fired on. This won't be provided if the event is not fired on any input field.
  • fieldChecked: Corresponding to the checked of the input element which the event is fired on. This won't be provided if the event is not fired on any input field.
  • Properties corresponding to the source event:
    • altKey
    • ctrlKey
    • metaKey
    • shiftKey
    • button (mouse events)
    • key (keyboard events)
    • isComposing (keyboard events and extra-contents-input)
    • locale (keyboard events and composition events)
    • location (keyboard events)
    • repeat (keyboard events)
    • data (composition events and extra-contents-input)
    • inputType (extra-contents-input)

You can cancel TST's built-in reaction for the event, with returning a boolean value true by the listener. Here is an example to listen events:

async function registerToTST() {
  try {
    await browser.runtime.sendMessage(TST_ID, {
      type: 'register-self',
      name: browser.i18n.getMessage('extensionName'),
      icons: browser.runtime.getManifest().icons,
      listeningTypes: [..., 'extra-contents-mousedown', 'extra-contents-dblclicked']
    });
  }
  catch(e) {
    // TST is not available
  }
}
registerToTST();

browser.runtime.onMessageExternal.addListener((message, sender) => {
  if (sender.id != TST_ID)
    return;

  switch (message.type) {
    ...
    case 'extra-contents-mousedown':
    case 'extra-contents-dbclicked':
      if (message.originalTarget) {
        console.log(message.originalTarget); // => '<button part="foo-button">foo</button>'
        console.log(message.originalTargetPart); // => "foo-button"
        return Promise.resolve(true); // cancel default event handling of TST
      }
      break;
  }
});

Provide drag data

You can make inserted extra contents draggable, with draggable and data-drag-data attributes.

Simple case: just single drag data

Here is an example to provide a plain text data as the drag data:

<span draggable="true"
      data-drag-data='{ "type":          "text/plain",
                        "data":          "http://example.com/",
                        "effectAllowed": "copyMove" }'>
  Foo
</span>

The value of the data-drag-data attribute should be a JSON string. It should have following properties:

  • type: String, the flavor type of the drag data.
  • data: String, the data to be dragged. Currently just a single string is supported.
  • effectAllowed (optional): String to be set for the dataTransfer.effectAllowed of the drag event, copy by default.

You can give multiple types data as an array. For example:

<span draggable="true"
      data-drag-data='[
        { "type": "text/x-moz-url",
          "data": "http://example.com/\nExample Link" }
        { "type": "text/plain",
          "data": "http://example.com/" }
      ]'>
  Foo
</span>

There is a special type tab, providing draggable data for TST's tab (tree node). tab-type drag data only has type and data properties, and the data is an object with some more properties. Here is an example:

<span part="tab"
      draggable="true"
      data-drag-data='{ "type": "tab",
                        "data": { "id": 10 } }'>
  Foo
</span>

Here is the list of available attributes of the data object:

  • id: Integer, the ID of the tab.
  • asTree (optional): Boolean, false by default. true means that descendant tabs are dragged together, false means that only an individual tab is dragged.
  • allowDetach (optional): Boolean, false by default. true means that dragged tabs are detached from the window when they are dropped outside the sidebar area.
  • allowLink (optional): Boolean, false by default. true means that links or bookmarks are created from dragged tabs when they are dropped outside the sidebar area (bookmarks toolbar, text input field, or others).

Complex case: different drag data for each action

You can provide different drag data for actions with specific modifier keys. For example:

<span draggable="true"
      data-drag-data='{
        "default":    [{ "type": "text/x-moz-url",
                         "data": "http://example.com/\nExample Link" },
                       { "type": "text/plain",
                         "data": "http://example.com/" }],
        "Shift":      { "type": "text/plain",
                        "data": "http://example.com/shifted" },
        "Ctrl+Shift": { "type": "text/plain",
                        "data": "http://example.com/new" }
      }'>
  Foo
</span>

Override context menu

You can override the context menu on an extra contents.

  • If you define an element with data-tab-id and the value is a valid tab ID, a tab context menu will be shown on the element. For example: <button data-tab-id="10">Reload</button>
  • If you define an element with data-bookmark-id and the value is a valid bookmark ID, a bookmark context menu will be shown on the element. For example: <button data-bookmark-id="aabbccdd">Bookmark</button>

How to clear extra contents

You can clear your extra contents from a tab or other places at arbitrary timing, with a message with the type clear-extra-contents. For example:

function clearContents(tabId) {
  browser.runtime.sendMessage(TST_ID, {
    type:  'clear-extra-contents',
    place: 'tab-front',
    tab:   tabId
  });
}

Parameters are:

  • place: same to set-extra-contents.
  • tab (optional, tabs, tabId and tabIds are aliases): Integer (tab ID), array of Integer(tab ID)s, or String (a query to specify tabs).
  • window (optional, windowId is an alias): Integer, the ID of a browser window.

Clear all extra contents from all places

You can clear all your extra contents from all tabs and other places, with a message with the type clear-all-extra-contents. For example:

browser.runtime.sendMessage(TST_ID, {
  type: 'clear-all-extra-contents'
});
Clone this wiki locally