chrome-devtools-frontend
Version:
Chrome DevTools UI
79 lines (67 loc) • 3.26 kB
text/typescript
// Copyright 2021 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.
import * as Common from '../common/common.js';
import * as Root from '../root/root.js';
import * as SDK from '../sdk/sdk.js';
let contrastCheckTriggerInstance: ContrastCheckTrigger|null = null;
export class ContrastCheckTrigger extends Common.ObjectWrapper.ObjectWrapper {
private pageLoadListeners: WeakMap<SDK.ResourceTreeModel.ResourceTreeModel, Common.EventTarget.EventDescriptor> =
new WeakMap();
private frameAddedListeners: WeakMap<SDK.ResourceTreeModel.ResourceTreeModel, Common.EventTarget.EventDescriptor> =
new WeakMap();
constructor() {
super();
SDK.SDKModel.TargetManager.instance().observeModels(SDK.ResourceTreeModel.ResourceTreeModel, this);
}
static instance({forceNew}: {forceNew: boolean} = {forceNew: false}): ContrastCheckTrigger {
if (!contrastCheckTriggerInstance || forceNew) {
contrastCheckTriggerInstance = new ContrastCheckTrigger();
}
return contrastCheckTriggerInstance;
}
async modelAdded(resourceTreeModel: SDK.ResourceTreeModel.ResourceTreeModel): Promise<void> {
this.pageLoadListeners.set(
resourceTreeModel,
resourceTreeModel.addEventListener(SDK.ResourceTreeModel.Events.Load, this.pageLoaded, this));
this.frameAddedListeners.set(
resourceTreeModel,
resourceTreeModel.addEventListener(SDK.ResourceTreeModel.Events.FrameAdded, this.frameAdded, this));
}
modelRemoved(resourceTreeModel: SDK.ResourceTreeModel.ResourceTreeModel): void {
const pageLoadListener = this.pageLoadListeners.get(resourceTreeModel);
if (pageLoadListener) {
Common.EventTarget.EventTarget.removeEventListeners([pageLoadListener]);
}
const frameAddedListeners = this.frameAddedListeners.get(resourceTreeModel);
if (frameAddedListeners) {
Common.EventTarget.EventTarget.removeEventListeners([frameAddedListeners]);
}
}
private checkContrast(resourceTreeModel: SDK.ResourceTreeModel.ResourceTreeModel): void {
if (!Root.Runtime.experiments.isEnabled('contrastIssues')) {
return;
}
resourceTreeModel.target().auditsAgent().invoke_checkContrast();
}
private pageLoaded(event: Common.EventTarget.EventTargetEvent): void {
const {resourceTreeModel} = event.data as {resourceTreeModel: SDK.ResourceTreeModel.ResourceTreeModel};
this.checkContrast(resourceTreeModel);
}
private async frameAdded(event: Common.EventTarget.EventTargetEvent): Promise<void> {
if (!Root.Runtime.experiments.isEnabled('contrastIssues')) {
return;
}
const frame = event.data as SDK.ResourceTreeModel.ResourceTreeFrame;
if (!frame.isMainFrame()) {
return;
}
// If the target document finished loading, check the contrast immediately.
// Otherwise, it should be triggered when the page load event fires.
const response = await frame.resourceTreeModel().target().runtimeAgent().invoke_evaluate(
{expression: 'document.readyState', returnByValue: true});
if (response.result && response.result.value === 'complete') {
this.checkContrast(frame.resourceTreeModel());
}
}
}