From 0e8091881a5cb3a8cba2d29993c71db3fc799442 Mon Sep 17 00:00:00 2001 From: Ville Brofeldt Date: Mon, 22 Nov 2021 09:50:18 +0200 Subject: [PATCH 1/4] chore(translations): fix translation literals --- .../src/SqlLab/components/App/index.jsx | 14 ++++---- .../src/SqlLab/components/ResultSet/index.tsx | 26 +++++++-------- .../components/SqlEditorLeftBar/index.jsx | 4 ++- .../src/dashboard/actions/dashboardState.js | 9 ++++-- .../nativeFilters/FilterBar/index.tsx | 8 +++-- .../FiltersConfigForm/getControlItemsMap.tsx | 2 +- .../src/views/CRUD/alert/AlertList.tsx | 24 +++++++++++--- .../src/views/CRUD/alert/AlertReportModal.tsx | 10 ++++-- .../src/views/CRUD/alert/ExecutionLog.tsx | 2 +- .../CRUD/alert/components/AlertStatusIcon.tsx | 32 +++++++++---------- .../CRUD/alert/components/RecipientIcon.tsx | 10 +++--- .../src/views/CRUD/alert/types.ts | 14 ++++---- superset-frontend/src/views/CRUD/hooks.ts | 2 +- .../src/views/CRUD/welcome/EmptyState.tsx | 14 ++++---- superset/databases/commands/exceptions.py | 2 +- 15 files changed, 103 insertions(+), 70 deletions(-) diff --git a/superset-frontend/src/SqlLab/components/App/index.jsx b/superset-frontend/src/SqlLab/components/App/index.jsx index 406ad740bd5c4..bce43f477800e 100644 --- a/superset-frontend/src/SqlLab/components/App/index.jsx +++ b/superset-frontend/src/SqlLab/components/App/index.jsx @@ -83,12 +83,14 @@ class App extends React.PureComponent { this.props.actions.addDangerToast( t( "SQL Lab uses your browser's local storage to store queries and results." + - `\n Currently, you are using ${currentUsage.toFixed( - 2, - )} KB out of ${LOCALSTORAGE_MAX_USAGE_KB} KB. storage space.` + - '\n To keep SQL Lab from crashing, please delete some query tabs.' + - '\n You can re-access these queries by using the Save feature before you delete the tab. ' + - 'Note that you will need to close other SQL Lab windows before you do this.', + '\nCurrently, you are using %(currentUsage)s KB out of %(maxStorage)d KB storage space.' + + '\nTo keep SQL Lab from crashing, please delete some query tabs.' + + '\nYou can re-access these queries by using the Save feature before you delete the tab.' + + '\nNote that you will need to close other SQL Lab windows before you do this.', + { + currentUsage: currentUsage.toFixed(2), + maxStorage: LOCALSTORAGE_MAX_USAGE_KB, + }, ), ); } diff --git a/superset-frontend/src/SqlLab/components/ResultSet/index.tsx b/superset-frontend/src/SqlLab/components/ResultSet/index.tsx index 987d87ea30071..2d15efc09be25 100644 --- a/superset-frontend/src/SqlLab/components/ResultSet/index.tsx +++ b/superset-frontend/src/SqlLab/components/ResultSet/index.tsx @@ -581,22 +581,22 @@ export default class ResultSet extends React.PureComponent< const isAdmin = !!this.props.user?.roles?.Admin; const displayMaxRowsReachedMessage = { withAdmin: t( - `The number of results displayed is limited to %(rows)d by the configuration DISPLAY_MAX_ROWS. `, + 'The number of results displayed is limited to %(rows)d by the configuration DISPLAY_MAX_ROWS. ', { rows }, ).concat( t( - `Please add additional limits/filters or download to csv to see more rows up to `, + 'Please add additional limits/filters or download to csv to see more rows up to ', ), - t(`the %(limit)d limit.`, { limit }), + t('the %(limit)d limit.', { limit }), ), withoutAdmin: t( - `The number of results displayed is limited to %(rows)d. `, + 'The number of results displayed is limited to %(rows)d. ', { rows }, ).concat( t( - `Please add additional limits/filters, download to csv, or contact an admin `, + 'Please add additional limits/filters, download to csv, or contact an admin ', ), - t(`to see more rows up to the %(limit)d limit.`, { + t('to see more rows up to the %(limit)d limit.', { limit, }), ), @@ -609,7 +609,7 @@ export default class ResultSet extends React.PureComponent< limitMessage = ( {t( - `The number of rows displayed is limited to %(rows)d by the query`, + 'The number of rows displayed is limited to %(rows)d by the query', { rows }, )} @@ -621,7 +621,7 @@ export default class ResultSet extends React.PureComponent< limitMessage = ( {t( - `The number of rows displayed is limited to %(rows)d by the limit dropdown.`, + 'The number of rows displayed is limited to %(rows)d by the limit dropdown.', { rows }, )} @@ -630,7 +630,7 @@ export default class ResultSet extends React.PureComponent< limitMessage = ( {t( - `The number of rows displayed is limited to %(rows)d by the query and limit dropdown.`, + 'The number of rows displayed is limited to %(rows)d by the query and limit dropdown.', { rows }, )} @@ -640,17 +640,17 @@ export default class ResultSet extends React.PureComponent< {!limitReached && !shouldUseDefaultDropdownAlert && ( - {t(`%(rows)d rows returned`, { rows })} {limitMessage} + {t('%(rows)d rows returned', { rows })} {limitMessage} )} {!limitReached && shouldUseDefaultDropdownAlert && (
@@ -661,7 +661,7 @@ export default class ResultSet extends React.PureComponent< { - const nowPublished = isPublished ? 'published' : 'hidden'; - dispatch(addSuccessToast(t(`This dashboard is now ${nowPublished}`))); + dispatch( + addSuccessToast( + isPublished + ? t('This dashboard is now published') + : t('This dashboard is now hidden'), + ), + ); dispatch(togglePublished(isPublished)); }) .catch(() => { diff --git a/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/index.tsx b/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/index.tsx index 703f47ec7aff8..9a8a9fde605b3 100644 --- a/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/index.tsx +++ b/superset-frontend/src/dashboard/components/nativeFilters/FilterBar/index.tsx @@ -320,7 +320,9 @@ const FilterBar: React.FC = ({ activeKey={editFilterSetId ? TabIds.AllFilters : undefined} > @@ -340,7 +342,9 @@ const FilterBar: React.FC = ({ diff --git a/superset-frontend/src/dashboard/components/nativeFilters/FiltersConfigModal/FiltersConfigForm/getControlItemsMap.tsx b/superset-frontend/src/dashboard/components/nativeFilters/FiltersConfigModal/FiltersConfigForm/getControlItemsMap.tsx index 65ac4fbca2592..ce8978abaa1ca 100644 --- a/superset-frontend/src/dashboard/components/nativeFilters/FiltersConfigModal/FiltersConfigForm/getControlItemsMap.tsx +++ b/superset-frontend/src/dashboard/components/nativeFilters/FiltersConfigModal/FiltersConfigForm/getControlItemsMap.tsx @@ -106,7 +106,7 @@ export default function getControlItemsMap({ initialValue={initColumn} label={ - {t(`${mainControlItem.config?.label}`) || t('Column')} + {mainControlItem.config?.label || t('Column')} } rules={[ diff --git a/superset-frontend/src/views/CRUD/alert/AlertList.tsx b/superset-frontend/src/views/CRUD/alert/AlertList.tsx index c9d8d63461934..7068787650657 100644 --- a/superset-frontend/src/views/CRUD/alert/AlertList.tsx +++ b/superset-frontend/src/views/CRUD/alert/AlertList.tsx @@ -50,6 +50,14 @@ import { AlertObject, AlertState } from './types'; const PAGE_SIZE = 25; +const AlertStateLabel: Record = { + [AlertState.Success]: t('Success'), + [AlertState.Working]: t('Working'), + [AlertState.Error]: t('Error'), + [AlertState.Noop]: t('Not triggered'), + [AlertState.Grace]: t('On Grace'), +}; + interface AlertListProps { addDangerToast: (msg: string) => void; addSuccessToast: (msg: string) => void; @@ -394,11 +402,17 @@ function AlertList({ operator: FilterOperator.equals, unfilteredLabel: 'Any', selects: [ - { label: t(`${AlertState.success}`), value: AlertState.success }, - { label: t(`${AlertState.working}`), value: AlertState.working }, - { label: t(`${AlertState.error}`), value: AlertState.error }, - { label: t(`${AlertState.noop}`), value: AlertState.noop }, - { label: t(`${AlertState.grace}`), value: AlertState.grace }, + { + label: AlertStateLabel[AlertState.Success], + value: AlertState.Success, + }, + { + label: AlertStateLabel[AlertState.Working], + value: AlertState.Working, + }, + { label: AlertStateLabel[AlertState.Error], value: AlertState.Error }, + { label: AlertStateLabel[AlertState.Noop], value: AlertState.Noop }, + { label: AlertStateLabel[AlertState.Grace], value: AlertState.Grace }, ], }, { diff --git a/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx b/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx index 775423ad76d57..201780289605f 100644 --- a/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx +++ b/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx @@ -1047,9 +1047,13 @@ const AlertReportModal: FunctionComponent = ({ ) : ( )} - {isEditMode - ? t(`Edit ${isReport ? 'Report' : 'Alert'}`) - : t(`Add ${isReport ? 'Report' : 'Alert'}`)} + {isEditMode && isReport + ? t('Edit Report') + : isEditMode + ? t('Edit Alert') + : isReport + ? 'Add Report' + : 'Add Alert'} } > diff --git a/superset-frontend/src/views/CRUD/alert/ExecutionLog.tsx b/superset-frontend/src/views/CRUD/alert/ExecutionLog.tsx index e9c7baff21d55..64e665aeb4c13 100644 --- a/superset-frontend/src/views/CRUD/alert/ExecutionLog.tsx +++ b/superset-frontend/src/views/CRUD/alert/ExecutionLog.tsx @@ -153,7 +153,7 @@ function ExecutionLog({ addDangerToast, isReportEnabled }: ExecutionLogProps) { name={ - {t(`${alertResource?.type}`)} {alertResource?.name} + {alertResource?.type} {alertResource?.name} Back to all diff --git a/superset-frontend/src/views/CRUD/alert/components/AlertStatusIcon.tsx b/superset-frontend/src/views/CRUD/alert/components/AlertStatusIcon.tsx index 586afc68e91ed..048a12f35e5e5 100644 --- a/superset-frontend/src/views/CRUD/alert/components/AlertStatusIcon.tsx +++ b/superset-frontend/src/views/CRUD/alert/components/AlertStatusIcon.tsx @@ -28,17 +28,17 @@ function getStatusColor( theme: typeof supersetTheme, ) { switch (status) { - case AlertState.working: + case AlertState.Working: return theme.colors.primary.base; - case AlertState.error: + case AlertState.Error: return theme.colors.error.base; - case AlertState.success: + case AlertState.Success: return isReportEnabled ? theme.colors.success.base : theme.colors.alert.base; - case AlertState.noop: + case AlertState.Noop: return theme.colors.success.base; - case AlertState.grace: + case AlertState.Grace: return theme.colors.alert.base; default: return theme.colors.grayscale.base; @@ -59,43 +59,43 @@ export default function AlertStatusIcon({ status: '', }; switch (state) { - case AlertState.success: + case AlertState.Success: lastStateConfig.icon = isReportEnabled ? Icons.Check : Icons.AlertSolidSmall; lastStateConfig.label = isReportEnabled ? t('Report sent') : t('Alert triggered, notification sent'); - lastStateConfig.status = AlertState.success; + lastStateConfig.status = AlertState.Success; break; - case AlertState.working: + case AlertState.Working: lastStateConfig.icon = Icons.Running; lastStateConfig.label = isReportEnabled ? t('Report sending') : t('Alert running'); - lastStateConfig.status = AlertState.working; + lastStateConfig.status = AlertState.Working; break; - case AlertState.error: + case AlertState.Error: lastStateConfig.icon = Icons.XSmall; lastStateConfig.label = isReportEnabled ? t('Report failed') : t('Alert failed'); - lastStateConfig.status = AlertState.error; + lastStateConfig.status = AlertState.Error; break; - case AlertState.noop: + case AlertState.Noop: lastStateConfig.icon = Icons.Check; lastStateConfig.label = t('Nothing triggered'); - lastStateConfig.status = AlertState.noop; + lastStateConfig.status = AlertState.Noop; break; - case AlertState.grace: + case AlertState.Grace: lastStateConfig.icon = Icons.AlertSolidSmall; lastStateConfig.label = t('Alert Triggered, In Grace Period'); - lastStateConfig.status = AlertState.grace; + lastStateConfig.status = AlertState.Grace; break; default: lastStateConfig.icon = Icons.Check; lastStateConfig.label = t('Nothing triggered'); - lastStateConfig.status = AlertState.noop; + lastStateConfig.status = AlertState.Noop; } const Icon = lastStateConfig.icon; return ( diff --git a/superset-frontend/src/views/CRUD/alert/components/RecipientIcon.tsx b/superset-frontend/src/views/CRUD/alert/components/RecipientIcon.tsx index b2e8652c537e9..96ca65ccb10cd 100644 --- a/superset-frontend/src/views/CRUD/alert/components/RecipientIcon.tsx +++ b/superset-frontend/src/views/CRUD/alert/components/RecipientIcon.tsx @@ -16,7 +16,7 @@ * specific language governing permissions and limitations * under the License. */ -import { t, SupersetTheme, css } from '@superset-ui/core'; +import { SupersetTheme, css } from '@superset-ui/core'; import React, { ReactElement } from 'react'; import { Tooltip } from 'src/components/Tooltip'; import Icons from 'src/components/Icons'; @@ -33,13 +33,13 @@ export default function RecipientIcon({ type }: { type: string }) { label: '', }; switch (type) { - case RecipientIconName.email: + case RecipientIconName.Email: recipientIconConfig.icon = ; - recipientIconConfig.label = t(`${RecipientIconName.email}`); + recipientIconConfig.label = RecipientIconName.Email; break; - case RecipientIconName.slack: + case RecipientIconName.Slack: recipientIconConfig.icon = ; - recipientIconConfig.label = t(`${RecipientIconName.slack}`); + recipientIconConfig.label = RecipientIconName.Slack; break; default: recipientIconConfig.icon = null; diff --git a/superset-frontend/src/views/CRUD/alert/types.ts b/superset-frontend/src/views/CRUD/alert/types.ts index 3f939a03bcfa1..1c40cdb1f9efa 100644 --- a/superset-frontend/src/views/CRUD/alert/types.ts +++ b/superset-frontend/src/views/CRUD/alert/types.ts @@ -100,14 +100,14 @@ export type LogObject = { }; export enum AlertState { - success = 'Success', - working = 'Working', - error = 'Error', - noop = 'Not triggered', - grace = 'On Grace', + Success = 'Success', + Working = 'Working', + Error = 'Error', + Noop = 'Not triggered', + Grace = 'On Grace', } export enum RecipientIconName { - email = 'Email', - slack = 'Slack', + Email = 'Email', + Slack = 'Slack', } diff --git a/superset-frontend/src/views/CRUD/hooks.ts b/superset-frontend/src/views/CRUD/hooks.ts index 057e4799aea40..f9b7a1ed98ccb 100644 --- a/superset-frontend/src/views/CRUD/hooks.ts +++ b/superset-frontend/src/views/CRUD/hooks.ts @@ -623,7 +623,7 @@ export const testDatabaseConnection = ( addSuccessToast(t('Connection looks good!')); }, createErrorHandler((errMsg: Record | string) => { - handleErrorMsg(t(`${t('ERROR: ')}${parsedErrorMessage(errMsg)}`)); + handleErrorMsg(t('ERROR: %s', parsedErrorMessage(errMsg))); }), ); }; diff --git a/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx b/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx index b55ac8a4a1f17..c89d6bc568670 100644 --- a/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx +++ b/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx @@ -57,11 +57,11 @@ export default function EmptyState({ tableName, tab }: EmptyStateProps) { SAVED_QUERIES: 'empty-queries.svg', }; const mine = ( - {`No ${ - tableName === 'SAVED_QUERIES' - ? t('saved queries') - : t(`${tableName.toLowerCase()}`) - } yet`} + + {tableName === 'SAVED_QUERIES' + ? t('No saved queries yet') + : t('No %(tableName)s yet', { tableName: tableName.toLowerCase() })} + ); const recent = ( @@ -77,7 +77,9 @@ export default function EmptyState({ tableName, tab }: EmptyStateProps) { ); } if (tab === 'Examples') { - return t(`Example ${tableName.toLowerCase()} will appear here`); + return t('Example %(tableName)s will appear here', { + tableName: tableName.toLowerCase(), + }); } if (tab === 'Edited') { return t( diff --git a/superset/databases/commands/exceptions.py b/superset/databases/commands/exceptions.py index 9cd8b3306f62e..9ba58373e197a 100644 --- a/superset/databases/commands/exceptions.py +++ b/superset/databases/commands/exceptions.py @@ -60,7 +60,7 @@ def __init__(self, json_error: str = "") -> None: super().__init__( [ _( - "Field cannot be decoded by JSON. %{json_error}s", + "Field cannot be decoded by JSON. %(json_error)s", json_error=json_error, ) ], From 031e14e9bea8d720d25279e1693e6853e79a322a Mon Sep 17 00:00:00 2001 From: Ville Brofeldt Date: Mon, 22 Nov 2021 14:26:24 +0200 Subject: [PATCH 2/4] fix table name types and translations --- .../src/views/CRUD/alert/AlertReportModal.tsx | 4 +- .../src/views/CRUD/welcome/EmptyState.tsx | 49 ++++++++++++------- .../src/views/CRUD/welcome/types.ts | 25 ++++++++++ 3 files changed, 57 insertions(+), 21 deletions(-) create mode 100644 superset-frontend/src/views/CRUD/welcome/types.ts diff --git a/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx b/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx index 201780289605f..99deaa8cc1fdb 100644 --- a/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx +++ b/superset-frontend/src/views/CRUD/alert/AlertReportModal.tsx @@ -1052,8 +1052,8 @@ const AlertReportModal: FunctionComponent = ({ : isEditMode ? t('Edit Alert') : isReport - ? 'Add Report' - : 'Add Alert'} + ? t('Add Report') + : t('Add Alert')} } > diff --git a/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx b/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx index c89d6bc568670..15546ca9495ef 100644 --- a/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx +++ b/superset-frontend/src/views/CRUD/welcome/EmptyState.tsx @@ -20,9 +20,17 @@ import React from 'react'; import Button from 'src/components/Button'; import { Empty } from 'src/common/components'; import { t, styled } from '@superset-ui/core'; +import { WelcomeTable } from './types'; + +const welcomeTableLabels: Record = { + [WelcomeTable.Charts]: t('charts'), + [WelcomeTable.Dashboards]: t('dashboards'), + [WelcomeTable.Recents]: t('recents'), + [WelcomeTable.SavedQueries]: t('saved queries'), +}; interface EmptyStateProps { - tableName: string; + tableName: WelcomeTable; tab?: string; } const EmptyContainer = styled.div` @@ -39,28 +47,31 @@ const ButtonContainer = styled.div` } `; +type Redirects = Record< + WelcomeTable.Charts | WelcomeTable.Dashboards | WelcomeTable.SavedQueries, + string +>; + export default function EmptyState({ tableName, tab }: EmptyStateProps) { - const mineRedirects = { - DASHBOARDS: '/dashboard/new', - CHARTS: '/chart/add', - SAVED_QUERIES: '/superset/sqllab?new=true', + const mineRedirects: Redirects = { + [WelcomeTable.Charts]: '/chart/add', + [WelcomeTable.Dashboards]: '/dashboard/new', + [WelcomeTable.SavedQueries]: '/superset/sqllab?new=true', }; - const favRedirects = { - DASHBOARDS: '/dashboard/list/', - CHARTS: '/chart/list', - SAVED_QUERIES: '/savedqueryview/list/', + const favRedirects: Redirects = { + [WelcomeTable.Charts]: '/chart/list', + [WelcomeTable.Dashboards]: '/dashboard/list/', + [WelcomeTable.SavedQueries]: '/savedqueryview/list/', }; - const tableIcon = { - RECENTS: 'union.svg', - DASHBOARDS: 'empty-dashboard.svg', - CHARTS: 'empty-charts.svg', - SAVED_QUERIES: 'empty-queries.svg', + const tableIcon: Record = { + [WelcomeTable.Charts]: 'empty-charts.svg', + [WelcomeTable.Dashboards]: 'empty-dashboard.svg', + [WelcomeTable.Recents]: 'union.svg', + [WelcomeTable.SavedQueries]: 'empty-queries.svg', }; const mine = ( - {tableName === 'SAVED_QUERIES' - ? t('No saved queries yet') - : t('No %(tableName)s yet', { tableName: tableName.toLowerCase() })} + {t('No %(tableName)s yet', { tableName: welcomeTableLabels[tableName] })} ); const recent = ( @@ -105,7 +116,7 @@ export default function EmptyState({ tableName, tab }: EmptyStateProps) {