source: trip-planner-front/node_modules/@angular/compiler/esm2015/src/render3/view/styling_builder.js@ 6a3a178

Last change on this file since 6a3a178 was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 65.7 KB
Line 
1import { ASTWithSource, BindingPipe, Interpolation } from '../../expression_parser/ast';
2import * as o from '../../output/output_ast';
3import { isEmptyExpression } from '../../template_parser/template_parser';
4import { Identifiers as R3 } from '../r3_identifiers';
5import { hyphenate, parse as parseStyle } from './style_parser';
6import { getInterpolationArgsLength } from './util';
7const IMPORTANT_FLAG = '!important';
8/**
9 * Minimum amount of binding slots required in the runtime for style/class bindings.
10 *
11 * Styling in Angular uses up two slots in the runtime LView/TData data structures to
12 * record binding data, property information and metadata.
13 *
14 * When a binding is registered it will place the following information in the `LView`:
15 *
16 * slot 1) binding value
17 * slot 2) cached value (all other values collected before it in string form)
18 *
19 * When a binding is registered it will place the following information in the `TData`:
20 *
21 * slot 1) prop name
22 * slot 2) binding index that points to the previous style/class binding (and some extra config
23 * values)
24 *
25 * Let's imagine we have a binding that looks like so:
26 *
27 * ```
28 * <div [style.width]="x" [style.height]="y">
29 * ```
30 *
31 * Our `LView` and `TData` data-structures look like so:
32 *
33 * ```typescript
34 * LView = [
35 * // ...
36 * x, // value of x
37 * "width: x",
38 *
39 * y, // value of y
40 * "width: x; height: y",
41 * // ...
42 * ];
43 *
44 * TData = [
45 * // ...
46 * "width", // binding slot 20
47 * 0,
48 *
49 * "height",
50 * 20,
51 * // ...
52 * ];
53 * ```
54 *
55 * */
56export const MIN_STYLING_BINDING_SLOTS_REQUIRED = 2;
57/**
58 * Produces creation/update instructions for all styling bindings (class and style)
59 *
60 * It also produces the creation instruction to register all initial styling values
61 * (which are all the static class="..." and style="..." attribute values that exist
62 * on an element within a template).
63 *
64 * The builder class below handles producing instructions for the following cases:
65 *
66 * - Static style/class attributes (style="..." and class="...")
67 * - Dynamic style/class map bindings ([style]="map" and [class]="map|string")
68 * - Dynamic style/class property bindings ([style.prop]="exp" and [class.name]="exp")
69 *
70 * Due to the complex relationship of all of these cases, the instructions generated
71 * for these attributes/properties/bindings must be done so in the correct order. The
72 * order which these must be generated is as follows:
73 *
74 * if (createMode) {
75 * styling(...)
76 * }
77 * if (updateMode) {
78 * styleMap(...)
79 * classMap(...)
80 * styleProp(...)
81 * classProp(...)
82 * }
83 *
84 * The creation/update methods within the builder class produce these instructions.
85 */
86export class StylingBuilder {
87 constructor(_directiveExpr) {
88 this._directiveExpr = _directiveExpr;
89 /** Whether or not there are any static styling values present */
90 this._hasInitialValues = false;
91 /**
92 * Whether or not there are any styling bindings present
93 * (i.e. `[style]`, `[class]`, `[style.prop]` or `[class.name]`)
94 */
95 this.hasBindings = false;
96 this.hasBindingsWithPipes = false;
97 /** the input for [class] (if it exists) */
98 this._classMapInput = null;
99 /** the input for [style] (if it exists) */
100 this._styleMapInput = null;
101 /** an array of each [style.prop] input */
102 this._singleStyleInputs = null;
103 /** an array of each [class.name] input */
104 this._singleClassInputs = null;
105 this._lastStylingInput = null;
106 this._firstStylingInput = null;
107 // maps are used instead of hash maps because a Map will
108 // retain the ordering of the keys
109 /**
110 * Represents the location of each style binding in the template
111 * (e.g. `<div [style.width]="w" [style.height]="h">` implies
112 * that `width=0` and `height=1`)
113 */
114 this._stylesIndex = new Map();
115 /**
116 * Represents the location of each class binding in the template
117 * (e.g. `<div [class.big]="b" [class.hidden]="h">` implies
118 * that `big=0` and `hidden=1`)
119 */
120 this._classesIndex = new Map();
121 this._initialStyleValues = [];
122 this._initialClassValues = [];
123 }
124 /**
125 * Registers a given input to the styling builder to be later used when producing AOT code.
126 *
127 * The code below will only accept the input if it is somehow tied to styling (whether it be
128 * style/class bindings or static style/class attributes).
129 */
130 registerBoundInput(input) {
131 // [attr.style] or [attr.class] are skipped in the code below,
132 // they should not be treated as styling-based bindings since
133 // they are intended to be written directly to the attr and
134 // will therefore skip all style/class resolution that is present
135 // with style="", [style]="" and [style.prop]="", class="",
136 // [class.prop]="". [class]="" assignments
137 let binding = null;
138 let name = input.name;
139 switch (input.type) {
140 case 0 /* Property */:
141 binding = this.registerInputBasedOnName(name, input.value, input.sourceSpan);
142 break;
143 case 3 /* Style */:
144 binding = this.registerStyleInput(name, false, input.value, input.sourceSpan, input.unit);
145 break;
146 case 2 /* Class */:
147 binding = this.registerClassInput(name, false, input.value, input.sourceSpan);
148 break;
149 }
150 return binding ? true : false;
151 }
152 registerInputBasedOnName(name, expression, sourceSpan) {
153 let binding = null;
154 const prefix = name.substring(0, 6);
155 const isStyle = name === 'style' || prefix === 'style.' || prefix === 'style!';
156 const isClass = !isStyle && (name === 'class' || prefix === 'class.' || prefix === 'class!');
157 if (isStyle || isClass) {
158 const isMapBased = name.charAt(5) !== '.'; // style.prop or class.prop makes this a no
159 const property = name.substr(isMapBased ? 5 : 6); // the dot explains why there's a +1
160 if (isStyle) {
161 binding = this.registerStyleInput(property, isMapBased, expression, sourceSpan);
162 }
163 else {
164 binding = this.registerClassInput(property, isMapBased, expression, sourceSpan);
165 }
166 }
167 return binding;
168 }
169 registerStyleInput(name, isMapBased, value, sourceSpan, suffix) {
170 if (isEmptyExpression(value)) {
171 return null;
172 }
173 // CSS custom properties are case-sensitive so we shouldn't normalize them.
174 // See: https://www.w3.org/TR/css-variables-1/#defining-variables
175 if (!isCssCustomProperty(name)) {
176 name = hyphenate(name);
177 }
178 const { property, hasOverrideFlag, suffix: bindingSuffix } = parseProperty(name);
179 suffix = typeof suffix === 'string' && suffix.length !== 0 ? suffix : bindingSuffix;
180 const entry = { name: property, suffix: suffix, value, sourceSpan, hasOverrideFlag };
181 if (isMapBased) {
182 this._styleMapInput = entry;
183 }
184 else {
185 (this._singleStyleInputs = this._singleStyleInputs || []).push(entry);
186 registerIntoMap(this._stylesIndex, property);
187 }
188 this._lastStylingInput = entry;
189 this._firstStylingInput = this._firstStylingInput || entry;
190 this._checkForPipes(value);
191 this.hasBindings = true;
192 return entry;
193 }
194 registerClassInput(name, isMapBased, value, sourceSpan) {
195 if (isEmptyExpression(value)) {
196 return null;
197 }
198 const { property, hasOverrideFlag } = parseProperty(name);
199 const entry = { name: property, value, sourceSpan, hasOverrideFlag, suffix: null };
200 if (isMapBased) {
201 this._classMapInput = entry;
202 }
203 else {
204 (this._singleClassInputs = this._singleClassInputs || []).push(entry);
205 registerIntoMap(this._classesIndex, property);
206 }
207 this._lastStylingInput = entry;
208 this._firstStylingInput = this._firstStylingInput || entry;
209 this._checkForPipes(value);
210 this.hasBindings = true;
211 return entry;
212 }
213 _checkForPipes(value) {
214 if ((value instanceof ASTWithSource) && (value.ast instanceof BindingPipe)) {
215 this.hasBindingsWithPipes = true;
216 }
217 }
218 /**
219 * Registers the element's static style string value to the builder.
220 *
221 * @param value the style string (e.g. `width:100px; height:200px;`)
222 */
223 registerStyleAttr(value) {
224 this._initialStyleValues = parseStyle(value);
225 this._hasInitialValues = true;
226 }
227 /**
228 * Registers the element's static class string value to the builder.
229 *
230 * @param value the className string (e.g. `disabled gold zoom`)
231 */
232 registerClassAttr(value) {
233 this._initialClassValues = value.trim().split(/\s+/g);
234 this._hasInitialValues = true;
235 }
236 /**
237 * Appends all styling-related expressions to the provided attrs array.
238 *
239 * @param attrs an existing array where each of the styling expressions
240 * will be inserted into.
241 */
242 populateInitialStylingAttrs(attrs) {
243 // [CLASS_MARKER, 'foo', 'bar', 'baz' ...]
244 if (this._initialClassValues.length) {
245 attrs.push(o.literal(1 /* Classes */));
246 for (let i = 0; i < this._initialClassValues.length; i++) {
247 attrs.push(o.literal(this._initialClassValues[i]));
248 }
249 }
250 // [STYLE_MARKER, 'width', '200px', 'height', '100px', ...]
251 if (this._initialStyleValues.length) {
252 attrs.push(o.literal(2 /* Styles */));
253 for (let i = 0; i < this._initialStyleValues.length; i += 2) {
254 attrs.push(o.literal(this._initialStyleValues[i]), o.literal(this._initialStyleValues[i + 1]));
255 }
256 }
257 }
258 /**
259 * Builds an instruction with all the expressions and parameters for `elementHostAttrs`.
260 *
261 * The instruction generation code below is used for producing the AOT statement code which is
262 * responsible for registering initial styles (within a directive hostBindings' creation block),
263 * as well as any of the provided attribute values, to the directive host element.
264 */
265 assignHostAttrs(attrs, definitionMap) {
266 if (this._directiveExpr && (attrs.length || this._hasInitialValues)) {
267 this.populateInitialStylingAttrs(attrs);
268 definitionMap.set('hostAttrs', o.literalArr(attrs));
269 }
270 }
271 /**
272 * Builds an instruction with all the expressions and parameters for `classMap`.
273 *
274 * The instruction data will contain all expressions for `classMap` to function
275 * which includes the `[class]` expression params.
276 */
277 buildClassMapInstruction(valueConverter) {
278 if (this._classMapInput) {
279 return this._buildMapBasedInstruction(valueConverter, true, this._classMapInput);
280 }
281 return null;
282 }
283 /**
284 * Builds an instruction with all the expressions and parameters for `styleMap`.
285 *
286 * The instruction data will contain all expressions for `styleMap` to function
287 * which includes the `[style]` expression params.
288 */
289 buildStyleMapInstruction(valueConverter) {
290 if (this._styleMapInput) {
291 return this._buildMapBasedInstruction(valueConverter, false, this._styleMapInput);
292 }
293 return null;
294 }
295 _buildMapBasedInstruction(valueConverter, isClassBased, stylingInput) {
296 // each styling binding value is stored in the LView
297 // map-based bindings allocate two slots: one for the
298 // previous binding value and another for the previous
299 // className or style attribute value.
300 let totalBindingSlotsRequired = MIN_STYLING_BINDING_SLOTS_REQUIRED;
301 // these values must be outside of the update block so that they can
302 // be evaluated (the AST visit call) during creation time so that any
303 // pipes can be picked up in time before the template is built
304 const mapValue = stylingInput.value.visit(valueConverter);
305 let reference;
306 if (mapValue instanceof Interpolation) {
307 totalBindingSlotsRequired += mapValue.expressions.length;
308 reference = isClassBased ? getClassMapInterpolationExpression(mapValue) :
309 getStyleMapInterpolationExpression(mapValue);
310 }
311 else {
312 reference = isClassBased ? R3.classMap : R3.styleMap;
313 }
314 return {
315 reference,
316 calls: [{
317 supportsInterpolation: true,
318 sourceSpan: stylingInput.sourceSpan,
319 allocateBindingSlots: totalBindingSlotsRequired,
320 params: (convertFn) => {
321 const convertResult = convertFn(mapValue);
322 const params = Array.isArray(convertResult) ? convertResult : [convertResult];
323 return params;
324 }
325 }]
326 };
327 }
328 _buildSingleInputs(reference, inputs, valueConverter, getInterpolationExpressionFn, isClassBased) {
329 const instructions = [];
330 inputs.forEach(input => {
331 const previousInstruction = instructions[instructions.length - 1];
332 const value = input.value.visit(valueConverter);
333 let referenceForCall = reference;
334 // each styling binding value is stored in the LView
335 // but there are two values stored for each binding:
336 // 1) the value itself
337 // 2) an intermediate value (concatenation of style up to this point).
338 // We need to store the intermediate value so that we don't allocate
339 // the strings on each CD.
340 let totalBindingSlotsRequired = MIN_STYLING_BINDING_SLOTS_REQUIRED;
341 if (value instanceof Interpolation) {
342 totalBindingSlotsRequired += value.expressions.length;
343 if (getInterpolationExpressionFn) {
344 referenceForCall = getInterpolationExpressionFn(value);
345 }
346 }
347 const call = {
348 sourceSpan: input.sourceSpan,
349 allocateBindingSlots: totalBindingSlotsRequired,
350 supportsInterpolation: !!getInterpolationExpressionFn,
351 params: (convertFn) => {
352 // params => stylingProp(propName, value, suffix)
353 const params = [];
354 params.push(o.literal(input.name));
355 const convertResult = convertFn(value);
356 if (Array.isArray(convertResult)) {
357 params.push(...convertResult);
358 }
359 else {
360 params.push(convertResult);
361 }
362 // [style.prop] bindings may use suffix values (e.g. px, em, etc...), therefore,
363 // if that is detected then we need to pass that in as an optional param.
364 if (!isClassBased && input.suffix !== null) {
365 params.push(o.literal(input.suffix));
366 }
367 return params;
368 }
369 };
370 // If we ended up generating a call to the same instruction as the previous styling property
371 // we can chain the calls together safely to save some bytes, otherwise we have to generate
372 // a separate instruction call. This is primarily a concern with interpolation instructions
373 // where we may start off with one `reference`, but end up using another based on the
374 // number of interpolations.
375 if (previousInstruction && previousInstruction.reference === referenceForCall) {
376 previousInstruction.calls.push(call);
377 }
378 else {
379 instructions.push({ reference: referenceForCall, calls: [call] });
380 }
381 });
382 return instructions;
383 }
384 _buildClassInputs(valueConverter) {
385 if (this._singleClassInputs) {
386 return this._buildSingleInputs(R3.classProp, this._singleClassInputs, valueConverter, null, true);
387 }
388 return [];
389 }
390 _buildStyleInputs(valueConverter) {
391 if (this._singleStyleInputs) {
392 return this._buildSingleInputs(R3.styleProp, this._singleStyleInputs, valueConverter, getStylePropInterpolationExpression, false);
393 }
394 return [];
395 }
396 /**
397 * Constructs all instructions which contain the expressions that will be placed
398 * into the update block of a template function or a directive hostBindings function.
399 */
400 buildUpdateLevelInstructions(valueConverter) {
401 const instructions = [];
402 if (this.hasBindings) {
403 const styleMapInstruction = this.buildStyleMapInstruction(valueConverter);
404 if (styleMapInstruction) {
405 instructions.push(styleMapInstruction);
406 }
407 const classMapInstruction = this.buildClassMapInstruction(valueConverter);
408 if (classMapInstruction) {
409 instructions.push(classMapInstruction);
410 }
411 instructions.push(...this._buildStyleInputs(valueConverter));
412 instructions.push(...this._buildClassInputs(valueConverter));
413 }
414 return instructions;
415 }
416}
417function registerIntoMap(map, key) {
418 if (!map.has(key)) {
419 map.set(key, map.size);
420 }
421}
422export function parseProperty(name) {
423 let hasOverrideFlag = false;
424 const overrideIndex = name.indexOf(IMPORTANT_FLAG);
425 if (overrideIndex !== -1) {
426 name = overrideIndex > 0 ? name.substring(0, overrideIndex) : '';
427 hasOverrideFlag = true;
428 }
429 let suffix = null;
430 let property = name;
431 const unitIndex = name.lastIndexOf('.');
432 if (unitIndex > 0) {
433 suffix = name.substr(unitIndex + 1);
434 property = name.substring(0, unitIndex);
435 }
436 return { property, suffix, hasOverrideFlag };
437}
438/**
439 * Gets the instruction to generate for an interpolated class map.
440 * @param interpolation An Interpolation AST
441 */
442function getClassMapInterpolationExpression(interpolation) {
443 switch (getInterpolationArgsLength(interpolation)) {
444 case 1:
445 return R3.classMap;
446 case 3:
447 return R3.classMapInterpolate1;
448 case 5:
449 return R3.classMapInterpolate2;
450 case 7:
451 return R3.classMapInterpolate3;
452 case 9:
453 return R3.classMapInterpolate4;
454 case 11:
455 return R3.classMapInterpolate5;
456 case 13:
457 return R3.classMapInterpolate6;
458 case 15:
459 return R3.classMapInterpolate7;
460 case 17:
461 return R3.classMapInterpolate8;
462 default:
463 return R3.classMapInterpolateV;
464 }
465}
466/**
467 * Gets the instruction to generate for an interpolated style map.
468 * @param interpolation An Interpolation AST
469 */
470function getStyleMapInterpolationExpression(interpolation) {
471 switch (getInterpolationArgsLength(interpolation)) {
472 case 1:
473 return R3.styleMap;
474 case 3:
475 return R3.styleMapInterpolate1;
476 case 5:
477 return R3.styleMapInterpolate2;
478 case 7:
479 return R3.styleMapInterpolate3;
480 case 9:
481 return R3.styleMapInterpolate4;
482 case 11:
483 return R3.styleMapInterpolate5;
484 case 13:
485 return R3.styleMapInterpolate6;
486 case 15:
487 return R3.styleMapInterpolate7;
488 case 17:
489 return R3.styleMapInterpolate8;
490 default:
491 return R3.styleMapInterpolateV;
492 }
493}
494/**
495 * Gets the instruction to generate for an interpolated style prop.
496 * @param interpolation An Interpolation AST
497 */
498function getStylePropInterpolationExpression(interpolation) {
499 switch (getInterpolationArgsLength(interpolation)) {
500 case 1:
501 return R3.styleProp;
502 case 3:
503 return R3.stylePropInterpolate1;
504 case 5:
505 return R3.stylePropInterpolate2;
506 case 7:
507 return R3.stylePropInterpolate3;
508 case 9:
509 return R3.stylePropInterpolate4;
510 case 11:
511 return R3.stylePropInterpolate5;
512 case 13:
513 return R3.stylePropInterpolate6;
514 case 15:
515 return R3.stylePropInterpolate7;
516 case 17:
517 return R3.stylePropInterpolate8;
518 default:
519 return R3.stylePropInterpolateV;
520 }
521}
522/**
523 * Checks whether property name is a custom CSS property.
524 * See: https://www.w3.org/TR/css-variables-1
525 */
526function isCssCustomProperty(name) {
527 return name.startsWith('--');
528}
529//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"styling_builder.js","sourceRoot":"","sources":["../../../../../../../../packages/compiler/src/render3/view/styling_builder.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 */\nimport {AttributeMarker} from '../../core';\nimport {AST, ASTWithSource, BindingPipe, BindingType, Interpolation} from '../../expression_parser/ast';\nimport * as o from '../../output/output_ast';\nimport {ParseSourceSpan} from '../../parse_util';\nimport {isEmptyExpression} from '../../template_parser/template_parser';\nimport * as t from '../r3_ast';\nimport {Identifiers as R3} from '../r3_identifiers';\n\nimport {hyphenate, parse as parseStyle} from './style_parser';\nimport {ValueConverter} from './template';\nimport {DefinitionMap, getInterpolationArgsLength} from './util';\n\nconst IMPORTANT_FLAG = '!important';\n\n/**\n * Minimum amount of binding slots required in the runtime for style/class bindings.\n *\n * Styling in Angular uses up two slots in the runtime LView/TData data structures to\n * record binding data, property information and metadata.\n *\n * When a binding is registered it will place the following information in the `LView`:\n *\n * slot 1) binding value\n * slot 2) cached value (all other values collected before it in string form)\n *\n * When a binding is registered it will place the following information in the `TData`:\n *\n * slot 1) prop name\n * slot 2) binding index that points to the previous style/class binding (and some extra config\n * values)\n *\n * Let's imagine we have a binding that looks like so:\n *\n * ```\n * <div [style.width]=\"x\" [style.height]=\"y\">\n * ```\n *\n * Our `LView` and `TData` data-structures look like so:\n *\n * ```typescript\n * LView = [\n *   // ...\n *   x, // value of x\n *   \"width: x\",\n *\n *   y, // value of y\n *   \"width: x; height: y\",\n *   // ...\n * ];\n *\n * TData = [\n *   // ...\n *   \"width\", // binding slot 20\n *   0,\n *\n *   \"height\",\n *   20,\n *   // ...\n * ];\n * ```\n *\n * */\nexport const MIN_STYLING_BINDING_SLOTS_REQUIRED = 2;\n\n/**\n * A styling expression summary that is to be processed by the compiler\n */\nexport interface StylingInstruction {\n  reference: o.ExternalReference;\n  /** Calls to individual styling instructions. Used when chaining calls to the same instruction. */\n  calls: StylingInstructionCall[];\n}\n\nexport interface StylingInstructionCall {\n  sourceSpan: ParseSourceSpan|null;\n  supportsInterpolation: boolean;\n  allocateBindingSlots: number;\n  params: ((convertFn: (value: any) => o.Expression | o.Expression[]) => o.Expression[]);\n}\n\n/**\n * An internal record of the input data for a styling binding\n */\ninterface BoundStylingEntry {\n  hasOverrideFlag: boolean;\n  name: string|null;\n  suffix: string|null;\n  sourceSpan: ParseSourceSpan;\n  value: AST;\n}\n\n/**\n * Produces creation/update instructions for all styling bindings (class and style)\n *\n * It also produces the creation instruction to register all initial styling values\n * (which are all the static class=\"...\" and style=\"...\" attribute values that exist\n * on an element within a template).\n *\n * The builder class below handles producing instructions for the following cases:\n *\n * - Static style/class attributes (style=\"...\" and class=\"...\")\n * - Dynamic style/class map bindings ([style]=\"map\" and [class]=\"map|string\")\n * - Dynamic style/class property bindings ([style.prop]=\"exp\" and [class.name]=\"exp\")\n *\n * Due to the complex relationship of all of these cases, the instructions generated\n * for these attributes/properties/bindings must be done so in the correct order. The\n * order which these must be generated is as follows:\n *\n * if (createMode) {\n *   styling(...)\n * }\n * if (updateMode) {\n *   styleMap(...)\n *   classMap(...)\n *   styleProp(...)\n *   classProp(...)\n * }\n *\n * The creation/update methods within the builder class produce these instructions.\n */\nexport class StylingBuilder {\n  /** Whether or not there are any static styling values present */\n  private _hasInitialValues = false;\n  /**\n   *  Whether or not there are any styling bindings present\n   *  (i.e. `[style]`, `[class]`, `[style.prop]` or `[class.name]`)\n   */\n  public hasBindings = false;\n  public hasBindingsWithPipes = false;\n\n  /** the input for [class] (if it exists) */\n  private _classMapInput: BoundStylingEntry|null = null;\n  /** the input for [style] (if it exists) */\n  private _styleMapInput: BoundStylingEntry|null = null;\n  /** an array of each [style.prop] input */\n  private _singleStyleInputs: BoundStylingEntry[]|null = null;\n  /** an array of each [class.name] input */\n  private _singleClassInputs: BoundStylingEntry[]|null = null;\n  private _lastStylingInput: BoundStylingEntry|null = null;\n  private _firstStylingInput: BoundStylingEntry|null = null;\n\n  // maps are used instead of hash maps because a Map will\n  // retain the ordering of the keys\n\n  /**\n   * Represents the location of each style binding in the template\n   * (e.g. `<div [style.width]=\"w\" [style.height]=\"h\">` implies\n   * that `width=0` and `height=1`)\n   */\n  private _stylesIndex = new Map<string, number>();\n\n  /**\n   * Represents the location of each class binding in the template\n   * (e.g. `<div [class.big]=\"b\" [class.hidden]=\"h\">` implies\n   * that `big=0` and `hidden=1`)\n   */\n  private _classesIndex = new Map<string, number>();\n  private _initialStyleValues: string[] = [];\n  private _initialClassValues: string[] = [];\n\n  constructor(private _directiveExpr: o.Expression|null) {}\n\n  /**\n   * Registers a given input to the styling builder to be later used when producing AOT code.\n   *\n   * The code below will only accept the input if it is somehow tied to styling (whether it be\n   * style/class bindings or static style/class attributes).\n   */\n  registerBoundInput(input: t.BoundAttribute): boolean {\n    // [attr.style] or [attr.class] are skipped in the code below,\n    // they should not be treated as styling-based bindings since\n    // they are intended to be written directly to the attr and\n    // will therefore skip all style/class resolution that is present\n    // with style=\"\", [style]=\"\" and [style.prop]=\"\", class=\"\",\n    // [class.prop]=\"\". [class]=\"\" assignments\n    let binding: BoundStylingEntry|null = null;\n    let name = input.name;\n    switch (input.type) {\n      case BindingType.Property:\n        binding = this.registerInputBasedOnName(name, input.value, input.sourceSpan);\n        break;\n      case BindingType.Style:\n        binding = this.registerStyleInput(name, false, input.value, input.sourceSpan, input.unit);\n        break;\n      case BindingType.Class:\n        binding = this.registerClassInput(name, false, input.value, input.sourceSpan);\n        break;\n    }\n    return binding ? true : false;\n  }\n\n  registerInputBasedOnName(name: string, expression: AST, sourceSpan: ParseSourceSpan) {\n    let binding: BoundStylingEntry|null = null;\n    const prefix = name.substring(0, 6);\n    const isStyle = name === 'style' || prefix === 'style.' || prefix === 'style!';\n    const isClass = !isStyle && (name === 'class' || prefix === 'class.' || prefix === 'class!');\n    if (isStyle || isClass) {\n      const isMapBased = name.charAt(5) !== '.';         // style.prop or class.prop makes this a no\n      const property = name.substr(isMapBased ? 5 : 6);  // the dot explains why there's a +1\n      if (isStyle) {\n        binding = this.registerStyleInput(property, isMapBased, expression, sourceSpan);\n      } else {\n        binding = this.registerClassInput(property, isMapBased, expression, sourceSpan);\n      }\n    }\n    return binding;\n  }\n\n  registerStyleInput(\n      name: string, isMapBased: boolean, value: AST, sourceSpan: ParseSourceSpan,\n      suffix?: string|null): BoundStylingEntry|null {\n    if (isEmptyExpression(value)) {\n      return null;\n    }\n    // CSS custom properties are case-sensitive so we shouldn't normalize them.\n    // See: https://www.w3.org/TR/css-variables-1/#defining-variables\n    if (!isCssCustomProperty(name)) {\n      name = hyphenate(name);\n    }\n    const {property, hasOverrideFlag, suffix: bindingSuffix} = parseProperty(name);\n    suffix = typeof suffix === 'string' && suffix.length !== 0 ? suffix : bindingSuffix;\n    const entry:\n        BoundStylingEntry = {name: property, suffix: suffix, value, sourceSpan, hasOverrideFlag};\n    if (isMapBased) {\n      this._styleMapInput = entry;\n    } else {\n      (this._singleStyleInputs = this._singleStyleInputs || []).push(entry);\n      registerIntoMap(this._stylesIndex, property);\n    }\n    this._lastStylingInput = entry;\n    this._firstStylingInput = this._firstStylingInput || entry;\n    this._checkForPipes(value);\n    this.hasBindings = true;\n    return entry;\n  }\n\n  registerClassInput(name: string, isMapBased: boolean, value: AST, sourceSpan: ParseSourceSpan):\n      BoundStylingEntry|null {\n    if (isEmptyExpression(value)) {\n      return null;\n    }\n    const {property, hasOverrideFlag} = parseProperty(name);\n    const entry:\n        BoundStylingEntry = {name: property, value, sourceSpan, hasOverrideFlag, suffix: null};\n    if (isMapBased) {\n      this._classMapInput = entry;\n    } else {\n      (this._singleClassInputs = this._singleClassInputs || []).push(entry);\n      registerIntoMap(this._classesIndex, property);\n    }\n    this._lastStylingInput = entry;\n    this._firstStylingInput = this._firstStylingInput || entry;\n    this._checkForPipes(value);\n    this.hasBindings = true;\n    return entry;\n  }\n\n  private _checkForPipes(value: AST) {\n    if ((value instanceof ASTWithSource) && (value.ast instanceof BindingPipe)) {\n      this.hasBindingsWithPipes = true;\n    }\n  }\n\n  /**\n   * Registers the element's static style string value to the builder.\n   *\n   * @param value the style string (e.g. `width:100px; height:200px;`)\n   */\n  registerStyleAttr(value: string) {\n    this._initialStyleValues = parseStyle(value);\n    this._hasInitialValues = true;\n  }\n\n  /**\n   * Registers the element's static class string value to the builder.\n   *\n   * @param value the className string (e.g. `disabled gold zoom`)\n   */\n  registerClassAttr(value: string) {\n    this._initialClassValues = value.trim().split(/\\s+/g);\n    this._hasInitialValues = true;\n  }\n\n  /**\n   * Appends all styling-related expressions to the provided attrs array.\n   *\n   * @param attrs an existing array where each of the styling expressions\n   * will be inserted into.\n   */\n  populateInitialStylingAttrs(attrs: o.Expression[]): void {\n    // [CLASS_MARKER, 'foo', 'bar', 'baz' ...]\n    if (this._initialClassValues.length) {\n      attrs.push(o.literal(AttributeMarker.Classes));\n      for (let i = 0; i < this._initialClassValues.length; i++) {\n        attrs.push(o.literal(this._initialClassValues[i]));\n      }\n    }\n\n    // [STYLE_MARKER, 'width', '200px', 'height', '100px', ...]\n    if (this._initialStyleValues.length) {\n      attrs.push(o.literal(AttributeMarker.Styles));\n      for (let i = 0; i < this._initialStyleValues.length; i += 2) {\n        attrs.push(\n            o.literal(this._initialStyleValues[i]), o.literal(this._initialStyleValues[i + 1]));\n      }\n    }\n  }\n\n  /**\n   * Builds an instruction with all the expressions and parameters for `elementHostAttrs`.\n   *\n   * The instruction generation code below is used for producing the AOT statement code which is\n   * responsible for registering initial styles (within a directive hostBindings' creation block),\n   * as well as any of the provided attribute values, to the directive host element.\n   */\n  assignHostAttrs(attrs: o.Expression[], definitionMap: DefinitionMap): void {\n    if (this._directiveExpr && (attrs.length || this._hasInitialValues)) {\n      this.populateInitialStylingAttrs(attrs);\n      definitionMap.set('hostAttrs', o.literalArr(attrs));\n    }\n  }\n\n  /**\n   * Builds an instruction with all the expressions and parameters for `classMap`.\n   *\n   * The instruction data will contain all expressions for `classMap` to function\n   * which includes the `[class]` expression params.\n   */\n  buildClassMapInstruction(valueConverter: ValueConverter): StylingInstruction|null {\n    if (this._classMapInput) {\n      return this._buildMapBasedInstruction(valueConverter, true, this._classMapInput);\n    }\n    return null;\n  }\n\n  /**\n   * Builds an instruction with all the expressions and parameters for `styleMap`.\n   *\n   * The instruction data will contain all expressions for `styleMap` to function\n   * which includes the `[style]` expression params.\n   */\n  buildStyleMapInstruction(valueConverter: ValueConverter): StylingInstruction|null {\n    if (this._styleMapInput) {\n      return this._buildMapBasedInstruction(valueConverter, false, this._styleMapInput);\n    }\n    return null;\n  }\n\n  private _buildMapBasedInstruction(\n      valueConverter: ValueConverter, isClassBased: boolean,\n      stylingInput: BoundStylingEntry): StylingInstruction {\n    // each styling binding value is stored in the LView\n    // map-based bindings allocate two slots: one for the\n    // previous binding value and another for the previous\n    // className or style attribute value.\n    let totalBindingSlotsRequired = MIN_STYLING_BINDING_SLOTS_REQUIRED;\n\n    // these values must be outside of the update block so that they can\n    // be evaluated (the AST visit call) during creation time so that any\n    // pipes can be picked up in time before the template is built\n    const mapValue = stylingInput.value.visit(valueConverter);\n    let reference: o.ExternalReference;\n    if (mapValue instanceof Interpolation) {\n      totalBindingSlotsRequired += mapValue.expressions.length;\n      reference = isClassBased ? getClassMapInterpolationExpression(mapValue) :\n                                 getStyleMapInterpolationExpression(mapValue);\n    } else {\n      reference = isClassBased ? R3.classMap : R3.styleMap;\n    }\n\n    return {\n      reference,\n      calls: [{\n        supportsInterpolation: true,\n        sourceSpan: stylingInput.sourceSpan,\n        allocateBindingSlots: totalBindingSlotsRequired,\n        params: (convertFn: (value: any) => o.Expression|o.Expression[]) => {\n          const convertResult = convertFn(mapValue);\n          const params = Array.isArray(convertResult) ? convertResult : [convertResult];\n          return params;\n        }\n      }]\n    };\n  }\n\n  private _buildSingleInputs(\n      reference: o.ExternalReference, inputs: BoundStylingEntry[], valueConverter: ValueConverter,\n      getInterpolationExpressionFn: ((value: Interpolation) => o.ExternalReference)|null,\n      isClassBased: boolean): StylingInstruction[] {\n    const instructions: StylingInstruction[] = [];\n\n    inputs.forEach(input => {\n      const previousInstruction: StylingInstruction|undefined =\n          instructions[instructions.length - 1];\n      const value = input.value.visit(valueConverter);\n      let referenceForCall = reference;\n\n      // each styling binding value is stored in the LView\n      // but there are two values stored for each binding:\n      //   1) the value itself\n      //   2) an intermediate value (concatenation of style up to this point).\n      //      We need to store the intermediate value so that we don't allocate\n      //      the strings on each CD.\n      let totalBindingSlotsRequired = MIN_STYLING_BINDING_SLOTS_REQUIRED;\n\n      if (value instanceof Interpolation) {\n        totalBindingSlotsRequired += value.expressions.length;\n\n        if (getInterpolationExpressionFn) {\n          referenceForCall = getInterpolationExpressionFn(value);\n        }\n      }\n\n      const call = {\n        sourceSpan: input.sourceSpan,\n        allocateBindingSlots: totalBindingSlotsRequired,\n        supportsInterpolation: !!getInterpolationExpressionFn,\n        params: (convertFn: (value: any) => o.Expression | o.Expression[]) => {\n          // params => stylingProp(propName, value, suffix)\n          const params: o.Expression[] = [];\n          params.push(o.literal(input.name));\n\n          const convertResult = convertFn(value);\n          if (Array.isArray(convertResult)) {\n            params.push(...convertResult);\n          } else {\n            params.push(convertResult);\n          }\n\n          // [style.prop] bindings may use suffix values (e.g. px, em, etc...), therefore,\n          // if that is detected then we need to pass that in as an optional param.\n          if (!isClassBased && input.suffix !== null) {\n            params.push(o.literal(input.suffix));\n          }\n\n          return params;\n        }\n      };\n\n      // If we ended up generating a call to the same instruction as the previous styling property\n      // we can chain the calls together safely to save some bytes, otherwise we have to generate\n      // a separate instruction call. This is primarily a concern with interpolation instructions\n      // where we may start off with one `reference`, but end up using another based on the\n      // number of interpolations.\n      if (previousInstruction && previousInstruction.reference === referenceForCall) {\n        previousInstruction.calls.push(call);\n      } else {\n        instructions.push({reference: referenceForCall, calls: [call]});\n      }\n    });\n\n    return instructions;\n  }\n\n  private _buildClassInputs(valueConverter: ValueConverter): StylingInstruction[] {\n    if (this._singleClassInputs) {\n      return this._buildSingleInputs(\n          R3.classProp, this._singleClassInputs, valueConverter, null, true);\n    }\n    return [];\n  }\n\n  private _buildStyleInputs(valueConverter: ValueConverter): StylingInstruction[] {\n    if (this._singleStyleInputs) {\n      return this._buildSingleInputs(\n          R3.styleProp, this._singleStyleInputs, valueConverter,\n          getStylePropInterpolationExpression, false);\n    }\n    return [];\n  }\n\n  /**\n   * Constructs all instructions which contain the expressions that will be placed\n   * into the update block of a template function or a directive hostBindings function.\n   */\n  buildUpdateLevelInstructions(valueConverter: ValueConverter) {\n    const instructions: StylingInstruction[] = [];\n    if (this.hasBindings) {\n      const styleMapInstruction = this.buildStyleMapInstruction(valueConverter);\n      if (styleMapInstruction) {\n        instructions.push(styleMapInstruction);\n      }\n      const classMapInstruction = this.buildClassMapInstruction(valueConverter);\n      if (classMapInstruction) {\n        instructions.push(classMapInstruction);\n      }\n      instructions.push(...this._buildStyleInputs(valueConverter));\n      instructions.push(...this._buildClassInputs(valueConverter));\n    }\n    return instructions;\n  }\n}\n\nfunction registerIntoMap(map: Map<string, number>, key: string) {\n  if (!map.has(key)) {\n    map.set(key, map.size);\n  }\n}\n\nexport function parseProperty(name: string):\n    {property: string, suffix: string|null, hasOverrideFlag: boolean} {\n  let hasOverrideFlag = false;\n  const overrideIndex = name.indexOf(IMPORTANT_FLAG);\n  if (overrideIndex !== -1) {\n    name = overrideIndex > 0 ? name.substring(0, overrideIndex) : '';\n    hasOverrideFlag = true;\n  }\n\n  let suffix: string|null = null;\n  let property = name;\n  const unitIndex = name.lastIndexOf('.');\n  if (unitIndex > 0) {\n    suffix = name.substr(unitIndex + 1);\n    property = name.substring(0, unitIndex);\n  }\n\n  return {property, suffix, hasOverrideFlag};\n}\n\n/**\n * Gets the instruction to generate for an interpolated class map.\n * @param interpolation An Interpolation AST\n */\nfunction getClassMapInterpolationExpression(interpolation: Interpolation): o.ExternalReference {\n  switch (getInterpolationArgsLength(interpolation)) {\n    case 1:\n      return R3.classMap;\n    case 3:\n      return R3.classMapInterpolate1;\n    case 5:\n      return R3.classMapInterpolate2;\n    case 7:\n      return R3.classMapInterpolate3;\n    case 9:\n      return R3.classMapInterpolate4;\n    case 11:\n      return R3.classMapInterpolate5;\n    case 13:\n      return R3.classMapInterpolate6;\n    case 15:\n      return R3.classMapInterpolate7;\n    case 17:\n      return R3.classMapInterpolate8;\n    default:\n      return R3.classMapInterpolateV;\n  }\n}\n\n/**\n * Gets the instruction to generate for an interpolated style map.\n * @param interpolation An Interpolation AST\n */\nfunction getStyleMapInterpolationExpression(interpolation: Interpolation): o.ExternalReference {\n  switch (getInterpolationArgsLength(interpolation)) {\n    case 1:\n      return R3.styleMap;\n    case 3:\n      return R3.styleMapInterpolate1;\n    case 5:\n      return R3.styleMapInterpolate2;\n    case 7:\n      return R3.styleMapInterpolate3;\n    case 9:\n      return R3.styleMapInterpolate4;\n    case 11:\n      return R3.styleMapInterpolate5;\n    case 13:\n      return R3.styleMapInterpolate6;\n    case 15:\n      return R3.styleMapInterpolate7;\n    case 17:\n      return R3.styleMapInterpolate8;\n    default:\n      return R3.styleMapInterpolateV;\n  }\n}\n\n/**\n * Gets the instruction to generate for an interpolated style prop.\n * @param interpolation An Interpolation AST\n */\nfunction getStylePropInterpolationExpression(interpolation: Interpolation) {\n  switch (getInterpolationArgsLength(interpolation)) {\n    case 1:\n      return R3.styleProp;\n    case 3:\n      return R3.stylePropInterpolate1;\n    case 5:\n      return R3.stylePropInterpolate2;\n    case 7:\n      return R3.stylePropInterpolate3;\n    case 9:\n      return R3.stylePropInterpolate4;\n    case 11:\n      return R3.stylePropInterpolate5;\n    case 13:\n      return R3.stylePropInterpolate6;\n    case 15:\n      return R3.stylePropInterpolate7;\n    case 17:\n      return R3.stylePropInterpolate8;\n    default:\n      return R3.stylePropInterpolateV;\n  }\n}\n\n/**\n * Checks whether property name is a custom CSS property.\n * See: https://www.w3.org/TR/css-variables-1\n */\nfunction isCssCustomProperty(name: string): boolean {\n  return name.startsWith('--');\n}\n"]}
Note: See TracBrowser for help on using the repository browser.