Web Apps mit
VAADIN 14

Tobse Fritz

  • 💰  Reich
  • 🦄 Gut aussehend
     
  • 🐱‍👤 Kotlin Fanboy
  • 🔨  Java Entwickler
  • 🧹 Clean Coder  
  • 📸 Hobby Photographer
  • 🎮 Gaming Nerd  
  • 💼 ITscope GmbH

Highlights

  • Vaadin what?
  • UI Entwicklung
    • Flow Componenten
    • Element API
    • Custom HTML
    • Polymer Templates
    • Web-Components
  • Progressive Web App
  • Vaadin und der Rest der Welt
  • Risiken und Nebenwirkungen
  • Ausblick

Vaadin what?

  • Vaadin 6
  • 2009
  • 2010
  • Vaadin Directory
  • 2020
  • Vaadin 14
  • 2007
  • IT Mill Toolkit
  • Ajax  → GWT
  • Swing like backend
  • Vaadin 10
  • Web Components
  • Modern mobile first web
  • 2018

Flow Componenten

34

Standard-
Komponenten

Clean Code App

VerticalLayout layout = new VerticalLayout();
TextField userName = new TextField();
userName.setCaption("Benutzername");
userName.setValue("Alex");
userName.addValueChangeListener(e -> profile.setName(e.getValue()));
layout.addComponent(userName);
ProfileView.java

Flow Componenten

Flow Databinding

Flow Databinding

GradeTopic
+ name: String
+ description: String
+ sectionWhys: String
GradeRating
+ Responsibility: 
   {Team, SingleDev}
+ evolvability: Int
+ correctness: String
Collection<GradeTopic> gradeTopics = loadData(); 
Grid<GradeTopic> table = new Grid<>();
table.setItems(gradeTopics);

table.addColumn(GradeTopic::getName)
        .setHeader(getTranslation("grade.name"))
        .setSortable(true).setAutoWidth(true);

table.addColumn(this::formatTopic)
        .setHeader(getTranslation("grade.grade"))
        .setSortable(true).setAutoWidth(true);

table.addColumn(topic -> topic.getGradeRating().getResponsibility()
        .setHeader(getTranslation("grade.rating.responsibility"))
        .setSortable(true).setAutoWidth(true);

table.sort(List.of(new GridSortOrder<>(columnName, ASCENDING)));

table.addSelectionListener(event -> event.getFirstSelectedItem()
                            .ifPresent(this::openGradeTopic));
CompendiumView.java

Flow Databinding

Vaadin Designer

https://vaadin.com/designer

Element API

Anchor
Article
Aside
Description
DescriptionList
Div
Emphasis
Footer
H1
H2
H3
H4
H5
H6
Header
Image
Label
ListItem
Main
NativeButton
Nav
OrderedList
Paragraph
Pre
Section
Span
Term
UnorderedList
HtmlContainer

Element API

public class GradeProgressBar extends FlexLayout {

    public GradeProgressBar() {
        setWidthFull();
        setJustifyContentMode(JustifyContentMode.EVENLY);
        setAlignItems(Alignment.END);
        addClassName("grade-progress");
    }
    
   public void setDays(List<ProgressDay> days) {
        days.forEach(this::add);
    }
}
GradeProgressBar : FlexLayout
ProgressDay : Div
Span

Element API

public static class ProgressDay extends Div {

  public ProgressDay(String tooltip) {
    addClassName("grade-progress-day");
    addToolTip(tooltip);
  }

  public void setDaySubmitted() {
  	addClassName("day-submitted");
  }

  public void setDayInFuture() {
  	addClassName("day-future");
  }
  
 private void addToolTip(String tooltip) {
    addClassName("tooltip");
    Span span = new Span(tooltip);
    span.addClassName("tooltiptext");
    span.addClassName("tooltip-bottom");
    add(span);
  }

}
.grade-progress {
    height: 45px;
}

.grade-progress-day {
    background-color: grey;
    color: grey;
    width: 10px;
    height: 40px;
    transition: border-top 0.8s;
    top: 0;
    bottom: 0;
}

.grade-progress-day:hover {
    border-top: solid currentColor 10px;
}

.day-submitted {
    background-color: green;
}

.day-future {
    background-color: black;
}

Custom HTML

content.add(new Html(getTranslation("about.info")));
content.add(new Html(getTranslation("about.info.technical")));
about.info = <p>Hast du schon einmal versucht den Weg eines <a href="https://clean-code-developer.de/" target="blank">Clean Code Developers</a> zu gehen und den weißen grad zu meistern? Diese Webapp hilft Dir diesen Weg einzuhalten und erinnert dich an deine aktuellen Ziele. Logge deinen Fortschritt und bekomme Belohnungen für deine erreichten Ziele 🏆.</p> 
about.info.technical = <p>Dieses Projekt ist eine Beispielapplikation für eine <a href="https://vaadin.com/" target="blank">Vaadin 14</a> Porgressive Web App. Es nutzt <a href="https://spring.io/projects/spring-boot" target="blank">Spring Boot</a> und läuft auf <a href="https://www.oracle.com/technetwork/java/javase/12-relnote-issues-5211422.html" target="blank">Java 12</a>. Diese Demo löscht alle Nutzerdaten sobald die Session verloren geht.</p>

Web Components

Benutzerdefinierte Elemente

HTML-Vorlagen

Shadow DOM

Web Components

Polymer WebComponent - l2t-paper-color

Web Components

developer.mozilla.org/de/docs/Web/Web_Components

Polymer Templates

web components library

  • Custom elements
  • Shadow DOM
  • Events
  • Data Binding

Polymer Templates

web components library

Polymer 2.0

  • HTML Imports
  • Bower

05.2017

Polymer 3.0

  • ES6 Modules
  • npm

01.2018

LitElement

  • Lightweigt
    Polymer

02.2019

Polymer Templates

import {PolymerElement, html} from '@polymer/polymer/polymer-element.js';

class MyElement extends PolymerElement {

  static get template() {
    return html`
      <style> .mood { color: orange; } </style>
      <h2>Web Components are <span class="mood">[[message]]</span>!</h2>
      `;
  }
  
   static get properties() { return { message: String }}
}

customElements.define('my-element', MyElement);

Polymer Templates

Polymer in Vaadin

Polymer in Vaadin

import {html, PolymerElement} from '@polymer/polymer/polymer-element.js';
import '@polymer/iron-icon/iron-icon';
import '@vaadin/vaadin-material-styles/color';

class LogEntry extends PolymerElement {

    static get template() {
        return html`<style>
    #logentry {
        width: 380px;
    }

    #header {
        display: flex;
    }

    #title {
        flex-grow: 1;
    }

    #username {
        font-size: 18px;
    }
    
    #grade {
        font-size: 16px;
        line-height: 10px;
    }

    #date {
        color: gray;
        align-self: flex-end;
    }

    #content-head {
        display: flex;
        justify-content: space-between;
    }

    #experience {
        display: flex;
        align-self: flex-end;
    }

    #xp {
        color: #22C4FA;
        margin-left: 8px;
        font-size: 18px;
    }

    #skill-points {
        color: #20E87B;
    }

    #topic {
        font-size: 18px;
    }

   #content {
        display: flex;
        flex-direction: column;
        background-color: var(--material-secondary-background-color);
        border-radius: 5px;
        padding: 5px;
    } 

    #actions {
        align-self: flex-end;
    }
    
    #avatar{
        margin-right: 5px;
    }
    
    iron-icon.avatar {
        --iron-icon-height: 38px;
        --iron-icon-width: 38px;
    }

    iron-icon.type {
        --iron-icon-height: 18px;
        --iron-icon-width: 18px;
    }
    iron-icon.action {
        --iron-icon-height: 22px;
        --iron-icon-width: 22px;
    }
</style>

<div id="logentry">
    <div id="header">
        <div id="avatar">
            <iron-icon class="avatar" icon="vaadin:user"></iron-icon>
        </div>
        <div id="title">
            <div id="username">[[username]]</div>
            <div id="grade">[[grade]]</div>
        </div>
        <div id="date">[[date]]</div>
    </div>
    <div id="content">
        <div id="content-head">
            <div id="topic">[[topic]]</div>
            <div id="type">
                <iron-icon class="type" icon="[[typeicon]]"></iron-icon>
            </div>
        </div>
        <div id="comment">[[comment]]</div>
           <div id="experience">
            <div id="skill-points">[[skills]]</div>
            <div id="xp">[[experience]]</div>
        </div>
        <div id="actions">
        <iron-icon class="action" icon="vaadin:close-small"></iron-icon>
        <iron-icon class="action" icon="vaadin:comment"></iron-icon>
        <iron-icon class="action" icon="vaadin:pencil"></iron-icon></div>
    </div>
</div>`;
    }

    static get is() {
        return 'log-entry';
    }
}

customElements.define(LogEntry.is, LogEntry);

Polymer in Vaadin

@Tag("log-entry")
@JsModule("./src/log-entry.js")
public class LogEntryComponent extends PolymerTemplate<LogEntryModel> {

    public LogEntryComponent() {
        setId("logentry");
    }

    @Override
    public LogEntryModel getModel() {
        return super.getModel();
    }

}
public interface LogEntryModel extends TemplateModel {

    String getTypeicon();
    
    void setTypeicon(String typeicon);

    String getSkills();
    
    void setSkills(String skills);

    String getTopic();

    void setTopic(String topic);

    String getGrade();

    void setGrade(String grade);

    String getExperience();

    void setExperience(String experience);

    String getComment();

    void setComment(String comment);

    String getUsername();

    void setUsername(String username);

    String getDate();

    void setDate(String date);
}

Polymer in Vaadin

private void addLogEntry(LogEntry logEntry) {
    var logEntryComponent = new LogEntryComponent();
    LogEntryModel logEntryModel = logEntryComponent.getModel();
    logEntryModel.setComment(logEntry.getComment());
    logEntryModel.setUsername(logEntry.getName());
    parent.add(logEntryComponent);
}
JournalView.java

Web Components

Web Components

@Tag("paper-fab-speed-dial")
@NpmPackage(value = "@cwmr/paper-fab-speed-dial", version = "3.0.0")
@JsModule("@cwmr/paper-fab-speed-dial/paper-fab-speed-dial.js")
public class SpeedDial extends Component implements HasEnabled {

  public SpeedDialAction addMenuItem(String item, Icon icon) {
    SpeedDialAction speedDialAction = new SpeedDialAction(item, icon);
    getElement().appendChild(speedDialAction.getElement());
    return speedDialAction;
  }

  public SpeedDialAction addMenuItem(String item, Icon icon, 
    ComponentEventListener<SpeedDialAction.ClickEvent> listener) {
    SpeedDialAction speedDialAction = addMenuItem(item, icon);
    speedDialAction.addClickListener(listener);
    return speedDialAction;
  }

  @Synchronize("opened-changed")
  public boolean isOpened() {
  	return getElement().getProperty("opened", false);
  }

  public void setBackdrop(boolean backdrop) {
    if (backdrop) {
    	getElement().setAttribute("with-backdrop", EMPTY);
    } else {
    	getElement().removeAttribute("with-backdrop");
    }
  }

  public void close() {
  	getElement().setProperty("opened", false);
  }

  public void open() {
  	getElement().setProperty("opened", true);
  }

  @Override
  public void setEnabled(boolean enabled) {
    if (enabled) {
    	getElement().setAttribute("disabled", "disabled");
    } else {
    	getElement().removeAttribute("disabled");
    }
  }

  public void setColorAction(String color) {
  	setStyle("--paper-fab-speed-dial-action-background", color);
  }

  public void setColor(String color) {
  	setStyle("--paper-fab-speed-dial-background", color);
  }

  public void setMarginRight(String marginRight) {
  	setStyle("--paper-fab-speed-dial-right", marginRight);
  }

  private Style setStyle(String name, String value) {
  	return getElement().getStyle().set(name, value);
  }
}

Web Components

@Tag("paper-fab-speed-dial-action")
@NpmPackage(value = "@cwmr/paper-fab-speed-dial", version = "3.0.0")
@JsModule("@cwmr/paper-fab-speed-dial/paper-fab-speed-dial-action.js")
public class SpeedDialAction extends Label implements HasEnabled, HasSpeedDialStyle {

  public SpeedDialAction(String text, Icon icon) {
    super(text);
    setIcon(icon);
  }

  public Element setIcon(Icon icon) {
  	return getElement().setAttribute("icon", getIconAttribute(icon));
  }

  public Registration addClickListener(ComponentEventListener<SpeedDialClickEvent> listener) {
  	return addListener(SpeedDialClickEvent.class, listener);
  }

  private String getIconAttribute(Icon icon) {
  	return icon.getElement().getAttribute("icon");
  }

}
SpeedDialAction.java

Web Components

SpeedDial speedDial = new SpeedDial();
speedDial.addMenuItem("Erfolg", VaadinIcon.TROPHY, e -> Notification.show("Clicked E"));
speedDial.addMenuItem("Log", VaadinIcon.NOTEBOOK, e -> Notification.show("Clicked L"));
speedDial.setBackdrop(true);
add(speedDial);
SpeedDialSampleView.java

Styling mit CSS Variablen

public void setColorAction(String color) {
   setStyle("--paper-fab-speed-dial-action-background", color);
}
SpeedDial speedDial = new SpeedDial();
setColor("var(--material-primary-text-color)");
setColorAction("var(--material-primary-text-color)");
add(speedDial);
SampleCssVar.java
SpeedDial.java
  • 1829 "Components"
  • 582 Webcompontens
  • 200 "Platform"

Vaadin Directory

vaadin.com/directory

Vaadin Directory

⭐ ServerSide ClickListener

Setters für alle 11 CSS Properties

Build in Support für Vaadin Icons

Vaadin Core - Pro Lizenz nicht erforderlich

Paper Speed Dial

Progressive Web App

  • Zuverlässig
    • Sofort einsatzbereit
    • Offlinefähig
  • Schnell
    • Schnelles Nutzerfeedback
  • Integriert
    • Fühlt sich wie lokale App an
    • Lässt sich installieren
    • Funktioniert auf Desktop & Mobil

Progressive Web Ap

@CssImport("./styles/shared-styles.css")
@Theme(value = Material.class, variant = Lumo.DARK)
@PWA(name = "Clean Code Developer Journal", backgroundColor = "#3B3B3B")
public class MainView extends AppLayout {

  public MainView() {
    addToNavbar(new DrawerToggle());
    addToNavbar(new H4(getTranslation("app.name")));
    addToDrawer(createMenuBar());
  }

  private VerticalLayout createMenuBar() {
    RouterLink journal = new RouterLink("Journal", JournalView.class);
    RouterLink profile = new RouterLink("Profil", ProfileView.class);
    RouterLink achievements = new RouterLink("Erfolge", AchievementsView.class);
    RouterLink compendium = new RouterLink("Compendium", CompendiumView.class);
    RouterLink about = new RouterLink("Über", AboutView.class);
    return new VerticalLayout(journal, profile, achievements, compendium, about);
  }

}
MainView.java

Progressive Web Ap

Integriert

Progressive Web Ap

Mobile & Web

Progressive Web Ap

Offlinefähigkeit

Vaadin
und der Rest der Welt

 Risiken und Nebenwirkungen

  • UI muss serialisierbar sein
  • Kein vorgegebenes UI Pattern
  • Speicherverbrauch
  • Hot Code Replacement
    • DCEVM
    • Spring Devtools
    • JRebel

 

 Ausblick

  • Vaadin 14
    • Router
    • @Push
    • LitElement 
  • Vaadin 15
    • Supportende für IE 11, Safari 9-12
    • Flow 3.0 - Typescript Support
    • pnpm - Schnelleres Setup
    • Vaadin on Kotlin (Vok)
    • Offline Components
    • Karibu-DSL
    • Karibu-Testing

THE END

Clean-Code.rocks

github.com/TobseF/CleanCodeDeveloperJournal

THE END

Quellen

Clean-Code.rocks

github.com/TobseF/CleanCodeDeveloperJournal

Web Apps mit Vaadin 14

By Tobse Fritz

Web Apps mit Vaadin 14

Moderne Progressive-Web-Apps mit Java 14 und Vaadin 14

  • 964