Compare commits

...

109 Commits

Author SHA1 Message Date
Jesse
92e5d78dde Update changelog details for snowflake (#5519) 2021-06-17 11:42:07 -07:00
Jesse
0983e6926f update changelog for v10-beta (#5517) 2021-06-17 10:45:17 -07:00
Jesse
dec88799ab Fix: pagination is broken on the dashboard list page (#5516)
* Add test that reproduces issue #5466

* Fix: Duplicate dashboard rows were returned by Dashboard.all() (#5466)
2021-06-15 13:04:36 -07:00
Jesse Whitehouse
64a1d7a6cd Update version for CircleCI build. 2021-06-01 11:21:49 -05:00
Shen Li
041b184d37 README.md: Add TiDB to the Supported Data Sources (#5477) 2021-05-14 06:52:29 -07:00
Omer Lachish
5085495dd4 Refine Dockerfile caching (#5484) 2021-05-14 06:48:10 -07:00
case-k-git
e62de4e4c3 fix big_query.py google api import error (#5482) 2021-05-14 06:47:38 -07:00
Jawshua
8cac6b555c Use the correct rq connection in get_queues_status (#5491) 2021-05-14 16:45:43 +03:00
adamzwakk
e4e567bbb9 Fixing failure report rendering (#5492) 2021-05-14 06:25:52 -07:00
Ben Herzberg
8e728308ab SFS-001: Adding support for the optional host connection property (#5490) 2021-05-14 06:07:30 -07:00
Omer Lachish
7ec86cf4bd Expire sessions after 6 hours of inactivity (#5159)
Configurable with environment variables
2021-05-10 13:36:34 -05:00
Omer Lachish
1c3f724f3e use ptpython instead of standard python shell (#5483) 2021-05-05 16:56:34 -07:00
Jesse
9c8c1bfa9a Adds rate limit to /forgot. (#5425)
Security vulnerability was disclosed by Sohail Ahmed <https://www.linkedin.com/in/sohail-ahmed-755776184/>
2021-04-26 12:02:47 -05:00
iwakiriK
f21f7e211f Athena: skip tables with no StorageDescriptor (#5447) 2021-04-21 15:01:57 -05:00
Nolan Nichols
a70eeb9530 Query Runner: SPARQL Endpoint Data Source (#5469) 2021-04-19 16:45:52 -05:00
Rafael Wendel
427c005c04 Replace hardcoded ids with hook (#5444)
* refactor: replace hardcoded ids with hook

* refactor: replace hard coded ids with lodash id (class)
2021-04-19 09:30:46 -03:00
Rafael Wendel
d8d7c78992 Replace <a> and <button> with <PlainButton> (#5433)
* Add PlainButton

* refactor close icons

* reorder import

* refactor remaining anchors

* refactor: replace remaining <button> and TODOs

* refactor: changed applicable elements to type link

* fix: minor details

* bug: fix tooltip ternary

* refactor: improve interactivity and semantics of schema list item
2021-04-10 16:43:58 -03:00
Rafael Wendel
23ced5db50 fix: treat possibly empty hrefs (#5468) 2021-04-10 13:00:15 -03:00
Rafael Wendel
f018c0a7b7 fix: rollback pip version to avoid legacy resolver problem (#5467)
Co-authored-by: Lingkai Kong <lingkai.kong@databricks.com>
2021-04-09 15:34:42 -03:00
Jesse
67263e1b0f Fixes issue #5445: Scheduled query not working (#5448)
* use 'query_id' everywhere instead of 'Query ID'
* some black while we're at it

Co-authored-by: Omer Lachish <omer@rauchy.net>
2021-04-08 13:32:34 -05:00
Rafael Wendel
bb1f8cbcf5 Fix Ace editor keyboard trap (#5451)
* bug: fix a11y and add sr notification

* refactor: improvements to sr notification
2021-04-07 09:50:54 -03:00
Rafael Wendel
a61a25dd32 Run prettier (#5436)
* run in /client

* run in /viz-lib

* bug: fix wrong line ts expect error

* bug: fixed search pattern for prettier
2021-03-31 16:44:19 -03:00
Jesse
21ea72fdc5 Get the user's current groups from props instead of useEffect(). (#5450)
useEffect() doesn't run until _after_ the component renders. Before the
hook runs, the value of `groups` === []. And this is passed to
<DynamicForm>'s `initialValue` prop. The `initialValue` is not re-evaluated
after useEffect() completes. So the users groups are never updated.

This change pulls the user's current groups from `user` prop on the
page.
2021-03-31 16:18:59 +03:00
Gabriel Dutra
fa8b24ea01 Prepare viz-lib release with Antd v4 (#5443) 2021-03-30 16:06:35 -03:00
Rafael Wendel
a2c96c1e6d Embed "external" link type into <Link> component (#5432)
* feature: add external link

* refactor: split external link into own component

* refactor: added link with icon

* refactor: remove reduntant tab index

* refactor: simplify props

* refactor: fix types

* refactor: bring types and components together

* refactor: improve treatment of target
2021-03-26 15:24:07 -03:00
Rafael Wendel
44178d9908 Improve input fields a11y (#5427)
* Added labels to params

* Added aria-label to inputs

* Linked unsemantic label with input

* Replaced span with label

* refactor: improve labels for schema browsers

* refactor: component accepts aria label

* refactor: add labels to sidebar search inputs
2021-03-26 11:45:24 -03:00
Rafael Wendel
6228f4cf71 Add live regions to tooltip (#5440)
* feature: add live regions to tooltip

* bug: treat null case
2021-03-25 17:47:49 -03:00
Rafael Wendel
c8df7a1c8a Add jsx/a11y eslint plugin (#5439)
* build: install eslint jsx/a11y

* chore: add ESlint rules for jsx/a11y

* bug: add exceptions
2021-03-24 18:50:21 -03:00
Rafael Wendel
a665253f50 Adds configuration for <Tooltip> trigger on focus (#5434)
* refactor: add tooltip

* refactor: replace imports

* feature: add focus trigger
2021-03-24 18:35:21 -03:00
Sebastian Tramp
70681294a3 Query Runner: eccenca Corporate Memory (SPARQL) - query RDF / Linked Data Knowledge Graphs with redash (#5415)
* add Corporate Memory Runner based on cmempy 21.2.3

* fix code style

* apply some code nice ups

* use extendedEnum, boolean and extra_options for schema description

* use lower case sorting for data source types list

This correctly orders data source names which starts with lower
chars (such as eccenca Corporate Memory)

* add missing dblogo
2021-03-24 00:15:24 -07:00
Rafael Wendel
fb90b501cb Improve icon a11y (#5424)
* Added screen reader CSS

* Added description to external links

* Added spinner icon accessibility

* Added accessibility to exclamation and big message

* Added question and exclamation accessibility

* Hide decorative icons

* Standardized link design

* Added a11y to refresh icons

* Added aria-label to anchors and buttons

* Added a11y to conditional icons

* Added applicable labels to Ant Icons

* Changed escape to interpolation

* Replaced external links with opens in new tab

* Improved Tooltip hosts

* Added aria live to temporary elements

* Removed mistakenly added redundant helper

* Undoes unnecessarily added interpolation

* Replaced empty label with hidden

* Improved full icon label

* Improved display of live regions

* Added note

* remove unused class

* Created unique id

* Remove TODOs

* Proper action label

* Improved feedback for autocomplete toggle

* feature: add id hook

* refactor: use id hook

* standardize white space
2021-03-22 19:49:36 -03:00
Rafael Wendel
0560e2410e Improve css and add focus styles (#5420)
* Add styles for focused ant menus

* Add disabled styles to clickable button

* Improved dashboard header syntax and added focus

* Improved CSS syntax

* Add interactive styles

* Improved anchor dependent styles

* Improved styles of widget (gray more/delete btns)

* Add interactive style for favorite star

* Improved style of delete btn

* Make table content fill all space

* Added focus and active styles

* Scoped query snippets list

* Fixed behavior for all major browsers

* Replaced button styles with plain button

* Scoped items list styles

* Added focus styles to ant table

* Add plain button (#5419)

* Minor syntax improvements

* Refactor of Link component (#5418)
2021-03-17 14:26:08 -03:00
Đặng Minh Dũng
a5ec506b60 feat: support Trino data-source (#5411)
* feat: add trino logo

* feat: add trino
2021-03-12 12:10:06 -08:00
Jiajie Zhong
d4f363854d Add setting to identify email block domain (#5377)
* Add setting to identify email block domain

ref: #5368

* rename

Co-authored-by: Levko Kravets <levko.ne@gmail.com>

* rename and add comment

Co-authored-by: Levko Kravets <levko.ne@gmail.com>

* Update redash/handlers/users.py

Co-authored-by: Levko Kravets <levko.ne@gmail.com>

* Update redash/handlers/users.py

Co-authored-by: Levko Kravets <levko.ne@gmail.com>

* Add more comment to settting

Co-authored-by: Levko Kravets <levko.ne@gmail.com>
2021-03-12 12:06:41 -08:00
Omer Lachish
9fdf1f341d Reset failure counter on adhoc success (#5394)
* reset failure counter when query completes successfully via adhoc

* Use "query_id" in metadata, but still allow "Query ID" for transition/legacy support
2021-03-12 12:02:29 -08:00
Rafael Wendel
10bce2d1ac Refactor of Link component (#5418)
* Refactor of link component

* Applied anchor-is-valid to Link component

* Fixed Eslint error

* Removed improper anchor uses

* Fixed TS errors
2021-03-11 11:07:01 -03:00
Rafael Wendel
b2636deef4 Add plain button (#5419)
* Add plain button

* Minor syntax improvements
2021-03-10 13:42:51 -03:00
Rafael Wendel
6cc69ec2c1 Initial a11y improvements (#5408)
* Fixed jsx-a11y problems

* Changed tabIndex to type number

* Initial improvements to DesktopNavbar accessibility

* Added accessibility to favorites list

* Improved accessibility in Desktop Navbar

* Improvements in Desktop navbar semantics

* Added aria roles to tags list

* Fixed tabindex type

* Improved aria labels in query control dropdown

* Added tab for help trigger close button

* Fixed typo

* Improved accessibility in query selector

* Changed resizable role to separator

* Added label to empty state close button

* Removed redundant and mistaken roles

* Used semantic components

* Removed tabIndex from anchor tags

* Removed mistakenly set menuitem role from anchors

* Removed tabIndex from Link components

* Removed improper hidden aria label from icon

* Reverted button and link roles in anchors for minimal merge conflicts

* Replaced alt attr with aria-label for icons

* Removed redundant menu role

* Improved accessibility of CodeBlock

* Removed improper role from schema browser

* Reverted favorites list to div

* Removed improper presentation role in query snippets

* Tracked changes for further PR

* Revert "Improved accessibility of CodeBlock"

* Add aria-labelledby to the associated code labels

This reverts commit 00a1685b1b.

* Wrapped close icon into button
2021-03-04 16:30:31 -03:00
Omer Lachish
46e97a08cc Upgrade RQ to v1.5 (#5207)
* upgrade RQ to v1.5

* set job's started_at

* update healthcheck to match string worker names

* delay worker healthcheck for 5 minutes from start to allow enough time to load in case many workers try to load simultaneously

* log when worker cannot be found
2021-02-15 22:52:53 +02:00
Levko Kravets
640fea5e47 Fix duplicate stylesheets (#5396) 2021-02-14 22:16:06 +02:00
Rafael Wendel
c865293aaa Revert "Updated axios (#5371)" (#5385)
This reverts commit 49536de1ed.
2021-02-02 17:59:53 -03:00
Omer Lachish
3d3f6b1916 extend sync_user_details expiry (#5330) 2021-02-02 16:30:38 +02:00
Justin Talbot
0e1587a068 Add My Dashboards filter option to the Dashboards list (#5375)
* Add My Dashboards filter option to the Dashboards list. Added API endpoint to get the list of a user's dashboards, similar to the My Queries feature.

* Update empty dashboard list state to show an invite to create a new dashboard, like My Queries

* Update to Levko's suggested approach. Clean up some of the formatting for consistency. Put the 'My Queries/Dashboards' item before the Favorites since that organization seems cleaner to me.

* Address Levko's comments
2021-02-02 12:37:48 +02:00
Rafael Wendel
04edf16ed4 Increased waiting time to avoid flakiness (#5370) 2021-01-28 15:02:36 -03:00
Rafael Wendel
49536de1ed Updated axios (#5371) 2021-01-28 14:48:36 -03:00
dependabot[bot]
2f1394a6f4 Bump axios from 0.19.0 to 0.21.1 (#5366)
Bumps [axios](https://github.com/axios/axios) from 0.19.0 to 0.21.1.
- [Release notes](https://github.com/axios/axios/releases)
- [Changelog](https://github.com/axios/axios/blob/v0.21.1/CHANGELOG.md)
- [Commits](https://github.com/axios/axios/compare/v0.19.0...v0.21.1)

Signed-off-by: dependabot[bot] <support@github.com>

Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2021-01-25 21:49:32 -03:00
dependabot[bot]
911f398006 Bump bl from 1.2.2 to 1.2.3 in /viz-lib (#5257)
Bumps [bl](https://github.com/rvagg/bl) from 1.2.2 to 1.2.3.
- [Release notes](https://github.com/rvagg/bl/releases)
- [Commits](https://github.com/rvagg/bl/compare/v1.2.2...v1.2.3)

Signed-off-by: dependabot[bot] <support@github.com>

Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2021-01-25 16:52:14 -03:00
dependabot[bot]
b0b1d6c81c Bump dompurify from 2.0.8 to 2.0.17 in /viz-lib (#5326)
Bumps [dompurify](https://github.com/cure53/DOMPurify) from 2.0.8 to 2.0.17.
- [Release notes](https://github.com/cure53/DOMPurify/releases)
- [Commits](https://github.com/cure53/DOMPurify/compare/2.0.8...2.0.17)

Signed-off-by: dependabot[bot] <support@github.com>

Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
2021-01-25 13:44:11 -03:00
Rafael Wendel
23a279f318 Fix for Cypress flakiness generated by param_spec (#5349) 2021-01-22 21:03:15 -03:00
Arik Fraimovich
e71ccf5de5 Fix: add a merge migration to solve multi head issue (#5364)
* Add unit test to test for multi-head migrations issue

* Add merge migration
2021-01-21 10:55:52 -08:00
Jiajie Zhong
bb42e92cd0 Remove unnecessary space in rq log (#5345) 2021-01-20 19:45:16 -08:00
Patrick Yang
4ec96caac5 Encrypt alert notification destinations (#5317) 2021-01-20 19:40:53 -08:00
Vipul Mathur
829247c2d2 Use legacy resolver in pip to fix broken build (#5309)
Fixes #5300 and fixes #5307 

There have been upstream (`python:37-slim` image) changes that
bring in `pip` version 20.3.1, which makes new `2020-resolver`
the default.  Due to that, un-resolvable dependency conflicts
in  `requirements_all_ds.txt` now cause the build to fail.

This is a workaround until the package versions can be updated
to work with the new pip resolver.
2021-01-20 12:17:39 -08:00
Rafael Wendel
7d33af4343 Fix inconsistent Sankey behavior (#5286)
* added type casting to coerce number string into nuber

* Merge branch 'master' into fix-inconsistent=sankey-behavior

* typed map viz options

* Partially typed what was possible

* reworked data coercion

* improved MapOptionsType types

* readaqueted sankey rows so as to allow strings again
2021-01-12 23:54:14 -03:00
Rafael Wendel
84c2abed59 Add reorder to dashboard parameter widgets (#5267)
* added paramOrder prop

* minor refactor

* moved logic to widget

* Added paramOrder to widget API call

* Update client/app/components/dashboards/dashboard-widget/VisualizationWidget.jsx

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>

* Merge branch 'master' into reorder-dashboard-parameters

* experimental removal of helper element

* cleaner comment

* Added dashboard global params logic

* Added backend logic for dashboard options

* Removed testing leftovers

* removed appending sortable to parent component behavior

* Revert "Added backend logic for dashboard options"

This reverts commit 41ae2ce475.

* Re-structured backend options

* removed temporary edits

* Added dashboard/widget param reorder cypress tests

* Separated edit and sorting permission

* added options to public dashboard serializer

* Removed undesirable events from drag

* Bring back attaching sortable to its parent

This reverts commit 163fb6fef5.

* Added prop to control draggable destination parent

* Removed paramOrder fallback

* WIP (for Netflify preview)

* fixup! Added prop to control draggable destination parent

* Better drag and drop styling and fix for the padding

* Revert "WIP (for Netflify preview)"

This reverts commit 433e11edc3.

* Improved dashboard parameter Cypress test

* Standardized reorder styling

* Changed dashboard param reorder to edit mode only

* fixup! Improved dashboard parameter Cypress test

* fixup! Improved dashboard parameter Cypress test

* Fix for Cypress CI error

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2021-01-11 15:18:50 -03:00
Christopher Grant
8b068dfd0b Truncate large Databricks ODBC result sizes (#5290)
Truncates results sets that exceed a limit taken from an environment
variable called DATABRICKS_ROW_LIMIT.
2021-01-08 15:20:11 -06:00
Rafael Wendel
06eb868120 Bar chart e2e test (#5279)
* created bar-chart e2e test boilerplate

* refactored assertions

* added snapshots and dashboard

* refactored assertions to properly deal with async

* replaced loops with getters for proper workings of cypress

* added a couple other bar charts

* ran prettier

* added a better query for bar charts

* removed leftovers

* moved helpers to support folder

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2021-01-06 15:13:33 -03:00
Patrick Yang
52ae7bedb2 Secret handling for Yandex, TreasureData, & Postgres/CockroachDB SSL (#5312) 2021-01-05 11:47:54 -08:00
Tim Gates
fbe57de53c docs: fix simple typo, possbily -> possibly (#5329)
There is a small typo in redash/settings/__init__.py.

Should read `possibly` rather than `possbily`.
2021-01-05 12:43:14 +02:00
Patrick Yang
db0cb98ed3 Add Username and Password fields to MongoDB config (#5314) 2021-01-04 23:14:16 -08:00
Rafael Wendel
dcdff66e62 Dropdown param search fix (#5304)
* fixed QueryBasedParamterInput optionFilterProp

* added optionFilterProp fallback for SelectWithVirtualScroll

* simplified syntax

* removed optionFilterProp from QueryBasedParameterInput.jsx

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>

* restricted SelectWithVirtualScroll props

* Added e2e test for parameter filters

* moved filter assertion to more suitable place

* created helper for option filter prop assertion

* moved option filter prop assertion to proper place, added result update assertion

* refactor openAndSearchAntdDropdown helper

* Fix parameter_spec

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2020-12-17 21:56:46 -03:00
Patrick Yang
d0793c4ba8 Obfuscate non-email alert destinations (#5318) 2020-12-16 15:39:30 -08:00
Lingkai Kong
7b8bcdf356 change item element in system status page (#5323) 2020-12-16 11:22:19 -08:00
Elad Ossadon
c290864ccd Convert viz-lib to TypeScript (#5310)
Co-authored-by: ts-migrate <>
2020-12-15 18:21:37 -08:00
Rafael Wendel
b70e95a323 added eslint no-console (#5305)
* added eslint no-console

* Update client/.eslintrc.js to allow warnings

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2020-12-14 10:09:43 -03:00
Elad Ossadon
18ee5343aa Sync date format from settings with clientConfig (#5299) 2020-12-10 11:16:31 -08:00
Elad Ossadon
fdf636a393 Fix disabled hot reload flow (#5306) 2020-12-07 16:02:52 -08:00
Rafael Wendel
88c13868a3 removed leftover console.log (#5303) 2020-12-07 17:21:40 -03:00
Elad Ossadon
aab11dc79b Add React Fast Refresh + Hot Module Reloading (#5291) 2020-12-07 11:46:46 -08:00
Elad Ossadon
00c77cf36e Redesign desktop nav bar (#5294) 2020-12-06 12:09:19 -08:00
Rafael Wendel
6e2631dec2 Changed 'Delete Alert' into 'Delete' for consistency (#5287) 2020-11-30 18:48:35 -03:00
Rafael Wendel
4b88959341 Fix QuerySourceDropdown value type (#5284) 2020-11-24 11:42:20 -03:00
Rafael Wendel
fa2b57a209 Remove unwanted props from Select component (#5277)
* Explicitly selected props so as to avoid errors from non-wanted props

* Simplified approach

* Ran prettier 😬

* Fixed minor issues
2020-11-22 13:07:56 -03:00
Jiajie Zhong
132fed64b3 Correct cleanup_query_results comment (#5276)
Correct comment from QUERY_RESULTS_MAX_AGE
to QUERY_RESULTS_CLEANUP_MAX_AGE
2020-11-20 23:11:13 +02:00
Gabriel Dutra
fa7ecca485 Frontend updates from internal fork (#5259)
* DynamicComponent for QuerySourceAlerts

* General Settings updates

* Dynamic Date[Range] updates

* EmptyState updates

* Query and SchemaBrowser updates

* Adjust page headers and add disablePublish

* Policy updates

* Separate Home FavoritesList component

* Update FormatQuery

* Autolimit frontend fixes

* Misc updates

* Keep registering of QuerySourceDropdown

* Undo changes in DynamicComponent

* Change sql-formatter package.json syntax

* Allow opening help trigger in new tab

* Don't run npm commands as root in Dockerfile

* Cypress: Remove extra execute query
2020-11-10 14:59:15 +02:00
deecay
8f484706b1 Enable Boxplot to be horizontal (#5262) 2020-11-08 23:17:08 +02:00
Josh Bohde
e2e8714155 Enable graceful shutdown of rq workers (#5214)
* Enable graceful shutdown of rq workers

* Use `exec` in the `worker` command of the entrypoint to propagate
  the `TERM` signal
* Allow rq processes managed by supervisor to exit without restart on
  expected status codes
* Allow supervisorctl to contact the running supervisor
* Add a `shutdown_worker` command that will send `TERM` to all running
  worker processes and then sleep. This allows orchestration systems
  to initiate a graceful shutdown before sending `SIGTERM` to
  supervisord

* Use Heroku worker as the BaseWorker

This implements a graceful shutdown on SIGTERM, which simplifies
external shutdown procedures.

* Fix imports based upon review

* Remove supervisorctl config
2020-11-05 11:49:45 +02:00
Jerry
c6bf8a1c55 bugfix: fix #5254 (#5255)
Co-authored-by: Jerry <jerry.yuan@webweye.com>
2020-11-04 20:56:41 +02:00
Rafael Wendel
12f71925c2 Multiselect dropdown slowness (fix) (#5221)
* created util to estimate reasonable width for dropdown

* removed unused import

* improved calculation of item percentile

* added getItemOfPercentileLength to relevant spots

* added getItemOfPercentileLength to relevant spots

* Added missing import

* created custom select element

* added check for property path

* removed uses of percentile util

* gave up on getting element reference

* finished testing Select component

* removed unused imports

* removed older uses of Option component

* added canvas calculation

* removed minWidth from Select

* improved calculation

* added fallbacks

* added estimated offset

* removed leftovers 😅

* replaced to percentiles to max value

* switched to memo and renamed component

* proper useMemo syntax

* Update client/app/components/Select.tsx

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>

* created custom restrictive types

* added quick const

* fixed style

* fixed generics

* added pos absolute to fix percy

* removed custom select from ParameterMappingInput

* applied prettier

* Revert "added pos absolute to fix percy"

This reverts commit 4daf1d4bef.

* Pin Percy version to 0.24.3

* Update client/app/components/ParameterMappingInput.jsx

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>

* renamed Select.jsx to SelectWithVirtualScroll

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2020-11-03 21:50:39 +02:00
Omer Lachish
cae088f35b extend the refresh_queries timeout from 3 minutes to 10 minutes (#5253) 2020-11-02 22:36:57 +02:00
Rafael Wendel
a3c79f26b9 Fix for the typo button in ParameterMappingInput (#5244) 2020-10-29 17:24:13 -03:00
Jonathan Hult
c7c92a3192 Fix annotation bug causing queries not to run - ORA-00933 (#5179) 2020-10-28 10:03:26 +02:00
Rafael Wendel
55cf17aa47 added required to Form.Item and Input for better UI (#5231)
* added required to Form.Item and Input for better UI

* removed required from input

* Revert "removed required from input"

This reverts commit b56cd76fa1.

* Redo "removed required from input"

* removed typo

Co-authored-by: rafawendel2010@gmail.com <rafawendel>
2020-10-28 09:37:16 +02:00
Levko Kravets
8dd76a00c5 Fix dashboard background grid (#5238) 2020-10-26 21:46:38 +02:00
Christopher Grant
e242ac2b10 Static SAML configuration and assertion encryption (#5175)
* Change front-end and data model for SAML2 auth - static configuration

* Add changes to use inline metadata.

* add switch for static and dynamic SAML configurations

* Fixed config of backend static/dynamic to match UI

* add ability to encrypt/decrypt SAML assertions with pem and crt files. Upgraded to pysaml2 6.1.0 to mitigate signature mismatch during decryption

* remove print debug statement

* Use utility to find xmlsec binary for encryption, formatting saml_auth module

* format SAML Javascript, revert want_signed_response to pre-PR value

* pysaml2's entityid should point to the sp, not the idp

* add logging for entityid for validation

* use mustache_render instead of string formatting. put all static logic into static branch

* move mustache template for inline saml metadata to the global level

* Incorporate SAML type with Enabled setting

* Update client/app/pages/settings/components/AuthSettings/SAMLSettings.jsx

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>

Co-authored-by: Chad Chen <chad.chen@databricks.com>
Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2020-10-25 12:06:45 -03:00
Gabriel Dutra
66463aedd4 Fix Home EmptyState help link (#5217) 2020-10-16 11:53:21 -03:00
Rafael Wendel
8a6524c1ba Add horizontal bar chart (#5154)
* added bar chart boilerplate

* added x/y manipulation

* replaced x/y management to inner series preparer

* added tests

* moved axis inversion to all charts series

* removed line and area

* inverted labels ui

* removed normalizer check, simplified inverted axes check

* finished working hbar

* minor review

* added conditional title to YAxis

* generalized horizontal chart for line charts, resetted state on globalSeriesType change

* fixed updates

* fixed updates to layout

* fixed minor issues

* removed right Y axis when axes inverted

* ran prettier

* fixed updater function conflict and misuse of getOptions

* renamed inverted to swapped

* created mappingtypes for swapped columns

* removed unused import

* minor polishing

* improved series behaviour in h-bar

* minor fix

* added basic filter to ChartTypeSelect

* final setup of filtered chart types

* Update viz-lib/src/components/visualizations/editor/createTabbedEditor.jsx

* added proptypes and renamed ChartTypeSelect props

* Add missing import

* fixed import, moved result array to global scope

* merged import

* clearer naming in ChartTypeSelect

* better lodash map syntax

* fixed global modification

* moved result inside useMemo

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
Co-authored-by: Levko Kravets <levko.ne@gmail.com>
2020-10-15 21:34:38 +03:00
Gabriel Dutra
9097feb100 Frontend updates from internal fork (#5209) 2020-10-15 14:25:22 -03:00
Gabriel Dutra
db4e97fa6f Remove build args from Cypress start script (#5203) 2020-10-09 12:23:14 -03:00
Levko Kravets
0d4615a482 Extra actions on Queries and Dashboards pages (#5201)
* Extra actions for Query View and Query Source pages

* Convert Queries List page to functional component

* Convert Dashboards List page to functional component

* Extra actions for Query List page

* Extra actions for Dashboard List page

* Extra actions for Dashboard page

* Pass some extra data to Dashboard.HeaderExtra component

* CR1
2020-10-09 12:12:56 +03:00
Alexander Rusanov
ff008a076b Updated Cypress to v5.3 and fixed e2e tests (#5199)
* Upgraded Cypress to v5.3 and fixed e2e tests

* Updated cypress image

* Fixed failing tests

* Updated NODE_VERSION in netlify

* Update client/cypress/integration/visualizations/choropleth_spec.js

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>

* fixed test in choropleth

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2020-10-06 16:06:47 -03:00
Gabriel Dutra
8d548ecbac Share Embed Spec: Make sure query is executed (#5191) 2020-10-04 16:01:30 +03:00
Gabriel Dutra
2992c382d1 ScheduleDialog: Filter empty interval groups (#5196) 2020-10-03 05:54:05 +03:00
Gabriel Dutra
f4dcb2918a Move Cypress to dev dependencies (#3991)
* Test Cypress on package list

* Skip Puppeteer Chromium as well

* Put back missing npm install on netlify.toml

* Netlify: move env vars to build.environment

* Remove cypress:install script

* Update Cypress dockerfile

* Copy package-lock.json to Cypress dockerfile
2020-09-29 09:51:28 +03:00
Gabriel Dutra
c821cab4cb Generate Code Coverage report for Cypress (#5137) 2020-09-28 21:43:04 -03:00
Levko Kravets
4fb77867b0 Align Y axes at zero (#5053)
* Align Y axes as zero

* Fix typo (with @deecay)

* Add alignYAxesAtZero function

* Avoid 0 division

Co-authored-by: Gabriel Dutra <nesk.frz@gmail.com>
2020-09-28 13:12:40 +03:00
Levko Kravets
a473611cb0 Some Choropleth improvements/refactoring (#5186)
* Directly map query results column to GeoJSON property

* Use cache for geoJson requests

* Don't handle bounds changes while loading geoJson data

* Choropleth: fix map "jumping" on load; don't save bounds if user didn't edit them; refine code a bit

* Improve cache

* Optimize Japan Perfectures map (remove irrelevant GeoJson properties)

* Improve getOptions for Choropleth; remove unused code

* Fix test

* Add US states map

* Convert USA map to Albers projection

* Allow to specify user-friendly field names for maps
2020-09-24 14:39:09 +03:00
Levko Kravets
210008c714 Ask user to log in when session expires (#5178)
* Ask user to log in when session expires

* Update implementation

* Update implementation

* Minor fix

* Update modal

* Do not intercept calls to api/session as Auth.requireSession() relies on it

* Refine code; adjust popup size and position
2020-09-23 16:30:08 +03:00
Omer Lachish
aa5d4f5f4e add 'cancelled' meta directive to all cancelled jobs (#5187) 2020-09-23 12:54:48 +03:00
Omer Lachish
6b811c5245 Refresh CSRF tokens (#5177)
* expire CSRF tokens after 6 hours

* use axios' built-in cookie to header copy mechanism

* add axios-auth-refresh

* retry CSRF-related 400 errors by refreshing the cookie

* export the auth refresh interceptor to support ejecting it if neccessary

* reject the original request if it's unrelated to CSRF
2020-09-21 23:21:14 +03:00
Levko Kravets
83726da48a Keep additional URL params when forking a query (#5184) 2020-09-21 12:54:55 +03:00
Levko Kravets
72dc157bbe Allow to clear selected tags on list pages (#5142)
* Convert TagsList to functional component

* Convert TagsList to typescript

* Allow to unselect all tags

* Add title to Tags block and explicit "clear filter" button

* Some tweaks
2020-09-17 14:01:15 +03:00
Lingkai Kong
1b8ff8e810 Add default limit (1000) to SQL queries (#5088)
* add default limit 1000

* Add frontend changes and connect to backend

* Fix query hash because of default limit

* fix CircleCI test

* adjust for comment
2020-09-14 14:18:31 +03:00
Omer Lachish
31ddd0fb79 prevent assigning queries to view_only data sources (#5152) 2020-09-10 15:43:25 +03:00
Levko Kravets
5cabf7a724 Keep selected filters when switching visualizations (#5146)
* getredash/redash#4944 Query pages: keep selected filters when switching visualizations

* Pass current filters to expanded widget modal
2020-09-10 13:42:53 +03:00
max-voronov
59b135ace7 Move CardsList to typescript (#5136)
* Refactor CardsList - pass a suffix for list item

Adding :id to an item to be used as a key suffix is redundant and the same
can be accomplished by using :index from the map function.

* Move CardsList to typescript

* Convert CardsList component to functional component

* CR1

* CR2
2020-09-05 20:08:01 +03:00
Gabriel Dutra
32b41e4112 Misc frontend changes from internal fork (#5143) 2020-09-04 08:00:30 -03:00
Gabriel Dutra
2e31b91054 Antd v4: Fix CreateUserDialog (#5139)
* Antd v4: Update CreateUserDialog

* Add Cypress test for user creation
2020-09-04 07:57:43 -03:00
Gabriel Dutra
205915e6db Add toggle to disable public URLs (#5140)
* Add toggle to disable public URLs

* Add Cypress tests
2020-09-01 08:49:30 -03:00
531 changed files with 21123 additions and 6252 deletions

View File

@@ -1,12 +1,12 @@
FROM cypress/browsers:chrome67 FROM cypress/browsers:node14.0.0-chrome84
ENV APP /usr/src/app ENV APP /usr/src/app
WORKDIR $APP WORKDIR $APP
COPY package.json $APP/package.json COPY package.json package-lock.json $APP/
RUN npm run cypress:install > /dev/null COPY viz-lib $APP/viz-lib
RUN npm ci > /dev/null
COPY client/cypress $APP/client/cypress COPY . $APP
COPY cypress.json $APP/cypress.json
RUN ./node_modules/.bin/cypress verify RUN ./node_modules/.bin/cypress verify

View File

@@ -57,6 +57,9 @@ jobs:
- store_artifacts: - store_artifacts:
path: coverage.xml path: coverage.xml
frontend-lint: frontend-lint:
environment:
CYPRESS_INSTALL_BINARY: 0
PUPPETEER_SKIP_CHROMIUM_DOWNLOAD: 1
docker: docker:
- image: circleci/node:12 - image: circleci/node:12
steps: steps:
@@ -67,6 +70,9 @@ jobs:
- store_test_results: - store_test_results:
path: /tmp/test-results path: /tmp/test-results
frontend-unit-tests: frontend-unit-tests:
environment:
CYPRESS_INSTALL_BINARY: 0
PUPPETEER_SKIP_CHROMIUM_DOWNLOAD: 1
docker: docker:
- image: circleci/node:12 - image: circleci/node:12
steps: steps:
@@ -90,11 +96,20 @@ jobs:
PERCY_TOKEN_ENCODED: ZGRiY2ZmZDQ0OTdjMzM5ZWE0ZGQzNTZiOWNkMDRjOTk4Zjg0ZjMxMWRmMDZiM2RjOTYxNDZhOGExMjI4ZDE3MA== PERCY_TOKEN_ENCODED: ZGRiY2ZmZDQ0OTdjMzM5ZWE0ZGQzNTZiOWNkMDRjOTk4Zjg0ZjMxMWRmMDZiM2RjOTYxNDZhOGExMjI4ZDE3MA==
CYPRESS_PROJECT_ID_ENCODED: OTI0Y2th CYPRESS_PROJECT_ID_ENCODED: OTI0Y2th
CYPRESS_RECORD_KEY_ENCODED: YzA1OTIxMTUtYTA1Yy00NzQ2LWEyMDMtZmZjMDgwZGI2ODgx CYPRESS_RECORD_KEY_ENCODED: YzA1OTIxMTUtYTA1Yy00NzQ2LWEyMDMtZmZjMDgwZGI2ODgx
CYPRESS_INSTALL_BINARY: 0
PUPPETEER_SKIP_CHROMIUM_DOWNLOAD: 1
docker: docker:
- image: circleci/node:12 - image: circleci/node:12
steps: steps:
- setup_remote_docker - setup_remote_docker
- checkout - checkout
- run:
name: Enable Code Coverage report for master branch
command: |
if [ "$CIRCLE_BRANCH" = "master" ]; then
echo 'export CODE_COVERAGE=true' >> $BASH_ENV
source $BASH_ENV
fi
- run: - run:
name: Install npm dependencies name: Install npm dependencies
command: | command: |
@@ -113,6 +128,13 @@ jobs:
command: | command: |
docker-compose logs docker-compose logs
when: on_fail when: on_fail
- run:
name: Copy Code Coverage results
command: |
docker cp cypress:/usr/src/app/coverage ./coverage || true
when: always
- store_artifacts:
path: coverage
build-docker-image: *build-docker-image-job build-docker-image: *build-docker-image-job
build-preview-docker-image: *build-docker-image-job build-preview-docker-image: *build-docker-image-job
workflows: workflows:

View File

@@ -1,7 +1,20 @@
version: '2.2' version: "2.2"
x-redash-service: &redash-service
build:
context: ../
args:
skip_dev_deps: "true"
skip_ds_deps: "true"
code_coverage: ${CODE_COVERAGE}
x-redash-environment: &redash-environment
REDASH_LOG_LEVEL: "INFO"
REDASH_REDIS_URL: "redis://redis:6379/0"
REDASH_DATABASE_URL: "postgresql://postgres@postgres/postgres"
REDASH_RATELIMIT_ENABLED: "false"
REDASH_ENFORCE_CSRF: "true"
services: services:
server: server:
build: ../ <<: *redash-service
command: server command: server
depends_on: depends_on:
- postgres - postgres
@@ -9,30 +22,25 @@ services:
ports: ports:
- "5000:5000" - "5000:5000"
environment: environment:
<<: *redash-environment
PYTHONUNBUFFERED: 0 PYTHONUNBUFFERED: 0
REDASH_LOG_LEVEL: "INFO"
REDASH_REDIS_URL: "redis://redis:6379/0"
REDASH_DATABASE_URL: "postgresql://postgres@postgres/postgres"
REDASH_RATELIMIT_ENABLED: "false"
REDASH_ENFORCE_CSRF: "true"
scheduler: scheduler:
build: ../ <<: *redash-service
command: scheduler command: scheduler
depends_on: depends_on:
- server - server
environment: environment:
REDASH_REDIS_URL: "redis://redis:6379/0" <<: *redash-environment
worker: worker:
build: ../ <<: *redash-service
command: worker command: worker
depends_on: depends_on:
- server - server
environment: environment:
<<: *redash-environment
PYTHONUNBUFFERED: 0 PYTHONUNBUFFERED: 0
REDASH_LOG_LEVEL: "INFO"
REDASH_REDIS_URL: "redis://redis:6379/0"
REDASH_DATABASE_URL: "postgresql://postgres@postgres/postgres"
cypress: cypress:
ipc: host
build: build:
context: ../ context: ../
dockerfile: .circleci/Dockerfile.cypress dockerfile: .circleci/Dockerfile.cypress
@@ -42,6 +50,7 @@ services:
- scheduler - scheduler
environment: environment:
CYPRESS_baseUrl: "http://server:5000" CYPRESS_baseUrl: "http://server:5000"
CYPRESS_coverage: ${CODE_COVERAGE}
PERCY_TOKEN: ${PERCY_TOKEN} PERCY_TOKEN: ${PERCY_TOKEN}
PERCY_BRANCH: ${CIRCLE_BRANCH} PERCY_BRANCH: ${CIRCLE_BRANCH}
PERCY_COMMIT: ${CIRCLE_SHA1} PERCY_COMMIT: ${CIRCLE_SHA1}

2
.gitignore vendored
View File

@@ -5,6 +5,8 @@ venv/
.coveralls.yml .coveralls.yml
.idea .idea
*.pyc *.pyc
.nyc_output
coverage
.coverage .coverage
coverage.xml coverage.xml
client/dist client/dist

View File

@@ -1,5 +1,129 @@
# 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

@@ -3,13 +3,24 @@ FROM node:12 as frontend-builder
# Controls whether to build the frontend assets # Controls whether to build the frontend assets
ARG skip_frontend_build ARG skip_frontend_build
ENV CYPRESS_INSTALL_BINARY=0
ENV PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1
RUN useradd -m -d /frontend redash
USER redash
WORKDIR /frontend WORKDIR /frontend
COPY package.json package-lock.json /frontend/ COPY --chown=redash package.json package-lock.json /frontend/
COPY viz-lib /frontend/viz-lib COPY --chown=redash viz-lib /frontend/viz-lib
# Controls whether to instrument code for coverage information
ARG code_coverage
ENV BABEL_ENV=${code_coverage:+test}
RUN if [ "x$skip_frontend_build" = "x" ] ; then npm ci --unsafe-perm; fi RUN if [ "x$skip_frontend_build" = "x" ] ; then npm ci --unsafe-perm; fi
COPY client /frontend/client COPY --chown=redash client /frontend/client
COPY 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
@@ -55,8 +66,9 @@ 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
ADD $databricks_odbc_driver_url /tmp/simba_odbc.zip RUN wget --quiet $databricks_odbc_driver_url -O /tmp/simba_odbc.zip \
RUN unzip /tmp/simba_odbc.zip -d /tmp/ \ && chmod 600 /tmp/simba_odbc.zip \
&& 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 \
@@ -68,12 +80,19 @@ 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
# We first copy only the requirements file, to avoid rebuilding on every file # rollback pip version to avoid legacy resolver problem
# change. RUN pip install pip==20.2.4;
COPY requirements.txt requirements_bundles.txt requirements_dev.txt requirements_all_ds.txt ./
RUN if [ "x$skip_dev_deps" = "x" ] ; then pip install -r requirements.txt -r requirements_dev.txt; else pip install -r requirements.txt; fi # We first copy only the requirements file, to avoid rebuilding on every file change.
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

@@ -35,7 +35,7 @@ backend-unit-tests: up test_db
docker-compose run --rm --name tests server tests docker-compose run --rm --name tests server tests
frontend-unit-tests: bundle frontend-unit-tests: bundle
npm ci CYPRESS_INSTALL_BINARY=0 PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=1 npm ci
npm run bundle npm run bundle
npm test npm test

View File

@@ -73,6 +73,7 @@ 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

@@ -18,8 +18,8 @@ worker() {
export WORKERS_COUNT=${WORKERS_COUNT:-2} export WORKERS_COUNT=${WORKERS_COUNT:-2}
export QUEUES=${QUEUES:-} export QUEUES=${QUEUES:-}
supervisord -c worker.conf exec supervisord -c worker.conf
} }
dev_worker() { dev_worker() {

View File

@@ -20,5 +20,10 @@
"globals": ["Error"] "globals": ["Error"]
} }
] ]
] ],
"env": {
"test": {
"plugins": ["istanbul"]
}
}
} }

View File

@@ -5,10 +5,11 @@ 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"], plugins: ["jest", "compat", "no-only-tests", "@typescript-eslint", "jsx-a11y"],
settings: { settings: {
"import/resolver": "webpack", "import/resolver": "webpack",
}, },
@@ -19,7 +20,35 @@ 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": "off", "jsx-a11y/anchor-is-valid": [
// 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": [
"error",
{
paths: [
{
name: "antd",
message: "Please use 'import XXX from antd/lib/XXX' import instead.",
},
{
name: "antd/lib",
message: "Please use 'import XXX from antd/lib/XXX' import instead.",
},
],
},
],
}, },
overrides: [ overrides: [
{ {
@@ -34,6 +63,8 @@ module.exports = {
// Do not complain about useless contructors in declaration files // Do not complain about useless contructors in declaration files
"no-useless-constructor": "off", "no-useless-constructor": "off",
"@typescript-eslint/no-useless-constructor": "error", "@typescript-eslint/no-useless-constructor": "error",
// Many API fields and generated types use camelcase
"@typescript-eslint/camelcase": "off",
}, },
}, },
], ],

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.5 KiB

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 31 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

View File

@@ -225,6 +225,16 @@
} }
} }
&-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 {
@@ -391,6 +401,18 @@
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,6 +98,10 @@ strong {
.clickable { .clickable {
cursor: pointer; cursor: pointer;
button&:disabled {
cursor: not-allowed;
}
} }
.resize-vertical { .resize-vertical {

View File

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

View File

@@ -2,163 +2,218 @@
Generate Margin Classes (0px - 25px) Generate Margin Classes (0px - 25px)
margin, margin-top, margin-bottom, margin-left, margin-right margin, margin-top, margin-bottom, margin-left, margin-right
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.margin (@label, @size: 1, @key:1) when (@size =< 30){ .margin (@label, @size: 1, @key:1) when (@size =< 30) {
.m-@{key} { .m-@{key} {
margin: @size !important; margin: @size !important;
} }
.m-t-@{key} { .m-t-@{key} {
margin-top: @size !important; margin-top: @size !important;
} }
.m-b-@{key} { .m-b-@{key} {
margin-bottom: @size !important; margin-bottom: @size !important;
} }
.m-l-@{key} { .m-l-@{key} {
margin-left: @size !important; margin-left: @size !important;
} }
.m-r-@{key} { .m-r-@{key} {
margin-right: @size !important; margin-right: @size !important;
} }
.margin(@label - 5; @size + 5; @key + 5); .margin(@label - 5; @size + 5; @key + 5);
} }
.margin(25, 0px, 0); .margin(25, 0px, 0);
.m-2{ .m-2 {
margin:2px; margin: 2px;
} }
/* -------------------------------------------------------- /* --------------------------------------------------------
Generate Padding Classes (0px - 25px) Generate Padding Classes (0px - 25px)
padding, padding-top, padding-bottom, padding-left, padding-right padding, padding-top, padding-bottom, padding-left, padding-right
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.padding (@label, @size: 1, @key:1) when (@size =< 30){ .padding (@label, @size: 1, @key:1) when (@size =< 30) {
.p-@{key} { .p-@{key} {
padding: @size !important; padding: @size !important;
} }
.p-t-@{key} { .p-t-@{key} {
padding-top: @size !important; padding-top: @size !important;
} }
.p-b-@{key} { .p-b-@{key} {
padding-bottom: @size !important; padding-bottom: @size !important;
} }
.p-l-@{key} { .p-l-@{key} {
padding-left: @size !important; padding-left: @size !important;
} }
.p-r-@{key} { .p-r-@{key} {
padding-right: @size !important; padding-right: @size !important;
} }
.padding(@label - 5; @size + 5; @key + 5); .padding(@label - 5; @size + 5; @key + 5);
} }
.padding(25, 0px, 0); .padding(25, 0px, 0);
/* -------------------------------------------------------- /* --------------------------------------------------------
Generate Font-Size Classes (8px - 20px) Generate Font-Size Classes (8px - 20px)
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.font-size (@label, @size: 8, @key:10) when (@size =< 20){ .font-size (@label, @size: 8, @key:10) when (@size =< 20) {
.f-@{key} { .f-@{key} {
font-size: @size !important; font-size: @size !important;
} }
.font-size(@label - 1; @size + 1; @key + 1); .font-size(@label - 1; @size + 1; @key + 1);
} }
.font-size(20, 8px, 8); .font-size(20, 8px, 8);
.f-inherit { font-size: inherit !important; } .f-inherit {
font-size: inherit !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Font Weight Font Weight
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.f-300 { font-weight: 300 !important; } .f-300 {
.f-400 { font-weight: 400 !important; } font-weight: 300 !important;
.f-500 { font-weight: 500 !important; } }
.f-700 { font-weight: 700 !important; } .f-400 {
font-weight: 400 !important;
}
.f-500 {
font-weight: 500 !important;
}
.f-700 {
font-weight: 700 !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Position Position
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.p-relative { position: relative !important; } .p-relative {
.p-absolute { position: absolute !important; } position: relative !important;
.p-fixed { position: fixed !important; } }
.p-static { position: static !important; } .p-absolute {
position: absolute !important;
}
.p-fixed {
position: fixed !important;
}
.p-static {
position: static !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Overflow Overflow
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.o-hidden { overflow: hidden !important; } .o-hidden {
.o-visible { overflow: visible !important; } overflow: hidden !important;
.o-auto { overflow: auto !important; } }
.o-visible {
overflow: visible !important;
}
.o-auto {
overflow: auto !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Display Display
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.di-block { display: inline-block !important; } .di-block {
.d-block { display: block; } display: inline-block !important;
}
.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, 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; @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;
.for(@array); .-each(@value) { .for(@array);
@name: extract(@value, 1); .-each(@value) {
@name2: extract(@value, 2); @name: extract(@value, 1);
@color: extract(@value, 3); @name2: extract(@value, 2);
&.@{name2} { @color: extract(@value, 3);
background-color: @color !important; &.@{name2} {
} background-color: @color !important;
}
&.@{name} {
color: @color !important; &.@{name} {
} color: @color !important;
}
} }
/* -------------------------------------------------------- /* --------------------------------------------------------
Background Colors Background Colors
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.bg-brand { background-color: @brand-bg; } .bg-brand {
.bg-black-trp { background-color: rgba(0,0,0,0.12) !important; } background-color: @brand-bg;
}
.bg-black-trp {
background-color: rgba(0, 0, 0, 0.12) !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Borders Borders
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.b-0 { border: 0 !important; } .b-0 {
border: 0 !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Width Width
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.w-100 { width: 100% !important; } .w-100 {
.w-50 { width: 50% !important; } width: 100% !important;
.w-25 { width: 25% !important; } }
.w-50 {
width: 50% !important;
}
.w-25 {
width: 25% !important;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Border Radius Border Radius
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.brd-2 { border-radius: 2px; } .brd-2 {
border-radius: 2px;
}
/* -------------------------------------------------------- /* --------------------------------------------------------
Alignment Alignment
-----------------------------------------------------------*/ -----------------------------------------------------------*/
.va-top { vertical-align: top; } .va-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,102 +1,107 @@
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;
border: none;
padding-top: 10px;
position: relative;
height: 100%;
.schema-loading-state {
display: flex;
align-items: center;
justify-content: center;
height: 100%;
}
.collapse.in {
background: transparent;
}
.copy-to-editor {
color: fade(@redash-gray, 90%);
cursor: pointer;
position: absolute;
top: 0;
right: 0;
bottom: 0;
width: 20px;
display: flex;
align-items: center;
justify-content: center;
}
.table-open {
padding: 0 22px 0 26px;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; border: none;
white-space: nowrap; padding-top: 10px;
position: relative; position: relative;
height: 18px; height: 100%;
.column-type { .schema-loading-state {
color: fade(@text-color, 80%); display: flex;
font-size: 10px; align-items: center;
margin-left: 2px; justify-content: center;
text-transform: uppercase; height: 100%;
}
.collapse.in {
background: transparent;
} }
.copy-to-editor { .copy-to-editor {
display: none; visibility: hidden;
color: fade(@redash-gray, 90%);
width: 20px;
display: flex;
align-items: center;
justify-content: center;
transition: none;
} }
&:hover { .schema-list-item {
background: fade(@redash-gray, 10%); display: flex;
border-radius: @redash-radius;
height: 22px;
.copy-to-editor { .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-item {
display: flex; display: flex;
height: 18px;
width: calc(100% - 22px);
padding-left: 22px;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
transition: none;
div:first-child {
flex: 1;
}
.column-type {
color: fade(@text-color, 80%);
font-size: 10px;
margin-left: 2px;
text-transform: uppercase;
}
&:hover,
&:focus,
&:focus-within {
background: fade(@redash-gray, 10%);
.copy-to-editor {
visibility: visible;
}
}
} }
} }
} }
}
.schema-control { .schema-control {
display: flex; display: flex;
flex-wrap: nowrap; flex-wrap: nowrap;
padding: 0; padding: 0;
.ant-btn { .ant-btn {
height: auto; height: auto;
}
}
.parameter-label {
display: block;
} }
} }
.parameter-label {
display: block;
}

View File

@@ -103,7 +103,7 @@
padding-top: 5px !important; padding-top: 5px !important;
} }
.btn-favourite, .btn-favorite,
.btn-archive { .btn-archive {
font-size: 15px; font-size: 15px;
} }
@@ -114,18 +114,23 @@
line-height: 1.7 !important; line-height: 1.7 !important;
} }
.btn-favourite { .btn-favorite {
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 {
color: @yellow-darker; filter: saturate(75%);
opacity: 0.75;
}
} }
} }

View File

@@ -127,11 +127,13 @@ body.fixed-layout {
} }
} }
a.label-tag { .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%);
} }
@@ -141,6 +143,7 @@ a.label-tag {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
flex-grow: 1; flex-grow: 1;
position: relative;
} }
.query-fullscreen { .query-fullscreen {

View File

@@ -1,10 +1,11 @@
import { first } from "lodash"; import React, { useMemo } from "react";
import React, { useState } from "react"; import { first, includes } from "lodash";
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";
@@ -15,83 +16,109 @@ 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 MenuUnfoldOutlinedIcon from "@ant-design/icons/MenuUnfoldOutlined";
import MenuFoldOutlinedIcon from "@ant-design/icons/MenuFoldOutlined";
import VersionInfo from "./VersionInfo"; import VersionInfo from "./VersionInfo";
import "./DesktopNavbar.less"; import "./DesktopNavbar.less";
function NavbarSection({ inlineCollapsed, children, ...props }) { function NavbarSection({ children, ...props }) {
return ( return (
<Menu <Menu selectable={false} mode="vertical" theme="dark" {...props}>
selectable={false}
mode={inlineCollapsed ? "inline" : "vertical"}
inlineCollapsed={inlineCollapsed}
theme="dark"
{...props}>
{children} {children}
</Menu> </Menu>
); );
} }
export default function DesktopNavbar() { function useNavbarActiveState() {
const [collapsed, setCollapsed] = useState(true); 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() {
const firstSettingsTab = first(settingsMenu.getAvailableItems()); const firstSettingsTab = first(settingsMenu.getAvailableItems());
const activeState = useNavbarActiveState();
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 (
<div className="desktop-navbar"> <nav className="desktop-navbar">
<NavbarSection inlineCollapsed={collapsed} className="desktop-navbar-logo"> <NavbarSection className="desktop-navbar-logo">
<div> <div role="menuitem">
<Link href="./"> <Link href="./">
<img src={logoUrl} alt="Redash" /> <img src={logoUrl} alt="Redash" />
</Link> </Link>
</div> </div>
</NavbarSection> </NavbarSection>
<NavbarSection inlineCollapsed={collapsed}> <NavbarSection>
{currentUser.hasPermission("list_dashboards") && ( {currentUser.hasPermission("list_dashboards") && (
<Menu.Item key="dashboards"> <Menu.Item key="dashboards" className={activeState.dashboards ? "navbar-active-item" : null}>
<Link href="dashboards"> <Link href="dashboards">
<DesktopOutlinedIcon /> <DesktopOutlinedIcon aria-label="Dashboard navigation button" />
<span>Dashboards</span> <span className="desktop-navbar-label">Dashboards</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
{currentUser.hasPermission("view_query") && ( {currentUser.hasPermission("view_query") && (
<Menu.Item key="queries"> <Menu.Item key="queries" className={activeState.queries ? "navbar-active-item" : null}>
<Link href="queries"> <Link href="queries">
<CodeOutlinedIcon /> <CodeOutlinedIcon aria-label="Queries navigation button" />
<span>Queries</span> <span className="desktop-navbar-label">Queries</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
{currentUser.hasPermission("list_alerts") && ( {currentUser.hasPermission("list_alerts") && (
<Menu.Item key="alerts"> <Menu.Item key="alerts" className={activeState.alerts ? "navbar-active-item" : null}>
<Link href="alerts"> <Link href="alerts">
<AlertOutlinedIcon /> <AlertOutlinedIcon aria-label="Alerts navigation button" />
<span>Alerts</span> <span className="desktop-navbar-label">Alerts</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
</NavbarSection> </NavbarSection>
<NavbarSection inlineCollapsed={collapsed} className="desktop-navbar-spacer"> <NavbarSection 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 && (
@@ -103,9 +130,9 @@ export default function DesktopNavbar() {
)} )}
{canCreateDashboard && ( {canCreateDashboard && (
<Menu.Item key="new-dashboard"> <Menu.Item key="new-dashboard">
<a data-test="CreateDashboardMenuItem" onMouseUp={() => CreateDashboardDialog.showModal()}> <PlainButton data-test="CreateDashboardMenuItem" onClick={() => CreateDashboardDialog.showModal()}>
New Dashboard New Dashboard
</a> </PlainButton>
</Menu.Item> </Menu.Item>
)} )}
{canCreateAlert && ( {canCreateAlert && (
@@ -119,32 +146,31 @@ export default function DesktopNavbar() {
)} )}
</NavbarSection> </NavbarSection>
<NavbarSection inlineCollapsed={collapsed}> <NavbarSection>
<Menu.Item key="help"> <Menu.Item key="help">
<HelpTrigger showTooltip={false} type="HOME"> <HelpTrigger showTooltip={false} type="HOME" tabIndex={0}>
<QuestionCircleOutlinedIcon /> <QuestionCircleOutlinedIcon />
<span>Help</span> <span className="desktop-navbar-label">Help</span>
</HelpTrigger> </HelpTrigger>
</Menu.Item> </Menu.Item>
{firstSettingsTab && ( {firstSettingsTab && (
<Menu.Item key="settings"> <Menu.Item key="settings" className={activeState.dataSources ? "navbar-active-item" : null}>
<Link href={firstSettingsTab.path} data-test="SettingsLink"> <Link href={firstSettingsTab.path} data-test="SettingsLink">
<SettingOutlinedIcon /> <SettingOutlinedIcon />
<span>Settings</span> <span className="desktop-navbar-label">Settings</span>
</Link> </Link>
</Menu.Item> </Menu.Item>
)} )}
<Menu.Divider />
</NavbarSection> </NavbarSection>
<NavbarSection inlineCollapsed={collapsed} className="desktop-navbar-profile-menu"> <NavbarSection 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">
@@ -157,20 +183,16 @@ export default function DesktopNavbar() {
)} )}
<Menu.Divider /> <Menu.Divider />
<Menu.Item key="logout"> <Menu.Item key="logout">
<a data-test="LogOutButton" onClick={() => Auth.logout()}> <PlainButton data-test="LogOutButton" onClick={() => Auth.logout()}>
Log out Log out
</a> </PlainButton>
</Menu.Item> </Menu.Item>
<Menu.Divider /> <Menu.Divider />
<Menu.Item key="version" disabled className="version-info"> <Menu.Item key="version" role="presentation" 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,12 +1,17 @@
@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;
@@ -21,12 +26,6 @@
height: 40px; height: 40px;
transition: all 270ms; transition: all 270ms;
} }
&.ant-menu-inline-collapsed {
img {
height: 20px;
}
}
} }
.help-trigger { .help-trigger {
@@ -34,33 +33,38 @@
} }
.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 {
@@ -71,21 +75,33 @@
.ant-menu-submenu-arrow { .ant-menu-submenu-arrow {
display: none; display: none;
} }
}
.ant-btn.desktop-navbar-collapse-button { .ant-menu-item,
background-color: @backgroundColor; .ant-menu-submenu {
border: 0; padding: 0;
border-radius: 0; height: 60px;
color: @textColor; display: flex;
align-items: center;
&:hover, flex-direction: column;
&:active { justify-content: center;
color: #fff;
} }
&:after { .ant-menu-submenu-title {
animation: 0s !important; width: 100%;
padding: 0;
}
a,
&.ant-menu-vertical > .ant-menu-submenu > .ant-menu-submenu-title,
.ant-menu-submenu-title {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
line-height: normal;
height: auto;
background: none;
color: inherit;
} }
} }
@@ -99,37 +115,8 @@
.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;
}
} }
} }
} }
@@ -146,7 +133,9 @@
color: @textColor; color: @textColor;
&:hover, &:hover,
&:active { &:active,
&:focus,
&:focus-within {
color: #fff; color: #fff;
} }
@@ -171,7 +160,9 @@
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 Update Available <i className="fa fa-external-link m-l-5" aria-hidden="true" />
<i className="fa fa-external-link m-l-5" /> <span className="sr-only">(opens in a new tab)</span>
</Link> </Link>
</div> </div>
)} )}

View File

@@ -13,19 +13,21 @@ export default function ApplicationLayout({ children }) {
return ( return (
<React.Fragment> <React.Fragment>
<div className="application-layout-side-menu"> <DynamicComponent name="ApplicationWrapper">
<DynamicComponent name="ApplicationDesktopNavbar"> <div className="application-layout-side-menu">
<DesktopNavbar /> <DynamicComponent name="ApplicationDesktopNavbar">
</DynamicComponent> <DesktopNavbar />
</div>
<div className="application-layout-content">
<nav className="application-layout-top-menu" ref={mobileNavbarContainerRef}>
<DynamicComponent name="ApplicationMobileNavbar" getPopupContainer={getMobileNavbarPopupContainer}>
<MobileNavbar getPopupContainer={getMobileNavbarPopupContainer} />
</DynamicComponent> </DynamicComponent>
</nav> </div>
{children} <div className="application-layout-content">
</div> <nav className="application-layout-top-menu" ref={mobileNavbarContainerRef}>
<DynamicComponent name="ApplicationMobileNavbar" getPopupContainer={getMobileNavbarPopupContainer}>
<MobileNavbar getPopupContainer={getMobileNavbarPopupContainer} />
</DynamicComponent>
</nav>
{children}
</div>
</DynamicComponent>
</React.Fragment> </React.Fragment>
); );
} }

View File

@@ -1,8 +1,10 @@
import { isObject, get } from "lodash"; import { get, isObject } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import "./ErrorMessage.less"; import "./ErrorMessage.less";
import DynamicComponent from "@/components/DynamicComponent";
import { ErrorMessageDetails } from "@/components/ApplicationArea/ErrorMessageDetails";
function getErrorMessageByStatus(status, defaultMessage) { function getErrorMessageByStatus(status, defaultMessage) {
switch (status) { switch (status) {
@@ -31,21 +33,30 @@ function getErrorMessage(error) {
return message; return message;
} }
export default function ErrorMessage({ error }) { export default function ErrorMessage({ error, message }) {
if (!error) { if (!error) {
return null; return null;
} }
console.error(error); console.error(error);
const errorDetailsProps = {
error,
message: message || getErrorMessage(error),
};
return ( return (
<div className="error-message-container" data-test="ErrorMessage"> <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" /> <i className="zmdi zmdi-alert-circle-o" aria-hidden="true" />
</div> </div>
<div className="error-state__details"> <div className="error-state__details">
<h4>{getErrorMessage(error)}</h4> <DynamicComponent
name="ErrorMessageDetails"
fallback={<ErrorMessageDetails {...errorDetailsProps} />}
{...errorDetailsProps}
/>
</div> </div>
</div> </div>
</div> </div>
@@ -54,4 +65,5 @@ export default function ErrorMessage({ error }) {
ErrorMessage.propTypes = { ErrorMessage.propTypes = {
error: PropTypes.object.isRequired, error: PropTypes.object.isRequired,
message: PropTypes.string,
}; };

View File

@@ -0,0 +1,11 @@
import React from "react";
import PropTypes from "prop-types";
export function ErrorMessageDetails(props) {
return <h4>{props.message}</h4>;
}
ErrorMessageDetails.propTypes = {
error: PropTypes.instanceOf(Error).isRequired,
message: PropTypes.string.isRequired,
};

View File

@@ -9,7 +9,7 @@ export default function handleNavigationIntent(event) {
} }
element = element.parentNode; element = element.parentNode;
} }
if (!element || !element.hasAttribute("href") || element.hasAttribute("download")) { if (!element || !element.hasAttribute("href") || element.hasAttribute("download") || element.dataset.skipRouter) {
return; return;
} }

View File

@@ -1,7 +1,7 @@
import React, { useEffect, useState, useContext } from "react"; import React, { useEffect, useState, useContext } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import { ErrorBoundaryContext } from "@redash/viz/lib/components/ErrorBoundary"; import { ErrorBoundaryContext } from "@redash/viz/lib/components/ErrorBoundary";
import { Auth } from "@/services/auth"; import { Auth, clientConfig } from "@/services/auth";
// This wrapper modifies `route.render` function and instead of passing `currentRoute` passes an object // This wrapper modifies `route.render` function and instead of passing `currentRoute` passes an object
// that contains: // that contains:
@@ -33,7 +33,7 @@ function ApiKeySessionWrapper({ apiKey, currentRoute, renderChildren }) {
}; };
}, [apiKey]); }, [apiKey]);
if (!isAuthenticated) { if (!isAuthenticated || clientConfig.disablePublicUrls) {
return null; return null;
} }

View File

@@ -1,21 +1,32 @@
import React, { useEffect, useState } from "react"; import React, { useEffect, useState } from "react";
import PropTypes from "prop-types";
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";
import { CurrentRoute } from "@/services/routes";
import organizationStatus from "@/services/organizationStatus"; import organizationStatus from "@/services/organizationStatus";
import DynamicComponent from "@/components/DynamicComponent";
import ApplicationLayout from "./ApplicationLayout"; import ApplicationLayout from "./ApplicationLayout";
import ErrorMessage from "./ErrorMessage"; import ErrorMessage from "./ErrorMessage";
export type UserSessionWrapperRenderChildrenProps<P> = {
pageTitle?: string;
onError: (error: Error) => void;
} & P;
export interface UserSessionWrapperProps<P> {
render: (props: UserSessionWrapperRenderChildrenProps<P>) => React.ReactNode;
currentRoute: CurrentRoute<P>;
bodyClass?: string;
}
// This wrapper modifies `route.render` function and instead of passing `currentRoute` passes an object // This wrapper modifies `route.render` function and instead of passing `currentRoute` passes an object
// that contains: // that contains:
// - `currentRoute.routeParams` // - `currentRoute.routeParams`
// - `pageTitle` field which is equal to `currentRoute.title` // - `pageTitle` field which is equal to `currentRoute.title`
// - `onError` field which is a `handleError` method of nearest error boundary // - `onError` field which is a `handleError` method of nearest error boundary
function UserSessionWrapper({ bodyClass, currentRoute, renderChildren }) { export function UserSessionWrapper<P>({ bodyClass, currentRoute, render }: UserSessionWrapperProps<P>) {
const [isAuthenticated, setIsAuthenticated] = useState(!!Auth.isAuthenticated()); const [isAuthenticated, setIsAuthenticated] = useState(!!Auth.isAuthenticated());
useEffect(() => { useEffect(() => {
let isCancelled = false; let isCancelled = false;
Promise.all([Auth.requireSession(), organizationStatus.refresh(), policy.refresh()]) Promise.all([Auth.requireSession(), organizationStatus.refresh(), policy.refresh()])
@@ -50,11 +61,14 @@ function UserSessionWrapper({ bodyClass, currentRoute, renderChildren }) {
return ( return (
<ApplicationLayout> <ApplicationLayout>
<React.Fragment key={currentRoute.key}> <React.Fragment key={currentRoute.key}>
<ErrorBoundary renderError={error => <ErrorMessage error={error} />}> {/* @ts-expect-error FIXME */}
<ErrorBoundary renderError={(error: Error) => <ErrorMessage error={error} />}>
<ErrorBoundaryContext.Consumer> <ErrorBoundaryContext.Consumer>
{({ handleError }) => {(
renderChildren({ ...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>
@@ -62,21 +76,35 @@ function UserSessionWrapper({ bodyClass, currentRoute, renderChildren }) {
); );
} }
UserSessionWrapper.propTypes = { export type RouteWithUserSessionOptions<P> = {
bodyClass: PropTypes.string, render: (props: UserSessionWrapperRenderChildrenProps<P>) => React.ReactNode;
renderChildren: PropTypes.func, bodyClass?: string;
title: string;
path: string;
}; };
UserSessionWrapper.defaultProps = { export const UserSessionWrapperDynamicComponentName = "UserSessionWrapper";
bodyClass: null,
renderChildren: () => null,
};
export default function routeWithUserSession({ render, bodyClass, ...rest }) { export default function routeWithUserSession<P extends {} = {}>({
render: originalRender,
bodyClass,
...rest
}: RouteWithUserSessionOptions<P>) {
return { return {
...rest, ...rest,
render: currentRoute => ( render: (currentRoute: CurrentRoute<P>) => {
<UserSessionWrapper bodyClass={bodyClass} currentRoute={currentRoute} renderChildren={render} /> const props = {
), render: originalRender,
bodyClass,
currentRoute,
};
return (
<DynamicComponent
{...props}
name={UserSessionWrapperDynamicComponentName}
fallback={<UserSessionWrapper {...props} />}
/>
);
},
}; };
} }

View File

@@ -1,14 +1,21 @@
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 className={"p-15 text-center " + className}> <div
<h3 className="m-t-0 m-b-0"> className={"big-message p-15 text-center " + className}
<i className={"fa " + icon} /> role="status"
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 />
{message} <span id={messageId}>{message}</span>
{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 "antd/lib/tooltip"; import Tooltip from "@/components/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 '~antd/lib/button/style/index'; @import (reference, less) "~@/assets/less/ant";
.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 } from "lodash"; import { isEmpty, toUpper, includes, get, uniqueId } 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,6 +45,8 @@ 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 });
}; };
@@ -82,6 +84,7 @@ 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"
@@ -111,11 +114,12 @@ 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" /> Setup Instructions <i className="fa fa-question-circle" aria-hidden="true" />
<span className="sr-only">(help)</span>
</HelpTrigger> </HelpTrigger>
)} )}
</div> </div>
<DynamicForm id="sourceForm" fields={fields} onSubmit={this.createSource} feedbackIcons hideSubmitButton /> <DynamicForm id={this.formId} 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{" "}
@@ -139,7 +143,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" /> <i className="fa fa-angle-double-right" aria-hidden="true" />
</PreviewCard> </PreviewCard>
</List.Item> </List.Item>
); );
@@ -169,7 +173,7 @@ class CreateSourceDialog extends React.Component {
<Button <Button
key="submit" key="submit"
htmlType="submit" htmlType="submit"
form="sourceForm" form={this.formId}
type="primary" type="primary"
loading={savingSource} loading={savingSource}
data-test="CreateSourceSaveButton"> data-test="CreateSourceSaveButton">

View File

@@ -22,8 +22,8 @@ export function wrap<ROk = void, P = {}, RCancel = void>(
props?: P props?: P
) => { ) => {
update: (props: P) => void; update: (props: P) => void;
onClose: (handler: (result: ROk) => Promise<void>) => void; onClose: (handler: (result: ROk) => Promise<void> | void) => void;
onDismiss: (handler: (result: RCancel) => Promise<void>) => void; onDismiss: (handler: (result: RCancel) => Promise<void> | void) => void;
close: (result: ROk) => void; close: (result: ROk) => void;
dismiss: (result: RCancel) => void; dismiss: (result: RCancel) => void;
}; };

View File

@@ -1,4 +1,4 @@
import { isFunction, isString } from "lodash"; import { isFunction, isString, isUndefined } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
@@ -24,6 +24,7 @@ export function unregisterComponent(name) {
export default class DynamicComponent extends React.Component { export default class DynamicComponent extends React.Component {
static propTypes = { static propTypes = {
name: PropTypes.string.isRequired, name: PropTypes.string.isRequired,
fallback: PropTypes.node,
children: PropTypes.node, children: PropTypes.node,
}; };
@@ -40,10 +41,11 @@ export default class DynamicComponent extends React.Component {
} }
render() { render() {
const { name, children, ...props } = this.props; const { name, children, fallback, ...props } = this.props;
const RealComponent = componentsRegistry.get(name); const RealComponent = componentsRegistry.get(name);
if (!RealComponent) { if (!RealComponent) {
return children; // return fallback if any, otherwise return children
return isUndefined(fallback) ? children : fallback;
} }
return <RealComponent {...props}>{children}</RealComponent>; return <RealComponent {...props}>{children}</RealComponent>;
} }

View File

@@ -86,6 +86,7 @@ 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

@@ -11,6 +11,7 @@ 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";
const { Option } = Select; const { Option } = Select;
const formItemProps = { labelCol: { span: 6 }, wrapperCol: { span: 16 } }; const formItemProps = { labelCol: { span: 6 }, wrapperCol: { span: 16 } };
@@ -111,6 +112,8 @@ 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}
@@ -125,12 +128,12 @@ function EditParameterSettingsDialog(props) {
htmlType="submit" htmlType="submit"
disabled={!isFulfilled()} disabled={!isFulfilled()}
type="primary" type="primary"
form="paramForm" form={paramFormId}
data-test="SaveParameterSettings"> data-test="SaveParameterSettings">
{isNew ? "Add Parameter" : "OK"} {isNew ? "Add Parameter" : "OK"}
</Button>, </Button>,
]}> ]}>
<Form layout="horizontal" onFinish={onConfirm} id="paramForm"> <Form layout="horizontal" onFinish={onConfirm} id={paramFormId}>
{isNew && ( {isNew && (
<NameInput <NameInput
name={param.name} name={param.name}
@@ -140,7 +143,7 @@ function EditParameterSettingsDialog(props) {
type={param.type} type={param.type}
/> />
)} )}
<Form.Item label="Title" {...formItemProps}> <Form.Item required label="Title" {...formItemProps}>
<Input <Input
value={isNull(param.title) ? getDefaultTitle(param.name) : param.title} value={isNull(param.title) ? getDefaultTitle(param.name) : param.title}
onChange={e => setParam({ ...param, title: e.target.value })} onChange={e => setParam({ ...param, title: e.target.value })}

View File

@@ -3,6 +3,8 @@ 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 PlusCircleFilledIcon from "@ant-design/icons/PlusCircleFilled"; import PlusCircleFilledIcon from "@ant-design/icons/PlusCircleFilled";
import ShareAltOutlinedIcon from "@ant-design/icons/ShareAltOutlined"; import ShareAltOutlinedIcon from "@ant-design/icons/ShareAltOutlined";
@@ -17,16 +19,18 @@ 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>
<a target="_self" onClick={() => props.openAddToDashboardForm(props.selectedTab)}> <PlainButton onClick={() => props.openAddToDashboardForm(props.selectedTab)}>
<PlusCircleFilledIcon /> Add to Dashboard <PlusCircleFilledIcon /> Add to Dashboard
</a> </PlainButton>
</Menu.Item> </Menu.Item>
)} )}
{!props.query.isNew() && ( {!clientConfig.disablePublicUrls && !props.query.isNew() && (
<Menu.Item> <Menu.Item>
<a onClick={() => props.showEmbedDialog(props.query, props.selectedTab)} data-test="ShowEmbedDialogButton"> <PlainButton
onClick={() => props.showEmbedDialog(props.query, props.selectedTab)}
data-test="ShowEmbedDialogButton">
<ShareAltOutlinedIcon /> Embed Elsewhere <ShareAltOutlinedIcon /> Embed Elsewhere
</a> </PlainButton>
</Menu.Item> </Menu.Item>
)} )}
<Menu.Item> <Menu.Item>

View File

@@ -1,12 +1,14 @@
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 "antd/lib/tooltip"; import Tooltip from "@/components/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;
} }
@@ -16,7 +18,7 @@ export default function EmailSettingsWarning({ featureName, className, mode, adm
} }
const message = ( const message = (
<span> <span id={messageDescriptionId}>
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>
@@ -24,8 +26,11 @@ export default function EmailSettingsWarning({ featureName, className, mode, adm
if (mode === "icon") { if (mode === "icon") {
return ( return (
<Tooltip title={message}> <Tooltip title={message} placement="topRight" arrowPointAtCenter>
<i className={cx("fa fa-exclamation-triangle", className)} /> {/* eslint-disable-next-line jsx-a11y/no-noninteractive-tabindex */}
<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,5 +1,6 @@
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 = {
@@ -29,12 +30,13 @@ 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 (
<a <PlainButton
title={title} title={title}
className="favorites-control btn-favourite" aria-label={title}
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" />
</a> </PlainButton>
); );
} }
} }

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" /> <i className="fa fa-square-o m-r-5" aria-hidden="true" />
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" /> <i className="fa fa-check-square-o m-r-5" aria-hidden="true" />
Select All Select All
</Select.Option>, </Select.Option>,
<Select.OptGroup key="Values" title="Values"> <Select.OptGroup key="Values" title="Values">

View File

@@ -1,13 +1,14 @@
import { startsWith, get } from "lodash"; 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 "antd/lib/tooltip"; import Tooltip from "@/components/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 from "@/components/DynamicComponent"; import DynamicComponent, { registerComponent } from "@/components/DynamicComponent";
import "./HelpTrigger.less"; import "./HelpTrigger.less";
@@ -16,204 +17,249 @@ const HELP_PATH = "/help";
const IFRAME_TIMEOUT = 20000; const IFRAME_TIMEOUT = 20000;
const IFRAME_URL_UPDATE_MESSAGE = "iframe_url"; const IFRAME_URL_UPDATE_MESSAGE = "iframe_url";
export const TYPES = { export const TYPES = mapValues(
HOME: ["", "Help"], {
VALUE_SOURCE_OPTIONS: ["/user-guide/querying/query-parameters#Value-Source-Options", "Guide: Value Source Options"], HOME: ["", "Help"],
SHARE_DASHBOARD: ["/user-guide/dashboards/sharing-dashboards", "Guide: Sharing and Embedding Dashboards"], VALUE_SOURCE_OPTIONS: ["/user-guide/querying/query-parameters#Value-Source-Options", "Guide: Value Source Options"],
AUTHENTICATION_OPTIONS: ["/user-guide/users/authentication-options", "Guide: Authentication Options"], SHARE_DASHBOARD: ["/user-guide/dashboards/sharing-dashboards", "Guide: Sharing and Embedding Dashboards"],
USAGE_DATA_SHARING: ["/open-source/admin-guide/usage-data", "Help: Anonymous Usage Data Sharing"], AUTHENTICATION_OPTIONS: ["/user-guide/users/authentication-options", "Guide: Authentication Options"],
DS_ATHENA: ["/data-sources/amazon-athena-setup", "Guide: Help Setting up Amazon Athena"], USAGE_DATA_SHARING: ["/open-source/admin-guide/usage-data", "Help: Anonymous Usage Data Sharing"],
DS_BIGQUERY: ["/data-sources/bigquery-setup", "Guide: Help Setting up BigQuery"], DS_ATHENA: ["/data-sources/amazon-athena-setup", "Guide: Help Setting up Amazon Athena"],
DS_URL: ["/data-sources/querying-urls", "Guide: Help Setting up URL"], DS_BIGQUERY: ["/data-sources/bigquery-setup", "Guide: Help Setting up BigQuery"],
DS_MONGODB: ["/data-sources/mongodb-setup", "Guide: Help Setting up MongoDB"], DS_URL: ["/data-sources/querying-urls", "Guide: Help Setting up URL"],
DS_GOOGLE_SPREADSHEETS: ["/data-sources/querying-a-google-spreadsheet", "Guide: Help Setting up Google Spreadsheets"], DS_MONGODB: ["/data-sources/mongodb-setup", "Guide: Help Setting up MongoDB"],
DS_GOOGLE_ANALYTICS: ["/data-sources/google-analytics-setup", "Guide: Help Setting up Google Analytics"], DS_GOOGLE_SPREADSHEETS: [
DS_AXIBASETSD: ["/data-sources/axibase-time-series-database", "Guide: Help Setting up Axibase Time Series"], "/data-sources/querying-a-google-spreadsheet",
DS_RESULTS: ["/user-guide/querying/query-results-data-source", "Guide: Help Setting up Query Results"], "Guide: Help Setting up Google Spreadsheets",
ALERT_SETUP: ["/user-guide/alerts/setting-up-an-alert", "Guide: Setting Up a New Alert"], ],
MAIL_CONFIG: ["/open-source/setup/#Mail-Configuration", "Guide: Mail Configuration"], DS_GOOGLE_ANALYTICS: ["/data-sources/google-analytics-setup", "Guide: Help Setting up Google Analytics"],
ALERT_NOTIF_TEMPLATE_GUIDE: ["/user-guide/alerts/custom-alert-notifications", "Guide: Custom Alerts Notifications"], DS_AXIBASETSD: ["/data-sources/axibase-time-series-database", "Guide: Help Setting up Axibase Time Series"],
FAVORITES: ["/user-guide/querying/favorites-tagging/#Favorites", "Guide: Favorites"], DS_RESULTS: ["/user-guide/querying/query-results-data-source", "Guide: Help Setting up Query Results"],
MANAGE_PERMISSIONS: [ ALERT_SETUP: ["/user-guide/alerts/setting-up-an-alert", "Guide: Setting Up a New Alert"],
"/user-guide/querying/writing-queries#Managing-Query-Permissions", MAIL_CONFIG: ["/open-source/setup/#Mail-Configuration", "Guide: Mail Configuration"],
"Guide: Managing Query Permissions", ALERT_NOTIF_TEMPLATE_GUIDE: ["/user-guide/alerts/custom-alert-notifications", "Guide: Custom Alerts Notifications"],
], FAVORITES: ["/user-guide/querying/favorites-tagging/#Favorites", "Guide: Favorites"],
NUMBER_FORMAT_SPECS: ["/user-guide/visualizations/formatting-numbers", "Formatting Numbers"], MANAGE_PERMISSIONS: [
"/user-guide/querying/writing-queries#Managing-Query-Permissions",
"Guide: Managing Query Permissions",
],
NUMBER_FORMAT_SPECS: ["/user-guide/visualizations/formatting-numbers", "Formatting Numbers"],
GETTING_STARTED: ["/user-guide/getting-started", "Guide: Getting Started"],
DASHBOARDS: ["/user-guide/dashboards", "Guide: Dashboards"],
QUERIES: ["/user-guide/querying", "Guide: Queries"],
ALERTS: ["/user-guide/alerts", "Guide: Alerts"],
},
([url, title]) => [DOMAIN + HELP_PATH + url, title]
);
const HelpTriggerPropTypes = {
type: PropTypes.string,
href: PropTypes.string,
title: PropTypes.node,
className: PropTypes.string,
showTooltip: PropTypes.bool,
renderAsLink: PropTypes.bool,
children: PropTypes.node,
}; };
export default class HelpTrigger extends React.Component { const HelpTriggerDefaultProps = {
static propTypes = { type: null,
type: PropTypes.oneOf(Object.keys(TYPES)), href: null,
href: PropTypes.string, title: null,
title: PropTypes.node, className: null,
className: PropTypes.string, showTooltip: true,
showTooltip: PropTypes.bool, renderAsLink: false,
children: PropTypes.node, children: <i className="fa fa-question-circle" aria-hidden="true" />,
}; };
static defaultProps = { export function helpTriggerWithTypes(types, allowedDomains = [], drawerClassName = null) {
type: null, return class HelpTrigger extends React.Component {
href: null, static propTypes = {
title: null, ...HelpTriggerPropTypes,
className: null, type: PropTypes.oneOf(Object.keys(types)),
showTooltip: true, };
children: <i className="fa fa-question-circle" />,
};
iframeRef = React.createRef(); static defaultProps = HelpTriggerDefaultProps;
iframeLoadingTimeout = null; iframeRef = React.createRef();
state = { iframeLoadingTimeout = null;
visible: false,
loading: false,
error: false,
currentUrl: null,
};
componentDidMount() { state = {
window.addEventListener("message", this.onPostMessageReceived, false); visible: false,
} loading: false,
error: false,
currentUrl: null,
};
componentWillUnmount() { componentDidMount() {
window.removeEventListener("message", this.onPostMessageReceived); window.addEventListener("message", this.onPostMessageReceived, false);
clearTimeout(this.iframeLoadingTimeout);
}
loadIframe = url => {
clearTimeout(this.iframeLoadingTimeout);
this.setState({ loading: true, error: false });
this.iframeRef.current.src = url;
this.iframeLoadingTimeout = setTimeout(() => {
this.setState({ error: url, loading: false });
}, IFRAME_TIMEOUT); // safety
};
onIframeLoaded = () => {
this.setState({ loading: false });
clearTimeout(this.iframeLoadingTimeout);
};
onPostMessageReceived = event => {
if (!startsWith(event.origin, DOMAIN)) {
return;
} }
const { type, message: currentUrl } = event.data || {}; componentWillUnmount() {
if (type !== IFRAME_URL_UPDATE_MESSAGE) { window.removeEventListener("message", this.onPostMessageReceived);
return; clearTimeout(this.iframeLoadingTimeout);
} }
this.setState({ currentUrl }); loadIframe = url => {
}; clearTimeout(this.iframeLoadingTimeout);
this.setState({ loading: true, error: false });
getUrl = () => { this.iframeRef.current.src = url;
const helpTriggerType = get(TYPES, this.props.type); this.iframeLoadingTimeout = setTimeout(() => {
return helpTriggerType ? DOMAIN + HELP_PATH + helpTriggerType[0] : this.props.href; this.setState({ error: url, loading: false });
}; }, IFRAME_TIMEOUT); // safety
};
openDrawer = () => { onIframeLoaded = () => {
this.setState({ visible: true }); this.setState({ loading: false });
// wait for drawer animation to complete so there's no animation jank clearTimeout(this.iframeLoadingTimeout);
setTimeout(() => this.loadIframe(this.getUrl()), 300); };
};
closeDrawer = event => { onPostMessageReceived = event => {
if (event) { if (!some(allowedDomains, domain => startsWith(event.origin, domain))) {
event.preventDefault(); return;
} }
this.setState({ visible: false });
this.setState({ visible: false, currentUrl: null });
};
render() { const { type, message: currentUrl } = event.data || {};
const tooltip = get(TYPES, `${this.props.type}[1]`, this.props.title); if (type !== IFRAME_URL_UPDATE_MESSAGE) {
const className = cx("help-trigger", this.props.className); return;
const url = this.state.currentUrl; }
const isAllowedDomain = startsWith(url || this.getUrl(), DOMAIN); this.setState({ currentUrl });
};
return ( getUrl = () => {
<React.Fragment> const helpTriggerType = get(types, this.props.type);
<Tooltip return helpTriggerType ? helpTriggerType[0] : this.props.href;
title={ };
this.props.showTooltip ? (
<> openDrawer = e => {
{tooltip} // keep "open in new tab" behavior
{!isAllowedDomain && <i className="fa fa-external-link" style={{ marginLeft: 5 }} />} if (!e.shiftKey && !e.ctrlKey && !e.metaKey) {
</> e.preventDefault();
) : null this.setState({ visible: true });
}> // wait for drawer animation to complete so there's no animation jank
{isAllowedDomain ? ( setTimeout(() => this.loadIframe(this.getUrl()), 300);
<a onClick={this.openDrawer} className={className}> }
{this.props.children} };
</a>
) : ( closeDrawer = event => {
<Link href={url || this.getUrl()} className={className} rel="noopener noreferrer" target="_blank"> if (event) {
event.preventDefault();
}
this.setState({ visible: false });
this.setState({ visible: false, currentUrl: null });
};
render() {
const targetUrl = this.getUrl();
if (!targetUrl) {
return null;
}
const tooltip = get(types, `${this.props.type}[1]`, this.props.title);
const className = cx("help-trigger", this.props.className);
const url = this.state.currentUrl;
const isAllowedDomain = some(allowedDomains, domain => startsWith(url || targetUrl, domain));
const shouldRenderAsLink = this.props.renderAsLink || !isAllowedDomain;
return (
<React.Fragment>
<Tooltip
title={
this.props.showTooltip ? (
<>
{tooltip}
{shouldRenderAsLink && (
<>
{" "}
<i className="fa fa-external-link" style={{ marginLeft: 5 }} aria-hidden="true" />
<span className="sr-only">(opens in a new tab)</span>
</>
)}
</>
) : null
}>
<Link
href={url || this.getUrl()}
className={className}
rel="noopener noreferrer"
target="_blank"
onClick={shouldRenderAsLink ? () => {} : this.openDrawer}>
{this.props.children} {this.props.children}
</Link> </Link>
)} </Tooltip>
</Tooltip> <Drawer
<Drawer placement="right"
placement="right" closable={false}
closable={false} onClose={this.closeDrawer}
onClose={this.closeDrawer} visible={this.state.visible}
visible={this.state.visible} className={cx("help-drawer", drawerClassName)}
className="help-drawer" destroyOnClose
destroyOnClose width={400}>
width={400}> <div className="drawer-wrapper">
<div className="drawer-wrapper"> <div className="drawer-menu">
<div className="drawer-menu"> {url && (
{url && ( <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 title="Close" placement="bottom">
<PlainButton onClick={this.closeDrawer}>
<CloseOutlinedIcon />
</PlainButton>
</Tooltip> </Tooltip>
</div>
{/* iframe */}
{!this.state.error && (
<iframe
ref={this.iframeRef}
title="Usage Help"
src="about:blank"
className={cx({ ready: !this.state.loading })}
onLoad={this.onIframeLoaded}
/>
)}
{/* loading indicator */}
{this.state.loading && (
<BigMessage icon="fa-spinner fa-2x fa-pulse" message="Loading..." className="help-message" />
)}
{/* error message */}
{this.state.error && (
<BigMessage icon="fa-exclamation-circle" className="help-message">
Something went wrong.
<br />
{/* eslint-disable-next-line react/jsx-no-target-blank */}
<Link href={this.state.error} target="_blank" rel="noopener">
Click here
</Link>{" "}
to open the page in a new window.
</BigMessage>
)} )}
<Tooltip title="Close" placement="bottom">
<a onClick={this.closeDrawer}>
<CloseOutlinedIcon />
</a>
</Tooltip>
</div> </div>
{/* iframe */} {/* extra content */}
{!this.state.error && ( <DynamicComponent name="HelpDrawerExtraContent" onLeave={this.closeDrawer} openPageUrl={this.loadIframe} />
<iframe </Drawer>
ref={this.iframeRef} </React.Fragment>
title="Redash Help" );
src="about:blank" }
className={cx({ ready: !this.state.loading })} };
onLoad={this.onIframeLoaded}
/>
)}
{/* loading indicator */}
{this.state.loading && (
<BigMessage icon="fa-spinner fa-2x fa-pulse" message="Loading..." className="help-message" />
)}
{/* error message */}
{this.state.error && (
<BigMessage icon="fa-exclamation-circle" className="help-message">
Something went wrong.
<br />
{/* eslint-disable-next-line react/jsx-no-target-blank */}
<Link href={this.state.error} target="_blank" rel="noopener">
Click here
</Link>{" "}
to open the page in a new window.
</BigMessage>
)}
</div>
{/* extra content */}
<DynamicComponent name="HelpDrawerExtraContent" onLeave={this.closeDrawer} openPageUrl={this.loadIframe} />
</Drawer>
</React.Fragment>
);
}
} }
registerComponent("HelpTrigger", helpTriggerWithTypes(TYPES, [DOMAIN]));
export default function HelpTrigger(props) {
return <DynamicComponent {...props} name="HelpTrigger" />;
}
HelpTrigger.propTypes = HelpTriggerPropTypes;
HelpTrigger.defaultProps = HelpTriggerDefaultProps;

View File

@@ -1,4 +1,4 @@
@import "~antd/lib/drawer/style/drawer"; @import (reference, less) "~@/assets/less/ant";
@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,7 +38,8 @@
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

@@ -1,7 +1,8 @@
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 "antd/lib/tooltip"; import Tooltip from "@/components/Tooltip";
import PlainButton from "./PlainButton";
export default class InputWithCopy extends React.Component { export default class InputWithCopy extends React.Component {
constructor(props) { constructor(props) {
@@ -42,7 +43,10 @@ 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"}>
<CopyOutlinedIcon style={{ cursor: "pointer" }} onClick={this.copy} /> <PlainButton onClick={this.copy}>
{/* TODO: lacks visual feedback */}
<CopyOutlinedIcon />
</PlainButton>
</Tooltip> </Tooltip>
); );

View File

@@ -1,26 +0,0 @@
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 {...props} />;
}
function ButtonLink(props) {
return <ButtonLink.Component {...props} />;
}
ButtonLink.Component = DefaultButtonLinkComponent;
Link.Button = ButtonLink;
export default Link;

View File

@@ -0,0 +1,61 @@
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,21 +2,26 @@ 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 "antd/lib/tooltip"; import Tooltip from "@/components/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 ? "spinner fa-pulse" : "check"; const icon = !paramCount ? (
<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}> <Button onClick={onClick}>{icon} Apply Changes</Button>
<i className={`fa fa-${icon}`} /> Apply Changes
</Button>
</span> </span>
</Tooltip> </Tooltip>
</Badge> </Badge>

View File

@@ -12,7 +12,7 @@ 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 "antd/lib/tooltip"; import Tooltip from "@/components/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";
@@ -25,8 +25,6 @@ import CheckOutlinedIcon from "@ant-design/icons/CheckOutlined";
import "./ParameterMappingInput.less"; import "./ParameterMappingInput.less";
const { Option } = Select;
export const MappingType = { export const MappingType = {
DashboardAddNew: "dashboard-add-new", DashboardAddNew: "dashboard-add-new",
DashboardMapToExisting: "dashboard-map-to-existing", DashboardMapToExisting: "dashboard-map-to-existing",
@@ -203,24 +201,20 @@ export class ParameterMappingInput extends React.Component {
const { const {
mapping: { mapTo }, mapping: { mapTo },
} = this.props; } = this.props;
return <Input value={mapTo} onChange={e => this.updateParamMapping({ mapTo: e.target.value })} />; return (
<Input
value={mapTo}
aria-label="Parameter name (key)"
onChange={e => this.updateParamMapping({ mapTo: e.target.value })}
/>
);
} }
renderDashboardMapToExisting() { renderDashboardMapToExisting() {
const { mapping, existingParamNames } = this.props; const { mapping, existingParamNames } = this.props;
const options = map(existingParamNames, paramName => ({ label: paramName, value: paramName }));
return ( return <Select value={mapping.mapTo} onChange={mapTo => this.updateParamMapping({ mapTo })} options={options} />;
<Select
value={mapping.mapTo}
onChange={mapTo => this.updateParamMapping({ mapTo })}
dropdownMatchSelectWidth={false}>
{map(existingParamNames, name => (
<Option value={name} key={name}>
{name}
</Option>
))}
</Select>
);
} }
renderStaticValue() { renderStaticValue() {
@@ -358,7 +352,7 @@ class MappingEditor extends React.Component {
content={this.renderContent()} content={this.renderContent()}
visible={visible} visible={visible}
onVisibleChange={this.onVisibleChange}> onVisibleChange={this.onVisibleChange}>
<Button size="small" type="dashed" data-test={`EditParamMappingButon-${mapping.param.name}`}> <Button size="small" type="dashed" data-test={`EditParamMappingButton-${mapping.param.name}`}>
<EditOutlinedIcon /> <EditOutlinedIcon />
</Button> </Button>
</Popover> </Popover>
@@ -432,6 +426,7 @@ 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}
@@ -452,7 +447,10 @@ 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">
<i className="fa fa-eye-slash" /> {/* eslint-disable-next-line jsx-a11y/no-noninteractive-tabindex */}
<span tabIndex={0}>
<i className="fa fa-eye-slash" aria-hidden="true" />
</span>
</Tooltip> </Tooltip>
); );
} }

View File

@@ -1,4 +1,4 @@
@import '~antd/lib/modal/style/index'; // for ant @vars @import (reference, less) "~@/assets/less/ant"; // for ant @vars
.parameters-mapping-list { .parameters-mapping-list {
.keyword { .keyword {
@@ -63,7 +63,8 @@
margin-right: 3px; margin-right: 3px;
} }
&.disabled, .fa { &.disabled,
.fa {
color: #a4a4a4; color: #a4a4a4;
} }

View File

@@ -1,7 +1,7 @@
import { isEqual, isEmpty } from "lodash"; import { isEqual, isEmpty, map } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Select from "antd/lib/select"; import SelectWithVirtualScroll from "@/components/SelectWithVirtualScroll";
import Input from "antd/lib/input"; import Input from "antd/lib/input";
import InputNumber from "antd/lib/input-number"; import InputNumber from "antd/lib/input-number";
import DateParameter from "@/components/dynamic-parameters/DateParameter"; import DateParameter from "@/components/dynamic-parameters/DateParameter";
@@ -10,8 +10,6 @@ import QueryBasedParameterInput from "./QueryBasedParameterInput";
import "./ParameterValueInput.less"; import "./ParameterValueInput.less";
const { Option } = Select;
const multipleValuesProps = { const multipleValuesProps = {
maxTagCount: 3, maxTagCount: 3,
maxTagTextLength: 10, maxTagTextLength: 10,
@@ -98,25 +96,19 @@ class ParameterValueInput extends React.Component {
const enumOptionsArray = enumOptions.split("\n").filter(v => v !== ""); const enumOptionsArray = enumOptions.split("\n").filter(v => v !== "");
// Antd Select doesn't handle null in multiple mode // Antd Select doesn't handle null in multiple mode
const normalize = val => (parameter.multiValuesOptions && val === null ? [] : val); const normalize = val => (parameter.multiValuesOptions && val === null ? [] : val);
return ( return (
<Select <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}
dropdownMatchSelectWidth={false} options={map(enumOptionsArray, opt => ({ label: String(opt), value: opt }))}
showSearch showSearch
showArrow showArrow
style={{ minWidth: 60 }}
notFoundContent={isEmpty(enumOptionsArray) ? "No options available" : null} notFoundContent={isEmpty(enumOptionsArray) ? "No options available" : null}
{...multipleValuesProps}> {...multipleValuesProps}
{enumOptionsArray.map(option => ( />
<Option key={option} value={option}>
{option}
</Option>
))}
</Select>
); );
} }
@@ -127,7 +119,6 @@ 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}
@@ -145,7 +136,12 @@ class ParameterValueInput extends React.Component {
const normalize = val => (isNaN(val) ? undefined : val); const normalize = val => (isNaN(val) ? undefined : val);
return ( return (
<InputNumber className={className} value={normalize(value)} onChange={val => this.onSelect(normalize(val))} /> <InputNumber
className={className}
value={normalize(value)}
aria-label="Parameter number value"
onChange={val => this.onSelect(normalize(val))}
/>
); );
} }
@@ -157,6 +153,7 @@ 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 "~antd/lib/input-number/style/index"; // for ant @vars @import (reference, less) "~@/assets/less/ant"; // for ant @vars
@input-dirty: #fffce1; @input-dirty: #fffce1;

View File

@@ -6,6 +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 { toHuman } from "@/lib/utils";
@@ -23,19 +24,23 @@ 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) {
@@ -85,7 +90,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(); onParametersEdit(parameters);
return { parameters }; return { parameters };
}); });
} }
@@ -110,7 +115,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(); onParametersEdit(parameters);
return { parameters }; return { parameters };
}); });
}); });
@@ -123,13 +128,14 @@ export default class Parameters extends React.Component {
<div className="parameter-heading"> <div className="parameter-heading">
<label>{param.title || toHuman(param.name)}</label> <label>{param.title || toHuman(param.name)}</label>
{editable && ( {editable && (
<button <PlainButton
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" /> <i className="fa fa-cog" aria-hidden="true" />
</button> </PlainButton>
)} )}
</div> </div>
<ParameterValueInput <ParameterValueInput
@@ -146,15 +152,17 @@ export default class Parameters extends React.Component {
render() { render() {
const { parameters } = this.state; const { parameters } = this.state;
const { editable } = this.props; const { sortable, appendSortableToParent } = this.props;
const dirtyParamCount = size(filter(parameters, "hasPendingValue")); const dirtyParamCount = size(filter(parameters, "hasPendingValue"));
return ( return (
<SortableContainer <SortableContainer
disabled={!editable} disabled={!sortable}
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={{
@@ -163,8 +171,11 @@ 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 className="parameter-block" data-editable={editable || null}> <div
{editable && <DragHandle data-test={`DragHandle-${param.name}`} />} className="parameter-block"
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 "../assets/less/ant"; @import (reference, less) "~@/assets/less/ant";
.parameter-block { .parameter-block {
display: inline-block; display: inline-block;
@@ -21,6 +21,8 @@
&.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,11 +7,12 @@ 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 "antd/lib/tooltip"; import Tooltip from "@/components/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";
@@ -102,7 +103,16 @@ function UserSelect({ onSelect, shouldShowUser }) {
placeholder="Add users..." placeholder="Add users..."
showSearch showSearch
onSearch={setSearchTerm} onSearch={setSearchTerm}
suffixIcon={loadingUsers ? <i className="fa fa-spinner fa-pulse" /> : <i className="fa fa-search" />} suffixIcon={
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}
@@ -156,7 +166,12 @@ 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 && <i className="fa fa-spinner fa-pulse" />} {loadingGrantees && (
<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
@@ -169,10 +184,11 @@ 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">
<i <PlainButton
className="fa fa-remove clickable" aria-label="Remove permissions"
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

@@ -0,0 +1,22 @@
@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

@@ -0,0 +1,20 @@
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,9 +1,7 @@
import { find, isArray, get, first, map, intersection, isEqual, isEmpty } from "lodash"; import { find, isArray, get, first, map, intersection, isEqual, isEmpty } from "lodash";
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import Select from "antd/lib/select"; import SelectWithVirtualScroll from "@/components/SelectWithVirtualScroll";
const { Option } = Select;
export default class QueryBasedParameterInput extends React.Component { export default class QueryBasedParameterInput extends React.Component {
static propTypes = { static propTypes = {
@@ -79,29 +77,23 @@ export default class QueryBasedParameterInput extends React.Component {
} }
render() { render() {
const { className, value, mode, onSelect, ...otherProps } = this.props; const { className, mode, onSelect, queryId, value, ...otherProps } = this.props;
const { loading, options } = this.state; const { loading, options } = this.state;
return ( return (
<span> <span>
<Select <SelectWithVirtualScroll
className={className} className={className}
disabled={loading} disabled={loading}
loading={loading} loading={loading}
mode={mode} mode={mode}
value={this.state.value} value={this.state.value}
onChange={onSelect} onChange={onSelect}
dropdownMatchSelectWidth={false} options={map(options, ({ value, name }) => ({ label: String(name), value }))}
optionFilterProp="children"
showSearch showSearch
showArrow showArrow
notFoundContent={isEmpty(options) ? "No options available" : null} notFoundContent={isEmpty(options) ? "No options available" : null}
{...otherProps}> {...otherProps}
{options.map(option => ( />
<Option value={option.value} key={option.value}>
{option.name}
</Option>
))}
</Select>
</span> </span>
); );
} }

View File

@@ -21,10 +21,12 @@ 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 (
<Link href={readOnly ? null : getUrl()} className="query-link"> <QueryLinkWrapper className="query-link">
<VisualizationName visualization={visualization} /> <span>{query.name}</span> <VisualizationName visualization={visualization} /> <span>{query.name}</span>
</Link> </QueryLinkWrapper>
); );
} }

View File

@@ -5,6 +5,7 @@ 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";
@@ -30,8 +31,21 @@ 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 = <i className="fa fa-times hide-in-percy" onClick={() => selectQuery(null)} />; const clearIcon = (
const spinIcon = <i className={cx("fa fa-spinner fa-pulse hide-in-percy", { hidden: !searching })} />; <i
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);
@@ -65,22 +79,25 @@ export default function QuerySelector(props) {
} }
return ( return (
<div className="list-group"> <ul className="list-group">
{searchResults.map(q => ( {searchResults.map(q => (
<a <PlainButton
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" />
</a> </PlainButton>
))} ))}
</div> </ul>
); );
} }
if (props.disabled) { if (props.disabled) {
return <Input value={selectedQuery && selectedQuery.name} placeholder={placeholder} disabled />; return (
<Input value={selectedQuery && selectedQuery.name} aria-label="Tied query" placeholder={placeholder} disabled />
);
} }
if (props.type === "select") { if (props.type === "select") {
@@ -127,11 +144,12 @@ export default function QuerySelector(props) {
return ( return (
<span data-test="QuerySelector"> <span data-test="QuerySelector">
{selectedQuery ? ( {selectedQuery ? (
<Input value={selectedQuery.name} suffix={clearIcon} readOnly /> <Input value={selectedQuery.name} aria-label="Tied query" 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,9 +51,12 @@ 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,13 +12,16 @@ 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("p-l-10", "p-r-10", { clickable: !isDisabled, disabled: isDisabled }, className)} className={classNames("select-items-list", "w-100", "p-l-10", "p-r-10", { disabled: isDisabled }, className)}
onClick={isDisabled ? null : () => onItemClick(item)}> onClick={isDisabled ? null : () => onItemClick(item)}>
{content} {content}
</List.Item> </List.Item>
@@ -117,7 +120,12 @@ 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 onChange={event => search(event.target.value)} placeholder={inputPlaceholder} autoFocus /> <Input.Search
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

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

View File

@@ -0,0 +1,45 @@
import React, { useMemo } from "react";
import { maxBy } from "lodash";
import AntdSelect, { SelectProps, LabeledValue } from "antd/lib/select";
import { calculateTextWidth } from "@/lib/calculateTextWidth";
const MIN_LEN_FOR_VIRTUAL_SCROLL = 400;
interface VirtualScrollLabeledValue extends LabeledValue {
label: string;
}
interface VirtualScrollSelectProps extends Omit<SelectProps<string>, "optionFilterProp" | "children"> {
options: Array<VirtualScrollLabeledValue>;
}
function SelectWithVirtualScroll({ options, ...props }: VirtualScrollSelectProps): JSX.Element {
const dropdownMatchSelectWidth = useMemo<number | boolean>(() => {
if (options && options.length > MIN_LEN_FOR_VIRTUAL_SCROLL) {
const largestOpt = maxBy(options, "label.length");
if (largestOpt) {
const offset = 40;
const optionText = largestOpt.label;
const width = calculateTextWidth(optionText);
if (width) {
return width + offset;
}
}
return true;
}
return false;
}, [options]);
return (
<AntdSelect<string>
dropdownMatchSelectWidth={dropdownMatchSelectWidth}
options={options}
optionFilterProp="label" // as this component expects "options" prop
{...props}
/>
);
}
export default SelectWithVirtualScroll;

View File

@@ -1,82 +0,0 @@
import { map } from "lodash";
import React from "react";
import PropTypes from "prop-types";
import Badge from "antd/lib/badge";
import Menu from "antd/lib/menu";
import getTags from "@/services/getTags";
import "./TagsList.less";
export default class TagsList extends React.Component {
static propTypes = {
tagsUrl: PropTypes.string.isRequired,
onUpdate: PropTypes.func,
};
static defaultProps = {
onUpdate: () => {},
};
constructor(props) {
super(props);
this.state = {
// An array of objects that with the name and count of the tagged items
allTags: [],
// A set of tag names
selectedTags: new Set(),
};
}
componentDidMount() {
getTags(this.props.tagsUrl).then(allTags => {
this.setState({ allTags });
});
}
toggleTag(event, tag) {
const { selectedTags } = this.state;
if (event.shiftKey) {
// toggle tag
if (selectedTags.has(tag)) {
selectedTags.delete(tag);
} else {
selectedTags.add(tag);
}
} else {
// if the tag is the only selected, deselect it, otherwise select only it
if (selectedTags.has(tag) && selectedTags.size === 1) {
selectedTags.clear();
} else {
selectedTags.clear();
selectedTags.add(tag);
}
}
this.forceUpdate();
this.props.onUpdate([...this.state.selectedTags]);
}
render() {
const { allTags, selectedTags } = this.state;
if (allTags.length > 0) {
return (
<div className="m-t-10 tags-list tiled">
<Menu className="invert-stripe-position" mode="inline" selectedKeys={[...selectedTags]}>
{map(allTags, tag => (
<Menu.Item key={tag.name} className="m-0">
<a
className="d-flex align-items-center justify-content-between"
onClick={event => this.toggleTag(event, tag.name)}>
<span className="max-character col-xs-11">{tag.name}</span>
<Badge count={tag.count} />
</a>
</Menu.Item>
))}
</Menu>
</div>
);
}
return null;
}
}

View File

@@ -1,15 +1,58 @@
@import '~@/assets/less/ant'; @import (reference, less) "~@/assets/less/ant";
.tags-list { .tags-list {
.tags-list-title {
margin: 15px 5px 5px 5px;
display: flex;
justify-content: space-between;
align-items: center;
.tags-list-label {
display: block;
white-space: nowrap;
margin: 0;
}
a,
.plain-button {
display: block;
white-space: nowrap;
cursor: pointer;
.anticon {
font-size: 75%;
margin-right: 2px;
}
}
}
.ant-badge-count { .ant-badge-count {
background-color: fade(@redash-gray, 10%); background-color: fade(@redash-gray, 10%);
color: fade(@redash-gray, 75%); color: fade(@redash-gray, 75%);
} }
.ant-menu-item-selected { .ant-menu.ant-menu-inline {
.ant-badge-count { border: none;
background-color: @primary-color;
color: white; .ant-menu-item {
width: 100%;
}
.ant-menu-item-selected {
.ant-badge-count {
background-color: @primary-color;
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

@@ -0,0 +1,108 @@
import { map, includes, difference } from "lodash";
import React, { useState, useCallback, useEffect } from "react";
import Badge from "antd/lib/badge";
import Menu from "antd/lib/menu";
import CloseOutlinedIcon from "@ant-design/icons/CloseOutlined";
import getTags from "@/services/getTags";
import PlainButton from "@/components/PlainButton";
import "./TagsList.less";
type Tag = {
name: string;
count?: number;
};
type TagsListProps = {
tagsUrl: string;
showUnselectAll: boolean;
onUpdate?: (selectedTags: string[]) => void;
};
function TagsList({ tagsUrl, showUnselectAll = false, onUpdate }: TagsListProps): JSX.Element | null {
const [allTags, setAllTags] = useState<Tag[]>([]);
const [selectedTags, setSelectedTags] = useState<string[]>([]);
useEffect(() => {
let isCancelled = false;
getTags(tagsUrl).then(tags => {
if (!isCancelled) {
setAllTags(tags);
}
});
return () => {
isCancelled = true;
};
}, [tagsUrl]);
const toggleTag = useCallback(
(event, tag) => {
let newSelectedTags;
if (event.shiftKey) {
// toggle tag
if (includes(selectedTags, tag)) {
newSelectedTags = difference(selectedTags, [tag]);
} else {
newSelectedTags = [...selectedTags, tag];
}
} else {
// if the tag is the only selected, deselect it, otherwise select only it
if (includes(selectedTags, tag) && selectedTags.length === 1) {
newSelectedTags = [];
} else {
newSelectedTags = [tag];
}
}
setSelectedTags(newSelectedTags);
if (onUpdate) {
onUpdate([...newSelectedTags]);
}
},
[selectedTags, onUpdate]
);
const unselectAll = useCallback(() => {
setSelectedTags([]);
if (onUpdate) {
onUpdate([]);
}
}, [onUpdate]);
if (allTags.length === 0) {
return null;
}
return (
<div className="tags-list">
<div className="tags-list-title">
<span className="tags-list-label">Tags</span>
{showUnselectAll && selectedTags.length > 0 && (
<PlainButton type="link" onClick={unselectAll}>
<CloseOutlinedIcon />
clear selection
</PlainButton>
)}
</div>
<div className="tiled">
<Menu className="invert-stripe-position" mode="inline" selectedKeys={selectedTags}>
{map(allTags, tag => (
<Menu.Item key={tag.name} className="m-0">
<PlainButton
className="d-flex align-items-center justify-content-between"
onClick={event => toggleTag(event, tag.name)}>
<span className="max-character col-xs-11">{tag.name}</span>
<Badge count={tag.count} />
</PlainButton>
</Menu.Item>
))}
</Menu>
</div>
</div>
);
}
export default TagsList;

View File

@@ -4,14 +4,14 @@ 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 "antd/lib/tooltip"; import Tooltip from "@/components/Tooltip";
function toMoment(value) { function toMoment(value) {
value = !isNil(value) ? moment(value) : null; value = !isNil(value) ? moment(value) : null;
return value && value.isValid() ? value : null; return value && value.isValid() ? value : null;
} }
export default function TimeAgo({ date, placeholder, autoUpdate }) { export default function TimeAgo({ date, placeholder, autoUpdate, variation }) {
const startDate = toMoment(date); const startDate = toMoment(date);
const [value, setValue] = useState(null); const [value, setValue] = useState(null);
const title = useMemo(() => (startDate ? startDate.format(clientConfig.dateTimeFormat) : null), [startDate]); const title = useMemo(() => (startDate ? startDate.format(clientConfig.dateTimeFormat) : null), [startDate]);
@@ -28,6 +28,13 @@ export default function TimeAgo({ date, placeholder, autoUpdate }) {
} }
}, [autoUpdate, startDate, placeholder]); }, [autoUpdate, startDate, placeholder]);
if (variation === "timeAgoInTooltip") {
return (
<Tooltip title={value}>
<span data-test="TimeAgo">{title}</span>
</Tooltip>
);
}
return ( return (
<Tooltip title={title}> <Tooltip title={title}>
<span data-test="TimeAgo">{value}</span> <span data-test="TimeAgo">{value}</span>
@@ -39,6 +46,7 @@ TimeAgo.propTypes = {
date: PropTypes.oneOfType([PropTypes.string, PropTypes.number, PropTypes.instanceOf(Date), Moment]), date: PropTypes.oneOfType([PropTypes.string, PropTypes.number, PropTypes.instanceOf(Date), Moment]),
placeholder: PropTypes.string, placeholder: PropTypes.string,
autoUpdate: PropTypes.bool, autoUpdate: PropTypes.bool,
variation: PropTypes.oneOf(["timeAgoInTooltip"]),
}; };
TimeAgo.defaultProps = { TimeAgo.defaultProps = {

View File

@@ -0,0 +1,13 @@
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

@@ -4,13 +4,13 @@ import PropTypes from "prop-types";
import Tag from "antd/lib/tag"; import Tag from "antd/lib/tag";
import Link from "@/components/Link"; import Link from "@/components/Link";
export default function UserGroups({ groups, ...props }) { import "./UserGroups.less";
export default function UserGroups({ groups, linkGroups, ...props }) {
return ( return (
<div {...props}> <div className="user-groups" {...props}>
{map(groups, group => ( {map(groups, group => (
<Tag className="m-b-5 m-r-5" key={group.id}> <Tag key={group.id}>{linkGroups ? <Link href={`groups/${group.id}`}>{group.name}</Link> : group.name}</Tag>
<Link href={`groups/${group.id}`}>{group.name}</Link>
</Tag>
))} ))}
</div> </div>
); );
@@ -19,12 +19,14 @@ export default function UserGroups({ groups, ...props }) {
UserGroups.propTypes = { UserGroups.propTypes = {
groups: PropTypes.arrayOf( groups: PropTypes.arrayOf(
PropTypes.shape({ PropTypes.shape({
id: PropTypes.number.isRequired, id: PropTypes.oneOfType([PropTypes.number, PropTypes.string]).isRequired,
name: PropTypes.string, name: PropTypes.string,
}) })
), ),
linkGroups: PropTypes.bool,
}; };
UserGroups.defaultProps = { UserGroups.defaultProps = {
groups: [], groups: [],
linkGroups: true,
}; };

View File

@@ -0,0 +1,7 @@
.user-groups {
margin: -5px 0 0 -5px;
.ant-tag {
margin: 5px 0 0 5px;
}
}

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,84 +0,0 @@
import Input from "antd/lib/input";
import { includes, isEmpty } from "lodash";
import PropTypes from "prop-types";
import React from "react";
import Link from "@/components/Link";
import EmptyState from "@/components/items-list/components/EmptyState";
import "./CardsList.less";
const { Search } = Input;
export default class CardsList extends React.Component {
static propTypes = {
items: PropTypes.arrayOf(
PropTypes.shape({
title: PropTypes.string.isRequired,
imgSrc: PropTypes.string.isRequired,
onClick: PropTypes.func,
href: PropTypes.string,
})
),
showSearch: PropTypes.bool,
};
static defaultProps = {
items: [],
showSearch: false,
};
state = {
searchText: "",
};
constructor(props) {
super(props);
this.items = [];
let itemId = 1;
props.items.forEach(item => {
this.items.push({ id: itemId, ...item });
itemId += 1;
});
}
// eslint-disable-next-line class-methods-use-this
renderListItem(item) {
return (
<Link key={`card${item.id}`} className="visual-card" onClick={item.onClick} href={item.href}>
<img alt={item.title} src={item.imgSrc} />
<h3>{item.title}</h3>
</Link>
);
}
render() {
const { showSearch } = this.props;
const { searchText } = this.state;
const filteredItems = this.items.filter(
item => isEmpty(searchText) || includes(item.title.toLowerCase(), searchText.toLowerCase())
);
return (
<div data-test="CardsList">
{showSearch && (
<div className="row p-10">
<div className="col-md-4 col-md-offset-4">
<Search placeholder="Search..." onChange={e => this.setState({ searchText: e.target.value })} autoFocus />
</div>
</div>
)}
{isEmpty(filteredItems) ? (
<EmptyState className="" />
) : (
<div className="row">
<div className="col-lg-12 d-inline-flex flex-wrap visual-card-list">
{filteredItems.map(item => this.renderListItem(item))}
</div>
</div>
)}
</div>
);
}
}

View File

@@ -1,5 +1,4 @@
@import (reference, less) "~@/assets/less/inc/variables";
@import '../../assets/less/inc/variables';
.visual-card-list { .visual-card-list {
width: 100%; width: 100%;
@@ -7,7 +6,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;
@@ -22,7 +21,9 @@
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;
} }
@@ -74,4 +75,4 @@
height: 48px; height: 48px;
} }
} }
} }

View File

@@ -0,0 +1,89 @@
import { includes, isEmpty } from "lodash";
import PropTypes from "prop-types";
import React, { useState } from "react";
import Input from "antd/lib/input";
import Link from "@/components/Link";
import PlainButton from "@/components/PlainButton";
import EmptyState from "@/components/items-list/components/EmptyState";
import "./CardsList.less";
export interface CardsListItem {
title: string;
imgSrc: string;
href?: string;
onClick?: React.MouseEventHandler<HTMLElement>;
}
export interface CardsListProps {
items?: CardsListItem[];
showSearch?: boolean;
}
interface ListItemProps {
item: CardsListItem;
keySuffix: string;
}
function ListItem({ item, keySuffix }: ListItemProps) {
const commonProps = {
key: `card${keySuffix}`,
className: "visual-card",
onClick: item.onClick,
children: (
<>
<img alt={item.title} src={item.imgSrc} />
<h3>{item.title}</h3>
</>
),
};
return item.href ? <Link href={item.href} {...commonProps} /> : <PlainButton type="link" {...commonProps} />;
}
export default function CardsList({ items = [], showSearch = false }: CardsListProps) {
const [searchText, setSearchText] = useState("");
const filteredItems = items.filter(
item => isEmpty(searchText) || includes(item.title.toLowerCase(), searchText.toLowerCase())
);
return (
<div data-test="CardsList">
{showSearch && (
<div className="row p-10">
<div className="col-md-4 col-md-offset-4">
<Input.Search
placeholder="Search..."
aria-label="Search cards"
onChange={(e: React.ChangeEvent<HTMLInputElement>) => setSearchText(e.target.value)}
autoFocus
/>
</div>
</div>
)}
{isEmpty(filteredItems) ? (
<EmptyState className="" />
) : (
<div className="row">
<div className="col-lg-12 d-inline-flex flex-wrap visual-card-list">
{filteredItems.map((item: CardsListItem, index: number) => (
<ListItem key={index} item={item} keySuffix={index.toString()} />
))}
</div>
</div>
)}
</div>
);
}
CardsList.propTypes = {
items: PropTypes.arrayOf(
PropTypes.shape({
title: PropTypes.string.isRequired,
imgSrc: PropTypes.string.isRequired,
onClick: PropTypes.func,
href: PropTypes.string,
})
),
showSearch: PropTypes.bool,
};

View File

@@ -8,12 +8,15 @@ 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;
@@ -29,9 +32,9 @@ function VisualizationSelect({ query, visualization, onChange }) {
return ( return (
<div> <div>
<div className="form-group"> <div className="form-group">
<label htmlFor="choose-visualization">Choose Visualization</label> <label htmlFor={vizSelectId}>Choose Visualization</label>
<Select <Select
id="choose-visualization" id={vizSelectId}
className="w-100" className="w-100"
value={visualization ? visualization.id : undefined} value={visualization ? visualization.id : undefined}
onChange={handleChange}> onChange={handleChange}>
@@ -108,6 +111,7 @@ 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
@@ -132,12 +136,12 @@ function AddWidgetDialog({ dialog, dashboard }) {
)} )}
{parameterMappings.length > 0 && [ {parameterMappings.length > 0 && [
<label key="parameters-title" htmlFor="parameter-mappings"> <label key="parameters-title" htmlFor={parameterMappingsId}>
Parameters Parameters
</label>, </label>,
<ParameterMappingListInput <ParameterMappingListInput
key="parameters-list" key="parameters-list"
id="parameter-mappings" id={parameterMappingsId}
mappings={parameterMappings} mappings={parameterMappings}
existingParams={existingParams} existingParams={existingParams}
onChange={setParameterMappings} onChange={setParameterMappings}

View File

@@ -60,6 +60,7 @@ 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,6 +41,7 @@ const DashboardWidget = React.memo(
onRefreshWidget, onRefreshWidget,
onRemoveWidget, onRemoveWidget,
onParameterMappingsChange, onParameterMappingsChange,
isEditing,
canEdit, canEdit,
isPublic, isPublic,
isLoading, isLoading,
@@ -57,6 +58,7 @@ 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}
@@ -77,7 +79,8 @@ 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 {
@@ -223,7 +226,6 @@ 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,
@@ -232,18 +234,21 @@ 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"
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={this.props.isEditing} isDraggable={isEditing}
isResizable={this.props.isEditing} isResizable={isEditing}
onResizeStart={this.autoHeightCtrl.stop} onResizeStart={this.autoHeightCtrl.stop}
onResizeStop={this.onWidgetResize} onResizeStop={this.onWidgetResize}
layouts={this.state.layouts} layouts={this.state.layouts}
@@ -265,6 +270,7 @@ 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

@@ -3,10 +3,11 @@ 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 { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper"; import { wrap as wrapDialog, DialogPropType } from "@/components/DialogWrapper";
import { FiltersType } from "@/components/Filters";
import VisualizationRenderer from "@/components/visualizations/VisualizationRenderer"; import VisualizationRenderer from "@/components/visualizations/VisualizationRenderer";
import VisualizationName from "@/components/visualizations/VisualizationName"; import VisualizationName from "@/components/visualizations/VisualizationName";
function ExpandedWidgetDialog({ dialog, widget }) { function ExpandedWidgetDialog({ dialog, widget, filters }) {
return ( return (
<Modal <Modal
{...dialog.props} {...dialog.props}
@@ -20,6 +21,7 @@ function ExpandedWidgetDialog({ dialog, widget }) {
<VisualizationRenderer <VisualizationRenderer
visualization={widget.visualization} visualization={widget.visualization}
queryResult={widget.getQueryResult()} queryResult={widget.getQueryResult()}
filters={filters}
context="widget" context="widget"
/> />
</Modal> </Modal>
@@ -29,6 +31,11 @@ function ExpandedWidgetDialog({ dialog, widget }) {
ExpandedWidgetDialog.propTypes = { ExpandedWidgetDialog.propTypes = {
dialog: DialogPropType.isRequired, dialog: DialogPropType.isRequired,
widget: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types widget: PropTypes.object.isRequired, // eslint-disable-line react/forbid-prop-types
filters: FiltersType,
};
ExpandedWidgetDialog.defaultProps = {
filters: [],
}; };
export default wrapDialog(ExpandedWidgetDialog); export default wrapDialog(ExpandedWidgetDialog);

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 "antd/lib/tooltip"; import Tooltip from "@/components/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,6 +73,7 @@ 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

@@ -48,10 +48,10 @@
top: 0; top: 0;
left: 0; left: 0;
bottom: 85px; bottom: 85px;
right: 15px; right: 0;
background: linear-gradient(to bottom, transparent, transparent 2px, #f6f8f9 2px, #f6f8f9 5px), background: linear-gradient(to bottom, transparent, transparent 2px, #f6f8f9 2px, #f6f8f9 5px),
linear-gradient(to left, #b3babf, #b3babf 1px, transparent 1px, transparent); linear-gradient(to left, #b3babf, #b3babf 1px, transparent 1px, transparent);
background-size: calc((100vw - 15px) / 6) 5px; background-size: calc((100% + 15px) / 6) 5px;
background-position: -7px 1px; background-position: -7px 1px;
} }
} }

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 } from "lodash"; import { compact, isEmpty, invoke, map } 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,9 +15,11 @@ 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 }) {
@@ -74,7 +76,8 @@ 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" /> <i className="zmdi zmdi-refresh zmdi-hc-spin" aria-hidden="true" />
<span className="sr-only">Refreshing...</span>
</div> </div>
<Timer from={refreshStartedAt} /> <Timer from={refreshStartedAt} />
</div> </div>
@@ -84,7 +87,14 @@ function RefreshIndicator({ refreshStartedAt }) {
RefreshIndicator.propTypes = { refreshStartedAt: Moment }; RefreshIndicator.propTypes = { refreshStartedAt: Moment };
RefreshIndicator.defaultProps = { refreshStartedAt: null }; RefreshIndicator.defaultProps = { refreshStartedAt: null };
function VisualizationWidgetHeader({ widget, refreshStartedAt, parameters, onParametersUpdate }) { function VisualizationWidgetHeader({
widget,
refreshStartedAt,
parameters,
isEditing,
onParametersUpdate,
onParametersEdit,
}) {
const canViewQuery = currentUser.hasPermission("view_query"); const canViewQuery = currentUser.hasPermission("view_query");
return ( return (
@@ -104,7 +114,13 @@ function VisualizationWidgetHeader({ widget, refreshStartedAt, parameters, onPar
</div> </div>
{!isEmpty(parameters) && ( {!isEmpty(parameters) && (
<div className="m-b-10"> <div className="m-b-10">
<Parameters parameters={parameters} onValuesChange={onParametersUpdate} /> <Parameters
parameters={parameters}
sortable={isEditing}
appendSortableToParent={false}
onValuesChange={onParametersUpdate}
onParametersEdit={onParametersEdit}
/>
</div> </div>
)} )}
</> </>
@@ -115,12 +131,16 @@ 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: [],
}; };
@@ -140,34 +160,40 @@ function VisualizationWidgetFooter({ widget, isPublic, onRefresh, onExpand }) {
<> <>
<span> <span>
{!isPublic && !!widgetQueryResult && ( {!isPublic && !!widgetQueryResult && (
<a <PlainButton
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 })} />{" "} <i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": refreshClickButtonId === 1 })} aria-hidden="true" />
<span className="sr-only">
{refreshClickButtonId === 1 ? "Refreshing, please wait. " : "Press to refresh. "}
</span>{" "}
<TimeAgo date={updatedAt} /> <TimeAgo date={updatedAt} />
</a> </PlainButton>
)} )}
<span className="visible-print"> <span className="visible-print">
<i className="zmdi zmdi-time-restore" /> {formatDateTime(updatedAt)} <i className="zmdi zmdi-time-restore" aria-hidden="true" /> {formatDateTime(updatedAt)}
</span> </span>
{isPublic && ( {isPublic && (
<span className="small hidden-print"> <span className="small hidden-print">
<i className="zmdi zmdi-time-restore" /> <TimeAgo date={updatedAt} /> <i className="zmdi zmdi-time-restore" aria-hidden="true" /> <TimeAgo date={updatedAt} />
</span> </span>
)} )}
</span> </span>
<span> <span>
{!isPublic && ( {!isPublic && (
<a <PlainButton
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 })} /> <i className={cx("zmdi zmdi-refresh", { "zmdi-hc-spin": refreshClickButtonId === 2 })} aria-hidden="true" />
</a> <span className="sr-only">
{refreshClickButtonId === 2 ? "Refreshing, please wait." : "Press to refresh."}
</span>
</PlainButton>
)} )}
<a className="btn btn-sm btn-default hidden-print btn-transparent btn__refresh" onClick={onExpand}> <PlainButton className="btn btn-sm btn-default hidden-print btn-transparent btn__refresh" onClick={onExpand}>
<i className="zmdi zmdi-fullscreen" /> <i className="zmdi zmdi-fullscreen" aria-hidden="true" />
</a> </PlainButton>
</span> </span>
</> </>
) : null; ) : null;
@@ -190,6 +216,7 @@ 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,
@@ -201,6 +228,7 @@ 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: () => {},
@@ -209,7 +237,10 @@ class VisualizationWidget extends React.Component {
constructor(props) { constructor(props) {
super(props); super(props);
this.state = { localParameters: props.widget.getLocalParameters() }; this.state = {
localParameters: props.widget.getLocalParameters(),
localFilters: props.filters,
};
} }
componentDidMount() { componentDidMount() {
@@ -219,8 +250,12 @@ class VisualizationWidget extends React.Component {
onLoad(); onLoad();
} }
onLocalFiltersChange = localFilters => {
this.setState({ localFilters });
};
expandWidget = () => { expandWidget = () => {
ExpandedWidgetDialog.showModal({ widget: this.props.widget }); ExpandedWidgetDialog.showModal({ widget: this.props.widget, filters: this.state.localFilters });
}; };
editParameterMappings = () => { editParameterMappings = () => {
@@ -260,15 +295,21 @@ class VisualizationWidget extends React.Component {
visualization={widget.visualization} visualization={widget.visualization}
queryResult={widgetQueryResult} queryResult={widgetQueryResult}
filters={filters} filters={filters}
onFiltersChange={this.onLocalFiltersChange}
context="widget" context="widget"
/> />
</div> </div>
); );
default: default:
return ( return (
<div className="body-row-auto spinner-container"> <div
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" /> <i className="zmdi zmdi-refresh zmdi-hc-spin zmdi-hc-5x" aria-hidden="true" />
<span className="sr-only">Loading...</span>
</div> </div>
</div> </div>
); );
@@ -276,10 +317,15 @@ class VisualizationWidget extends React.Component {
} }
render() { render() {
const { widget, isLoading, isPublic, canEdit, onRefresh } = this.props; const { widget, isLoading, isPublic, canEdit, isEditing, 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
@@ -295,7 +341,9 @@ 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,6 +7,7 @@ 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";
@@ -22,9 +23,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"]}>
<a className="action p-l-15 p-r-15" data-test="WidgetDropdownButton"> <PlainButton className="action p-l-15 p-r-15" data-test="WidgetDropdownButton" aria-label="More options">
<i className="zmdi zmdi-more-vert" /> <i className="zmdi zmdi-more-vert" aria-hidden="true" />
</a> </PlainButton>
</Dropdown> </Dropdown>
</div> </div>
); );
@@ -45,9 +46,14 @@ WidgetDropdownButton.defaultProps = {
function WidgetDeleteButton({ onClick }) { function WidgetDeleteButton({ onClick }) {
return ( return (
<div className="widget-menu-remove"> <div className="widget-menu-remove">
<a className="action" title="Remove From Dashboard" onClick={onClick} data-test="WidgetDeleteButton"> <PlainButton
<i className="zmdi zmdi-close" /> className="action"
</a> title="Remove From Dashboard"
onClick={onClick}
data-test="WidgetDeleteButton"
aria-label="Close">
<i className="zmdi zmdi-close" aria-hidden="true" />
</PlainButton>
</div> </div>
); );
} }

View File

@@ -1,12 +1,4 @@
@import "../../../assets/less/inc/variables"; @import (reference, less) "~@/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 {
@@ -22,10 +14,19 @@
line-height: 100%; line-height: 100%;
display: block; display: block;
padding: 4px 10px 3px; padding: 4px 10px 3px;
}
.action:hover { &:focus {
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%);
}
} }
} }
@@ -83,7 +84,7 @@
display: block; display: block;
} }
a.query-link { .query-link {
pointer-events: none; pointer-events: none;
cursor: move; cursor: move;
} }
@@ -190,10 +191,18 @@
.th-title { .th-title {
padding-right: 23px; // no overlap on RefreshIndicator padding-right: 23px; // no overlap on RefreshIndicator
a { .hidden-print {
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%);
}
} }
} }
@@ -212,7 +221,10 @@
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;
@@ -240,10 +252,12 @@
} }
} }
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,7 +201,10 @@ 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 className={cx("fa m-l-5", { "fa-caret-up": showExtraFields, "fa-caret-down": !showExtraFields })} /> <i
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 "~@/assets/less/ant"; @import (reference, less) "~@/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

@@ -1,14 +1,7 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import classNames from "classnames"; import { getDynamicDateFromString } from "@/services/parameters/DateParameter";
import moment from "moment"; import DynamicDatePicker from "@/components/dynamic-parameters/DynamicDatePicker";
import { includes } from "lodash";
import { isDynamicDate, getDynamicDateFromString } from "@/services/parameters/DateParameter";
import DateInput from "@/components/DateInput";
import DateTimeInput from "@/components/DateTimeInput";
import DynamicButton from "@/components/dynamic-parameters/DynamicButton";
import "./DynamicParameters.less";
const DYNAMIC_DATE_OPTIONS = [ const DYNAMIC_DATE_OPTIONS = [
{ {
@@ -29,87 +22,30 @@ const DYNAMIC_DATE_OPTIONS = [
}, },
]; ];
class DateParameter extends React.Component { function DateParameter(props) {
static propTypes = { return (
type: PropTypes.string, <DynamicDatePicker
className: PropTypes.string, dynamicButtonOptions={{ options: DYNAMIC_DATE_OPTIONS }}
value: PropTypes.any, // eslint-disable-line react/forbid-prop-types {...props}
parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types dateOptions={{ "aria-label": "Parameter date value" }}
onSelect: PropTypes.func, />
}; );
static defaultProps = {
type: "",
className: "",
value: null,
parameter: null,
onSelect: () => {},
};
constructor(props) {
super(props);
this.dateComponentRef = React.createRef();
}
onDynamicValueSelect = dynamicValue => {
const { onSelect, parameter } = this.props;
if (dynamicValue === "static") {
const parameterValue = parameter.getExecutionValue();
if (parameterValue) {
onSelect(moment(parameterValue));
} else {
onSelect(null);
}
} else {
onSelect(dynamicValue.value);
}
// give focus to the DatePicker to get keyboard shortcuts to work
this.dateComponentRef.current.focus();
};
render() {
const { type, value, className, onSelect } = this.props;
const hasDynamicValue = isDynamicDate(value);
const isDateTime = includes(type, "datetime");
const additionalAttributes = {};
let DateComponent = DateInput;
if (isDateTime) {
DateComponent = DateTimeInput;
if (includes(type, "with-seconds")) {
additionalAttributes.withSeconds = true;
}
}
if (moment.isMoment(value) || value === null) {
additionalAttributes.value = value;
}
if (hasDynamicValue) {
const dynamicDate = value;
additionalAttributes.placeholder = dynamicDate && dynamicDate.name;
additionalAttributes.value = null;
}
return (
<div className="date-parameter">
<DateComponent
ref={this.dateComponentRef}
className={classNames("redash-datepicker", { "dynamic-value": hasDynamicValue }, className)}
onSelect={onSelect}
suffixIcon={null}
{...additionalAttributes}
/>
<DynamicButton
options={DYNAMIC_DATE_OPTIONS}
selectedDynamicValue={hasDynamicValue ? value : null}
enabled={hasDynamicValue}
onSelect={this.onDynamicValueSelect}
/>
</div>
);
}
} }
DateParameter.propTypes = {
type: PropTypes.string,
className: PropTypes.string,
value: PropTypes.any, // eslint-disable-line react/forbid-prop-types
parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types
onSelect: PropTypes.func,
};
DateParameter.defaultProps = {
type: "",
className: "",
value: null,
parameter: null,
onSelect: () => {},
};
export default DateParameter; export default DateParameter;

View File

@@ -1,14 +1,8 @@
import React from "react"; import React from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import classNames from "classnames"; import { includes } from "lodash";
import moment from "moment"; import { getDynamicDateRangeFromString } from "@/services/parameters/DateRangeParameter";
import { includes, isArray, isObject } from "lodash"; import DynamicDateRangePicker from "@/components/dynamic-parameters/DynamicDateRangePicker";
import { isDynamicDateRange, getDynamicDateRangeFromString } from "@/services/parameters/DateRangeParameter";
import DateRangeInput from "@/components/DateRangeInput";
import DateTimeRangeInput from "@/components/DateTimeRangeInput";
import DynamicButton from "@/components/dynamic-parameters/DynamicButton";
import "./DynamicParameters.less";
const DYNAMIC_DATE_OPTIONS = [ const DYNAMIC_DATE_OPTIONS = [
{ {
@@ -134,98 +128,25 @@ const DYNAMIC_DATETIME_OPTIONS = [
...DYNAMIC_DATE_OPTIONS, ...DYNAMIC_DATE_OPTIONS,
]; ];
const widthByType = { function DateRangeParameter(props) {
"date-range": 294, const options = includes(props.type, "datetime-range") ? DYNAMIC_DATETIME_OPTIONS : DYNAMIC_DATE_OPTIONS;
"datetime-range": 352, return <DynamicDateRangePicker {...props} dynamicButtonOptions={{ options }} />;
"datetime-range-with-seconds": 382, }
DateRangeParameter.propTypes = {
type: PropTypes.string,
className: PropTypes.string,
value: PropTypes.any, // eslint-disable-line react/forbid-prop-types
parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types
onSelect: PropTypes.func,
}; };
function isValidDateRangeValue(value) { DateRangeParameter.defaultProps = {
return isArray(value) && value.length === 2 && moment.isMoment(value[0]) && moment.isMoment(value[1]); type: "",
} className: "",
value: null,
class DateRangeParameter extends React.Component { parameter: null,
static propTypes = { onSelect: () => {},
type: PropTypes.string, };
className: PropTypes.string,
value: PropTypes.any, // eslint-disable-line react/forbid-prop-types
parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types
onSelect: PropTypes.func,
};
static defaultProps = {
type: "",
className: "",
value: null,
parameter: null,
onSelect: () => {},
};
constructor(props) {
super(props);
this.dateRangeComponentRef = React.createRef();
}
onDynamicValueSelect = dynamicValue => {
const { onSelect, parameter } = this.props;
if (dynamicValue === "static") {
const parameterValue = parameter.getExecutionValue();
if (isObject(parameterValue) && parameterValue.start && parameterValue.end) {
onSelect([moment(parameterValue.start), moment(parameterValue.end)]);
} else {
onSelect(null);
}
} else {
onSelect(dynamicValue.value);
}
// give focus to the DatePicker to get keyboard shortcuts to work
this.dateRangeComponentRef.current.focus();
};
render() {
const { type, value, onSelect, className } = this.props;
const isDateTimeRange = includes(type, "datetime-range");
const hasDynamicValue = isDynamicDateRange(value);
const options = isDateTimeRange ? DYNAMIC_DATETIME_OPTIONS : DYNAMIC_DATE_OPTIONS;
const additionalAttributes = {};
let DateRangeComponent = DateRangeInput;
if (isDateTimeRange) {
DateRangeComponent = DateTimeRangeInput;
if (includes(type, "with-seconds")) {
additionalAttributes.withSeconds = true;
}
}
if (isValidDateRangeValue(value) || value === null) {
additionalAttributes.value = value;
}
if (hasDynamicValue) {
additionalAttributes.placeholder = [value && value.name];
additionalAttributes.value = null;
}
return (
<div className="data-range-parameter">
<DateRangeComponent
ref={this.dateRangeComponentRef}
className={classNames("redash-datepicker date-range-input", { "dynamic-value": hasDynamicValue }, className)}
onSelect={onSelect}
style={{ width: hasDynamicValue ? 195 : widthByType[type] }}
suffixIcon={null}
{...additionalAttributes}
/>
<DynamicButton
options={options}
selectedDynamicValue={hasDynamicValue ? value : null}
enabled={hasDynamicValue}
onSelect={this.onDynamicValueSelect}
/>
</div>
);
}
}
export default DateRangeParameter; export default DateRangeParameter;

View File

@@ -15,7 +15,7 @@ import "./DynamicButton.less";
const { Text } = Typography; const { Text } = Typography;
function DynamicButton({ options, selectedDynamicValue, onSelect, enabled }) { function DynamicButton({ options, selectedDynamicValue, onSelect, enabled, staticValueLabel }) {
const menu = ( const menu = (
<Menu <Menu
className="dynamic-menu" className="dynamic-menu"
@@ -32,7 +32,7 @@ function DynamicButton({ options, selectedDynamicValue, onSelect, enabled }) {
{enabled && ( {enabled && (
<Menu.Item> <Menu.Item>
<ArrowLeftOutlinedIcon /> <ArrowLeftOutlinedIcon />
<Text type="secondary">Back to Static Value</Text> <Text type="secondary">{staticValueLabel}</Text>
</Menu.Item> </Menu.Item>
)} )}
</Menu> </Menu>
@@ -42,7 +42,7 @@ function DynamicButton({ options, selectedDynamicValue, onSelect, enabled }) {
return ( return (
<div ref={containerRef}> <div ref={containerRef}>
<a onClick={e => e.stopPropagation()}> <div role="presentation" 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 }) {
getPopupContainer={() => containerRef.current} getPopupContainer={() => containerRef.current}
data-test="DynamicButton" data-test="DynamicButton"
/> />
</a> </div>
</div> </div>
); );
} }
@@ -68,6 +68,7 @@ DynamicButton.propTypes = {
selectedDynamicValue: PropTypes.oneOfType([DynamicDateType, DynamicDateRangeType]), selectedDynamicValue: PropTypes.oneOfType([DynamicDateType, DynamicDateRangeType]),
onSelect: PropTypes.func, onSelect: PropTypes.func,
enabled: PropTypes.bool, enabled: PropTypes.bool,
staticValueLabel: PropTypes.string,
}; };
DynamicButton.defaultProps = { DynamicButton.defaultProps = {
@@ -75,6 +76,7 @@ DynamicButton.defaultProps = {
selectedDynamicValue: null, selectedDynamicValue: null,
onSelect: () => {}, onSelect: () => {},
enabled: false, enabled: false,
staticValueLabel: "Back to Static Value",
}; };
export default DynamicButton; export default DynamicButton;

View File

@@ -34,3 +34,9 @@
font-size: 11px; font-size: 11px;
} }
} }
.dynamic-icon {
display: flex !important;
align-items: center;
justify-content: center;
}

View File

@@ -0,0 +1,112 @@
import React from "react";
import PropTypes from "prop-types";
import classNames from "classnames";
import moment from "moment";
import { includes } from "lodash";
import { isDynamicDate } from "@/services/parameters/DateParameter";
import DateInput from "@/components/DateInput";
import DateTimeInput from "@/components/DateTimeInput";
import DynamicButton from "@/components/dynamic-parameters/DynamicButton";
import "./DynamicParameters.less";
class DynamicDatePicker extends React.Component {
static propTypes = {
type: PropTypes.string,
className: PropTypes.string,
value: PropTypes.any, // eslint-disable-line react/forbid-prop-types
parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types
onSelect: PropTypes.func,
dynamicButtonOptions: PropTypes.shape({
staticValueLabel: PropTypes.string,
options: PropTypes.arrayOf(
PropTypes.shape({
name: PropTypes.string,
value: PropTypes.object,
label: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),
})
),
}),
dateOptions: PropTypes.any, // eslint-disable-line react/forbid-prop-types
};
static defaultProps = {
type: "",
className: "",
value: null,
parameter: null,
dynamicButtonOptions: {
options: [],
},
onSelect: () => {},
};
constructor(props) {
super(props);
this.dateComponentRef = React.createRef();
}
onDynamicValueSelect = dynamicValue => {
const { onSelect, parameter } = this.props;
if (dynamicValue === "static") {
const parameterValue = parameter.getExecutionValue();
if (parameterValue) {
onSelect(moment(parameterValue));
} else {
onSelect(null);
}
} else {
onSelect(dynamicValue.value);
}
// give focus to the DatePicker to get keyboard shortcuts to work
this.dateComponentRef.current.focus();
};
render() {
const { type, value, className, dateOptions, dynamicButtonOptions, onSelect } = this.props;
const hasDynamicValue = isDynamicDate(value);
const isDateTime = includes(type, "datetime");
const additionalAttributes = {};
let DateComponent = DateInput;
if (isDateTime) {
DateComponent = DateTimeInput;
if (includes(type, "with-seconds")) {
additionalAttributes.withSeconds = true;
}
}
if (moment.isMoment(value) || value === null) {
additionalAttributes.value = value;
}
if (hasDynamicValue) {
const dynamicDate = value;
additionalAttributes.placeholder = dynamicDate && dynamicDate.name;
additionalAttributes.value = null;
}
return (
<div className={classNames("date-parameter", className)}>
<DateComponent
{...dateOptions}
ref={this.dateComponentRef}
className={classNames("redash-datepicker", type, { "dynamic-value": hasDynamicValue })}
onSelect={onSelect}
suffixIcon={null}
{...additionalAttributes}
/>
<DynamicButton
options={dynamicButtonOptions.options}
staticValueLabel={dynamicButtonOptions.staticValueLabel}
selectedDynamicValue={hasDynamicValue ? value : null}
enabled={hasDynamicValue}
onSelect={this.onDynamicValueSelect}
/>
</div>
);
}
}
export default DynamicDatePicker;

View File

@@ -0,0 +1,115 @@
import React from "react";
import PropTypes from "prop-types";
import classNames from "classnames";
import moment from "moment";
import { includes, isArray, isObject } from "lodash";
import { isDynamicDateRange } from "@/services/parameters/DateRangeParameter";
import DateRangeInput from "@/components/DateRangeInput";
import DateTimeRangeInput from "@/components/DateTimeRangeInput";
import DynamicButton from "@/components/dynamic-parameters/DynamicButton";
import "./DynamicParameters.less";
function isValidDateRangeValue(value) {
return isArray(value) && value.length === 2 && moment.isMoment(value[0]) && moment.isMoment(value[1]);
}
class DynamicDateRangePicker extends React.Component {
static propTypes = {
type: PropTypes.oneOf(["date-range", "datetime-range", "datetime-range-with-seconds"]).isRequired,
className: PropTypes.string,
value: PropTypes.any, // eslint-disable-line react/forbid-prop-types
parameter: PropTypes.any, // eslint-disable-line react/forbid-prop-types
onSelect: PropTypes.func,
dynamicButtonOptions: PropTypes.shape({
staticValueLabel: PropTypes.string,
options: PropTypes.arrayOf(
PropTypes.shape({
name: PropTypes.string,
value: PropTypes.object,
label: PropTypes.oneOfType([PropTypes.string, PropTypes.func]),
})
),
}),
dateRangeOptions: PropTypes.any, // eslint-disable-line react/forbid-prop-types
};
static defaultProps = {
type: "date-range",
className: "",
value: null,
parameter: null,
dynamicButtonOptions: {
options: [],
},
onSelect: () => {},
};
constructor(props) {
super(props);
this.dateRangeComponentRef = React.createRef();
}
onDynamicValueSelect = dynamicValue => {
const { onSelect, parameter } = this.props;
if (dynamicValue === "static") {
const parameterValue = parameter.getExecutionValue();
if (isObject(parameterValue) && parameterValue.start && parameterValue.end) {
onSelect([moment(parameterValue.start), moment(parameterValue.end)]);
} else {
onSelect(null);
}
} else {
onSelect(dynamicValue.value);
}
// give focus to the DatePicker to get keyboard shortcuts to work
this.dateRangeComponentRef.current.focus();
};
render() {
const { type, value, onSelect, className, dynamicButtonOptions, dateRangeOptions, parameter, ...rest } = this.props;
const isDateTimeRange = includes(type, "datetime-range");
const hasDynamicValue = isDynamicDateRange(value);
const additionalAttributes = {};
let DateRangeComponent = DateRangeInput;
if (isDateTimeRange) {
DateRangeComponent = DateTimeRangeInput;
if (includes(type, "with-seconds")) {
additionalAttributes.withSeconds = true;
}
}
if (isValidDateRangeValue(value) || value === null) {
additionalAttributes.value = value;
}
if (hasDynamicValue) {
additionalAttributes.placeholder = [value && value.name];
additionalAttributes.value = null;
}
return (
<div {...rest} className={classNames("date-range-parameter", className)}>
<DateRangeComponent
{...dateRangeOptions}
ref={this.dateRangeComponentRef}
className={classNames("redash-datepicker date-range-input", type, { "dynamic-value": hasDynamicValue })}
onSelect={onSelect}
suffixIcon={null}
{...additionalAttributes}
/>
<DynamicButton
options={dynamicButtonOptions.options}
staticValueLabel={dynamicButtonOptions.staticValueLabel}
selectedDynamicValue={hasDynamicValue ? value : null}
enabled={hasDynamicValue}
onSelect={this.onDynamicValueSelect}
/>
</div>
);
}
}
export default DynamicDateRangePicker;

View File

@@ -1,8 +1,26 @@
@import "../../assets/less/inc/variables"; @import (reference, less) "~@/assets/less/inc/variables";
.date-range-parameter,
.date-parameter {
position: relative;
}
.redash-datepicker { .redash-datepicker {
padding-right: 35px !important; padding-right: 35px !important;
&.date-range {
width: 294px;
}
&.datetime-range {
width: 352px;
}
&.datetime-range-with-seconds {
width: 382px;
}
&.dynamic-value {
width: 195px;
}
&.ant-picker-range .ant-picker-clear { &.ant-picker-range .ant-picker-clear {
right: 35px !important; right: 35px !important;
background: transparent; background: transparent;
@@ -14,7 +32,7 @@
&.dynamic-value { &.dynamic-value {
& ::placeholder { & ::placeholder {
color: @text-color !important; color: @input-color !important;
} }
&.date-range-input { &.date-range-input {
@@ -22,7 +40,8 @@
opacity: 0; opacity: 0;
} }
.ant-picker-separator { .ant-picker-separator,
.ant-picker-range-separator {
display: none; display: none;
} }

View File

@@ -8,13 +8,21 @@ export interface StepItem<K> {
node: React.ReactNode; node: React.ReactNode;
} }
export interface EmptyStateHelpMessageProps {
helpTriggerType: string;
}
export declare const EmptyStateHelpMessage: React.FunctionComponent<EmptyStateHelpMessageProps>;
export interface EmptyStateProps<K = unknown> { export interface EmptyStateProps<K = unknown> {
header?: string; header?: string;
icon?: string; icon?: string;
description: string; description: string;
illustration: string; illustration: string;
illustrationPath?: string; illustrationPath?: string;
helpLink: string; helpMessage?: React.ReactNode;
closable?: boolean;
onClose?: () => void;
onboardingMode?: boolean; onboardingMode?: boolean;
showAlertStep?: boolean; showAlertStep?: boolean;
@@ -33,8 +41,9 @@ export interface StepProps {
show: boolean; show: boolean;
completed: boolean; completed: boolean;
url?: string; url?: string;
urlText?: string; urlTarget?: string;
text: string; urlText?: React.ReactNode;
text?: React.ReactNode;
onClick?: () => void; onClick?: () => void;
} }

View File

@@ -2,10 +2,14 @@ import { keys, some } from "lodash";
import React, { useCallback } from "react"; import React, { useCallback } from "react";
import PropTypes from "prop-types"; import PropTypes from "prop-types";
import classNames from "classnames"; import classNames from "classnames";
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 { 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, urlText, onClick }) {
@@ -13,12 +17,11 @@ export function Step({ show, completed, text, url, urlText, onClick }) {
return null; return null;
} }
const commonProps = { children: urlText, onClick };
return ( return (
<li className={classNames({ done: completed })}> <li className={classNames({ done: completed })}>
<Link href={url} onClick={onClick}> {url ? <Link href={url} {...commonProps} /> : <PlainButton type="link" {...commonProps} />} {text}
{urlText}
</Link>{" "}
{text}
</li> </li>
); );
} }
@@ -26,24 +29,44 @@ export function Step({ show, completed, text, url, urlText, onClick }) {
Step.propTypes = { Step.propTypes = {
show: PropTypes.bool.isRequired, show: PropTypes.bool.isRequired,
completed: PropTypes.bool.isRequired, completed: PropTypes.bool.isRequired,
text: PropTypes.string.isRequired, text: PropTypes.node,
url: PropTypes.string, url: PropTypes.string,
urlText: PropTypes.string, urlTarget: PropTypes.string,
urlText: PropTypes.node,
onClick: PropTypes.func, onClick: PropTypes.func,
}; };
Step.defaultProps = { Step.defaultProps = {
url: null, url: null,
urlTarget: null,
urlText: null, urlText: null,
text: null,
onClick: null, onClick: null,
}; };
export function EmptyStateHelpMessage({ helpTriggerType }) {
return (
<p>
Need more support?{" "}
<HelpTrigger className="f-14" type={helpTriggerType} showTooltip={false}>
See our Help
</HelpTrigger>
</p>
);
}
EmptyStateHelpMessage.propTypes = {
helpTriggerType: PropTypes.string.isRequired,
};
function EmptyState({ function EmptyState({
icon, icon,
header, header,
description, description,
illustration, illustration,
helpLink, helpMessage,
closable,
onClose,
onboardingMode, onboardingMode,
showAlertStep, showAlertStep,
showDashboardStep, showDashboardStep,
@@ -87,8 +110,7 @@ function EmptyState({
show={isAvailable.dataSource} show={isAvailable.dataSource}
completed={isCompleted.dataSource} completed={isCompleted.dataSource}
url="data_sources/new" url="data_sources/new"
urlText="Connect" urlText="Connect a Data Source"
text="a Data Source"
/> />
); );
} }
@@ -116,8 +138,7 @@ function EmptyState({
show={isAvailable.query} show={isAvailable.query}
completed={isCompleted.query} completed={isCompleted.query}
url="queries/new" url="queries/new"
urlText="Create" urlText="Create your first Query"
text="your first Query"
/> />
), ),
}, },
@@ -129,8 +150,7 @@ function EmptyState({
show={isAvailable.alert} show={isAvailable.alert}
completed={isCompleted.alert} completed={isCompleted.alert}
url="alerts/new" url="alerts/new"
urlText="Create" urlText="Create your first Alert"
text="your first Alert"
/> />
), ),
}, },
@@ -142,8 +162,7 @@ function EmptyState({
show={isAvailable.dashboard} show={isAvailable.dashboard}
completed={isCompleted.dashboard} completed={isCompleted.dashboard}
onClick={showCreateDashboardDialog} onClick={showCreateDashboardDialog}
urlText="Create" urlText="Create your first Dashboard"
text="your first Dashboard"
/> />
), ),
}, },
@@ -155,8 +174,7 @@ function EmptyState({
show={isAvailable.inviteUsers} show={isAvailable.inviteUsers}
completed={isCompleted.inviteUsers} completed={isCompleted.inviteUsers}
url="users/new" url="users/new"
urlText="Invite" urlText="Invite your team members"
text="your team members"
/> />
), ),
}, },
@@ -166,26 +184,27 @@ function EmptyState({
const imageSource = illustrationPath ? illustrationPath : "static/images/illustrations/" + illustration + ".svg"; const imageSource = illustrationPath ? illustrationPath : "static/images/illustrations/" + illustration + ".svg";
return ( return (
<div className="empty-state bg-white tiled"> <div className="empty-state-wrapper">
<div className="empty-state__summary"> <div className="empty-state bg-white tiled">
{header && <h4>{header}</h4>} <div className="empty-state__summary">
<h2> {header && <h4>{header}</h4>}
<i className={icon} /> <h2>
</h2> <i className={icon} aria-hidden="true" />
<p>{description}</p> </h2>
<img src={imageSource} alt={illustration + " Illustration"} width="75%" /> <p>{description}</p>
</div> <img src={imageSource} alt={illustration + " Illustration"} width="75%" />
<div className="empty-state__steps"> </div>
<h4>Let&apos;s get started</h4> <div className="empty-state__steps">
<ol>{stepsItems.map(item => item.node)}</ol> <h4>Let&apos;s get started</h4>
<p> <ol>{stepsItems.map(item => item.node)}</ol>
Need more support?{" "} {helpMessage}
<Link href={helpLink} target="_blank" rel="noopener noreferrer"> </div>
See our Help
<i className="fa fa-external-link m-l-5" aria-hidden="true" />
</Link>
</p>
</div> </div>
{closable && (
<PlainButton className="close-button" aria-label="Close" onClick={onClose}>
<CloseOutlinedIcon />
</PlainButton>
)}
</div> </div>
); );
} }
@@ -196,7 +215,9 @@ EmptyState.propTypes = {
description: PropTypes.string.isRequired, description: PropTypes.string.isRequired,
illustration: PropTypes.string.isRequired, illustration: PropTypes.string.isRequired,
illustrationPath: PropTypes.string, illustrationPath: PropTypes.string,
helpLink: PropTypes.string.isRequired, helpMessage: PropTypes.node,
closable: PropTypes.bool,
onClose: PropTypes.func,
onboardingMode: PropTypes.bool, onboardingMode: PropTypes.bool,
showAlertStep: PropTypes.bool, showAlertStep: PropTypes.bool,
@@ -210,6 +231,9 @@ EmptyState.propTypes = {
EmptyState.defaultProps = { EmptyState.defaultProps = {
icon: null, icon: null,
header: null, header: null,
helpMessage: null,
closable: false,
onClose: () => {},
onboardingMode: false, onboardingMode: false,
showAlertStep: false, showAlertStep: false,

View File

@@ -1,7 +1,9 @@
@import (reference, less) "~@/assets/less/ant";
// Empty states // Empty states
.empty-state { .empty-state {
width: 100%; width: 100%;
margin: 0px auto 10px; margin: 0 auto 10px;
display: flex; display: flex;
flex-direction: row; flex-direction: row;
justify-content: space-between; justify-content: space-between;
@@ -16,14 +18,17 @@
} }
.empty-state__steps { .empty-state__steps {
padding-left: 0px; padding-left: 0;
} }
.empty-state__summary { .empty-state__summary {
align-self: flex-start; align-self: flex-start;
text-align: center; text-align: center;
background: rgba(102, 136, 153, 0.025); background: rgba(102, 136, 153, 0.025);
p {
margin-bottom: 0;
}
} }
ol { ol {
@@ -44,10 +49,6 @@
margin-bottom: 15px; margin-bottom: 15px;
} }
p {
margin-bottom: 0;
}
a:hover { a:hover {
cursor: pointer; cursor: pointer;
} }
@@ -71,3 +72,27 @@
} }
} }
} }
// close button
.empty-state-wrapper {
position: relative;
.close-button {
position: absolute;
top: 15px;
right: 25px;
font-size: 15px;
color: @text-color-secondary;
cursor: pointer;
transition: color @animation-duration-slow;
&:hover,
&:focus {
color: @text-color;
}
&:active {
filter: contrast(200%);
}
}
}

View File

@@ -28,6 +28,7 @@ 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 "antd/lib/tooltip"; import Tooltip from "@/components/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" /> <i className="fa fa-plus m-r-5" aria-hidden="true" />
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" /> <i className="fa fa-plus m-r-5" aria-hidden="true" />
Add Data Sources Add Data Sources
</Button> </Button>
)} )}

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