# -*- coding: utf-8 -*- """ /*************************************************************************** DataPlotlyDialog A QGIS plugin D3 Plots for QGIS ------------------- begin : 2017-03-05 git sha : $Format:%H$ copyright : (C) 2017 by matteo ghetta email : matteo.ghetta@gmail.com ***************************************************************************/ /*************************************************************************** * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; either version 2 of the License, or * * (at your option) any later version. * * * ***************************************************************************/ """ import json from collections import OrderedDict from shutil import copyfile from functools import partial import sys from qgis.PyQt import uic from qgis.PyQt.QtWidgets import ( QListWidgetItem, QVBoxLayout, QFileDialog, QMenu ) from qgis.PyQt.QtXml import QDomDocument from qgis.PyQt.QtGui import ( QFont, QImage, QPainter, QColor ) from qgis.PyQt.QtCore import ( QUrl, pyqtSignal, QDir ) from qgis.PyQt.QtWebKit import QWebSettings from qgis.PyQt.QtWebKitWidgets import ( QWebView ) from qgis.core import ( Qgis, QgsNetworkAccessManager, QgsFeatureRequest, QgsMapLayerProxyModel, QgsProject, QgsFileUtils, QgsReferencedRectangle, QgsExpressionContextGenerator, QgsPropertyCollection, QgsLayoutItemRegistry, QgsPropertyDefinition ) from qgis.gui import ( QgsPanelWidget, QgsMessageBar, QgsPropertyOverrideButton ) from qgis.utils import iface from DataPlotly.core.plot_factory import PlotFactory from DataPlotly.core.plot_settings import PlotSettings from DataPlotly.gui.gui_utils import GuiUtils WIDGET, _ = uic.loadUiType(GuiUtils.get_ui_file_path('dataplotly_dockwidget_base.ui')) class DataPlotlyPanelWidget(QgsPanelWidget, WIDGET): # pylint: disable=too-many-lines,too-many-instance-attributes,too-many-public-methods """ Main configuration panel widget for plot settings """ MODE_CANVAS = 'CANVAS' MODE_LAYOUT = 'LAYOUT' # emit signal when dialog is resized resizeWindow = pyqtSignal() def __init__(self, mode=MODE_CANVAS, parent=None, override_iface=None, message_bar: QgsMessageBar = None): # pylint: disable=too-many-statements """Constructor.""" super().__init__(parent) self.setupUi(self) if override_iface is None: self.iface = iface else: self.iface = override_iface self.mode = mode self.message_bar = message_bar self.setPanelTitle(self.tr('Plot Properties')) self.save_to_project = True self.read_from_project = True self.data_defined_properties = QgsPropertyCollection() # listen out for project save/restore, and update our state accordingly QgsProject.instance().writeProject.connect(self.write_project) QgsProject.instance().readProject.connect(self.read_project) if self.iface is not None: self.listWidget.setIconSize(self.iface.iconSize(False)) self.listWidget.setMaximumWidth(int(self.listWidget.iconSize().width() * 1.18)) # connect signal to function to reload the plot view self.resizeWindow.connect(self.reloadPlotCanvas) # create the reload button with text and icon self.reload_btn.setText("Reload") self.reload_btn.setIcon(GuiUtils.get_icon('reload.svg')) self.clear_btn.setIcon(GuiUtils.get_icon('clean.svg')) self.update_btn.setIcon(GuiUtils.get_icon('refresh.svg')) self.draw_btn.setIcon(GuiUtils.get_icon('create_plot.svg')) # connect the button to the reload function self.reload_btn.clicked.connect(self.reloadPlotCanvas2) self.configuration_menu = QMenu(self) action_load_configuration = self.configuration_menu.addAction(self.tr("Load Configuration…")) action_load_configuration.triggered.connect(self.load_configuration) action_save_configuration = self.configuration_menu.addAction(self.tr("Save Configuration…")) action_save_configuration.triggered.connect(self.save_configuration) self.configuration_btn.setMenu(self.configuration_menu) # ListWidget icons and themes self.listWidget_icons = [ QListWidgetItem(GuiUtils.get_icon('list_properties.svg'), ""), QListWidgetItem(GuiUtils.get_icon('list_custom.svg'), ""), ] if self.mode == DataPlotlyPanelWidget.MODE_CANVAS: self.listWidget_icons.extend([ QListWidgetItem(GuiUtils.get_icon('list_plot.svg'), ""), QListWidgetItem(GuiUtils.get_icon('list_help.svg'), ""), QListWidgetItem(GuiUtils.get_icon('list_code.svg'), "") ]) # fill the QListWidget with items and icons for i in self.listWidget_icons: self.listWidget.addItem(i) # highlight the first row when starting the first time self.listWidget.setCurrentRow(0) self.marker_size.setValue(10) self.marker_size.setSingleStep(0.2) self.marker_size.setClearValue(10) self.marker_width.setValue(1) self.marker_width.setSingleStep(0.2) self.marker_width.setClearValue(0, self.tr('None')) # Populate PlotTypes combobox # we sort available types by translated name type_classes = [clazz for _, clazz in PlotFactory.PLOT_TYPES.items()] type_classes.sort(key=lambda x: x.name().lower()) for clazz in type_classes: self.plot_combo.addItem(clazz.icon(), clazz.name(), clazz.type_name()) # default to scatter plots self.set_plot_type('scatter') # SubPlots combobox self.subcombo.addItem(self.tr('Single Plot'), 'single') self.subcombo.addItem(self.tr('Subplots'), 'subplots') # connect to the functions to clean the UI and fill with the correct # widgets self.refreshWidgets() self.refreshWidgets2() self.plot_combo.currentIndexChanged.connect(self.refreshWidgets) self.plot_combo.currentIndexChanged.connect(self.helpPage) self.subcombo.currentIndexChanged.connect(self.refreshWidgets2) self.marker_type_combo.currentIndexChanged.connect(self.refreshWidgets3) self.properties_group_box.collapsedStateChanged.connect(self.refreshWidgets) # fill the layer combobox with vector layers self.layer_combo.setFilters(QgsMapLayerProxyModel.VectorLayer) # connect the combo boxes to the setLegend function self.x_combo.fieldChanged.connect(self.setLegend) self.y_combo.fieldChanged.connect(self.setLegend) self.z_combo.fieldChanged.connect(self.setLegend) self.draw_btn.clicked.connect(self.create_plot) self.update_btn.clicked.connect(self.UpdatePlot) self.clear_btn.clicked.connect(self.clearPlotView) self.save_plot_btn.clicked.connect(self.save_plot_as_image) self.save_plot_html_btn.clicked.connect(self.save_plot_as_html) self.save_plot_btn.setIcon(GuiUtils.get_icon('save_as_image.svg')) self.save_plot_html_btn.setIcon(GuiUtils.get_icon('save_as_html.svg')) # initialize the empty dictionary of plots self.plot_factories = {} # start the index counter self.idx = 1 # load the help html page into the help widget self.layouth = QVBoxLayout() self.layouth.setContentsMargins(0, 0, 0, 0) self.help_widget.setLayout(self.layouth) self.help_view = QWebView() self.layouth.addWidget(self.help_view) self.helpPage() # load the webview of the plot a the first running of the plugin self.layoutw = QVBoxLayout() self.layoutw.setContentsMargins(0, 0, 0, 0) self.plot_qview.setLayout(self.layoutw) self.plot_view = QWebView() self.plot_view.page().setNetworkAccessManager(QgsNetworkAccessManager.instance()) self.plot_view.statusBarMessage.connect(self.getJSmessage) plot_view_settings = self.plot_view.settings() plot_view_settings.setAttribute(QWebSettings.WebGLEnabled, True) plot_view_settings.setAttribute(QWebSettings.DeveloperExtrasEnabled, True) plot_view_settings.setAttribute(QWebSettings.Accelerated2dCanvasEnabled, True) self.layoutw.addWidget(self.plot_view) # get the plot type from the combobox self.ptype = self.plot_combo.currentData() self.layer_combo.layerChanged.connect(self.selected_layer_changed) # fill combo boxes when launching the UI self.selected_layer_changed(self.layer_combo.currentLayer()) self.register_data_defined_button(self.feature_subset_defined_button, PlotSettings.PROPERTY_FILTER) self.register_data_defined_button(self.size_defined_button, PlotSettings.PROPERTY_MARKER_SIZE) self.size_defined_button.registerEnabledWidget(self.marker_size, natural=False) self.register_data_defined_button(self.stroke_defined_button, PlotSettings.PROPERTY_STROKE_WIDTH) self.stroke_defined_button.registerEnabledWidget(self.marker_width, natural=False) self.register_data_defined_button(self.in_color_defined_button, PlotSettings.PROPERTY_COLOR) self.in_color_defined_button.changed.connect(self.data_defined_color_updated) self.register_data_defined_button(self.out_color_defined_button, PlotSettings.PROPERTY_STROKE_COLOR) self.out_color_defined_button.registerEnabledWidget(self.out_color_combo, natural=False) self.register_data_defined_button(self.plot_title_defined_button, PlotSettings.PROPERTY_TITLE) self.plot_title_defined_button.registerEnabledWidget(self.plot_title_line, natural=False) self.register_data_defined_button(self.legend_title_defined_button, PlotSettings.PROPERTY_LEGEND_TITLE) self.legend_title_defined_button.registerEnabledWidget(self.legend_title, natural=False) self.register_data_defined_button(self.x_axis_title_defined_button, PlotSettings.PROPERTY_X_TITLE) self.x_axis_title_defined_button.registerEnabledWidget(self.x_axis_title, natural=False) self.register_data_defined_button(self.y_axis_title_defined_button, PlotSettings.PROPERTY_Y_TITLE) self.y_axis_title_defined_button.registerEnabledWidget(self.y_axis_title, natural=False) self.register_data_defined_button(self.z_axis_title_defined_button, PlotSettings.PROPERTY_Z_TITLE) self.z_axis_title_defined_button.registerEnabledWidget(self.z_axis_title, natural=False) self.register_data_defined_button(self.x_axis_min_defined_button, PlotSettings.PROPERTY_X_MIN) self.x_axis_min_defined_button.registerEnabledWidget(self.x_axis_min, natural=False) self.register_data_defined_button(self.x_axis_max_defined_button, PlotSettings.PROPERTY_X_MAX) self.x_axis_max_defined_button.registerEnabledWidget(self.x_axis_max, natural=False) self.register_data_defined_button(self.y_axis_min_defined_button, PlotSettings.PROPERTY_Y_MIN) self.y_axis_min_defined_button.registerEnabledWidget(self.y_axis_min, natural=False) self.register_data_defined_button(self.y_axis_max_defined_button, PlotSettings.PROPERTY_Y_MAX) self.y_axis_max_defined_button.registerEnabledWidget(self.y_axis_max, natural=False) # connect to refreshing function of listWidget and stackedWidgets self.listWidget.currentRowChanged.connect(self.updateStacked) # connect the plot changing to the color data defined buttons self.plot_combo.currentIndexChanged.connect(self.data_defined_color_updated) # better default colors self.in_color_combo.setColor(QColor('#8EBAD9')) self.out_color_combo.setColor(QColor('#1F77B4')) # set range of axis min/max spin boxes self.x_axis_min.setRange(sys.float_info.max * -1, sys.float_info.max) self.x_axis_max.setRange(sys.float_info.max * -1, sys.float_info.max) self.y_axis_min.setRange(sys.float_info.max * -1, sys.float_info.max) self.y_axis_max.setRange(sys.float_info.max * -1, sys.float_info.max) self.pid = None self.plot_path = None self.plot_url = None self.plot_file = None if self.mode == DataPlotlyPanelWidget.MODE_LAYOUT: self.update_btn.setEnabled(True) # hide for now self.draw_btn.setVisible(False) self.clear_btn.setVisible(False) self.subcombo.setVisible(False) self.subcombo_label.setVisible(False) self.visible_feature_check.setVisible(False) self.selected_feature_check.setVisible(False) else: self.iface.mapCanvas().extentsChanged.connect(self.update_plot_visible_rect) self.label_linked_map.setVisible(False) self.linked_map_combo.setVisible(False) self.filter_by_map_check.setVisible(False) self.filter_by_atlas_check.setVisible(False) QgsProject.instance().layerWillBeRemoved.connect(self.layer_will_be_removed) def updateStacked(self, row): """ according to the listWdiget row change the stackedWidget and nestedStackedWidget """ # stackedWidget index = 1 and change just the nestedStackedWidgets if 0 <= row <= 1: self.stackedPlotWidget.setCurrentIndex(0) self.stackedNestedPlotWidget.setCurrentIndex(row) # change the stackedWidgets index elif row > 1: self.stackedPlotWidget.setCurrentIndex(row - 1) def registerExpressionContextGenerator(self, generator: QgsExpressionContextGenerator): """ Register the panel's expression context generator with all relevant children """ self.x_combo.registerExpressionContextGenerator(generator) self.y_combo.registerExpressionContextGenerator(generator) self.z_combo.registerExpressionContextGenerator(generator) self.additional_info_combo.registerExpressionContextGenerator(generator) buttons = self.findChildren(QgsPropertyOverrideButton) for button in buttons: button.registerExpressionContextGenerator(generator) def register_data_defined_button(self, button, property_key: int): """ Registers a new data defined button, linked to the given property key (see values in PlotSettings) """ button.init(property_key, self.data_defined_properties, PlotSettings.DYNAMIC_PROPERTIES, None, False) button.changed.connect(self._update_property) def _update_property(self): """ Triggered when a property override button value is changed """ button = self.sender() self.data_defined_properties.setProperty(button.propertyKey(), button.toProperty()) def update_data_defined_button(self, button): """ Updates the current state of a property override button to reflect the current property value """ if button.propertyKey() < 0: return button.blockSignals(True) button.setToProperty(self.data_defined_properties.property(button.propertyKey())) button.blockSignals(False) def set_print_layout(self, print_layout): """ Sets the print layout linked with the widget, if in print layout mode """ self.linked_map_combo.setCurrentLayout(print_layout) self.linked_map_combo.setItemType(QgsLayoutItemRegistry.LayoutMap) def set_plot_type(self, plot_type: str): """ Sets the current plot type shown in the dialog """ self.plot_combo.setCurrentIndex(self.plot_combo.findData(plot_type)) def data_defined_color_updated(self): """ refreshing function for color data defined button checks is the datadefined button is active and check also the plot type in order to deactivate the color when not needed """ # if data defined button is active if self.in_color_defined_button.isActive(): # if plot is type for which using an expression for the color selection makes sense if self.ptype in ['scatter', 'bar', 'pie', 'ternary', 'histogram']: self.in_color_combo.setEnabled(False) self.color_scale_data_defined_in.setVisible(True) self.color_scale_data_defined_in.setEnabled(True) self.color_scale_data_defined_in_label.setVisible(True) self.color_scale_data_defined_in_label.setEnabled(True) self.color_scale_data_defined_in_check.setVisible(True) self.color_scale_data_defined_in_check.setEnabled(True) self.color_scale_data_defined_in_invert_check.setVisible(True) self.color_scale_data_defined_in_invert_check.setEnabled(True) # if plot is type for which using an expression for the color selection does not make sense else: self.in_color_combo.setEnabled(True) self.color_scale_data_defined_in.setVisible(False) self.color_scale_data_defined_in.setEnabled(False) self.color_scale_data_defined_in_label.setVisible(False) self.color_scale_data_defined_in_label.setEnabled(False) self.color_scale_data_defined_in_check.setVisible(False) self.color_scale_data_defined_in_check.setEnabled(False) self.color_scale_data_defined_in_invert_check.setVisible(False) self.color_scale_data_defined_in_invert_check.setEnabled(False) # if datadefined button is deactivated else: self.in_color_combo.setEnabled(True) self.color_scale_data_defined_in.setVisible(False) self.color_scale_data_defined_in.setEnabled(False) self.color_scale_data_defined_in_label.setVisible(False) self.color_scale_data_defined_in_check.setVisible(False) self.color_scale_data_defined_in_invert_check.setVisible(False) def selected_layer_changed(self, layer): """ Trigger actions after selected layer changes """ self.x_combo.setLayer(layer) self.y_combo.setLayer(layer) self.z_combo.setLayer(layer) self.additional_info_combo.setLayer(layer) buttons = self.findChildren(QgsPropertyOverrideButton) for button in buttons: button.setVectorLayer(layer) def layer_will_be_removed(self, layer_id): """ Triggered when a layer is about to be removed """ self.plot_factories = {k: v for k, v in self.plot_factories.items() if not v.source_layer or v.source_layer.id() != layer_id} def getJSmessage(self, status): """ landing method for statusBarMessage signal coming from PLOT.js_callback it decodes feature ids of clicked or selected plot elements, selects on map canvas and triggers a pan/zoom to them the method handles several exceptions: the first try/except is due to the connection to the init method second try/except looks into the decoded status, that is, it decodes the js dictionary and loop where it is necessary the dic js dictionary contains several information useful to handle correctly every operation """ try: dic = json.JSONDecoder().decode(status) except: # pylint: disable=bare-except # noqa: F401 dic = None # print('STATUS', status, dic) try: # check the user behavior linked to the js script # if a selection event is performed if dic['mode'] == 'selection': if dic['type'] == 'scatter': self.layer_combo.currentLayer().selectByIds(dic['id']) else: self.layer_combo.currentLayer().selectByIds(dic['tid']) # if a clicking event is performed depending on the plot type elif dic["mode"] == 'clicking': if dic['type'] == 'scatter': self.layer_combo.currentLayer().selectByIds([dic['fidd']]) elif dic["type"] == 'pie': exp = """ "{}" = '{}' """.format(dic['field'], dic['label']) # set the iterator with the expression as filter in feature request request = QgsFeatureRequest().setFilterExpression(exp) it = self.layer_combo.currentLayer().getFeatures(request) self.layer_combo.currentLayer().selectByIds([f.id() for f in it]) elif dic["type"] == 'histogram': vmin = dic['id'] - dic['bin_step'] / 2 vmax = dic['id'] + dic['bin_step'] / 2 exp = """ "{}" <= {} AND "{}" > {} """.format(dic['field'], vmax, dic['field'], vmin) request = QgsFeatureRequest().setFilterExpression(exp) it = self.layer_combo.currentLayer().getFeatures(request) self.layer_combo.currentLayer().selectByIds([f.id() for f in it]) elif dic["type"] == 'scatterternary': self.layer_combo.currentLayer().selectByIds([dic['fid']]) else: # build the expression from the js dic (customdata) exp = """ "{}" = '{}' """.format(dic['field'], dic['id']) # set the iterator with the expression as filter in feature request request = QgsFeatureRequest().setFilterExpression(exp) it = self.layer_combo.currentLayer().getFeatures(request) self.layer_combo.currentLayer().selectByIds([f.id() for f in it]) # print(exp) except: # pylint: disable=bare-except # noqa: F401 pass def helpPage(self): """ change the page of the manual according to the plot type selected and the language (looks for translations) """ # locale = QSettings().value('locale/userLocale', 'en_US')[0:2] self.help_view.load(QUrl('')) self.layouth.addWidget(self.help_view) help_url = QUrl('https://dataplotly-docs.readthedocs.io/en/latest/{}.html'.format(self.ptype)) self.help_view.load(help_url) def resizeEvent(self, event): """ reimplemented event to detect the dialog resizing """ self.resizeWindow.emit() return super(DataPlotlyPanelWidget, self).resizeEvent(event) def reloadPlotCanvas(self): """ just reload the plot view controlling the check state """ if self.live_update_check.isChecked(): self.plot_view.reload() def reloadPlotCanvas2(self): """ just reload the plot view """ self.plot_view.reload() def refreshListWidget(self): """ highlight the item in the QListWidget when the QStackWidget changes needed to highligh the correct icon when the plot is rendered """ self.listWidget.setCurrentRow(self.stackedPlotWidget.currentIndex()) def refreshWidgets(self): # pylint: disable=too-many-statements,too-many-branches """ just for refreshing the UI widgets depending on the plot type in the combobox to have a cleaner interface self.widgetType is a dict of widget depending on the plot type chosen 'all': is for all the plot type, else the name of the plot is explicitated BE AWARE: if loops are just for widgets that already exist! If a widget is proper to a specific plot and is put within the if statement, the method p.buildProperties will fail! In the statement there have to be only widgets that, for example, need to be re-rendered (label name...) """ # get the plot type from the combobox self.ptype = self.plot_combo.currentData() # BoxPlot BarPlot and Histogram orientation (same values) self.orientation_combo.clear() self.orientation_combo.addItem(self.tr('Vertical'), 'v') self.orientation_combo.addItem(self.tr('Horizontal'), 'h') # BoxPlot and Violin outliers self.outliers_combo.clear() self.outliers_combo.addItem(self.tr('No Outliers'), False) self.outliers_combo.addItem(self.tr('Standard Outliers'), 'outliers') self.outliers_combo.addItem(self.tr('Suspected Outliers'), 'suspectedoutliers') self.outliers_combo.addItem(self.tr('All Points'), 'all') # BoxPlot statistic types self.box_statistic_combo.clear() self.box_statistic_combo.addItem(self.tr('None'), False) self.box_statistic_combo.addItem(self.tr('Mean'), True) self.box_statistic_combo.addItem(self.tr('Standard Deviation'), 'sd') # BoxPlot and ScatterPlot X axis type self.x_axis_mode_combo.clear() self.x_axis_mode_combo.addItem(self.tr('Linear'), 'linear') self.x_axis_mode_combo.addItem(self.tr('Logarithmic'), 'log') self.x_axis_mode_combo.addItem(self.tr('Categorized'), 'category') self.y_axis_mode_combo.clear() self.y_axis_mode_combo.addItem(self.tr('Linear'), 'linear') self.y_axis_mode_combo.addItem(self.tr('Logarithmic'), 'log') self.y_axis_mode_combo.addItem(self.tr('Categorized'), 'category') # ScatterPlot marker types self.marker_types = OrderedDict([ (self.tr('Points'), 'markers'), (self.tr('Lines'), 'lines'), (self.tr('Points and Lines'), 'lines+markers') ]) self.marker_type_combo.clear() for k, v in self.marker_types.items(): self.marker_type_combo.addItem(k, v) # Point types self.point_types = OrderedDict([ (GuiUtils.get_icon('circle.svg'), 'circle'), (GuiUtils.get_icon('square.svg'), 'square'), (GuiUtils.get_icon('diamond.svg'), 'diamond'), (GuiUtils.get_icon('cross.svg'), 'cross'), (GuiUtils.get_icon('x.svg'), 'x'), (GuiUtils.get_icon('triangle.svg'), 'triangle'), (GuiUtils.get_icon('penta.svg'), 'penta'), (GuiUtils.get_icon('star.svg'), 'star'), ]) self.point_types2 = OrderedDict([ ('circle', 0), ('square', 1), ('diamond', 2), ('cross', 3), ('x', 4), ('triangle', 5), ('penta', 13), ('star', 17), ]) self.point_combo.clear() for k, v in self.point_types.items(): self.point_combo.addItem(k, '', v) self.line_types = OrderedDict([ (GuiUtils.get_icon('solid.png'), self.tr('Solid Line')), (GuiUtils.get_icon('dot.png'), self.tr('Dot Line')), (GuiUtils.get_icon('dash.png'), self.tr('Dash Line')), (GuiUtils.get_icon('longdash.png'), self.tr('Long Dash Line')), (GuiUtils.get_icon('dotdash.png'), self.tr('Dot Dash Line')), (GuiUtils.get_icon('longdashdot.png'), self.tr('Long Dash Dot Line')), ]) self.line_types2 = OrderedDict([ (self.tr('Solid Line'), 'solid'), (self.tr('Dot Line'), 'dot'), (self.tr('Dash Line'), 'dash'), (self.tr('Long Dash Line'), 'longdash'), (self.tr('Dot Dash Line'), 'dashdot'), (self.tr('Long Dash Dot Line'), 'longdashdot'), ]) self.line_combo.clear() for k, v in self.line_types.items(): self.line_combo.addItem(k, v) # BarPlot bar mode self.bar_mode_combo.clear() self.bar_mode_combo.addItem(self.tr('Grouped'), 'group') self.bar_mode_combo.addItem(self.tr('Stacked'), 'stack') self.bar_mode_combo.addItem(self.tr('Overlay'), 'overlay') # Histogram normalization mode self.hist_norm_combo.clear() self.hist_norm_combo.addItem(self.tr('Enumerated'), '') self.hist_norm_combo.addItem(self.tr('Percents'), 'percent') self.hist_norm_combo.addItem(self.tr('Probability'), 'probability') self.hist_norm_combo.addItem(self.tr('Density'), 'density') self.hist_norm_combo.addItem(self.tr('Prob Density'), 'probability density') # Contour Plot rendering type self.contour_type = OrderedDict([ (self.tr('Fill'), 'fill'), (self.tr('Heatmap'), 'heatmap'), (self.tr('Only Lines'), 'lines'), ]) self.contour_type_combo.clear() for k, v in self.contour_type.items(): self.contour_type_combo.addItem(k, v) # Contour Plot color scale and Data Defined Color scale scale_color_dict = {'Grey Scale': 'Greys', 'Green Scale': 'Greens', 'Fire Scale': 'Hot', 'BlueYellowRed': 'Portland', 'BlueGreenRed': 'Jet', 'BlueToRed': 'RdBu', 'BlueToRed Soft': 'Bluered', 'BlackRedYellowBlue': 'Blackbody', 'Terrain': 'Earth', 'Electric Scale': 'Electric', 'RedOrangeYellow': 'YIOrRd', 'DeepblueBlueWhite': 'YIGnBu', 'BlueWhitePurple': 'Picnic'} self.color_scale_combo.clear() self.color_scale_data_defined_in.clear() for k, v in scale_color_dict.items(): self.color_scale_combo.addItem(k, v) self.color_scale_data_defined_in.addItem(k, v) # according to the plot type, change the label names # BoxPlot if self.ptype == 'box' or self.ptype == 'violin': self.x_label.setText(self.tr('Grouping field \n(optional)')) # set the horizontal and vertical size of the label and reduce the label font size ff = QFont() ff.setPointSizeF(8) self.x_label.setFont(ff) self.x_label.setFixedWidth(100) self.orientation_label.setText(self.tr('Box orientation')) self.in_color_lab.setText(self.tr('Box color')) self.register_data_defined_button(self.in_color_defined_button, PlotSettings.PROPERTY_COLOR) elif self.ptype in ('scatter', 'ternary', 'bar', '2dhistogram', 'contour', 'polar'): self.x_label.setText(self.tr('X field')) self.x_label.setFont(self.font()) if self.ptype in ('scatter', 'ternary'): self.in_color_lab.setText(self.tr('Marker color')) elif self.ptype == 'bar': self.orientation_label.setText(self.tr('Bar orientation')) self.in_color_lab.setText(self.tr('Bar color')) self.register_data_defined_button(self.in_color_defined_button, PlotSettings.PROPERTY_COLOR) elif self.ptype == 'pie': self.x_label.setText(self.tr('Grouping field')) ff = QFont() ff.setPointSizeF(8.5) self.x_label.setFont(ff) self.x_label.setFixedWidth(80) # Register button again with more specific help text self.in_color_defined_button.init( PlotSettings.PROPERTY_COLOR, self.data_defined_properties.property(PlotSettings.PROPERTY_COLOR), QgsPropertyDefinition( 'color', QgsPropertyDefinition.DataType.DataTypeString, 'Color Array', "string [r,g,b,a] as int 0-255 or #AARRGGBB as hex or color as color's name, " "or an array of such strings" ), None, False ) self.in_color_defined_button.changed.connect(self._update_property) elif self.ptype == 'histogram': # Register button again with more specific help text self.in_color_defined_button.init( PlotSettings.PROPERTY_COLOR, self.data_defined_properties.property(PlotSettings.PROPERTY_COLOR), QgsPropertyDefinition( 'color', QgsPropertyDefinition.DataType.DataTypeString, 'Color Array', "string [r,g,b,a] as int 0-255 or #AARRGGBB as hex or color as color's name, " "or an array of such strings" ), None, False ) # info combo for data hovering self.info_combo.clear() self.info_combo.addItem(self.tr('All Values'), 'all') self.info_combo.addItem(self.tr('X Values'), 'x') self.info_combo.addItem(self.tr('Y Values'), 'y') self.info_combo.addItem(self.tr('No Data'), 'none') # label text position choice self.combo_text_position.clear() self.combo_text_position.addItem(self.tr('Automatic'), 'auto') self.combo_text_position.addItem(self.tr('Inside bar'), 'inside') self.combo_text_position.addItem(self.tr('Outside bar'), 'outside') # Violin side self.violinSideCombo.clear() self.violinSideCombo.addItem(self.tr('Both Sides'), 'both') self.violinSideCombo.addItem(self.tr('Only Left'), 'negative') self.violinSideCombo.addItem(self.tr('Only right'), 'positive') # dictionary with all the widgets and the plot they belong to self.widgetType = { # plot properties self.layer_combo: ['all'], self.feature_subset_defined_button: ['all'], self.x_label: ['all'], self.x_combo: ['all'], self.y_label: ['scatter', 'bar', 'box', 'pie', '2dhistogram', 'polar', 'ternary', 'contour', 'violin'], self.y_combo: ['scatter', 'bar', 'box', 'pie', '2dhistogram', 'polar', 'ternary', 'contour', 'violin'], self.z_label: ['ternary'], self.z_combo: ['ternary'], self.info_label: ['scatter'], self.info_combo: ['scatter'], self.in_color_lab: ['scatter', 'bar', 'box', 'pie', 'histogram', 'polar', 'ternary', 'violin'], self.in_color_combo: ['scatter', 'bar', 'box', 'pie', 'histogram', 'polar', 'ternary', 'violin'], self.in_color_defined_button: ['scatter', 'bar', 'box', 'pie', 'histogram', 'ternary'], self.color_scale_data_defined_in: ['scatter', 'bar', 'pie', 'histogram', 'ternary'], self.color_scale_data_defined_in_label: ['scatter', 'bar', 'ternary'], self.color_scale_data_defined_in_check: ['scatter', 'bar', 'ternary'], self.color_scale_data_defined_in_invert_check: ['bar', 'ternary'], self.out_color_lab: ['scatter', 'bar', 'box', 'pie', 'histogram', 'polar', 'ternary', 'violin'], self.out_color_combo: ['scatter', 'bar', 'box', 'pie', 'histogram', 'polar', 'ternary', 'violin'], self.marker_width_lab: ['scatter', 'bar', 'box', 'histogram', 'polar', 'ternary', 'violin'], self.marker_width: ['scatter', 'bar', 'box', 'histogram', 'polar', 'ternary', 'violin'], self.stroke_defined_button: ['scatter', 'bar', 'box', 'histogram', 'polar', 'ternary', 'violin'], self.marker_size_lab: ['scatter', 'polar', 'ternary'], self.marker_size: ['scatter', 'polar', 'ternary'], self.size_defined_button: ['scatter', 'ternary'], self.marker_type_lab: ['scatter', 'polar'], self.marker_type_combo: ['scatter', 'polar'], self.alpha_lab: ['scatter', 'bar', 'box', 'histogram', 'polar', 'ternary', 'violin', 'contour'], self.opacity_widget: ['scatter', 'bar', 'box', 'pie', 'histogram', 'polar', 'ternary', 'violin', 'contour'], self.properties_group_box: ['scatter', 'bar', 'box', 'pie', 'histogram', 'polar', 'ternary', 'contour', '2dhistogram', 'violin'], self.bar_mode_lab: ['bar', 'histogram'], self.bar_mode_combo: ['bar', 'histogram'], self.legend_label: ['all'], self.legend_title: ['all'], self.legend_title_defined_button: ['all'], self.point_lab: ['scatter', 'ternary', 'polar'], self.point_combo: ['scatter', 'ternary', 'polar'], self.line_lab: ['scatter', 'polar'], self.line_combo: ['scatter', 'polar'], self.color_scale_label: ['contour', '2dhistogram'], self.color_scale_combo: ['contour', '2dhistogram'], self.contour_type_label: ['contour'], self.contour_type_combo: ['contour'], self.show_lines_check: ['contour'], # layout customization self.show_legend_check: ['all'], self.orientation_legend_check: ['scatter', 'bar', 'box', 'histogram', 'ternary', 'pie', 'violin'], self.plot_title_lab: ['all'], self.plot_title_line: ['all'], self.plot_title_defined_button: ['all'], self.x_axis_label: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'ternary', 'violin'], self.x_axis_title: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'ternary', 'violin'], self.x_axis_title_defined_button: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'ternary', 'violin'], self.y_axis_label: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'ternary', 'violin'], self.y_axis_title: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'ternary', 'violin'], self.y_axis_title_defined_button: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'ternary', 'violin'], self.z_axis_label: ['ternary'], self.z_axis_title: ['ternary'], self.z_axis_title_defined_button: ['ternary'], self.x_axis_mode_label: ['scatter', 'box'], self.y_axis_mode_label: ['scatter', 'box'], self.x_axis_mode_combo: ['scatter', 'box'], self.y_axis_mode_combo: ['scatter', 'box'], self.invert_x_check: ['scatter', 'bar', 'box', 'histogram', '2dhistogram'], self.invert_y_check: ['scatter', 'bar', 'box', 'histogram', '2dhistogram'], self.x_axis_min: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.x_axis_max: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.x_axis_min_defined_button: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.x_axis_max_defined_button: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.y_axis_min: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.y_axis_max: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.y_axis_min_defined_button: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.y_axis_max_defined_button: ['scatter', 'bar', 'box', 'histogram', '2dhistogram', 'violin'], self.orientation_label: ['bar', 'box', 'histogram', 'violin'], self.orientation_combo: ['bar', 'box', 'histogram', 'violin'], self.box_statistic_label: ['box'], self.box_statistic_combo: ['box'], self.outliers_label: ['box', 'violin'], self.outliers_combo: ['box', 'violin'], self.showMeanCheck: ['violin'], self.range_slider_combo: ['scatter'], self.hist_norm_label: ['histogram'], self.hist_norm_combo: ['histogram'], self.additional_info_label: ['scatter', 'ternary', 'bar'], self.additional_info_combo: ['scatter', 'ternary', 'bar'], self.hover_as_text_check: ['scatter'], self.label_text_position: ['bar'], self.combo_text_position: ['bar'], self.cumulative_hist_check: ['histogram'], self.invert_hist_check: ['histogram'], self.bins_check: ['histogram'], self.bins_value: ['histogram'], self.bar_gap_label: ['histogram'], self.bar_gap: ['histogram'], self.violinSideLabel: ['violin'], self.violinSideCombo: ['violin'], self.violinBox: ['violin'], } # enable the widget according to the plot type for k, v in self.widgetType.items(): if 'all' in v or self.ptype in v: k.setEnabled(True) k.setVisible(True) else: k.setEnabled(False) k.setVisible(False) # disable by default the bins value box # if not explicit, the upper loop will enable it self.bins_value.setEnabled(False) # disable at first run the color data defined buttons self.color_scale_data_defined_in.setVisible(False) self.color_scale_data_defined_in_label.setVisible(False) self.color_scale_data_defined_in_check.setVisible(False) self.color_scale_data_defined_in_invert_check.setVisible(False) def refreshWidgets2(self): """ just refresh the UI to make the radiobuttons visible when SubPlots """ # enable radio buttons for subplots if self.subcombo.currentData() == 'subplots': self.radio_rows.setEnabled(True) self.radio_rows.setVisible(True) self.radio_columns.setEnabled(True) self.radio_columns.setVisible(True) else: self.radio_rows.setEnabled(False) self.radio_rows.setVisible(False) self.radio_columns.setEnabled(False) self.radio_columns.setVisible(False) def refreshWidgets3(self): """ refresh the UI according to Point, Lines or both choosen for the symbols of scatterplot """ if self.marker_type_combo.currentText() == self.tr('Points'): self.point_lab.setEnabled(True) self.point_lab.setVisible(True) self.point_combo.setEnabled(True) self.point_combo.setVisible(True) self.line_lab.setEnabled(False) self.line_lab.setVisible(False) self.line_combo.setEnabled(False) self.line_combo.setVisible(False) elif self.marker_type_combo.currentText() == self.tr('Lines'): self.point_lab.setEnabled(False) self.point_lab.setVisible(False) self.point_combo.setEnabled(False) self.point_combo.setVisible(False) self.line_lab.setEnabled(True) self.line_lab.setVisible(True) self.line_combo.setEnabled(True) self.line_combo.setVisible(True) else: self.point_lab.setEnabled(True) self.point_lab.setVisible(True) self.point_combo.setEnabled(True) self.point_combo.setVisible(True) self.line_lab.setEnabled(True) self.line_lab.setVisible(True) self.line_combo.setEnabled(True) self.line_combo.setVisible(True) def setLegend(self): """ set the legend from the fields combo boxes """ if self.ptype in ('box', 'bar'): self.legend_title.setText(self.y_combo.currentText()) elif self.ptype == 'histogram': self.legend_title.setText(self.x_combo.currentText()) else: legend_title_string = ('{} - {}'.format(self.x_combo.currentText(), self.y_combo.currentText())) self.legend_title.setText(legend_title_string) def get_settings(self) -> PlotSettings: # pylint: disable=R0915 """ Returns the plot settings as currently defined in the dialog """ # get the plot type from the combo box self.ptype = self.plot_combo.currentData() # if colorscale should be visible or not color_scale_visible = self.color_scale_data_defined_in_check.isVisible() and self.color_scale_data_defined_in_check.isChecked() # dictionary of all the plot properties plot_properties = {'custom': [self.x_combo.currentText()], 'hover_text': self.info_combo.currentData(), 'hover_label_text': '+text' if self.hover_as_text_check.isChecked() else None, 'hover_label_position': self.combo_text_position.currentData(), 'x_name': self.x_combo.currentText(), 'y_name': self.y_combo.currentText(), 'z_name': self.z_combo.currentText(), 'in_color': self.in_color_combo.color().name(), 'show_colorscale_legend': color_scale_visible, 'invert_color_scale': self.color_scale_data_defined_in_invert_check.isChecked(), 'out_color': self.out_color_combo.color().name(), 'marker_width': self.marker_width.value(), 'marker_size': self.marker_size.value(), 'marker_symbol': self.point_types2[self.point_combo.currentData()], 'line_dash': self.line_types2[self.line_combo.currentText()], 'box_orientation': self.orientation_combo.currentData(), 'marker': self.marker_types[self.marker_type_combo.currentText()], 'opacity': self.opacity_widget.opacity(), 'box_stat': self.box_statistic_combo.currentData(), 'box_outliers': self.outliers_combo.currentData(), 'name': self.legend_title.text(), 'normalization': self.hist_norm_combo.currentData(), 'cont_type': self.contour_type[self.contour_type_combo.currentText()], 'color_scale': None, 'show_lines': self.show_lines_check.isChecked(), 'cumulative': self.cumulative_hist_check.isChecked(), 'invert_hist': 'decreasing' if self.invert_hist_check.isChecked() else 'increasing', 'bins': self.bins_value.value(), 'show_mean_line': self.showMeanCheck.isChecked(), 'violin_side': self.violinSideCombo.currentData(), 'violin_box': self.violinBox.isChecked(), 'selected_features_only': self.selected_feature_check.isChecked(), 'visible_features_only': self.visible_feature_check.isChecked(), 'color_scale_data_defined_in_check': False, 'color_scale_data_defined_in_invert_check': False, 'marker_type_combo': self.marker_type_combo.currentText(), 'point_combo': self.point_combo.currentText(), 'line_combo': self.line_combo.currentText(), 'contour_type_combo': self.contour_type_combo.currentText(), 'show_lines_check': self.show_lines_check.isChecked() } if self.in_color_defined_button.isActive(): plot_properties['color_scale_data_defined_in_check'] = self.color_scale_data_defined_in_check.isChecked() plot_properties[ 'color_scale_data_defined_in_invert_check'] = self.color_scale_data_defined_in_invert_check.isChecked() if self.ptype in self.widgetType[self.color_scale_data_defined_in]: plot_properties['color_scale'] = self.color_scale_data_defined_in.currentData() else: plot_properties['color_scale'] = self.color_scale_combo.currentData() # add widgets properties to the dictionary # build the layout customizations layout_properties = {'legend': self.show_legend_check.isChecked(), 'legend_orientation': 'h' if self.orientation_legend_check.isChecked() else 'v', 'title': self.plot_title_line.text(), 'x_title': self.x_axis_title.text(), 'y_title': self.y_axis_title.text(), 'z_title': self.z_axis_title.text(), 'range_slider': {'visible': self.range_slider_combo.isChecked(), 'borderwidth': 1}, 'bar_mode': self.bar_mode_combo.currentData(), 'x_type': self.x_axis_mode_combo.currentData(), 'y_type': self.y_axis_mode_combo.currentData(), 'x_inv': None if not self.invert_x_check.isChecked() else 'reversed', 'y_inv': None if not self.invert_y_check.isChecked() else 'reversed', 'x_min': self.x_axis_min.value() if self.x_axis_bounds_check.isChecked() else None, 'x_max': self.x_axis_max.value() if self.x_axis_bounds_check.isChecked() else None, 'y_min': self.y_axis_min.value() if self.y_axis_bounds_check.isChecked() else None, 'y_max': self.y_axis_max.value() if self.y_axis_bounds_check.isChecked() else None, 'bargaps': self.bar_gap.value(), 'additional_info_expression': self.additional_info_combo.expression(), 'bins_check': self.bins_check.isChecked()} settings = PlotSettings(plot_type=self.ptype, properties=plot_properties, layout=layout_properties, source_layer_id=self.layer_combo.currentLayer().id() if self.layer_combo.currentLayer() else None) settings.data_defined_properties = self.data_defined_properties return settings def set_layer_id(self, layer_id: str): """ Sets the layer to use for plotting, by layer ID """ layer = QgsProject.instance().mapLayer(layer_id) if not layer: return self.layer_combo.setLayer(layer) def set_settings(self, settings: PlotSettings): # pylint: disable=too-many-statements """ Takes a PlotSettings object and fill the widgets with the settings """ self.set_plot_type(settings.plot_type) # Set the plot properties self.set_layer_id(settings.source_layer_id) self.selected_feature_check.setChecked(settings.properties.get('selected_features_only', False)) self.visible_feature_check.setChecked(settings.properties.get('visible_features_only', False)) self.data_defined_properties = settings.data_defined_properties buttons = self.findChildren(QgsPropertyOverrideButton) for button in buttons: self.update_data_defined_button(button) self.x_combo.setExpression(settings.properties.get('x_name', '')) self.y_combo.setExpression(settings.properties.get('y_name', '')) self.z_combo.setExpression(settings.properties.get('z_name', '')) self.in_color_combo.setColor( QColor(settings.properties.get('in_color', '#8ebad9'))) self.marker_size.setValue(settings.properties.get('marker_size', 10)) self.color_scale_data_defined_in.setCurrentIndex( self.color_scale_data_defined_in.findData(settings.properties.get('color_scale', None))) self.color_scale_data_defined_in_check.setChecked(settings.properties.get('color_scale_data_defined_in_check', False)) self.color_scale_data_defined_in_invert_check.setChecked( settings.properties.get('color_scale_data_defined_in_invert_check', False)) self.out_color_combo.setColor( QColor(settings.properties.get('out_color', '#1f77b4'))) self.marker_width.setValue(settings.properties.get('marker_width', 1)) self.marker_type_combo.setCurrentText( settings.properties.get('marker_type_combo', 'Points')) self.point_combo.setCurrentText(settings.properties.get('point_combo', '')) self.line_combo.setCurrentText( settings.properties.get('line_combo', 'Solid Line')) self.contour_type_combo.setCurrentText( settings.properties.get('contour_type_combo', 'Fill')) self.show_lines_check.setChecked(settings.properties.get('show_lines_check', False)) self.color_scale_combo.setCurrentIndex(self.color_scale_combo.findData(settings.properties.get('color_scale', None))) self.opacity_widget.setOpacity(settings.properties.get('opacity', 1)) self.orientation_legend_check.setChecked(settings.layout.get('legend_orientation') == 'h') self.range_slider_combo.setChecked(settings.layout['range_slider']['visible']) self.plot_title_line.setText( settings.layout.get('title', 'Plot Title')) self.legend_title.setText(settings.properties.get('name', '')) self.x_axis_title.setText(settings.layout.get('x_title', '')) self.y_axis_title.setText(settings.layout.get('y_title', '')) self.z_axis_title.setText(settings.layout.get('z_title', '')) self.info_combo.setCurrentIndex(self.info_combo.findData(settings.properties.get('hover_text', None))) self.additional_info_combo.setExpression(settings.layout.get('additional_info_expression', '')) self.hover_as_text_check.setChecked(settings.properties.get('hover_label_text') == '+text') self.combo_text_position.setCurrentIndex( self.combo_text_position.findData(settings.layout.get('hover_label_position', 'auto'))) self.invert_x_check.setChecked(settings.layout.get('x_inv') == 'reversed') self.x_axis_mode_combo.setCurrentIndex(self.x_axis_mode_combo.findData(settings.layout.get('x_type', None))) self.invert_y_check.setChecked(settings.layout.get('y_inv') == 'reversed') self.y_axis_mode_combo.setCurrentIndex(self.y_axis_mode_combo.findData(settings.layout.get('y_type', None))) self.x_axis_bounds_check.setChecked(settings.layout.get('x_min', None) is not None) self.x_axis_bounds_check.setCollapsed(settings.layout.get('x_min', None) is None) self.x_axis_min.setValue(settings.layout.get('x_min') if settings.layout.get('x_min', None) is not None else 0.0) self.x_axis_max.setValue(settings.layout.get('x_max') if settings.layout.get('x_max', None) is not None else 0.0) self.y_axis_bounds_check.setChecked(settings.layout.get('y_min', None) is not None) self.y_axis_bounds_check.setCollapsed(settings.layout.get('y_min', None) is None) self.y_axis_min.setValue(settings.layout.get('y_min') if settings.layout.get('y_min', None) is not None else 0.0) self.y_axis_max.setValue(settings.layout.get('y_max') if settings.layout.get('y_max', None) is not None else 0.0) self.orientation_combo.setCurrentIndex(self.orientation_combo.findData(settings.properties.get('box_orientation', 'v'))) self.bar_mode_combo.setCurrentIndex(self.bar_mode_combo.findData(settings.layout.get('bar_mode', None))) self.hist_norm_combo.setCurrentIndex(self.hist_norm_combo.findData(settings.properties.get('normalization', None))) self.box_statistic_combo.setCurrentIndex(self.box_statistic_combo.findData(settings.properties.get('box_stat', None))) self.outliers_combo.setCurrentIndex(self.outliers_combo.findData(settings.properties.get('box_outliers', None))) self.violinSideCombo.setCurrentIndex(self.violinSideCombo.findData(settings.properties.get('violin_side', None))) self.violinBox.setChecked(settings.properties.get('violin_box', False)) self.showMeanCheck.setChecked(settings.properties.get('show_mean_line', False)) self.cumulative_hist_check.setChecked(settings.properties.get('cumulative', False)) self.invert_hist_check.setChecked(settings.properties.get('invert_hist') == 'decreasing') self.bins_check.setChecked(settings.layout.get('bins_check', False)) self.bins_value.setValue(settings.properties.get('bins', 0)) self.bar_gap.setValue(settings.layout.get('bargaps', 0)) self.show_legend_check.setChecked(settings.layout.get('legend', True)) def create_plot_factory(self) -> PlotFactory: """ Creates a PlotFactory based on the settings defined in the dialog """ settings = self.get_settings() visible_region = None if settings.properties['visible_features_only']: visible_region = QgsReferencedRectangle(self.iface.mapCanvas().extent(), self.iface.mapCanvas().mapSettings().destinationCrs()) # plot instance plot_factory = PlotFactory(settings, visible_region=visible_region) # unique name for each plot trace (name is idx_plot, e.g. 1_scatter) self.pid = ('{}_{}'.format(str(self.idx), settings.plot_type)) # create default dictionary that contains all the plot and properties self.plot_factories[self.pid] = plot_factory plot_factory.plot_built.connect(partial(self.refresh_plot, plot_factory)) # just add 1 to the index self.idx += 1 # enable the Update Plot button self.update_btn.setEnabled(True) return plot_factory def update_plot_visible_rect(self): """ Called when the canvas rect changes, and we may need to update filtered plots """ region = QgsReferencedRectangle(self.iface.mapCanvas().extent(), self.iface.mapCanvas().mapSettings().destinationCrs()) for _, factory in self.plot_factories.items(): factory.set_visible_region(region) def refresh_plot(self, factory): """ Refreshes the plot built by the specified factory """ self.plot_path = factory.build_figure() self.refreshPlotView() def create_plot(self): """ call the method to effectively draw the final plot before creating the plot, it calls the method plotProperties in order to create the plot instance with all the properties taken from the UI """ # call the method to build all the Plot plotProperties plot_factory = self.create_plot_factory() # set the correct index page of the widget self.stackedPlotWidget.setCurrentIndex(1) # highlight the correct plot row in the listwidget self.listWidget.setCurrentRow(2) if self.subcombo.currentData() == 'single': # plot single plot, check the object dictionary length if len(self.plot_factories) <= 1: self.plot_path = plot_factory.build_figure() # to plot many plots in the same figure else: # plot list ready to be called within go.Figure pl = [] for _, v in self.plot_factories.items(): pl.append(v.trace[0]) self.plot_path = plot_factory.build_figures(self.ptype, pl) # choice to draw subplots instead depending on the combobox elif self.subcombo.currentData() == 'subplots': try: gr = len(self.plot_factories) pl = [] for _, v in self.plot_factories.items(): pl.append(v.trace[0]) # plot in single row and many columns if self.radio_rows.isChecked(): self.plot_path = plot_factory.build_sub_plots('row', 1, gr, pl) # plot in single column and many rows elif self.radio_columns.isChecked(): self.plot_path = plot_factory.build_sub_plots('col', gr, 1, pl) except: # pylint: disable=bare-except # noqa: F401 if self.message_bar: self.message_bar.pushMessage( self.tr("{} plot is not compatible for subplotting\n see ".format(self.ptype)), Qgis.MessageLevel(2), duration=5) return # connect to simple function that reloads the view self.refreshPlotView() def UpdatePlot(self): """ updates only the LAST plot created get the key of the last plot created and delete it from the plot container and call the method to create the plot with the updated settings """ if self.mode == DataPlotlyPanelWidget.MODE_CANVAS: plot_to_update = (sorted(self.plot_factories.keys())[-1]) del self.plot_factories[plot_to_update] self.create_plot() else: self.widgetChanged.emit() def refreshPlotView(self): """ just refresh the view, if the reload method is called immediately after the view creation it won't reload the page """ self.plot_url = QUrl.fromLocalFile(self.plot_path) self.plot_view.load(self.plot_url) self.layoutw.addWidget(self.plot_view) self.raw_plot_text.clear() with open(self.plot_path, 'r') as myfile: plot_text = myfile.read() self.raw_plot_text.setPlainText(plot_text) def clearPlotView(self): """ clear the content of the QWebView by loading an empty url and clear the raw text of the QPlainTextEdit """ self.plot_factories = {} try: self.plot_view.load(QUrl('')) self.layoutw.addWidget(self.plot_view) self.raw_plot_text.clear() if self.mode == DataPlotlyPanelWidget.MODE_CANVAS: # disable the Update Plot Button self.update_btn.setEnabled(False) except: # pylint: disable=bare-except # noqa: F401 pass def save_plot_as_image(self): """ Save the current plot view as a png image. The user can choose the path and the file name """ plot_file, _ = QFileDialog.getSaveFileName(self, self.tr("Save Plot"), "", "*.png") if not plot_file: return plot_file = QgsFileUtils.ensureFileNameHasExtension(plot_file, ['png']) frame = self.plot_view.page().mainFrame() self.plot_view.page().setViewportSize(frame.contentsSize()) # render image image = QImage(self.plot_view.page().viewportSize(), QImage.Format_ARGB32) painter = QPainter(image) frame.render(painter) painter.end() image.save(plot_file) if self.message_bar: self.message_bar.pushSuccess(self.tr('DataPlotly'), self.tr('Plot saved to {}').format( QUrl.fromLocalFile(plot_file).toString(), QDir.toNativeSeparators(plot_file))) def save_plot_as_html(self): """ Saves the plot as a local html file. Basically just let the user choose where to save the already existing html file created by plotly """ plot_file, _ = QFileDialog.getSaveFileName(self, self.tr("Save Plot"), "", "*.html") if not plot_file: return plot_file = QgsFileUtils.ensureFileNameHasExtension(plot_file, ['html']) copyfile(self.plot_path, plot_file) if self.message_bar: self.message_bar.pushSuccess(self.tr('DataPlotly'), self.tr('Saved plot to {}').format( QUrl.fromLocalFile(plot_file).toString(), QDir.toNativeSeparators(plot_file))) def showPlotFromDic(self, plot_input_dic): """ Allows to call the plugin from the python console param: plot_input_dic (dictionary): dictionary with specific structure, see below Code usage example: #import the plugins of QGIS from qgis.utils import plugins # create the instace of the plugin myplugin = plugins['DataPlotly'] # create a dictionary of values that will be elaborated by the method vl = iface.activeLayer() # empty dictionary that should filled up with GUI values dq = {} dq['plot_prop'] = {} dq['layout_prop'] = {} # plot type dq['plot_type'] = 'scatter' # vector layer dq["layer"] = vl # minimum X and Y axis values dq['plot_prop']['x'] = [i["some_field"] for i in vl.getFeatures()] dq['plot_prop']['y'] = [i["some_field"] for i in vl.getFeatures()] # call the final method myplugin.loadPlotFromDic(dq) """ # keys of the nested plot_prop and layout_prop have to be the SAME of # those created in buildProperties and buildLayout method # set some dialog widget from the input dictionary # plot type in the plot_combo combobox self.plot_combo.setCurrentIndex(self.plot_combo.findData(plot_input_dic["plot_type"])) try: self.layer_combo.setLayer(plot_input_dic["layer"]) if 'x_name' in plot_input_dic["plot_prop"] and plot_input_dic["plot_prop"]["x_name"]: self.x_combo.setField(plot_input_dic["plot_prop"]["x_name"]) if 'y_name' in plot_input_dic["plot_prop"] and plot_input_dic["plot_prop"]["y_name"]: self.y_combo.setField(plot_input_dic["plot_prop"]["y_name"]) if 'z_name' in plot_input_dic["plot_prop"] and plot_input_dic["plot_prop"]["z_name"]: self.z_combo.setField(plot_input_dic["plot_prop"]["z_name"]) except: # pylint: disable=bare-except # noqa: F401 pass settings = PlotSettings(plot_input_dic['plot_type'], properties=plot_input_dic["plot_prop"], layout=plot_input_dic["layout_prop"]) # create Plot instance factory = PlotFactory(settings) standalone_plot_path = factory.build_figure() standalone_plot_url = QUrl.fromLocalFile(standalone_plot_path) self.plot_view.load(standalone_plot_url) self.layoutw.addWidget(self.plot_view) # enable the Update Button to allow the updating of the plot self.update_btn.setEnabled(True) # the following code is necessary to let the user add other plots in # different plot canvas after the creation of the standolone plot # unique name for each plot trace (name is idx_plot, e.g. 1_scatter) self.pid = ('{}_{}'.format(str(self.idx), plot_input_dic["plot_type"])) # create default dictionary that contains all the plot and properties self.plot_factories[self.pid] = factory # just add 1 to the index self.idx += 1 def write_project(self, document: QDomDocument): """ Called when the current project is being saved. If the dialog opts to store its current settings in the project, it will use this hook to store them in the project XML """ if not self.save_to_project: return settings = self.get_settings() settings.write_to_project(document) def read_project(self, document: QDomDocument): """ Called when the current project is being read. If the dialog opts to read its current settings from a project, it will use this hook to read them from the project XML """ if not self.read_from_project: return settings = PlotSettings() if settings.read_from_project(document): # update the dock state to match the read settings self.set_settings(settings) def load_configuration(self): """ Loads configuration settings from a file """ file, _ = QFileDialog.getOpenFileName(self, self.tr("Load Configuration"), "", "XML files (*.xml)") if file: settings = PlotSettings() if settings.read_from_file(file): self.set_settings(settings) else: if self.message_bar: self.message_bar.pushWarning(self.tr('DataPlotly'), self.tr('Could not read settings from file')) def save_configuration(self): """ Saves configuration settings to a file """ file, _ = QFileDialog.getSaveFileName(self, self.tr("Save Configuration"), "", "XML files (*.xml)") if file: file = QgsFileUtils.ensureFileNameHasExtension(file, ['xml']) self.get_settings().write_to_file(file) if self.message_bar: self.message_bar.pushSuccess(self.tr('DataPlotly'), self.tr('Saved configuration to {}').format( QUrl.fromLocalFile(file).toString(), QDir.toNativeSeparators(file)))