UNPKG

chrome-devtools-frontend

Version:
254 lines (222 loc) • 9.47 kB
// Copyright 2021 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. import * as i18n from '../../core/i18n/i18n.js'; import type * as SDK from '../../core/sdk/sdk.js'; import * as Protocol from '../../generated/protocol.js'; import {Issue, IssueCategory, IssueKind} from './Issue.js'; import { resolveLazyDescription, type LazyMarkdownIssueDescription, type MarkdownIssueDescription, } from './MarkdownIssueDescription.js'; const UIStrings = { /** * @description title for autofill documentation page */ howDoesAutofillWorkPageTitle: 'How does autofill work?', /** * @description title for label form elements usage example page */ labelFormlementsPageTitle: 'The label elements', /** * @description title for input form elements usage example page */ inputFormElementPageTitle: 'The form input element', /** * @description title for autocomplete attribute documentation page. */ autocompleteAttributePageTitle: 'HTML attribute: autocomplete', /** * @description title for CORB explainer. */ corbExplainerPageTitle: 'CORB explainer', /** * @description title for history intervention documentation page. */ historyManipulationInterventionPageTitle: 'History manipulation intervention explainer' } as const; const str_ = i18n.i18n.registerUIStrings('models/issues_manager/GenericIssue.ts', UIStrings); const i18nLazyString = i18n.i18n.getLazilyComputedLocalizedString.bind(undefined, str_); export class GenericIssue extends Issue<Protocol.Audits.GenericIssueDetails> { constructor( issueDetails: Protocol.Audits.GenericIssueDetails, issuesModel: SDK.IssuesModel.IssuesModel|null, issueId?: Protocol.Audits.IssueId) { const issueCode = [ Protocol.Audits.InspectorIssueCode.GenericIssue, issueDetails.errorType, ].join('::'); super(issueCode, issueDetails, issuesModel, issueId); } override requests(): Iterable<Protocol.Audits.AffectedRequest> { const details = this.details(); if (details.request) { return [details.request]; } return []; } getCategory(): IssueCategory { return IssueCategory.GENERIC; } primaryKey(): string { const details = this.details(); const requestId = details.request ? details.request.requestId : 'no-request'; return `${this.code()}-(${details.frameId})-(${details.violatingNodeId})-(${details.violatingNodeAttribute})-(${ requestId})`; } getDescription(): MarkdownIssueDescription|null { const description = issueDescriptions.get(this.details().errorType); if (!description) { return null; } return resolveLazyDescription(description); } getKind(): IssueKind { return issueTypes.get(this.details().errorType) || IssueKind.IMPROVEMENT; } static fromInspectorIssue( issuesModel: SDK.IssuesModel.IssuesModel|null, inspectorIssue: Protocol.Audits.InspectorIssue): GenericIssue[] { const genericDetails = inspectorIssue.details.genericIssueDetails; if (!genericDetails) { console.warn('Generic issue without details received.'); return []; } return [new GenericIssue(genericDetails, issuesModel, inspectorIssue.issueId)]; } } export const genericFormLabelForNameError = { file: 'genericFormLabelForNameError.md', links: [{ link: 'https://html.spec.whatwg.org/multipage/forms.html#attr-label-for', // Since the link points to a page with the same title, the 'HTML Standard' // string doesn't need to be translated. linkTitle: i18n.i18n.lockedLazyString('HTML Standard'), }], }; export const genericFormInputWithNoLabelError = { file: 'genericFormInputWithNoLabelError.md', links: [], }; export const genericFormAutocompleteAttributeEmptyError = { file: 'genericFormAutocompleteAttributeEmptyError.md', links: [], }; export const genericFormDuplicateIdForInputError = { file: 'genericFormDuplicateIdForInputError.md', links: [{ link: 'https://web.dev/learn/forms/autofill/#how-does-autofill-work', linkTitle: i18nLazyString(UIStrings.howDoesAutofillWorkPageTitle), }], }; export const genericFormAriaLabelledByToNonExistingIdError = { file: 'genericFormAriaLabelledByToNonExistingIdError.md', links: [{ link: 'https://developer.mozilla.org/en-US/docs/Web/HTML/Element/label', linkTitle: i18nLazyString(UIStrings.labelFormlementsPageTitle), }], }; export const genericFormEmptyIdAndNameAttributesForInputError = { file: 'genericFormEmptyIdAndNameAttributesForInputError.md', links: [{ link: 'https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input', linkTitle: i18nLazyString(UIStrings.inputFormElementPageTitle), }], }; export const genericFormInputAssignedAutocompleteValueToIdOrNameAttributeError = { file: 'genericFormInputAssignedAutocompleteValueToIdOrNameAttributeError.md', links: [{ link: 'https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/autocomplete#values', linkTitle: i18nLazyString(UIStrings.autocompleteAttributePageTitle), }], }; export const genericFormInputHasWrongButWellIntendedAutocompleteValue = { file: 'genericFormInputHasWrongButWellIntendedAutocompleteValueError.md', links: [{ link: 'https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/autocomplete#values', linkTitle: i18nLazyString(UIStrings.autocompleteAttributePageTitle), }], }; export const genericFormLabelForMatchesNonExistingIdError = { file: 'genericFormLabelForMatchesNonExistingIdError.md', links: [{ link: 'https://developer.mozilla.org/en-US/docs/Web/HTML/Element/label', linkTitle: i18nLazyString(UIStrings.labelFormlementsPageTitle), }], }; export const genericFormLabelHasNeitherForNorNestedInputError = { file: 'genericFormLabelHasNeitherForNorNestedInputError.md', links: [{ link: 'https://developer.mozilla.org/en-US/docs/Web/HTML/Element/label', linkTitle: i18nLazyString(UIStrings.labelFormlementsPageTitle), }], }; export const genericResponseWasBlockedbyORB = { file: 'genericResponseWasBlockedByORB.md', links: [{ link: 'https://www.chromium.org/Home/chromium-security/corb-for-developers/', linkTitle: i18nLazyString(UIStrings.corbExplainerPageTitle), }], }; export const genericNavigationEntryMarkedSkippable = { file: 'genericNavigationEntryMarkedSkippable.md', links: [{ link: 'https://chromium.googlesource.com/chromium/src/+/main/docs/history_manipulation_intervention.md', linkTitle: i18nLazyString(UIStrings.historyManipulationInterventionPageTitle), }], }; const issueDescriptions = new Map<Protocol.Audits.GenericIssueErrorType, LazyMarkdownIssueDescription>([ [Protocol.Audits.GenericIssueErrorType.FormLabelForNameError, genericFormLabelForNameError], [Protocol.Audits.GenericIssueErrorType.FormInputWithNoLabelError, genericFormInputWithNoLabelError], [ Protocol.Audits.GenericIssueErrorType.FormAutocompleteAttributeEmptyError, genericFormAutocompleteAttributeEmptyError, ], [Protocol.Audits.GenericIssueErrorType.FormDuplicateIdForInputError, genericFormDuplicateIdForInputError], [ Protocol.Audits.GenericIssueErrorType.FormAriaLabelledByToNonExistingIdError, genericFormAriaLabelledByToNonExistingIdError ], [ Protocol.Audits.GenericIssueErrorType.FormEmptyIdAndNameAttributesForInputError, genericFormEmptyIdAndNameAttributesForInputError, ], [ Protocol.Audits.GenericIssueErrorType.FormInputAssignedAutocompleteValueToIdOrNameAttributeError, genericFormInputAssignedAutocompleteValueToIdOrNameAttributeError, ], [ Protocol.Audits.GenericIssueErrorType.FormLabelForMatchesNonExistingIdError, genericFormLabelForMatchesNonExistingIdError, ], [ Protocol.Audits.GenericIssueErrorType.FormLabelHasNeitherForNorNestedInputError, genericFormLabelHasNeitherForNorNestedInputError, ], [ Protocol.Audits.GenericIssueErrorType.FormInputHasWrongButWellIntendedAutocompleteValueError, genericFormInputHasWrongButWellIntendedAutocompleteValue, ], [ Protocol.Audits.GenericIssueErrorType.ResponseWasBlockedByORB, genericResponseWasBlockedbyORB, ], [ Protocol.Audits.GenericIssueErrorType.NavigationEntryMarkedSkippable, genericNavigationEntryMarkedSkippable, ], ]); const issueTypes = new Map<Protocol.Audits.GenericIssueErrorType, IssueKind>([ [Protocol.Audits.GenericIssueErrorType.FormLabelForNameError, IssueKind.PAGE_ERROR], [Protocol.Audits.GenericIssueErrorType.FormInputWithNoLabelError, IssueKind.IMPROVEMENT], [Protocol.Audits.GenericIssueErrorType.FormAutocompleteAttributeEmptyError, IssueKind.PAGE_ERROR], [Protocol.Audits.GenericIssueErrorType.FormDuplicateIdForInputError, IssueKind.PAGE_ERROR], [Protocol.Audits.GenericIssueErrorType.FormAriaLabelledByToNonExistingIdError, IssueKind.IMPROVEMENT], [Protocol.Audits.GenericIssueErrorType.FormEmptyIdAndNameAttributesForInputError, IssueKind.IMPROVEMENT], [ Protocol.Audits.GenericIssueErrorType.FormInputAssignedAutocompleteValueToIdOrNameAttributeError, IssueKind.IMPROVEMENT, ], [Protocol.Audits.GenericIssueErrorType.FormLabelForMatchesNonExistingIdError, IssueKind.PAGE_ERROR], [Protocol.Audits.GenericIssueErrorType.FormLabelHasNeitherForNorNestedInputError, IssueKind.IMPROVEMENT], [Protocol.Audits.GenericIssueErrorType.FormInputHasWrongButWellIntendedAutocompleteValueError, IssueKind.IMPROVEMENT], ]);