diff --git a/seed/models/compliance_metrics.py b/seed/models/compliance_metrics.py index 98016a697c..840e20abdb 100644 --- a/seed/models/compliance_metrics.py +++ b/seed/models/compliance_metrics.py @@ -66,29 +66,26 @@ def evaluate(self, user_ali): # get properties (no filter) # property_response = properties_across_cycles(self.organization_id, -1, cycle_ids) # get properties (applies filter group) - display_field_id = Column.objects.get( + display_field = Column.objects.get( table_name="PropertyState", column_name=self.organization.property_display_field, organization=self.organization - ).id + ) # array of columns to return - column_ids = [display_field_id] + columns = [display_field] if self.actual_energy_column is not None: - column_ids.append(self.actual_energy_column.id) + columns.append(self.actual_energy_column) if self.target_energy_column is not None: - column_ids.append(self.target_energy_column.id) + columns.append(self.target_energy_column) if self.actual_emission_column is not None: - column_ids.append(self.actual_emission_column.id) + columns.append(self.actual_emission_column) if self.target_emission_column is not None: - column_ids.append(self.target_emission_column.id) + columns.append(self.target_emission_column) for col in self.x_axis_columns.all(): - column_ids.append(col.id) + columns.append(col) - # Unique ids - column_ids = [*set(column_ids)] - - property_response = properties_across_cycles_with_filters(self.organization_id, user_ali, cycle_ids, query_dict, column_ids) + property_response = properties_across_cycles_with_filters(self.organization_id, user_ali, cycle_ids, query_dict, columns) datasets = { "y": {"data": [], "label": "compliant"}, @@ -141,24 +138,24 @@ def evaluate(self, user_ali): for p in property_response[cyc]: # initialize - properties[p["property_view_id"]] = None + properties[p["id"]] = None # energy metric if metric["energy_metric"]: - properties[p["property_view_id"]] = self._calculate_compliance(p, metric["energy_bool"], "energy") + properties[p["id"]] = self._calculate_compliance(p, metric["energy_bool"], "energy") # emission metric - if metric["emission_metric"] and properties[p["property_view_id"]] != "u": + if metric["emission_metric"] and properties[p["id"]] != "u": temp_val = self._calculate_compliance(p, metric["emission_bool"], "emission") # reconcile if temp_val == "u": # unknown stays unknown (missing data) - properties[p["property_view_id"]] = "u" - elif properties[p["property_view_id"]] is None: + properties[p["id"]] = "u" + elif properties[p["id"]] is None: # only emission metric (not energy metric) - properties[p["property_view_id"]] = temp_val + properties[p["id"]] = temp_val else: # compliant if both are compliant - properties[p["property_view_id"]] = temp_val if temp_val == "n" else properties[p["property_view_id"]] + properties[p["id"]] = temp_val if temp_val == "n" else properties[p["id"]] # count compliant, non-compliant, unknown for each property with data for key in cnts: diff --git a/seed/static/seed/js/controllers/insights_property_controller.js b/seed/static/seed/js/controllers/insights_property_controller.js index d6089f2915..fb77d53a34 100644 --- a/seed/static/seed/js/controllers/insights_property_controller.js +++ b/seed/static/seed/js/controllers/insights_property_controller.js @@ -14,6 +14,9 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro 'filter_groups', 'property_columns', 'cycles', + 'access_level_tree', + 'user_service', + 'ah_service', 'spinner_utility', 'auth_payload', // eslint-disable-next-line func-names @@ -29,6 +32,9 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro filter_groups, property_columns, cycles, + access_level_tree, + user_service, + ah_service, spinner_utility, auth_payload ) { @@ -37,6 +43,13 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro $scope.organization = organization_payload.organization; $scope.auth = auth_payload.auth; + $scope.access_level_tree = access_level_tree.access_level_tree; + $scope.level_names = access_level_tree.access_level_names; + $scope.level_name_index = null; + $scope.potential_level_instances = []; + $scope.access_level_instance_id = null; + $scope.users_access_level_instance_id = user_service.get_access_level_instance().id; + // used by modal $scope.filter_groups = filter_groups; $scope.property_columns = property_columns; @@ -48,6 +61,28 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro $scope.show_help = !$scope.show_help; }; + function path_to_string(path) { + const orderedPath = []; + for (const i in $scope.level_names) { + if (Object.prototype.hasOwnProperty.call(path, $scope.level_names[i])) { + orderedPath.push(path[$scope.level_names[i]]); + } + } + return orderedPath.join(' : '); + } + + const access_level_instances_by_depth = ah_service.calculate_access_level_instances_by_depth($scope.access_level_tree); + // cannot select parents alis + const [users_depth] = Object.entries(access_level_instances_by_depth).find(([, x]) => x.length === 1 && x[0].id === parseInt($scope.users_access_level_instance_id, 10)); + $scope.change_selected_level_index = () => { + const new_level_instance_depth = parseInt($scope.level_name_index, 10) + parseInt(users_depth, 10); + $scope.potential_level_instances = access_level_instances_by_depth[new_level_instance_depth]; + for (const key in $scope.potential_level_instances) { + $scope.potential_level_instances[key].name = path_to_string($scope.potential_level_instances[key].path); + } + $scope.access_level_instance_id = null; + }; + // configs ($scope.configs set to saved_configs where still applies. // for example, if saved_configs.compliance_metric is 1, but 1 has been deleted, it does apply.) const saved_configs = JSON.parse(localStorage.getItem(`insights.property.configs.${$scope.organization.id}`)); @@ -154,6 +189,12 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro true ); + $scope.change_ali = () => { + localStorage.setItem(localStorageALIndex, JSON.stringify($scope.level_name_index)); + localStorage.setItem(localStorageALIID, JSON.stringify($scope.access_level_instance_id)); + _load_data(); + }; + // load data const _load_data = () => { if (_.isEmpty($scope.configs.compliance_metric)) { @@ -163,11 +204,19 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro } spinner_utility.show(); compliance_metric_service - .evaluate_compliance_metric($scope.configs.compliance_metric.id) + .evaluate_compliance_metric($scope.configs.compliance_metric.id, $scope.organization.id, $scope.access_level_instance_id) .then((data) => { $scope.data = data; }) .then(() => { + // if there's +3k properties, dont even bother charting them. + $scope.chartStatusMessage = ''; + const num_properties = Object.values($scope.data.properties_by_cycles).reduce((acc, curr) => acc + curr.length, 0); + if (num_properties > 3000) { + $scope.data.properties_by_cycles = Object.keys($scope.data.properties_by_cycles).reduce((acc, k) => ({ ...acc, [k]: [] }), {}); + $scope.chartStatusMessage = 'Too much data, try a different ali'; + } + if ($scope.data) { // set options // cycles @@ -307,7 +356,7 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro $scope.annotations = {}; _.forEach($scope.data.properties_by_cycles[$scope.configs.chart_cycle], (prop) => { - const item = { id: prop.property_view_id }; + const item = { id: prop.id }; item.name = _.find(prop, (v, k) => k.startsWith($scope.organization.property_display_field)); // x axis is easy item.x = _.find(prop, (v, k) => k.endsWith(`_${String($scope.configs.chart_xaxis)}`)); @@ -333,10 +382,10 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro } // place in appropriate dataset - if (_.includes($scope.data.results_by_cycles[$scope.configs.chart_cycle].y, prop.property_view_id)) { + if (_.includes($scope.data.results_by_cycles[$scope.configs.chart_cycle].y, prop.id)) { // compliant dataset datasets[0].data.push(item); - } else if (_.includes($scope.data.results_by_cycles[$scope.configs.chart_cycle].n, prop.property_view_id)) { + } else if (_.includes($scope.data.results_by_cycles[$scope.configs.chart_cycle].n, prop.id)) { // non-compliant dataset datasets[1].data.push(item); } else { @@ -606,6 +655,13 @@ angular.module('SEED.controller.insights_property', []).controller('insights_pro $scope.configs.annotation_visibility = $scope.display_annotation; }; + const localStorageALIndex = 'insights.property.configs.ALIndex'; + const localStorageALIID = 'insights.property.configs.ALIID'; + $scope.level_name_index = JSON.parse(localStorage.getItem(localStorageALIndex)) || '0'; + const new_level_instance_depth = parseInt($scope.level_name_index, 10) + parseInt(users_depth, 10); + $scope.potential_level_instances = access_level_instances_by_depth[new_level_instance_depth]; + $scope.access_level_instance_id = JSON.parse(localStorage.getItem(localStorageALIID)) || parseInt($scope.users_access_level_instance_id, 10); + setTimeout(_load_data, 0); // avoid race condition with route transition spinner. $scope.visibleIds = () => { diff --git a/seed/static/seed/js/controllers/program_setup_controller.js b/seed/static/seed/js/controllers/program_setup_controller.js index c7c85ca6a0..62ae820373 100644 --- a/seed/static/seed/js/controllers/program_setup_controller.js +++ b/seed/static/seed/js/controllers/program_setup_controller.js @@ -44,8 +44,14 @@ angular.module('SEED.controller.program_setup', []).controller('program_setup_co $scope.valid_column_data_types = ['number', 'float', 'integer', 'ghg', 'ghg_intensity', 'area', 'eui', 'boolean']; $scope.valid_x_axis_data_types = ['number', 'string', 'float', 'integer', 'ghg', 'ghg_intensity', 'area', 'eui', 'boolean']; - $scope.property_columns = _.reject(property_columns, (item) => (item.related || !$scope.valid_column_data_types.includes(item.data_type)) && item.derived_column == null).sort((a, b) => naturalSort(a.displayName, b.displayName)); - $scope.x_axis_columns = _.reject(property_columns, (item) => (item.related || !$scope.valid_x_axis_data_types.includes(item.data_type)) && item.derived_column == null).sort((a, b) => naturalSort(a.displayName, b.displayName)); + $scope.property_columns = _.reject(property_columns, (item) => ( + (item.related || !$scope.valid_column_data_types.includes(item.data_type)) && + item.derived_column == null + )).sort((a, b) => naturalSort(a.displayName, b.displayName)); + $scope.x_axis_columns = _.reject(property_columns, (item) => ( + (item.related || !$scope.valid_x_axis_data_types.includes(item.data_type)) && + item.derived_column == null + )).sort((a, b) => naturalSort(a.displayName, b.displayName)); $scope.x_axis_selection = ''; $scope.cycle_selection = ''; $scope.compliance_metrics_error = []; diff --git a/seed/static/seed/js/seed.js b/seed/static/seed/js/seed.js index b169b7e022..bb57381bb1 100644 --- a/seed/static/seed/js/seed.js +++ b/seed/static/seed/js/seed.js @@ -2586,7 +2586,15 @@ return inventory_service.get_property_columns_for_org(organization_id); } ], - cycles: ['cycle_service', (cycle_service) => cycle_service.get_cycles()] + cycles: ['cycle_service', (cycle_service) => cycle_service.get_cycles()], + access_level_tree: [ + 'organization_service', + 'user_service', + (organization_service, user_service) => { + const organization_id = user_service.get_organization().id; + return organization_service.get_organization_access_level_tree(organization_id); + } + ] } }) .state({ diff --git a/seed/static/seed/js/services/compliance_metric_service.js b/seed/static/seed/js/services/compliance_metric_service.js index df5344dcec..af64c45ae8 100644 --- a/seed/static/seed/js/services/compliance_metric_service.js +++ b/seed/static/seed/js/services/compliance_metric_service.js @@ -44,10 +44,11 @@ angular.module('SEED.service.compliance_metric', []).factory('compliance_metric_ }; // evaluate - const evaluate_compliance_metric = (metric_id, organization_id = user_service.get_organization().id) => $http + const evaluate_compliance_metric = (metric_id, organization_id = user_service.get_organization().id, access_level_instance_id = null) => $http .get(`/api/v3/compliance_metrics/${metric_id}/evaluate/`, { params: { - organization_id + organization_id, + ...(access_level_instance_id ? { access_level_instance_id } : {}) } }) .then((response) => response.data.data) diff --git a/seed/static/seed/partials/insights_property.html b/seed/static/seed/partials/insights_property.html index be636d22b7..1234fe013b 100644 --- a/seed/static/seed/partials/insights_property.html +++ b/seed/static/seed/partials/insights_property.html @@ -62,6 +62,22 @@ + +