| # Copyright 2019 The Chromium Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| from style_variable_generator.base_generator import Color, Modes, BaseGenerator, VariableType |
| import collections |
| |
| |
| class CSSStyleGenerator(BaseGenerator): |
| '''Generator for CSS Variables''' |
| |
| @staticmethod |
| def GetName(): |
| return 'CSS' |
| |
| def Render(self): |
| self.Validate() |
| return self.ApplyTemplate(self, 'css_generator.tmpl', |
| self.GetParameters()) |
| |
| def GetParameters(self): |
| if self.generate_single_mode: |
| resolved_colors = self.model[VariableType.COLOR].Flatten( |
| resolve_missing=True) |
| resolved_opacities = self.model[VariableType.OPACITY].Flatten( |
| resolve_missing=True) |
| colors = { |
| Modes.DEFAULT: resolved_colors[self.generate_single_mode] |
| } |
| opacities = { |
| Modes.DEFAULT: resolved_opacities[self.generate_single_mode] |
| } |
| else: |
| colors = self.model[VariableType.COLOR].Flatten() |
| opacities = self.model[VariableType.OPACITY].Flatten() |
| |
| return { |
| 'opacities': opacities, |
| 'colors': colors, |
| 'typography': self.model[VariableType.TYPOGRAPHY], |
| 'untyped_css': self.model[VariableType.UNTYPED_CSS], |
| } |
| |
| def GetFilters(self): |
| return { |
| 'to_css_var_name': self.ToCSSVarName, |
| 'css_color': self._CSSColor, |
| 'css_opacity': self._CSSOpacity, |
| 'css_color_rgb': self.CSSColorRGB, |
| 'process_simple_ref': self.ProcessSimpleRef, |
| } |
| |
| def GetGlobals(self): |
| return { |
| 'css_color_var': |
| self.CSSColorVar, |
| 'in_files': |
| sorted(self.in_file_to_context.keys()), |
| 'dark_mode_selector': |
| self.generator_options.get('dark_mode_selector', None), |
| 'debug_placeholder': |
| self.generator_options.get('debug_placeholder', ''), |
| 'suppress_sources_comment': |
| self.generator_options.get('suppress_sources_comment', False), |
| 'Modes': |
| Modes, |
| } |
| |
| def AddGeneratedVars(self, var_names, variable_type): |
| def AddVarNames(model_names, variations): |
| for model_name in model_names: |
| for v in variations: |
| var_name = v.replace('$css_name', |
| self.ToCSSVarName(model_name)) |
| if var_name in var_names: |
| raise ValueError(name + " is defined multiple times") |
| var_names[var_name] = model_name |
| |
| submodel = self.model[variable_type] |
| if variable_type == VariableType.OPACITY: |
| AddVarNames(submodel.keys(), ['$css_name']) |
| elif variable_type == VariableType.COLOR: |
| AddVarNames(submodel.keys(), ['$css_name', '$css_name-rgb']) |
| elif variable_type == VariableType.UNTYPED_CSS: |
| for category in submodel.values(): |
| AddVarNames(category.keys(), ['$css_name']) |
| elif variable_type == VariableType.TYPOGRAPHY: |
| AddVarNames(submodel.typefaces.keys(), [ |
| '$css_name-font', |
| '$css_name-font-family', |
| '$css_name-font-size', |
| '$css_name-font-weight', |
| '$css_name-line-height', |
| ]) |
| else: |
| raise ValueError("GetGeneratedVars() for '%s' not implemented") |
| |
| def GetCSSVarNames(self): |
| '''Returns a map of all generated names to the model names that |
| generated them. |
| ''' |
| var_names = dict() |
| for vt in VariableType.ALL: |
| generated = self.AddGeneratedVars(var_names, vt) |
| |
| return var_names |
| |
| def ProcessSimpleRef(self, value): |
| '''If |value| is a simple '$other_variable' reference, returns a |
| CSS variable that points to '$other_variable'.''' |
| if value.startswith('$'): |
| ref_name = value[1:] |
| assert self.context_map[ref_name] |
| value = 'var({0})'.format(self.ToCSSVarName(ref_name)) |
| |
| return value |
| |
| def _GetCSSVarPrefix(self, model_name): |
| prefix = self.context_map[model_name].get(CSSStyleGenerator.GetName(), |
| {}).get('prefix') |
| return prefix + '-' if prefix else '' |
| |
| def ToCSSVarName(self, model_name): |
| return '--%s%s' % (self._GetCSSVarPrefix(model_name), |
| model_name.replace('_', '-')) |
| |
| def _CSSOpacity(self, opacity): |
| if opacity.var: |
| return 'var(%s)' % self.ToCSSVarName(opacity.var) |
| |
| return ('%f' % opacity.a).rstrip('0').rstrip('.') |
| |
| def _CSSColor(self, c): |
| '''Returns the CSS color representation of |c|''' |
| assert (isinstance(c, Color)) |
| if c.var: |
| return 'var(%s)' % self.ToCSSVarName(c.var) |
| |
| if c.rgb_var: |
| if c.opacity.a != 1: |
| return 'rgba(var(%s-rgb), %g)' % (self.ToCSSVarName( |
| c.RGBVarToVar()), self._CSSOpacity(c.opacity)) |
| else: |
| return 'rgb(var(%s-rgb))' % self.ToCSSVarName(c.RGBVarToVar()) |
| |
| elif c.a != 1: |
| return 'rgba(%d, %d, %d, %g)' % (c.r, c.g, c.b, |
| self._CSSOpacity(c.opacity)) |
| else: |
| return 'rgb(%d, %d, %d)' % (c.r, c.g, c.b) |
| |
| def CSSColorRGB(self, c): |
| '''Returns the CSS rgb representation of |c|''' |
| if c.var: |
| return 'var(%s-rgb)' % self.ToCSSVarName(c.var) |
| |
| if c.rgb_var: |
| return 'var(%s-rgb)' % self.ToCSSVarName(c.RGBVarToVar()) |
| |
| return '%d, %d, %d' % (c.r, c.g, c.b) |
| |
| def CSSColorVar(self, model_name, color): |
| '''Returns the CSS color representation given a color name and color''' |
| if color.var: |
| return 'var(%s)' % self.ToCSSVarName(color.var) |
| if color.opacity and color.opacity.a != 1: |
| return 'rgba(var(%s-rgb), %s)' % (self.ToCSSVarName(model_name), |
| self._CSSOpacity(color.opacity)) |
| else: |
| return 'rgb(var(%s-rgb))' % self.ToCSSVarName(model_name) |