Exploring Predix UI seed
GE China Predix Bootcamp - Day 5
Shanghai foundry front-end team
data:image/s3,"s3://crabby-images/08304/0830432d34c23dca7cd56f32160eb1c7f57ba15c" alt=""
Required Environment
- Devbox
- CloudFoundry CLI
- NodeJS
- Git
- Sublime Text (or any code editor)
what does seed app offer
- Polymer web app-shell scaffold
- Example Predix UI Components usages
- Express web server connect to Predix services and APIs
- Built-in support of Predix UAA
seed-app overview
BFF - Backend for front-end
- NodeJS
- bower/npm
- Express Web Server
data:image/s3,"s3://crabby-images/569f3/569f3cc9e9545dbbaaa73767aacd5db7c8b7d7dc" alt=""
data:image/s3,"s3://crabby-images/6ed21/6ed21bad5317e0bf1ff6c9f78a61d0176d8b4890" alt=""
data:image/s3,"s3://crabby-images/87920/87920890d26b6b2f4a6210ff0652f03a1a67b011" alt=""
data:image/s3,"s3://crabby-images/f2705/f2705ee24d8019753211c652d65f1341d484c429" alt=""
data:image/s3,"s3://crabby-images/59bb9/59bb98aaf129460255a5bf04d2f6eb4ce3021201" alt=""
data:image/s3,"s3://crabby-images/56e4b/56e4bd3f16acf7aa498e95ddb01b9d00c3a6cc6d" alt=""
data:image/s3,"s3://crabby-images/02358/023583ca24eccccc1083b2bb8b49b4d7ce674597" alt=""
bower vs npm
data:image/s3,"s3://crabby-images/20f74/20f7482a61d3fe01f61dcf7deb5db80768b715db" alt=""
- non-blocking I/O
- Express middleware
- http-proxy middleware
Express Server
(request) blocking I/O
data:image/s3,"s3://crabby-images/149ab/149ab90a5d17976f4d6029a89b750e68937ff361" alt=""
(request) non-blocking I/O
data:image/s3,"s3://crabby-images/6d027/6d0273144088a5927e6ddeb638d0b2eb12fff2c6" alt=""
data:image/s3,"s3://crabby-images/32456/32456693b625da5affd549d1a58aa6cce4de75b5" alt=""
Middleware functions
- Execute any code
- Make changes to request/response
- End the request-response cycle
- Call the next middleware in stack
data:image/s3,"s3://crabby-images/dc220/dc2202325eab2b896cecb2ee7623d19a66f73776" alt=""
Express "hello world"
data:image/s3,"s3://crabby-images/38333/38333c0a60f4e46f4f1520c22d1d1dcca784df4a" alt=""
Express routes
data:image/s3,"s3://crabby-images/ca38b/ca38baed673417e14fd6266286acbf7e349c350d" alt=""
HTTP Proxy Middleware
data:image/s3,"s3://crabby-images/7938a/7938a231fb9e548c580808e886bac805ea645948" alt=""
data:image/s3,"s3://crabby-images/6bc7f/6bc7fd9f6d926fb53975fda9ddd0785984e4c539" alt=""
JSON-Server Middleware
Enable UAA in seed app
- Config UAA for authentication
- Secure Express routes
- Credentials in manifest file for Deploy
app.use('/login', {...});
app.use('/api', {...});
//Use this route to make the entire app secure. This forces login for any path in the entire app.
app.use('/', passport.authenticate('main', {
noredirect: false //Don't redirect a user to the authentication page, just show an error
}),
// other middleware
);
Authenticating All Routes
data:image/s3,"s3://crabby-images/dd325/dd325a470283c61ee5f42e4d6848c10d2c173bc3" alt=""
data:image/s3,"s3://crabby-images/dd325/dd325a470283c61ee5f42e4d6848c10d2c173bc3" alt=""
data:image/s3,"s3://crabby-images/042b6/042b64ed8305bad4831251871605e6c963f7f0d4" alt=""
---
applications:
- name: predix-ui-seed
memory: 64M
buildpack: nodejs_buildpack
command: node server/app.js
path: dist
#services:
# - <your-name>-secure-uaa-instance
# - <your-name>-timeseries-instance
# - <your-name>-asset-instance
env:
node_env: cloud
uaa_service_label: predix-uaa
# Add these values for authentication in the cloud
#clientId: {Enter client ID, e.g. app-client-id, and place it here}
#base64ClientCredential: {Get clientID:clientSecret then base64 encode and place it here}
#windServiceURL: "{URL of the microservice <your-name>-winddata-timeseries-service}, e.g. https://your-name-winddata-timeseries-service.run.asw-usw02-pr.predix.io"
Config manifest
data:image/s3,"s3://crabby-images/edab5/edab5ed47a61ae3e895c9c98f5e26260ce7a0f27" alt=""
Px UI components
(used in seed app)
- px-app-nav
- px-context-browser
- px-simple-line-chart
- px-simple-horizontal-bar-chart
- px-vis-timeseries
- px-data-table
data:image/s3,"s3://crabby-images/6a1fa/6a1fa4e9e9a7459eb5fa61270f1fa9220cb3b172" alt=""
data:image/s3,"s3://crabby-images/c467b/c467b0f66d4c11bb92fe0cfa9f42ee3c6679ed03" alt=""
data:image/s3,"s3://crabby-images/1a1fe/1a1fe5652d8f40cf16375afc66c172a78262d474" alt=""
data:image/s3,"s3://crabby-images/7eeff/7eeff92b5ca0ccf691cc1e8e9b1c0edfb42886ed" alt=""
data:image/s3,"s3://crabby-images/7103c/7103ca7b55a24adf2c40a187311840dc9bcdc0c2" alt=""
Time-series chart
- px-rangepicker
- px-vis-data-convertor
- chart features
data:image/s3,"s3://crabby-images/08eec/08eecccc4e6de266555144433280a27fbed8a539" alt=""
time-series chart
data:image/s3,"s3://crabby-images/65cad/65cadb78a54cfb615e20a79a403df01339df64dd" alt=""
two time-series format
UI dashboard
- view service hierarchy
- context-browser
- deck-selector
data:image/s3,"s3://crabby-images/66222/66222344575645d5b1af4e75ffea70bb49b4090c" alt=""
dashboard view hierarchy
dashboard elements
- View – Visual summary of the information displayed in a web application. It consists of one or more decks.
- Deck – Component that supplies simple one-dimensional layout control for cards and components arranged in a specified order.
- Card – Smallest functional unit of the page. You can create interactions between components within an individual card, as well as interactions with other cards based on user context.
- Component – User-interface element that allows you to add concrete features or access services.
dashboard & assets
data:image/s3,"s3://crabby-images/a8e0a/a8e0ac9a0799a19ef8f22c91c6e3acd3fb27f449" alt=""
Application Shell
- Web components
- HTML Import
- Polymer
data:image/s3,"s3://crabby-images/b0aa9/b0aa91710b9c270ba4b3f3f8b0335e5fdcbaba44" alt=""
data:image/s3,"s3://crabby-images/cf498/cf498aa328ba8ba076da83e195980756d10c9bd1" alt=""
data:image/s3,"s3://crabby-images/f6ed7/f6ed75e1745ac689c4f391acf8bcffec52cb3c14" alt=""
data:image/s3,"s3://crabby-images/77d3f/77d3fb1afb2963221fa7bf8c8ace303e83f85f39" alt=""
data:image/s3,"s3://crabby-images/f1584/f15845ba6dce28cc29bd47c81c076caf4962c74e" alt=""
HTML Import - linking
data:image/s3,"s3://crabby-images/1f406/1f406f7f52e4bdb65bb585820b024498676cb0d8" alt=""
App Shell - structure
data:image/s3,"s3://crabby-images/6b1ed/6b1ede8db88f9beca5f8f0698fe159d55311f7f1" alt=""
App Shell - bundling
data:image/s3,"s3://crabby-images/2f89a/2f89a5213e021f29fc6a88a8e274d44d1694cb34" alt=""
layers of Polymer application
data:image/s3,"s3://crabby-images/a7837/a78378eefd22c2e9855fabd2359aa4987b29dbe5" alt=""
Polymer Web Component
<dom-module id="element-name">
<template>
<style>
/* CSS rules for your element */
</style>
<!-- local DOM for your element -->
<div>{{greeting}}</div> <!-- data bindings in local DOM -->
</template>
<script>
// element registration
Polymer({
is: "element-name",
// add properties and methods on the element's prototype
properties: {
// declare properties for the element's public API
greeting: {
type: String,
value: "Hello!"
}
}
});
</script>
</dom-module>
Predix CSS Framework
- Sass Lang
- Predix Design
- InuitCSS - BEM/ITCSS
data:image/s3,"s3://crabby-images/ddeae/ddeaefd9f65089c7ba91740d75f236cfd11f4762" alt=""
SASS - Preprocessing CSS
data:image/s3,"s3://crabby-images/21780/21780ad9235b80345f207063b6d95db75197c22e" alt=""
Predix Design system
data:image/s3,"s3://crabby-images/ecc27/ecc276cb481a7a8b65dc36af7df27c44656fc7c6" alt=""
data:image/s3,"s3://crabby-images/a04f9/a04f9cf4fbeb4e12f73273e958713c9cb2ff1505" alt=""
ITCSS - Inverted Triangle Cascading Style Sheet
- Objects – class-based selectors which define undecorated design patterns, for example media object known from OOCSS
- Components – specific UI components. This is where majority of our work takes place and our UI components are often composed of Objects and Components
- Trumps – utilities and helper classes with ability to override anything which goes before in the triangle, eg. hide helper class
- Settings – used with preprocessors and contain font, colors definitions, etc.
- Tools – globally used mixins and functions. It’s important not to output any CSS in the first 2 layers.
- Generic – reset and/or normalize styles, box-sizing definition, etc. This is the first layer which generates actual CSS.
- Elements – styling for bare HTML elements (like H1, A, etc.). These come with default styling from the browser so we can redefine them here.
Law of ITCSS
data:image/s3,"s3://crabby-images/b4b34/b4b346dcad86c1e1f527700445f78be183990bf6" alt=""
BEM style convention
data:image/s3,"s3://crabby-images/ef52f/ef52fb5ee99d8c924f7dc230d3f4e89b895c42a0" alt=""
data:image/s3,"s3://crabby-images/a3ce9/a3ce99728f4e1586556d309e7a510cd17c0ce233" alt=""
example: starter-kit-design-demo
data:image/s3,"s3://crabby-images/f35dc/f35dccd56501a0f9cc497cd73087e70a24a1308c" alt=""
example: predix-button-design
data:image/s3,"s3://crabby-images/2af6a/2af6adfbb26fa6f6f273436ac0d409a566f89a20" alt=""
Scoped Styling
- Shadow DOM & Scoped Styling
- Shared styles module
- Custom CSS Variables & Mix-ins
- Document-wide Styling
<dom-module id="my-element">
<template>
<style>
:host {
display: block;
border: 1px solid red;
}
#child-element {
background: yellow;
}
/* styling elements distributed to content (via ::content) requires */
/* selecting the parent of the <content> element for compatibility with */
/* shady DOM . This can be :host or a wrapper element. */
.content-wrapper ::content > .special {
background: orange;
}
</style>
<div id="child-element">In local DOM!</div>
<div class="content-wrapper"><content></content></div>
</template>
<script>
Polymer({
is: 'my-element'
});
</script>
</dom-module>
component isolated local styles
<dom-module id="my-toolbar">
<template>
<style>
:host {
padding: 4px;
background-color: gray;
}
.title {
color: var(--my-toolbar-title-color);
}
</style>
<span class="title">{{title}}</span>
</template>
<script>
Polymer({
is: 'my-toolbar',
properties: {
title: String
}
});
</script>
</dom-module>
cross-scope styling - variables
<!-- usage -->
<style>
my-toolbar {
--my-toolbar-theme: {
/* rules */
};
--my-toolbar-title-theme: {
/* rules */
};
}
</style>
<dom-module id="my-toolbar">
<template>
<style>
:host {
padding: 4px;
background-color: gray;
/* apply a mixin */
@apply(--my-toolbar-theme);
}
.title {
@apply(--my-toolbar-title-theme);
}
</style>
<span class="title">{{title}}</span>
</template>
</dom-module>
cross-scope styling - mixins
<!-- shared-styles.html -->
<dom-module id="shared-styles">
<template>
<style>
.red { color: red; }
</style>
</template>
</dom-module>
<!-- import the module -->
<link rel="import" href="../shared-styles/shared-styles.html">
<dom-module id="x-foo">
<template>
<!-- include the style module by name -->
<style include="shared-styles"></style>
<style>:host { display: block; }</style>
Hi
</template>
<script>Polymer({is: 'x-foo'});</script>
</dom-module>
shared style module
Data binding (MVVM)
- Declarative data-binding
- Observers v.s. Computed Property
- Control flow in data-binding
<dom-module id="user-view">
<template>
<div>[[name]]</div>
</template>
<script>
Polymer({
is: 'user-view',
properties: {
name: String
}
});
</script>
</dom-module>
<!-- usage -->
<user-view name="Samuel"></user-view>
<dom-module id="main-view">
<template>
<user-view first="{{user.first}}"
last="{{user.last}}"></user-view>
</template>
<script>
Polymer({
is: 'main-view',
properties: {
user: Object
}
});
</script>
</dom-module>
text-binding v.s. property binding
Polymer({
is: 'x-custom',
properties: {
disabled: {
type: Boolean,
observer: '_disabledChanged'
},
highlight: {
observer: '_highlightChanged'
}
},
_disabledChanged: function(newValue, oldValue) {
this.toggleClass('disabled', newValue);
this.highlight = true;
},
_highlightChanged: function() {
this.classList.add('highlight');
this.async(function() {
this.classList.remove('highlight');
}, 300);
}
});
Simple Property Observer
Polymer({
is: 'x-custom',
properties: {
preload: Boolean,
src: String,
size: String
},
observers: [
'updateImage(preload, src, size)'
],
updateImage: function(preload, src, size) {
// ... do work using dependent values
}
});
Complex Property Observer
<dom-module id="x-deep-observer">
<template>
<input value="{{user.name.first::input}}"
placeholder="First Name">
<input value="{{user.name.last::input}}"
placeholder="Last Name">
</template>
<script>
Polymer({
is: 'x-deep-observer',
properties: {
user: {
type: Object,
value: function() {
return {'name':{}};
}
}
},
observers: [
'userNameChanged(user.name.*)'
],
userNameChanged: function(changeRecord) {
console.log('path: ' + changeRecord.path);
console.log('value: ' + changeRecord.value);
},
});
</script>
</dom-module>
Deep Property Observer
<dom-module id="name-card">
<template>
<div>[[name.first]] [[name.last]]</div>
</template>
<script>Polymer({ is: 'name-card' });</script>
</dom-module>
Data flow within element itself
data:image/s3,"s3://crabby-images/bf252/bf2528ac101010a3b1803f35ddb5c82c0ee04765" alt=""
<dom-module id="user-profile">
<template>
…
<address-card
address="{{primaryAddress}}"></address-card>
</template>
…
</dom-module>
Data flow between elements
data:image/s3,"s3://crabby-images/452da/452da86fb493a27a46703f08758916139bcd8587" alt=""
If <address-card> change it's "address", property "primaryAddress" is changed on <user-profile> as well.
routing & page-flow
- app-location & app-route
- px-view
- HTML5 history API
URL and View Routing
<!-- proxy to window.location -->
<app-location
route="{{_route}}">
</app-location>
<!-- match URL segment and convert to data -->
<app-route
route="[[_route]]"
pattern="/:page"
data="{{routeData}}">
</app-route>
<!-- select the view that matches route to display -->
<template is="dom-repeat" items="[[_routePages]]">
<px-view
active="[[_equals(routeData.page, item.page)]]"
element-href="[[item.elementHref]]">
</px-view>
</template>
data:image/s3,"s3://crabby-images/09942/0994216910679dc5ae8cfd98b72c852fc1c8613d" alt=""
data:image/s3,"s3://crabby-images/71cec/71cecd881b222fb68dbac14b780b8cfee2959c70" alt=""
Build the seed app
- build system
- gulp-sass
- gulp-vulcanizer
data:image/s3,"s3://crabby-images/171f4/171f4bb5222c4efcad4a574c40fca6f4a05efa59" alt=""
build system
data:image/s3,"s3://crabby-images/4944c/4944c87ba2b770064f7882ac76d883a1cd4afc1c" alt=""
data:image/s3,"s3://crabby-images/1206e/1206e3ff2d788ad0bc55d1c97a1546e8d023e247" alt=""
data:image/s3,"s3://crabby-images/a0884/a08849e09c8fb142f440d8e6433f74865a9061c1" alt=""
gulp-sass task
gulp-vulcanizer
data:image/s3,"s3://crabby-images/0c5c8/0c5c8aa09d8c3e02c1a52cb688d679c5c5627dcd" alt=""
gulp.task('vulcanize', function() {
return gulp.src('app/elements/elements.html')
.pipe(vulcanize({
stripComments: true,
inlineScripts: true,
inlineCss: true
}))
.pipe(gulp.dest('dist/elements'));
});
Deploy the seed app
- CF Deploy
- Jenkins CI integration
---
applications:
- name: predix-ui-app
memory: 64M
buildpack: nodejs_buildpack
command: node server/app.js
#services:
# - <your-name>-secure-uaa-instance
# - <your-name>-timeseries-instance
# - <your-name>-asset-instance
env:
node_env: cloud
uaa_service_label : predix-uaa
# Add these values for authentication in the cloud
#clientId: {Enter client ID, e.g. app-client-id, and place it here}
#base64ClientCredential: dWFhLWNsaWVudC1pZDp1YWEtY2xpZW50LWlkLXNlY3JldA==
#windServiceURL: "{URL of the microservice <your-name>-winddata-timeseries-service}, e.g. https://your-name-winddata-timeseries-service.run.asw-usw02-pr.predix.io"
Nodejs buildpack
Jenkins CI build
data:image/s3,"s3://crabby-images/516d9/516d9dc69d183a7a8b797caf13f251fb84fbd075" alt=""
data:image/s3,"s3://crabby-images/38413/38413c2d04cd7f536d552844762c63bf402addbd" alt=""
Jenkins CI build log
Q & A
data:image/s3,"s3://crabby-images/55902/55902079bc1584566880eb97f037721be226fd16" alt=""
Exploring Predix Seed App
By Garry Yao
Exploring Predix Seed App
UI seed app structure and features explained
- 2,042