import { Menu, dialog, app, getCurrentWindow } from '@electron/remote'; import React, { Component } from 'react'; import { defineMessages, injectIntl } from 'react-intl'; import PropTypes from 'prop-types'; import { observer } from 'mobx-react'; import classnames from 'classnames'; import { SortableElement } from 'react-sortable-hoc'; import injectSheet from 'react-jss'; import ms from 'ms'; import { observable, autorun } from 'mobx'; import ServiceModel from '../../../models/Service'; import { cmdOrCtrlShortcutKey } from '../../../environment'; import globalMessages from '../../../i18n/globalMessages'; const IS_SERVICE_DEBUGGING_ENABLED = ( localStorage.getItem('debug') || '' ).includes('Ferdi:Service'); const messages = defineMessages({ reload: { id: 'tabs.item.reload', defaultMessage: 'Reload', }, disableNotifications: { id: 'tabs.item.disableNotifications', defaultMessage: 'Disable notifications', }, enableNotifications: { id: 'tabs.item.enableNotification', defaultMessage: 'Enable notifications', }, disableAudio: { id: 'tabs.item.disableAudio', defaultMessage: 'Disable audio', }, enableAudio: { id: 'tabs.item.enableAudio', defaultMessage: 'Enable audio', }, enableDarkMode: { id: 'tabs.item.enableDarkMode', defaultMessage: 'Enable Dark mode', }, disableDarkMode: { id: 'tabs.item.disableDarkMode', defaultMessage: 'Disable Dark mode', }, disableService: { id: 'tabs.item.disableService', defaultMessage: 'Disable service', }, enableService: { id: 'tabs.item.enableService', defaultMessage: 'Enable service', }, hibernateService: { id: 'tabs.item.hibernateService', defaultMessage: 'Hibernate service', }, wakeUpService: { id: 'tabs.item.wakeUpService', defaultMessage: 'Wake up service', }, deleteService: { id: 'tabs.item.deleteService', defaultMessage: 'Delete service', }, confirmDeleteService: { id: 'tabs.item.confirmDeleteService', defaultMessage: 'Do you really want to delete the {serviceName} service?', }, }); let pollIndicatorTransition = 'none'; let polledTransition = 'none'; let pollAnsweredTransition = 'none'; if (window && window.matchMedia('(prefers-reduced-motion: no-preference)')) { pollIndicatorTransition = 'background 0.5s'; polledTransition = 'background 0.1s'; pollAnsweredTransition = 'background 0.1s'; } const styles = { pollIndicator: { position: 'absolute', bottom: 2, width: 10, height: 10, borderRadius: 5, background: 'gray', transition: pollIndicatorTransition, }, pollIndicatorPoll: { left: 2, }, pollIndicatorAnswer: { left: 14, }, polled: { background: 'yellow !important', transition: polledTransition, }, pollAnswered: { background: 'green !important', transition: pollAnsweredTransition, }, stale: { background: 'red !important', }, }; @injectSheet(styles) @observer class TabItem extends Component { static propTypes = { classes: PropTypes.object.isRequired, service: PropTypes.instanceOf(ServiceModel).isRequired, clickHandler: PropTypes.func.isRequired, shortcutIndex: PropTypes.number.isRequired, reload: PropTypes.func.isRequired, toggleNotifications: PropTypes.func.isRequired, toggleAudio: PropTypes.func.isRequired, toggleDarkMode: PropTypes.func.isRequired, openSettings: PropTypes.func.isRequired, deleteService: PropTypes.func.isRequired, disableService: PropTypes.func.isRequired, enableService: PropTypes.func.isRequired, hibernateService: PropTypes.func.isRequired, wakeUpService: PropTypes.func.isRequired, showMessageBadgeWhenMutedSetting: PropTypes.bool.isRequired, showMessageBadgesEvenWhenMuted: PropTypes.bool.isRequired, }; @observable isPolled = false; @observable isPollAnswered = false; componentDidMount() { const { service } = this.props; if (IS_SERVICE_DEBUGGING_ENABLED) { autorun(() => { if (Date.now() - service.lastPoll < ms('0.2s')) { this.isPolled = true; setTimeout(() => { this.isPolled = false; }, ms('1s')); } if (Date.now() - service.lastPollAnswer < ms('0.2s')) { this.isPollAnswered = true; setTimeout(() => { this.isPollAnswered = false; }, ms('1s')); } }); } } render() { const { classes, service, clickHandler, shortcutIndex, reload, toggleNotifications, toggleAudio, toggleDarkMode, deleteService, disableService, enableService, hibernateService, wakeUpService, openSettings, showMessageBadgeWhenMutedSetting, showMessageBadgesEvenWhenMuted, } = this.props; const { intl } = this.props; const menuTemplate = [ { label: service.name || service.recipe.name, enabled: false, }, { type: 'separator', }, { label: intl.formatMessage(messages.reload), click: reload, accelerator: `${cmdOrCtrlShortcutKey()}+R`, }, { label: intl.formatMessage(globalMessages.edit), click: () => openSettings({ path: `services/edit/${service.id}`, }), }, { type: 'separator', }, { label: service.isNotificationEnabled ? intl.formatMessage(messages.disableNotifications) : intl.formatMessage(messages.enableNotifications), click: () => toggleNotifications(), }, { label: service.isMuted ? intl.formatMessage(messages.enableAudio) : intl.formatMessage(messages.disableAudio), click: () => toggleAudio(), }, { label: service.isDarkModeEnabled ? intl.formatMessage(messages.disableDarkMode) : intl.formatMessage(messages.enableDarkMode), click: () => toggleDarkMode(), }, { label: intl.formatMessage( service.isEnabled ? messages.disableService : messages.enableService, ), click: () => (service.isEnabled ? disableService() : enableService()), }, { label: intl.formatMessage( service.isHibernating ? messages.wakeUpService : messages.hibernateService, ), click: () => (service.isHibernating ? wakeUpService() : hibernateService()), enabled: service.canHibernate, }, { type: 'separator', }, { label: intl.formatMessage(messages.deleteService), click: () => { const selection = dialog.showMessageBoxSync(app.mainWindow, { type: 'question', message: intl.formatMessage(messages.deleteService), detail: intl.formatMessage(messages.confirmDeleteService, { serviceName: service.name || service.recipe.name, }), buttons: [ intl.formatMessage(globalMessages.yes), intl.formatMessage(globalMessages.no), ], }); if (selection === 0) { deleteService(); } }, }, ]; const menu = Menu.buildFromTemplate(menuTemplate); let notificationBadge = null; if ( (showMessageBadgeWhenMutedSetting || service.isNotificationEnabled) && showMessageBadgesEvenWhenMuted && service.isBadgeEnabled ) { notificationBadge = ( {service.unreadDirectMessageCount > 0 && ( {service.unreadDirectMessageCount} )} {service.unreadIndirectMessageCount > 0 && service.unreadDirectMessageCount === 0 && service.isIndirectMessageBadgeEnabled && ( )} {service.isHibernating && ( )} ); } return (
  • menu.popup(getCurrentWindow())} data-tip={`${service.name} ${ shortcutIndex <= 9 ? `(${cmdOrCtrlShortcutKey(false)}+${shortcutIndex})` : '' }`} > {notificationBadge} {IS_SERVICE_DEBUGGING_ENABLED && ( <>
    )}
  • ); } } export default injectIntl(SortableElement(TabItem));