feat: Export chart option in desktop view
This commit is contained in:
parent
57cb3ac023
commit
37198159aa
@ -1,3 +1,4 @@
|
|||||||
|
import html2canvas from 'html2canvas';
|
||||||
erpnext.HierarchyChart = class {
|
erpnext.HierarchyChart = class {
|
||||||
/* Options:
|
/* Options:
|
||||||
- doctype
|
- doctype
|
||||||
@ -11,16 +12,20 @@ erpnext.HierarchyChart = class {
|
|||||||
this.method = method;
|
this.method = method;
|
||||||
this.doctype = doctype;
|
this.doctype = doctype;
|
||||||
|
|
||||||
|
this.setup_page_style();
|
||||||
|
this.page.main.addClass('frappe-card');
|
||||||
|
|
||||||
|
this.nodes = {};
|
||||||
|
this.setup_node_class();
|
||||||
|
}
|
||||||
|
|
||||||
|
setup_page_style() {
|
||||||
this.page.main.css({
|
this.page.main.css({
|
||||||
'min-height': '300px',
|
'min-height': '300px',
|
||||||
'max-height': '600px',
|
'max-height': '600px',
|
||||||
'overflow': 'auto',
|
'overflow': 'auto',
|
||||||
'position': 'relative'
|
'position': 'relative'
|
||||||
});
|
});
|
||||||
this.page.main.addClass('frappe-card');
|
|
||||||
|
|
||||||
this.nodes = {};
|
|
||||||
this.setup_node_class();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
setup_node_class() {
|
setup_node_class() {
|
||||||
@ -84,7 +89,7 @@ erpnext.HierarchyChart = class {
|
|||||||
|
|
||||||
// svg for connectors
|
// svg for connectors
|
||||||
me.make_svg_markers();
|
me.make_svg_markers();
|
||||||
me.setup_hierarchy()
|
me.setup_hierarchy();
|
||||||
me.render_root_nodes();
|
me.render_root_nodes();
|
||||||
me.all_nodes_expanded = false;
|
me.all_nodes_expanded = false;
|
||||||
}
|
}
|
||||||
@ -97,6 +102,10 @@ erpnext.HierarchyChart = class {
|
|||||||
|
|
||||||
setup_actions() {
|
setup_actions() {
|
||||||
let me = this;
|
let me = this;
|
||||||
|
this.page.add_inner_button(__('Export'), function() {
|
||||||
|
me.export_chart();
|
||||||
|
});
|
||||||
|
|
||||||
this.page.add_inner_button(__('Expand All'), function() {
|
this.page.add_inner_button(__('Expand All'), function() {
|
||||||
me.load_children(me.root_node, true);
|
me.load_children(me.root_node, true);
|
||||||
me.all_nodes_expanded = true;
|
me.all_nodes_expanded = true;
|
||||||
@ -113,6 +122,36 @@ erpnext.HierarchyChart = class {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export_chart() {
|
||||||
|
this.page.main.css({
|
||||||
|
'min-height': '',
|
||||||
|
'max-height': '',
|
||||||
|
'overflow': 'visible',
|
||||||
|
'position': 'fixed',
|
||||||
|
'left': '0',
|
||||||
|
'top': '0'
|
||||||
|
});
|
||||||
|
|
||||||
|
$('.node-card').addClass('exported');
|
||||||
|
|
||||||
|
html2canvas(document.querySelector('#hierarchy-chart-wrapper'), {
|
||||||
|
scrollY: -window.scrollY,
|
||||||
|
scrollX: 0
|
||||||
|
}).then(function(canvas) {
|
||||||
|
// Export the canvas to its data URI representation
|
||||||
|
let dataURL = canvas.toDataURL('image/png');
|
||||||
|
|
||||||
|
// download the image
|
||||||
|
let a = document.createElement('a');
|
||||||
|
a.href = dataURL;
|
||||||
|
a.download = 'hierarchy_chart';
|
||||||
|
a.click();
|
||||||
|
});
|
||||||
|
|
||||||
|
this.setup_page_style();
|
||||||
|
$('.node-card').removeClass('exported');
|
||||||
|
}
|
||||||
|
|
||||||
setup_hierarchy() {
|
setup_hierarchy() {
|
||||||
if (this.$hierarchy)
|
if (this.$hierarchy)
|
||||||
this.$hierarchy.remove();
|
this.$hierarchy.remove();
|
||||||
@ -127,33 +166,37 @@ erpnext.HierarchyChart = class {
|
|||||||
</li>
|
</li>
|
||||||
</ul>`);
|
</ul>`);
|
||||||
|
|
||||||
this.page.main.append(this.$hierarchy);
|
this.page.main
|
||||||
|
.find('#hierarchy-chart-wrapper')
|
||||||
|
.append(this.$hierarchy);
|
||||||
this.nodes = {};
|
this.nodes = {};
|
||||||
}
|
}
|
||||||
|
|
||||||
make_svg_markers() {
|
make_svg_markers() {
|
||||||
$('#arrows').remove();
|
$('#arrows').remove();
|
||||||
|
|
||||||
this.page.main.prepend(`
|
this.page.main.append(`
|
||||||
<svg id="arrows" width="100%" height="100%">
|
<div id="hierarchy-chart-wrapper">
|
||||||
<defs>
|
<svg id="arrows" width="100%" height="100%">
|
||||||
<marker id="arrowhead-active" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="6" markerHeight="6" orient="auto" fill="var(--blue-500)">
|
<defs>
|
||||||
<path d="M 0 0 L 10 5 L 0 10 z"></path>
|
<marker id="arrowhead-active" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="6" markerHeight="6" orient="auto" fill="var(--blue-500)">
|
||||||
</marker>
|
<path d="M 0 0 L 10 5 L 0 10 z"></path>
|
||||||
<marker id="arrowhead-collapsed" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="6" markerHeight="6" orient="auto" fill="var(--blue-300)">
|
</marker>
|
||||||
<path d="M 0 0 L 10 5 L 0 10 z"></path>
|
<marker id="arrowhead-collapsed" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="6" markerHeight="6" orient="auto" fill="var(--blue-300)">
|
||||||
</marker>
|
<path d="M 0 0 L 10 5 L 0 10 z"></path>
|
||||||
|
</marker>
|
||||||
|
|
||||||
<marker id="arrowstart-active" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="8" markerHeight="8" orient="auto" fill="var(--blue-500)">
|
<marker id="arrowstart-active" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="8" markerHeight="8" orient="auto" fill="var(--blue-500)">
|
||||||
<circle cx="4" cy="4" r="3.5" fill="white" stroke="var(--blue-500)"/>
|
<circle cx="4" cy="4" r="3.5" fill="white" stroke="var(--blue-500)"/>
|
||||||
</marker>
|
</marker>
|
||||||
<marker id="arrowstart-collapsed" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="8" markerHeight="8" orient="auto" fill="var(--blue-300)">
|
<marker id="arrowstart-collapsed" viewBox="0 0 10 10" refX="3" refY="5" markerWidth="8" markerHeight="8" orient="auto" fill="var(--blue-300)">
|
||||||
<circle cx="4" cy="4" r="3.5" fill="white" stroke="var(--blue-300)"/>
|
<circle cx="4" cy="4" r="3.5" fill="white" stroke="var(--blue-300)"/>
|
||||||
</marker>
|
</marker>
|
||||||
</defs>
|
</defs>
|
||||||
<g id="connectors" fill="none">
|
<g id="connectors" fill="none">
|
||||||
</g>
|
</g>
|
||||||
</svg>`);
|
</svg>
|
||||||
|
</div>`);
|
||||||
}
|
}
|
||||||
|
|
||||||
render_root_nodes(expanded_view=false) {
|
render_root_nodes(expanded_view=false) {
|
||||||
@ -310,7 +353,7 @@ erpnext.HierarchyChart = class {
|
|||||||
let entry = undefined;
|
let entry = undefined;
|
||||||
let node = undefined;
|
let node = undefined;
|
||||||
|
|
||||||
while(data_list.length) {
|
while (data_list.length) {
|
||||||
// to avoid overlapping connectors
|
// to avoid overlapping connectors
|
||||||
entry = data_list.shift();
|
entry = data_list.shift();
|
||||||
node = this.nodes[entry.parent];
|
node = this.nodes[entry.parent];
|
||||||
@ -323,7 +366,7 @@ erpnext.HierarchyChart = class {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render_child_nodes_for_expanded_view(node, child_nodes) {
|
render_child_nodes_for_expanded_view(node, child_nodes) {
|
||||||
node.$children = $('<ul class="node-children"></ul>')
|
node.$children = $('<ul class="node-children"></ul>');
|
||||||
|
|
||||||
const last_level = this.$hierarchy.find('.level:last').index();
|
const last_level = this.$hierarchy.find('.level:last').index();
|
||||||
const node_level = $(`#${node.id}`).parent().parent().parent().index();
|
const node_level = $(`#${node.id}`).parent().parent().parent().index();
|
||||||
|
@ -21,6 +21,10 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.node-card.exported {
|
||||||
|
box-shadow: none
|
||||||
|
}
|
||||||
|
|
||||||
.node-image {
|
.node-image {
|
||||||
width: 3.0rem;
|
width: 3.0rem;
|
||||||
height: 3.0rem;
|
height: 3.0rem;
|
||||||
@ -178,9 +182,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// horizontal hierarchy tree view
|
// horizontal hierarchy tree view
|
||||||
|
#hierarchy-chart-wrapper {
|
||||||
|
padding-top: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
.hierarchy {
|
.hierarchy {
|
||||||
display: flex;
|
display: flex;
|
||||||
padding-top: 30px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.hierarchy li {
|
.hierarchy li {
|
||||||
@ -200,6 +207,7 @@
|
|||||||
#arrows {
|
#arrows {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
overflow: visible;
|
overflow: visible;
|
||||||
|
margin-top: -80px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.active-connector {
|
.active-connector {
|
||||||
|
@ -3,14 +3,16 @@
|
|||||||
|
|
||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
import frappe
|
import frappe
|
||||||
|
import os
|
||||||
from frappe import _
|
from frappe import _
|
||||||
|
from frappe.utils.pdf import get_pdf
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def get_all_nodes(parent, parent_name, method, company):
|
def get_all_nodes(parent, parent_name, method, company):
|
||||||
'''Recursively gets all data from nodes'''
|
'''Recursively gets all data from nodes'''
|
||||||
method = frappe.get_attr(method)
|
method = frappe.get_attr(method)
|
||||||
|
|
||||||
if not method in frappe.whitelisted:
|
if method not in frappe.whitelisted:
|
||||||
frappe.throw(_('Not Permitted'), frappe.PermissionError)
|
frappe.throw(_('Not Permitted'), frappe.PermissionError)
|
||||||
|
|
||||||
data = method(parent, company)
|
data = method(parent, company)
|
||||||
|
Loading…
Reference in New Issue
Block a user