Skip to content

Commit

Permalink
Update
Browse files Browse the repository at this point in the history
  • Loading branch information
Marou-Fer committed Feb 28, 2024
1 parent 76f310e commit 0e70802
Show file tree
Hide file tree
Showing 4 changed files with 376 additions and 0 deletions.
Binary file not shown.
73 changes: 73 additions & 0 deletions Custom-Widget-Samples/Sunbrust Chart with Styling Panel/index.json
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 Custom-Widget-Samples/Sunbrust Chart with Styling Panel/main.js
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 Custom-Widget-Samples/Sunbrust Chart with Styling Panel/styling.js
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);
})();

0 comments on commit 0e70802

Please sign in to comment.