-
Notifications
You must be signed in to change notification settings - Fork 281
Extra Tab Contents API
- Abstract
- How to insert extra contents
- How to clear extra contents
(generated by Table of Contents Generator for GitHub Wiki)
Important Notes:
- This document assumes that you have experienced to develop one or more Firefox extensions. If you've never done it, I recommend you to see the basic document for extensions authors.
- APIs around extra contents are totally reconstructed at TST 3.9.0. There is the old version of this document for TST 3.8.x and older versions.
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.
There are some example usecases made by me:
- TST Active Tab in Collapsed Tree
- TST Tab Drag Handle
- TST Indent Line
- A code snippet implementing a search field below tabs
You can set extra contents for a tab with a message with the type set-extra-contents
. For example:
const TST_ID = '[email protected]';
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
andtabIds
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.- Only limited safe type elements are allowed, and all others (for example
<script>
) will be rejected. There is a list of allowed element types.
- Only limited safe type elements are allowed, and all others (for example
-
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 duplicatedid
s. To avoid this concern, an alternative attributeanonid
(named fromanonymous-id
) is available.
- Currently the namespace of
- 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
andtab-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.
// 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;
}
});
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 toset-extra-contents
. -
part
: String, thepart
of the target element. -
properties
: Object, a hash of updated properties. Keys are treated as property names. -
tab
(optional,tabs
,tabId
andtabIds
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.
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 toset-extra-contents
. -
part
: String, thepart
of the target element. -
tab
(optional,tabs
,tabId
andtabIds
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.
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.
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.
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.
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 [email protected]
, 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;
}
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) oroutside
(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 thepart
attribute of the closest element which the event fired on. (TST 4.0 and later) -
windowId
(andwindow
): 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 thevalue
of the input field (input
,textarea
orselect
) which the event is fired on. This won't be provided if the event is not fired on any input field. -
fieldChecked
: Corresponding to thechecked
of theinput
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 andextra-contents-input
) -
locale
(keyboard events and composition events) -
location
(keyboard events) -
repeat
(keyboard events) -
data
(composition events andextra-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;
}
});
You can make inserted extra contents draggable, with draggable
and data-drag-data
attributes.
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 thedataTransfer.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).
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>
-
default
: drag data for no modifier case. - Combinations of
Alt
,Ctrl
, 'Command',MacCtrl
andShift
, connected with+
: drag data for the modifiers combination. Modifiers are compatible to the one on modifier keys for the "commands" manifest feature.
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>
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 toset-extra-contents
. -
tab
(optional,tabs
,tabId
andtabIds
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.
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'
});