electron/lib/browser/api/touch-bar.js

342 lines
10 KiB
JavaScript
Raw Normal View History

const {EventEmitter} = require('events')
let nextItemID = 1
class TouchBar extends EventEmitter {
// Bind a touch bar to a window
static _setOnWindow (touchBar, window) {
if (window._touchBar != null) {
window._touchBar._removeFromWindow(window)
}
if (touchBar == null) {
window._setTouchBarItems([])
return
}
if (Array.isArray(touchBar)) {
touchBar = new TouchBar(touchBar)
}
touchBar._addToWindow(window)
}
constructor (options) {
super()
if (options == null) {
throw new Error('Must specify options object as first argument')
}
let {items, escapeItem} = options
// FIXME Support array as first argument, remove in 2.0
if (Array.isArray(options)) {
items = options
escapeItem = null
}
if (!Array.isArray(items)) {
items = []
}
this.changeListener = (item) => {
this.emit('change', item.id, item.type)
}
2017-03-01 18:55:28 +00:00
this.windowListeners = {}
this.items = {}
this.ordereredItems = []
this.escapeItem = escapeItem
2017-03-02 17:30:21 +00:00
const registerItem = (item) => {
this.items[item.id] = item
item.on('change', this.changeListener)
if (item.child instanceof TouchBar) {
item.child.ordereredItems.forEach(registerItem)
}
}
items.forEach((item) => {
if (!(item instanceof TouchBarItem)) {
2017-03-03 18:22:25 +00:00
throw new Error('Each item must be an instance of TouchBarItem')
}
2017-03-02 17:30:21 +00:00
this.ordereredItems.push(item)
registerItem(item)
})
2017-03-01 18:55:28 +00:00
}
set escapeItem (item) {
if (item != null && !(item instanceof TouchBarItem)) {
throw new Error('Escape item must be an instance of TouchBarItem')
}
if (this.escapeItem != null) {
this.escapeItem.removeListener('change', this.changeListener)
}
this._escapeItem = item
if (this.escapeItem != null) {
this.escapeItem.on('change', this.changeListener)
}
this.emit('escape-item-change', item)
}
get escapeItem () {
return this._escapeItem
}
2017-03-01 18:55:28 +00:00
_addToWindow (window) {
const {id} = window
// Already added to window
if (this.windowListeners.hasOwnProperty(id)) return
window._touchBar = this
2017-03-01 18:55:28 +00:00
const changeListener = (itemID) => {
window._refreshTouchBarItem(itemID)
}
this.on('change', changeListener)
const escapeItemListener = (item) => {
window._setEscapeTouchBarItem(item != null ? item : {})
}
this.on('escape-item-change', escapeItemListener)
2017-03-01 18:55:28 +00:00
const interactionListener = (event, itemID, details) => {
let item = this.items[itemID]
if (item == null && this.escapeItem != null && this.escapeItem.id === itemID) {
item = this.escapeItem
}
if (item != null && item.onInteraction != null) {
item.onInteraction(details)
}
2017-03-01 18:55:28 +00:00
}
window.on('-touch-bar-interaction', interactionListener)
const removeListeners = () => {
this.removeListener('change', changeListener)
this.removeListener('escape-item-change', escapeItemListener)
2017-03-01 18:55:28 +00:00
window.removeListener('-touch-bar-interaction', interactionListener)
window.removeListener('closed', removeListeners)
window._touchBar = null
delete this.windowListeners[id]
const unregisterItems = (items) => {
for (const item of items) {
item.removeListener('change', this.changeListener)
if (item.child instanceof TouchBar) {
unregisterItems(item.child.ordereredItems)
}
}
}
unregisterItems(this.ordereredItems)
if (this.escapeItem) {
this.escapeItem.removeListener('change', this.changeListener)
}
2017-03-01 18:55:28 +00:00
}
window.once('closed', removeListeners)
this.windowListeners[id] = removeListeners
window._setTouchBarItems(this.ordereredItems)
escapeItemListener(this.escapeItem)
2017-03-01 18:55:28 +00:00
}
_removeFromWindow (window) {
const removeListeners = this.windowListeners[window.id]
if (removeListeners != null) removeListeners()
}
}
2017-03-01 00:08:12 +00:00
class TouchBarItem extends EventEmitter {
2017-03-02 22:29:59 +00:00
constructor () {
2017-03-01 00:08:12 +00:00
super()
this._addImmutableProperty('id', `${nextItemID++}`)
this._parents = []
}
_addImmutableProperty (name, value) {
Object.defineProperty(this, name, {
get: function () {
return value
},
set: function () {
throw new Error(`Cannot override property ${name}`)
},
enumerable: true,
configurable: false
})
}
2017-03-01 22:54:43 +00:00
_addLiveProperty (name, initialValue) {
const privateName = `_${name}`
this[privateName] = initialValue
Object.defineProperty(this, name, {
get: function () {
return this[privateName]
},
set: function (value) {
this[privateName] = value
this.emit('change', this)
2017-03-01 22:54:43 +00:00
},
enumerable: true
})
}
_addParent (item) {
const existing = this._parents.some(test => test.id === item.id)
if (!existing) {
this._parents.push({
id: item.id,
type: item.type
})
}
}
}
TouchBar.TouchBarButton = class TouchBarButton extends TouchBarItem {
constructor (config) {
2017-03-02 22:29:59 +00:00
super()
if (config == null) config = {}
this._addImmutableProperty('type', 'button')
const {click, icon, iconPosition, label, backgroundColor} = config
2017-03-01 22:54:43 +00:00
this._addLiveProperty('label', label)
this._addLiveProperty('backgroundColor', backgroundColor)
this._addLiveProperty('icon', icon)
this._addLiveProperty('iconPosition', iconPosition)
2017-03-01 22:54:43 +00:00
if (typeof click === 'function') {
this._addImmutableProperty('onInteraction', () => {
2017-03-03 18:22:25 +00:00
config.click()
})
2017-03-01 22:54:43 +00:00
}
}
}
TouchBar.TouchBarColorPicker = class TouchBarColorPicker extends TouchBarItem {
constructor (config) {
2017-03-02 22:29:59 +00:00
super()
if (config == null) config = {}
this._addImmutableProperty('type', 'colorpicker')
2017-03-01 22:54:43 +00:00
const {availableColors, change, selectedColor} = config
this._addLiveProperty('availableColors', availableColors)
this._addLiveProperty('selectedColor', selectedColor)
if (typeof change === 'function') {
this._addImmutableProperty('onInteraction', (details) => {
2017-03-01 22:54:43 +00:00
this._selectedColor = details.color
change(details.color)
})
}
}
}
TouchBar.TouchBarGroup = class TouchBarGroup extends TouchBarItem {
2016-11-28 07:24:48 +00:00
constructor (config) {
2017-03-02 22:29:59 +00:00
super()
if (config == null) config = {}
this._addImmutableProperty('type', 'group')
const defaultChild = (config.items instanceof TouchBar) ? config.items : new TouchBar(config.items)
this._addLiveProperty('child', defaultChild)
this.child.ordereredItems.forEach((item) => item._addParent(this))
2016-11-28 07:24:48 +00:00
}
}
TouchBar.TouchBarLabel = class TouchBarLabel extends TouchBarItem {
constructor (config) {
2017-03-02 22:29:59 +00:00
super()
if (config == null) config = {}
this._addImmutableProperty('type', 'label')
2017-03-01 22:54:43 +00:00
this._addLiveProperty('label', config.label)
this._addLiveProperty('textColor', config.textColor)
}
}
TouchBar.TouchBarPopover = class TouchBarPopover extends TouchBarItem {
constructor (config) {
2017-03-02 22:29:59 +00:00
super()
if (config == null) config = {}
this._addImmutableProperty('type', 'popover')
2017-03-01 22:54:43 +00:00
this._addLiveProperty('label', config.label)
2017-03-02 21:37:34 +00:00
this._addLiveProperty('icon', config.icon)
this._addLiveProperty('showCloseButton', config.showCloseButton)
const defaultChild = (config.items instanceof TouchBar) ? config.items : new TouchBar(config.items)
this._addLiveProperty('child', defaultChild)
this.child.ordereredItems.forEach((item) => item._addParent(this))
}
}
TouchBar.TouchBarSlider = class TouchBarSlider extends TouchBarItem {
constructor (config) {
2017-03-02 22:29:59 +00:00
super()
if (config == null) config = {}
this._addImmutableProperty('type', 'slider')
2017-03-01 22:54:43 +00:00
const {change, label, minValue, maxValue, value} = config
this._addLiveProperty('label', label)
this._addLiveProperty('minValue', minValue)
this._addLiveProperty('maxValue', maxValue)
this._addLiveProperty('value', value)
if (typeof change === 'function') {
this._addImmutableProperty('onInteraction', (details) => {
2017-03-01 22:54:43 +00:00
this._value = details.value
change(details.value)
})
}
}
}
2017-03-03 17:54:46 +00:00
TouchBar.TouchBarSpacer = class TouchBarSpacer extends TouchBarItem {
constructor (config) {
super()
if (config == null) config = {}
this._addImmutableProperty('type', 'spacer')
this._addImmutableProperty('size', config.size)
2017-03-03 17:54:46 +00:00
}
}
2017-03-10 06:40:39 +00:00
TouchBar.TouchBarSegmentedControl = class TouchBarSegmentedControl extends TouchBarItem {
constructor (config) {
super()
if (config == null) config = {}
const {segmentStyle, segments, selectedIndex, change, mode} = config
this._addImmutableProperty('type', 'segmented_control')
2017-03-10 06:56:26 +00:00
this._addLiveProperty('segmentStyle', segmentStyle)
this._addLiveProperty('segments', segments || [])
2017-03-10 06:40:39 +00:00
this._addLiveProperty('selectedIndex', selectedIndex)
this._addLiveProperty('mode', mode)
2017-03-10 06:40:39 +00:00
if (typeof change === 'function') {
this._addImmutableProperty('onInteraction', (details) => {
2017-03-10 06:56:26 +00:00
this._selectedIndex = details.selectedIndex
2017-05-03 10:25:50 +00:00
change(details.selectedIndex, details.isSelected)
})
2017-03-10 06:40:39 +00:00
}
}
}
2017-03-12 23:51:12 +00:00
TouchBar.TouchBarScrubber = class TouchBarScrubber extends TouchBarItem {
2017-03-13 00:00:10 +00:00
constructor (config) {
2017-03-12 23:51:12 +00:00
super()
if (config == null) config = {}
2017-03-15 16:47:07 +00:00
const {items, selectedStyle, overlayStyle, showArrowButtons, continuous, mode} = config
2017-03-14 21:02:48 +00:00
let {select, highlight} = config
this._addImmutableProperty('type', 'scrubber')
2017-03-12 23:51:12 +00:00
this._addLiveProperty('items', items)
2017-03-14 07:57:57 +00:00
this._addLiveProperty('selectedStyle', selectedStyle || null)
2017-03-15 16:47:07 +00:00
this._addLiveProperty('overlayStyle', overlayStyle || null)
2017-03-14 07:57:57 +00:00
this._addLiveProperty('showArrowButtons', showArrowButtons || false)
this._addLiveProperty('mode', mode || 'free')
this._addLiveProperty('continuous', typeof continuous === 'undefined' ? true : continuous)
2017-03-12 23:51:12 +00:00
2017-03-14 21:02:48 +00:00
if (typeof select === 'function' || typeof highlight === 'function') {
if (select == null) select = () => {}
if (highlight == null) highlight = () => {}
this._addImmutableProperty('onInteraction', (details) => {
if (details.type === 'select' && typeof select === 'function') {
2017-03-14 21:02:48 +00:00
select(details.selectedIndex)
} else if (details.type === 'highlight' && typeof highlight === 'function') {
2017-03-14 21:02:48 +00:00
highlight(details.highlightedIndex)
2017-03-12 23:51:12 +00:00
}
})
2017-03-12 23:51:12 +00:00
}
}
}
2016-11-29 07:55:07 +00:00
module.exports = TouchBar