feat: Export chart option in desktop view

This commit is contained in:
Rucha Mahabal 2021-07-25 20:28:01 +05:30
parent 57cb3ac023
commit 37198159aa
3 changed files with 82 additions and 29 deletions

View File

@ -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();

View File

@ -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 {

View File

@ -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)