import { basicButton, textButton } from 'app/client/ui2018/buttons'; import { icon } from 'app/client/ui2018/icons'; import { confirmModal } from 'app/client/ui2018/modals'; import { Disposable, dom, IDomArgs, makeTestId, Observable, observable, styled } from 'grainjs'; interface IWidgetOptions { apiKey: Observable; onDelete: () => Promise; onCreate: () => Promise; anonymous?: boolean; // Configure appearance and available options for anonymous use. // When anonymous, no modifications are permitted to profile information. // TODO: add browser test for this option. inputArgs?: IDomArgs; } const testId = makeTestId('test-apikey-'); /** * ApiKey component shows an api key with controls to change it. Expects `options.apiKey` the api * key and shows it if value is truthy along with a 'Delete' button that triggers the * `options.onDelete` callback. When `options.apiKey` is falsy, hides it and show a 'Create' button * that triggers the `options.onCreate` callback. It is the responsibility of the caller to update * the `options.apiKey` to its new value. */ export class ApiKey extends Disposable { private _apiKey: Observable; private _onDeleteCB: () => Promise; private _onCreateCB: () => Promise; private _anonymous: boolean; private _inputArgs: IDomArgs; private _loading = observable(false); private _isHidden: Observable = Observable.create(this, true); constructor(options: IWidgetOptions) { super(); this._apiKey = options.apiKey; this._onDeleteCB = options.onDelete; this._onCreateCB = options.onCreate; this._anonymous = Boolean(options.anonymous); this._inputArgs = options.inputArgs ?? []; } public buildDom() { return dom('div', testId('container'), dom.style('position', 'relative'), dom.maybe(this._apiKey, (apiKey) => dom('div', cssRow( cssInput( { readonly: true, value: this._apiKey.get(), }, dom.attr('type', (use) => use(this._isHidden) ? 'password' : 'text'), testId('key'), {title: 'Click to show'}, dom.on('click', (_ev, el) => { this._isHidden.set(false); setTimeout(() => el.select(), 0); }), dom.on('blur', (ev) => { // Hide the key when it is no longer selected. if (ev.target !== document.activeElement) { this._isHidden.set(true); } }), this._inputArgs ), cssTextBtn( cssTextBtnIcon('Remove'), 'Remove', dom.on('click', () => this._showRemoveKeyModal()), testId('delete'), dom.boolAttr('disabled', (use) => use(this._loading) || this._anonymous) ), ), description(this._getDescription(), testId('description')), )), dom.maybe((use) => !(use(this._apiKey) || this._anonymous), () => [ basicButton('Create', dom.on('click', () => this._onCreate()), testId('create'), dom.boolAttr('disabled', this._loading)), description('By generating an API key, you will be able to make API calls ' + 'for your own account.', testId('description')), ]), ); } // Switch the `_loading` flag to `true` and later, once promise resolves, switch it back to // `false`. private async _switchLoadingFlag(promise: Promise) { this._loading.set(true); try { await promise; } finally { this._loading.set(false); } } private _onDelete(): Promise { return this._switchLoadingFlag(this._onDeleteCB()); } private _onCreate(): Promise { return this._switchLoadingFlag(this._onCreateCB()); } private _getDescription(): string { if (!this._anonymous) { return 'This API key can be used to access your account via the API. ' + 'Don’t share your API key with anyone.'; } else { return 'This API key can be used to access this account anonymously via the API.'; } } private _showRemoveKeyModal(): void { confirmModal( `Remove API Key`, 'Remove', () => this._onDelete(), `You're about to delete an API key. This will cause all future ` + `requests using this API key to be rejected. Do you still want to delete?` ); } } const description = styled('div', ` color: #8a8a8a; font-size: 13px; `); const cssInput = styled('input', ` outline: none; flex: 1 0 0; `); const cssRow = styled('div', ` display: flex; `); const cssTextBtn = styled(textButton, ` text-align: left; width: 90px; margin-left: 16px; `); const cssTextBtnIcon = styled(icon, ` margin: 0 4px 2px 0; `);