diff --git a/ckanext/charts/assets/css/charts.css b/ckanext/charts/assets/css/charts.css index ee0a3a0..edb041b 100644 --- a/ckanext/charts/assets/css/charts.css +++ b/ckanext/charts/assets/css/charts.css @@ -1 +1 @@ -.row.wrapper.charts-view .dataset-resource-form{position:relative;padding:0}.resource-view-charts_builder_view label.no-dots:after,.resource-view-charts_view label.no-dots:after,#content div.charts-view label.no-dots:after{display:none}.resource-view-charts_builder_view #chart-container,.resource-view-charts_view #chart-container,#content div.charts-view #chart-container{max-height:600px}.resource-view-charts_builder_view select,.resource-view-charts_view select,#content div.charts-view select{border:1px solid #d0d0d0;padding:8px;border-radius:3px;width:100%}.resource-view-charts_builder_view #chart-clear,.resource-view-charts_view #chart-clear,#content div.charts-view #chart-clear{margin-left:auto}.resource-view-charts_builder_view .tab-content,.resource-view-charts_view .tab-content,#content div.charts-view .tab-content{display:block !important}.resource-view-charts_builder_view #chart-indicator,.resource-view-charts_view #chart-indicator,#content div.charts-view #chart-indicator{display:none}.resource-view-charts_builder_view .htmx-request#chart-indicator,.resource-view-charts_view .htmx-request#chart-indicator,#content div.charts-view .htmx-request#chart-indicator{display:inline}.charts-filters .filter-container{margin-bottom:1rem}.charts-filters .filter-container .filter-pair{display:flex;gap:.5rem;margin-bottom:1rem;padding-bottom:1rem;border-bottom:1px solid beige}.charts-filters .filter-container .filter-pair:last-of-type{margin-bottom:0;padding-bottom:0;border-bottom:unset}.charts-filters .filter-container .filter-pair .remove-pair{top:0;height:fit-content}.charts-filters .filter-container .filter-pair .ts-wrapper{flex-basis:45%}.charts-view--form .ts-wrapper .ts-control{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 20 20'%3e%3cpath stroke='%236b7280' stroke-linecap='round' stroke-linejoin='round' stroke-width='1.5' d='M6 8l4 4 4-4'/%3e%3c/svg%3e");background-position:right .5rem center;background-repeat:no-repeat;background-size:1.5em 1.5em}.charts-view--form .ts-wrapper.plugin-clear_button .ts-control{background:#fff}.charts-view--form .ts-wrapper.plugin-clear_button .ts-control .clear-button{right:.55rem;font-size:1.25rem}.charts-view--form .ts-wrapper .ts-dropdown{top:unset;margin-top:1px} \ No newline at end of file +.modal-open .module-resource{z-index:unset}.row.wrapper.charts-view .dataset-resource-form{position:relative;padding:0}.resource-view-charts_builder_view label.no-dots:after,.resource-view-charts_view label.no-dots:after,#content div.charts-view label.no-dots:after{display:none}.resource-view-charts_builder_view #chart-container,.resource-view-charts_view #chart-container,#content div.charts-view #chart-container{max-height:600px}.resource-view-charts_builder_view select,.resource-view-charts_view select,#content div.charts-view select{border:1px solid #d0d0d0;padding:8px;border-radius:3px;width:100%}.resource-view-charts_builder_view #chart-clear,.resource-view-charts_view #chart-clear,#content div.charts-view #chart-clear{margin-left:auto}.resource-view-charts_builder_view .tab-content,.resource-view-charts_view .tab-content,#content div.charts-view .tab-content{display:block !important}.resource-view-charts_builder_view #chart-indicator,.resource-view-charts_view #chart-indicator,#content div.charts-view #chart-indicator{display:none}.resource-view-charts_builder_view .htmx-request#chart-indicator,.resource-view-charts_view .htmx-request#chart-indicator,#content div.charts-view .htmx-request#chart-indicator{display:inline}.charts-filters .filter-container{margin-bottom:1rem}.charts-filters .filter-container .filter-pair{display:flex;gap:.5rem;margin-bottom:1rem;padding-bottom:1rem;border-bottom:1px solid beige}.charts-filters .filter-container .filter-pair:last-of-type{margin-bottom:0;padding-bottom:0;border-bottom:unset}.charts-filters .filter-container .filter-pair .remove-pair{top:0;height:fit-content}.charts-filters .filter-container .filter-pair .ts-wrapper{flex-basis:45%}.charts-view--form .ts-wrapper .ts-control{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 20 20'%3e%3cpath stroke='%236b7280' stroke-linecap='round' stroke-linejoin='round' stroke-width='1.5' d='M6 8l4 4 4-4'/%3e%3c/svg%3e");background-position:right .5rem center;background-repeat:no-repeat;background-size:1.5em 1.5em}.charts-view--form .ts-wrapper.plugin-clear_button .ts-control{background:#fff}.charts-view--form .ts-wrapper.plugin-clear_button .ts-control .clear-button{right:.55rem;font-size:1.25rem}.charts-view--form .ts-wrapper .ts-dropdown{top:unset;margin-top:1px} \ No newline at end of file diff --git a/ckanext/charts/assets/js/vendor/chartjs-adapter-moment.js b/ckanext/charts/assets/js/vendor/chartjs-adapter-moment.js new file mode 100644 index 0000000..9ea8681 --- /dev/null +++ b/ckanext/charts/assets/js/vendor/chartjs-adapter-moment.js @@ -0,0 +1,111 @@ +/* + +Purpose: +To handle and display date and time data in Chart.js scatter and bubble charts with type: 'time' on the x-axis. + +Why We Use It: +1. Date Parsing and Formatting: +- Enables robust parsing and formatting of date strings using the moment library. + +2. Chart.js Compatibility: +- Integrates with Chart.js for time-based data handling. +- Necessary for using type: 'time' in chart configurations. + +3. Advanced Date Manipulation: +- Provides functions for date manipulation like adding/subtracting time and calculating date differences. + +Example Usage: +To use type: 'time' for the x-axis in scatter and bubble charts: + +const config = { + type: 'scatter', + data: {...}, + options: { + scales: { + x: { + type: 'time', + time: { + unit: 'day' + } + } + } + } +}; + +The moment adapter ensures accurate processing and display of date and time data in Chart.js. + +*/ + + +/*! + * chartjs-adapter-moment v1.0.0 + * https://www.chartjs.org + * (c) 2021 chartjs-adapter-moment Contributors + * Released under the MIT license + */ +(function (global, factory) { + typeof exports === 'object' && typeof module !== 'undefined' ? factory(require('moment'), require('chart.js')) : + typeof define === 'function' && define.amd ? define(['moment', 'chart.js'], factory) : + (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.moment, global.Chart)); + }(this, (function (moment, chart_js) { 'use strict'; + + function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; } + + var moment__default = /*#__PURE__*/_interopDefaultLegacy(moment); + + const FORMATS = { + datetime: 'MMM D, YYYY, h:mm:ss a', + millisecond: 'h:mm:ss.SSS a', + second: 'h:mm:ss a', + minute: 'h:mm a', + hour: 'hA', + day: 'MMM D', + week: 'll', + month: 'MMM YYYY', + quarter: '[Q]Q - YYYY', + year: 'YYYY' + }; + + chart_js._adapters._date.override(typeof moment__default['default'] === 'function' ? { + _id: 'moment', // DEBUG ONLY + + formats: function() { + return FORMATS; + }, + + parse: function(value, format) { + if (typeof value === 'string' && typeof format === 'string') { + value = moment__default['default'](value, format); + } else if (!(value instanceof moment__default['default'])) { + value = moment__default['default'](value); + } + return value.isValid() ? value.valueOf() : null; + }, + + format: function(time, format) { + return moment__default['default'](time).format(format); + }, + + add: function(time, amount, unit) { + return moment__default['default'](time).add(amount, unit).valueOf(); + }, + + diff: function(max, min, unit) { + return moment__default['default'](max).diff(moment__default['default'](min), unit); + }, + + startOf: function(time, unit, weekday) { + time = moment__default['default'](time); + if (unit === 'isoWeek') { + weekday = Math.trunc(Math.min(Math.max(0, weekday), 6)); + return time.isoWeekday(weekday).startOf('day').valueOf(); + } + return time.startOf(unit).valueOf(); + }, + + endOf: function(time, unit) { + return moment__default['default'](time).endOf(unit).valueOf(); + } + } : {}); + +}))); diff --git a/ckanext/charts/assets/webassets.yml b/ckanext/charts/assets/webassets.yml index 3f954aa..b7c46c5 100644 --- a/ckanext/charts/assets/webassets.yml +++ b/ckanext/charts/assets/webassets.yml @@ -3,6 +3,7 @@ chartjs: output: ckanext-charts/%(version)s-chartjs.js contents: - js/vendor/chartjs.min.js + - js/vendor/chartjs-adapter-moment.js - js/charts-render-chartjs.js extra: preload: diff --git a/ckanext/charts/chart_builders/base.py b/ckanext/charts/chart_builders/base.py index 58d4690..f6dd23f 100644 --- a/ckanext/charts/chart_builders/base.py +++ b/ckanext/charts/chart_builders/base.py @@ -631,6 +631,18 @@ def height_field(self) -> dict[str, Any]: "group": "Data", } + def more_info_button_field(self) -> dict[str, Any]: + """ + Adds a "More info" button to the Data tab in the form, which triggers a pop-up. + This pop-up provides users with information about supported date formats. + """ + return { + "field_name": "more_info", + "label": "More info", + "form_snippet": "chart_more_info_button.html", + "group": "Data", + } + def size_field(self, choices: list[dict[str, str]]) -> dict[str, Any]: field = self.column_field(choices) field.update({"field_name": "size", "label": "Size", "group": "Structure"}) diff --git a/ckanext/charts/chart_builders/chartjs.py b/ckanext/charts/chart_builders/chartjs.py index d278a3c..d1b58dc 100644 --- a/ckanext/charts/chart_builders/chartjs.py +++ b/ckanext/charts/chart_builders/chartjs.py @@ -90,6 +90,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_multi_axis_field(columns), + self.more_info_button_field(), self.limit_field(), self.filter_field(columns), ] @@ -155,6 +156,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_multi_axis_field(columns), + self.more_info_button_field(), self.sort_x_field(), self.sort_y_field(), self.limit_field(), @@ -213,6 +215,7 @@ def get_form_fields(self): self.type_field(chart_types), self.values_field(columns), self.names_field(columns), + self.more_info_button_field(), self.limit_field(), self.filter_field(columns), ] @@ -254,8 +257,32 @@ def to_json(self) -> str: "data": dataset_data, } ] + return json.dumps(self._configure_date_axis(data)) + + def _configure_date_axis(self, data: dict[str, Any]) -> dict[str, Any]: + """ + Configure date settings for the x-axis if it uses 'date_time'. + """ + x_axis = data["options"]["x"] + scales = data["options"].get("scales", {}) + + if x_axis == "date_time": + x_scale = scales.get("x", {}) + x_scale.update( + { + "type": "time", + "time": { + "unit": "day", + "displayFormats": {"day": "YYYY-MM-DD"}, + }, + } + ) + scales["x"] = x_scale - return json.dumps(data) + if scales: + data["options"]["scales"] = scales + + return data class ChartJSScatterForm(BaseChartForm): @@ -277,6 +304,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_axis_field(columns), + self.more_info_button_field(), self.sort_x_field(), self.sort_y_field(), self.limit_field(), @@ -312,8 +340,7 @@ def to_json(self) -> str: data["data"]["datasets"] = [ {"label": self.settings["y"], "data": dataset_data}, ] - - return json.dumps(data) + return json.dumps(self._configure_date_axis(data)) def _calculate_bubble_radius(self, data_series: pd.Series, max_size: int) -> int: """Calculate bubble radius based on the size column""" @@ -403,6 +430,7 @@ def get_form_fields(self): columns, help_text="Select 3 or more different categorical variables (dimensions)", ), + self.more_info_button_field(), self.limit_field(), self.filter_field(columns), ] diff --git a/ckanext/charts/chart_builders/observable.py b/ckanext/charts/chart_builders/observable.py index 286da57..040ff5d 100644 --- a/ckanext/charts/chart_builders/observable.py +++ b/ckanext/charts/chart_builders/observable.py @@ -54,6 +54,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_axis_field(columns), + self.more_info_button_field(), self.sort_x_field(), self.sort_y_field(), self.fill_field(columns), @@ -112,6 +113,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_axis_field(columns), + self.more_info_button_field(), self.invert_x_field(), self.invert_y_field(), self.sort_x_field(), @@ -187,6 +189,7 @@ def get_form_fields(self): self.type_field(chart_types), self.values_field(columns), self.names_field(columns), + self.more_info_button_field(), self.opacity_field(), self.inner_radius_field(), self.stroke_width_field(), @@ -227,6 +230,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_axis_field(columns), + self.more_info_button_field(), self.sort_x_field(), self.sort_y_field(), self.color_field(columns), @@ -265,6 +269,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_axis_field(columns), + self.more_info_button_field(), self.sort_x_field(), self.sort_y_field(), self.color_field(columns), diff --git a/ckanext/charts/chart_builders/plotly.py b/ckanext/charts/chart_builders/plotly.py index 781971c..a355d41 100644 --- a/ckanext/charts/chart_builders/plotly.py +++ b/ckanext/charts/chart_builders/plotly.py @@ -112,6 +112,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_axis_field(columns), + self.more_info_button_field(), self.log_x_field(), self.log_y_field(), self.sort_x_field(), @@ -143,6 +144,7 @@ def get_form_fields(self): self.type_field(chart_types), self.values_field(columns), self.names_field(columns), + self.more_info_button_field(), self.opacity_field(), self.limit_field(), self.filter_field(columns), @@ -185,6 +187,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.plotly_y_multi_axis_field(columns, 2), + self.more_info_button_field(), self.invert_x_field(), self.invert_y_field(), self.sort_x_field(), @@ -229,6 +232,7 @@ def get_form_fields(self): self.type_field(chart_types), self.x_axis_field(columns), self.y_axis_field(columns), + self.more_info_button_field(), self.log_x_field(), self.log_y_field(), self.sort_x_field(), diff --git a/ckanext/charts/templates/scheming/form_snippets/chart_more_info_button.html b/ckanext/charts/templates/scheming/form_snippets/chart_more_info_button.html new file mode 100644 index 0000000..992583f --- /dev/null +++ b/ckanext/charts/templates/scheming/form_snippets/chart_more_info_button.html @@ -0,0 +1,63 @@ +
+
+ +
+
+ +{% set datetime_mapping = { + "YYYY-MM-DD": "2023-07-24", + "YYYY/MM/DD": "2023/07/24", + "MM-DD-YYYY": "07-24-2023", + "MM/DD/YYYY": "07/24/2023", + "DD-MM-YYYY": "24-07-2023", + "DD/MM/YYYY": "24/07/2023", + "dd/MMM/yyyy": "25/Nov/2023", + "YYYYMMDD": "20231125", + "YYYY-MM-DDTHH:MM:SS": "2023-07-24T14:30:00", + "YYYY-MM-DD HH:mm:ss": "2023-11-25 12:34:56", + "YYYY-MM-DDTHH:mm:ssZ": "2023-11-25T12:34:56Z", + "YYYY-MM-DD HH:mm:ss.SSS": "2023-11-25 12:34:56.789", + "MM/dd/yyyy hh:mm:ss a": "9/28/2023 2:23:15 PM", + "MM/dd/yyyy hh:mm:ss a:SSS": "8/5/2023 3:31:18 AM:234", + "MMdd_HH:mm:ss.SSS": "0423_11:42:35.883", + "dd MMM yyyy HH:mm:ss*SSS": "23 Apr 2023 10:32:35*311", + "dd MMM yyyy HH:mm:ss": "23 Apr 2023 11:42:35", + "yyMMdd HH:mm:ss": "220423 11:42:35", + "yy-MM-dd HH:mm:ss": "23-04-19 12:00:17" +} %} + + diff --git a/ckanext/charts/theme/mixins.scss b/ckanext/charts/theme/mixins.scss index e69de29..bd9ca72 100644 --- a/ckanext/charts/theme/mixins.scss +++ b/ckanext/charts/theme/mixins.scss @@ -0,0 +1,5 @@ +.modal-open { + .module-resource { + z-index: unset; + } +}