| // Copyright 2016 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| /** |
| * @fileoverview 'settings-add-languages-dialog' is a dialog for enabling |
| * languages. |
| */ |
| import 'chrome://resources/cr_elements/cr_button/cr_button.m.js'; |
| import 'chrome://resources/cr_elements/cr_checkbox/cr_checkbox.m.js'; |
| import 'chrome://resources/cr_elements/cr_dialog/cr_dialog.m.js'; |
| import 'chrome://resources/cr_elements/cr_search_field/cr_search_field.js'; |
| import 'chrome://resources/cr_elements/shared_vars_css.m.js'; |
| import 'chrome://resources/polymer/v3_0/iron-list/iron-list.js'; |
| import '../settings_shared_css.js'; |
| |
| import {CrCheckboxElement} from 'chrome://resources/cr_elements/cr_checkbox/cr_checkbox.m.js'; |
| import {CrDialogElement} from 'chrome://resources/cr_elements/cr_dialog/cr_dialog.m.js'; |
| import {CrScrollableBehavior} from 'chrome://resources/cr_elements/cr_scrollable_behavior.m.js'; |
| import {CrSearchFieldElement} from 'chrome://resources/cr_elements/cr_search_field/cr_search_field.js'; |
| import {FindShortcutMixin, FindShortcutMixinInterface} from 'chrome://resources/cr_elements/find_shortcut_mixin.js'; |
| import {html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js'; |
| import {getTemplate} from './add_languages_dialog.html.js'; |
| |
| export interface SettingsAddLanguagesDialogElement { |
| $: { |
| dialog: CrDialogElement, |
| search: CrSearchFieldElement, |
| }; |
| } |
| |
| // Workaround for the fact that TypeScript definitions are missing |
| // |scrollIntoViewIfNeeded|. |
| interface HTMLElementWithScroll extends HTMLElement { |
| scrollIntoViewIfNeeded(): void; |
| } |
| |
| interface Repeaterevent extends Event { |
| model: { |
| item: chrome.languageSettingsPrivate.Language, |
| }; |
| } |
| |
| const SettingsAddLanguagesDialogElementBase = |
| mixinBehaviors([CrScrollableBehavior], FindShortcutMixin(PolymerElement)) as |
| {new (): PolymerElement & FindShortcutMixinInterface}; |
| |
| export class SettingsAddLanguagesDialogElement extends |
| SettingsAddLanguagesDialogElementBase { |
| static get is() { |
| return 'settings-add-languages-dialog'; |
| } |
| |
| static get template() { |
| return getTemplate(); |
| } |
| |
| static get properties() { |
| return { |
| languages: { |
| type: Array, |
| notify: true, |
| }, |
| |
| languagesToAdd_: { |
| type: Object, |
| value() { |
| return new Set(); |
| }, |
| }, |
| |
| disableActionButton_: { |
| type: Boolean, |
| value: true, |
| }, |
| |
| filterValue_: { |
| type: String, |
| value: '', |
| }, |
| }; |
| } |
| |
| languages: Array<chrome.languageSettingsPrivate.Language>; |
| private languagesToAdd_: Set<string>; |
| private disableActionButton_: boolean; |
| private filterValue_: string; |
| |
| connectedCallback() { |
| super.connectedCallback(); |
| |
| this.$.dialog.showModal(); |
| } |
| |
| // Override FindShortcutMixin methods. |
| handleFindShortcut(_modalContextOpen: boolean) { |
| // Assumes this is the only open modal. |
| const searchInput = this.$.search.getSearchInput(); |
| (searchInput as unknown as HTMLElementWithScroll).scrollIntoViewIfNeeded(); |
| if (!this.searchInputHasFocus()) { |
| searchInput.focus(); |
| } |
| return true; |
| } |
| |
| // Override FindShortcutMixin methods. |
| searchInputHasFocus() { |
| return this.$.search.getSearchInput() === |
| this.$.search.shadowRoot!.activeElement; |
| } |
| |
| private onSearchChanged_(e: CustomEvent<string>) { |
| this.filterValue_ = e.detail; |
| } |
| |
| /** |
| * @return A list of languages to be displayed. |
| */ |
| private getLanguages_(): Array<chrome.languageSettingsPrivate.Language> { |
| if (!this.filterValue_) { |
| return this.languages; |
| } |
| |
| const filterValue = this.filterValue_.toLowerCase(); |
| |
| return this.languages.filter(language => { |
| return language.displayName.toLowerCase().includes(filterValue) || |
| language.nativeDisplayName.toLowerCase().includes(filterValue); |
| }); |
| } |
| |
| private getDisplayText_(language: chrome.languageSettingsPrivate.Language): |
| string { |
| let displayText = language.displayName; |
| // If the native name is different, add it. |
| if (language.displayName !== language.nativeDisplayName) { |
| displayText += ' - ' + language.nativeDisplayName; |
| } |
| return displayText; |
| } |
| |
| /** |
| * @return Whether the user has chosen to add this language (checked its |
| * checkbox). |
| */ |
| private willAdd_(languageCode: string): boolean { |
| return this.languagesToAdd_.has(languageCode); |
| } |
| |
| /** |
| * Handler for checking or unchecking a language item. |
| */ |
| private onLanguageCheckboxChange_(e: Repeaterevent) { |
| // Add or remove the item to the Set. No need to worry about data binding: |
| // willAdd_ is called to initialize the checkbox state (in case the |
| // iron-list re-uses a previous checkbox), and the checkbox can only be |
| // changed after that by user action. |
| const language = e.model.item; |
| if ((e.target as CrCheckboxElement).checked) { |
| this.languagesToAdd_.add(language.code); |
| } else { |
| this.languagesToAdd_.delete(language.code); |
| } |
| |
| this.disableActionButton_ = !this.languagesToAdd_.size; |
| } |
| |
| private onCancelButtonTap_() { |
| this.$.dialog.close(); |
| } |
| |
| /** |
| * Enables the checked languages. |
| */ |
| private onActionButtonTap_() { |
| this.dispatchEvent(new CustomEvent('languages-added', { |
| bubbles: true, |
| composed: true, |
| detail: Array.from(this.languagesToAdd_), |
| })); |
| this.$.dialog.close(); |
| } |
| |
| private onKeydown_(e: KeyboardEvent) { |
| // Close dialog if 'esc' is pressed and the search box is already empty. |
| if (e.key === 'Escape' && !this.$.search.getValue().trim()) { |
| this.$.dialog.close(); |
| } else if (e.key !== 'PageDown' && e.key !== 'PageUp') { |
| (this.$.search as unknown as HTMLElementWithScroll) |
| .scrollIntoViewIfNeeded(); |
| } |
| } |
| } |
| |
| declare global { |
| interface HTMLElementTagNameMap { |
| 'settings-add-languages-dialog': SettingsAddLanguagesDialogElement; |
| } |
| } |
| |
| customElements.define( |
| SettingsAddLanguagesDialogElement.is, SettingsAddLanguagesDialogElement); |