Compare commits

..

31 Commits

Author SHA1 Message Date
Gabriel Dutra
19343a0520 Clear QueryBasedParameterInput 2020-11-23 19:18:35 -03:00
Gabriel Dutra
c1ed8848f0 Merge branch 'master' into query-based-dropdown--parameters 2020-11-23 16:42:06 -03:00
Gabriel Dutra
b40070d7f5 Use LabeledValues for parameterized queries 2020-11-23 16:40:18 -03:00
Gabriel Dutra
bd9ce68f68 Don't filter out values when param has search 2020-11-20 15:14:17 -03:00
Gabriel Dutra
0c0b62ae1a Remove searchTerm from structure 2020-11-20 15:13:47 -03:00
Gabriel Dutra
08bcdf77d0 Mock query instead of query_has_parameters 2020-11-12 09:11:54 -03:00
Gabriel Dutra
aa2064b1ab Fix other dropdown_values usages to use query obj 2020-11-11 13:58:01 -03:00
Gabriel Dutra
d0a787cab1 Make NoResultFound invalid parameters 2020-11-10 22:10:47 -03:00
Gabriel Dutra
a741341938 Oops 2020-11-10 20:46:51 -03:00
Gabriel Dutra
53385fa24b Merge branch 'master' into query-based-dropdown--parameters 2020-11-10 15:19:43 -03:00
Gabriel Dutra
f396c96457 Merge branch 'master' into query-based-dropdown--parameters 2020-02-25 07:49:36 -03:00
Gabriel Dutra
8bfcbf21e3 Remove redundant import 2020-02-24 11:44:28 -03:00
Gabriel Dutra
8a1640c4e7 Separate InputPopover component 2020-02-24 11:44:18 -03:00
Gabriel Dutra
a37e7f93dc Add is_safe test for queries with params 2020-02-22 15:47:29 -03:00
Gabriel Dutra
cc34e781d3 Small updates
- Change searchTerm separator
- Add cy.wait
2020-02-22 15:23:43 -03:00
Gabriel Dutra
6aa0ea715e Invert tooltip messages order 2020-02-22 14:08:19 -03:00
Gabriel Dutra
6c27619671 Make Parameter Mapping required in UI 2020-02-21 23:00:26 -03:00
Gabriel Dutra
6eeb3b3eb2 Separate UI components 2020-02-21 15:49:23 -03:00
Gabriel Dutra
d40edb81c2 Fix backend tests 2020-02-21 14:18:59 -03:00
Gabriel Dutra
f128b4b85f Only allow search for Text Parameters 2020-02-21 13:36:06 -03:00
Gabriel Dutra
264fb5798d Merge branch 'master' into query-based-dropdown--parameters 2020-02-21 13:31:49 -03:00
Gabriel Dutra
90023ac435 Make sure Table updates correctly 2020-02-21 11:03:37 -03:00
Gabriel Dutra
df755fbc17 Add try except for NoResultFound 2020-02-21 09:40:52 -03:00
Gabriel Dutra
e555642844 Add is_safe check for parameterized query based 2020-02-21 09:27:12 -03:00
Gabriel Dutra
bdd7b146ae Change stored mapping attributes 2020-02-20 21:59:19 -03:00
Gabriel Dutra
b7478defec Don't validade query params with params 2020-02-20 19:29:43 -03:00
Gabriel Dutra
bb0d7830c9 Fixes + temp remove validation for Query param 2020-02-20 18:49:29 -03:00
Gabriel Dutra
137aa22dd4 Parameter Mapping UI (2/2) 2020-02-18 17:55:27 -03:00
Gabriel Dutra
9cf396599a Parameter Mapping UI (1/2) 2020-02-17 23:32:10 -03:00
Gabriel Dutra
b70f0fa921 Iterate over backend verification 2020-02-16 13:39:05 -03:00
Gabriel Dutra
5e3613d6cb Start experiements with a 'search' parameter 2020-02-13 16:29:50 -03:00
443 changed files with 4296 additions and 11018 deletions

View File

@@ -1,129 +1,5 @@
# Change Log # Change Log
## v10.0.0-beta - 2021-06-16
Just over a year since our last release, the V10 beta is ready. Since we never made a non-beta release of V9, we expect many users will upgrade directly from V8 -> V10. This will bring a lot of exciting features. Please check out the V9 beta release notes below to learn more.
This V10 beta incorporates fixes for the feedback we received on the V9 beta along with a few long-requested features (horizontal bar charts!) and other changes to improve UX and reliability.
This release was made possible by contributions from 35+ people (the Github API didn't let us pull handles this time around): Alex Kovar, Alexander Rusanov, Arik Fraimovich, Ben Amor, Christopher Grant, Đặng Minh Dũng, Daniel Lang, deecay, Elad Ossadon, Gabriel Dutra, iwakiriK, Jannis Leidel, Jerry, Jesse Whitehouse, Jiajie Zhong, Jim Sparkman, Jonathan Hult, Josh Bohde, Justin Talbot, koooge, Lei Ni, Levko Kravets, Lingkai Kong, max-voronov, Mike Nason, Nolan Nichols, Omer Lachish, Patrick Yang, peterlee, Rafael Wendel, Sebastian Tramp, simonschneider-db, Tim Gates, Tobias Macey, Vipul Mathur, and Vladislav Denisov
Our special thanks to [Sohail Ahmed](https://pk.linkedin.com/in/sohail-ahmed-755776184) for reporting a vulnerability in our "forgot password" page (#5425)
### Upgrading
(This section is duplicated from the previous release - since many users will upgrade directly from V8 -> V10)
Typically, if you are running your own instance of Redash and wish to upgrade, you would simply modify the Docker tag in your `docker-compose.yml` file. Since RQ has replaced Celery in this version, there are a couple extra modifications that need to be done in your `docker-compose.yml`:
1. Under `services/scheduler/environment`, omit `QUEUES` and `WORKERS_COUNT` (and omit `environment` altogether if it is empty).
2. Under `services`, add a new service for general RQ jobs:
```yaml
worker:
<<: *redash-service
command: worker
environment:
QUEUES: "periodic emails default"
WORKERS_COUNT: 1
```
Following that, force a recreation of your containers with `docker-compose up --force-recreate --build` and you should be good to go.
### UX
- Redash now uses a vertical navbar
- Dashboard list now includes “My Dashboards” filter
- Dashboard parameters can now be re-ordered
- Queries can now be executed with Shift + Enter on all platforms.
- Added New Dashboard/Query/Alert buttons to corresponding list pages
- Dashboard text widgets now prompt to confirm before closing the text editor
- A plus sign is now shown between tags used for search
- On the queries list view “My Queries” has moved above “Archived”
- Improved behavior for filtering by tags in list views
- When a users session expires for inactivity, they are prompted to log-in with a pop-up so they dont lose their place in the app
- Numerous accessibility changes towards the a11y standard
- Hide the “Create” menu button if current user doesnt have permission to any data sources
### Visualizations
- Feature: Added support for horizontal box plots
- Feature: Added support for horizontal bar charts
- Feature: Added “Reverse” option for Chart visualization legend
- Feature: Added option to align Chart Y-axes at zero
- Feature: The table visualization header is now fixed when scrolling
- Feature: Added USA map to choropleth visualization
- Fix: Selected filters were reset when switching visualizations
- Fix: Stacked bar chart showed the wrong Y-axis range in some cases
- Fix: Bar chart with second y axis overlapped data series
- Fix: Y-axis autoscale failed when min or max was set
- Fix: Custom JS visualization was broken because of a typo
- Fix: Too large visualization caused filters block to collapse
- Fix: Sankey visualization looked inconsistent if the data source returned VARCHAR instead of numeric types
### Structural Updates
- Redash now prevents CSRF attacks
- Migration to TypeScript
- Upgrade to Antd version 4
### Data Sources
- New Data Sources: SPARQL Endpoint, Eccenca Corporate Memory, TrinoDB
- Databricks
- Custom Schema Browser that allows switching between databases
- Option added to truncate large results
- Support for multiple-statement queries
- Schema browser can now use eventlet instead of RQ
- MongoDB:
- Moved Username and Password out of the connection string so that password can be stored secretly
- Oracle:
- Fix: Annotated queries always failed. Annotation is now disabled
- Postgres/CockroachDB:
- SSL certfile/keyfile fields are now handled as secret
- Python:
- Feature: Custom built-ins are now supported
- Fix: Query runner was not compatible with Python 3
- Snowflake:
- Data source now accepts a custom host address (for use with proxies)
- TreasureData:
- API key field is now handled as secret
- Yandex:
- OAuth token field is now handled as secret
### Alerts
- Feature: Added ability to mute alerts without deleting them
- Change: Non-email alert destination details are now obfuscated to avoid leaking sensitive information (webhook URLs, tokens etc.)
- Fix: numerical comparisons failed if value from query was a string
### Parameters
- Added “Last 12 months” option for dynamic date ranges
### Bug Fixes
- Fix: Private addresses were not allowed even when enforcing was disabled
- Fix: Python query runner wasnt updated for Python 3
- Fix: Sorting queries by schedule returned the wrong order
- Fix: Counter visualization was enormous in some cases
- Fix: Dashboard URL will now change when the dashboard title changes
- Fix: URL parameters were removed when forking a query
- Fix: Create link on data sources page was broken
- Fix: Queries could be reassigned to read-only data sources
- Fix: Multi-select dropdown was very slow if there were 1k+ options
- Fix: Search Input couldnt be focused or updated while editing a dashboard
- Fix: The CLI command for “status” did not work
- Fix: The dashboard list screen displayed too few items under certain pagination configurations
### Other
- Added an environment variable to disable public sharing links for queries and dashboards
- Alert destinations are now encrypted at the database
- The base query runner now has stubs to implement result truncating for other data sources
- Static SAML configuration and assertion encryption are now supported
- Adds new component for adding extra actions to the query and dashboard pages
- Non-admins with at least view_only permission on a dashboard can now make GET requests to the data source resource
- Added a BLOCKED_DOMAINS setting to prevent sign-ups from emails at specific domains
- Added a rate limit to the “forgot password” page
- RQ workers will now shutdown gracefully for known error codes
- Scheduled execution failure counter now resets following a successful ad hoc execution
- Redash now deletes locks for cancelled queries
- Upgraded Ace Editor from v6 to v9
- Added a periodic job to remove ghost locks
- Removed content width limit on all pages
- Introduce a <Link> React component
## v9.0.0-beta - 2020-06-11 ## v9.0.0-beta - 2020-06-11
This release was long time in the making and has several major changes: This release was long time in the making and has several major changes:

View File

@@ -22,8 +22,7 @@ RUN if [ "x$skip_frontend_build" = "x" ] ; then npm ci --unsafe-perm; fi
COPY --chown=redash client /frontend/client COPY --chown=redash client /frontend/client
COPY --chown=redash webpack.config.js /frontend/ COPY --chown=redash webpack.config.js /frontend/
RUN if [ "x$skip_frontend_build" = "x" ] ; then npm run build; else mkdir -p /frontend/client/dist && touch /frontend/client/dist/multi_org.html && touch /frontend/client/dist/index.html; fi RUN if [ "x$skip_frontend_build" = "x" ] ; then npm run build; else mkdir -p /frontend/client/dist && touch /frontend/client/dist/multi_org.html && touch /frontend/client/dist/index.html; fi
FROM python:3.7-slim
FROM python:3.7-slim-buster
EXPOSE 5000 EXPOSE 5000
@@ -67,9 +66,8 @@ RUN apt-get update && \
rm -rf /var/lib/apt/lists/* rm -rf /var/lib/apt/lists/*
ARG databricks_odbc_driver_url=https://databricks.com/wp-content/uploads/2.6.10.1010-2/SimbaSparkODBC-2.6.10.1010-2-Debian-64bit.zip ARG databricks_odbc_driver_url=https://databricks.com/wp-content/uploads/2.6.10.1010-2/SimbaSparkODBC-2.6.10.1010-2-Debian-64bit.zip
RUN wget --quiet $databricks_odbc_driver_url -O /tmp/simba_odbc.zip \ ADD $databricks_odbc_driver_url /tmp/simba_odbc.zip
&& chmod 600 /tmp/simba_odbc.zip \ RUN unzip /tmp/simba_odbc.zip -d /tmp/ \
&& unzip /tmp/simba_odbc.zip -d /tmp/ \
&& dpkg -i /tmp/SimbaSparkODBC-*/*.deb \ && dpkg -i /tmp/SimbaSparkODBC-*/*.deb \
&& echo "[Simba]\nDriver = /opt/simba/spark/lib/64/libsparkodbc_sb64.so" >> /etc/odbcinst.ini \ && echo "[Simba]\nDriver = /opt/simba/spark/lib/64/libsparkodbc_sb64.so" >> /etc/odbcinst.ini \
&& rm /tmp/simba_odbc.zip \ && rm /tmp/simba_odbc.zip \
@@ -81,19 +79,12 @@ WORKDIR /app
ENV PIP_DISABLE_PIP_VERSION_CHECK=1 ENV PIP_DISABLE_PIP_VERSION_CHECK=1
ENV PIP_NO_CACHE_DIR=1 ENV PIP_NO_CACHE_DIR=1
# rollback pip version to avoid legacy resolver problem # We first copy only the requirements file, to avoid rebuilding on every file
RUN pip install pip==20.2.4; # change.
COPY requirements.txt requirements_bundles.txt requirements_dev.txt requirements_all_ds.txt ./
# We first copy only the requirements file, to avoid rebuilding on every file change. RUN if [ "x$skip_dev_deps" = "x" ] ; then pip install -r requirements.txt -r requirements_dev.txt; else pip install -r requirements.txt; fi
COPY requirements_all_ds.txt ./
RUN if [ "x$skip_ds_deps" = "x" ] ; then pip install -r requirements_all_ds.txt ; else echo "Skipping pip install -r requirements_all_ds.txt" ; fi RUN if [ "x$skip_ds_deps" = "x" ] ; then pip install -r requirements_all_ds.txt ; else echo "Skipping pip install -r requirements_all_ds.txt" ; fi
COPY requirements_bundles.txt requirements_dev.txt ./
RUN if [ "x$skip_dev_deps" = "x" ] ; then pip install -r requirements_dev.txt ; fi
COPY requirements.txt ./
RUN pip install -r requirements.txt
COPY . /app COPY . /app
COPY --from=frontend-builder /frontend/client/dist /app/client/dist COPY --from=frontend-builder /frontend/client/dist /app/client/dist
RUN chown -R redash /app RUN chown -R redash /app

View File

@@ -73,7 +73,6 @@ Redash supports more than 35 SQL and NoSQL [data sources](https://redash.io/help
- Shell Scripts - Shell Scripts
- Snowflake - Snowflake
- SQLite - SQLite
- TiDB
- TreasureData - TreasureData
- Vertica - Vertica
- Yandex AppMetrrica - Yandex AppMetrrica

View File

@@ -5,11 +5,10 @@ module.exports = {
"react-app", "react-app",
"plugin:compat/recommended", "plugin:compat/recommended",
"prettier", "prettier",
"plugin:jsx-a11y/recommended",
// Remove any typescript-eslint rules that would conflict with prettier // Remove any typescript-eslint rules that would conflict with prettier
"prettier/@typescript-eslint", "prettier/@typescript-eslint",
], ],
plugins: ["jest", "compat", "no-only-tests", "@typescript-eslint", "jsx-a11y"], plugins: ["jest", "compat", "no-only-tests", "@typescript-eslint"],
settings: { settings: {
"import/resolver": "webpack", "import/resolver": "webpack",
}, },
@@ -20,20 +19,7 @@ module.exports = {
rules: { rules: {
// allow debugger during development // allow debugger during development
"no-debugger": process.env.NODE_ENV === "production" ? 2 : 0, "no-debugger": process.env.NODE_ENV === "production" ? 2 : 0,
"jsx-a11y/anchor-is-valid": [ "jsx-a11y/anchor-is-valid": "off",
// TMP
"off",
{
components: ["Link"],
aspects: ["noHref", "invalidHref", "preferButton"],
},
],
"jsx-a11y/no-redundant-roles": "error",
"jsx-a11y/no-autofocus": "off",
"jsx-a11y/click-events-have-key-events": "off", // TMP
"jsx-a11y/no-static-element-interactions": "off", // TMP
"jsx-a11y/no-noninteractive-element-interactions": "off", // TMP
"no-console": ["warn", { allow: ["warn", "error"] }],
"no-restricted-imports": [ "no-restricted-imports": [
"error", "error",
{ {

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.6 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 31 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 23 KiB

View File

@@ -225,16 +225,6 @@
} }
} }
&-tbody > tr&-row {
&:hover,
&:focus,
&:focus-within {
& > td {
background: @table-row-hover-bg;
}
}
}
// Custom styles // Custom styles
&-headerless &-tbody > tr:first-child > td { &-headerless &-tbody > tr:first-child > td {
@@ -401,18 +391,6 @@
left: 0; left: 0;
} }
} }
&:focus,
&:focus-within {
color: @menu-highlight-color;
}
}
}
.@{dropdown-prefix-cls}-menu-item {
&:focus,
&:focus-within {
background-color: @item-hover-bg;
} }
} }

View File

@@ -98,10 +98,6 @@ strong {
.clickable { .clickable {
cursor: pointer; cursor: pointer;
button&:disabled {
cursor: not-allowed;
}
} }
.resize-vertical { .resize-vertical {

View File

@@ -1,23 +1,26 @@
.edit-in-place { .edit-in-place span {
white-space: pre-line; white-space: pre-line;
display: inline-block;
p { p {
margin-bottom: 0; margin-bottom: 0;
} }
}
.editable { .edit-in-place span.editable {
display: inline-block; display: inline-block;
cursor: pointer; cursor: pointer;
}
&:hover { .edit-in-place span.editable:hover {
background: @redash-yellow; background: @redash-yellow;
border-radius: @redash-radius; border-radius: @redash-radius;
} }
}
&.active input, .edit-in-place.active input,
&.active textarea { .edit-in-place.active textarea {
display: inline-block; display: inline-block;
} }
.edit-in-place {
display: inline-block;
} }

View File

@@ -62,6 +62,7 @@
.padding(25, 0px, 0); .padding(25, 0px, 0);
/* -------------------------------------------------------- /* --------------------------------------------------------
Generate Font-Size Classes (8px - 20px) Generate Font-Size Classes (8px - 20px)
-----------------------------------------------------------*/ -----------------------------------------------------------*/
@@ -75,78 +76,47 @@
.font-size(20, 8px, 8); .font-size(20, 8px, 8);
.f-inherit { .f-inherit { font-size: inherit !important; }
font-size: inherit !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Font Weight Font Weight
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.f-300 { .f-300 { font-weight: 300 !important; }
font-weight: 300 !important; .f-400 { font-weight: 400 !important; }
} .f-500 { font-weight: 500 !important; }
.f-400 { .f-700 { font-weight: 700 !important; }
font-weight: 400 !important;
}
.f-500 {
font-weight: 500 !important;
}
.f-700 {
font-weight: 700 !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Position Position
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.p-relative { .p-relative { position: relative !important; }
position: relative !important; .p-absolute { position: absolute !important; }
} .p-fixed { position: fixed !important; }
.p-absolute { .p-static { position: static !important; }
position: absolute !important;
}
.p-fixed {
position: fixed !important;
}
.p-static {
position: static !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Overflow Overflow
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.o-hidden { .o-hidden { overflow: hidden !important; }
overflow: hidden !important; .o-visible { overflow: visible !important; }
} .o-auto { overflow: auto !important; }
.o-visible {
overflow: visible !important;
}
.o-auto {
overflow: auto !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Display Display
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.di-block { .di-block { display: inline-block !important; }
display: inline-block !important; .d-block { display: block; }
}
.d-block {
display: block;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Background Colors and Colors Background Colors and Colors
-----------------------------------------------------------*/ -----------------------------------------------------------*/
@array: c-white bg-white @white, c-ace bg-ace @ace, c-black bg-black @black, c-brown bg-brown @brown, @array: c-white bg-white @white, c-ace bg-ace @ace, c-black bg-black @black, c-brown bg-brown @brown, c-pink bg-pink @pink, c-red bg-red @red, c-blue bg-blue @blue, c-purple bg-purple @purple, c-deeppurple bg-deeppurple @deeppurple, c-lightblue bg-lightblue @lightblue, c-cyan bg-cyan @cyan, c-teal bg-teal @teal, c-green bg-green @green, c-lightgreen bg-lightgreen @lightgreen, c-lime bg-lime @lime, c-yellow bg-yellow @yellow, c-amber bg-amber @amber, c-orange bg-orange @orange, c-deeporange bg-deeporange @deeporange, c-gray bg-gray @gray, c-bluegray bg-bluegray @bluegray, c-indigo bg-indigo @indigo;
c-pink bg-pink @pink, c-red bg-red @red, c-blue bg-blue @blue, c-purple bg-purple @purple,
c-deeppurple bg-deeppurple @deeppurple, c-lightblue bg-lightblue @lightblue, c-cyan bg-cyan @cyan,
c-teal bg-teal @teal, c-green bg-green @green, c-lightgreen bg-lightgreen @lightgreen, c-lime bg-lime @lime,
c-yellow bg-yellow @yellow, c-amber bg-amber @amber, c-orange bg-orange @orange,
c-deeporange bg-deeporange @deeporange, c-gray bg-gray @gray, c-bluegray bg-bluegray @bluegray,
c-indigo bg-indigo @indigo;
.for(@array); .for(@array); .-each(@value) {
.-each(@value) {
@name: extract(@value, 1); @name: extract(@value, 1);
@name2: extract(@value, 2); @name2: extract(@value, 2);
@color: extract(@value, 3); @color: extract(@value, 3);
@@ -159,61 +129,36 @@
} }
} }
/* -------------------------------------------------------- /* --------------------------------------------------------
Background Colors Background Colors
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.bg-brand { .bg-brand { background-color: @brand-bg; }
background-color: @brand-bg; .bg-black-trp { background-color: rgba(0,0,0,0.12) !important; }
}
.bg-black-trp {
background-color: rgba(0, 0, 0, 0.12) !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Borders Borders
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.b-0 { .b-0 { border: 0 !important; }
border: 0 !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Width Width
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.w-100 { .w-100 { width: 100% !important; }
width: 100% !important; .w-50 { width: 50% !important; }
} .w-25 { width: 25% !important; }
.w-50 {
width: 50% !important;
}
.w-25 {
width: 25% !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Border Radius Border Radius
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.brd-2 { .brd-2 { border-radius: 2px; }
border-radius: 2px;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Alignment Alignment
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.va-top { .va-top { vertical-align: top; }
vertical-align: top;
}
/* --------------------------------------------------------
Screen readers
-----------------------------------------------------------*/
.sr-only {
position: absolute;
width: 1px;
height: 1px;
padding: 0;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
white-space: nowrap;
border-width: 0;
}

View File

@@ -1,7 +1,31 @@
div.table-name {
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
cursor: pointer;
padding: 2px 22px 2px 10px;
border-radius: @redash-radius;
position: relative;
height: 22px;
.copy-to-editor {
display: none;
}
&:hover {
background: fade(@redash-gray, 10%);
.copy-to-editor {
display: flex;
}
}
}
.schema-container { .schema-container {
height: 100%; height: 100%;
z-index: 10; z-index: 10;
background-color: white; background-color: white;
}
.schema-browser { .schema-browser {
overflow: hidden; overflow: hidden;
@@ -22,54 +46,25 @@
} }
.copy-to-editor { .copy-to-editor {
visibility: hidden;
color: fade(@redash-gray, 90%); color: fade(@redash-gray, 90%);
cursor: pointer;
position: absolute;
top: 0;
right: 0;
bottom: 0;
width: 20px; width: 20px;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
transition: none;
}
.schema-list-item {
display: flex;
border-radius: @redash-radius;
height: 22px;
.table-name {
flex: 1;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
cursor: pointer;
padding: 2px 22px 2px 10px;
}
&:hover,
&:focus,
&:focus-within {
background: fade(@redash-gray, 10%);
.copy-to-editor {
visibility: visible;
}
}
} }
.table-open { .table-open {
.table-open-item { padding: 0 22px 0 26px;
display: flex;
height: 18px;
width: calc(100% - 22px);
padding-left: 22px;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
white-space: nowrap; white-space: nowrap;
transition: none; position: relative;
height: 18px;
div:first-child {
flex: 1;
}
.column-type { .column-type {
color: fade(@text-color, 80%); color: fade(@text-color, 80%);
@@ -78,14 +73,15 @@
text-transform: uppercase; text-transform: uppercase;
} }
&:hover, .copy-to-editor {
&:focus, display: none;
&:focus-within { }
&:hover {
background: fade(@redash-gray, 10%); background: fade(@redash-gray, 10%);
.copy-to-editor { .copy-to-editor {
visibility: visible; display: flex;
}
} }
} }
} }
@@ -104,4 +100,3 @@
.parameter-label { .parameter-label {
display: block; display: block;
} }
}

View File

@@ -103,7 +103,7 @@
padding-top: 5px !important; padding-top: 5px !important;
} }
.btn-favorite, .btn-favourite,
.btn-archive { .btn-archive {
font-size: 15px; font-size: 15px;
} }
@@ -114,23 +114,18 @@
line-height: 1.7 !important; line-height: 1.7 !important;
} }
.btn-favorite { .btn-favourite {
color: #d4d4d4; color: #d4d4d4;
transition: all 0.25s ease-in-out; transition: all 0.25s ease-in-out;
.fa-star {
color: @yellow-darker;
}
&:hover, &:hover,
&:focus { &:focus {
color: @yellow-darker; color: @yellow-darker;
cursor: pointer; cursor: pointer;
}
.fa-star { .fa-star {
filter: saturate(75%); color: @yellow-darker;
opacity: 0.75;
}
} }
} }

View File

@@ -127,13 +127,11 @@ body.fixed-layout {
} }
} }
.label-tag { a.label-tag {
background: fade(@redash-gray, 15%); background: fade(@redash-gray, 15%);
color: darken(@redash-gray, 15%); color: darken(@redash-gray, 15%);
&:hover, &:hover {
&:focus,
&:active {
color: darken(@redash-gray, 15%); color: darken(@redash-gray, 15%);
background: fade(@redash-gray, 25%); background: fade(@redash-gray, 25%);
} }

View File

@@ -1,11 +1,10 @@
import React, { useMemo } from "react"; import { first } from "lodash";
import { first, includes } from "lodash"; import React, { useState } from "react";
import Button from "antd/lib/button";
import Menu from "antd/lib/menu"; import Menu from "antd/lib/menu";
import Link from "@/components/Link"; import Link from "@/components/Link";
import PlainButton from "@/components/PlainButton";
import HelpTrigger from "@/components/HelpTrigger"; import HelpTrigger from "@/components/HelpTrigger";
import CreateDashboardDialog from "@/components/dashboards/CreateDashboardDialog"; import CreateDashboardDialog from "@/components/dashboards/CreateDashboardDialog";
import { useCurrentRoute } from "@/components/ApplicationArea/Router";
import { Auth, currentUser } from "@/services/auth"; import { Auth, currentUser } from "@/services/auth";
import settingsMenu from "@/services/settingsMenu"; import settingsMenu from "@/services/settingsMenu";
import logoUrl from "@/assets/images/redash_icon_small.png"; import logoUrl from "@/assets/images/redash_icon_small.png";
@@ -16,109 +15,83 @@ import AlertOutlinedIcon from "@ant-design/icons/AlertOutlined";
import PlusOutlinedIcon from "@ant-design/icons/PlusOutlined"; import PlusOutlinedIcon from "@ant-design/icons/PlusOutlined";
import QuestionCircleOutlinedIcon from "@ant-design/icons/QuestionCircleOutlined"; import QuestionCircleOutlinedIcon from "@ant-design/icons/QuestionCircleOutlined";
import SettingOutlinedIcon from "@ant-design/icons/SettingOutlined"; import SettingOutlinedIcon from "@ant-design/icons/SettingOutlined";
import VersionInfo from "./VersionInfo"; import MenuUnfoldOutlinedIcon from "@ant-design/icons/MenuUnfoldOutlined";
import MenuFoldOutlinedIcon from "@ant-design/icons/MenuFoldOutlined";
import VersionInfo from "./VersionInfo";
import "./DesktopNavbar.less"; import "./DesktopNavbar.less";
function NavbarSection({ children, ...props }) { function NavbarSection({ inlineCollapsed, children, ...props }) {
return ( return (
<Menu selectable={false} mode="vertical" theme="dark" {...props}> <Menu
selectable={false}
mode={inlineCollapsed ? "inline" : "vertical"}
inlineCollapsed={inlineCollapsed}
theme="dark"
{...props}>
{children} {children}
</Menu> </Menu>
); );
} }
function useNavbarActiveState() {
const currentRoute = useCurrentRoute();
return useMemo(
() => ({
dashboards: includes(
[
"Dashboards.List",
"Dashboards.Favorites",
"Dashboards.My",
"Dashboards.ViewOrEdit",
"Dashboards.LegacyViewOrEdit",
],
currentRoute.id
),
queries: includes(
[
"Queries.List",
"Queries.Favorites",
"Queries.Archived",
"Queries.My",
"Queries.View",
"Queries.New",
"Queries.Edit",
],
currentRoute.id
),
dataSources: includes(["DataSources.List"], currentRoute.id),
alerts: includes(["Alerts.List", "Alerts.New", "Alerts.View", "Alerts.Edit"], currentRoute.id),
}),
[currentRoute.id]
);
}
export default function DesktopNavbar() { export default function DesktopNavbar() {
const firstSettingsTab = first(settingsMenu.getAvailableItems()); const [collapsed, setCollapsed] = useState(true);
const activeState = useNavbarActiveState(); const firstSettingsTab = first(settingsMenu.getAvailableItems());
const canCreateQuery = currentUser.hasPermission("create_query"); const canCreateQuery = currentUser.hasPermission("create_query");
const canCreateDashboard = currentUser.hasPermission("create_dashboard"); const canCreateDashboard = currentUser.hasPermission("create_dashboard");
const canCreateAlert = currentUser.hasPermission("list_alerts"); const canCreateAlert = currentUser.hasPermission("list_alerts");
return ( return (
<nav className="desktop-navbar"> <div className="desktop-navbar">
<NavbarSection className="desktop-navbar-logo"> <NavbarSection inlineCollapsed={collapsed} className="desktop-navbar-logo">
<div role="menuitem"> <div>
<Link href="./"> <Link href="./">
<img src={logoUrl} alt="Redash" /> <img src={logoUrl} alt="Redash" />
</Link> </Link>
</div> </div>
</NavbarSection> </NavbarSection>
<NavbarSection> <NavbarSection inlineCollapsed={collapsed}>
{currentUser.hasPermission("list_dashboards") && ( {currentUser.hasPermission("list_dashboards") && (
<Menu.Item key="dashboards" className={activeState.dashboards ? "navbar-active-item" : null}> <Menu.Item key="dashboards">
<Link href="dashboards"> <Link href="dashboards">
<DesktopOutlinedIcon aria-label="Dashboard navigation button" /> <DesktopOutlinedIcon />
<span className="desktop-navbar-label">Dashboards</span> <span>Dashboards</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
{currentUser.hasPermission("view_query") && ( {currentUser.hasPermission("view_query") && (
<Menu.Item key="queries" className={activeState.queries ? "navbar-active-item" : null}> <Menu.Item key="queries">
<Link href="queries"> <Link href="queries">
<CodeOutlinedIcon aria-label="Queries navigation button" /> <CodeOutlinedIcon />
<span className="desktop-navbar-label">Queries</span> <span>Queries</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
{currentUser.hasPermission("list_alerts") && ( {currentUser.hasPermission("list_alerts") && (
<Menu.Item key="alerts" className={activeState.alerts ? "navbar-active-item" : null}> <Menu.Item key="alerts">
<Link href="alerts"> <Link href="alerts">
<AlertOutlinedIcon aria-label="Alerts navigation button" /> <AlertOutlinedIcon />
<span className="desktop-navbar-label">Alerts</span> <span>Alerts</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
</NavbarSection> </NavbarSection>
<NavbarSection className="desktop-navbar-spacer"> <NavbarSection inlineCollapsed={collapsed} className="desktop-navbar-spacer">
{(canCreateQuery || canCreateDashboard || canCreateAlert) && <Menu.Divider />}
{(canCreateQuery || canCreateDashboard || canCreateAlert) && ( {(canCreateQuery || canCreateDashboard || canCreateAlert) && (
<Menu.SubMenu <Menu.SubMenu
key="create" key="create"
popupClassName="desktop-navbar-submenu" popupClassName="desktop-navbar-submenu"
data-test="CreateButton"
tabIndex={0}
title={ title={
<React.Fragment> <React.Fragment>
<span data-test="CreateButton">
<PlusOutlinedIcon /> <PlusOutlinedIcon />
<span className="desktop-navbar-label">Create</span> <span>Create</span>
</span>
</React.Fragment> </React.Fragment>
}> }>
{canCreateQuery && ( {canCreateQuery && (
@@ -130,9 +103,9 @@ export default function DesktopNavbar() {
)} )}
{canCreateDashboard && ( {canCreateDashboard && (
<Menu.Item key="new-dashboard"> <Menu.Item key="new-dashboard">
<PlainButton data-test="CreateDashboardMenuItem" onClick={() => CreateDashboardDialog.showModal()}> <a data-test="CreateDashboardMenuItem" onMouseUp={() => CreateDashboardDialog.showModal()}>
New Dashboard New Dashboard
</PlainButton> </a>
</Menu.Item> </Menu.Item>
)} )}
{canCreateAlert && ( {canCreateAlert && (
@@ -146,31 +119,32 @@ export default function DesktopNavbar() {
)} )}
</NavbarSection> </NavbarSection>
<NavbarSection> <NavbarSection inlineCollapsed={collapsed}>
<Menu.Item key="help"> <Menu.Item key="help">
<HelpTrigger showTooltip={false} type="HOME" tabIndex={0}> <HelpTrigger showTooltip={false} type="HOME">
<QuestionCircleOutlinedIcon /> <QuestionCircleOutlinedIcon />
<span className="desktop-navbar-label">Help</span> <span>Help</span>
</HelpTrigger> </HelpTrigger>
</Menu.Item> </Menu.Item>
{firstSettingsTab && ( {firstSettingsTab && (
<Menu.Item key="settings" className={activeState.dataSources ? "navbar-active-item" : null}> <Menu.Item key="settings">
<Link href={firstSettingsTab.path} data-test="SettingsLink"> <Link href={firstSettingsTab.path} data-test="SettingsLink">
<SettingOutlinedIcon /> <SettingOutlinedIcon />
<span className="desktop-navbar-label">Settings</span> <span>Settings</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
<Menu.Divider />
</NavbarSection> </NavbarSection>
<NavbarSection className="desktop-navbar-profile-menu"> <NavbarSection inlineCollapsed={collapsed} className="desktop-navbar-profile-menu">
<Menu.SubMenu <Menu.SubMenu
key="profile" key="profile"
popupClassName="desktop-navbar-submenu" popupClassName="desktop-navbar-submenu"
tabIndex={0}
title={ title={
<span data-test="ProfileDropdown" className="desktop-navbar-profile-menu-title"> <span data-test="ProfileDropdown" className="desktop-navbar-profile-menu-title">
<img className="profile__image_thumb" src={currentUser.profile_image_url} alt={currentUser.name} /> <img className="profile__image_thumb" src={currentUser.profile_image_url} alt={currentUser.name} />
<span>{currentUser.name}</span>
</span> </span>
}> }>
<Menu.Item key="profile"> <Menu.Item key="profile">
@@ -183,16 +157,20 @@ export default function DesktopNavbar() {
)} )}
<Menu.Divider /> <Menu.Divider />
<Menu.Item key="logout"> <Menu.Item key="logout">
<PlainButton data-test="LogOutButton" onClick={() => Auth.logout()}> <a data-test="LogOutButton" onClick={() => Auth.logout()}>
Log out Log out
</PlainButton> </a>
</Menu.Item> </Menu.Item>
<Menu.Divider /> <Menu.Divider />
<Menu.Item key="version" role="presentation" disabled className="version-info"> <Menu.Item key="version" disabled className="version-info">
<VersionInfo /> <VersionInfo />
</Menu.Item> </Menu.Item>
</Menu.SubMenu> </Menu.SubMenu>
</NavbarSection> </NavbarSection>
</nav>
<Button onClick={() => setCollapsed(!collapsed)} className="desktop-navbar-collapse-button">
{collapsed ? <MenuUnfoldOutlinedIcon /> : <MenuFoldOutlinedIcon />}
</Button>
</div>
); );
} }

View File

@@ -1,17 +1,12 @@
@backgroundColor: #001529; @backgroundColor: #001529;
@dividerColor: rgba(255, 255, 255, 0.5); @dividerColor: rgba(255, 255, 255, 0.5);
@textColor: rgba(255, 255, 255, 0.75); @textColor: rgba(255, 255, 255, 0.75);
@brandColor: #ff7964; // Redash logo color
@activeItemColor: @brandColor;
@iconSize: 26px;
.desktop-navbar { .desktop-navbar {
background: @backgroundColor; background: @backgroundColor;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
height: 100%; height: 100%;
width: 80px;
overflow: hidden;
&-spacer { &-spacer {
flex: 1 1 auto; flex: 1 1 auto;
@@ -26,6 +21,12 @@
height: 40px; height: 40px;
transition: all 270ms; transition: all 270ms;
} }
&.ant-menu-inline-collapsed {
img {
height: 20px;
}
}
} }
.help-trigger { .help-trigger {
@@ -33,38 +34,33 @@
} }
.ant-menu { .ant-menu {
&:not(.ant-menu-inline-collapsed) {
width: 170px;
}
&.ant-menu-inline-collapsed > .ant-menu-submenu-title span img + span,
&.ant-menu-inline-collapsed > .ant-menu-item i + span {
display: inline-block;
max-width: 0;
opacity: 0;
}
.ant-menu-item-divider {
background: @dividerColor;
}
.ant-menu-item, .ant-menu-item,
.ant-menu-submenu { .ant-menu-submenu {
font-weight: 500; font-weight: 500;
color: @textColor; color: @textColor;
&.navbar-active-item {
box-shadow: inset 3px 0 0 @activeItemColor;
.anticon {
color: @activeItemColor;
}
}
&.ant-menu-submenu-open, &.ant-menu-submenu-open,
&.ant-menu-submenu-active, &.ant-menu-submenu-active,
&:hover, &:hover,
&:active, &:active {
&:focus,
&:focus-within {
color: #fff; color: #fff;
} }
.anticon {
font-size: @iconSize;
margin: 0;
}
.desktop-navbar-label {
margin-top: 4px;
font-size: 11px;
}
a, a,
span, span,
.anticon { .anticon {
@@ -75,33 +71,21 @@
.ant-menu-submenu-arrow { .ant-menu-submenu-arrow {
display: none; display: none;
} }
.ant-menu-item,
.ant-menu-submenu {
padding: 0;
height: 60px;
display: flex;
align-items: center;
flex-direction: column;
justify-content: center;
} }
.ant-menu-submenu-title { .ant-btn.desktop-navbar-collapse-button {
width: 100%; background-color: @backgroundColor;
padding: 0; border: 0;
border-radius: 0;
color: @textColor;
&:hover,
&:active {
color: #fff;
} }
a, &:after {
&.ant-menu-vertical > .ant-menu-submenu > .ant-menu-submenu-title, animation: 0s !important;
.ant-menu-submenu-title {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
line-height: normal;
height: auto;
background: none;
color: inherit;
} }
} }
@@ -115,8 +99,37 @@
.profile__image_thumb { .profile__image_thumb {
margin: 0; margin: 0;
vertical-align: middle; vertical-align: middle;
width: @iconSize; }
height: @iconSize;
.profile__image_thumb + span {
flex: 1 1 auto;
overflow: hidden;
white-space: nowrap;
text-overflow: ellipsis;
margin-left: 10px;
vertical-align: middle;
display: inline-block;
// styles from Antd
opacity: 1;
transition: opacity 0.3s cubic-bezier(0.645, 0.045, 0.355, 1),
margin-left 0.3s cubic-bezier(0.645, 0.045, 0.355, 1), width 0.3s cubic-bezier(0.645, 0.045, 0.355, 1);
}
}
&.ant-menu-inline-collapsed {
.ant-menu-submenu-title {
padding-left: 16px !important;
padding-right: 16px !important;
}
.desktop-navbar-profile-menu-title {
.profile__image_thumb + span {
opacity: 0;
max-width: 0;
margin-left: 0;
}
} }
} }
} }
@@ -133,9 +146,7 @@
color: @textColor; color: @textColor;
&:hover, &:hover,
&:active, &:active {
&:focus,
&:focus-within {
color: #fff; color: #fff;
} }
@@ -160,9 +171,7 @@
color: rgba(255, 255, 255, 0.8); color: rgba(255, 255, 255, 0.8);
&:hover, &:hover,
&:active, &:active {
&:focus,
&:focus-within {
color: rgba(255, 255, 255, 1); color: rgba(255, 255, 255, 1);
} }
} }

View File

@@ -14,8 +14,8 @@ export default function VersionInfo() {
<div className="m-t-10"> <div className="m-t-10">
{/* eslint-disable react/jsx-no-target-blank */} {/* eslint-disable react/jsx-no-target-blank */}
<Link href="https://version.redash.io/" className="update-available" target="_blank" rel="noopener"> <Link href="https://version.redash.io/" className="update-available" target="_blank" rel="noopener">
Update Available <i className="fa fa-external-link m-l-5" aria-hidden="true" /> Update Available
<span className="sr-only">(opens in a new tab)</span> <i className="fa fa-external-link m-l-5" />
</Link> </Link>
</div> </div>
)} )}

View File

@@ -49,7 +49,7 @@ export default function ErrorMessage({ error, message }) {
<div className="error-message-container" data-test="ErrorMessage" role="alert"> <div className="error-message-container" data-test="ErrorMessage" role="alert">
<div className="error-state bg-white tiled"> <div className="error-state bg-white tiled">
<div className="error-state__icon"> <div className="error-state__icon">
<i className="zmdi zmdi-alert-circle-o" aria-hidden="true" /> <i className="zmdi zmdi-alert-circle-o" />
</div> </div>
<div className="error-state__details"> <div className="error-state__details">
<DynamicComponent <DynamicComponent

View File

@@ -1,4 +1,5 @@
import React, { useEffect, useState } from "react"; import React, { useEffect, useState } from "react";
// @ts-expect-error (Must be removed after adding @redash/viz typing)
import ErrorBoundary, { ErrorBoundaryContext } from "@redash/viz/lib/components/ErrorBoundary"; import ErrorBoundary, { ErrorBoundaryContext } from "@redash/viz/lib/components/ErrorBoundary";
import { Auth } from "@/services/auth"; import { Auth } from "@/services/auth";
import { policy } from "@/services/policy"; import { policy } from "@/services/policy";
@@ -61,14 +62,11 @@ export function UserSessionWrapper<P>({ bodyClass, currentRoute, render }: UserS
return ( return (
<ApplicationLayout> <ApplicationLayout>
<React.Fragment key={currentRoute.key}> <React.Fragment key={currentRoute.key}>
{/* @ts-expect-error FIXME */}
<ErrorBoundary renderError={(error: Error) => <ErrorMessage error={error} />}> <ErrorBoundary renderError={(error: Error) => <ErrorMessage error={error} />}>
<ErrorBoundaryContext.Consumer> <ErrorBoundaryContext.Consumer>
{( {({ handleError }: { handleError: UserSessionWrapperRenderChildrenProps<P>["onError"] }) =>
{ render({ ...currentRoute.routeParams, pageTitle: currentRoute.title, onError: handleError })
handleError, }
} /* : { handleError: UserSessionWrapperRenderChildrenProps<P>["onError"] } FIXME bring back type */
) => render({ ...currentRoute.routeParams, pageTitle: currentRoute.title, onError: handleError })}
</ErrorBoundaryContext.Consumer> </ErrorBoundaryContext.Consumer>
</ErrorBoundary> </ErrorBoundary>
</React.Fragment> </React.Fragment>

View File

@@ -1,21 +1,14 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import { useUniqueId } from "@/lib/hooks/useUniqueId";
import cx from "classnames";
function BigMessage({ message, icon, children, className }) { function BigMessage({ message, icon, children, className }) {
const messageId = useUniqueId("bm-message");
return ( return (
<div <div className={"p-15 text-center " + className}>
className={"big-message p-15 text-center " + className} <h3 className="m-t-0 m-b-0">
role="status" <i className={"fa " + icon} />
aria-live="assertive"
aria-relevant="additions removals">
<h3 className="m-t-0 m-b-0" aria-labelledby={messageId}>
<i className={cx("fa", icon)} aria-hidden="true" />
</h3> </h3>
<br /> <br />
<span id={messageId}>{message}</span> {message}
{children} {children}
</div> </div>
); );

View File

@@ -1,7 +1,7 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import CopyOutlinedIcon from "@ant-design/icons/CopyOutlined"; import CopyOutlinedIcon from "@ant-design/icons/CopyOutlined";
import "./CodeBlock.less"; import "./CodeBlock.less";

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; @import '~antd/lib/button/style/index';
.code-block { .code-block {
background: rgba(0, 0, 0, 0.06); background: rgba(0, 0, 0, 0.06);

View File

@@ -1,6 +1,6 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import { isEmpty, toUpper, includes, get, uniqueId } from "lodash"; import { isEmpty, toUpper, includes, get } from "lodash";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import List from "antd/lib/list"; import List from "antd/lib/list";
import Modal from "antd/lib/modal"; import Modal from "antd/lib/modal";
@@ -45,8 +45,6 @@ class CreateSourceDialog extends React.Component {
currentStep: StepEnum.SELECT_TYPE, currentStep: StepEnum.SELECT_TYPE,
}; };
formId = uniqueId("sourceForm");
selectType = selectedType => { selectType = selectedType => {
this.setState({ selectedType, currentStep: StepEnum.CONFIGURE_IT }); this.setState({ selectedType, currentStep: StepEnum.CONFIGURE_IT });
}; };
@@ -84,7 +82,6 @@ class CreateSourceDialog extends React.Component {
<div className="m-t-10"> <div className="m-t-10">
<Search <Search
placeholder="Search..." placeholder="Search..."
aria-label="Search"
onChange={e => this.setState({ searchText: e.target.value })} onChange={e => this.setState({ searchText: e.target.value })}
autoFocus autoFocus
data-test="SearchSource" data-test="SearchSource"
@@ -114,12 +111,11 @@ class CreateSourceDialog extends React.Component {
<div className="text-right"> <div className="text-right">
{HELP_TRIGGER_TYPES[helpTriggerType] && ( {HELP_TRIGGER_TYPES[helpTriggerType] && (
<HelpTrigger className="f-13" type={helpTriggerType}> <HelpTrigger className="f-13" type={helpTriggerType}>
Setup Instructions <i className="fa fa-question-circle" aria-hidden="true" /> Setup Instructions <i className="fa fa-question-circle" />
<span className="sr-only">(help)</span>
</HelpTrigger> </HelpTrigger>
)} )}
</div> </div>
<DynamicForm id={this.formId} fields={fields} onSubmit={this.createSource} feedbackIcons hideSubmitButton /> <DynamicForm id="sourceForm" fields={fields} onSubmit={this.createSource} feedbackIcons hideSubmitButton />
{selectedType.type === "databricks" && ( {selectedType.type === "databricks" && (
<small> <small>
By using the Databricks Data Source you agree to the Databricks JDBC/ODBC{" "} By using the Databricks Data Source you agree to the Databricks JDBC/ODBC{" "}
@@ -143,7 +139,7 @@ class CreateSourceDialog extends React.Component {
roundedImage={false} roundedImage={false}
data-test="PreviewItem" data-test="PreviewItem"
data-test-type={item.type}> data-test-type={item.type}>
<i className="fa fa-angle-double-right" aria-hidden="true" /> <i className="fa fa-angle-double-right" />
</PreviewCard> </PreviewCard>
</List.Item> </List.Item>
); );
@@ -173,7 +169,7 @@ class CreateSourceDialog extends React.Component {
<Button <Button
key="submit" key="submit"
htmlType="submit" htmlType="submit"
form={this.formId} form="sourceForm"
type="primary" type="primary"
loading={savingSource} loading={savingSource}
data-test="CreateSourceSaveButton"> data-test="CreateSourceSaveButton">

View File

@@ -86,7 +86,6 @@ export default class EditInPlace extends React.Component {
return ( return (
<InputComponent <InputComponent
defaultValue={value} defaultValue={value}
aria-label="Editing"
onBlur={e => this.stopEditing(e.target.value)} onBlur={e => this.stopEditing(e.target.value)}
onKeyDown={this.handleKeyDown} onKeyDown={this.handleKeyDown}
autoFocus autoFocus

View File

@@ -1,5 +1,5 @@
import { includes, words, capitalize, clone, isNull } from "lodash"; import { includes, words, capitalize, clone, isNull, map, get, find } from "lodash";
import React, { useState, useEffect } from "react"; import React, { useState, useEffect, useRef, useMemo } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Checkbox from "antd/lib/checkbox"; import Checkbox from "antd/lib/checkbox";
import Modal from "antd/lib/modal"; import Modal from "antd/lib/modal";
@@ -11,7 +11,8 @@ import Divider from "antd/lib/divider";
import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper"; import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper";
import QuerySelector from "@/components/QuerySelector"; import QuerySelector from "@/components/QuerySelector";
import { Query } from "@/services/query"; import { Query } from "@/services/query";
import { useUniqueId } from "@/lib/hooks/useUniqueId"; import { QueryBasedParameterMappingType } from "@/services/parameters/QueryBasedDropdownParameter";
import QueryBasedParameterMappingTable from "./query-based-parameter/QueryBasedParameterMappingTable";
const { Option } = Select; const { Option } = Select;
const formItemProps = { labelCol: { span: 6 }, wrapperCol: { span: 16 } }; const formItemProps = { labelCol: { span: 6 }, wrapperCol: { span: 16 } };
@@ -70,17 +71,27 @@ NameInput.propTypes = {
function EditParameterSettingsDialog(props) { function EditParameterSettingsDialog(props) {
const [param, setParam] = useState(clone(props.parameter)); const [param, setParam] = useState(clone(props.parameter));
const [isNameValid, setIsNameValid] = useState(true); const [isNameValid, setIsNameValid] = useState(true);
const [initialQuery, setInitialQuery] = useState(); const [paramQuery, setParamQuery] = useState();
const mappingParameters = useMemo(
() =>
map(paramQuery && paramQuery.getParametersDefs(), mappingParam => ({
mappingParam,
existingMapping: get(param.parameterMapping, mappingParam.name, {
mappingType: QueryBasedParameterMappingType.UNDEFINED,
}),
})),
[param.parameterMapping, paramQuery]
);
const isNew = !props.parameter.name; const isNew = !props.parameter.name;
// fetch query by id // fetch query by id
const initialQueryId = useRef(props.parameter.queryId);
useEffect(() => { useEffect(() => {
const queryId = props.parameter.queryId; if (initialQueryId.current) {
if (queryId) { Query.get({ id: initialQueryId.current }).then(setParamQuery);
Query.get({ id: queryId }).then(setInitialQuery);
} }
}, [props.parameter.queryId]); }, []);
function isFulfilled() { function isFulfilled() {
// name // name
@@ -94,10 +105,16 @@ function EditParameterSettingsDialog(props) {
} }
// query // query
if (param.type === "query" && !param.queryId) { if (param.type === "query") {
if (!param.queryId) {
return false; return false;
} }
if (find(mappingParameters, { existingMapping: { mappingType: QueryBasedParameterMappingType.UNDEFINED } })) {
return false;
}
}
return true; return true;
} }
@@ -112,8 +129,6 @@ function EditParameterSettingsDialog(props) {
props.dialog.close(param); props.dialog.close(param);
} }
const paramFormId = useUniqueId("paramForm");
return ( return (
<Modal <Modal
{...props.dialog.props} {...props.dialog.props}
@@ -128,12 +143,12 @@ function EditParameterSettingsDialog(props) {
htmlType="submit" htmlType="submit"
disabled={!isFulfilled()} disabled={!isFulfilled()}
type="primary" type="primary"
form={paramFormId} form="paramForm"
data-test="SaveParameterSettings"> data-test="SaveParameterSettings">
{isNew ? "Add Parameter" : "OK"} {isNew ? "Add Parameter" : "OK"}
</Button>, </Button>,
]}> ]}>
<Form layout="horizontal" onFinish={onConfirm} id={paramFormId}> <Form layout="horizontal" onFinish={onConfirm} id="paramForm">
{isNew && ( {isNew && (
<NameInput <NameInput
name={param.name} name={param.name}
@@ -190,14 +205,28 @@ function EditParameterSettingsDialog(props) {
</Form.Item> </Form.Item>
)} )}
{param.type === "query" && ( {param.type === "query" && (
<Form.Item label="Query" help="Select query to load dropdown values from" {...formItemProps}> <Form.Item label="Query" help="Select query to load dropdown values from" required {...formItemProps}>
<QuerySelector <QuerySelector
selectedQuery={initialQuery} selectedQuery={paramQuery}
onChange={q => setParam({ ...param, queryId: q && q.id })} onChange={q => {
if (q) {
setParamQuery(q);
setParam({ ...param, queryId: q.id, parameterMapping: {} });
}
}}
type="select" type="select"
/> />
</Form.Item> </Form.Item>
)} )}
{param.type === "query" && paramQuery && paramQuery.hasParameters() && (
<Form.Item className="m-t-15 m-b-5" label="Parameters" required {...formItemProps}>
<QueryBasedParameterMappingTable
param={param}
mappingParameters={mappingParameters}
onChangeParam={setParam}
/>
</Form.Item>
)}
{(param.type === "enum" || param.type === "query") && ( {(param.type === "enum" || param.type === "query") && (
<Form.Item className="m-b-0" label=" " colon={false} {...formItemProps}> <Form.Item className="m-b-0" label=" " colon={false} {...formItemProps}>
<Checkbox <Checkbox

View File

@@ -0,0 +1,134 @@
import React, { useState, useEffect, useRef, useReducer } from "react";
import PropTypes from "prop-types";
import { values } from "lodash";
import Button from "antd/lib/button";
import Tooltip from "antd/lib/tooltip";
import Radio from "antd/lib/radio";
import Typography from "antd/lib/typography/Typography";
import ParameterValueInput from "@/components/ParameterValueInput";
import InputPopover from "@/components/InputPopover";
import Form from "antd/lib/form";
import { QueryBasedParameterMappingType } from "@/services/parameters/QueryBasedDropdownParameter";
import QuestionCircleFilledIcon from "@ant-design/icons/QuestionCircleFilled";
import EditOutlinedIcon from "@ant-design/icons/EditOutlined";
const { Text } = Typography;
const formItemProps = { labelCol: { span: 6 }, wrapperCol: { span: 16 } };
export default function QueryBasedParameterMappingEditor({ parameter, mapping, searchAvailable, onChange }) {
const [showPopover, setShowPopover] = useState(false);
const [newMapping, setNewMapping] = useReducer((prevState, updates) => ({ ...prevState, ...updates }), mapping);
const newMappingRef = useRef(newMapping);
useEffect(() => {
if (
mapping.mappingType !== newMappingRef.current.mappingType ||
mapping.staticValue !== newMappingRef.current.staticValue
) {
setNewMapping(mapping);
}
}, [mapping]);
const parameterRef = useRef(parameter);
useEffect(() => {
parameterRef.current.setValue(mapping.staticValue);
}, [mapping.staticValue]);
const onCancel = () => {
setNewMapping(mapping);
setShowPopover(false);
};
const onOk = () => {
onChange(newMapping);
setShowPopover(false);
};
let currentState = <Text type="secondary">Pick a type</Text>;
if (mapping.mappingType === QueryBasedParameterMappingType.DROPDOWN_SEARCH) {
currentState = "Dropdown Search";
} else if (mapping.mappingType === QueryBasedParameterMappingType.STATIC) {
currentState = `Value: ${mapping.staticValue}`;
}
return (
<>
{currentState}
<InputPopover
placement="left"
trigger="click"
header="Edit Parameter Source"
okButtonProps={{
disabled: newMapping.mappingType === QueryBasedParameterMappingType.STATIC && parameter.isEmpty,
}}
onOk={onOk}
onCancel={onCancel}
content={
<Form>
<Form.Item className="m-b-15" label="Source" {...formItemProps}>
<Radio.Group
value={newMapping.mappingType}
onChange={({ target }) => setNewMapping({ mappingType: target.value })}>
<Radio
className="radio"
value={QueryBasedParameterMappingType.DROPDOWN_SEARCH}
disabled={!searchAvailable || parameter.type !== "text"}>
Dropdown Search{" "}
{(!searchAvailable || parameter.type !== "text") && (
<Tooltip
title={
parameter.type !== "text"
? "Dropdown Search is only available for Text Parameters"
: "There is already a parameter mapped with the Dropdown Search type."
}>
<QuestionCircleFilledIcon />
</Tooltip>
)}
</Radio>
<Radio className="radio" value={QueryBasedParameterMappingType.STATIC}>
Static Value
</Radio>
</Radio.Group>
</Form.Item>
{newMapping.mappingType === QueryBasedParameterMappingType.STATIC && (
<Form.Item label="Value" required {...formItemProps}>
<ParameterValueInput
type={parameter.type}
value={parameter.normalizedValue}
enumOptions={parameter.enumOptions}
queryId={parameter.queryId}
parameter={parameter}
onSelect={value => {
parameter.setValue(value);
setNewMapping({ staticValue: parameter.getExecutionValue({ joinListValues: true }) });
}}
/>
</Form.Item>
)}
</Form>
}
visible={showPopover}
onVisibleChange={setShowPopover}>
<Button className="m-l-5" size="small" type="dashed">
<EditOutlinedIcon />
</Button>
</InputPopover>
</>
);
}
QueryBasedParameterMappingEditor.propTypes = {
parameter: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
mapping: PropTypes.shape({
mappingType: PropTypes.oneOf(values(QueryBasedParameterMappingType)),
staticValue: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
}),
searchAvailable: PropTypes.bool,
onChange: PropTypes.func,
};
QueryBasedParameterMappingEditor.defaultProps = {
mapping: { mappingType: QueryBasedParameterMappingType.UNDEFINED, staticValue: undefined },
searchAvailable: false,
onChange: () => {},
};

View File

@@ -0,0 +1,56 @@
import React from "react";
import { findKey } from "lodash";
import PropTypes from "prop-types";
import Table from "antd/lib/table";
import { QueryBasedParameterMappingType } from "@/services/parameters/QueryBasedDropdownParameter";
import QueryBasedParameterMappingEditor from "./QueryBasedParameterMappingEditor";
export default function QueryBasedParameterMappingTable({ param, mappingParameters, onChangeParam }) {
return (
<Table
dataSource={mappingParameters}
size="middle"
pagination={false}
rowKey={({ mappingParam }) => `param${mappingParam.name}`}>
<Table.Column title="Title" key="title" render={({ mappingParam }) => mappingParam.getTitle()} />
<Table.Column
title="Keyword"
key="keyword"
className="keyword"
render={({ mappingParam }) => <code>{`{{ ${mappingParam.name} }}`}</code>}
/>
<Table.Column
title="Value Source"
key="source"
render={({ mappingParam, existingMapping }) => (
<QueryBasedParameterMappingEditor
parameter={mappingParam.setValue(existingMapping.staticValue)}
mapping={existingMapping}
searchAvailable={
!findKey(param.parameterMapping, {
mappingType: QueryBasedParameterMappingType.DROPDOWN_SEARCH,
}) || existingMapping.mappingType === QueryBasedParameterMappingType.DROPDOWN_SEARCH
}
onChange={mapping =>
onChangeParam({
...param,
parameterMapping: { ...param.parameterMapping, [mappingParam.name]: mapping },
})
}
/>
)}
/>
</Table>
);
}
QueryBasedParameterMappingTable.propTypes = {
param: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
mappingParameters: PropTypes.arrayOf(PropTypes.object), // eslint-disable-line react/forbid-prop-types
onChangeParam: PropTypes.func,
};
QueryBasedParameterMappingTable.defaultProps = {
mappingParameters: [],
onChangeParam: () => {},
};

View File

@@ -3,7 +3,6 @@ import PropTypes from "prop-types";
import Dropdown from "antd/lib/dropdown"; import Dropdown from "antd/lib/dropdown";
import Menu from "antd/lib/menu"; import Menu from "antd/lib/menu";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import PlainButton from "@/components/PlainButton";
import { clientConfig } from "@/services/auth"; import { clientConfig } from "@/services/auth";
import PlusCircleFilledIcon from "@ant-design/icons/PlusCircleFilled"; import PlusCircleFilledIcon from "@ant-design/icons/PlusCircleFilled";
@@ -19,18 +18,16 @@ export default function QueryControlDropdown(props) {
<Menu> <Menu>
{!props.query.isNew() && (!props.query.is_draft || !props.query.is_archived) && ( {!props.query.isNew() && (!props.query.is_draft || !props.query.is_archived) && (
<Menu.Item> <Menu.Item>
<PlainButton onClick={() => props.openAddToDashboardForm(props.selectedTab)}> <a target="_self" onClick={() => props.openAddToDashboardForm(props.selectedTab)}>
<PlusCircleFilledIcon /> Add to Dashboard <PlusCircleFilledIcon /> Add to Dashboard
</PlainButton> </a>
</Menu.Item> </Menu.Item>
)} )}
{!clientConfig.disablePublicUrls && !props.query.isNew() && ( {!clientConfig.disablePublicUrls && !props.query.isNew() && (
<Menu.Item> <Menu.Item>
<PlainButton <a onClick={() => props.showEmbedDialog(props.query, props.selectedTab)} data-test="ShowEmbedDialogButton">
onClick={() => props.showEmbedDialog(props.query, props.selectedTab)}
data-test="ShowEmbedDialogButton">
<ShareAltOutlinedIcon /> Embed Elsewhere <ShareAltOutlinedIcon /> Embed Elsewhere
</PlainButton> </a>
</Menu.Item> </Menu.Item>
)} )}
<Menu.Item> <Menu.Item>

View File

@@ -1,14 +1,12 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import cx from "classnames";
import { clientConfig, currentUser } from "@/services/auth"; import { clientConfig, currentUser } from "@/services/auth";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import Alert from "antd/lib/alert"; import Alert from "antd/lib/alert";
import HelpTrigger from "@/components/HelpTrigger"; import HelpTrigger from "@/components/HelpTrigger";
import { useUniqueId } from "@/lib/hooks/useUniqueId";
export default function EmailSettingsWarning({ featureName, className, mode, adminOnly }) { export default function EmailSettingsWarning({ featureName, className, mode, adminOnly }) {
const messageDescriptionId = useUniqueId("sr-mail-description");
if (!clientConfig.mailSettingsMissing) { if (!clientConfig.mailSettingsMissing) {
return null; return null;
} }
@@ -18,7 +16,7 @@ export default function EmailSettingsWarning({ featureName, className, mode, adm
} }
const message = ( const message = (
<span id={messageDescriptionId}> <span>
Your mail server isn&apos;t configured correctly, and is needed for {featureName} to work.{" "} Your mail server isn&apos;t configured correctly, and is needed for {featureName} to work.{" "}
<HelpTrigger type="MAIL_CONFIG" className="f-inherit" /> <HelpTrigger type="MAIL_CONFIG" className="f-inherit" />
</span> </span>
@@ -26,11 +24,8 @@ export default function EmailSettingsWarning({ featureName, className, mode, adm
if (mode === "icon") { if (mode === "icon") {
return ( return (
<Tooltip title={message} placement="topRight" arrowPointAtCenter> <Tooltip title={message}>
{/* eslint-disable-next-line jsx-a11y/no-noninteractive-tabindex */} <i className={cx("fa fa-exclamation-triangle", className)} />
<span className={className} aria-label="Mail alert" aria-describedby={messageDescriptionId} tabIndex={0}>
<i className={"fa fa-exclamation-triangle"} aria-hidden="true" />
</span>
</Tooltip> </Tooltip>
); );
} }

View File

@@ -1,6 +1,5 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import PlainButton from "@/components/PlainButton";
export default class FavoritesControl extends React.Component { export default class FavoritesControl extends React.Component {
static propTypes = { static propTypes = {
@@ -30,13 +29,12 @@ export default class FavoritesControl extends React.Component {
const icon = item.is_favorite ? "fa fa-star" : "fa fa-star-o"; const icon = item.is_favorite ? "fa fa-star" : "fa fa-star-o";
const title = item.is_favorite ? "Remove from favorites" : "Add to favorites"; const title = item.is_favorite ? "Remove from favorites" : "Add to favorites";
return ( return (
<PlainButton <a
title={title} title={title}
aria-label={title} className="favorites-control btn-favourite"
className="favorites-control btn-favorite"
onClick={event => this.toggleItem(event, item, onChange)}> onClick={event => this.toggleItem(event, item, onChange)}>
<i className={icon} aria-hidden="true" /> <i className={icon} aria-hidden="true" />
</PlainButton> </a>
); );
} }
} }

View File

@@ -112,11 +112,11 @@ function Filters({ filters, onChange }) {
{!filter.multiple && options} {!filter.multiple && options}
{filter.multiple && [ {filter.multiple && [
<Select.Option key={NONE_VALUES} data-test="ClearOption"> <Select.Option key={NONE_VALUES} data-test="ClearOption">
<i className="fa fa-square-o m-r-5" aria-hidden="true" /> <i className="fa fa-square-o m-r-5" />
Clear Clear
</Select.Option>, </Select.Option>,
<Select.Option key={ALL_VALUES} data-test="SelectAllOption"> <Select.Option key={ALL_VALUES} data-test="SelectAllOption">
<i className="fa fa-check-square-o m-r-5" aria-hidden="true" /> <i className="fa fa-check-square-o m-r-5" />
Select All Select All
</Select.Option>, </Select.Option>,
<Select.OptGroup key="Values" title="Values"> <Select.OptGroup key="Values" title="Values">

View File

@@ -2,10 +2,9 @@ import { startsWith, get, some, mapValues } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import cx from "classnames"; import cx from "classnames";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import Drawer from "antd/lib/drawer"; import Drawer from "antd/lib/drawer";
import Link from "@/components/Link"; import Link from "@/components/Link";
import PlainButton from "@/components/PlainButton";
import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined"; import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined";
import BigMessage from "@/components/BigMessage"; import BigMessage from "@/components/BigMessage";
import DynamicComponent, { registerComponent } from "@/components/DynamicComponent"; import DynamicComponent, { registerComponent } from "@/components/DynamicComponent";
@@ -46,7 +45,7 @@ export const TYPES = mapValues(
NUMBER_FORMAT_SPECS: ["/user-guide/visualizations/formatting-numbers", "Formatting Numbers"], NUMBER_FORMAT_SPECS: ["/user-guide/visualizations/formatting-numbers", "Formatting Numbers"],
GETTING_STARTED: ["/user-guide/getting-started", "Guide: Getting Started"], GETTING_STARTED: ["/user-guide/getting-started", "Guide: Getting Started"],
DASHBOARDS: ["/user-guide/dashboards", "Guide: Dashboards"], DASHBOARDS: ["/user-guide/dashboards", "Guide: Dashboards"],
QUERIES: ["/user-guide/querying", "Guide: Queries"], QUERIES: ["/help/user-guide/querying", "Guide: Queries"],
ALERTS: ["/user-guide/alerts", "Guide: Alerts"], ALERTS: ["/user-guide/alerts", "Guide: Alerts"],
}, },
([url, title]) => [DOMAIN + HELP_PATH + url, title] ([url, title]) => [DOMAIN + HELP_PATH + url, title]
@@ -69,7 +68,7 @@ const HelpTriggerDefaultProps = {
className: null, className: null,
showTooltip: true, showTooltip: true,
renderAsLink: false, renderAsLink: false,
children: <i className="fa fa-question-circle" aria-hidden="true" />, children: <i className="fa fa-question-circle" />,
}; };
export function helpTriggerWithTypes(types, allowedDomains = [], drawerClassName = null) { export function helpTriggerWithTypes(types, allowedDomains = [], drawerClassName = null) {
@@ -171,13 +170,7 @@ export function helpTriggerWithTypes(types, allowedDomains = [], drawerClassName
this.props.showTooltip ? ( this.props.showTooltip ? (
<> <>
{tooltip} {tooltip}
{shouldRenderAsLink && ( {shouldRenderAsLink && <i className="fa fa-external-link" style={{ marginLeft: 5 }} />}
<>
{" "}
<i className="fa fa-external-link" style={{ marginLeft: 5 }} aria-hidden="true" />
<span className="sr-only">(opens in a new tab)</span>
</>
)}
</> </>
) : null ) : null
}> }>
@@ -204,15 +197,14 @@ export function helpTriggerWithTypes(types, allowedDomains = [], drawerClassName
<Tooltip title="Open page in a new window" placement="left"> <Tooltip title="Open page in a new window" placement="left">
{/* eslint-disable-next-line react/jsx-no-target-blank */} {/* eslint-disable-next-line react/jsx-no-target-blank */}
<Link href={url} target="_blank"> <Link href={url} target="_blank">
<i className="fa fa-external-link" aria-hidden="true" /> <i className="fa fa-external-link" />
<span className="sr-only">(opens in a new tab)</span>
</Link> </Link>
</Tooltip> </Tooltip>
)} )}
<Tooltip title="Close" placement="bottom"> <Tooltip title="Close" placement="bottom">
<PlainButton onClick={this.closeDrawer}> <a onClick={this.closeDrawer}>
<CloseOutlinedIcon /> <CloseOutlinedIcon />
</PlainButton> </a>
</Tooltip> </Tooltip>
</div> </div>

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; @import "~antd/lib/drawer/style/drawer";
@help-doc-bg: #f7f7f7; // according to https://github.com/getredash/website/blob/13daff2d8b570956565f482236f6245042e8477f/src/scss/_components/_variables.scss#L15 @help-doc-bg: #f7f7f7; // according to https://github.com/getredash/website/blob/13daff2d8b570956565f482236f6245042e8477f/src/scss/_components/_variables.scss#L15
@@ -38,8 +38,7 @@
border: 2px solid @help-doc-bg; border: 2px solid @help-doc-bg;
display: flex; display: flex;
a, a {
.plain-button {
height: 26px; height: 26px;
width: 26px; width: 26px;
display: flex; display: flex;

View File

@@ -0,0 +1,57 @@
import React from "react";
import PropTypes from "prop-types";
import Button from "antd/lib/button";
import Popover from "antd/lib/popover";
import "./index.less";
export default function InputPopover({
header,
content,
children,
okButtonProps,
cancelButtonProps,
onCancel,
onOk,
...props
}) {
return (
<Popover
{...props}
content={
<div className="input-popover-content" data-test="InputPopoverContent">
{header && <header>{header}</header>}
{content}
<footer>
<Button onClick={onCancel} {...cancelButtonProps}>
Cancel
</Button>
<Button onClick={onOk} type="primary" {...okButtonProps}>
OK
</Button>
</footer>
</div>
}>
{children}
</Popover>
);
}
InputPopover.propTypes = {
header: PropTypes.node,
content: PropTypes.node,
children: PropTypes.node,
okButtonProps: PropTypes.object,
cancelButtonProps: PropTypes.object,
onOk: PropTypes.func,
onCancel: PropTypes.func,
};
InputPopover.defaultProps = {
header: null,
children: null,
okButtonProps: null,
cancelButtonProps: null,
onOk: () => {},
onCancel: () => {},
};

View File

@@ -0,0 +1,37 @@
@import "~antd/lib/modal/style/index"; // for ant @vars
.input-popover-content {
width: 390px;
.radio {
display: block;
height: 30px;
line-height: 30px;
}
.form-item {
margin-bottom: 10px;
}
header {
padding: 0 16px 10px;
margin: 0 -16px 20px;
border-bottom: @border-width-base @border-style-base @border-color-split;
font-size: @font-size-lg;
font-weight: 500;
color: @heading-color;
display: flex;
justify-content: space-between;
}
footer {
border-top: @border-width-base @border-style-base @border-color-split;
padding: 10px 16px 0;
margin: 0 -16px;
text-align: right;
button {
margin-left: 8px;
}
}
}

View File

@@ -1,8 +1,7 @@
import React from "react"; import React from "react";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import CopyOutlinedIcon from "@ant-design/icons/CopyOutlined"; import CopyOutlinedIcon from "@ant-design/icons/CopyOutlined";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import PlainButton from "./PlainButton";
export default class InputWithCopy extends React.Component { export default class InputWithCopy extends React.Component {
constructor(props) { constructor(props) {
@@ -43,10 +42,7 @@ export default class InputWithCopy extends React.Component {
render() { render() {
const copyButton = ( const copyButton = (
<Tooltip title={this.state.copied || "Copy"}> <Tooltip title={this.state.copied || "Copy"}>
<PlainButton onClick={this.copy}> <CopyOutlinedIcon style={{ cursor: "pointer" }} onClick={this.copy} />
{/* TODO: lacks visual feedback */}
<CopyOutlinedIcon />
</PlainButton>
</Tooltip> </Tooltip>
); );

View File

@@ -0,0 +1,26 @@
import React from "react";
import Button from "antd/lib/button";
function DefaultLinkComponent(props) {
return <a {...props} />; // eslint-disable-line jsx-a11y/anchor-has-content
}
function Link(props) {
return <Link.Component {...props} />;
}
Link.Component = DefaultLinkComponent;
function DefaultButtonLinkComponent(props) {
return <Button role="button" {...props} />;
}
function ButtonLink(props) {
return <ButtonLink.Component {...props} />;
}
ButtonLink.Component = DefaultButtonLinkComponent;
Link.Button = ButtonLink;
export default Link;

View File

@@ -1,61 +0,0 @@
import React from "react";
import Button, { ButtonProps as AntdButtonProps } from "antd/lib/button";
function DefaultLinkComponent({ children, ...props }: React.AnchorHTMLAttributes<HTMLAnchorElement>) {
return <a {...props}>{children}</a>;
}
Link.Component = DefaultLinkComponent;
interface LinkProps extends Omit<React.AnchorHTMLAttributes<HTMLAnchorElement>, "role" | "type" | "target"> {
href: string;
}
function Link({ children, ...props }: LinkProps) {
return <Link.Component {...props}>{children}</Link.Component>;
}
interface LinkWithIconProps extends LinkProps {
children: string;
icon: JSX.Element;
alt: string;
target?: "_self" | "_blank" | "_parent" | "_top";
}
function LinkWithIcon({ icon, alt, children, ...props }: LinkWithIconProps) {
return (
<Link.Component {...props}>
{children} {icon} <span className="sr-only">{alt}</span>
</Link.Component>
);
}
Link.WithIcon = LinkWithIcon;
function ExternalLink({
icon = <i className="fa fa-external-link" aria-hidden="true" />,
alt = "(opens in a new tab)",
...props
}: Omit<LinkWithIconProps, "target">) {
return <Link.WithIcon target="_blank" rel="noopener noreferrer" icon={icon} alt={alt} {...props} />;
}
Link.External = ExternalLink;
// Ant Button will render an <a> if href is present.
function DefaultButtonLinkComponent(props: ButtonProps) {
return <Button {...props} />;
}
ButtonLink.Component = DefaultButtonLinkComponent;
interface ButtonProps extends AntdButtonProps {
href: string;
}
function ButtonLink(props: ButtonProps) {
return <ButtonLink.Component {...props} />;
}
Link.Button = ButtonLink;
export default Link;

View File

@@ -2,26 +2,21 @@ import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import Badge from "antd/lib/badge"; import Badge from "antd/lib/badge";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import KeyboardShortcuts from "@/services/KeyboardShortcuts"; import KeyboardShortcuts from "@/services/KeyboardShortcuts";
function ParameterApplyButton({ paramCount, onClick }) { function ParameterApplyButton({ paramCount, onClick }) {
// show spinner when count is empty so the fade out is consistent // show spinner when count is empty so the fade out is consistent
const icon = !paramCount ? ( const icon = !paramCount ? "spinner fa-pulse" : "check";
<span role="status" aria-live="polite" aria-relevant="additions removals">
<i className="fa fa-spinner fa-pulse" aria-hidden="true" />
<span className="sr-only">Loading...</span>
</span>
) : (
<i className="fa fa-check" aria-hidden="true" />
);
return ( return (
<div className="parameter-apply-button" data-show={!!paramCount} data-test="ParameterApplyButton"> <div className="parameter-apply-button" data-show={!!paramCount} data-test="ParameterApplyButton">
<Badge count={paramCount}> <Badge count={paramCount}>
<Tooltip title={paramCount ? `${KeyboardShortcuts.modKey} + Enter` : null}> <Tooltip title={paramCount ? `${KeyboardShortcuts.modKey} + Enter` : null}>
<span> <span>
<Button onClick={onClick}>{icon} Apply Changes</Button> <Button onClick={onClick}>
<i className={`fa fa-${icon}`} /> Apply Changes
</Button>
</span> </span>
</Tooltip> </Tooltip>
</Badge> </Badge>

View File

@@ -12,11 +12,12 @@ import Tag from "antd/lib/tag";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import Radio from "antd/lib/radio"; import Radio from "antd/lib/radio";
import Form from "antd/lib/form"; import Form from "antd/lib/form";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import ParameterValueInput from "@/components/ParameterValueInput"; import ParameterValueInput from "@/components/ParameterValueInput";
import { ParameterMappingType } from "@/services/widget"; import { ParameterMappingType } from "@/services/widget";
import { Parameter, cloneParameter } from "@/services/parameters"; import { Parameter, cloneParameter } from "@/services/parameters";
import HelpTrigger from "@/components/HelpTrigger"; import HelpTrigger from "@/components/HelpTrigger";
import InputPopover from "@/components/InputPopover";
import QuestionCircleFilledIcon from "@ant-design/icons/QuestionCircleFilled"; import QuestionCircleFilledIcon from "@ant-design/icons/QuestionCircleFilled";
import EditOutlinedIcon from "@ant-design/icons/EditOutlined"; import EditOutlinedIcon from "@ant-design/icons/EditOutlined";
@@ -201,13 +202,7 @@ export class ParameterMappingInput extends React.Component {
const { const {
mapping: { mapTo }, mapping: { mapTo },
} = this.props; } = this.props;
return ( return <Input value={mapTo} onChange={e => this.updateParamMapping({ mapTo: e.target.value })} />;
<Input
value={mapTo}
aria-label="Parameter name (key)"
onChange={e => this.updateParamMapping({ mapTo: e.target.value })}
/>
);
} }
renderDashboardMapToExisting() { renderDashboardMapToExisting() {
@@ -319,43 +314,34 @@ class MappingEditor extends React.Component {
this.setState({ visible: false }); this.setState({ visible: false });
}; };
renderContent() { render() {
const { mapping, inputError } = this.state; const { visible, mapping, inputError } = this.state;
return ( return (
<div className="parameter-mapping-editor" data-test="EditParamMappingPopover"> <InputPopover
<header> placement="left"
trigger="click"
header={
<>
Edit Source and Value <HelpTrigger type="VALUE_SOURCE_OPTIONS" /> Edit Source and Value <HelpTrigger type="VALUE_SOURCE_OPTIONS" />
</header> </>
}
content={
<ParameterMappingInput <ParameterMappingInput
mapping={mapping} mapping={mapping}
existingParamNames={this.props.existingParamNames} existingParamNames={this.props.existingParamNames}
onChange={this.onChange} onChange={this.onChange}
inputError={inputError} inputError={inputError}
/> />
<footer>
<Button onClick={this.hide}>Cancel</Button>
<Button onClick={this.save} disabled={!!inputError} type="primary">
OK
</Button>
</footer>
</div>
);
} }
onOk={this.save}
render() { onCancel={this.hide}
const { visible, mapping } = this.state; okButtonProps={{ disabled: !!inputError }}
return (
<Popover
placement="left"
trigger="click"
content={this.renderContent()}
visible={visible} visible={visible}
onVisibleChange={this.onVisibleChange}> onVisibleChange={this.onVisibleChange}>
<Button size="small" type="dashed" data-test={`EditParamMappingButton-${mapping.param.name}`}> <Button size="small" type="dashed" data-test={`EditParamMappingButton-${mapping.param.name}`}>
<EditOutlinedIcon /> <EditOutlinedIcon />
</Button> </Button>
</Popover> </InputPopover>
); );
} }
} }
@@ -426,7 +412,6 @@ class TitleEditor extends React.Component {
size="small" size="small"
value={this.state.title} value={this.state.title}
placeholder={paramTitle} placeholder={paramTitle}
aria-label="Edit parameter title"
onChange={this.onEditingTitleChange} onChange={this.onEditingTitleChange}
onPressEnter={this.save} onPressEnter={this.save}
maxLength={100} maxLength={100}
@@ -447,10 +432,7 @@ class TitleEditor extends React.Component {
if (mapping.type === MappingType.StaticValue) { if (mapping.type === MappingType.StaticValue) {
return ( return (
<Tooltip placement="right" title="Titles for static values don't appear in widgets"> <Tooltip placement="right" title="Titles for static values don't appear in widgets">
{/* eslint-disable-next-line jsx-a11y/no-noninteractive-tabindex */} <i className="fa fa-eye-slash" />
<span tabIndex={0}>
<i className="fa fa-eye-slash" aria-hidden="true" />
</span>
</Tooltip> </Tooltip>
); );
} }

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; // for ant @vars @import "~antd/lib/modal/style/index"; // for ant @vars
.parameters-mapping-list { .parameters-mapping-list {
.keyword { .keyword {
@@ -22,42 +22,6 @@
} }
} }
.parameter-mapping-editor {
width: 390px;
.radio {
display: block;
height: 30px;
line-height: 30px;
}
.form-item {
margin-bottom: 10px;
}
header {
padding: 0 16px 10px;
margin: 0 -16px 20px;
border-bottom: @border-width-base @border-style-base @border-color-split;
font-size: @font-size-lg;
font-weight: 500;
color: @heading-color;
display: flex;
justify-content: space-between;
}
footer {
border-top: @border-width-base @border-style-base @border-color-split;
padding: 10px 16px 0;
margin: 0 -16px;
text-align: right;
button {
margin-left: 8px;
}
}
}
.parameter-mapping-title { .parameter-mapping-title {
.text { .text {
margin-right: 3px; margin-right: 3px;

View File

@@ -101,6 +101,7 @@ class ParameterValueInput extends React.Component {
<SelectWithVirtualScroll <SelectWithVirtualScroll
className={this.props.className} className={this.props.className}
mode={parameter.multiValuesOptions ? "multiple" : "default"} mode={parameter.multiValuesOptions ? "multiple" : "default"}
optionFilterProp="children"
value={normalize(value)} value={normalize(value)}
onChange={this.onSelect} onChange={this.onSelect}
options={map(enumOptionsArray, opt => ({ label: String(opt), value: opt }))} options={map(enumOptionsArray, opt => ({ label: String(opt), value: opt }))}
@@ -119,6 +120,7 @@ class ParameterValueInput extends React.Component {
<QueryBasedParameterInput <QueryBasedParameterInput
className={this.props.className} className={this.props.className}
mode={parameter.multiValuesOptions ? "multiple" : "default"} mode={parameter.multiValuesOptions ? "multiple" : "default"}
optionFilterProp="children"
parameter={parameter} parameter={parameter}
value={value} value={value}
queryId={queryId} queryId={queryId}
@@ -136,12 +138,7 @@ class ParameterValueInput extends React.Component {
const normalize = val => (isNaN(val) ? undefined : val); const normalize = val => (isNaN(val) ? undefined : val);
return ( return (
<InputNumber <InputNumber className={className} value={normalize(value)} onChange={val => this.onSelect(normalize(val))} />
className={className}
value={normalize(value)}
aria-label="Parameter number value"
onChange={val => this.onSelect(normalize(val))}
/>
); );
} }
@@ -153,7 +150,6 @@ class ParameterValueInput extends React.Component {
<Input <Input
className={className} className={className}
value={value} value={value}
aria-label="Parameter text value"
data-test="TextParamInput" data-test="TextParamInput"
onChange={e => this.onSelect(e.target.value)} onChange={e => this.onSelect(e.target.value)}
/> />

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; // for ant @vars @import "~antd/lib/input-number/style/index"; // for ant @vars
@input-dirty: #fffce1; @input-dirty: #fffce1;
@@ -21,7 +21,7 @@
.@{ant-prefix}-input-number, .@{ant-prefix}-input-number,
.@{ant-prefix}-select-selector, .@{ant-prefix}-select-selector,
.@{ant-prefix}-picker { .@{ant-prefix}-picker {
background-color: @input-dirty; background-color: @input-dirty !important;
} }
} }
} }

View File

@@ -6,9 +6,7 @@ import location from "@/services/location";
import { Parameter, createParameter } from "@/services/parameters"; import { Parameter, createParameter } from "@/services/parameters";
import ParameterApplyButton from "@/components/ParameterApplyButton"; import ParameterApplyButton from "@/components/ParameterApplyButton";
import ParameterValueInput from "@/components/ParameterValueInput"; import ParameterValueInput from "@/components/ParameterValueInput";
import PlainButton from "@/components/PlainButton";
import EditParameterSettingsDialog from "./EditParameterSettingsDialog"; import EditParameterSettingsDialog from "./EditParameterSettingsDialog";
import { toHuman } from "@/lib/utils";
import "./Parameters.less"; import "./Parameters.less";
@@ -24,23 +22,19 @@ export default class Parameters extends React.Component {
static propTypes = { static propTypes = {
parameters: PropTypes.arrayOf(PropTypes.instanceOf(Parameter)), parameters: PropTypes.arrayOf(PropTypes.instanceOf(Parameter)),
editable: PropTypes.bool, editable: PropTypes.bool,
sortable: PropTypes.bool,
disableUrlUpdate: PropTypes.bool, disableUrlUpdate: PropTypes.bool,
onValuesChange: PropTypes.func, onValuesChange: PropTypes.func,
onPendingValuesChange: PropTypes.func, onPendingValuesChange: PropTypes.func,
onParametersEdit: PropTypes.func, onParametersEdit: PropTypes.func,
appendSortableToParent: PropTypes.bool,
}; };
static defaultProps = { static defaultProps = {
parameters: [], parameters: [],
editable: false, editable: false,
sortable: false,
disableUrlUpdate: false, disableUrlUpdate: false,
onValuesChange: () => {}, onValuesChange: () => {},
onPendingValuesChange: () => {}, onPendingValuesChange: () => {},
onParametersEdit: () => {}, onParametersEdit: () => {},
appendSortableToParent: true,
}; };
constructor(props) { constructor(props) {
@@ -90,7 +84,7 @@ export default class Parameters extends React.Component {
if (oldIndex !== newIndex) { if (oldIndex !== newIndex) {
this.setState(({ parameters }) => { this.setState(({ parameters }) => {
parameters.splice(newIndex, 0, parameters.splice(oldIndex, 1)[0]); parameters.splice(newIndex, 0, parameters.splice(oldIndex, 1)[0]);
onParametersEdit(parameters); onParametersEdit();
return { parameters }; return { parameters };
}); });
} }
@@ -115,7 +109,7 @@ export default class Parameters extends React.Component {
this.setState(({ parameters }) => { this.setState(({ parameters }) => {
const updatedParameter = extend(parameter, updated); const updatedParameter = extend(parameter, updated);
parameters[index] = createParameter(updatedParameter, updatedParameter.parentQueryId); parameters[index] = createParameter(updatedParameter, updatedParameter.parentQueryId);
onParametersEdit(parameters); onParametersEdit();
return { parameters }; return { parameters };
}); });
}); });
@@ -126,16 +120,15 @@ export default class Parameters extends React.Component {
return ( return (
<div key={param.name} className="di-block" data-test={`ParameterName-${param.name}`}> <div key={param.name} className="di-block" data-test={`ParameterName-${param.name}`}>
<div className="parameter-heading"> <div className="parameter-heading">
<label>{param.title || toHuman(param.name)}</label> <label>{param.getTitle()}</label>
{editable && ( {editable && (
<PlainButton <button
className="btn btn-default btn-xs m-l-5" className="btn btn-default btn-xs m-l-5"
aria-label="Edit"
onClick={() => this.showParameterSettings(param, index)} onClick={() => this.showParameterSettings(param, index)}
data-test={`ParameterSettings-${param.name}`} data-test={`ParameterSettings-${param.name}`}
type="button"> type="button">
<i className="fa fa-cog" aria-hidden="true" /> <i className="fa fa-cog" />
</PlainButton> </button>
)} )}
</div> </div>
<ParameterValueInput <ParameterValueInput
@@ -152,17 +145,15 @@ export default class Parameters extends React.Component {
render() { render() {
const { parameters } = this.state; const { parameters } = this.state;
const { sortable, appendSortableToParent } = this.props; const { editable } = this.props;
const dirtyParamCount = size(filter(parameters, "hasPendingValue")); const dirtyParamCount = size(filter(parameters, "hasPendingValue"));
return ( return (
<SortableContainer <SortableContainer
disabled={!sortable} disabled={!editable}
axis="xy" axis="xy"
useDragHandle useDragHandle
lockToContainerEdges lockToContainerEdges
helperClass="parameter-dragged" helperClass="parameter-dragged"
helperContainer={containerEl => (appendSortableToParent ? containerEl : document.body)}
updateBeforeSortStart={this.onBeforeSortStart} updateBeforeSortStart={this.onBeforeSortStart}
onSortEnd={this.moveParameter} onSortEnd={this.moveParameter}
containerProps={{ containerProps={{
@@ -171,11 +162,8 @@ export default class Parameters extends React.Component {
}}> }}>
{parameters.map((param, index) => ( {parameters.map((param, index) => (
<SortableElement key={param.name} index={index}> <SortableElement key={param.name} index={index}>
<div <div className="parameter-block" data-editable={editable || null}>
className="parameter-block" {editable && <DragHandle data-test={`DragHandle-${param.name}`} />}
data-editable={sortable || null}
data-test={`ParameterBlock-${param.name}`}>
{sortable && <DragHandle data-test={`DragHandle-${param.name}`} />}
{this.renderParameter(param, index)} {this.renderParameter(param, index)}
</div> </div>
</SortableElement> </SortableElement>

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; @import "../assets/less/ant";
.parameter-block { .parameter-block {
display: inline-block; display: inline-block;
@@ -21,8 +21,6 @@
&.parameter-dragged { &.parameter-dragged {
z-index: 2; z-index: 2;
margin: 4px 0 0 4px;
padding: 3px 6px 6px;
box-shadow: 0 4px 9px -3px rgba(102, 136, 153, 0.15); box-shadow: 0 4px 9px -3px rgba(102, 136, 153, 0.15);
} }
} }

View File

@@ -7,12 +7,11 @@ import List from "antd/lib/list";
import Modal from "antd/lib/modal"; import Modal from "antd/lib/modal";
import Select from "antd/lib/select"; import Select from "antd/lib/select";
import Tag from "antd/lib/tag"; import Tag from "antd/lib/tag";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper"; import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper";
import { toHuman } from "@/lib/utils"; import { toHuman } from "@/lib/utils";
import HelpTrigger from "@/components/HelpTrigger"; import HelpTrigger from "@/components/HelpTrigger";
import { UserPreviewCard } from "@/components/PreviewCard"; import { UserPreviewCard } from "@/components/PreviewCard";
import PlainButton from "@/components/PlainButton";
import notification from "@/services/notification"; import notification from "@/services/notification";
import User from "@/services/user"; import User from "@/services/user";
@@ -103,16 +102,7 @@ function UserSelect({ onSelect, shouldShowUser }) {
placeholder="Add users..." placeholder="Add users..."
showSearch showSearch
onSearch={setSearchTerm} onSearch={setSearchTerm}
suffixIcon={ suffixIcon={loadingUsers ? <i className="fa fa-spinner fa-pulse" /> : <i className="fa fa-search" />}
loadingUsers ? (
<span role="status" aria-live="polite" aria-relevant="additions removals">
<i className="fa fa-spinner fa-pulse" aria-hidden="true" />
<span className="sr-only">Loading...</span>
</span>
) : (
<i className="fa fa-search" aria-hidden="true" />
)
}
filterOption={false} filterOption={false}
notFoundContent={null} notFoundContent={null}
value={undefined} value={undefined}
@@ -166,12 +156,7 @@ function PermissionsEditorDialog({ dialog, author, context, aclUrl }) {
/> />
<div className="d-flex align-items-center m-t-5"> <div className="d-flex align-items-center m-t-5">
<h5 className="flex-fill">Users with permissions</h5> <h5 className="flex-fill">Users with permissions</h5>
{loadingGrantees && ( {loadingGrantees && <i className="fa fa-spinner fa-pulse" />}
<span role="status" aria-live="polite" aria-relevant="additions removals">
<i className="fa fa-spinner fa-pulse" aria-hidden="true" />
<span className="sr-only">Loading...</span>
</span>
)}
</div> </div>
<div className="scrollbox p-5" style={{ maxHeight: "40vh" }}> <div className="scrollbox p-5" style={{ maxHeight: "40vh" }}>
<List <List
@@ -184,11 +169,10 @@ function PermissionsEditorDialog({ dialog, author, context, aclUrl }) {
<Tag className="m-0">Author</Tag> <Tag className="m-0">Author</Tag>
) : ( ) : (
<Tooltip title="Remove user permissions"> <Tooltip title="Remove user permissions">
<PlainButton <i
aria-label="Remove permissions" className="fa fa-remove clickable"
onClick={() => removePermission(user.id).then(loadUsersWithPermissions)}> onClick={() => removePermission(user.id).then(loadUsersWithPermissions)}
<i className="fa fa-remove clickable" aria-hidden="true" /> />
</PlainButton>
</Tooltip> </Tooltip>
)} )}
</UserPreviewCard> </UserPreviewCard>

View File

@@ -1,22 +0,0 @@
@import (reference, less) "~@/assets/less/ant";
.plain-button {
all: unset;
transition: all 0.3s cubic-bezier(0.645, 0.045, 0.355, 1);
.@{dropdown-prefix-cls}-menu-item > & {
width: 100%;
margin: -5px -12px;
padding: 5px 12px;
}
.@{menu-prefix-cls}-item > & {
width: 100%;
margin: 0 -16px;
padding: 0 16px;
}
}
.plain-button-link {
.btn-link();
}

View File

@@ -1,20 +0,0 @@
import classNames from "classnames";
import React from "react";
import "./PlainButton.less";
export interface PlainButtonProps extends Omit<React.ButtonHTMLAttributes<HTMLButtonElement>, "type"> {
type?: "link" | "button";
}
function PlainButton({ className, type, ...rest }: PlainButtonProps) {
return (
<button
className={classNames("plain-button", "clickable", { "plain-button-link": type === "link" }, className)}
type="button"
{...rest}
/>
);
}
export default PlainButton;

View File

@@ -1,8 +1,19 @@
import { find, isArray, get, first, map, intersection, isEqual, isEmpty } from "lodash"; import { find, isArray, get, first, map, intersection, isEqual, isEmpty, trim, debounce, isNil } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import SelectWithVirtualScroll from "@/components/SelectWithVirtualScroll"; import SelectWithVirtualScroll from "@/components/SelectWithVirtualScroll";
const SEARCH_DEBOUNCE_TIME = 300;
function filterValuesThatAreNotInOptions(value, options) {
if (isArray(value)) {
const optionValues = map(options, option => option.value);
return intersection(value, optionValues);
}
const found = find(options, option => option.value === value) !== undefined;
return found ? value : get(first(options), "value");
}
export default class QueryBasedParameterInput extends React.Component { export default class QueryBasedParameterInput extends React.Component {
static propTypes = { static propTypes = {
parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types
@@ -28,6 +39,7 @@ export default class QueryBasedParameterInput extends React.Component {
options: [], options: [],
value: null, value: null,
loading: false, loading: false,
currentSearchTerm: null,
}; };
} }
@@ -36,9 +48,10 @@ export default class QueryBasedParameterInput extends React.Component {
} }
componentDidUpdate(prevProps) { componentDidUpdate(prevProps) {
if (this.props.queryId !== prevProps.queryId) { if (this.props.queryId !== prevProps.queryId || this.props.parameter !== prevProps.parameter) {
this._loadOptions(this.props.queryId); this._loadOptions(this.props.queryId);
} }
if (this.props.value !== prevProps.value) { if (this.props.value !== prevProps.value) {
this.setValue(this.props.value); this.setValue(this.props.value);
} }
@@ -46,26 +59,26 @@ export default class QueryBasedParameterInput extends React.Component {
setValue(value) { setValue(value) {
const { options } = this.state; const { options } = this.state;
if (this.props.mode === "multiple") { const { mode, parameter } = this.props;
value = isArray(value) ? value : [value];
const optionValues = map(options, option => option.value); if (mode === "multiple") {
const validValues = intersection(value, optionValues); if (isNil(value)) {
this.setState({ value: validValues }); value = [];
return validValues;
} }
const found = find(options, option => option.value === this.props.value) !== undefined;
value = found ? value : get(first(options), "value"); value = isArray(value) ? value : [value];
}
// parameters with search don't have options available, so we trust what we get
if (!parameter.searchFunction) {
value = filterValuesThatAreNotInOptions(value, options);
}
this.setState({ value }); this.setState({ value });
return value; return value;
} }
async _loadOptions(queryId) { updateOptions(options) {
if (queryId && queryId !== this.state.queryId) {
this.setState({ loading: true });
const options = await this.props.parameter.loadDropdownValues();
// stale queryId check
if (this.props.queryId === queryId) {
this.setState({ options, loading: false }, () => { this.setState({ options, loading: false }, () => {
const updatedValue = this.setValue(this.props.value); const updatedValue = this.setValue(this.props.value);
if (!isEqual(updatedValue, this.props.value)) { if (!isEqual(updatedValue, this.props.value)) {
@@ -73,26 +86,58 @@ export default class QueryBasedParameterInput extends React.Component {
} }
}); });
} }
async _loadOptions(queryId) {
if (queryId && queryId !== this.state.queryId) {
this.setState({ loading: true });
const options = await this.props.parameter.loadDropdownValues(this.state.currentSearchTerm);
// stale queryId check
if (this.props.queryId === queryId) {
this.updateOptions(options);
}
} }
} }
searchFunction = debounce(searchTerm => {
const { parameter } = this.props;
if (parameter.searchFunction && trim(searchTerm)) {
this.setState({ loading: true, currentSearchTerm: searchTerm });
parameter.searchFunction(searchTerm).then(options => {
if (this.state.currentSearchTerm === searchTerm) {
this.updateOptions(options);
}
});
}
}, SEARCH_DEBOUNCE_TIME);
render() { render() {
const { className, mode, onSelect, queryId, value, ...otherProps } = this.props; const { parameter, className, mode, onSelect, queryId, value, ...otherProps } = this.props;
const { loading, options } = this.state; const { loading, options } = this.state;
const selectProps = { ...otherProps };
if (parameter.searchColumn) {
selectProps.filterOption = false;
selectProps.onSearch = this.searchFunction;
selectProps.onChange = value => onSelect(parameter.normalizeValue(value));
selectProps.notFoundContent = null;
selectProps.labelInValue = true;
}
return ( return (
<span> <span>
<SelectWithVirtualScroll <SelectWithVirtualScroll
className={className} className={className}
disabled={loading} disabled={!parameter.searchFunction && loading}
loading={loading} loading={loading}
mode={mode} mode={mode}
value={this.state.value} value={this.state.value || undefined}
onChange={onSelect} onChange={onSelect}
options={map(options, ({ value, name }) => ({ label: String(name), value }))} options={options}
optionFilterProp="children"
showSearch showSearch
showArrow showArrow
notFoundContent={isEmpty(options) ? "No options available" : null} notFoundContent={isEmpty(options) ? "No options available" : null}
{...otherProps} {...selectProps}
/> />
</span> </span>
); );

View File

@@ -21,12 +21,10 @@ function QueryLink({ query, visualization, readOnly }) {
return query.getUrl(false, hash); return query.getUrl(false, hash);
}; };
const QueryLinkWrapper = props => (readOnly ? <span {...props} /> : <Link href={getUrl()} {...props} />);
return ( return (
<QueryLinkWrapper className="query-link"> <Link href={readOnly ? null : getUrl()} className="query-link">
<VisualizationName visualization={visualization} /> <span>{query.name}</span> <VisualizationName visualization={visualization} /> <span>{query.name}</span>
</QueryLinkWrapper> </Link>
); );
} }

View File

@@ -5,7 +5,6 @@ import cx from "classnames";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import Select from "antd/lib/select"; import Select from "antd/lib/select";
import { Query } from "@/services/query"; import { Query } from "@/services/query";
import PlainButton from "@/components/PlainButton";
import notification from "@/services/notification"; import notification from "@/services/notification";
import { QueryTagsControl } from "@/components/tags-control/TagsControl"; import { QueryTagsControl } from "@/components/tags-control/TagsControl";
import useSearchResults from "@/lib/hooks/useSearchResults"; import useSearchResults from "@/lib/hooks/useSearchResults";
@@ -31,21 +30,8 @@ export default function QuerySelector(props) {
const [doSearch, searchResults, searching] = useSearchResults(search, { initialResults: [] }); const [doSearch, searchResults, searching] = useSearchResults(search, { initialResults: [] });
const placeholder = "Search a query by name"; const placeholder = "Search a query by name";
const clearIcon = ( const clearIcon = <i className="fa fa-times hide-in-percy" onClick={() => selectQuery(null)} />;
<i const spinIcon = <i className={cx("fa fa-spinner fa-pulse hide-in-percy", { hidden: !searching })} />;
className="fa fa-times hide-in-percy"
role="button"
tabIndex={0}
aria-label="Clear"
onClick={() => selectQuery(null)}
/>
);
const spinIcon = (
<span role="status" aria-live="polite" aria-relevant="additions removals">
<i className={cx("fa fa-spinner fa-pulse hide-in-percy", { hidden: !searching })} aria-hidden="true" />
<span className="sr-only">Searching...</span>
</span>
);
useEffect(() => { useEffect(() => {
doSearch(searchTerm); doSearch(searchTerm);
@@ -79,25 +65,22 @@ export default function QuerySelector(props) {
} }
return ( return (
<ul className="list-group"> <div className="list-group">
{searchResults.map(q => ( {searchResults.map(q => (
<PlainButton <a
className={cx("query-selector-result", "list-group-item", { inactive: q.is_draft })} className={cx("query-selector-result", "list-group-item", { inactive: q.is_draft })}
key={q.id} key={q.id}
role="listitem"
onClick={() => selectQuery(q.id)} onClick={() => selectQuery(q.id)}
data-test={`QueryId${q.id}`}> data-test={`QueryId${q.id}`}>
{q.name} <QueryTagsControl isDraft={q.is_draft} tags={q.tags} className="inline-tags-control" /> {q.name} <QueryTagsControl isDraft={q.is_draft} tags={q.tags} className="inline-tags-control" />
</PlainButton> </a>
))} ))}
</ul> </div>
); );
} }
if (props.disabled) { if (props.disabled) {
return ( return <Input value={selectedQuery && selectedQuery.name} placeholder={placeholder} disabled />;
<Input value={selectedQuery && selectedQuery.name} aria-label="Tied query" placeholder={placeholder} disabled />
);
} }
if (props.type === "select") { if (props.type === "select") {
@@ -144,12 +127,11 @@ export default function QuerySelector(props) {
return ( return (
<span data-test="QuerySelector"> <span data-test="QuerySelector">
{selectedQuery ? ( {selectedQuery ? (
<Input value={selectedQuery.name} aria-label="Tied query" suffix={clearIcon} readOnly /> <Input value={selectedQuery.name} suffix={clearIcon} readOnly />
) : ( ) : (
<Input <Input
placeholder={placeholder} placeholder={placeholder}
value={searchTerm} value={searchTerm}
aria-label="Tied query"
onChange={e => setSearchTerm(e.target.value)} onChange={e => setSearchTerm(e.target.value)}
suffix={spinIcon} suffix={spinIcon}
/> />

View File

@@ -51,12 +51,9 @@ export default function Resizable({ toggleShortcut, direction, sizeAttribute, ch
const resizeHandle = useMemo( const resizeHandle = useMemo(
() => ( () => (
// eslint-disable-next-line jsx-a11y/click-events-have-key-events, jsx-a11y/no-noninteractive-element-interactions
<span <span
className={`react-resizable-handle react-resizable-handle-${direction}`} className={`react-resizable-handle react-resizable-handle-${direction}`}
role="separator"
onClick={() => { onClick={() => {
// TODO: add key controls
// On desktops resize uses `mousedown`/`mousemove`/`mouseup` events, and there is a conflict // On desktops resize uses `mousedown`/`mousemove`/`mouseup` events, and there is a conflict
// with this `click` handler: after user releases mouse - this handler will be executed. // with this `click` handler: after user releases mouse - this handler will be executed.
// So we use `wasResized` flag to check if there was actual resize or user just pressed and released // So we use `wasResized` flag to check if there was actual resize or user just pressed and released

View File

@@ -12,16 +12,13 @@ import LoadingState from "@/components/items-list/components/LoadingState";
import notification from "@/services/notification"; import notification from "@/services/notification";
import useSearchResults from "@/lib/hooks/useSearchResults"; import useSearchResults from "@/lib/hooks/useSearchResults";
import "./SelectItemsDialog.less";
function ItemsList({ items, renderItem, onItemClick }) { function ItemsList({ items, renderItem, onItemClick }) {
const renderListItem = useCallback( const renderListItem = useCallback(
item => { item => {
const { content, className, isDisabled } = renderItem(item); const { content, className, isDisabled } = renderItem(item);
return ( return (
<List.Item <List.Item
className={classNames("select-items-list", "w-100", "p-l-10", "p-r-10", { disabled: isDisabled }, className)} className={classNames("p-l-10", "p-r-10", { clickable: !isDisabled, disabled: isDisabled }, className)}
onClick={isDisabled ? null : () => onItemClick(item)}> onClick={isDisabled ? null : () => onItemClick(item)}>
{content} {content}
</List.Item> </List.Item>
@@ -120,12 +117,7 @@ function SelectItemsDialog({
}> }>
<div className="d-flex align-items-center m-b-10"> <div className="d-flex align-items-center m-b-10">
<div className="flex-fill"> <div className="flex-fill">
<Input.Search <Input.Search onChange={event => search(event.target.value)} placeholder={inputPlaceholder} autoFocus />
onChange={event => search(event.target.value)}
placeholder={inputPlaceholder}
aria-label={inputPlaceholder}
autoFocus
/>
</div> </div>
{renderStagedItem && ( {renderStagedItem && (
<div className="w-50 m-l-20"> <div className="w-50 m-l-20">

View File

@@ -1,9 +0,0 @@
.select-items-list {
&:hover,
&:focus,
&:focus-within {
color: #555;
background-color: #f5f5f5;
transition: all 150ms ease-in-out;
}
}

View File

@@ -9,7 +9,7 @@ interface VirtualScrollLabeledValue extends LabeledValue {
label: string; label: string;
} }
interface VirtualScrollSelectProps extends Omit<SelectProps<string>, "optionFilterProp" | "children"> { interface VirtualScrollSelectProps extends SelectProps<string> {
options: Array<VirtualScrollLabeledValue>; options: Array<VirtualScrollLabeledValue>;
} }
function SelectWithVirtualScroll({ options, ...props }: VirtualScrollSelectProps): JSX.Element { function SelectWithVirtualScroll({ options, ...props }: VirtualScrollSelectProps): JSX.Element {
@@ -32,14 +32,7 @@ function SelectWithVirtualScroll({ options, ...props }: VirtualScrollSelectProps
return false; return false;
}, [options]); }, [options]);
return ( return <AntdSelect<string> dropdownMatchSelectWidth={dropdownMatchSelectWidth} options={options} {...props} />;
<AntdSelect<string>
dropdownMatchSelectWidth={dropdownMatchSelectWidth}
options={options}
optionFilterProp="label" // as this component expects "options" prop
{...props}
/>
);
} }
export default SelectWithVirtualScroll; export default SelectWithVirtualScroll;

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; @import "~@/assets/less/ant";
.tags-list { .tags-list {
.tags-list-title { .tags-list-title {
@@ -7,14 +7,13 @@
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
.tags-list-label { label {
display: block; display: block;
white-space: nowrap; white-space: nowrap;
margin: 0; margin: 0;
} }
a, a {
.plain-button {
display: block; display: block;
white-space: nowrap; white-space: nowrap;
cursor: pointer; cursor: pointer;
@@ -44,15 +43,5 @@
color: white; color: white;
} }
} }
.ant-menu-item {
&:hover,
&:active,
&:focus,
&:focus-within {
color: @primary-color;
transition: all 0.3s cubic-bezier(0.645, 0.045, 0.355, 1);
}
}
} }
} }

View File

@@ -4,7 +4,6 @@ import Badge from "antd/lib/badge";
import Menu from "antd/lib/menu"; import Menu from "antd/lib/menu";
import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined"; import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined";
import getTags from "@/services/getTags"; import getTags from "@/services/getTags";
import PlainButton from "@/components/PlainButton";
import "./TagsList.less"; import "./TagsList.less";
@@ -78,12 +77,12 @@ function TagsList({ tagsUrl, showUnselectAll = false, onUpdate }: TagsListProps)
return ( return (
<div className="tags-list"> <div className="tags-list">
<div className="tags-list-title"> <div className="tags-list-title">
<span className="tags-list-label">Tags</span> <label>Tags</label>
{showUnselectAll && selectedTags.length > 0 && ( {showUnselectAll && selectedTags.length > 0 && (
<PlainButton type="link" onClick={unselectAll}> <a onClick={unselectAll}>
<CloseOutlinedIcon /> <CloseOutlinedIcon />
clear selection clear selection
</PlainButton> </a>
)} )}
</div> </div>
@@ -91,12 +90,12 @@ function TagsList({ tagsUrl, showUnselectAll = false, onUpdate }: TagsListProps)
<Menu className="invert-stripe-position" mode="inline" selectedKeys={selectedTags}> <Menu className="invert-stripe-position" mode="inline" selectedKeys={selectedTags}>
{map(allTags, tag => ( {map(allTags, tag => (
<Menu.Item key={tag.name} className="m-0"> <Menu.Item key={tag.name} className="m-0">
<PlainButton <a
className="d-flex align-items-center justify-content-between" className="d-flex align-items-center justify-content-between"
onClick={event => toggleTag(event, tag.name)}> onClick={event => toggleTag(event, tag.name)}>
<span className="max-character col-xs-11">{tag.name}</span> <span className="max-character col-xs-11">{tag.name}</span>
<Badge count={tag.count} /> <Badge count={tag.count} />
</PlainButton> </a>
</Menu.Item> </Menu.Item>
))} ))}
</Menu> </Menu>

View File

@@ -4,7 +4,7 @@ import React, { useEffect, useMemo, useState } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import { Moment } from "@/components/proptypes"; import { Moment } from "@/components/proptypes";
import { clientConfig } from "@/services/auth"; import { clientConfig } from "@/services/auth";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
function toMoment(value) { function toMoment(value) {
value = !isNil(value) ? moment(value) : null; value = !isNil(value) ? moment(value) : null;

View File

@@ -1,13 +0,0 @@
import React from "react";
import AntTooltip, { TooltipProps } from "antd/lib/tooltip";
import { isNil } from "lodash";
export default function Tooltip({ title, ...restProps }: TooltipProps) {
const liveTitle = !isNil(title) ? (
<span role="status" aria-live="assertive" aria-relevant="additions">
{title}
</span>
) : null;
return <AntTooltip trigger={["hover", "focus"]} title={liveTitle} {...restProps} />;
}

View File

@@ -35,11 +35,11 @@ CounterCard.defaultProps = {
const queryJobsColumns = [ const queryJobsColumns = [
{ title: "Queue", dataIndex: "origin" }, { title: "Queue", dataIndex: "origin" },
{ title: "Query ID", dataIndex: ["meta", "query_id"] }, { title: "Query ID", dataIndex: "meta.query_id" },
{ title: "Org ID", dataIndex: ["meta", "org_id"] }, { title: "Org ID", dataIndex: "meta.org_id" },
{ title: "Data Source ID", dataIndex: ["meta", "data_source_id"] }, { title: "Data Source ID", dataIndex: "meta.data_source_id" },
{ title: "User ID", dataIndex: ["meta", "user_id"] }, { title: "User ID", dataIndex: "meta.user_id" },
Columns.custom(scheduled => scheduled.toString(), { title: "Scheduled", dataIndex: ["meta", "scheduled"] }), Columns.custom(scheduled => scheduled.toString(), { title: "Scheduled", dataIndex: "meta.scheduled" }),
Columns.timeAgo({ title: "Start Time", dataIndex: "started_at" }), Columns.timeAgo({ title: "Start Time", dataIndex: "started_at" }),
Columns.timeAgo({ title: "Enqueue Time", dataIndex: "enqueued_at" }), Columns.timeAgo({ title: "Enqueue Time", dataIndex: "enqueued_at" }),
]; ];

View File

@@ -1,4 +1,5 @@
@import (reference, less) "~@/assets/less/inc/variables";
@import '../../assets/less/inc/variables';
.visual-card-list { .visual-card-list {
width: 100%; width: 100%;
@@ -6,7 +7,7 @@
} }
.visual-card { .visual-card {
background: #ffffff; background: #FFFFFF;
border: 1px solid fade(@redash-gray, 15%); border: 1px solid fade(@redash-gray, 15%);
border-radius: 3px; border-radius: 3px;
margin: 5px; margin: 5px;
@@ -21,9 +22,7 @@
display: flex; display: flex;
align-items: center; align-items: center;
&:hover, &:hover {
&:focus,
&:focus-within {
box-shadow: rgba(102, 136, 153, 0.15) 0px 4px 9px -3px; box-shadow: rgba(102, 136, 153, 0.15) 0px 4px 9px -3px;
} }

View File

@@ -3,7 +3,6 @@ import PropTypes from "prop-types";
import React, { useState } from "react"; import React, { useState } from "react";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import Link from "@/components/Link"; import Link from "@/components/Link";
import PlainButton from "@/components/PlainButton";
import EmptyState from "@/components/items-list/components/EmptyState"; import EmptyState from "@/components/items-list/components/EmptyState";
import "./CardsList.less"; import "./CardsList.less";
@@ -11,8 +10,8 @@ import "./CardsList.less";
export interface CardsListItem { export interface CardsListItem {
title: string; title: string;
imgSrc: string; imgSrc: string;
onClick?: () => void;
href?: string; href?: string;
onClick?: React.MouseEventHandler<HTMLElement>;
} }
export interface CardsListProps { export interface CardsListProps {
@@ -26,19 +25,12 @@ interface ListItemProps {
} }
function ListItem({ item, keySuffix }: ListItemProps) { function ListItem({ item, keySuffix }: ListItemProps) {
const commonProps = { return (
key: `card${keySuffix}`, <Link key={`card${keySuffix}`} className="visual-card" onClick={item.onClick} href={item.href}>
className: "visual-card",
onClick: item.onClick,
children: (
<>
<img alt={item.title} src={item.imgSrc} /> <img alt={item.title} src={item.imgSrc} />
<h3>{item.title}</h3> <h3>{item.title}</h3>
</> </Link>
), );
};
return item.href ? <Link href={item.href} {...commonProps} /> : <PlainButton type="link" {...commonProps} />;
} }
export default function CardsList({ items = [], showSearch = false }: CardsListProps) { export default function CardsList({ items = [], showSearch = false }: CardsListProps) {
@@ -54,7 +46,6 @@ export default function CardsList({ items = [], showSearch = false }: CardsListP
<div className="col-md-4 col-md-offset-4"> <div className="col-md-4 col-md-offset-4">
<Input.Search <Input.Search
placeholder="Search..." placeholder="Search..."
aria-label="Search cards"
onChange={(e: React.ChangeEvent<HTMLInputElement>) => setSearchText(e.target.value)} onChange={(e: React.ChangeEvent<HTMLInputElement>) => setSearchText(e.target.value)}
autoFocus autoFocus
/> />

View File

@@ -8,15 +8,12 @@ import { MappingType, ParameterMappingListInput } from "@/components/ParameterMa
import QuerySelector from "@/components/QuerySelector"; import QuerySelector from "@/components/QuerySelector";
import notification from "@/services/notification"; import notification from "@/services/notification";
import { Query } from "@/services/query"; import { Query } from "@/services/query";
import { useUniqueId } from "@/lib/hooks/useUniqueId";
function VisualizationSelect({ query, visualization, onChange }) { function VisualizationSelect({ query, visualization, onChange }) {
const visualizationGroups = useMemo(() => { const visualizationGroups = useMemo(() => {
return query ? groupBy(query.visualizations, "type") : {}; return query ? groupBy(query.visualizations, "type") : {};
}, [query]); }, [query]);
const vizSelectId = useUniqueId("visualization-select");
const handleChange = useCallback( const handleChange = useCallback(
visualizationId => { visualizationId => {
const selectedVisualization = query ? find(query.visualizations, { id: visualizationId }) : null; const selectedVisualization = query ? find(query.visualizations, { id: visualizationId }) : null;
@@ -32,9 +29,9 @@ function VisualizationSelect({ query, visualization, onChange }) {
return ( return (
<div> <div>
<div className="form-group"> <div className="form-group">
<label htmlFor={vizSelectId}>Choose Visualization</label> <label htmlFor="choose-visualization">Choose Visualization</label>
<Select <Select
id={vizSelectId} id="choose-visualization"
className="w-100" className="w-100"
value={visualization ? visualization.id : undefined} value={visualization ? visualization.id : undefined}
onChange={handleChange}> onChange={handleChange}>
@@ -111,7 +108,6 @@ function AddWidgetDialog({ dialog, dashboard }) {
}, [dialog, selectedVisualization, parameterMappings]); }, [dialog, selectedVisualization, parameterMappings]);
const existingParams = dashboard.getParametersDefs(); const existingParams = dashboard.getParametersDefs();
const parameterMappingsId = useUniqueId("parameter-mappings");
return ( return (
<Modal <Modal
@@ -136,12 +132,12 @@ function AddWidgetDialog({ dialog, dashboard }) {
)} )}
{parameterMappings.length > 0 && [ {parameterMappings.length > 0 && [
<label key="parameters-title" htmlFor={parameterMappingsId}> <label key="parameters-title" htmlFor="parameter-mappings">
Parameters Parameters
</label>, </label>,
<ParameterMappingListInput <ParameterMappingListInput
key="parameters-list" key="parameters-list"
id={parameterMappingsId} id="parameter-mappings"
mappings={parameterMappings} mappings={parameterMappings}
existingParams={existingParams} existingParams={existingParams}
onChange={setParameterMappings} onChange={setParameterMappings}

View File

@@ -60,7 +60,6 @@ function CreateDashboardDialog({ dialog }) {
onChange={handleNameChange} onChange={handleNameChange}
onPressEnter={save} onPressEnter={save}
placeholder="Dashboard Name" placeholder="Dashboard Name"
aria-label="Dashboard name"
disabled={saveInProgress} disabled={saveInProgress}
autoFocus autoFocus
/> />

View File

@@ -41,7 +41,6 @@ const DashboardWidget = React.memo(
onRefreshWidget, onRefreshWidget,
onRemoveWidget, onRemoveWidget,
onParameterMappingsChange, onParameterMappingsChange,
isEditing,
canEdit, canEdit,
isPublic, isPublic,
isLoading, isLoading,
@@ -58,7 +57,6 @@ const DashboardWidget = React.memo(
widget={widget} widget={widget}
dashboard={dashboard} dashboard={dashboard}
filters={filters} filters={filters}
isEditing={isEditing}
canEdit={canEdit} canEdit={canEdit}
isPublic={isPublic} isPublic={isPublic}
isLoading={isLoading} isLoading={isLoading}
@@ -79,8 +77,7 @@ const DashboardWidget = React.memo(
prevProps.canEdit === nextProps.canEdit && prevProps.canEdit === nextProps.canEdit &&
prevProps.isPublic === nextProps.isPublic && prevProps.isPublic === nextProps.isPublic &&
prevProps.isLoading === nextProps.isLoading && prevProps.isLoading === nextProps.isLoading &&
prevProps.filters === nextProps.filters && prevProps.filters === nextProps.filters
prevProps.isEditing === nextProps.isEditing
); );
class DashboardGrid extends React.Component { class DashboardGrid extends React.Component {
@@ -226,6 +223,7 @@ class DashboardGrid extends React.Component {
}); });
render() { render() {
const className = cx("dashboard-wrapper", this.props.isEditing ? "editing-mode" : "preview-mode");
const { const {
onLoadWidget, onLoadWidget,
onRefreshWidget, onRefreshWidget,
@@ -234,21 +232,19 @@ class DashboardGrid extends React.Component {
filters, filters,
dashboard, dashboard,
isPublic, isPublic,
isEditing,
widgets, widgets,
} = this.props; } = this.props;
const className = cx("dashboard-wrapper", isEditing ? "editing-mode" : "preview-mode");
return ( return (
<div className={className}> <div className={className}>
<ResponsiveGridLayout <ResponsiveGridLayout
draggableCancel="input,.sortable-container" draggableCancel="input"
className={cx("layout", { "disable-animations": this.state.disableAnimations })} className={cx("layout", { "disable-animations": this.state.disableAnimations })}
cols={{ [MULTI]: cfg.columns, [SINGLE]: 1 }} cols={{ [MULTI]: cfg.columns, [SINGLE]: 1 }}
rowHeight={cfg.rowHeight - cfg.margins} rowHeight={cfg.rowHeight - cfg.margins}
margin={[cfg.margins, cfg.margins]} margin={[cfg.margins, cfg.margins]}
isDraggable={isEditing} isDraggable={this.props.isEditing}
isResizable={isEditing} isResizable={this.props.isEditing}
onResizeStart={this.autoHeightCtrl.stop} onResizeStart={this.autoHeightCtrl.stop}
onResizeStop={this.onWidgetResize} onResizeStop={this.onWidgetResize}
layouts={this.state.layouts} layouts={this.state.layouts}
@@ -270,7 +266,6 @@ class DashboardGrid extends React.Component {
filters={filters} filters={filters}
isPublic={isPublic} isPublic={isPublic}
isLoading={widget.loading} isLoading={widget.loading}
isEditing={isEditing}
canEdit={dashboard.canEdit()} canEdit={dashboard.canEdit()}
onLoadWidget={onLoadWidget} onLoadWidget={onLoadWidget}
onRefreshWidget={onRefreshWidget} onRefreshWidget={onRefreshWidget}

View File

@@ -5,7 +5,7 @@ import PropTypes from "prop-types";
import { useDebouncedCallback } from "use-debounce"; import { useDebouncedCallback } from "use-debounce";
import Modal from "antd/lib/modal"; import Modal from "antd/lib/modal";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import Divider from "antd/lib/divider"; import Divider from "antd/lib/divider";
import Link from "@/components/Link"; import Link from "@/components/Link";
import HtmlContent from "@redash/viz/lib/components/HtmlContent"; import HtmlContent from "@redash/viz/lib/components/HtmlContent";
@@ -73,7 +73,6 @@ function TextboxDialog({ dialog, isNew, ...props }) {
className="resize-vertical" className="resize-vertical"
rows="5" rows="5"
value={text} value={text}
aria-label="Textbox widget content"
onChange={handleInputChange} onChange={handleInputChange}
autoFocus autoFocus
placeholder="This is where you write some text" placeholder="This is where you write some text"

View File

@@ -1,6 +1,6 @@
import React, { useState } from "react"; import React, { useState } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import { compact, isEmpty, invoke, map } from "lodash"; import { compact, isEmpty, invoke } from "lodash";
import { markdown } from "markdown"; import { markdown } from "markdown";
import cx from "classnames"; import cx from "classnames";
import Menu from "antd/lib/menu"; import Menu from "antd/lib/menu";
@@ -15,11 +15,9 @@ import Timer from "@/components/Timer";
import { Moment } from "@/components/proptypes"; import { Moment } from "@/components/proptypes";
import QueryLink from "@/components/QueryLink"; import QueryLink from "@/components/QueryLink";
import { FiltersType } from "@/components/Filters"; import { FiltersType } from "@/components/Filters";
import PlainButton from "@/components/PlainButton";
import ExpandedWidgetDialog from "@/components/dashboards/ExpandedWidgetDialog"; import ExpandedWidgetDialog from "@/components/dashboards/ExpandedWidgetDialog";
import EditParameterMappingsDialog from "@/components/dashboards/EditParameterMappingsDialog"; import EditParameterMappingsDialog from "@/components/dashboards/EditParameterMappingsDialog";
import VisualizationRenderer from "@/components/visualizations/VisualizationRenderer"; import VisualizationRenderer from "@/components/visualizations/VisualizationRenderer";
import Widget from "./Widget"; import Widget from "./Widget";
function visualizationWidgetMenuOptions({ widget, canEditDashboard, onParametersEdit }) { function visualizationWidgetMenuOptions({ widget, canEditDashboard, onParametersEdit }) {
@@ -76,8 +74,7 @@ function RefreshIndicator({ refreshStartedAt }) {
return ( return (
<div className="refresh-indicator"> <div className="refresh-indicator">
<div className="refresh-icon"> <div className="refresh-icon">
<i className="zmdi zmdi-refresh zmdi-hc-spin" aria-hidden="true" /> <i className="zmdi zmdi-refresh zmdi-hc-spin" />
<span className="sr-only">Refreshing...</span>
</div> </div>
<Timer from={refreshStartedAt} /> <Timer from={refreshStartedAt} />
</div> </div>
@@ -87,14 +84,7 @@ function RefreshIndicator({ refreshStartedAt }) {
RefreshIndicator.propTypes = { refreshStartedAt: Moment }; RefreshIndicator.propTypes = { refreshStartedAt: Moment };
RefreshIndicator.defaultProps = { refreshStartedAt: null }; RefreshIndicator.defaultProps = { refreshStartedAt: null };
function VisualizationWidgetHeader({ function VisualizationWidgetHeader({ widget, refreshStartedAt, parameters, onParametersUpdate }) {
widget,
refreshStartedAt,
parameters,
isEditing,
onParametersUpdate,
onParametersEdit,
}) {
const canViewQuery = currentUser.hasPermission("view_query"); const canViewQuery = currentUser.hasPermission("view_query");
return ( return (
@@ -114,13 +104,7 @@ function VisualizationWidgetHeader({
</div> </div>
{!isEmpty(parameters) && ( {!isEmpty(parameters) && (
<div className="m-b-10"> <div className="m-b-10">
<Parameters <Parameters parameters={parameters} onValuesChange={onParametersUpdate} />
parameters={parameters}
sortable={isEditing}
appendSortableToParent={false}
onValuesChange={onParametersUpdate}
onParametersEdit={onParametersEdit}
/>
</div> </div>
)} )}
</> </>
@@ -131,16 +115,12 @@ VisualizationWidgetHeader.propTypes = {
widget: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types widget: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
refreshStartedAt: Moment, refreshStartedAt: Moment,
parameters: PropTypes.arrayOf(PropTypes.object), parameters: PropTypes.arrayOf(PropTypes.object),
isEditing: PropTypes.bool,
onParametersUpdate: PropTypes.func, onParametersUpdate: PropTypes.func,
onParametersEdit: PropTypes.func,
}; };
VisualizationWidgetHeader.defaultProps = { VisualizationWidgetHeader.defaultProps = {
refreshStartedAt: null, refreshStartedAt: null,
onParametersUpdate: () => {}, onParametersUpdate: () => {},
onParametersEdit: () => {},
isEditing: false,
parameters: [], parameters: [],
}; };
@@ -160,40 +140,34 @@ function VisualizationWidgetFooter({ widget, isPublic, onRefresh, onExpand }) {
<> <>
<span> <span>
{!isPublic && !!widgetQueryResult && ( {!isPublic && !!widgetQueryResult && (
<PlainButton <a
className="refresh-button hidden-print btn btn-sm btn-default btn-transparent" className="refresh-button hidden-print btn btn-sm btn-default btn-transparent"
onClick={() => refreshWidget(1)} onClick={() => refreshWidget(1)}
data-test="RefreshButton"> data-test="RefreshButton">
<i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": refreshClickButtonId === 1 })} aria-hidden="true" /> <i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": refreshClickButtonId === 1 })} />{" "}
<span className="sr-only">
{refreshClickButtonId === 1 ? "Refreshing, please wait. " : "Press to refresh. "}
</span>{" "}
<TimeAgo date={updatedAt} /> <TimeAgo date={updatedAt} />
</PlainButton> </a>
)} )}
<span className="visible-print"> <span className="visible-print">
<i className="zmdi zmdi-time-restore" aria-hidden="true" /> {formatDateTime(updatedAt)} <i className="zmdi zmdi-time-restore" /> {formatDateTime(updatedAt)}
</span> </span>
{isPublic && ( {isPublic && (
<span className="small hidden-print"> <span className="small hidden-print">
<i className="zmdi zmdi-time-restore" aria-hidden="true" /> <TimeAgo date={updatedAt} /> <i className="zmdi zmdi-time-restore" /> <TimeAgo date={updatedAt} />
</span> </span>
)} )}
</span> </span>
<span> <span>
{!isPublic && ( {!isPublic && (
<PlainButton <a
className="btn btn-sm btn-default hidden-print btn-transparent btn__refresh" className="btn btn-sm btn-default hidden-print btn-transparent btn__refresh"
onClick={() => refreshWidget(2)}> onClick={() => refreshWidget(2)}>
<i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": refreshClickButtonId === 2 })} aria-hidden="true" /> <i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": refreshClickButtonId === 2 })} />
<span className="sr-only"> </a>
{refreshClickButtonId === 2 ? "Refreshing, please wait." : "Press to refresh."}
</span>
</PlainButton>
)} )}
<PlainButton className="btn btn-sm btn-default hidden-print btn-transparent btn__refresh" onClick={onExpand}> <a className="btn btn-sm btn-default hidden-print btn-transparent btn__refresh" onClick={onExpand}>
<i className="zmdi zmdi-fullscreen" aria-hidden="true" /> <i className="zmdi zmdi-fullscreen" />
</PlainButton> </a>
</span> </span>
</> </>
) : null; ) : null;
@@ -216,7 +190,6 @@ class VisualizationWidget extends React.Component {
isPublic: PropTypes.bool, isPublic: PropTypes.bool,
isLoading: PropTypes.bool, isLoading: PropTypes.bool,
canEdit: PropTypes.bool, canEdit: PropTypes.bool,
isEditing: PropTypes.bool,
onLoad: PropTypes.func, onLoad: PropTypes.func,
onRefresh: PropTypes.func, onRefresh: PropTypes.func,
onDelete: PropTypes.func, onDelete: PropTypes.func,
@@ -228,7 +201,6 @@ class VisualizationWidget extends React.Component {
isPublic: false, isPublic: false,
isLoading: false, isLoading: false,
canEdit: false, canEdit: false,
isEditing: false,
onLoad: () => {}, onLoad: () => {},
onRefresh: () => {}, onRefresh: () => {},
onDelete: () => {}, onDelete: () => {},
@@ -302,14 +274,9 @@ class VisualizationWidget extends React.Component {
); );
default: default:
return ( return (
<div <div className="body-row-auto spinner-container">
className="body-row-auto spinner-container"
role="status"
aria-live="polite"
aria-relevant="additions removals">
<div className="spinner"> <div className="spinner">
<i className="zmdi zmdi-refresh zmdi-hc-spin zmdi-hc-5x" aria-hidden="true" /> <i className="zmdi zmdi-refresh zmdi-hc-spin zmdi-hc-5x" />
<span className="sr-only">Loading...</span>
</div> </div>
</div> </div>
); );
@@ -317,15 +284,10 @@ class VisualizationWidget extends React.Component {
} }
render() { render() {
const { widget, isLoading, isPublic, canEdit, isEditing, onRefresh } = this.props; const { widget, isLoading, isPublic, canEdit, onRefresh } = this.props;
const { localParameters } = this.state; const { localParameters } = this.state;
const widgetQueryResult = widget.getQueryResult(); const widgetQueryResult = widget.getQueryResult();
const isRefreshing = isLoading && !!(widgetQueryResult && widgetQueryResult.getStatus()); const isRefreshing = isLoading && !!(widgetQueryResult && widgetQueryResult.getStatus());
const onParametersEdit = parameters => {
const paramOrder = map(parameters, "name");
widget.options.paramOrder = paramOrder;
widget.save("options", { paramOrder });
};
return ( return (
<Widget <Widget
@@ -341,9 +303,7 @@ class VisualizationWidget extends React.Component {
widget={widget} widget={widget}
refreshStartedAt={isRefreshing ? widget.refreshStartedAt : null} refreshStartedAt={isRefreshing ? widget.refreshStartedAt : null}
parameters={localParameters} parameters={localParameters}
isEditing={isEditing}
onParametersUpdate={onRefresh} onParametersUpdate={onRefresh}
onParametersEdit={onParametersEdit}
/> />
} }
footer={ footer={

View File

@@ -7,7 +7,6 @@ import Modal from "antd/lib/modal";
import Menu from "antd/lib/menu"; import Menu from "antd/lib/menu";
import recordEvent from "@/services/recordEvent"; import recordEvent from "@/services/recordEvent";
import { Moment } from "@/components/proptypes"; import { Moment } from "@/components/proptypes";
import PlainButton from "@/components/PlainButton";
import "./Widget.less"; import "./Widget.less";
@@ -23,9 +22,9 @@ function WidgetDropdownButton({ extraOptions, showDeleteOption, onDelete }) {
return ( return (
<div className="widget-menu-regular"> <div className="widget-menu-regular">
<Dropdown overlay={WidgetMenu} placement="bottomRight" trigger={["click"]}> <Dropdown overlay={WidgetMenu} placement="bottomRight" trigger={["click"]}>
<PlainButton className="action p-l-15 p-r-15" data-test="WidgetDropdownButton" aria-label="More options"> <a className="action p-l-15 p-r-15" data-test="WidgetDropdownButton">
<i className="zmdi zmdi-more-vert" aria-hidden="true" /> <i className="zmdi zmdi-more-vert" />
</PlainButton> </a>
</Dropdown> </Dropdown>
</div> </div>
); );
@@ -46,14 +45,9 @@ WidgetDropdownButton.defaultProps = {
function WidgetDeleteButton({ onClick }) { function WidgetDeleteButton({ onClick }) {
return ( return (
<div className="widget-menu-remove"> <div className="widget-menu-remove">
<PlainButton <a className="action" title="Remove From Dashboard" onClick={onClick} data-test="WidgetDeleteButton">
className="action" <i className="zmdi zmdi-close" />
title="Remove From Dashboard" </a>
onClick={onClick}
data-test="WidgetDeleteButton"
aria-label="Close">
<i className="zmdi zmdi-close" aria-hidden="true" />
</PlainButton>
</div> </div>
); );
} }

View File

@@ -1,4 +1,12 @@
@import (reference, less) "~@/assets/less/inc/variables"; @import "../../../assets/less/inc/variables";
.tile .t-header .th-title a.query-link {
color: rgba(0, 0, 0, 0.5);
}
.th-title p.hidden-print {
margin-bottom: 0;
}
.widget-wrapper { .widget-wrapper {
.widget-actions { .widget-actions {
@@ -14,20 +22,11 @@
line-height: 100%; line-height: 100%;
display: block; display: block;
padding: 4px 10px 3px; padding: 4px 10px 3px;
}
&:focus { .action:hover {
background-color: rgba(0, 0, 0, 0.1); background-color: rgba(0, 0, 0, 0.1);
} }
&:hover {
background-color: transparent;
color: @blue;
}
&:active {
filter: brightness(75%);
}
}
} }
.parameter-container { .parameter-container {
@@ -84,7 +83,7 @@
display: block; display: block;
} }
.query-link { a.query-link {
pointer-events: none; pointer-events: none;
cursor: move; cursor: move;
} }
@@ -191,18 +190,10 @@
.th-title { .th-title {
padding-right: 23px; // no overlap on RefreshIndicator padding-right: 23px; // no overlap on RefreshIndicator
.hidden-print { a {
margin-bottom: 0;
}
.query-link {
color: fade(@redash-black, 80%); color: fade(@redash-black, 80%);
font-size: 15px; font-size: 15px;
font-weight: 500; font-weight: 500;
&:not(.visualization-name) {
color: fade(@redash-black, 50%);
}
} }
} }
@@ -221,10 +212,7 @@
padding: 15px; padding: 15px;
} }
&:hover, &:hover {
&:focus,
&:active,
&:focus-within {
.widget-menu-regular, .widget-menu-regular,
.btn__refresh { .btn__refresh {
opacity: 1 !important; opacity: 1 !important;
@@ -252,12 +240,10 @@
} }
} }
a, a {
.plain-button {
color: fade(@redash-black, 65%); color: fade(@redash-black, 65%);
&:hover, &:hover {
&:focus {
color: fade(@redash-black, 95%); color: fade(@redash-black, 95%);
} }
} }

View File

@@ -201,10 +201,7 @@ export default function DynamicForm({
className="extra-options-button" className="extra-options-button"
onClick={() => setShowExtraFields(currentShowExtraFields => !currentShowExtraFields)}> onClick={() => setShowExtraFields(currentShowExtraFields => !currentShowExtraFields)}>
Additional Settings Additional Settings
<i <i className={cx("fa m-l-5", { "fa-caret-up": showExtraFields, "fa-caret-down": !showExtraFields })} />
className={cx("fa m-l-5", { "fa-caret-up": showExtraFields, "fa-caret-down": !showExtraFields })}
aria-hidden="true"
/>
</Button> </Button>
<Collapse collapsed={!showExtraFields} className="extra-options-content"> <Collapse collapsed={!showExtraFields} className="extra-options-content">
<DynamicFormFields fields={extraFields} feedbackIcons={feedbackIcons} form={form} /> <DynamicFormFields fields={extraFields} feedbackIcons={feedbackIcons} form={form} />

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; @import "~@/assets/less/ant";
@btn-extra-options-bg: fade(@redash-gray, 10%); @btn-extra-options-bg: fade(@redash-gray, 10%);
@btn-extra-options-border: fade(@redash-gray, 15%); @btn-extra-options-border: fade(@redash-gray, 15%);

View File

@@ -23,13 +23,7 @@ const DYNAMIC_DATE_OPTIONS = [
]; ];
function DateParameter(props) { function DateParameter(props) {
return ( return <DynamicDatePicker dynamicButtonOptions={{ options: DYNAMIC_DATE_OPTIONS }} {...props} />;
<DynamicDatePicker
dynamicButtonOptions={{ options: DYNAMIC_DATE_OPTIONS }}
{...props}
dateOptions={{ "aria-label": "Parameter date value" }}
/>
);
} }
DateParameter.propTypes = { DateParameter.propTypes = {

View File

@@ -42,7 +42,7 @@ function DynamicButton({ options, selectedDynamicValue, onSelect, enabled, stati
return ( return (
<div ref={containerRef}> <div ref={containerRef}>
<div role="presentation" onClick={e => e.stopPropagation()}> <a onClick={e => e.stopPropagation()}>
<Dropdown.Button <Dropdown.Button
overlay={menu} overlay={menu}
className="dynamic-button" className="dynamic-button"
@@ -58,7 +58,7 @@ function DynamicButton({ options, selectedDynamicValue, onSelect, enabled, stati
getPopupContainer={() => containerRef.current} getPopupContainer={() => containerRef.current}
data-test="DynamicButton" data-test="DynamicButton"
/> />
</div> </a>
</div> </div>
); );
} }

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/inc/variables"; @import "../../assets/less/inc/variables";
.date-range-parameter, .date-range-parameter,
.date-parameter { .date-parameter {

View File

@@ -4,24 +4,23 @@ import PropTypes from "prop-types";
import classNames from "classnames"; import classNames from "classnames";
import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined"; import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined";
import Link from "@/components/Link"; import Link from "@/components/Link";
import PlainButton from "@/components/PlainButton";
import CreateDashboardDialog from "@/components/dashboards/CreateDashboardDialog"; import CreateDashboardDialog from "@/components/dashboards/CreateDashboardDialog";
import HelpTrigger from "@/components/HelpTrigger"; import HelpTrigger from "@/components/HelpTrigger";
import { currentUser } from "@/services/auth"; import { currentUser } from "@/services/auth";
import organizationStatus from "@/services/organizationStatus"; import organizationStatus from "@/services/organizationStatus";
import "./empty-state.less"; import "./empty-state.less";
export function Step({ show, completed, text, url, urlText, onClick }) { export function Step({ show, completed, text, url, urlTarget, urlText, onClick }) {
if (!show) { if (!show) {
return null; return null;
} }
const commonProps = { children: urlText, onClick };
return ( return (
<li className={classNames({ done: completed })}> <li className={classNames({ done: completed })}>
{url ? <Link href={url} {...commonProps} /> : <PlainButton type="link" {...commonProps} />} {text} <Link href={url} onClick={onClick} target={urlTarget}>
{urlText}
</Link>{" "}
{text}
</li> </li>
); );
} }
@@ -189,7 +188,7 @@ function EmptyState({
<div className="empty-state__summary"> <div className="empty-state__summary">
{header && <h4>{header}</h4>} {header && <h4>{header}</h4>}
<h2> <h2>
<i className={icon} aria-hidden="true" /> <i className={icon} />
</h2> </h2>
<p>{description}</p> <p>{description}</p>
<img src={imageSource} alt={illustration + " Illustration"} width="75%" /> <img src={imageSource} alt={illustration + " Illustration"} width="75%" />
@@ -201,9 +200,9 @@ function EmptyState({
</div> </div>
</div> </div>
{closable && ( {closable && (
<PlainButton className="close-button" aria-label="Close" onClick={onClose}> <a className="close-button" onClick={onClose}>
<CloseOutlinedIcon /> <CloseOutlinedIcon />
</PlainButton> </a>
)} )}
</div> </div>
); );

View File

@@ -3,7 +3,7 @@
// Empty states // Empty states
.empty-state { .empty-state {
width: 100%; width: 100%;
margin: 0 auto 10px; margin: 0px auto 10px;
display: flex; display: flex;
flex-direction: row; flex-direction: row;
justify-content: space-between; justify-content: space-between;
@@ -18,7 +18,7 @@
} }
.empty-state__steps { .empty-state__steps {
padding-left: 0; padding-left: 0px;
} }
.empty-state__summary { .empty-state__summary {
@@ -86,13 +86,8 @@
cursor: pointer; cursor: pointer;
transition: color @animation-duration-slow; transition: color @animation-duration-slow;
&:hover, &:hover {
&:focus {
color: @text-color; color: @text-color;
} }
&:active {
filter: contrast(200%);
}
} }
} }

View File

@@ -28,7 +28,6 @@ class CreateGroupDialog extends React.Component {
onChange={event => this.setState({ name: event.target.value })} onChange={event => this.setState({ name: event.target.value })}
onPressEnter={() => this.save()} onPressEnter={() => this.save()}
placeholder="Group Name" placeholder="Group Name"
aria-label="Group name"
autoFocus autoFocus
/> />
</Modal> </Modal>

View File

@@ -3,7 +3,7 @@ import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import Modal from "antd/lib/modal"; import Modal from "antd/lib/modal";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import notification from "@/services/notification"; import notification from "@/services/notification";
import Group from "@/services/group"; import Group from "@/services/group";

View File

@@ -26,13 +26,13 @@ export default function DetailsPageSidebar({
<Sidebar.Menu items={items} selected={controller.params.currentPage} /> <Sidebar.Menu items={items} selected={controller.params.currentPage} />
{canAddMembers && ( {canAddMembers && (
<Button className="w-100 m-t-5" type="primary" onClick={onAddMembersClick}> <Button className="w-100 m-t-5" type="primary" onClick={onAddMembersClick}>
<i className="fa fa-plus m-r-5" aria-hidden="true" /> <i className="fa fa-plus m-r-5" />
Add Members Add Members
</Button> </Button>
)} )}
{canAddDataSources && ( {canAddDataSources && (
<Button className="w-100 m-t-5" type="primary" onClick={onAddDataSourcesClick}> <Button className="w-100 m-t-5" type="primary" onClick={onAddDataSourcesClick}>
<i className="fa fa-plus m-r-5" aria-hidden="true" /> <i className="fa fa-plus m-r-5" />
Add Data Sources Add Data Sources
</Button> </Button>
)} )}

View File

@@ -1,38 +1,19 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
export default function ListItemAddon({ isSelected, isStaged, alreadyInGroup, deselectedIcon }) { export default function ListItemAddon({ isSelected, isStaged, alreadyInGroup, deselectedIcon }) {
if (isStaged) { if (isStaged) {
return ( return <i className="fa fa-remove" />;
<>
<i className="fa fa-remove" aria-hidden="true" />
<span className="sr-only">Remove</span>
</>
);
} }
if (alreadyInGroup) { if (alreadyInGroup) {
return ( return (
<Tooltip title="Already selected"> <Tooltip title="Already selected">
{/* eslint-disable-next-line jsx-a11y/no-noninteractive-tabindex */} <i className="fa fa-check" />
<span tabIndex={0}>
<i className="fa fa-check" aria-hidden="true" />
<span className="sr-only">Already selected</span>
</span>
</Tooltip> </Tooltip>
); );
} }
return isSelected ? ( return isSelected ? <i className="fa fa-check" /> : <i className={`fa ${deselectedIcon}`} />;
<>
<i className="fa fa-check" aria-hidden="true" />
<span className="sr-only">Selected</span>
</>
) : (
<>
<i className={`fa ${deselectedIcon}`} aria-hidden="true" />
<span className="sr-only">Select</span>
</>
);
} }
ListItemAddon.propTypes = { ListItemAddon.propTypes = {

View File

@@ -10,7 +10,7 @@ import TagsList from "@/components/TagsList";
SearchInput SearchInput
*/ */
export function SearchInput({ placeholder, value, showIcon, onChange, label }) { export function SearchInput({ placeholder, value, showIcon, onChange }) {
const [currentValue, setCurrentValue] = useState(value); const [currentValue, setCurrentValue] = useState(value);
useEffect(() => { useEffect(() => {
@@ -29,29 +29,21 @@ export function SearchInput({ placeholder, value, showIcon, onChange, label }) {
const InputControl = showIcon ? Input.Search : Input; const InputControl = showIcon ? Input.Search : Input;
return ( return (
<div className="m-b-10"> <div className="m-b-10">
<InputControl <InputControl className="form-control" placeholder={placeholder} value={currentValue} onChange={onInputChange} />
className="form-control"
placeholder={placeholder}
value={currentValue}
aria-label={label}
onChange={onInputChange}
/>
</div> </div>
); );
} }
SearchInput.propTypes = { SearchInput.propTypes = {
value: PropTypes.string.isRequired,
placeholder: PropTypes.string, placeholder: PropTypes.string,
value: PropTypes.string.isRequired,
showIcon: PropTypes.bool, showIcon: PropTypes.bool,
onChange: PropTypes.func.isRequired, onChange: PropTypes.func.isRequired,
label: PropTypes.string,
}; };
SearchInput.defaultProps = { SearchInput.defaultProps = {
placeholder: "Search...", placeholder: "Search...",
showIcon: false, showIcon: false,
label: "Search",
}; };
/* /*
@@ -70,7 +62,7 @@ export function Menu({ items, selected }) {
<AntdMenu.Item key={item.key} className="m-0"> <AntdMenu.Item key={item.key} className="m-0">
<Link href={item.href}> <Link href={item.href}>
{isString(item.icon) && item.icon !== "" && ( {isString(item.icon) && item.icon !== "" && (
<span className="btn-favorite m-r-5"> <span className="btn-favourite m-r-5">
<i className={item.icon} aria-hidden="true" /> <i className={item.icon} aria-hidden="true" />
</span> </span>
)} )}
@@ -108,7 +100,7 @@ Menu.defaultProps = {
export function MenuIcon({ icon }) { export function MenuIcon({ icon }) {
return ( return (
<span className="btn-favorite m-r-5"> <span className="btn-favourite m-r-5">
<i className={icon} aria-hidden="true" /> <i className={icon} aria-hidden="true" />
</span> </span>
); );

View File

@@ -5,7 +5,6 @@ import Modal from "antd/lib/modal";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import List from "antd/lib/list"; import List from "antd/lib/list";
import Link from "@/components/Link"; import Link from "@/components/Link";
import PlainButton from "@/components/PlainButton";
import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined"; import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined";
import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper"; import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper";
import { QueryTagsControl } from "@/components/tags-control/TagsControl"; import { QueryTagsControl } from "@/components/tags-control/TagsControl";
@@ -90,9 +89,7 @@ function AddToDashboardDialog({ dialog, visualization }) {
value={searchTerm} value={searchTerm}
onChange={event => setSearchTerm(event.target.value)} onChange={event => setSearchTerm(event.target.value)}
suffix={ suffix={
<PlainButton className={searchTerm === "" ? "hidden" : null} onClick={() => setSearchTerm("")}> <CloseOutlinedIcon className={searchTerm === "" ? "hidden" : null} onClick={() => setSearchTerm("")} />
<CloseOutlinedIcon />
</PlainButton>
} }
/> />
)} )}
@@ -107,15 +104,7 @@ function AddToDashboardDialog({ dialog, visualization }) {
renderItem={d => ( renderItem={d => (
<List.Item <List.Item
key={`dashboard-${d.id}`} key={`dashboard-${d.id}`}
actions={ actions={selectedDashboard ? [<CloseOutlinedIcon onClick={() => setSelectedDashboard(null)} />] : []}
selectedDashboard
? [
<PlainButton onClick={() => setSelectedDashboard(null)}>
<CloseOutlinedIcon />
</PlainButton>,
]
: []
}
onClick={selectedDashboard ? null : () => setSelectedDashboard(d)}> onClick={selectedDashboard ? null : () => setSelectedDashboard(d)}>
<div className="add-to-dashboard-dialog-item-content"> <div className="add-to-dashboard-dialog-item-content">
{d.name} {d.name}

View File

@@ -9,7 +9,6 @@ import CodeBlock from "@/components/CodeBlock";
import { axios } from "@/services/axios"; import { axios } from "@/services/axios";
import { clientConfig } from "@/services/auth"; import { clientConfig } from "@/services/auth";
import notification from "@/services/notification"; import notification from "@/services/notification";
import { useUniqueId } from "@/lib/hooks/useUniqueId";
import "./index.less"; import "./index.less";
import { policy } from "@/services/policy"; import { policy } from "@/services/policy";
@@ -40,16 +39,13 @@ function ApiKeyDialog({ dialog, ...props }) {
[query.id, query.api_key] [query.id, query.api_key]
); );
const csvResultsLabelId = useUniqueId("csv-results-label");
const jsonResultsLabelId = useUniqueId("json-results-label");
return ( return (
<Modal {...dialog.props} width={600} footer={<Button onClick={() => dialog.close(query)}>Close</Button>}> <Modal {...dialog.props} width={600} footer={<Button onClick={() => dialog.close(query)}>Close</Button>}>
<div className="query-api-key-dialog-wrapper"> <div className="query-api-key-dialog-wrapper">
<h5>API Key</h5> <h5>API Key</h5>
<div className="m-b-20"> <div className="m-b-20">
<Input.Group compact> <Input.Group compact>
<Input readOnly value={query.api_key} aria-label="Query API Key" /> <Input readOnly value={query.api_key} />
{policy.canEdit(query) && ( {policy.canEdit(query) && (
<Button disabled={updatingApiKey} loading={updatingApiKey} onClick={regenerateQueryApiKey}> <Button disabled={updatingApiKey} loading={updatingApiKey} onClick={regenerateQueryApiKey}>
Regenerate Regenerate
@@ -60,16 +56,12 @@ function ApiKeyDialog({ dialog, ...props }) {
<h5>Example API Calls:</h5> <h5>Example API Calls:</h5>
<div className="m-b-10"> <div className="m-b-10">
<span id={csvResultsLabelId}>Results in CSV format:</span> <label>Results in CSV format:</label>
<CodeBlock aria-labelledby={csvResultsLabelId} copyable> <CodeBlock copyable>{csvUrl}</CodeBlock>
{csvUrl}
</CodeBlock>
</div> </div>
<div> <div>
<span id={jsonResultsLabelId}>Results in JSON format:</span> <label>Results in JSON format:</label>
<CodeBlock aria-labelledby={jsonResultsLabelId} copyable> <CodeBlock copyable>{jsonUrl}</CodeBlock>
{jsonUrl}
</CodeBlock>
</div> </div>
</div> </div>
</Modal> </Modal>

View File

@@ -1,4 +1,3 @@
import { uniqueId } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Alert from "antd/lib/alert"; import Alert from "antd/lib/alert";
@@ -10,7 +9,6 @@ import Modal from "antd/lib/modal";
import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper"; import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper";
import { clientConfig } from "@/services/auth"; import { clientConfig } from "@/services/auth";
import CodeBlock from "@/components/CodeBlock"; import CodeBlock from "@/components/CodeBlock";
import "./EmbedQueryDialog.less"; import "./EmbedQueryDialog.less";
class EmbedQueryDialog extends React.Component { class EmbedQueryDialog extends React.Component {
@@ -38,9 +36,6 @@ class EmbedQueryDialog extends React.Component {
} }
} }
urlEmbedLabelId = uniqueId("url-embed-label");
iframeEmbedLabelId = uniqueId("iframe-embed-label");
render() { render() {
const { query, dialog } = this.props; const { query, dialog } = this.props;
const { enableChangeIframeSize, iframeWidth, iframeHeight } = this.state; const { enableChangeIframeSize, iframeWidth, iframeHeight } = this.state;
@@ -53,19 +48,15 @@ class EmbedQueryDialog extends React.Component {
footer={<Button onClick={dialog.dismiss}>Close</Button>}> footer={<Button onClick={dialog.dismiss}>Close</Button>}>
{query.is_safe ? ( {query.is_safe ? (
<React.Fragment> <React.Fragment>
<h5 id={this.urlEmbedLabelId} className="m-t-0"> <h5 className="m-t-0">Public URL</h5>
Public URL
</h5>
<div className="m-b-30"> <div className="m-b-30">
<CodeBlock aria-labelledby={this.urlEmbedLabelId} data-test="EmbedIframe" copyable> <CodeBlock data-test="EmbedIframe" copyable>
{this.embedUrl} {this.embedUrl}
</CodeBlock> </CodeBlock>
</div> </div>
<h5 id={this.iframeEmbedLabelId} className="m-t-0"> <h5 className="m-t-0">IFrame Embed</h5>
IFrame Embed
</h5>
<div> <div>
<CodeBlock aria-labelledby={this.iframeEmbedLabelId} copyable> <CodeBlock copyable>
{`<iframe src="${this.embedUrl}" width="${iframeWidth}" height="${iframeHeight}"></iframe>`} {`<iframe src="${this.embedUrl}" width="${iframeWidth}" height="${iframeHeight}"></iframe>`}
</CodeBlock> </CodeBlock>
<Form className="m-t-10" layout="inline"> <Form className="m-t-10" layout="inline">

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; @import '~antd/lib/button/style/index';
.embed-query-dialog { .embed-query-dialog {
label { label {

View File

@@ -2,7 +2,7 @@ import React, { useCallback } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import recordEvent from "@/services/recordEvent"; import recordEvent from "@/services/recordEvent";
import Checkbox from "antd/lib/checkbox"; import Checkbox from "antd/lib/checkbox";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
export default function AutoLimitCheckbox({ available, checked, onChange }) { export default function AutoLimitCheckbox({ available, checked, onChange }) {
const handleClick = useCallback(() => { const handleClick = useCallback(() => {

View File

@@ -1,5 +1,5 @@
import React, { useCallback } from "react"; import React, { useCallback } from "react";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import "@/redash-font/style.less"; import "@/redash-font/style.less";
@@ -25,12 +25,8 @@ export default function AutocompleteToggle({ available, enabled, onToggle }) {
return ( return (
<Tooltip placement="top" title={tooltipMessage}> <Tooltip placement="top" title={tooltipMessage}>
<Button <Button className="query-editor-controls-button m-r-5" disabled={!available} onClick={handleClick}>
className="query-editor-controls-button m-r-5" <i className={"icon " + icon} />
disabled={!available}
onClick={handleClick}
aria-label={enabled ? "Disable live autocomplete" : "Enable live autocomplete"}>
<i className={"icon " + icon} aria-hidden="true" />
</Button> </Button>
</Tooltip> </Tooltip>
); );

View File

@@ -1,7 +1,7 @@
import { isFunction, map, filter, fromPairs, noop } from "lodash"; import { isFunction, map, filter, fromPairs, noop } from "lodash";
import React, { useEffect } from "react"; import React, { useEffect } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import Select from "antd/lib/select"; import Select from "antd/lib/select";
import KeyboardShortcuts, { humanReadableShortcut } from "@/services/KeyboardShortcuts"; import KeyboardShortcuts, { humanReadableShortcut } from "@/services/KeyboardShortcuts";

View File

@@ -2,7 +2,6 @@ import React, { useEffect, useMemo, useState, useCallback, useImperativeHandle }
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import cx from "classnames"; import cx from "classnames";
import { AceEditor, snippetsModule, updateSchemaCompleter } from "./ace"; import { AceEditor, snippetsModule, updateSchemaCompleter } from "./ace";
import { srNotify } from "@/lib/accessibility";
import { SchemaItemType } from "@/components/queries/SchemaBrowser"; import { SchemaItemType } from "@/components/queries/SchemaBrowser";
import resizeObserver from "@/services/resizeObserver"; import resizeObserver from "@/services/resizeObserver";
import QuerySnippet from "@/services/query-snippet"; import QuerySnippet from "@/services/query-snippet";
@@ -90,25 +89,6 @@ const QueryEditor = React.forwardRef(function(
// Lineup only mac // Lineup only mac
editor.commands.bindKey({ win: null, mac: "Ctrl+P" }, "golineup"); editor.commands.bindKey({ win: null, mac: "Ctrl+P" }, "golineup");
// Esc for exiting
editor.commands.bindKey({ win: "Esc", mac: "Esc" }, () => {
editor.blur();
});
let notificationCleanup = null;
editor.on("focus", () => {
notificationCleanup = srNotify({
text: "You've entered the SQL editor. To exit press the ESC key.",
politeness: "assertive",
});
});
editor.on("blur", () => {
if (notificationCleanup) {
notificationCleanup();
}
});
// Reset Completer in case dot is pressed // Reset Completer in case dot is pressed
editor.commands.on("afterExec", e => { editor.commands.on("afterExec", e => {
if (e.command.name === "insertstring" && e.args === "." && editor.completer) { if (e.command.name === "insertstring" && e.args === "." && editor.completer) {

View File

@@ -1,7 +1,6 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import PlainButton from "@/components/PlainButton";
import { localizeTime, durationHumanize } from "@/lib/utils"; import { localizeTime, durationHumanize } from "@/lib/utils";
import { RefreshScheduleType, RefreshScheduleDefault } from "../proptypes"; import { RefreshScheduleType, RefreshScheduleDefault } from "../proptypes";
@@ -52,9 +51,9 @@ export default class SchedulePhrase extends React.Component {
const content = full ? <Tooltip title={full}>{short}</Tooltip> : short; const content = full ? <Tooltip title={full}>{short}</Tooltip> : short;
return this.props.isLink ? ( return this.props.isLink ? (
<PlainButton type="link" className="schedule-phrase" onClick={this.props.onClick} data-test="EditSchedule"> <a className="schedule-phrase" onClick={this.props.onClick} data-test="EditSchedule">
{content} {content}
</PlainButton> </a>
) : ( ) : (
content content
); );

View File

@@ -5,10 +5,9 @@ import PropTypes from "prop-types";
import { useDebouncedCallback } from "use-debounce"; import { useDebouncedCallback } from "use-debounce";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import Button from "antd/lib/button"; import Button from "antd/lib/button";
import Tooltip from "antd/lib/tooltip";
import AutoSizer from "react-virtualized/dist/commonjs/AutoSizer"; import AutoSizer from "react-virtualized/dist/commonjs/AutoSizer";
import List from "react-virtualized/dist/commonjs/List"; import List from "react-virtualized/dist/commonjs/List";
import PlainButton from "@/components/PlainButton";
import Tooltip from "@/components/Tooltip";
import useDataSourceSchema from "@/pages/queries/hooks/useDataSourceSchema"; import useDataSourceSchema from "@/pages/queries/hooks/useDataSourceSchema";
import useImmutableCallback from "@/lib/hooks/useImmutableCallback"; import useImmutableCallback from "@/lib/hooks/useImmutableCallback";
import LoadingState from "../items-list/components/LoadingState"; import LoadingState from "../items-list/components/LoadingState";
@@ -46,27 +45,23 @@ function SchemaItem({ item, expanded, onToggle, onSelect, ...props }) {
return ( return (
<div {...props}> <div {...props}>
<div className="schema-list-item"> <div className="table-name" onClick={onToggle}>
<PlainButton className="table-name" onClick={onToggle}> <i className="fa fa-table m-r-5" />
<i className="fa fa-table m-r-5" aria-hidden="true" />
<strong> <strong>
<span title={item.name}>{tableDisplayName}</span> <span title={item.name}>{tableDisplayName}</span>
{!isNil(item.size) && <span> ({item.size})</span>} {!isNil(item.size) && <span> ({item.size})</span>}
</strong> </strong>
</PlainButton>
<Tooltip <Tooltip title="Insert table name into query text" mouseEnterDelay={0} mouseLeaveDelay={0}>
title="Insert table name into query text" <i
mouseEnterDelay={0} className="fa fa-angle-double-right copy-to-editor"
mouseLeaveDelay={0} aria-hidden="true"
placement="topRight" onClick={e => handleSelect(e, item.name)}
arrowPointAtCenter> />
<PlainButton className="copy-to-editor" onClick={e => handleSelect(e, item.name)}>
<i className="fa fa-angle-double-right" aria-hidden="true" />
</PlainButton>
</Tooltip> </Tooltip>
</div> </div>
{expanded && ( {expanded && (
<div className="table-open"> <div>
{item.loading ? ( {item.loading ? (
<div className="table-open">Loading...</div> <div className="table-open">Loading...</div>
) : ( ) : (
@@ -74,21 +69,16 @@ function SchemaItem({ item, expanded, onToggle, onSelect, ...props }) {
const columnName = get(column, "name"); const columnName = get(column, "name");
const columnType = get(column, "type"); const columnType = get(column, "type");
return ( return (
<Tooltip <div key={columnName} className="table-open">
title="Insert column name into query text"
mouseEnterDelay={0}
mouseLeaveDelay={0}
placement="rightTop">
<PlainButton key={columnName} className="table-open-item" onClick={e => handleSelect(e, columnName)}>
<div>
{columnName} {columnType && <span className="column-type">{columnType}</span>} {columnName} {columnType && <span className="column-type">{columnType}</span>}
</div> <Tooltip title="Insert column name into query text" mouseEnterDelay={0} mouseLeaveDelay={0}>
<i
<div className="copy-to-editor"> className="fa fa-angle-double-right copy-to-editor"
<i className="fa fa-angle-double-right" aria-hidden="true" /> aria-hidden="true"
</div> onClick={e => handleSelect(e, columnName)}
</PlainButton> />
</Tooltip> </Tooltip>
</div>
); );
}) })
)} )}
@@ -241,15 +231,13 @@ export default function SchemaBrowser({
<Input <Input
className="m-r-5" className="m-r-5"
placeholder="Search schema..." placeholder="Search schema..."
aria-label="Search schema"
disabled={schema.length === 0} disabled={schema.length === 0}
onChange={event => handleFilterChange(event.target.value)} onChange={event => handleFilterChange(event.target.value)}
/> />
<Tooltip title="Refresh Schema"> <Tooltip title="Refresh Schema">
<Button onClick={() => refreshSchema(true)}> <Button onClick={() => refreshSchema(true)}>
<i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": isLoading })} aria-hidden="true" /> <i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": isLoading })} />
<span className="sr-only">{isLoading ? "Loading, please wait." : "Press to refresh."}</span>
</Button> </Button>
</Tooltip> </Tooltip>
</div> </div>

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/main.less"; @import (reference, less) '~@/assets/less/main.less';
.ant-list { .ant-list {
&.add-to-dashboard-dialog-search-results { &.add-to-dashboard-dialog-search-results {
@@ -13,8 +13,7 @@
padding: 12px; padding: 12px;
cursor: pointer; cursor: pointer;
&:hover, &:hover, &:active {
&:active {
@table-row-hover-bg: fade(@redash-gray, 5%); @table-row-hover-bg: fade(@redash-gray, 5%);
background-color: @table-row-hover-bg; background-color: @table-row-hover-bg;
} }

View File

@@ -6,7 +6,7 @@ import Button from "antd/lib/button";
import SyncOutlinedIcon from "@ant-design/icons/SyncOutlined"; import SyncOutlinedIcon from "@ant-design/icons/SyncOutlined";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import Select from "antd/lib/select"; import Select from "antd/lib/select";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import { SchemaList, applyFilterOnSchema } from "@/components/queries/SchemaBrowser"; import { SchemaList, applyFilterOnSchema } from "@/components/queries/SchemaBrowser";
import useImmutableCallback from "@/lib/hooks/useImmutableCallback"; import useImmutableCallback from "@/lib/hooks/useImmutableCallback";
import useDatabricksSchema from "./useDatabricksSchema"; import useDatabricksSchema from "./useDatabricksSchema";
@@ -84,7 +84,6 @@ export default function DatabricksSchemaBrowser({
<Input <Input
className={isDatabaseSelectOpen ? "database-select-open" : ""} className={isDatabaseSelectOpen ? "database-select-open" : ""}
placeholder="Filter tables & columns..." placeholder="Filter tables & columns..."
aria-label="Search schema"
disabled={loadingDatabases || loadingSchema} disabled={loadingDatabases || loadingSchema}
onChange={event => handleFilterChange(event.target.value)} onChange={event => handleFilterChange(event.target.value)}
addonBefore={ addonBefore={
@@ -99,12 +98,12 @@ export default function DatabricksSchemaBrowser({
onDropdownVisibleChange={setIsDatabaseSelectOpen} onDropdownVisibleChange={setIsDatabaseSelectOpen}
placeholder={ placeholder={
<> <>
<i className="fa fa-database m-r-5" aria-hidden="true" /> Database <i className="fa fa-database m-r-5" /> Database
</> </>
}> }>
{filteredDatabases.map(database => ( {filteredDatabases.map(database => (
<Select.Option key={database}> <Select.Option key={database}>
<i className="fa fa-database m-r-5" aria-hidden="true" /> <i className="fa fa-database m-r-5" />
{database} {database}
</Select.Option> </Select.Option>
))} ))}

View File

@@ -1,4 +1,4 @@
@import (reference, less) "~@/assets/less/ant"; @import "~@/assets/less/ant";
.databricks-schema-browser { .databricks-schema-browser {
.schema-control { .schema-control {

View File

@@ -5,7 +5,6 @@ import Button from "antd/lib/button";
import Modal from "antd/lib/modal"; import Modal from "antd/lib/modal";
import DynamicForm from "@/components/dynamic-form/DynamicForm"; import DynamicForm from "@/components/dynamic-form/DynamicForm";
import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper"; import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper";
import { useUniqueId } from "@/lib/hooks/useUniqueId";
function QuerySnippetDialog({ querySnippet, dialog, readOnly }) { function QuerySnippetDialog({ querySnippet, dialog, readOnly }) {
const handleSubmit = useCallback( const handleSubmit = useCallback(
@@ -32,8 +31,6 @@ function QuerySnippetDialog({ querySnippet, dialog, readOnly }) {
{ name: "snippet", title: "Snippet", type: "ace", required: true }, { name: "snippet", title: "Snippet", type: "ace", required: true },
].map(field => ({ ...field, readOnly, initialValue: get(querySnippet, field.name, "") })); ].map(field => ({ ...field, readOnly, initialValue: get(querySnippet, field.name, "") }));
const querySnippetsFormId = useUniqueId("querySnippetForm");
return ( return (
<Modal <Modal
{...dialog.props} {...dialog.props}
@@ -49,7 +46,7 @@ function QuerySnippetDialog({ querySnippet, dialog, readOnly }) {
disabled={readOnly || dialog.props.okButtonProps.disabled} disabled={readOnly || dialog.props.okButtonProps.disabled}
htmlType="submit" htmlType="submit"
type="primary" type="primary"
form={querySnippetsFormId} form="querySnippetForm"
data-test="SaveQuerySnippetButton"> data-test="SaveQuerySnippetButton">
{isEditing ? "Save" : "Create"} {isEditing ? "Save" : "Create"}
</Button> </Button>
@@ -58,13 +55,7 @@ function QuerySnippetDialog({ querySnippet, dialog, readOnly }) {
wrapProps={{ wrapProps={{
"data-test": "QuerySnippetDialog", "data-test": "QuerySnippetDialog",
}}> }}>
<DynamicForm <DynamicForm id="querySnippetForm" fields={formFields} onSubmit={handleSubmit} hideSubmitButton feedbackIcons />
id={querySnippetsFormId}
fields={formFields}
onSubmit={handleSubmit}
hideSubmitButton
feedbackIcons
/>
</Modal> </Modal>
); );
} }

View File

@@ -1,9 +1,8 @@
import { map, trim } from "lodash"; import { map, trim } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Tooltip from "@/components/Tooltip"; import Tooltip from "antd/lib/tooltip";
import EditTagsDialog from "./EditTagsDialog"; import EditTagsDialog from "./EditTagsDialog";
import PlainButton from "@/components/PlainButton";
export class TagsControl extends React.Component { export class TagsControl extends React.Component {
static propTypes = { static propTypes = {
@@ -35,23 +34,19 @@ export class TagsControl extends React.Component {
renderEditButton() { renderEditButton() {
const tags = map(this.props.tags, trim); const tags = map(this.props.tags, trim);
return ( return (
<PlainButton <a
className="label label-tag hidden-xs" className="label label-tag hidden-xs"
role="none"
onClick={() => this.editTags(tags, this.props.getAvailableTags)} onClick={() => this.editTags(tags, this.props.getAvailableTags)}
data-test="EditTagsButton"> data-test="EditTagsButton">
{tags.length === 0 && ( {tags.length === 0 && (
<React.Fragment> <React.Fragment>
<i className="zmdi zmdi-plus m-r-5" aria-hidden="true" /> <i className="zmdi zmdi-plus m-r-5" />
Add tag Add tag
</React.Fragment> </React.Fragment>
)} )}
{tags.length > 0 && ( {tags.length > 0 && <i className="zmdi zmdi-edit" />}
<> </a>
<i className="zmdi zmdi-edit" aria-hidden="true" />
<span className="sr-only">Edit</span>
</>
)}
</PlainButton>
); );
} }

View File

@@ -10,7 +10,6 @@ import notification from "@/services/notification";
import Visualization from "@/services/visualization"; import Visualization from "@/services/visualization";
import recordEvent from "@/services/recordEvent"; import recordEvent from "@/services/recordEvent";
import useQueryResultData from "@/lib/useQueryResultData"; import useQueryResultData from "@/lib/useQueryResultData";
import { useUniqueId } from "@/lib/hooks/useUniqueId";
import { import {
registeredVisualizations, registeredVisualizations,
getDefaultVisualization, getDefaultVisualization,
@@ -157,9 +156,6 @@ function EditVisualizationDialog({ dialog, visualization, query, queryResult })
? filter(sortBy(registeredVisualizations, ["name"]), vis => !vis.isDeprecated) ? filter(sortBy(registeredVisualizations, ["name"]), vis => !vis.isDeprecated)
: pick(registeredVisualizations, [type]); : pick(registeredVisualizations, [type]);
const vizTypeId = useUniqueId("visualization-type");
const vizNameId = useUniqueId("visualization-name");
return ( return (
<Modal <Modal
{...dialog.props} {...dialog.props}
@@ -176,10 +172,10 @@ function EditVisualizationDialog({ dialog, visualization, query, queryResult })
<div className="edit-visualization-dialog"> <div className="edit-visualization-dialog">
<div className="visualization-settings"> <div className="visualization-settings">
<div className="m-b-15"> <div className="m-b-15">
<label htmlFor={vizTypeId}>Visualization Type</label> <label htmlFor="visualization-type">Visualization Type</label>
<Select <Select
data-test="VisualizationType" data-test="VisualizationType"
id={vizTypeId} id="visualization-type"
className="w-100" className="w-100"
disabled={!isNew} disabled={!isNew}
value={type} value={type}
@@ -192,10 +188,10 @@ function EditVisualizationDialog({ dialog, visualization, query, queryResult })
</Select> </Select>
</div> </div>
<div className="m-b-15"> <div className="m-b-15">
<label htmlFor={vizNameId}>Visualization Name</label> <label htmlFor="visualization-name">Visualization Name</label>
<Input <Input
data-test="VisualizationName" data-test="VisualizationName"
id={vizNameId} id="visualization-name"
className="w-100" className="w-100"
value={name} value={name}
onChange={event => onNameChanged(event.target.value)} onChange={event => onNameChanged(event.target.value)}

View File

@@ -1,9 +1,4 @@
import React from "react"; import React from "react";
import Spin from "antd/lib/spin"; import Spin from "antd/lib/spin";
Spin.setDefaultIndicator( Spin.setDefaultIndicator(<i className="fa fa-spinner fa-pulse" />);
<span role="status" aria-live="polite" aria-relevant="additions removals">
<i className="fa fa-spinner fa-pulse" aria-hidden="true" />
<span className="sr-only">Loading...</span>
</span>
);

View File

@@ -1,45 +0,0 @@
import { HTMLAttributes } from "react";
interface SrNotifyProps {
text: string;
expiry: number;
container: HTMLElement;
politeness: HTMLAttributes<HTMLDivElement>["aria-live"];
}
export function srNotify({ text, expiry = 1000, container = document.body, politeness = "polite" }: SrNotifyProps) {
const element = document.createElement("div");
const id = `speak-${Date.now()}`;
element.id = id;
element.className = "sr-only";
element.textContent = text;
element.setAttribute("role", "alert");
element.setAttribute("aria-live", politeness);
container.appendChild(element);
let timer: null | number = null;
let isDone = false;
const cleanupFn = () => {
if (isDone) {
return;
}
isDone = true;
try {
container.removeChild(element);
} catch (e) {
console.error(e);
}
if (timer) {
window.clearTimeout(timer);
}
};
timer = window.setTimeout(cleanupFn, expiry);
return cleanupFn;
}

Some files were not shown because too many files have changed in this diff Show More