@angular/material 12.0.0 → 12.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/_index.scss +2 -1
- package/_theming.scss +39 -4
- package/autocomplete/_autocomplete-legacy-index.scss +2 -0
- package/autocomplete/autocomplete.d.ts +4 -2
- package/autocomplete/index.metadata.json +1 -1
- package/badge/_badge-legacy-index.scss +3 -0
- package/bottom-sheet/_bottom-sheet-legacy-index.scss +2 -0
- package/bottom-sheet/bottom-sheet-container.d.ts +0 -2
- package/bottom-sheet/index.metadata.json +1 -1
- package/bundles/material-autocomplete.umd.js.map +1 -1
- package/bundles/material-bottom-sheet.umd.js +7 -15
- package/bundles/material-bottom-sheet.umd.js.map +1 -1
- package/bundles/material-button-toggle.umd.js.map +1 -1
- package/bundles/material-checkbox.umd.js +1 -1
- package/bundles/material-checkbox.umd.js.map +1 -1
- package/bundles/material-core.umd.js +2 -2
- package/bundles/material-core.umd.js.map +1 -1
- package/bundles/material-datepicker.umd.js +17 -19
- package/bundles/material-datepicker.umd.js.map +1 -1
- package/bundles/material-dialog.umd.js +11 -19
- package/bundles/material-dialog.umd.js.map +1 -1
- package/bundles/material-input.umd.js +8 -2
- package/bundles/material-input.umd.js.map +1 -1
- package/bundles/material-radio.umd.js +4 -4
- package/bundles/material-radio.umd.js.map +1 -1
- package/bundles/material-slide-toggle.umd.js +4 -4
- package/bundles/material-slide-toggle.umd.js.map +1 -1
- package/bundles/material-sort.umd.js +7 -1
- package/bundles/material-sort.umd.js.map +1 -1
- package/bundles/material-stepper.umd.js +2 -2
- package/bundles/material-stepper.umd.js.map +1 -1
- package/bundles/material-table.umd.js.map +1 -1
- package/bundles/material-tabs.umd.js +4 -2
- package/bundles/material-tabs.umd.js.map +1 -1
- package/bundles/material-tooltip.umd.js +19 -15
- package/bundles/material-tooltip.umd.js.map +1 -1
- package/button/_button-legacy-index.scss +19 -0
- package/button-toggle/_button-toggle-legacy-index.scss +2 -0
- package/button-toggle/button-toggle.d.ts +4 -0
- package/button-toggle/index.metadata.json +1 -1
- package/card/_card-legacy-index.scss +2 -0
- package/checkbox/_checkbox-legacy-index.scss +2 -0
- package/checkbox/checkbox-config.d.ts +2 -0
- package/checkbox/index.metadata.json +1 -1
- package/chips/_chips-legacy-index.scss +6 -0
- package/core/_core-legacy-index.scss +17 -0
- package/core/color/_color-legacy-index.scss +1 -0
- package/core/density/private/_density-legacy-index.scss +5 -0
- package/core/focus-indicators/_focus-indicators-legacy-index.scss +5 -0
- package/core/focus-indicators/_focus-indicators.import.scss +1 -1
- package/core/focus-indicators/_focus-indicators.scss +4 -0
- package/core/index.metadata.json +1 -1
- package/core/option/_option-legacy-index.scss +5 -0
- package/core/ripple/_ripple-legacy-index.scss +3 -0
- package/core/ripple/ripple.d.ts +8 -6
- package/core/selection/pseudo-checkbox/_pseudo-checkbox-legacy-index.scss +2 -0
- package/core/style/_style-legacy-index.scss +24 -0
- package/core/theming/_theming-legacy-index.scss +22 -0
- package/core/theming/_theming.scss +45 -22
- package/core/typography/_typography-legacy-index.scss +18 -0
- package/core/typography/_typography-utils.scss +27 -8
- package/core/typography/_typography.scss +36 -7
- package/datepicker/_datepicker-legacy-index.scss +9 -0
- package/datepicker/calendar.d.ts +2 -4
- package/datepicker/datepicker-base.d.ts +6 -2
- package/datepicker/index.metadata.json +1 -1
- package/dialog/_dialog-legacy-index.scss +2 -0
- package/dialog/dialog-container.d.ts +0 -2
- package/dialog/index.metadata.json +1 -1
- package/divider/_divider-legacy-index.scss +3 -0
- package/esm2015/autocomplete/autocomplete.js +1 -1
- package/esm2015/bottom-sheet/bottom-sheet-container.js +5 -12
- package/esm2015/button-toggle/button-toggle.js +1 -1
- package/esm2015/checkbox/checkbox-config.js +1 -1
- package/esm2015/checkbox/checkbox.js +2 -2
- package/esm2015/core/common-behaviors/common-module.js +1 -1
- package/esm2015/core/ripple/ripple.js +1 -1
- package/esm2015/core/version.js +1 -1
- package/esm2015/datepicker/calendar.js +3 -5
- package/esm2015/datepicker/date-range-input-parts.js +3 -2
- package/esm2015/datepicker/datepicker-base.js +9 -9
- package/esm2015/datepicker/datepicker-intl.js +3 -3
- package/esm2015/dialog/dialog-container.js +5 -12
- package/esm2015/input/input.js +10 -4
- package/esm2015/radio/radio.js +5 -5
- package/esm2015/slide-toggle/slide-toggle.js +5 -5
- package/esm2015/sort/sort-header.js +8 -2
- package/esm2015/stepper/step-header.js +3 -3
- package/esm2015/table/table-data-source.js +1 -1
- package/esm2015/tabs/tab-group.js +4 -2
- package/esm2015/tabs/tab.js +3 -3
- package/esm2015/tooltip/tooltip.js +12 -8
- package/expansion/_expansion-legacy-index.scss +3 -0
- package/fesm2015/autocomplete.js.map +1 -1
- package/fesm2015/bottom-sheet.js +4 -11
- package/fesm2015/bottom-sheet.js.map +1 -1
- package/fesm2015/button-toggle.js.map +1 -1
- package/fesm2015/checkbox.js +1 -1
- package/fesm2015/checkbox.js.map +1 -1
- package/fesm2015/core.js +2 -2
- package/fesm2015/core.js.map +1 -1
- package/fesm2015/datepicker.js +14 -15
- package/fesm2015/datepicker.js.map +1 -1
- package/fesm2015/dialog.js +4 -11
- package/fesm2015/dialog.js.map +1 -1
- package/fesm2015/input.js +9 -3
- package/fesm2015/input.js.map +1 -1
- package/fesm2015/radio.js +4 -4
- package/fesm2015/radio.js.map +1 -1
- package/fesm2015/slide-toggle.js +4 -4
- package/fesm2015/slide-toggle.js.map +1 -1
- package/fesm2015/sort.js +7 -1
- package/fesm2015/sort.js.map +1 -1
- package/fesm2015/stepper.js +2 -2
- package/fesm2015/stepper.js.map +1 -1
- package/fesm2015/table.js.map +1 -1
- package/fesm2015/tabs.js +4 -2
- package/fesm2015/tabs.js.map +1 -1
- package/fesm2015/tooltip.js +11 -7
- package/fesm2015/tooltip.js.map +1 -1
- package/form-field/_form-field-legacy-index.scss +24 -0
- package/grid-list/_grid-list-legacy-index.scss +2 -0
- package/icon/_icon-legacy-index.scss +2 -0
- package/input/_input-legacy-index.scss +2 -0
- package/input/index.metadata.json +1 -1
- package/list/_list-legacy-index.scss +2 -0
- package/menu/_menu-legacy-index.scss +2 -0
- package/package.json +2 -2
- package/paginator/_paginator-legacy-index.scss +2 -0
- package/progress-bar/_progress-bar-legacy-index.scss +2 -0
- package/progress-spinner/_progress-spinner-legacy-index.scss +2 -0
- package/radio/_radio-legacy-index.scss +2 -0
- package/radio/index.metadata.json +1 -1
- package/radio/radio.d.ts +3 -2
- package/schematics/ng-add/fonts/material-fonts.mjs +46 -0
- package/schematics/ng-add/index.js +1 -1
- package/schematics/ng-add/index.mjs +55 -0
- package/schematics/ng-add/package-config.mjs +47 -0
- package/schematics/ng-add/schema.mjs +10 -0
- package/schematics/ng-add/setup-project.mjs +121 -0
- package/schematics/ng-add/theming/create-custom-theme.mjs +51 -0
- package/schematics/ng-add/theming/theming.mjs +165 -0
- package/schematics/ng-generate/address-form/index.mjs +49 -0
- package/schematics/ng-generate/address-form/schema.mjs +10 -0
- package/schematics/ng-generate/dashboard/index.mjs +49 -0
- package/schematics/ng-generate/dashboard/schema.mjs +10 -0
- package/schematics/ng-generate/navigation/index.mjs +49 -0
- package/schematics/ng-generate/navigation/schema.mjs +10 -0
- package/schematics/ng-generate/table/index.mjs +46 -0
- package/schematics/ng-generate/table/schema.mjs +10 -0
- package/schematics/ng-generate/tree/index.mjs +46 -0
- package/schematics/ng-generate/tree/schema.mjs +10 -0
- package/schematics/ng-update/data/attribute-selectors.mjs +12 -0
- package/schematics/ng-update/data/class-names.mjs +32 -0
- package/schematics/ng-update/data/constructor-checks.mjs +143 -0
- package/schematics/ng-update/data/css-selectors.mjs +46 -0
- package/schematics/ng-update/data/element-selectors.mjs +18 -0
- package/schematics/ng-update/data/index.mjs +29 -0
- package/schematics/ng-update/data/input-names.mjs +76 -0
- package/schematics/ng-update/data/method-call-checks.mjs +22 -0
- package/schematics/ng-update/data/output-names.mjs +101 -0
- package/schematics/ng-update/data/property-names.mjs +173 -0
- package/schematics/ng-update/index.mjs +78 -0
- package/schematics/ng-update/migrations/hammer-gestures-v9/find-hammer-script-tags.mjs +42 -0
- package/schematics/ng-update/migrations/hammer-gestures-v9/find-main-module.mjs +32 -0
- package/schematics/ng-update/migrations/hammer-gestures-v9/hammer-gestures-migration.js +3 -3
- package/schematics/ng-update/migrations/hammer-gestures-v9/hammer-gestures-migration.mjs +778 -0
- package/schematics/ng-update/migrations/hammer-gestures-v9/hammer-template-check.mjs +56 -0
- package/schematics/ng-update/migrations/hammer-gestures-v9/import-manager.mjs +370 -0
- package/schematics/ng-update/migrations/hammer-gestures-v9/remove-array-element.mjs +66 -0
- package/schematics/ng-update/migrations/hammer-gestures-v9/remove-element-from-html.mjs +28 -0
- package/schematics/ng-update/migrations/misc-checks/misc-class-inheritance.mjs +48 -0
- package/schematics/ng-update/migrations/misc-checks/misc-class-names.mjs +43 -0
- package/schematics/ng-update/migrations/misc-checks/misc-imports.mjs +54 -0
- package/schematics/ng-update/migrations/misc-checks/misc-property-names.mjs +47 -0
- package/schematics/ng-update/migrations/misc-checks/misc-template.mjs +60 -0
- package/schematics/ng-update/migrations/misc-ripples-v7/ripple-speed-factor-migration.mjs +138 -0
- package/schematics/ng-update/migrations/misc-ripples-v7/ripple-speed-factor.mjs +30 -0
- package/schematics/ng-update/migrations/package-imports-v8/secondary-entry-points-migration.js +6 -2
- package/schematics/ng-update/migrations/package-imports-v8/secondary-entry-points-migration.mjs +173 -0
- package/schematics/ng-update/migrations/theming-api-v12/config.mjs +206 -0
- package/schematics/ng-update/migrations/theming-api-v12/migration.js +6 -1
- package/schematics/ng-update/migrations/theming-api-v12/migration.mjs +239 -0
- package/schematics/ng-update/migrations/theming-api-v12/theming-api-migration.mjs +45 -0
- package/schematics/ng-update/typescript/module-specifiers.mjs +35 -0
- package/schematics/ng-update/upgrade-data.mjs +24 -0
- package/schematics/paths.mjs +16 -0
- package/schematics/schematics.externs.js +0 -0
- package/select/_select-legacy-index.scss +2 -0
- package/sidenav/_sidenav-legacy-index.scss +2 -0
- package/slide-toggle/_slide-toggle-legacy-index.scss +3 -0
- package/slide-toggle/index.metadata.json +1 -1
- package/slide-toggle/slide-toggle.d.ts +3 -2
- package/slider/_slider-legacy-index.scss +2 -0
- package/snack-bar/_snack-bar-legacy-index.scss +2 -0
- package/sort/_sort-legacy-index.scss +2 -0
- package/sort/sort-header.d.ts +1 -1
- package/stepper/_stepper-legacy-index.scss +7 -0
- package/stepper/index.metadata.json +1 -1
- package/table/_table-legacy-index.scss +3 -0
- package/table/index.metadata.json +1 -1
- package/table/table-data-source.d.ts +17 -5
- package/tabs/_tabs-legacy-index.scss +6 -0
- package/tabs/index.metadata.json +1 -1
- package/toolbar/_toolbar-legacy-index.scss +4 -0
- package/tooltip/_tooltip-legacy-index.scss +3 -0
- package/tooltip/index.metadata.json +1 -1
- package/tree/_tree-legacy-index.scss +2 -0
|
@@ -0,0 +1,778 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* @license
|
|
4
|
+
* Copyright Google LLC All Rights Reserved.
|
|
5
|
+
*
|
|
6
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
7
|
+
* found in the LICENSE file at https://angular.io/license
|
|
8
|
+
*/
|
|
9
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
10
|
+
exports.HammerGesturesMigration = void 0;
|
|
11
|
+
const core_1 = require("@angular-devkit/core");
|
|
12
|
+
const schematics_1 = require("@angular/cdk/schematics");
|
|
13
|
+
const change_1 = require("@schematics/angular/utility/change");
|
|
14
|
+
const fs_1 = require("fs");
|
|
15
|
+
const ts = require("typescript");
|
|
16
|
+
const find_hammer_script_tags_1 = require("./find-hammer-script-tags");
|
|
17
|
+
const find_main_module_1 = require("./find-main-module");
|
|
18
|
+
const hammer_template_check_1 = require("./hammer-template-check");
|
|
19
|
+
const import_manager_1 = require("./import-manager");
|
|
20
|
+
const remove_array_element_1 = require("./remove-array-element");
|
|
21
|
+
const remove_element_from_html_1 = require("./remove-element-from-html");
|
|
22
|
+
const GESTURE_CONFIG_CLASS_NAME = 'GestureConfig';
|
|
23
|
+
const GESTURE_CONFIG_FILE_NAME = 'gesture-config';
|
|
24
|
+
const GESTURE_CONFIG_TEMPLATE_PATH = './gesture-config.template';
|
|
25
|
+
const HAMMER_CONFIG_TOKEN_NAME = 'HAMMER_GESTURE_CONFIG';
|
|
26
|
+
const HAMMER_CONFIG_TOKEN_MODULE = '@angular/platform-browser';
|
|
27
|
+
const HAMMER_MODULE_NAME = 'HammerModule';
|
|
28
|
+
const HAMMER_MODULE_IMPORT = '@angular/platform-browser';
|
|
29
|
+
const HAMMER_MODULE_SPECIFIER = 'hammerjs';
|
|
30
|
+
const CANNOT_REMOVE_REFERENCE_ERROR = `Cannot remove reference to "GestureConfig". Please remove manually.`;
|
|
31
|
+
class HammerGesturesMigration extends schematics_1.DevkitMigration {
|
|
32
|
+
constructor() {
|
|
33
|
+
super(...arguments);
|
|
34
|
+
// Only enable this rule if the migration targets v9 or v10 and is running for a non-test
|
|
35
|
+
// target. We cannot migrate test targets since they have a limited scope
|
|
36
|
+
// (in regards to source files) and therefore the HammerJS usage detection can be incorrect.
|
|
37
|
+
this.enabled = (this.targetVersion === schematics_1.TargetVersion.V9 || this.targetVersion === schematics_1.TargetVersion.V10) &&
|
|
38
|
+
!this.context.isTestTarget;
|
|
39
|
+
this._printer = ts.createPrinter();
|
|
40
|
+
this._importManager = new import_manager_1.ImportManager(this.fileSystem, this._printer);
|
|
41
|
+
this._nodeFailures = [];
|
|
42
|
+
/**
|
|
43
|
+
* Whether custom HammerJS events provided by the Material gesture
|
|
44
|
+
* config are used in a template.
|
|
45
|
+
*/
|
|
46
|
+
this._customEventsUsedInTemplate = false;
|
|
47
|
+
/** Whether standard HammerJS events are used in a template. */
|
|
48
|
+
this._standardEventsUsedInTemplate = false;
|
|
49
|
+
/** Whether HammerJS is accessed at runtime. */
|
|
50
|
+
this._usedInRuntime = false;
|
|
51
|
+
/**
|
|
52
|
+
* List of imports that make "hammerjs" available globally. We keep track of these
|
|
53
|
+
* since we might need to remove them if Hammer is not used.
|
|
54
|
+
*/
|
|
55
|
+
this._installImports = [];
|
|
56
|
+
/**
|
|
57
|
+
* List of identifiers which resolve to the gesture config from Angular Material.
|
|
58
|
+
*/
|
|
59
|
+
this._gestureConfigReferences = [];
|
|
60
|
+
/**
|
|
61
|
+
* List of identifiers which resolve to the "HAMMER_GESTURE_CONFIG" token from
|
|
62
|
+
* "@angular/platform-browser".
|
|
63
|
+
*/
|
|
64
|
+
this._hammerConfigTokenReferences = [];
|
|
65
|
+
/**
|
|
66
|
+
* List of identifiers which resolve to the "HammerModule" from
|
|
67
|
+
* "@angular/platform-browser".
|
|
68
|
+
*/
|
|
69
|
+
this._hammerModuleReferences = [];
|
|
70
|
+
/**
|
|
71
|
+
* List of identifiers that have been deleted from source files. This can be
|
|
72
|
+
* used to determine if certain imports are still used or not.
|
|
73
|
+
*/
|
|
74
|
+
this._deletedIdentifiers = [];
|
|
75
|
+
}
|
|
76
|
+
visitTemplate(template) {
|
|
77
|
+
if (!this._customEventsUsedInTemplate || !this._standardEventsUsedInTemplate) {
|
|
78
|
+
const { standardEvents, customEvents } = hammer_template_check_1.isHammerJsUsedInTemplate(template.content);
|
|
79
|
+
this._customEventsUsedInTemplate = this._customEventsUsedInTemplate || customEvents;
|
|
80
|
+
this._standardEventsUsedInTemplate = this._standardEventsUsedInTemplate || standardEvents;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
visitNode(node) {
|
|
84
|
+
this._checkHammerImports(node);
|
|
85
|
+
this._checkForRuntimeHammerUsage(node);
|
|
86
|
+
this._checkForMaterialGestureConfig(node);
|
|
87
|
+
this._checkForHammerGestureConfigToken(node);
|
|
88
|
+
this._checkForHammerModuleReference(node);
|
|
89
|
+
}
|
|
90
|
+
postAnalysis() {
|
|
91
|
+
// Walk through all hammer config token references and check if there
|
|
92
|
+
// is a potential custom gesture config setup.
|
|
93
|
+
const hasCustomGestureConfigSetup = this._hammerConfigTokenReferences.some(r => this._checkForCustomGestureConfigSetup(r));
|
|
94
|
+
const usedInTemplate = this._standardEventsUsedInTemplate || this._customEventsUsedInTemplate;
|
|
95
|
+
/*
|
|
96
|
+
Possible scenarios and how the migration should change the project:
|
|
97
|
+
1. We detect that a custom HammerJS gesture config is set up:
|
|
98
|
+
- Remove references to the Material gesture config if no HammerJS event is used.
|
|
99
|
+
- Print a warning about ambiguous configuration that cannot be handled completely
|
|
100
|
+
if there are references to the Material gesture config.
|
|
101
|
+
2. We detect that HammerJS is only used programmatically:
|
|
102
|
+
- Remove references to GestureConfig of Material.
|
|
103
|
+
- Remove references to the "HammerModule" if present.
|
|
104
|
+
3. We detect that standard HammerJS events are used in a template:
|
|
105
|
+
- Set up the "HammerModule" from platform-browser.
|
|
106
|
+
- Remove all gesture config references.
|
|
107
|
+
4. We detect that custom HammerJS events provided by the Material gesture
|
|
108
|
+
config are used.
|
|
109
|
+
- Copy the Material gesture config into the app.
|
|
110
|
+
- Rewrite all gesture config references to the newly copied one.
|
|
111
|
+
- Set up the new gesture config in the root app module.
|
|
112
|
+
- Set up the "HammerModule" from platform-browser.
|
|
113
|
+
4. We detect no HammerJS usage at all:
|
|
114
|
+
- Remove Hammer imports
|
|
115
|
+
- Remove Material gesture config references
|
|
116
|
+
- Remove HammerModule setup if present.
|
|
117
|
+
- Remove Hammer script imports in "index.html" files.
|
|
118
|
+
*/
|
|
119
|
+
if (hasCustomGestureConfigSetup) {
|
|
120
|
+
// If a custom gesture config is provided, we always assume that HammerJS is used.
|
|
121
|
+
HammerGesturesMigration.globalUsesHammer = true;
|
|
122
|
+
if (!usedInTemplate && this._gestureConfigReferences.length) {
|
|
123
|
+
// If the Angular Material gesture events are not used and we found a custom
|
|
124
|
+
// gesture config, we can safely remove references to the Material gesture config
|
|
125
|
+
// since events provided by the Material gesture config are guaranteed to be unused.
|
|
126
|
+
this._removeMaterialGestureConfigSetup();
|
|
127
|
+
this.printInfo('The HammerJS v9 migration for Angular Components detected that HammerJS is ' +
|
|
128
|
+
'manually set up in combination with references to the Angular Material gesture ' +
|
|
129
|
+
'config. This target cannot be migrated completely, but all references to the ' +
|
|
130
|
+
'deprecated Angular Material gesture have been removed. Read more here: ' +
|
|
131
|
+
'https://git.io/ng-material-v9-hammer-ambiguous-usage');
|
|
132
|
+
}
|
|
133
|
+
else if (usedInTemplate && this._gestureConfigReferences.length) {
|
|
134
|
+
// Since there is a reference to the Angular Material gesture config, and we detected
|
|
135
|
+
// usage of a gesture event that could be provided by Angular Material, we *cannot*
|
|
136
|
+
// automatically remove references. This is because we do *not* know whether the
|
|
137
|
+
// event is actually provided by the custom config or by the Material config.
|
|
138
|
+
this.printInfo('The HammerJS v9 migration for Angular Components detected that HammerJS is ' +
|
|
139
|
+
'manually set up in combination with references to the Angular Material gesture ' +
|
|
140
|
+
'config. This target cannot be migrated completely. Please manually remove ' +
|
|
141
|
+
'references to the deprecated Angular Material gesture config. Read more here: ' +
|
|
142
|
+
'https://git.io/ng-material-v9-hammer-ambiguous-usage');
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
else if (this._usedInRuntime || usedInTemplate) {
|
|
146
|
+
// We keep track of whether Hammer is used globally. This is necessary because we
|
|
147
|
+
// want to only remove Hammer from the "package.json" if it is not used in any project
|
|
148
|
+
// target. Just because it isn't used in one target doesn't mean that we can safely
|
|
149
|
+
// remove the dependency.
|
|
150
|
+
HammerGesturesMigration.globalUsesHammer = true;
|
|
151
|
+
// If hammer is only used at runtime, we don't need the gesture config or "HammerModule"
|
|
152
|
+
// and can remove it (along with the hammer config token import if no longer needed).
|
|
153
|
+
if (!usedInTemplate) {
|
|
154
|
+
this._removeMaterialGestureConfigSetup();
|
|
155
|
+
this._removeHammerModuleReferences();
|
|
156
|
+
}
|
|
157
|
+
else if (this._standardEventsUsedInTemplate && !this._customEventsUsedInTemplate) {
|
|
158
|
+
this._setupHammerWithStandardEvents();
|
|
159
|
+
}
|
|
160
|
+
else {
|
|
161
|
+
this._setupHammerWithCustomEvents();
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
else {
|
|
165
|
+
this._removeHammerSetup();
|
|
166
|
+
}
|
|
167
|
+
// Record the changes collected in the import manager. Changes need to be applied
|
|
168
|
+
// once the import manager registered all import modifications. This avoids collisions.
|
|
169
|
+
this._importManager.recordChanges();
|
|
170
|
+
// Create migration failures that will be printed by the update-tool on migration
|
|
171
|
+
// completion. We need special logic for updating failure positions to reflect
|
|
172
|
+
// the new source file after modifications from the import manager.
|
|
173
|
+
this.failures.push(...this._createMigrationFailures());
|
|
174
|
+
// The template check for HammerJS events is not completely reliable as the event
|
|
175
|
+
// output could also be from a component having an output named similarly to a known
|
|
176
|
+
// hammerjs event (e.g. "@Output() slide"). The usage is therefore somewhat ambiguous
|
|
177
|
+
// and we want to print a message that developers might be able to remove Hammer manually.
|
|
178
|
+
if (!hasCustomGestureConfigSetup && !this._usedInRuntime && usedInTemplate) {
|
|
179
|
+
this.printInfo('The HammerJS v9 migration for Angular Components migrated the ' +
|
|
180
|
+
'project to keep HammerJS installed, but detected ambiguous usage of HammerJS. Please ' +
|
|
181
|
+
'manually check if you can remove HammerJS from your application. More details: ' +
|
|
182
|
+
'https://git.io/ng-material-v9-hammer-ambiguous-usage');
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Sets up the hammer gesture config in the current project. To achieve this, the
|
|
187
|
+
* following steps are performed:
|
|
188
|
+
* 1) Create copy of Angular Material gesture config.
|
|
189
|
+
* 2) Rewrite all references to the Angular Material gesture config to the
|
|
190
|
+
* new gesture config.
|
|
191
|
+
* 3) Setup the HAMMER_GESTURE_CONFIG in the root app module (if not done already).
|
|
192
|
+
* 4) Setup the "HammerModule" in the root app module (if not done already).
|
|
193
|
+
*/
|
|
194
|
+
_setupHammerWithCustomEvents() {
|
|
195
|
+
const project = this.context.project;
|
|
196
|
+
const sourceRoot = this.fileSystem.resolve(project.sourceRoot || project.root);
|
|
197
|
+
const newConfigPath = core_1.join(sourceRoot, this._getAvailableGestureConfigFileName(sourceRoot));
|
|
198
|
+
// Copy gesture config template into the CLI project.
|
|
199
|
+
this.fileSystem.create(newConfigPath, fs_1.readFileSync(require.resolve(GESTURE_CONFIG_TEMPLATE_PATH), 'utf8'));
|
|
200
|
+
// Replace all Material gesture config references to resolve to the
|
|
201
|
+
// newly copied gesture config.
|
|
202
|
+
this._gestureConfigReferences.forEach(i => {
|
|
203
|
+
const filePath = this.fileSystem.resolve(i.node.getSourceFile().fileName);
|
|
204
|
+
return this._replaceGestureConfigReference(i, GESTURE_CONFIG_CLASS_NAME, getModuleSpecifier(newConfigPath, filePath));
|
|
205
|
+
});
|
|
206
|
+
// Setup the gesture config provider and the "HammerModule" in the root module
|
|
207
|
+
// if not done already. The "HammerModule" is needed in v9 since it enables the
|
|
208
|
+
// Hammer event plugin that was previously enabled by default in v8.
|
|
209
|
+
this._setupNewGestureConfigInRootModule(newConfigPath);
|
|
210
|
+
this._setupHammerModuleInRootModule();
|
|
211
|
+
}
|
|
212
|
+
/**
|
|
213
|
+
* Sets up the standard hammer module in the project and removes all
|
|
214
|
+
* references to the deprecated Angular Material gesture config.
|
|
215
|
+
*/
|
|
216
|
+
_setupHammerWithStandardEvents() {
|
|
217
|
+
// Setup the HammerModule. The HammerModule enables support for
|
|
218
|
+
// the standard HammerJS events.
|
|
219
|
+
this._setupHammerModuleInRootModule();
|
|
220
|
+
this._removeMaterialGestureConfigSetup();
|
|
221
|
+
}
|
|
222
|
+
/**
|
|
223
|
+
* Removes Hammer from the current project. The following steps are performed:
|
|
224
|
+
* 1) Delete all TypeScript imports to "hammerjs".
|
|
225
|
+
* 2) Remove references to the Angular Material gesture config.
|
|
226
|
+
* 3) Remove "hammerjs" from all index HTML files of the current project.
|
|
227
|
+
*/
|
|
228
|
+
_removeHammerSetup() {
|
|
229
|
+
this._installImports.forEach(i => this._importManager.deleteImportByDeclaration(i));
|
|
230
|
+
this._removeMaterialGestureConfigSetup();
|
|
231
|
+
this._removeHammerModuleReferences();
|
|
232
|
+
this._removeHammerFromIndexFile();
|
|
233
|
+
}
|
|
234
|
+
/**
|
|
235
|
+
* Removes the gesture config setup by deleting all found references to the Angular
|
|
236
|
+
* Material gesture config. Additionally, unused imports to the hammer gesture config
|
|
237
|
+
* token from "@angular/platform-browser" will be removed as well.
|
|
238
|
+
*/
|
|
239
|
+
_removeMaterialGestureConfigSetup() {
|
|
240
|
+
this._gestureConfigReferences.forEach(r => this._removeGestureConfigReference(r));
|
|
241
|
+
this._hammerConfigTokenReferences.forEach(r => {
|
|
242
|
+
if (r.isImport) {
|
|
243
|
+
this._removeHammerConfigTokenImportIfUnused(r);
|
|
244
|
+
}
|
|
245
|
+
});
|
|
246
|
+
}
|
|
247
|
+
/** Removes all references to the "HammerModule" from "@angular/platform-browser". */
|
|
248
|
+
_removeHammerModuleReferences() {
|
|
249
|
+
this._hammerModuleReferences.forEach(({ node, isImport, importData }) => {
|
|
250
|
+
const sourceFile = node.getSourceFile();
|
|
251
|
+
const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));
|
|
252
|
+
// Only remove the import for the HammerModule if the module has been accessed
|
|
253
|
+
// through a non-namespaced identifier access.
|
|
254
|
+
if (!isNamespacedIdentifierAccess(node)) {
|
|
255
|
+
this._importManager.deleteNamedBindingImport(sourceFile, HAMMER_MODULE_NAME, importData.moduleName);
|
|
256
|
+
}
|
|
257
|
+
// For references from within an import, we do not need to do anything other than
|
|
258
|
+
// removing the import. For other references, we remove the import and the actual
|
|
259
|
+
// identifier in the module imports.
|
|
260
|
+
if (isImport) {
|
|
261
|
+
return;
|
|
262
|
+
}
|
|
263
|
+
// If the "HammerModule" is referenced within an array literal, we can
|
|
264
|
+
// remove the element easily. Otherwise if it's outside of an array literal,
|
|
265
|
+
// we need to replace the reference with an empty object literal w/ todo to
|
|
266
|
+
// not break the application.
|
|
267
|
+
if (ts.isArrayLiteralExpression(node.parent)) {
|
|
268
|
+
// Removes the "HammerModule" from the parent array expression. Removes
|
|
269
|
+
// the trailing comma token if present.
|
|
270
|
+
remove_array_element_1.removeElementFromArrayExpression(node, recorder);
|
|
271
|
+
}
|
|
272
|
+
else {
|
|
273
|
+
recorder.remove(node.getStart(), node.getWidth());
|
|
274
|
+
recorder.insertRight(node.getStart(), `/* TODO: remove */ {}`);
|
|
275
|
+
this._nodeFailures.push({
|
|
276
|
+
node: node,
|
|
277
|
+
message: 'Unable to delete reference to "HammerModule".',
|
|
278
|
+
});
|
|
279
|
+
}
|
|
280
|
+
});
|
|
281
|
+
}
|
|
282
|
+
/**
|
|
283
|
+
* Checks if the given node is a reference to the hammer gesture config
|
|
284
|
+
* token from platform-browser. If so, keeps track of the reference.
|
|
285
|
+
*/
|
|
286
|
+
_checkForHammerGestureConfigToken(node) {
|
|
287
|
+
if (ts.isIdentifier(node)) {
|
|
288
|
+
const importData = schematics_1.getImportOfIdentifier(node, this.typeChecker);
|
|
289
|
+
if (importData && importData.symbolName === HAMMER_CONFIG_TOKEN_NAME &&
|
|
290
|
+
importData.moduleName === HAMMER_CONFIG_TOKEN_MODULE) {
|
|
291
|
+
this._hammerConfigTokenReferences.push({ node, importData, isImport: ts.isImportSpecifier(node.parent) });
|
|
292
|
+
}
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
/**
|
|
296
|
+
* Checks if the given node is a reference to the HammerModule from
|
|
297
|
+
* "@angular/platform-browser". If so, keeps track of the reference.
|
|
298
|
+
*/
|
|
299
|
+
_checkForHammerModuleReference(node) {
|
|
300
|
+
if (ts.isIdentifier(node)) {
|
|
301
|
+
const importData = schematics_1.getImportOfIdentifier(node, this.typeChecker);
|
|
302
|
+
if (importData && importData.symbolName === HAMMER_MODULE_NAME &&
|
|
303
|
+
importData.moduleName === HAMMER_MODULE_IMPORT) {
|
|
304
|
+
this._hammerModuleReferences.push({ node, importData, isImport: ts.isImportSpecifier(node.parent) });
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
/**
|
|
309
|
+
* Checks if the given node is an import to the HammerJS package. Imports to
|
|
310
|
+
* HammerJS which load specific symbols from the package are considered as
|
|
311
|
+
* runtime usage of Hammer. e.g. `import {Symbol} from "hammerjs";`.
|
|
312
|
+
*/
|
|
313
|
+
_checkHammerImports(node) {
|
|
314
|
+
if (ts.isImportDeclaration(node) && ts.isStringLiteral(node.moduleSpecifier) &&
|
|
315
|
+
node.moduleSpecifier.text === HAMMER_MODULE_SPECIFIER) {
|
|
316
|
+
// If there is an import to HammerJS that imports symbols, or is namespaced
|
|
317
|
+
// (e.g. "import {A, B} from ..." or "import * as hammer from ..."), then we
|
|
318
|
+
// assume that some exports are used at runtime.
|
|
319
|
+
if (node.importClause &&
|
|
320
|
+
!(node.importClause.namedBindings && ts.isNamedImports(node.importClause.namedBindings) &&
|
|
321
|
+
node.importClause.namedBindings.elements.length === 0)) {
|
|
322
|
+
this._usedInRuntime = true;
|
|
323
|
+
}
|
|
324
|
+
else {
|
|
325
|
+
this._installImports.push(node);
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
/**
|
|
330
|
+
* Checks if the given node accesses the global "Hammer" symbol at runtime. If so,
|
|
331
|
+
* the migration rule state will be updated to reflect that Hammer is used at runtime.
|
|
332
|
+
*/
|
|
333
|
+
_checkForRuntimeHammerUsage(node) {
|
|
334
|
+
if (this._usedInRuntime) {
|
|
335
|
+
return;
|
|
336
|
+
}
|
|
337
|
+
// Detects usages of "window.Hammer".
|
|
338
|
+
if (ts.isPropertyAccessExpression(node) && node.name.text === 'Hammer') {
|
|
339
|
+
const originExpr = unwrapExpression(node.expression);
|
|
340
|
+
if (ts.isIdentifier(originExpr) && originExpr.text === 'window') {
|
|
341
|
+
this._usedInRuntime = true;
|
|
342
|
+
}
|
|
343
|
+
return;
|
|
344
|
+
}
|
|
345
|
+
// Detects usages of "window['Hammer']".
|
|
346
|
+
if (ts.isElementAccessExpression(node) && ts.isStringLiteral(node.argumentExpression) &&
|
|
347
|
+
node.argumentExpression.text === 'Hammer') {
|
|
348
|
+
const originExpr = unwrapExpression(node.expression);
|
|
349
|
+
if (ts.isIdentifier(originExpr) && originExpr.text === 'window') {
|
|
350
|
+
this._usedInRuntime = true;
|
|
351
|
+
}
|
|
352
|
+
return;
|
|
353
|
+
}
|
|
354
|
+
// Handles usages of plain identifier with the name "Hammer". These usage
|
|
355
|
+
// are valid if they resolve to "@types/hammerjs". e.g. "new Hammer(myElement)".
|
|
356
|
+
if (ts.isIdentifier(node) && node.text === 'Hammer' &&
|
|
357
|
+
!ts.isPropertyAccessExpression(node.parent) && !ts.isElementAccessExpression(node.parent)) {
|
|
358
|
+
const symbol = this._getDeclarationSymbolOfNode(node);
|
|
359
|
+
if (symbol && symbol.valueDeclaration &&
|
|
360
|
+
symbol.valueDeclaration.getSourceFile().fileName.includes('@types/hammerjs')) {
|
|
361
|
+
this._usedInRuntime = true;
|
|
362
|
+
}
|
|
363
|
+
}
|
|
364
|
+
}
|
|
365
|
+
/**
|
|
366
|
+
* Checks if the given node references the gesture config from Angular Material.
|
|
367
|
+
* If so, we keep track of the found symbol reference.
|
|
368
|
+
*/
|
|
369
|
+
_checkForMaterialGestureConfig(node) {
|
|
370
|
+
if (ts.isIdentifier(node)) {
|
|
371
|
+
const importData = schematics_1.getImportOfIdentifier(node, this.typeChecker);
|
|
372
|
+
if (importData && importData.symbolName === GESTURE_CONFIG_CLASS_NAME &&
|
|
373
|
+
importData.moduleName.startsWith('@angular/material/')) {
|
|
374
|
+
this._gestureConfigReferences.push({ node, importData, isImport: ts.isImportSpecifier(node.parent) });
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
/**
|
|
379
|
+
* Checks if the given Hammer gesture config token reference is part of an
|
|
380
|
+
* Angular provider definition that sets up a custom gesture config.
|
|
381
|
+
*/
|
|
382
|
+
_checkForCustomGestureConfigSetup(tokenRef) {
|
|
383
|
+
// Walk up the tree to look for a parent property assignment of the
|
|
384
|
+
// reference to the hammer gesture config token.
|
|
385
|
+
let propertyAssignment = tokenRef.node;
|
|
386
|
+
while (propertyAssignment && !ts.isPropertyAssignment(propertyAssignment)) {
|
|
387
|
+
propertyAssignment = propertyAssignment.parent;
|
|
388
|
+
}
|
|
389
|
+
if (!propertyAssignment || !ts.isPropertyAssignment(propertyAssignment) ||
|
|
390
|
+
getPropertyNameText(propertyAssignment.name) !== 'provide') {
|
|
391
|
+
return false;
|
|
392
|
+
}
|
|
393
|
+
const objectLiteralExpr = propertyAssignment.parent;
|
|
394
|
+
const matchingIdentifiers = findMatchingChildNodes(objectLiteralExpr, ts.isIdentifier);
|
|
395
|
+
// We naively assume that if there is a reference to the "GestureConfig" export
|
|
396
|
+
// from Angular Material in the provider literal, that the provider sets up the
|
|
397
|
+
// Angular Material gesture config.
|
|
398
|
+
return !this._gestureConfigReferences.some(r => matchingIdentifiers.includes(r.node));
|
|
399
|
+
}
|
|
400
|
+
/**
|
|
401
|
+
* Determines an available file name for the gesture config which should
|
|
402
|
+
* be stored in the specified file path.
|
|
403
|
+
*/
|
|
404
|
+
_getAvailableGestureConfigFileName(sourceRoot) {
|
|
405
|
+
if (!this.fileSystem.exists(core_1.join(sourceRoot, `${GESTURE_CONFIG_FILE_NAME}.ts`))) {
|
|
406
|
+
return `${GESTURE_CONFIG_FILE_NAME}.ts`;
|
|
407
|
+
}
|
|
408
|
+
let possibleName = `${GESTURE_CONFIG_FILE_NAME}-`;
|
|
409
|
+
let index = 1;
|
|
410
|
+
while (this.fileSystem.exists(core_1.join(sourceRoot, `${possibleName}-${index}.ts`))) {
|
|
411
|
+
index++;
|
|
412
|
+
}
|
|
413
|
+
return `${possibleName + index}.ts`;
|
|
414
|
+
}
|
|
415
|
+
/** Replaces a given gesture config reference with a new import. */
|
|
416
|
+
_replaceGestureConfigReference({ node, importData, isImport }, symbolName, moduleSpecifier) {
|
|
417
|
+
const sourceFile = node.getSourceFile();
|
|
418
|
+
const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));
|
|
419
|
+
// List of all identifiers referring to the gesture config in the current file. This
|
|
420
|
+
// allows us to add an import for the copied gesture configuration without generating a
|
|
421
|
+
// new identifier for the import to avoid collisions. i.e. "GestureConfig_1". The import
|
|
422
|
+
// manager checks for possible name collisions, but is able to ignore specific identifiers.
|
|
423
|
+
// We use this to ignore all references to the original Angular Material gesture config,
|
|
424
|
+
// because these will be replaced and therefore will not interfere.
|
|
425
|
+
const gestureIdentifiersInFile = this._getGestureConfigIdentifiersOfFile(sourceFile);
|
|
426
|
+
// If the parent of the identifier is accessed through a namespace, we can just
|
|
427
|
+
// import the new gesture config without rewriting the import declaration because
|
|
428
|
+
// the config has been imported through a namespaced import.
|
|
429
|
+
if (isNamespacedIdentifierAccess(node)) {
|
|
430
|
+
const newExpression = this._importManager.addImportToSourceFile(sourceFile, symbolName, moduleSpecifier, false, gestureIdentifiersInFile);
|
|
431
|
+
recorder.remove(node.parent.getStart(), node.parent.getWidth());
|
|
432
|
+
recorder.insertRight(node.parent.getStart(), this._printNode(newExpression, sourceFile));
|
|
433
|
+
return;
|
|
434
|
+
}
|
|
435
|
+
// Delete the old import to the "GestureConfig".
|
|
436
|
+
this._importManager.deleteNamedBindingImport(sourceFile, GESTURE_CONFIG_CLASS_NAME, importData.moduleName);
|
|
437
|
+
// If the current reference is not from inside of a import, we need to add a new
|
|
438
|
+
// import to the copied gesture config and replace the identifier. For references
|
|
439
|
+
// within an import, we do nothing but removing the actual import. This allows us
|
|
440
|
+
// to remove unused imports to the Material gesture config.
|
|
441
|
+
if (!isImport) {
|
|
442
|
+
const newExpression = this._importManager.addImportToSourceFile(sourceFile, symbolName, moduleSpecifier, false, gestureIdentifiersInFile);
|
|
443
|
+
recorder.remove(node.getStart(), node.getWidth());
|
|
444
|
+
recorder.insertRight(node.getStart(), this._printNode(newExpression, sourceFile));
|
|
445
|
+
}
|
|
446
|
+
}
|
|
447
|
+
/**
|
|
448
|
+
* Removes a given gesture config reference and its corresponding import from
|
|
449
|
+
* its containing source file. Imports will be always removed, but in some cases,
|
|
450
|
+
* where it's not guaranteed that a removal can be performed safely, we just
|
|
451
|
+
* create a migration failure (and add a TODO if possible).
|
|
452
|
+
*/
|
|
453
|
+
_removeGestureConfigReference({ node, importData, isImport }) {
|
|
454
|
+
const sourceFile = node.getSourceFile();
|
|
455
|
+
const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));
|
|
456
|
+
// Only remove the import for the gesture config if the gesture config has
|
|
457
|
+
// been accessed through a non-namespaced identifier access.
|
|
458
|
+
if (!isNamespacedIdentifierAccess(node)) {
|
|
459
|
+
this._importManager.deleteNamedBindingImport(sourceFile, GESTURE_CONFIG_CLASS_NAME, importData.moduleName);
|
|
460
|
+
}
|
|
461
|
+
// For references from within an import, we do not need to do anything other than
|
|
462
|
+
// removing the import. For other references, we remove the import and the reference
|
|
463
|
+
// identifier if used inside of a provider definition.
|
|
464
|
+
if (isImport) {
|
|
465
|
+
return;
|
|
466
|
+
}
|
|
467
|
+
const providerAssignment = node.parent;
|
|
468
|
+
// Only remove references to the gesture config which are part of a statically
|
|
469
|
+
// analyzable provider definition. We only support the common case of a gesture
|
|
470
|
+
// config provider definition where the config is set up through "useClass".
|
|
471
|
+
// Otherwise, it's not guaranteed that we can safely remove the provider definition.
|
|
472
|
+
if (!ts.isPropertyAssignment(providerAssignment) ||
|
|
473
|
+
getPropertyNameText(providerAssignment.name) !== 'useClass') {
|
|
474
|
+
this._nodeFailures.push({ node, message: CANNOT_REMOVE_REFERENCE_ERROR });
|
|
475
|
+
return;
|
|
476
|
+
}
|
|
477
|
+
const objectLiteralExpr = providerAssignment.parent;
|
|
478
|
+
const provideToken = objectLiteralExpr.properties.find((p) => ts.isPropertyAssignment(p) && getPropertyNameText(p.name) === 'provide');
|
|
479
|
+
// Do not remove the reference if the gesture config is not part of a provider definition,
|
|
480
|
+
// or if the provided toke is not referring to the known HAMMER_GESTURE_CONFIG token
|
|
481
|
+
// from platform-browser.
|
|
482
|
+
if (!provideToken || !this._isReferenceToHammerConfigToken(provideToken.initializer)) {
|
|
483
|
+
this._nodeFailures.push({ node, message: CANNOT_REMOVE_REFERENCE_ERROR });
|
|
484
|
+
return;
|
|
485
|
+
}
|
|
486
|
+
// Collect all nested identifiers which will be deleted. This helps us
|
|
487
|
+
// determining if we can remove imports for the "HAMMER_GESTURE_CONFIG" token.
|
|
488
|
+
this._deletedIdentifiers.push(...findMatchingChildNodes(objectLiteralExpr, ts.isIdentifier));
|
|
489
|
+
// In case the found provider definition is not part of an array literal,
|
|
490
|
+
// we cannot safely remove the provider. This is because it could be declared
|
|
491
|
+
// as a variable. e.g. "const gestureProvider = {provide: .., useClass: GestureConfig}".
|
|
492
|
+
// In that case, we just add an empty object literal with TODO and print a failure.
|
|
493
|
+
if (!ts.isArrayLiteralExpression(objectLiteralExpr.parent)) {
|
|
494
|
+
recorder.remove(objectLiteralExpr.getStart(), objectLiteralExpr.getWidth());
|
|
495
|
+
recorder.insertRight(objectLiteralExpr.getStart(), `/* TODO: remove */ {}`);
|
|
496
|
+
this._nodeFailures.push({
|
|
497
|
+
node: objectLiteralExpr,
|
|
498
|
+
message: `Unable to delete provider definition for "GestureConfig" completely. ` +
|
|
499
|
+
`Please clean up the provider.`
|
|
500
|
+
});
|
|
501
|
+
return;
|
|
502
|
+
}
|
|
503
|
+
// Removes the object literal from the parent array expression. Removes
|
|
504
|
+
// the trailing comma token if present.
|
|
505
|
+
remove_array_element_1.removeElementFromArrayExpression(objectLiteralExpr, recorder);
|
|
506
|
+
}
|
|
507
|
+
/** Removes the given hammer config token import if it is not used. */
|
|
508
|
+
_removeHammerConfigTokenImportIfUnused({ node, importData }) {
|
|
509
|
+
const sourceFile = node.getSourceFile();
|
|
510
|
+
const isTokenUsed = this._hammerConfigTokenReferences.some(r => !r.isImport && !isNamespacedIdentifierAccess(r.node) &&
|
|
511
|
+
r.node.getSourceFile() === sourceFile && !this._deletedIdentifiers.includes(r.node));
|
|
512
|
+
// We don't want to remove the import for the token if the token is
|
|
513
|
+
// still used somewhere.
|
|
514
|
+
if (!isTokenUsed) {
|
|
515
|
+
this._importManager.deleteNamedBindingImport(sourceFile, HAMMER_CONFIG_TOKEN_NAME, importData.moduleName);
|
|
516
|
+
}
|
|
517
|
+
}
|
|
518
|
+
/** Removes Hammer from all index HTML files of the current project. */
|
|
519
|
+
_removeHammerFromIndexFile() {
|
|
520
|
+
const indexFilePaths = schematics_1.getProjectIndexFiles(this.context.project);
|
|
521
|
+
indexFilePaths.forEach(filePath => {
|
|
522
|
+
if (!this.fileSystem.exists(filePath)) {
|
|
523
|
+
return;
|
|
524
|
+
}
|
|
525
|
+
const htmlContent = this.fileSystem.read(filePath);
|
|
526
|
+
const recorder = this.fileSystem.edit(filePath);
|
|
527
|
+
find_hammer_script_tags_1.findHammerScriptImportElements(htmlContent)
|
|
528
|
+
.forEach(el => remove_element_from_html_1.removeElementFromHtml(el, recorder));
|
|
529
|
+
});
|
|
530
|
+
}
|
|
531
|
+
/** Sets up the Hammer gesture config in the root module if needed. */
|
|
532
|
+
_setupNewGestureConfigInRootModule(gestureConfigPath) {
|
|
533
|
+
const { project } = this.context;
|
|
534
|
+
const mainFilePath = schematics_1.getProjectMainFile(project);
|
|
535
|
+
const rootModuleSymbol = this._getRootModuleSymbol(mainFilePath);
|
|
536
|
+
if (rootModuleSymbol === null || rootModuleSymbol.valueDeclaration === undefined) {
|
|
537
|
+
this.failures.push({
|
|
538
|
+
filePath: mainFilePath,
|
|
539
|
+
message: `Could not setup Hammer gestures in module. Please ` +
|
|
540
|
+
`manually ensure that the Hammer gesture config is set up.`,
|
|
541
|
+
});
|
|
542
|
+
return;
|
|
543
|
+
}
|
|
544
|
+
const sourceFile = rootModuleSymbol.valueDeclaration.getSourceFile();
|
|
545
|
+
const metadata = schematics_1.getDecoratorMetadata(sourceFile, 'NgModule', '@angular/core');
|
|
546
|
+
// If no "NgModule" definition is found inside the source file, we just do nothing.
|
|
547
|
+
if (!metadata.length) {
|
|
548
|
+
return;
|
|
549
|
+
}
|
|
550
|
+
const filePath = this.fileSystem.resolve(sourceFile.fileName);
|
|
551
|
+
const recorder = this.fileSystem.edit(filePath);
|
|
552
|
+
const providersField = schematics_1.getMetadataField(metadata[0], 'providers')[0];
|
|
553
|
+
const providerIdentifiers = providersField ? findMatchingChildNodes(providersField, ts.isIdentifier) : null;
|
|
554
|
+
const gestureConfigExpr = this._importManager.addImportToSourceFile(sourceFile, GESTURE_CONFIG_CLASS_NAME, getModuleSpecifier(gestureConfigPath, filePath), false, this._getGestureConfigIdentifiersOfFile(sourceFile));
|
|
555
|
+
const hammerConfigTokenExpr = this._importManager.addImportToSourceFile(sourceFile, HAMMER_CONFIG_TOKEN_NAME, HAMMER_CONFIG_TOKEN_MODULE);
|
|
556
|
+
const newProviderNode = ts.createObjectLiteral([
|
|
557
|
+
ts.createPropertyAssignment('provide', hammerConfigTokenExpr),
|
|
558
|
+
ts.createPropertyAssignment('useClass', gestureConfigExpr)
|
|
559
|
+
]);
|
|
560
|
+
// If the providers field exists and already contains references to the hammer gesture
|
|
561
|
+
// config token and the gesture config, we naively assume that the gesture config is
|
|
562
|
+
// already set up. We only want to add the gesture config provider if it is not set up.
|
|
563
|
+
if (!providerIdentifiers ||
|
|
564
|
+
!(this._hammerConfigTokenReferences.some(r => providerIdentifiers.includes(r.node)) &&
|
|
565
|
+
this._gestureConfigReferences.some(r => providerIdentifiers.includes(r.node)))) {
|
|
566
|
+
const symbolName = this._printNode(newProviderNode, sourceFile);
|
|
567
|
+
schematics_1.addSymbolToNgModuleMetadata(sourceFile, sourceFile.fileName, 'providers', symbolName, null)
|
|
568
|
+
.forEach(change => {
|
|
569
|
+
if (change instanceof change_1.InsertChange) {
|
|
570
|
+
recorder.insertRight(change.pos, change.toAdd);
|
|
571
|
+
}
|
|
572
|
+
});
|
|
573
|
+
}
|
|
574
|
+
}
|
|
575
|
+
/**
|
|
576
|
+
* Gets the TypeScript symbol of the root module by looking for the module
|
|
577
|
+
* bootstrap expression in the specified source file.
|
|
578
|
+
*/
|
|
579
|
+
_getRootModuleSymbol(mainFilePath) {
|
|
580
|
+
const mainFile = this.program.getSourceFile(mainFilePath);
|
|
581
|
+
if (!mainFile) {
|
|
582
|
+
return null;
|
|
583
|
+
}
|
|
584
|
+
const appModuleExpr = find_main_module_1.findMainModuleExpression(mainFile);
|
|
585
|
+
if (!appModuleExpr) {
|
|
586
|
+
return null;
|
|
587
|
+
}
|
|
588
|
+
const appModuleSymbol = this._getDeclarationSymbolOfNode(unwrapExpression(appModuleExpr));
|
|
589
|
+
if (!appModuleSymbol || !appModuleSymbol.valueDeclaration) {
|
|
590
|
+
return null;
|
|
591
|
+
}
|
|
592
|
+
return appModuleSymbol;
|
|
593
|
+
}
|
|
594
|
+
/** Sets up the "HammerModule" in the root module of the current project. */
|
|
595
|
+
_setupHammerModuleInRootModule() {
|
|
596
|
+
const { project } = this.context;
|
|
597
|
+
const mainFilePath = schematics_1.getProjectMainFile(project);
|
|
598
|
+
const rootModuleSymbol = this._getRootModuleSymbol(mainFilePath);
|
|
599
|
+
if (rootModuleSymbol === null || rootModuleSymbol.valueDeclaration === undefined) {
|
|
600
|
+
this.failures.push({
|
|
601
|
+
filePath: mainFilePath,
|
|
602
|
+
message: `Could not setup HammerModule. Please manually set up the "HammerModule" ` +
|
|
603
|
+
`from "@angular/platform-browser".`,
|
|
604
|
+
});
|
|
605
|
+
return;
|
|
606
|
+
}
|
|
607
|
+
const sourceFile = rootModuleSymbol.valueDeclaration.getSourceFile();
|
|
608
|
+
const metadata = schematics_1.getDecoratorMetadata(sourceFile, 'NgModule', '@angular/core');
|
|
609
|
+
if (!metadata.length) {
|
|
610
|
+
return;
|
|
611
|
+
}
|
|
612
|
+
const importsField = schematics_1.getMetadataField(metadata[0], 'imports')[0];
|
|
613
|
+
const importIdentifiers = importsField ? findMatchingChildNodes(importsField, ts.isIdentifier) : null;
|
|
614
|
+
const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));
|
|
615
|
+
const hammerModuleExpr = this._importManager.addImportToSourceFile(sourceFile, HAMMER_MODULE_NAME, HAMMER_MODULE_IMPORT);
|
|
616
|
+
// If the "HammerModule" is not already imported in the app module, we set it up
|
|
617
|
+
// by adding it to the "imports" field of the app module.
|
|
618
|
+
if (!importIdentifiers ||
|
|
619
|
+
!this._hammerModuleReferences.some(r => importIdentifiers.includes(r.node))) {
|
|
620
|
+
const symbolName = this._printNode(hammerModuleExpr, sourceFile);
|
|
621
|
+
schematics_1.addSymbolToNgModuleMetadata(sourceFile, sourceFile.fileName, 'imports', symbolName, null)
|
|
622
|
+
.forEach(change => {
|
|
623
|
+
if (change instanceof change_1.InsertChange) {
|
|
624
|
+
recorder.insertRight(change.pos, change.toAdd);
|
|
625
|
+
}
|
|
626
|
+
});
|
|
627
|
+
}
|
|
628
|
+
}
|
|
629
|
+
/** Prints a given node within the specified source file. */
|
|
630
|
+
_printNode(node, sourceFile) {
|
|
631
|
+
return this._printer.printNode(ts.EmitHint.Unspecified, node, sourceFile);
|
|
632
|
+
}
|
|
633
|
+
/** Gets all referenced gesture config identifiers of a given source file */
|
|
634
|
+
_getGestureConfigIdentifiersOfFile(sourceFile) {
|
|
635
|
+
return this._gestureConfigReferences.filter(d => d.node.getSourceFile() === sourceFile)
|
|
636
|
+
.map(d => d.node);
|
|
637
|
+
}
|
|
638
|
+
/** Gets the symbol that contains the value declaration of the specified node. */
|
|
639
|
+
_getDeclarationSymbolOfNode(node) {
|
|
640
|
+
const symbol = this.typeChecker.getSymbolAtLocation(node);
|
|
641
|
+
// Symbols can be aliases of the declaration symbol. e.g. in named import specifiers.
|
|
642
|
+
// We need to resolve the aliased symbol back to the declaration symbol.
|
|
643
|
+
// tslint:disable-next-line:no-bitwise
|
|
644
|
+
if (symbol && (symbol.flags & ts.SymbolFlags.Alias) !== 0) {
|
|
645
|
+
return this.typeChecker.getAliasedSymbol(symbol);
|
|
646
|
+
}
|
|
647
|
+
return symbol;
|
|
648
|
+
}
|
|
649
|
+
/**
|
|
650
|
+
* Checks whether the given expression resolves to a hammer gesture config
|
|
651
|
+
* token reference from "@angular/platform-browser".
|
|
652
|
+
*/
|
|
653
|
+
_isReferenceToHammerConfigToken(expr) {
|
|
654
|
+
const unwrapped = unwrapExpression(expr);
|
|
655
|
+
if (ts.isIdentifier(unwrapped)) {
|
|
656
|
+
return this._hammerConfigTokenReferences.some(r => r.node === unwrapped);
|
|
657
|
+
}
|
|
658
|
+
else if (ts.isPropertyAccessExpression(unwrapped)) {
|
|
659
|
+
return this._hammerConfigTokenReferences.some(r => r.node === unwrapped.name);
|
|
660
|
+
}
|
|
661
|
+
return false;
|
|
662
|
+
}
|
|
663
|
+
/**
|
|
664
|
+
* Creates migration failures of the collected node failures. The returned migration
|
|
665
|
+
* failures are updated to reflect the post-migration state of source files. Meaning
|
|
666
|
+
* that failure positions are corrected if source file modifications shifted lines.
|
|
667
|
+
*/
|
|
668
|
+
_createMigrationFailures() {
|
|
669
|
+
return this._nodeFailures.map(({ node, message }) => {
|
|
670
|
+
const sourceFile = node.getSourceFile();
|
|
671
|
+
const offset = node.getStart();
|
|
672
|
+
const position = ts.getLineAndCharacterOfPosition(sourceFile, node.getStart());
|
|
673
|
+
return {
|
|
674
|
+
position: this._importManager.correctNodePosition(node, offset, position),
|
|
675
|
+
message: message,
|
|
676
|
+
filePath: this.fileSystem.resolve(sourceFile.fileName),
|
|
677
|
+
};
|
|
678
|
+
});
|
|
679
|
+
}
|
|
680
|
+
/**
|
|
681
|
+
* Static migration rule method that will be called once all project targets
|
|
682
|
+
* have been migrated individually. This method can be used to make changes based
|
|
683
|
+
* on the analysis of the individual targets. For example: we only remove Hammer
|
|
684
|
+
* from the "package.json" if it is not used in *any* project target.
|
|
685
|
+
*/
|
|
686
|
+
static globalPostMigration(tree, context) {
|
|
687
|
+
// Always notify the developer that the Hammer v9 migration does not migrate tests.
|
|
688
|
+
context.logger.info('\n⚠ General notice: The HammerJS v9 migration for Angular Components is not able to ' +
|
|
689
|
+
'migrate tests. Please manually clean up tests in your project if they rely on ' +
|
|
690
|
+
(this.globalUsesHammer ? 'the deprecated Angular Material gesture config.' : 'HammerJS.'));
|
|
691
|
+
context.logger.info('Read more about migrating tests: https://git.io/ng-material-v9-hammer-migrate-tests');
|
|
692
|
+
if (!this.globalUsesHammer && this._removeHammerFromPackageJson(tree)) {
|
|
693
|
+
// Since Hammer has been removed from the workspace "package.json" file,
|
|
694
|
+
// we schedule a node package install task to refresh the lock file.
|
|
695
|
+
return { runPackageManager: true };
|
|
696
|
+
}
|
|
697
|
+
// Clean global state once the workspace has been migrated. This is technically
|
|
698
|
+
// not necessary in "ng update", but in tests we re-use the same rule class.
|
|
699
|
+
this.globalUsesHammer = false;
|
|
700
|
+
}
|
|
701
|
+
/**
|
|
702
|
+
* Removes the hammer package from the workspace "package.json".
|
|
703
|
+
* @returns Whether Hammer was set up and has been removed from the "package.json"
|
|
704
|
+
*/
|
|
705
|
+
static _removeHammerFromPackageJson(tree) {
|
|
706
|
+
if (!tree.exists('/package.json')) {
|
|
707
|
+
return false;
|
|
708
|
+
}
|
|
709
|
+
const packageJson = JSON.parse(tree.read('/package.json').toString('utf8'));
|
|
710
|
+
// We do not handle the case where someone manually added "hammerjs" to the dev dependencies.
|
|
711
|
+
if (packageJson.dependencies && packageJson.dependencies[HAMMER_MODULE_SPECIFIER]) {
|
|
712
|
+
delete packageJson.dependencies[HAMMER_MODULE_SPECIFIER];
|
|
713
|
+
tree.overwrite('/package.json', JSON.stringify(packageJson, null, 2));
|
|
714
|
+
return true;
|
|
715
|
+
}
|
|
716
|
+
return false;
|
|
717
|
+
}
|
|
718
|
+
}
|
|
719
|
+
exports.HammerGesturesMigration = HammerGesturesMigration;
|
|
720
|
+
/** Global state of whether Hammer is used in any analyzed project target. */
|
|
721
|
+
HammerGesturesMigration.globalUsesHammer = false;
|
|
722
|
+
/**
|
|
723
|
+
* Recursively unwraps a given expression if it is wrapped
|
|
724
|
+
* by parenthesis, type casts or type assertions.
|
|
725
|
+
*/
|
|
726
|
+
function unwrapExpression(node) {
|
|
727
|
+
if (ts.isParenthesizedExpression(node)) {
|
|
728
|
+
return unwrapExpression(node.expression);
|
|
729
|
+
}
|
|
730
|
+
else if (ts.isAsExpression(node)) {
|
|
731
|
+
return unwrapExpression(node.expression);
|
|
732
|
+
}
|
|
733
|
+
else if (ts.isTypeAssertion(node)) {
|
|
734
|
+
return unwrapExpression(node.expression);
|
|
735
|
+
}
|
|
736
|
+
return node;
|
|
737
|
+
}
|
|
738
|
+
/**
|
|
739
|
+
* Converts the specified path to a valid TypeScript module specifier which is
|
|
740
|
+
* relative to the given containing file.
|
|
741
|
+
*/
|
|
742
|
+
function getModuleSpecifier(newPath, containingFile) {
|
|
743
|
+
let result = core_1.relative(core_1.dirname(containingFile), newPath).replace(/\\/g, '/').replace(/\.ts$/, '');
|
|
744
|
+
if (!result.startsWith('.')) {
|
|
745
|
+
result = `./${result}`;
|
|
746
|
+
}
|
|
747
|
+
return result;
|
|
748
|
+
}
|
|
749
|
+
/**
|
|
750
|
+
* Gets the text of the given property name.
|
|
751
|
+
* @returns Text of the given property name. Null if not statically analyzable.
|
|
752
|
+
*/
|
|
753
|
+
function getPropertyNameText(node) {
|
|
754
|
+
if (ts.isIdentifier(node) || ts.isStringLiteralLike(node)) {
|
|
755
|
+
return node.text;
|
|
756
|
+
}
|
|
757
|
+
return null;
|
|
758
|
+
}
|
|
759
|
+
/** Checks whether the given identifier is part of a namespaced access. */
|
|
760
|
+
function isNamespacedIdentifierAccess(node) {
|
|
761
|
+
return ts.isQualifiedName(node.parent) || ts.isPropertyAccessExpression(node.parent);
|
|
762
|
+
}
|
|
763
|
+
/**
|
|
764
|
+
* Walks through the specified node and returns all child nodes which match the
|
|
765
|
+
* given predicate.
|
|
766
|
+
*/
|
|
767
|
+
function findMatchingChildNodes(parent, predicate) {
|
|
768
|
+
const result = [];
|
|
769
|
+
const visitNode = (node) => {
|
|
770
|
+
if (predicate(node)) {
|
|
771
|
+
result.push(node);
|
|
772
|
+
}
|
|
773
|
+
ts.forEachChild(node, visitNode);
|
|
774
|
+
};
|
|
775
|
+
ts.forEachChild(parent, visitNode);
|
|
776
|
+
return result;
|
|
777
|
+
}
|
|
778
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hammer-gestures-migration.js","sourceRoot":"","sources":["../../../../../../../../../src/material/schematics/ng-update/migrations/hammer-gestures-v9/hammer-gestures-migration.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {\n  join,\n  Path,\n  relative,\n  dirname\n} from '@angular-devkit/core';\nimport {SchematicContext, Tree} from '@angular-devkit/schematics';\nimport {\n  addSymbolToNgModuleMetadata,\n  DevkitMigration,\n  getDecoratorMetadata,\n  getImportOfIdentifier,\n  getMetadataField,\n  getProjectIndexFiles,\n  getProjectMainFile,\n  Import,\n  MigrationFailure,\n  PostMigrationAction,\n  ResolvedResource,\n  TargetVersion,\n} from '@angular/cdk/schematics';\nimport {InsertChange} from '@schematics/angular/utility/change';\nimport {readFileSync} from 'fs';\nimport * as ts from 'typescript';\n\nimport {findHammerScriptImportElements} from './find-hammer-script-tags';\nimport {findMainModuleExpression} from './find-main-module';\nimport {isHammerJsUsedInTemplate} from './hammer-template-check';\nimport {ImportManager} from './import-manager';\nimport {removeElementFromArrayExpression} from './remove-array-element';\nimport {removeElementFromHtml} from './remove-element-from-html';\n\nconst GESTURE_CONFIG_CLASS_NAME = 'GestureConfig';\nconst GESTURE_CONFIG_FILE_NAME = 'gesture-config';\nconst GESTURE_CONFIG_TEMPLATE_PATH = './gesture-config.template';\n\nconst HAMMER_CONFIG_TOKEN_NAME = 'HAMMER_GESTURE_CONFIG';\nconst HAMMER_CONFIG_TOKEN_MODULE = '@angular/platform-browser';\n\nconst HAMMER_MODULE_NAME = 'HammerModule';\nconst HAMMER_MODULE_IMPORT = '@angular/platform-browser';\n\nconst HAMMER_MODULE_SPECIFIER = 'hammerjs';\n\nconst CANNOT_REMOVE_REFERENCE_ERROR =\n    `Cannot remove reference to \"GestureConfig\". Please remove manually.`;\n\ninterface IdentifierReference {\n  node: ts.Identifier;\n  importData: Import;\n  isImport: boolean;\n}\n\ninterface PackageJson {\n  dependencies: Record<string, string>;\n}\n\nexport class HammerGesturesMigration extends DevkitMigration<null> {\n  // Only enable this rule if the migration targets v9 or v10 and is running for a non-test\n  // target. We cannot migrate test targets since they have a limited scope\n  // (in regards to source files) and therefore the HammerJS usage detection can be incorrect.\n  enabled =\n      (this.targetVersion === TargetVersion.V9 || this.targetVersion === TargetVersion.V10) &&\n      !this.context.isTestTarget;\n\n  private _printer = ts.createPrinter();\n  private _importManager = new ImportManager(this.fileSystem, this._printer);\n  private _nodeFailures: {node: ts.Node, message: string}[] = [];\n\n  /**\n   * Whether custom HammerJS events provided by the Material gesture\n   * config are used in a template.\n   */\n  private _customEventsUsedInTemplate = false;\n\n  /** Whether standard HammerJS events are used in a template. */\n  private _standardEventsUsedInTemplate = false;\n\n  /** Whether HammerJS is accessed at runtime. */\n  private _usedInRuntime = false;\n\n  /**\n   * List of imports that make \"hammerjs\" available globally. We keep track of these\n   * since we might need to remove them if Hammer is not used.\n   */\n  private _installImports: ts.ImportDeclaration[] = [];\n\n  /**\n   * List of identifiers which resolve to the gesture config from Angular Material.\n   */\n  private _gestureConfigReferences: IdentifierReference[] = [];\n\n  /**\n   * List of identifiers which resolve to the \"HAMMER_GESTURE_CONFIG\" token from\n   * \"@angular/platform-browser\".\n   */\n  private _hammerConfigTokenReferences: IdentifierReference[] = [];\n\n  /**\n   * List of identifiers which resolve to the \"HammerModule\" from\n   * \"@angular/platform-browser\".\n   */\n  private _hammerModuleReferences: IdentifierReference[] = [];\n\n  /**\n   * List of identifiers that have been deleted from source files. This can be\n   * used to determine if certain imports are still used or not.\n   */\n  private _deletedIdentifiers: ts.Identifier[] = [];\n\n  visitTemplate(template: ResolvedResource): void {\n    if (!this._customEventsUsedInTemplate || !this._standardEventsUsedInTemplate) {\n      const {standardEvents, customEvents} = isHammerJsUsedInTemplate(template.content);\n      this._customEventsUsedInTemplate = this._customEventsUsedInTemplate || customEvents;\n      this._standardEventsUsedInTemplate = this._standardEventsUsedInTemplate || standardEvents;\n    }\n  }\n\n  visitNode(node: ts.Node): void {\n    this._checkHammerImports(node);\n    this._checkForRuntimeHammerUsage(node);\n    this._checkForMaterialGestureConfig(node);\n    this._checkForHammerGestureConfigToken(node);\n    this._checkForHammerModuleReference(node);\n  }\n\n  postAnalysis(): void {\n    // Walk through all hammer config token references and check if there\n    // is a potential custom gesture config setup.\n    const hasCustomGestureConfigSetup =\n        this._hammerConfigTokenReferences.some(r => this._checkForCustomGestureConfigSetup(r));\n    const usedInTemplate = this._standardEventsUsedInTemplate || this._customEventsUsedInTemplate;\n\n    /*\n      Possible scenarios and how the migration should change the project:\n        1. We detect that a custom HammerJS gesture config is set up:\n            - Remove references to the Material gesture config if no HammerJS event is used.\n            - Print a warning about ambiguous configuration that cannot be handled completely\n              if there are references to the Material gesture config.\n        2. We detect that HammerJS is only used programmatically:\n            - Remove references to GestureConfig of Material.\n            - Remove references to the \"HammerModule\" if present.\n        3. We detect that standard HammerJS events are used in a template:\n            - Set up the \"HammerModule\" from platform-browser.\n            - Remove all gesture config references.\n        4. We detect that custom HammerJS events provided by the Material gesture\n           config are used.\n            - Copy the Material gesture config into the app.\n            - Rewrite all gesture config references to the newly copied one.\n            - Set up the new gesture config in the root app module.\n            - Set up the \"HammerModule\" from platform-browser.\n        4. We detect no HammerJS usage at all:\n            - Remove Hammer imports\n            - Remove Material gesture config references\n            - Remove HammerModule setup if present.\n            - Remove Hammer script imports in \"index.html\" files.\n    */\n\n    if (hasCustomGestureConfigSetup) {\n      // If a custom gesture config is provided, we always assume that HammerJS is used.\n      HammerGesturesMigration.globalUsesHammer = true;\n      if (!usedInTemplate && this._gestureConfigReferences.length) {\n        // If the Angular Material gesture events are not used and we found a custom\n        // gesture config, we can safely remove references to the Material gesture config\n        // since events provided by the Material gesture config are guaranteed to be unused.\n        this._removeMaterialGestureConfigSetup();\n        this.printInfo(\n            'The HammerJS v9 migration for Angular Components detected that HammerJS is ' +\n            'manually set up in combination with references to the Angular Material gesture ' +\n            'config. This target cannot be migrated completely, but all references to the ' +\n            'deprecated Angular Material gesture have been removed. Read more here: ' +\n            'https://git.io/ng-material-v9-hammer-ambiguous-usage');\n      } else if (usedInTemplate && this._gestureConfigReferences.length) {\n        // Since there is a reference to the Angular Material gesture config, and we detected\n        // usage of a gesture event that could be provided by Angular Material, we *cannot*\n        // automatically remove references. This is because we do *not* know whether the\n        // event is actually provided by the custom config or by the Material config.\n        this.printInfo(\n            'The HammerJS v9 migration for Angular Components detected that HammerJS is ' +\n            'manually set up in combination with references to the Angular Material gesture ' +\n            'config. This target cannot be migrated completely. Please manually remove ' +\n            'references to the deprecated Angular Material gesture config. Read more here: ' +\n            'https://git.io/ng-material-v9-hammer-ambiguous-usage');\n      }\n    } else if (this._usedInRuntime || usedInTemplate) {\n      // We keep track of whether Hammer is used globally. This is necessary because we\n      // want to only remove Hammer from the \"package.json\" if it is not used in any project\n      // target. Just because it isn't used in one target doesn't mean that we can safely\n      // remove the dependency.\n      HammerGesturesMigration.globalUsesHammer = true;\n\n      // If hammer is only used at runtime, we don't need the gesture config or \"HammerModule\"\n      // and can remove it (along with the hammer config token import if no longer needed).\n      if (!usedInTemplate) {\n        this._removeMaterialGestureConfigSetup();\n        this._removeHammerModuleReferences();\n      } else if (this._standardEventsUsedInTemplate && !this._customEventsUsedInTemplate) {\n        this._setupHammerWithStandardEvents();\n      } else {\n        this._setupHammerWithCustomEvents();\n      }\n    } else {\n      this._removeHammerSetup();\n    }\n\n    // Record the changes collected in the import manager. Changes need to be applied\n    // once the import manager registered all import modifications. This avoids collisions.\n    this._importManager.recordChanges();\n\n    // Create migration failures that will be printed by the update-tool on migration\n    // completion. We need special logic for updating failure positions to reflect\n    // the new source file after modifications from the import manager.\n    this.failures.push(...this._createMigrationFailures());\n\n    // The template check for HammerJS events is not completely reliable as the event\n    // output could also be from a component having an output named similarly to a known\n    // hammerjs event (e.g. \"@Output() slide\"). The usage is therefore somewhat ambiguous\n    // and we want to print a message that developers might be able to remove Hammer manually.\n    if (!hasCustomGestureConfigSetup && !this._usedInRuntime && usedInTemplate) {\n      this.printInfo(\n          'The HammerJS v9 migration for Angular Components migrated the ' +\n          'project to keep HammerJS installed, but detected ambiguous usage of HammerJS. Please ' +\n          'manually check if you can remove HammerJS from your application. More details: ' +\n          'https://git.io/ng-material-v9-hammer-ambiguous-usage');\n    }\n  }\n\n  /**\n   * Sets up the hammer gesture config in the current project. To achieve this, the\n   * following steps are performed:\n   *   1) Create copy of Angular Material gesture config.\n   *   2) Rewrite all references to the Angular Material gesture config to the\n   *      new gesture config.\n   *   3) Setup the HAMMER_GESTURE_CONFIG in the root app module (if not done already).\n   *   4) Setup the \"HammerModule\" in the root app module (if not done already).\n   */\n  private _setupHammerWithCustomEvents() {\n    const project = this.context.project;\n    const sourceRoot = this.fileSystem.resolve(project.sourceRoot || project.root);\n    const newConfigPath =\n        join(sourceRoot, this._getAvailableGestureConfigFileName(sourceRoot));\n\n    // Copy gesture config template into the CLI project.\n    this.fileSystem.create(\n        newConfigPath, readFileSync(require.resolve(GESTURE_CONFIG_TEMPLATE_PATH), 'utf8'));\n\n    // Replace all Material gesture config references to resolve to the\n    // newly copied gesture config.\n    this._gestureConfigReferences.forEach(i => {\n      const filePath = this.fileSystem.resolve(i.node.getSourceFile().fileName);\n      return this._replaceGestureConfigReference(i, GESTURE_CONFIG_CLASS_NAME,\n        getModuleSpecifier(newConfigPath, filePath));\n    });\n\n    // Setup the gesture config provider and the \"HammerModule\" in the root module\n    // if not done already. The \"HammerModule\" is needed in v9 since it enables the\n    // Hammer event plugin that was previously enabled by default in v8.\n    this._setupNewGestureConfigInRootModule(newConfigPath);\n    this._setupHammerModuleInRootModule();\n  }\n\n  /**\n   * Sets up the standard hammer module in the project and removes all\n   * references to the deprecated Angular Material gesture config.\n   */\n  private _setupHammerWithStandardEvents() {\n    // Setup the HammerModule. The HammerModule enables support for\n    // the standard HammerJS events.\n    this._setupHammerModuleInRootModule();\n    this._removeMaterialGestureConfigSetup();\n  }\n\n  /**\n   * Removes Hammer from the current project. The following steps are performed:\n   *   1) Delete all TypeScript imports to \"hammerjs\".\n   *   2) Remove references to the Angular Material gesture config.\n   *   3) Remove \"hammerjs\" from all index HTML files of the current project.\n   */\n  private _removeHammerSetup() {\n    this._installImports.forEach(i => this._importManager.deleteImportByDeclaration(i));\n\n    this._removeMaterialGestureConfigSetup();\n    this._removeHammerModuleReferences();\n    this._removeHammerFromIndexFile();\n  }\n\n  /**\n   * Removes the gesture config setup by deleting all found references to the Angular\n   * Material gesture config. Additionally, unused imports to the hammer gesture config\n   * token from \"@angular/platform-browser\" will be removed as well.\n   */\n  private _removeMaterialGestureConfigSetup() {\n    this._gestureConfigReferences.forEach(r => this._removeGestureConfigReference(r));\n\n    this._hammerConfigTokenReferences.forEach(r => {\n      if (r.isImport) {\n        this._removeHammerConfigTokenImportIfUnused(r);\n      }\n    });\n  }\n\n  /** Removes all references to the \"HammerModule\" from \"@angular/platform-browser\". */\n  private _removeHammerModuleReferences() {\n    this._hammerModuleReferences.forEach(({node, isImport, importData}) => {\n      const sourceFile = node.getSourceFile();\n      const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));\n\n      // Only remove the import for the HammerModule if the module has been accessed\n      // through a non-namespaced identifier access.\n      if (!isNamespacedIdentifierAccess(node)) {\n        this._importManager.deleteNamedBindingImport(\n            sourceFile, HAMMER_MODULE_NAME, importData.moduleName);\n      }\n\n      // For references from within an import, we do not need to do anything other than\n      // removing the import. For other references, we remove the import and the actual\n      // identifier in the module imports.\n      if (isImport) {\n        return;\n      }\n\n      // If the \"HammerModule\" is referenced within an array literal, we can\n      // remove the element easily. Otherwise if it's outside of an array literal,\n      // we need to replace the reference with an empty object literal w/ todo to\n      // not break the application.\n      if (ts.isArrayLiteralExpression(node.parent)) {\n        // Removes the \"HammerModule\" from the parent array expression. Removes\n        // the trailing comma token if present.\n        removeElementFromArrayExpression(node, recorder);\n      } else {\n        recorder.remove(node.getStart(), node.getWidth());\n        recorder.insertRight(node.getStart(), `/* TODO: remove */ {}`);\n        this._nodeFailures.push({\n          node: node,\n          message: 'Unable to delete reference to \"HammerModule\".',\n        });\n      }\n    });\n  }\n\n  /**\n   * Checks if the given node is a reference to the hammer gesture config\n   * token from platform-browser. If so, keeps track of the reference.\n   */\n  private _checkForHammerGestureConfigToken(node: ts.Node) {\n    if (ts.isIdentifier(node)) {\n      const importData = getImportOfIdentifier(node, this.typeChecker);\n      if (importData && importData.symbolName === HAMMER_CONFIG_TOKEN_NAME &&\n          importData.moduleName === HAMMER_CONFIG_TOKEN_MODULE) {\n        this._hammerConfigTokenReferences.push(\n            {node, importData, isImport: ts.isImportSpecifier(node.parent)});\n      }\n    }\n  }\n\n  /**\n   * Checks if the given node is a reference to the HammerModule from\n   * \"@angular/platform-browser\". If so, keeps track of the reference.\n   */\n  private _checkForHammerModuleReference(node: ts.Node) {\n    if (ts.isIdentifier(node)) {\n      const importData = getImportOfIdentifier(node, this.typeChecker);\n      if (importData && importData.symbolName === HAMMER_MODULE_NAME &&\n          importData.moduleName === HAMMER_MODULE_IMPORT) {\n        this._hammerModuleReferences.push(\n            {node, importData, isImport: ts.isImportSpecifier(node.parent)});\n      }\n    }\n  }\n\n  /**\n   * Checks if the given node is an import to the HammerJS package. Imports to\n   * HammerJS which load specific symbols from the package are considered as\n   * runtime usage of Hammer. e.g. `import {Symbol} from \"hammerjs\";`.\n   */\n  private _checkHammerImports(node: ts.Node) {\n    if (ts.isImportDeclaration(node) && ts.isStringLiteral(node.moduleSpecifier) &&\n        node.moduleSpecifier.text === HAMMER_MODULE_SPECIFIER) {\n      // If there is an import to HammerJS that imports symbols, or is namespaced\n      // (e.g. \"import {A, B} from ...\" or \"import * as hammer from ...\"), then we\n      // assume that some exports are used at runtime.\n      if (node.importClause &&\n          !(node.importClause.namedBindings && ts.isNamedImports(node.importClause.namedBindings) &&\n            node.importClause.namedBindings.elements.length === 0)) {\n        this._usedInRuntime = true;\n      } else {\n        this._installImports.push(node);\n      }\n    }\n  }\n\n  /**\n   * Checks if the given node accesses the global \"Hammer\" symbol at runtime. If so,\n   * the migration rule state will be updated to reflect that Hammer is used at runtime.\n   */\n  private _checkForRuntimeHammerUsage(node: ts.Node) {\n    if (this._usedInRuntime) {\n      return;\n    }\n\n    // Detects usages of \"window.Hammer\".\n    if (ts.isPropertyAccessExpression(node) && node.name.text === 'Hammer') {\n      const originExpr = unwrapExpression(node.expression);\n      if (ts.isIdentifier(originExpr) && originExpr.text === 'window') {\n        this._usedInRuntime = true;\n      }\n      return;\n    }\n\n    // Detects usages of \"window['Hammer']\".\n    if (ts.isElementAccessExpression(node) && ts.isStringLiteral(node.argumentExpression) &&\n        node.argumentExpression.text === 'Hammer') {\n      const originExpr = unwrapExpression(node.expression);\n      if (ts.isIdentifier(originExpr) && originExpr.text === 'window') {\n        this._usedInRuntime = true;\n      }\n      return;\n    }\n\n    // Handles usages of plain identifier with the name \"Hammer\". These usage\n    // are valid if they resolve to \"@types/hammerjs\". e.g. \"new Hammer(myElement)\".\n    if (ts.isIdentifier(node) && node.text === 'Hammer' &&\n        !ts.isPropertyAccessExpression(node.parent) && !ts.isElementAccessExpression(node.parent)) {\n      const symbol = this._getDeclarationSymbolOfNode(node);\n      if (symbol && symbol.valueDeclaration &&\n          symbol.valueDeclaration.getSourceFile().fileName.includes('@types/hammerjs')) {\n        this._usedInRuntime = true;\n      }\n    }\n  }\n\n  /**\n   * Checks if the given node references the gesture config from Angular Material.\n   * If so, we keep track of the found symbol reference.\n   */\n  private _checkForMaterialGestureConfig(node: ts.Node) {\n    if (ts.isIdentifier(node)) {\n      const importData = getImportOfIdentifier(node, this.typeChecker);\n      if (importData && importData.symbolName === GESTURE_CONFIG_CLASS_NAME &&\n          importData.moduleName.startsWith('@angular/material/')) {\n        this._gestureConfigReferences.push(\n            {node, importData, isImport: ts.isImportSpecifier(node.parent)});\n      }\n    }\n  }\n\n  /**\n   * Checks if the given Hammer gesture config token reference is part of an\n   * Angular provider definition that sets up a custom gesture config.\n   */\n  private _checkForCustomGestureConfigSetup(tokenRef: IdentifierReference): boolean {\n    // Walk up the tree to look for a parent property assignment of the\n    // reference to the hammer gesture config token.\n    let propertyAssignment: ts.Node = tokenRef.node;\n    while (propertyAssignment && !ts.isPropertyAssignment(propertyAssignment)) {\n      propertyAssignment = propertyAssignment.parent;\n    }\n\n    if (!propertyAssignment || !ts.isPropertyAssignment(propertyAssignment) ||\n        getPropertyNameText(propertyAssignment.name) !== 'provide') {\n      return false;\n    }\n\n    const objectLiteralExpr = propertyAssignment.parent;\n    const matchingIdentifiers = findMatchingChildNodes(objectLiteralExpr, ts.isIdentifier);\n\n    // We naively assume that if there is a reference to the \"GestureConfig\" export\n    // from Angular Material in the provider literal, that the provider sets up the\n    // Angular Material gesture config.\n    return !this._gestureConfigReferences.some(r => matchingIdentifiers.includes(r.node));\n  }\n\n  /**\n   * Determines an available file name for the gesture config which should\n   * be stored in the specified file path.\n   */\n  private _getAvailableGestureConfigFileName(sourceRoot: Path) {\n    if (!this.fileSystem.exists(join(sourceRoot, `${GESTURE_CONFIG_FILE_NAME}.ts`))) {\n      return `${GESTURE_CONFIG_FILE_NAME}.ts`;\n    }\n\n    let possibleName = `${GESTURE_CONFIG_FILE_NAME}-`;\n    let index = 1;\n    while (this.fileSystem.exists(join(sourceRoot, `${possibleName}-${index}.ts`))) {\n      index++;\n    }\n    return `${possibleName + index}.ts`;\n  }\n\n  /** Replaces a given gesture config reference with a new import. */\n  private _replaceGestureConfigReference(\n      {node, importData, isImport}: IdentifierReference, symbolName: string,\n      moduleSpecifier: string) {\n    const sourceFile = node.getSourceFile();\n    const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));\n\n    // List of all identifiers referring to the gesture config in the current file. This\n    // allows us to add an import for the copied gesture configuration without generating a\n    // new identifier for the import to avoid collisions. i.e. \"GestureConfig_1\". The import\n    // manager checks for possible name collisions, but is able to ignore specific identifiers.\n    // We use this to ignore all references to the original Angular Material gesture config,\n    // because these will be replaced and therefore will not interfere.\n    const gestureIdentifiersInFile = this._getGestureConfigIdentifiersOfFile(sourceFile);\n\n    // If the parent of the identifier is accessed through a namespace, we can just\n    // import the new gesture config without rewriting the import declaration because\n    // the config has been imported through a namespaced import.\n    if (isNamespacedIdentifierAccess(node)) {\n      const newExpression = this._importManager.addImportToSourceFile(\n          sourceFile, symbolName, moduleSpecifier, false, gestureIdentifiersInFile);\n\n      recorder.remove(node.parent.getStart(), node.parent.getWidth());\n      recorder.insertRight(node.parent.getStart(), this._printNode(newExpression, sourceFile));\n      return;\n    }\n\n    // Delete the old import to the \"GestureConfig\".\n    this._importManager.deleteNamedBindingImport(\n        sourceFile, GESTURE_CONFIG_CLASS_NAME, importData.moduleName);\n\n    // If the current reference is not from inside of a import, we need to add a new\n    // import to the copied gesture config and replace the identifier. For references\n    // within an import, we do nothing but removing the actual import. This allows us\n    // to remove unused imports to the Material gesture config.\n    if (!isImport) {\n      const newExpression = this._importManager.addImportToSourceFile(\n          sourceFile, symbolName, moduleSpecifier, false, gestureIdentifiersInFile);\n\n      recorder.remove(node.getStart(), node.getWidth());\n      recorder.insertRight(node.getStart(), this._printNode(newExpression, sourceFile));\n    }\n  }\n\n  /**\n   * Removes a given gesture config reference and its corresponding import from\n   * its containing source file. Imports will be always removed, but in some cases,\n   * where it's not guaranteed that a removal can be performed safely, we just\n   * create a migration failure (and add a TODO if possible).\n   */\n  private _removeGestureConfigReference({node, importData, isImport}: IdentifierReference) {\n    const sourceFile = node.getSourceFile();\n    const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));\n    // Only remove the import for the gesture config if the gesture config has\n    // been accessed through a non-namespaced identifier access.\n    if (!isNamespacedIdentifierAccess(node)) {\n      this._importManager.deleteNamedBindingImport(\n          sourceFile, GESTURE_CONFIG_CLASS_NAME, importData.moduleName);\n    }\n\n    // For references from within an import, we do not need to do anything other than\n    // removing the import. For other references, we remove the import and the reference\n    // identifier if used inside of a provider definition.\n    if (isImport) {\n      return;\n    }\n\n    const providerAssignment = node.parent;\n\n    // Only remove references to the gesture config which are part of a statically\n    // analyzable provider definition. We only support the common case of a gesture\n    // config provider definition where the config is set up through \"useClass\".\n    // Otherwise, it's not guaranteed that we can safely remove the provider definition.\n    if (!ts.isPropertyAssignment(providerAssignment) ||\n        getPropertyNameText(providerAssignment.name) !== 'useClass') {\n      this._nodeFailures.push({node, message: CANNOT_REMOVE_REFERENCE_ERROR});\n      return;\n    }\n\n    const objectLiteralExpr = providerAssignment.parent;\n    const provideToken = objectLiteralExpr.properties.find(\n        (p): p is ts.PropertyAssignment =>\n            ts.isPropertyAssignment(p) && getPropertyNameText(p.name) === 'provide');\n\n    // Do not remove the reference if the gesture config is not part of a provider definition,\n    // or if the provided toke is not referring to the known HAMMER_GESTURE_CONFIG token\n    // from platform-browser.\n    if (!provideToken || !this._isReferenceToHammerConfigToken(provideToken.initializer)) {\n      this._nodeFailures.push({node, message: CANNOT_REMOVE_REFERENCE_ERROR});\n      return;\n    }\n\n    // Collect all nested identifiers which will be deleted. This helps us\n    // determining if we can remove imports for the \"HAMMER_GESTURE_CONFIG\" token.\n    this._deletedIdentifiers.push(...findMatchingChildNodes(objectLiteralExpr, ts.isIdentifier));\n\n    // In case the found provider definition is not part of an array literal,\n    // we cannot safely remove the provider. This is because it could be declared\n    // as a variable. e.g. \"const gestureProvider = {provide: .., useClass: GestureConfig}\".\n    // In that case, we just add an empty object literal with TODO and print a failure.\n    if (!ts.isArrayLiteralExpression(objectLiteralExpr.parent)) {\n      recorder.remove(objectLiteralExpr.getStart(), objectLiteralExpr.getWidth());\n      recorder.insertRight(objectLiteralExpr.getStart(), `/* TODO: remove */ {}`);\n      this._nodeFailures.push({\n        node: objectLiteralExpr,\n        message: `Unable to delete provider definition for \"GestureConfig\" completely. ` +\n            `Please clean up the provider.`\n      });\n      return;\n    }\n\n    // Removes the object literal from the parent array expression. Removes\n    // the trailing comma token if present.\n    removeElementFromArrayExpression(objectLiteralExpr, recorder);\n  }\n\n  /** Removes the given hammer config token import if it is not used. */\n  private _removeHammerConfigTokenImportIfUnused({node, importData}: IdentifierReference) {\n    const sourceFile = node.getSourceFile();\n    const isTokenUsed = this._hammerConfigTokenReferences.some(\n        r => !r.isImport && !isNamespacedIdentifierAccess(r.node) &&\n            r.node.getSourceFile() === sourceFile && !this._deletedIdentifiers.includes(r.node));\n\n    // We don't want to remove the import for the token if the token is\n    // still used somewhere.\n    if (!isTokenUsed) {\n      this._importManager.deleteNamedBindingImport(\n          sourceFile, HAMMER_CONFIG_TOKEN_NAME, importData.moduleName);\n    }\n  }\n\n  /** Removes Hammer from all index HTML files of the current project. */\n  private _removeHammerFromIndexFile() {\n    const indexFilePaths = getProjectIndexFiles(this.context.project);\n    indexFilePaths.forEach(filePath => {\n      if (!this.fileSystem.exists(filePath)) {\n        return;\n      }\n\n      const htmlContent = this.fileSystem.read(filePath)!;\n      const recorder = this.fileSystem.edit(filePath);\n\n      findHammerScriptImportElements(htmlContent)\n          .forEach(el => removeElementFromHtml(el, recorder));\n    });\n  }\n\n  /** Sets up the Hammer gesture config in the root module if needed. */\n  private _setupNewGestureConfigInRootModule(gestureConfigPath: Path) {\n    const {project} = this.context;\n    const mainFilePath = getProjectMainFile(project);\n    const rootModuleSymbol = this._getRootModuleSymbol(mainFilePath);\n\n    if (rootModuleSymbol === null || rootModuleSymbol.valueDeclaration === undefined) {\n      this.failures.push({\n        filePath: mainFilePath,\n        message: `Could not setup Hammer gestures in module. Please ` +\n            `manually ensure that the Hammer gesture config is set up.`,\n      });\n      return;\n    }\n\n    const sourceFile = rootModuleSymbol.valueDeclaration.getSourceFile();\n    const metadata = getDecoratorMetadata(sourceFile, 'NgModule', '@angular/core') as\n        ts.ObjectLiteralExpression[];\n\n    // If no \"NgModule\" definition is found inside the source file, we just do nothing.\n    if (!metadata.length) {\n      return;\n    }\n\n    const filePath = this.fileSystem.resolve(sourceFile.fileName);\n    const recorder = this.fileSystem.edit(filePath);\n    const providersField = getMetadataField(metadata[0], 'providers')[0];\n    const providerIdentifiers =\n        providersField ? findMatchingChildNodes(providersField, ts.isIdentifier) : null;\n    const gestureConfigExpr = this._importManager.addImportToSourceFile(\n        sourceFile, GESTURE_CONFIG_CLASS_NAME,\n        getModuleSpecifier(gestureConfigPath, filePath), false,\n        this._getGestureConfigIdentifiersOfFile(sourceFile));\n    const hammerConfigTokenExpr = this._importManager.addImportToSourceFile(\n        sourceFile, HAMMER_CONFIG_TOKEN_NAME, HAMMER_CONFIG_TOKEN_MODULE);\n    const newProviderNode = ts.createObjectLiteral([\n      ts.createPropertyAssignment('provide', hammerConfigTokenExpr),\n      ts.createPropertyAssignment('useClass', gestureConfigExpr)\n    ]);\n\n    // If the providers field exists and already contains references to the hammer gesture\n    // config token and the gesture config, we naively assume that the gesture config is\n    // already set up. We only want to add the gesture config provider if it is not set up.\n    if (!providerIdentifiers ||\n        !(this._hammerConfigTokenReferences.some(r => providerIdentifiers.includes(r.node)) &&\n          this._gestureConfigReferences.some(r => providerIdentifiers.includes(r.node)))) {\n      const symbolName = this._printNode(newProviderNode, sourceFile);\n      addSymbolToNgModuleMetadata(sourceFile, sourceFile.fileName, 'providers', symbolName, null)\n        .forEach(change => {\n          if (change instanceof InsertChange) {\n            recorder.insertRight(change.pos, change.toAdd);\n          }\n        });\n    }\n  }\n\n  /**\n   * Gets the TypeScript symbol of the root module by looking for the module\n   * bootstrap expression in the specified source file.\n   */\n  private _getRootModuleSymbol(mainFilePath: Path): ts.Symbol|null {\n    const mainFile = this.program.getSourceFile(mainFilePath);\n    if (!mainFile) {\n      return null;\n    }\n\n    const appModuleExpr = findMainModuleExpression(mainFile);\n    if (!appModuleExpr) {\n      return null;\n    }\n\n    const appModuleSymbol = this._getDeclarationSymbolOfNode(unwrapExpression(appModuleExpr));\n    if (!appModuleSymbol || !appModuleSymbol.valueDeclaration) {\n      return null;\n    }\n    return appModuleSymbol;\n  }\n\n  /** Sets up the \"HammerModule\" in the root module of the current project. */\n  private _setupHammerModuleInRootModule() {\n    const {project} = this.context;\n    const mainFilePath = getProjectMainFile(project);\n    const rootModuleSymbol = this._getRootModuleSymbol(mainFilePath);\n\n    if (rootModuleSymbol === null || rootModuleSymbol.valueDeclaration === undefined) {\n      this.failures.push({\n        filePath: mainFilePath,\n        message: `Could not setup HammerModule. Please manually set up the \"HammerModule\" ` +\n            `from \"@angular/platform-browser\".`,\n      });\n      return;\n    }\n\n    const sourceFile = rootModuleSymbol.valueDeclaration.getSourceFile();\n    const metadata = getDecoratorMetadata(sourceFile, 'NgModule', '@angular/core') as\n        ts.ObjectLiteralExpression[];\n    if (!metadata.length) {\n      return;\n    }\n\n    const importsField = getMetadataField(metadata[0], 'imports')[0];\n    const importIdentifiers =\n        importsField ? findMatchingChildNodes(importsField, ts.isIdentifier) : null;\n    const recorder = this.fileSystem.edit(this.fileSystem.resolve(sourceFile.fileName));\n    const hammerModuleExpr = this._importManager.addImportToSourceFile(\n        sourceFile, HAMMER_MODULE_NAME, HAMMER_MODULE_IMPORT);\n\n    // If the \"HammerModule\" is not already imported in the app module, we set it up\n    // by adding it to the \"imports\" field of the app module.\n    if (!importIdentifiers ||\n        !this._hammerModuleReferences.some(r => importIdentifiers.includes(r.node))) {\n      const symbolName = this._printNode(hammerModuleExpr, sourceFile);\n      addSymbolToNgModuleMetadata(sourceFile, sourceFile.fileName, 'imports', symbolName, null)\n        .forEach(change => {\n          if (change instanceof InsertChange) {\n            recorder.insertRight(change.pos, change.toAdd);\n          }\n        });\n    }\n  }\n\n  /** Prints a given node within the specified source file. */\n  private _printNode(node: ts.Node, sourceFile: ts.SourceFile): string {\n    return this._printer.printNode(ts.EmitHint.Unspecified, node, sourceFile);\n  }\n\n  /** Gets all referenced gesture config identifiers of a given source file */\n  private _getGestureConfigIdentifiersOfFile(sourceFile: ts.SourceFile): ts.Identifier[] {\n    return this._gestureConfigReferences.filter(d => d.node.getSourceFile() === sourceFile)\n        .map(d => d.node);\n  }\n\n  /** Gets the symbol that contains the value declaration of the specified node. */\n  private _getDeclarationSymbolOfNode(node: ts.Node): ts.Symbol|undefined {\n    const symbol = this.typeChecker.getSymbolAtLocation(node);\n\n    // Symbols can be aliases of the declaration symbol. e.g. in named import specifiers.\n    // We need to resolve the aliased symbol back to the declaration symbol.\n    // tslint:disable-next-line:no-bitwise\n    if (symbol && (symbol.flags & ts.SymbolFlags.Alias) !== 0) {\n      return this.typeChecker.getAliasedSymbol(symbol);\n    }\n    return symbol;\n  }\n\n  /**\n   * Checks whether the given expression resolves to a hammer gesture config\n   * token reference from \"@angular/platform-browser\".\n   */\n  private _isReferenceToHammerConfigToken(expr: ts.Expression) {\n    const unwrapped = unwrapExpression(expr);\n    if (ts.isIdentifier(unwrapped)) {\n      return this._hammerConfigTokenReferences.some(r => r.node === unwrapped);\n    } else if (ts.isPropertyAccessExpression(unwrapped)) {\n      return this._hammerConfigTokenReferences.some(r => r.node === unwrapped.name);\n    }\n    return false;\n  }\n\n  /**\n   * Creates migration failures of the collected node failures. The returned migration\n   * failures are updated to reflect the post-migration state of source files. Meaning\n   * that failure positions are corrected if source file modifications shifted lines.\n   */\n  private _createMigrationFailures(): MigrationFailure[] {\n    return this._nodeFailures.map(({node, message}) => {\n      const sourceFile = node.getSourceFile();\n      const offset = node.getStart();\n      const position = ts.getLineAndCharacterOfPosition(sourceFile, node.getStart());\n      return {\n        position: this._importManager.correctNodePosition(node, offset, position),\n        message: message,\n        filePath: this.fileSystem.resolve(sourceFile.fileName),\n      };\n    });\n  }\n\n  /** Global state of whether Hammer is used in any analyzed project target. */\n  static globalUsesHammer = false;\n\n  /**\n   * Static migration rule method that will be called once all project targets\n   * have been migrated individually. This method can be used to make changes based\n   * on the analysis of the individual targets. For example: we only remove Hammer\n   * from the \"package.json\" if it is not used in *any* project target.\n   */\n  static globalPostMigration(tree: Tree, context: SchematicContext): PostMigrationAction {\n    // Always notify the developer that the Hammer v9 migration does not migrate tests.\n    context.logger.info(\n        '\\n⚠  General notice: The HammerJS v9 migration for Angular Components is not able to ' +\n        'migrate tests. Please manually clean up tests in your project if they rely on ' +\n        (this.globalUsesHammer ? 'the deprecated Angular Material gesture config.' : 'HammerJS.'));\n    context.logger.info(\n        'Read more about migrating tests: https://git.io/ng-material-v9-hammer-migrate-tests');\n\n    if (!this.globalUsesHammer && this._removeHammerFromPackageJson(tree)) {\n      // Since Hammer has been removed from the workspace \"package.json\" file,\n      // we schedule a node package install task to refresh the lock file.\n      return {runPackageManager: true};\n    }\n\n    // Clean global state once the workspace has been migrated. This is technically\n    // not necessary in \"ng update\", but in tests we re-use the same rule class.\n    this.globalUsesHammer = false;\n  }\n\n  /**\n   * Removes the hammer package from the workspace \"package.json\".\n   * @returns Whether Hammer was set up and has been removed from the \"package.json\"\n   */\n  private static _removeHammerFromPackageJson(tree: Tree): boolean {\n    if (!tree.exists('/package.json')) {\n      return false;\n    }\n\n    const packageJson = JSON.parse(tree.read('/package.json')!.toString('utf8')) as PackageJson;\n\n    // We do not handle the case where someone manually added \"hammerjs\" to the dev dependencies.\n    if (packageJson.dependencies && packageJson.dependencies[HAMMER_MODULE_SPECIFIER]) {\n      delete packageJson.dependencies[HAMMER_MODULE_SPECIFIER];\n      tree.overwrite('/package.json', JSON.stringify(packageJson, null, 2));\n      return true;\n    }\n    return false;\n  }\n}\n\n/**\n * Recursively unwraps a given expression if it is wrapped\n * by parenthesis, type casts or type assertions.\n */\nfunction unwrapExpression(node: ts.Node): ts.Node {\n  if (ts.isParenthesizedExpression(node)) {\n    return unwrapExpression(node.expression);\n  } else if (ts.isAsExpression(node)) {\n    return unwrapExpression(node.expression);\n  } else if (ts.isTypeAssertion(node)) {\n    return unwrapExpression(node.expression);\n  }\n  return node;\n}\n\n/**\n * Converts the specified path to a valid TypeScript module specifier which is\n * relative to the given containing file.\n */\nfunction getModuleSpecifier(newPath: Path, containingFile: Path) {\n  let result = relative(dirname(containingFile), newPath).replace(/\\\\/g, '/').replace(/\\.ts$/, '');\n  if (!result.startsWith('.')) {\n    result = `./${result}`;\n  }\n  return result;\n}\n\n/**\n * Gets the text of the given property name.\n * @returns Text of the given property name. Null if not statically analyzable.\n */\nfunction getPropertyNameText(node: ts.PropertyName): string|null {\n  if (ts.isIdentifier(node) || ts.isStringLiteralLike(node)) {\n    return node.text;\n  }\n  return null;\n}\n\n/** Checks whether the given identifier is part of a namespaced access. */\nfunction isNamespacedIdentifierAccess(node: ts.Identifier): boolean {\n  return ts.isQualifiedName(node.parent) || ts.isPropertyAccessExpression(node.parent);\n}\n\n/**\n * Walks through the specified node and returns all child nodes which match the\n * given predicate.\n */\nfunction findMatchingChildNodes<T extends ts.Node>(\n    parent: ts.Node, predicate: (node: ts.Node) => node is T): T[] {\n  const result: T[] = [];\n  const visitNode = (node: ts.Node) => {\n    if (predicate(node)) {\n      result.push(node);\n    }\n    ts.forEachChild(node, visitNode);\n  };\n  ts.forEachChild(parent, visitNode);\n  return result;\n}\n"]}
|