2021-12-10 02:37:01 +00:00
|
|
|
/**
|
|
|
|
|
* @license
|
|
|
|
|
* Copyright Google LLC All Rights Reserved.
|
|
|
|
|
*
|
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
|
*/
|
|
|
|
|
|
2021-12-09 05:44:17 +00:00
|
|
|
import {ComponentExplorerViewQuery, DirectiveMetadata, DirectivesProperties, ElementPosition, PropertyQueryTypes, UpdatedStateData,} from 'protocol';
|
|
|
|
|
|
|
|
|
|
import {buildDirectiveTree, getLViewFromDirectiveOrElementInstance} from './directive-forest/index';
|
|
|
|
|
import {deeplySerializeSelectedProperties, serializeDirectiveState} from './state-serializer/state-serializer';
|
2021-10-11 18:11:29 +00:00
|
|
|
|
|
|
|
|
// Need to be kept in sync with Angular framework
|
|
|
|
|
// We can't directly import it from framework now
|
|
|
|
|
// because this also pulls up the security policies
|
|
|
|
|
// for Trusted Types, which we reinstantiate.
|
|
|
|
|
enum ChangeDetectionStrategy {
|
|
|
|
|
OnPush = 0,
|
|
|
|
|
Default = 1,
|
|
|
|
|
}
|
2020-01-27 18:40:18 +00:00
|
|
|
|
2021-12-09 05:44:17 +00:00
|
|
|
import {ComponentTreeNode, DirectiveInstanceType, ComponentInstanceType} from './interfaces';
|
2020-01-27 18:40:18 +00:00
|
|
|
|
2021-09-08 22:36:40 +00:00
|
|
|
const ngDebug = () => (window as any).ng;
|
2020-01-27 18:40:18 +00:00
|
|
|
|
2021-12-09 05:44:17 +00:00
|
|
|
export const getLatestComponentState =
|
|
|
|
|
(query: ComponentExplorerViewQuery, directiveForest?: ComponentTreeNode[]):
|
|
|
|
|
DirectivesProperties|undefined => {
|
|
|
|
|
// if a directive forest is passed in we don't have to build the forest again.
|
|
|
|
|
directiveForest = directiveForest ?? buildDirectiveForest();
|
|
|
|
|
|
|
|
|
|
const node = queryDirectiveForest(query.selectedElement, directiveForest);
|
|
|
|
|
if (!node) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const result: DirectivesProperties = {};
|
|
|
|
|
|
|
|
|
|
const populateResultSet = (dir: DirectiveInstanceType|ComponentInstanceType) => {
|
|
|
|
|
if (query.propertyQuery.type === PropertyQueryTypes.All) {
|
|
|
|
|
result[dir.name] = {
|
|
|
|
|
props: serializeDirectiveState(dir.instance),
|
|
|
|
|
metadata: getDirectiveMetadata(dir.instance),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
if (query.propertyQuery.type === PropertyQueryTypes.Specified) {
|
|
|
|
|
result[dir.name] = {
|
|
|
|
|
props: deeplySerializeSelectedProperties(
|
|
|
|
|
dir.instance, query.propertyQuery.properties[dir.name] || []),
|
|
|
|
|
metadata: getDirectiveMetadata(dir.instance),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
node.directives.forEach(populateResultSet);
|
|
|
|
|
if (node.component) {
|
|
|
|
|
populateResultSet(node.component);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
};
|
2020-01-27 18:40:18 +00:00
|
|
|
|
2020-04-08 18:35:50 +00:00
|
|
|
const enum DirectiveMetadataKey {
|
|
|
|
|
INPUTS = 'inputs',
|
|
|
|
|
OUTPUTS = 'outputs',
|
|
|
|
|
ENCAPSULATION = 'encapsulation',
|
|
|
|
|
ON_PUSH = 'onPush',
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-16 01:03:46 +00:00
|
|
|
// Gets directive metadata. For newer versions of Angular (v12+) it uses
|
|
|
|
|
// the global `getDirectiveMetadata`. For prior versions of the framework
|
|
|
|
|
// the method directly interacts with the directive/component definition.
|
2020-04-08 18:35:50 +00:00
|
|
|
export const getDirectiveMetadata = (dir: any): DirectiveMetadata => {
|
2021-04-16 22:39:16 +00:00
|
|
|
const getMetadata = (window as any).ng.getDirectiveMetadata;
|
|
|
|
|
if (getMetadata) {
|
|
|
|
|
const metadata = getMetadata(dir);
|
2021-04-16 01:03:46 +00:00
|
|
|
if (metadata) {
|
|
|
|
|
return {
|
|
|
|
|
inputs: metadata.inputs,
|
|
|
|
|
outputs: metadata.outputs,
|
|
|
|
|
encapsulation: metadata.encapsulation,
|
|
|
|
|
onPush: metadata.changeDetection === ChangeDetectionStrategy.OnPush,
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Used in older Angular versions, prior to the introduction of `getDirectiveMetadata`.
|
2020-04-08 18:35:50 +00:00
|
|
|
const safelyGrabMetadata = (key: DirectiveMetadataKey) => {
|
2020-03-20 13:53:46 +00:00
|
|
|
try {
|
2020-04-08 18:35:50 +00:00
|
|
|
return dir.constructor.ɵcmp ? dir.constructor.ɵcmp[key] : dir.constructor.ɵdir[key];
|
2020-03-20 13:53:46 +00:00
|
|
|
} catch {
|
2020-04-08 18:35:50 +00:00
|
|
|
console.warn(`Could not find metadata for key: ${key} in directive:`, dir);
|
|
|
|
|
return undefined;
|
2020-03-20 13:53:46 +00:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return {
|
2020-04-08 18:35:50 +00:00
|
|
|
inputs: safelyGrabMetadata(DirectiveMetadataKey.INPUTS),
|
|
|
|
|
outputs: safelyGrabMetadata(DirectiveMetadataKey.OUTPUTS),
|
|
|
|
|
encapsulation: safelyGrabMetadata(DirectiveMetadataKey.ENCAPSULATION),
|
|
|
|
|
onPush: safelyGrabMetadata(DirectiveMetadataKey.ON_PUSH),
|
2020-03-20 13:53:46 +00:00
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2020-03-12 15:47:40 +00:00
|
|
|
const getRootLViewsHelper = (element: Element, rootLViews = new Set<any>()): Set<any> => {
|
2020-03-11 03:08:56 +00:00
|
|
|
if (!(element instanceof HTMLElement)) {
|
2020-03-12 15:47:40 +00:00
|
|
|
return rootLViews;
|
2020-03-11 03:08:56 +00:00
|
|
|
}
|
|
|
|
|
const lView = getLViewFromDirectiveOrElementInstance(element);
|
|
|
|
|
if (lView) {
|
2020-03-12 01:13:06 +00:00
|
|
|
rootLViews.add(lView);
|
2020-03-12 15:47:40 +00:00
|
|
|
return rootLViews;
|
2020-03-11 03:08:56 +00:00
|
|
|
}
|
|
|
|
|
// tslint:disable-next-line: prefer-for-of
|
|
|
|
|
for (let i = 0; i < element.children.length; i++) {
|
|
|
|
|
getRootLViewsHelper(element.children[i], rootLViews);
|
|
|
|
|
}
|
|
|
|
|
return rootLViews;
|
|
|
|
|
};
|
|
|
|
|
|
2021-03-03 23:39:16 +00:00
|
|
|
const getRoots = () => {
|
2021-12-16 07:00:43 +00:00
|
|
|
const roots =
|
|
|
|
|
Array.from(document.documentElement.querySelectorAll('[ng-version]')) as HTMLElement[];
|
|
|
|
|
|
2021-03-03 23:39:16 +00:00
|
|
|
const isTopLevel = (element: HTMLElement) => {
|
2021-12-09 05:44:17 +00:00
|
|
|
let parent: HTMLElement|null = element;
|
2021-12-16 07:00:43 +00:00
|
|
|
|
2021-04-16 22:39:16 +00:00
|
|
|
while (parent?.parentElement) {
|
|
|
|
|
parent = parent.parentElement;
|
2021-03-03 23:39:16 +00:00
|
|
|
if (parent.hasAttribute('ng-version')) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-12-16 07:00:43 +00:00
|
|
|
|
2021-03-03 23:39:16 +00:00
|
|
|
return true;
|
|
|
|
|
};
|
2021-12-16 07:00:43 +00:00
|
|
|
|
2021-03-03 23:39:16 +00:00
|
|
|
return roots.filter(isTopLevel);
|
2020-03-12 01:13:06 +00:00
|
|
|
};
|
|
|
|
|
|
2020-03-30 18:43:00 +00:00
|
|
|
export const buildDirectiveForest = (): ComponentTreeNode[] => {
|
2021-03-03 23:39:16 +00:00
|
|
|
const roots = getRoots();
|
|
|
|
|
return Array.prototype.concat.apply([], Array.from(roots).map(buildDirectiveTree));
|
2020-01-27 18:40:18 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Based on an ElementID we return a specific component node.
|
|
|
|
|
// If we can't find any, we return null.
|
2021-12-09 05:44:17 +00:00
|
|
|
export const queryDirectiveForest =
|
|
|
|
|
(position: ElementPosition, forest: ComponentTreeNode[]): ComponentTreeNode|null => {
|
|
|
|
|
if (!position.length) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
let node: null|ComponentTreeNode = null;
|
|
|
|
|
for (const i of position) {
|
|
|
|
|
node = forest[i];
|
|
|
|
|
if (!node) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
forest = node.children;
|
|
|
|
|
}
|
|
|
|
|
return node;
|
|
|
|
|
};
|
2020-02-07 19:43:49 +00:00
|
|
|
|
2021-12-09 05:44:17 +00:00
|
|
|
export const findNodeInForest =
|
|
|
|
|
(position: ElementPosition, forest: ComponentTreeNode[]): HTMLElement|null => {
|
|
|
|
|
const foundComponent: ComponentTreeNode|null = queryDirectiveForest(position, forest);
|
|
|
|
|
return foundComponent ? (foundComponent.nativeElement as HTMLElement) : null;
|
|
|
|
|
};
|
2020-02-07 19:43:49 +00:00
|
|
|
|
2021-12-09 05:44:17 +00:00
|
|
|
export const findNodeFromSerializedPosition =
|
|
|
|
|
(serializedPosition: string): ComponentTreeNode|null => {
|
|
|
|
|
const position: number[] = serializedPosition.split(',').map((index) => parseInt(index, 10));
|
|
|
|
|
return queryDirectiveForest(position, buildDirectiveForest());
|
|
|
|
|
};
|
2020-02-19 20:07:01 +00:00
|
|
|
|
2020-03-20 18:54:37 +00:00
|
|
|
export const updateState = (updatedStateData: UpdatedStateData): void => {
|
2020-03-11 15:40:42 +00:00
|
|
|
const ngd = ngDebug();
|
2020-03-30 18:43:00 +00:00
|
|
|
const node = queryDirectiveForest(updatedStateData.directiveId.element, buildDirectiveForest());
|
2020-03-20 18:54:37 +00:00
|
|
|
if (!node) {
|
2021-12-09 05:44:17 +00:00
|
|
|
console.warn(
|
|
|
|
|
'Could not update the state of component', updatedStateData,
|
|
|
|
|
'because the component was not found');
|
2020-03-20 18:54:37 +00:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (updatedStateData.directiveId.directive !== undefined) {
|
2020-02-19 20:07:01 +00:00
|
|
|
const directive = node.directives[updatedStateData.directiveId.directive].instance;
|
|
|
|
|
mutateComponentOrDirective(updatedStateData, directive);
|
2020-03-11 15:40:42 +00:00
|
|
|
ngd.applyChanges(ngd.getOwningComponent(directive));
|
2020-03-20 18:54:37 +00:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (node.component) {
|
|
|
|
|
const comp = node.component.instance;
|
|
|
|
|
mutateComponentOrDirective(updatedStateData, comp);
|
|
|
|
|
ngd.applyChanges(comp);
|
|
|
|
|
return;
|
2020-02-19 20:07:01 +00:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2020-03-20 18:54:37 +00:00
|
|
|
const mutateComponentOrDirective = (updatedStateData: UpdatedStateData, compOrDirective: any) => {
|
2020-02-19 20:07:01 +00:00
|
|
|
const valueKey = updatedStateData.keyPath.pop();
|
2020-03-20 18:54:37 +00:00
|
|
|
if (valueKey === undefined) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-02-19 20:07:01 +00:00
|
|
|
|
|
|
|
|
let parentObjectOfValueToUpdate = compOrDirective;
|
2020-03-27 22:59:03 +00:00
|
|
|
updatedStateData.keyPath.forEach((key) => {
|
2020-02-19 20:07:01 +00:00
|
|
|
parentObjectOfValueToUpdate = parentObjectOfValueToUpdate[key];
|
|
|
|
|
});
|
2020-02-24 17:38:22 +00:00
|
|
|
|
2021-05-03 17:38:30 +00:00
|
|
|
// When we try to set a property which only has a getter
|
|
|
|
|
// the line below could throw an error.
|
|
|
|
|
try {
|
|
|
|
|
parentObjectOfValueToUpdate[valueKey] = updatedStateData.newValue;
|
2021-12-09 05:44:17 +00:00
|
|
|
} catch {
|
|
|
|
|
}
|
2020-02-19 20:07:01 +00:00
|
|
|
};
|