Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Theme component with app theme. #9418

Merged
merged 7 commits into from
Nov 24, 2020
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
],
"repository": "vaadin/flow",
"name": "@vaadin/application-theme-plugin",
"version": "0.1.0",
"version": "0.1.2",
"main": "application-theme-plugin.js",
"author": "Vaadin Ltd",
"license": "Apache-2.0",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@
const glob = require('glob');
const path = require('path');

// Special folder inside a theme for component themes that go inside the component shadow root
const themeComponentsFolder = 'components';
// The contents of a global CSS file with this name in a theme is always added to
// the document. E.g. @font-face must be in this
const themeFileAlwaysAddToDocument = 'document.css';
Expand Down Expand Up @@ -48,13 +50,22 @@ function generateThemeFile(themeFolder, themeName) {
cwd: themeFolder,
nodir: true,
});
const componentsFiles = glob.sync('*.css', {
cwd: path.resolve(themeFolder, themeComponentsFolder),
nodir: true,
});

let themeFile = headerImport;

if(componentsFiles.length > 0){
themeFile += 'import { css, unsafeCSS, registerStyles } from \'@vaadin/vaadin-themable-mixin/register-styles\';';
}

themeFile += injectGlobalCssMethod;

const imports = [];
const globalCssCode = [];
const componentCssCode = [];

globalFiles.forEach((global) => {
const filename = path.basename(global);
Expand All @@ -66,8 +77,27 @@ function generateThemeFile(themeFolder, themeName) {
globalCssCode.push(`injectGlobalCss(${variable}.toString(), target);\n`);
});

componentsFiles.forEach((componentCss) => {
const filename = path.basename(componentCss);
const tag = filename.replace('.css', '');
const variable = camelCase(filename);
imports.push(
`import ${variable} from './${themeComponentsFolder}/${filename}';\n`
);
// Don't format as the generated file formatting will get wonky!
const componentString = `registerStyles(
'${tag}',
css\`
\${unsafeCSS(${variable}.toString())}
\`
);
`;
componentCssCode.push(componentString);
});

const themeIdentifier = '_vaadinds_' + themeName + '_';
const globalCssFlag = themeIdentifier + 'globalCss';
const componentCssFlag = themeIdentifier + 'componentCss';

themeFile += imports.join('');

Expand All @@ -77,6 +107,10 @@ function generateThemeFile(themeFolder, themeName) {
${globalCssCode.join('')}
target['${globalCssFlag}'] = true;
}
if (!document['${componentCssFlag}']) {
${componentCssCode.join('')}
document['${componentCssFlag}'] = true;
}
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While not entirely connected to this PR, we should probably document this thing a bit, as it is quite unclear what does this applyTheme do and when it is used. At least when just reading the code again after I while, it makes me go wat, what is this for and why ...

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added some comment as there was need to merge and do conflict resolution.

`;

Expand All @@ -94,7 +128,7 @@ function generateThemeFile(themeFolder, themeName) {
function camelCase(str) {
return str.replace(/(?:^\w|[A-Z]|\b\w)/g, function(word, index) {
return index === 0 ? word.toLowerCase() : word.toUpperCase();
}).replace(/\s+/g, '').replace(/\./g, '');
}).replace(/\s+/g, '').replace(/\.|\-/g, '');
}

module.exports = generateThemeFile;
210 changes: 210 additions & 0 deletions flow-tests/test-themes/frontend/my-field.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,210 @@
/*
* Copyright 2000-2020 Vaadin Ltd.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/

import '@vaadin/vaadin-lumo-styles/color.js';
import '@vaadin/vaadin-lumo-styles/sizing.js';
import '@vaadin/vaadin-lumo-styles/spacing.js';
import '@vaadin/vaadin-lumo-styles/style.js';
import '@vaadin/vaadin-lumo-styles/typography.js';
import '@vaadin/vaadin-lumo-styles/mixins/required-field.js';
import '@vaadin/vaadin-lumo-styles/font-icons.js';
import '@vaadin/vaadin-lumo-styles/mixins/field-button.js';
import { html } from '@polymer/polymer/lib/utils/html-tag.js';

const $_documentContainer = html`<dom-module id="lumo-text-field" theme-for="my-field">
caalador marked this conversation as resolved.
Show resolved Hide resolved
<template>
<style include="lumo-required-field lumo-field-button">
:host {
--lumo-text-field-size: var(--lumo-size-m);
color: var(--lumo-body-text-color);
font-size: var(--lumo-font-size-m);
font-family: var(--lumo-font-family);
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
-webkit-tap-highlight-color: transparent;
padding: var(--lumo-space-xs) 0;
display: inline-flex;
outline: none;
}

:host::before {
height: var(--lumo-text-field-size);
box-sizing: border-box;
display: inline-flex;
align-items: center;
content: "\\2003";
width: 0;
display: inline-block;
/* Size and position this element on the same vertical position as the input-field element
to make vertical align for the host element work as expected */
}

:host([focused]:not([readonly])) [part="label"] {
color: var(--lumo-primary-text-color);
}

[part="value"],
[part="input-field"] ::slotted(input),
[part="input-field"] ::slotted(textarea),
/* Slotted by vaadin-select-text-field */
[part="input-field"] ::slotted([part="value"]) {
cursor: inherit;
min-height: var(--lumo-text-field-size);
padding: 0 0.25em;
--_lumo-text-field-overflow-mask-image: linear-gradient(to left, transparent, #000 1.25em);
-webkit-mask-image: var(--_lumo-text-field-overflow-mask-image);
}

[part="value"]:focus,
:host([focused]) [part="input-field"] ::slotted(input),
:host([focused]) [part="input-field"] ::slotted(textarea) {
-webkit-mask-image: none;
mask-image: none;
}

[part="input-field"] {
border-radius: var(--lumo-border-radius);
background-color: var(--lumo-contrast-10pct);
padding: 0 calc(0.375em + var(--lumo-border-radius) / 4 - 1px);
font-weight: 500;
line-height: 1;
position: relative;
cursor: text;
box-sizing: border-box;
}

/* Used for hover and activation effects */
[part="input-field"]::after {
content: "";
position: absolute;
top: 0;
right: 0;
bottom: 0;
left: 0;
border-radius: inherit;
pointer-events: none;
background-color: var(--lumo-contrast-50pct);
opacity: 0;
transition: transform 0.15s, opacity 0.2s;
transform-origin: 100% 0;
}

/* Slotted content */

[part="input-field"] ::slotted(:not([part]):not(iron-icon):not(input):not(textarea)) {
color: var(--lumo-secondary-text-color);
font-weight: 400;
}

.my-field-container {
display: flex;
flex-direction: column;
min-width: 100%;
max-width: 100%;
width: var(--vaadin-text-field-default-width, 12em);
}

[part="input-field"] {
display: flex;
align-items: center;
flex: auto;
}

.my-field-container [part="input-field"] {
flex-grow: 0;
}

/* Reset the native input styles */
[part="value"],
[part="input-field"] ::slotted(input),
[part="input-field"] ::slotted(textarea) {
-webkit-appearance: none;
-moz-appearance: none;
outline: none;
margin: 0;
padding: 0;
border: 0;
border-radius: 0;
min-width: 0;
font: inherit;
font-size: 1em;
line-height: normal;
color: inherit;
background-color: transparent;
/* Disable default invalid style in Firefox */
box-shadow: none;
}

[part="input-field"] ::slotted(*) {
flex: none;
}

[part="value"],
[part="input-field"] ::slotted(input),
[part="input-field"] ::slotted(textarea),
/* Slotted by vaadin-select-text-field */
[part="input-field"] ::slotted([part="value"]) {
flex: auto;
white-space: nowrap;
overflow: hidden;
width: 100%;
height: 100%;
}

[part="input-field"] ::slotted(textarea) {
resize: none;
}

[part="value"]::-ms-clear,
[part="input-field"] ::slotted(input)::-ms-clear {
display: none;
}
</style>
</template>
</dom-module>`;

document.head.appendChild($_documentContainer.content);

import { PolymerElement } from '@polymer/polymer/polymer-element.js';
caalador marked this conversation as resolved.
Show resolved Hide resolved
import { ThemableMixin } from '@vaadin/vaadin-themable-mixin/vaadin-themable-mixin.js';

class MyField extends
ThemableMixin(PolymerElement) {
static get template() {
return html`

<div class="my-field-container">
<div part="input-field" id="input">
<slot name="input">
<input part="value">
</slot>
</div>
</div>
`;
}

static get is() {
return 'my-field';
}

static get version() {
return '0.0.1';
}
}

customElements.define(MyField.is, MyField);

export { MyField };
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
[part="input-field"] {
background: red;
}
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,11 @@

package com.vaadin.flow.uitest.ui.theme;

import com.vaadin.flow.component.dependency.NpmPackage;
import com.vaadin.flow.component.page.AppShellConfigurator;
import com.vaadin.flow.server.PWA;
import com.vaadin.flow.theme.Theme;

@Theme(value ="app-theme")
@PWA(name = "Project Base for Vaadin", shortName = "Project Base")
@NpmPackage(value = "@vaadin/vaadin-themable-mixin", version = "1.6.1")
public class AppShell implements AppShellConfigurator {
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
/*
* Copyright 2000-2020 Vaadin Ltd.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
* use this file except in compliance with the License. You may obtain a copy of
* the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations under
* the License.
*/
package com.vaadin.flow.uitest.ui.theme;

import com.vaadin.flow.component.Component;
import com.vaadin.flow.component.Tag;
import com.vaadin.flow.component.dependency.JsModule;

/**
* My field is a minimal text filed for component theme test with the app theme.
*/
@JsModule("./my-field.js")
@Tag("my-field")
public class MyField extends Component {

/**
* Set the component id.
*
* @param id
* value to set
* @return this component
*/
public Component withId(String id) {
setId(id);
return this;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -20,10 +20,14 @@
import com.vaadin.flow.component.html.Span;
import com.vaadin.flow.router.Route;

@Route("com.vaadin.flow.uitest.ui.theme.Theme")
@Route("com.vaadin.flow.uitest.ui.theme.ThemeView")
public class ThemeView extends Div {

public static final String MY_FIELD_ID = "field";

public ThemeView() {
add(new Span("This is the theme test view"));
add(new Div());
add(new MyField().withId(MY_FIELD_ID));
}
}
Loading