-
Notifications
You must be signed in to change notification settings - Fork 18
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
4 changed files
with
376 additions
and
0 deletions.
There are no files selected for viewing
Binary file not shown.
73 changes: 73 additions & 0 deletions
73
Custom-Widget-Samples/Sunbrust Chart with Styling Panel/index.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
{ | ||
"eula": "", | ||
"vendor": "SAP", | ||
"license": "", | ||
"id": "com.sap.sac.sample.init.example.echarts.sunburst", | ||
"icon": "", | ||
"version": "1.0.0", | ||
"supportsMobile": true, | ||
"supportsBookmark": true, | ||
"supportsLinkedAnalysisFilterOnSelection": true, | ||
"name": "EChartsSunburstStyling", | ||
"newInstancePrefix": "EChartsSunburstStyling", | ||
"description": "A Sample Sunburst Chart with Styling Panel", | ||
"webcomponents": [ | ||
{ | ||
"kind": "main", | ||
"tag": "com-sap-sample-init-example-echarts-sunburst", | ||
"url": "/main.js", | ||
"integrity": "", | ||
"ignoreIntegrity": true | ||
}, | ||
{ | ||
"kind": "styling", | ||
"tag": "com-sap-sample-init-example-echarts-sunburst-styling", | ||
"url": "/styling.js", | ||
"integrity": "", | ||
"ignoreIntegrity": true | ||
} | ||
], | ||
"properties": { | ||
"width": { | ||
"type": "integer", | ||
"default": 600 | ||
}, | ||
"height": { | ||
"type": "integer", | ||
"default": 320 | ||
}, | ||
"drillUpArea": { | ||
"type": "string", | ||
"default": "#ffffff" | ||
}, | ||
"stops": { | ||
"type": "string[]", | ||
"default": ["#0070F2", "#4CB1FF", "#A6E0FF", "#D2EFFF", "#0057D2"] | ||
}, | ||
"showAll": { | ||
"type": "boolean", | ||
"default": false | ||
}, | ||
"labelAll": { | ||
"type": "string" | ||
} | ||
}, | ||
"methods": {}, | ||
"events": {}, | ||
"dataBindings": { | ||
"dataBinding": { | ||
"feeds": [ | ||
{ | ||
"id": "dimensions", | ||
"description": "Dimensions", | ||
"type": "dimension" | ||
}, | ||
{ | ||
"id": "measures", | ||
"description": "Measures", | ||
"type": "mainStructureMember" | ||
} | ||
] | ||
} | ||
} | ||
} |
189 changes: 189 additions & 0 deletions
189
Custom-Widget-Samples/Sunbrust Chart with Styling Panel/main.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,189 @@ | ||
/*var getScriptPromisify = (src) => { | ||
return new Promise((resolve) => { | ||
$.getScript(src, resolve); | ||
}); | ||
};*/ | ||
|
||
var getScriptPromisify = (src) => { | ||
// Workaround with conflict between geo widget and echarts. | ||
const __define = define | ||
define = undefined | ||
return new Promise(resolve => { | ||
$.getScript(src, () => { | ||
define = __define | ||
resolve() | ||
}) | ||
}) | ||
} | ||
|
||
(function () { | ||
const parseMetadata = metadata => { | ||
const { dimensions: dimensionsMap, mainStructureMembers: measuresMap } = metadata | ||
const dimensions = [] | ||
for (const key in dimensionsMap) { | ||
const dimension = dimensionsMap[key] | ||
dimensions.push({ key, ...dimension }) | ||
} | ||
const measures = [] | ||
for (const key in measuresMap) { | ||
const measure = measuresMap[key] | ||
measures.push({ key, ...measure }) | ||
} | ||
return { dimensions, measures, dimensionsMap, measuresMap } | ||
} | ||
|
||
const appendSuperRoot = (data, labelAll) => { | ||
data = JSON.parse(JSON.stringify(data)) | ||
const superRoot = { | ||
dimensions_0: { id: 'superRoot', label: labelAll }, | ||
measures_0: { raw: 0 } | ||
} | ||
data.forEach(data => { | ||
if (data.dimensions_0.parentId) { return } | ||
data.dimensions_0.parentId = 'superRoot' | ||
superRoot.measures_0.raw += data.measures_0.raw | ||
}) | ||
return [superRoot].concat(data) | ||
} | ||
|
||
const findDp = (seriesData, id, depth = 1) => { | ||
for (let i = 0; i < seriesData.length; i++) { | ||
const dp = seriesData[i] | ||
if (dp.id === id) { return { dp, depth } } | ||
if (dp.children) { | ||
const found = findDp(dp.children, id, depth + 1) | ||
if (found) { return found } | ||
} | ||
} | ||
} | ||
|
||
class Renderer { | ||
constructor(root) { | ||
this._root = root | ||
this._echart = null | ||
} | ||
|
||
async render(dataBinding, drillUpArea, stops, showAll, labelAll) { | ||
await getScriptPromisify("https://cdn.staticfile.org/echarts/5.3.0/echarts.min.js"); | ||
this.dispose() | ||
|
||
if (dataBinding.state !== 'success') { return } | ||
|
||
let { data, metadata } = dataBinding | ||
const { dimensions, measures } = parseMetadata(metadata) | ||
|
||
if (dimensions.length !== 1) { return } | ||
|
||
const levels = [{ | ||
itemStyle: { color: drillUpArea } | ||
}] | ||
stops.forEach(stop => { | ||
levels.push({ | ||
itemStyle: { color: stop } | ||
}) | ||
}) | ||
const seriesData = [] | ||
if (showAll) { | ||
data = appendSuperRoot(data, labelAll) | ||
} | ||
data.forEach(row => { | ||
const { id, label, parentId } = row[dimensions[0].key] | ||
const { raw } = row[measures[0].key] | ||
const dp = findDp(seriesData, id) || { id } | ||
dp.name = label | ||
dp.value = raw | ||
|
||
if (parentId) { | ||
const parent = findDp(seriesData, parentId) | ||
if (!parent) { console.error('Please check: "Include Parent Node"') } | ||
|
||
parent.dp.children = parent.dp.children || [] | ||
parent.dp.children.push(dp) | ||
} else { | ||
seriesData.push(dp) | ||
} | ||
}) | ||
|
||
this._echart = echarts.init(this._root) | ||
// https://echarts.apache.org/en/option.html | ||
this._echart.setOption({ | ||
series: { | ||
type: 'sunburst', | ||
data: seriesData, | ||
levels, | ||
radius: [0, '90%'], | ||
label: { | ||
rotate: 'radial' | ||
} | ||
} | ||
}) | ||
} | ||
|
||
dispose() { | ||
if (this._echart) { | ||
echarts.dispose(this._echart) | ||
} | ||
} | ||
} | ||
|
||
const template = document.createElement('template') | ||
template.innerHTML = ` | ||
<style> | ||
#chart { | ||
width: 100%; | ||
height: 100%; | ||
} | ||
</style> | ||
<div id="root" style="width: 100%; height: 100%;"> | ||
<div id="chart"></div> | ||
</div> | ||
` | ||
|
||
class Main extends HTMLElement { | ||
constructor () { | ||
super() | ||
|
||
this._shadowRoot = this.attachShadow({ mode: 'open' }) | ||
this._shadowRoot.appendChild(template.content.cloneNode(true)) | ||
this._root = this._shadowRoot.getElementById('root') | ||
this._renderer = new Renderer(this._root) | ||
} | ||
|
||
// ------------------ | ||
// LifecycleCallbacks | ||
// ------------------ | ||
async onCustomWidgetBeforeUpdate(changedProps) { | ||
} | ||
|
||
async onCustomWidgetAfterUpdate(changedProps) { | ||
this.render() | ||
} | ||
|
||
async onCustomWidgetResize(width, height) { | ||
this.render() | ||
} | ||
|
||
async onCustomWidgetDestroy() { | ||
this.dispose() | ||
} | ||
|
||
// ------------------ | ||
// | ||
// ------------------ | ||
dispose() { | ||
this._renderer.dispose() | ||
} | ||
|
||
render() { | ||
if (!document.contains(this)) { | ||
// Delay the render to assure the custom widget is appended on dom | ||
setTimeout(this.render.bind(this), 0) | ||
return | ||
} | ||
|
||
this._renderer.render(this.dataBinding, this.drillUpArea, this.stops, this.showAll, this.labelAll) | ||
} | ||
} | ||
|
||
customElements.define('com-sap-sample-init-example-echarts-sunburst', Main) | ||
})() |
114 changes: 114 additions & 0 deletions
114
Custom-Widget-Samples/Sunbrust Chart with Styling Panel/styling.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,114 @@ | ||
(function () { | ||
let template = document.createElement("template"); | ||
template.innerHTML = ` | ||
<style> | ||
#root div { | ||
margin: 0.5rem; | ||
} | ||
#root .title { | ||
font-weight: bold; | ||
} | ||
</style> | ||
<div id="root" style="width: 100%; height: 100%;"> | ||
<div class="title">All</div> | ||
<div> | ||
<input id="showAll" type="checkbox" checked /><label for="reverse">Show</label> | ||
</div> | ||
<div class="title">Label</div> | ||
<div> | ||
<input id="labelAll" type="text" value="All" /> | ||
</div> | ||
<div class="title">Drill Up Area</div> | ||
<div> | ||
<input id="drillUpArea" type="color" value="#ffffff" /> | ||
</div> | ||
<div class="title">Colors</div> | ||
<!-- http://zhongguose.com/#chahuahong --> | ||
<div> | ||
<input id="stop0" type="color" value="#c04851" /> | ||
</div> | ||
<div> | ||
<input id="stop1" type="color" value="#ed5a65" /> | ||
</div> | ||
<div> | ||
<input id="stop2" type="color" value="#f07c82" /> | ||
</div> | ||
<div> | ||
<input id="stop3" type="color" value="#eea2a4" /> | ||
</div> | ||
<div> | ||
<input id="stop4" type="color" value="#ee3f4d" /> | ||
</div> | ||
<div> | ||
<button id="button">Apply</button> | ||
</div> | ||
</div> | ||
`; | ||
|
||
class Styling extends HTMLElement { | ||
constructor () { | ||
super() | ||
|
||
this._shadowRoot = this.attachShadow({ mode: 'open' }) | ||
this._shadowRoot.appendChild(template.content.cloneNode(true)) | ||
this._root = this._shadowRoot.getElementById('root') | ||
|
||
this._button = this._shadowRoot.getElementById('button') | ||
this._button.addEventListener('click', () => { | ||
const drillUpArea = this._shadowRoot.getElementById('drillUpArea').value | ||
const stops = [ | ||
this._shadowRoot.getElementById('stop0').value, | ||
this._shadowRoot.getElementById('stop1').value, | ||
this._shadowRoot.getElementById('stop2').value, | ||
this._shadowRoot.getElementById('stop3').value, | ||
this._shadowRoot.getElementById('stop4').value | ||
] | ||
const showAll = this._shadowRoot.getElementById('showAll').checked | ||
const labelAll = this._shadowRoot.getElementById('labelAll').value | ||
this.dispatchEvent(new CustomEvent('propertiesChanged', { detail: { properties: { drillUpArea, stops, showAll, labelAll } } })) | ||
}) | ||
} | ||
|
||
// ------------------ | ||
// LifecycleCallbacks | ||
// ------------------ | ||
async onCustomWidgetBeforeUpdate (changedProps) { | ||
} | ||
|
||
async onCustomWidgetAfterUpdate (changedProps) { | ||
if (changedProps.drillUpArea) { | ||
this._shadowRoot.getElementById('drillUpArea').value = changedProps.drillUpArea | ||
} | ||
if (changedProps.stops) { | ||
this._shadowRoot.getElementById('stop0').value = changedProps.stops[0] | ||
this._shadowRoot.getElementById('stop1').value = changedProps.stops[1] | ||
this._shadowRoot.getElementById('stop2').value = changedProps.stops[2] | ||
this._shadowRoot.getElementById('stop3').value = changedProps.stops[3] | ||
this._shadowRoot.getElementById('stop4').value = changedProps.stops[4] | ||
} | ||
if (changedProps.showAll !== undefined) { | ||
this._shadowRoot.getElementById('showAll').checked = changedProps.showAll | ||
} | ||
if (changedProps.labelAll !== undefined) { | ||
this._shadowRoot.getElementById('labelAll').value = changedProps.labelAll | ||
} | ||
} | ||
|
||
async onCustomWidgetResize (width, height) { | ||
} | ||
|
||
async onCustomWidgetDestroy () { | ||
this.dispose() | ||
} | ||
|
||
// ------------------ | ||
// | ||
// ------------------ | ||
|
||
dispose () { | ||
} | ||
} | ||
|
||
customElements.define('com-sap-sample-init-example-echarts-sunburst-styling', Styling); | ||
})(); |