From a85266284b5f935b589b32a2771e0a7db298ce26 Mon Sep 17 00:00:00 2001 From: simeng-li Date: Fri, 5 Jan 2024 10:46:27 +0800 Subject: [PATCH] feat(console,phrases): add new third-party application to the guide Library (#5144) * feat(console,phrases): add new third-party applicaiton to the guide library add new third-party applicaiton to the guide library * fix(test): adjust the test order adjust the test order * fix(phrases): add untranslated tag add untranslated tag --- .../assets/docs/guides/generate-metadata.js | 9 ++++- .../console/src/assets/docs/guides/index.ts | 8 ++++ .../docs/guides/third-party-oidc/README.mdx | 1 + .../docs/guides/third-party-oidc/index.ts | 12 ++++++ .../docs/guides/third-party-oidc/logo.svg | 12 ++++++ .../console/src/assets/docs/guides/types.ts | 3 ++ .../src/components/Guide/GuideCard/index.tsx | 5 ++- .../console/src/components/Guide/hooks.ts | 40 +++++++++++++++---- .../components/GuideDrawer/index.tsx | 4 +- .../components/CreateForm/index.tsx | 13 +++++- .../components/GuideLibrary/index.tsx | 16 +++++++- packages/console/src/types/applications.ts | 3 ++ .../tests/console/applications/constants.ts | 11 +++++ .../tests/console/applications/index.test.ts | 33 +++++++++++++++ .../de/translation/admin-console/guide.ts | 2 + .../en/translation/admin-console/guide.ts | 1 + .../es/translation/admin-console/guide.ts | 2 + .../fr/translation/admin-console/guide.ts | 2 + .../it/translation/admin-console/guide.ts | 2 + .../ja/translation/admin-console/guide.ts | 2 + .../ko/translation/admin-console/guide.ts | 2 + .../pl-pl/translation/admin-console/guide.ts | 2 + .../pt-br/translation/admin-console/guide.ts | 2 + .../pt-pt/translation/admin-console/guide.ts | 2 + .../ru/translation/admin-console/guide.ts | 2 + .../tr-tr/translation/admin-console/guide.ts | 2 + .../zh-cn/translation/admin-console/guide.ts | 2 + .../zh-hk/translation/admin-console/guide.ts | 2 + .../zh-tw/translation/admin-console/guide.ts | 3 ++ 29 files changed, 182 insertions(+), 18 deletions(-) create mode 100644 packages/console/src/assets/docs/guides/third-party-oidc/README.mdx create mode 100644 packages/console/src/assets/docs/guides/third-party-oidc/index.ts create mode 100644 packages/console/src/assets/docs/guides/third-party-oidc/logo.svg diff --git a/packages/console/src/assets/docs/guides/generate-metadata.js b/packages/console/src/assets/docs/guides/generate-metadata.js index 76eb4a05d..a0be81e57 100644 --- a/packages/console/src/assets/docs/guides/generate-metadata.js +++ b/packages/console/src/assets/docs/guides/generate-metadata.js @@ -24,7 +24,9 @@ const data = await Promise.all( const logo = ['logo.svg'].find((logo) => existsSync(`${directory}/${logo}`)); const config = existsSync(`${directory}/config.json`) - ? await import(`./${directory}/config.json`, { assert: { type: 'json' } }).then((module) => module.default) + ? await import(`./${directory}/config.json`, { assert: { type: 'json' } }).then( + (module) => module.default + ) : undefined; return { @@ -34,7 +36,10 @@ const data = await Promise.all( }; }) ); -const metadata = data.filter(Boolean).slice().sort((a, b) => a.order - b.order); +const metadata = data + .filter(Boolean) + .slice() + .sort((a, b) => a.order - b.order); const camelCase = (value) => value.replaceAll(/-./g, (x) => x[1].toUpperCase()); const filename = 'index.ts'; diff --git a/packages/console/src/assets/docs/guides/index.ts b/packages/console/src/assets/docs/guides/index.ts index 1dbb1b22a..fdd61ef65 100644 --- a/packages/console/src/assets/docs/guides/index.ts +++ b/packages/console/src/assets/docs/guides/index.ts @@ -14,6 +14,7 @@ import nativeIosSwift from './native-ios-swift/index'; import spaReact from './spa-react/index'; import spaVanilla from './spa-vanilla/index'; import spaVue from './spa-vue/index'; +import thirdPartyOidc from './third-party-oidc/index'; import { type Guide } from './types'; import webAspNetCore from './web-asp-net-core/index'; import webAspNetCoreMvc from './web-asp-net-core-mvc/index'; @@ -197,6 +198,13 @@ const guides: Readonly = Object.freeze([ Component: lazy(async () => import('./api-spring-boot/README.mdx')), metadata: apiSpringBoot, }, + { + order: Number.POSITIVE_INFINITY, + id: 'third-party-oidc', + Logo: lazy(async () => import('./third-party-oidc/logo.svg')), + Component: lazy(async () => import('./third-party-oidc/README.mdx')), + metadata: thirdPartyOidc, + }, ]); export default guides; diff --git a/packages/console/src/assets/docs/guides/third-party-oidc/README.mdx b/packages/console/src/assets/docs/guides/third-party-oidc/README.mdx new file mode 100644 index 000000000..4148efbf9 --- /dev/null +++ b/packages/console/src/assets/docs/guides/third-party-oidc/README.mdx @@ -0,0 +1 @@ +# Place holder for third party OIDC guide diff --git a/packages/console/src/assets/docs/guides/third-party-oidc/index.ts b/packages/console/src/assets/docs/guides/third-party-oidc/index.ts new file mode 100644 index 000000000..26a2c18a0 --- /dev/null +++ b/packages/console/src/assets/docs/guides/third-party-oidc/index.ts @@ -0,0 +1,12 @@ +import { ApplicationType } from '@logto/schemas'; + +import { type GuideMetadata } from '../types'; + +const metadata: Readonly = Object.freeze({ + name: 'OIDC', + description: 'Use Logto as a third-party OIDC identity provider (IdP) for your application. ', + target: ApplicationType.Traditional, + isThirdParty: true, +}); + +export default metadata; diff --git a/packages/console/src/assets/docs/guides/third-party-oidc/logo.svg b/packages/console/src/assets/docs/guides/third-party-oidc/logo.svg new file mode 100644 index 000000000..f6eee6e16 --- /dev/null +++ b/packages/console/src/assets/docs/guides/third-party-oidc/logo.svg @@ -0,0 +1,12 @@ + + + + + + + + + + + diff --git a/packages/console/src/assets/docs/guides/types.ts b/packages/console/src/assets/docs/guides/types.ts index 638e626ad..833bede67 100644 --- a/packages/console/src/assets/docs/guides/types.ts +++ b/packages/console/src/assets/docs/guides/types.ts @@ -27,6 +27,9 @@ export type GuideMetadata = { }; /** Whether the guide is displayed in featured group. */ isFeatured?: boolean; + + /** Indicate whether the application is for third-party use */ + isThirdParty?: boolean; }; /** The guide instance to build in the console. */ diff --git a/packages/console/src/components/Guide/GuideCard/index.tsx b/packages/console/src/components/Guide/GuideCard/index.tsx index 9cae58a13..eb0462e9e 100644 --- a/packages/console/src/components/Guide/GuideCard/index.tsx +++ b/packages/console/src/components/Guide/GuideCard/index.tsx @@ -11,6 +11,7 @@ export type SelectedGuide = { id: Guide['id']; target: GuideMetadata['target']; name: GuideMetadata['name']; + isThirdParty: GuideMetadata['isThirdParty']; }; type Props = { @@ -24,13 +25,13 @@ function GuideCard({ data, onClick, hasBorder, hasButton }: Props) { const { id, Logo, - metadata: { target, name, description }, + metadata: { target, name, description, isThirdParty }, } = data; const buttonText = target === 'API' ? 'guide.get_started' : 'guide.start_building'; const handleClick = useCallback(() => { - onClick({ id, target, name }); + onClick({ id, target, name, isThirdParty }); }, [id, name, target, onClick]); return ( diff --git a/packages/console/src/components/Guide/hooks.ts b/packages/console/src/components/Guide/hooks.ts index a6f750aca..4672be0b8 100644 --- a/packages/console/src/components/Guide/hooks.ts +++ b/packages/console/src/components/Guide/hooks.ts @@ -2,7 +2,12 @@ import { useCallback, useMemo } from 'react'; import guides from '@/assets/docs/guides'; import { type Guide } from '@/assets/docs/guides/types'; -import { type AppGuideCategory, type StructuredAppGuideMetadata } from '@/types/applications'; +import { isDevFeaturesEnabled } from '@/consts/env'; +import { + thirdPartyAppCategory, + type AppGuideCategory, + type StructuredAppGuideMetadata, +} from '@/types/applications'; const defaultStructuredMetadata: StructuredAppGuideMetadata = { featured: [], @@ -11,6 +16,7 @@ const defaultStructuredMetadata: StructuredAppGuideMetadata = { Native: [], MachineToMachine: [], Protected: [], + ThirdParty: [], }; type FilterOptions = { @@ -28,7 +34,12 @@ export const useAppGuideMetadata = (): { ) => Record; } => { const appGuides = useMemo( - () => guides.filter(({ metadata: { target } }) => target !== 'API'), + () => + guides.filter( + ({ metadata: { target, isThirdParty } }) => + // @simeng-li #FIXME: remove isDevFeaturesEnabled check once we have all guides ready + target !== 'API' && (isDevFeaturesEnabled || !isThirdParty) + ), [] ); @@ -49,11 +60,15 @@ export const useAppGuideMetadata = (): { // Categories only, return selected categories if (!keyword && filterCategories?.length) { - return appGuides.filter(({ metadata: { target, isFeatured } }) => - filterCategories.some( - (filterCategory) => - filterCategory === target || (isFeatured && filterCategory === 'featured') - ) + return appGuides.filter(({ metadata: { target, isFeatured, isThirdParty } }) => + filterCategories.some((filterCategory) => { + return ( + filterCategory === target || + // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing + (isFeatured && filterCategory === 'featured') || + (isThirdParty && filterCategory === 'ThirdParty') + ); + }) ); } @@ -76,13 +91,22 @@ export const useAppGuideMetadata = (): { const getStructuredAppGuideMetadata = useCallback( (filters?: FilterOptions) => { const filteredMetadata = getFilteredAppGuideMetadata(filters); + return filteredMetadata.reduce((accumulated, guide) => { - const { target, isFeatured } = guide.metadata; + const { target, isFeatured, isThirdParty } = guide.metadata; // Rule out API target guides to make TypeScript happy if (target === 'API') { return accumulated; } + + if (isThirdParty) { + return { + ...accumulated, + [thirdPartyAppCategory]: [...accumulated[thirdPartyAppCategory], guide], + }; + } + return { ...accumulated, [target]: [...accumulated[target], guide], diff --git a/packages/console/src/pages/ApplicationDetails/components/GuideDrawer/index.tsx b/packages/console/src/pages/ApplicationDetails/components/GuideDrawer/index.tsx index 3eaa5c430..19635b03c 100644 --- a/packages/console/src/pages/ApplicationDetails/components/GuideDrawer/index.tsx +++ b/packages/console/src/pages/ApplicationDetails/components/GuideDrawer/index.tsx @@ -39,9 +39,9 @@ function GuideDrawer({ app, onClose }: Props) { if (guide) { const { id, - metadata: { target, name }, + metadata: { target, name, isThirdParty }, } = guide; - setSelectedGuide({ id, target, name }); + setSelectedGuide({ id, target, name, isThirdParty }); } } }, [hasSingleGuide, app.type, structuredMetadata]); diff --git a/packages/console/src/pages/Applications/components/CreateForm/index.tsx b/packages/console/src/pages/Applications/components/CreateForm/index.tsx index 9bdfcb65f..db98efed4 100644 --- a/packages/console/src/pages/Applications/components/CreateForm/index.tsx +++ b/packages/console/src/pages/Applications/components/CreateForm/index.tsx @@ -25,21 +25,30 @@ type FormData = { type: ApplicationType; name: string; description?: string; + isThirdParty?: boolean; }; type Props = { + isDefaultCreateThirdParty?: boolean; defaultCreateType?: ApplicationType; defaultCreateFrameworkName?: string; onClose?: (createdApp?: Application) => void; }; -function CreateForm({ defaultCreateType, defaultCreateFrameworkName, onClose }: Props) { +function CreateForm({ + defaultCreateType, + defaultCreateFrameworkName, + isDefaultCreateThirdParty, + onClose, +}: Props) { const { handleSubmit, control, register, formState: { errors, isSubmitting }, - } = useForm({ defaultValues: { type: defaultCreateType } }); + } = useForm({ + defaultValues: { type: defaultCreateType, isThirdParty: isDefaultCreateThirdParty }, + }); const { field: { onChange, value, name, ref }, diff --git a/packages/console/src/pages/Applications/components/GuideLibrary/index.tsx b/packages/console/src/pages/Applications/components/GuideLibrary/index.tsx index 33f6f7695..d2d1e4462 100644 --- a/packages/console/src/pages/Applications/components/GuideLibrary/index.tsx +++ b/packages/console/src/pages/Applications/components/GuideLibrary/index.tsx @@ -16,6 +16,7 @@ import OverlayScrollbar from '@/ds-components/OverlayScrollbar'; import TextInput from '@/ds-components/TextInput'; import useTenantPathname from '@/hooks/use-tenant-pathname'; import { allAppGuideCategories, type AppGuideCategory } from '@/types/applications'; +import { thirdPartyAppCategory } from '@/types/applications'; import CreateForm from '../CreateForm'; @@ -56,7 +57,13 @@ function GuideLibrary({ className, hasCardBorder, hasCardButton, hasFilters }: P const onCloseCreateForm = useCallback( (newApp?: Application) => { if (newApp && selectedGuide) { - navigate(`/applications/${newApp.id}/guide/${selectedGuide.id}`, { replace: true }); + navigate( + // Third party app directly goes to the app detail page + selectedGuide.isThirdParty + ? `/applications/${newApp.id}` + : `/applications/${newApp.id}/guide/${selectedGuide.id}`, + { replace: true } + ); return; } setShowCreateForm(false); @@ -86,7 +93,11 @@ function GuideLibrary({ className, hasCardBorder, hasCardButton, hasFilters }: P isDevFeaturesEnabled || category !== 'Protected') + .filter( + (category) => + isDevFeaturesEnabled || + (category !== 'Protected' && category !== thirdPartyAppCategory) + ) .map((category) => ({ title: `guide.categories.${category}`, value: category, @@ -145,6 +156,7 @@ function GuideLibrary({ className, hasCardBorder, hasCardButton, hasFilters }: P )} diff --git a/packages/console/src/types/applications.ts b/packages/console/src/types/applications.ts index 267e3b5ce..89895a7e9 100644 --- a/packages/console/src/types/applications.ts +++ b/packages/console/src/types/applications.ts @@ -2,6 +2,8 @@ import { ApplicationType } from '@logto/schemas'; import { type Guide } from '@/assets/docs/guides/types'; +export const thirdPartyAppCategory = 'ThirdParty' as const; + export const applicationTypeI18nKey = Object.freeze({ [ApplicationType.Native]: 'applications.type.native', [ApplicationType.SPA]: 'applications.type.spa', @@ -22,6 +24,7 @@ export const allAppGuideCategories = Object.freeze([ 'Native', 'MachineToMachine', 'Protected', + thirdPartyAppCategory, ] as const); export type AppGuideCategory = (typeof allAppGuideCategories)[number]; diff --git a/packages/integration-tests/src/tests/console/applications/constants.ts b/packages/integration-tests/src/tests/console/applications/constants.ts index dc4c8720e..97da4e9bb 100644 --- a/packages/integration-tests/src/tests/console/applications/constants.ts +++ b/packages/integration-tests/src/tests/console/applications/constants.ts @@ -39,12 +39,23 @@ export const testApp: ApplicationCase = { postSignOutRedirectUri: 'https://my.test.app/sign-out', }; +export const thirdPartyApp: Omit< + ApplicationCase, + 'sample' | 'redirectUri' | 'postSignOutRedirectUri' +> = { + framework: 'OIDC', + name: 'OIDC third party app', + description: 'This is a OIDC third party app', + guideFilename: 'third-party-oidc', +}; + export const frameworkGroupLabels = [ 'Popular and for you', 'Traditional web app', 'Single page app', 'Native', 'Machine-to-machine', + 'Third-party', ] as const; export type ApplicationMetadata = { diff --git a/packages/integration-tests/src/tests/console/applications/index.test.ts b/packages/integration-tests/src/tests/console/applications/index.test.ts index 1a27990bb..b85c559c2 100644 --- a/packages/integration-tests/src/tests/console/applications/index.test.ts +++ b/packages/integration-tests/src/tests/console/applications/index.test.ts @@ -19,6 +19,7 @@ import { applicationTypesMetadata, initialApp, testApp, + thirdPartyApp, } from './constants.js'; import { expectFrameworkExists, @@ -260,6 +261,38 @@ describe('applications', () => { } ); + it('can create an third party application', async () => { + await expect(page).toClick('div[class$=main] div[class$=headline] button span', { + text: 'Create application', + }); + + await expectModalWithTitle(page, 'Start with SDK and guides'); + + await expectFrameworksInGroup(page, '.ReactModalPortal div[class$=guideGroup]:has(>label)'); + + // Expect the framework contains on the page + await expectFrameworkExists(page, thirdPartyApp.framework); + + // Filter + await expect(page).toFill('div[class$=searchInput] input', thirdPartyApp.framework); + + // Expect the framework exists after filtering + await expectFrameworkExists(page, thirdPartyApp.framework); + + await expectToChooseAndClickApplicationFramework(page, thirdPartyApp.framework); + + // Expect the app can be created successfully + await expectToProceedApplicationCreationFrom(page, thirdPartyApp); + + await expect(page).toMatchElement('div[class$=main] div[class$=header] div[class$=name]', { + text: thirdPartyApp.name, + }); + + await expectToProceedAppDeletion(page, thirdPartyApp.name); + + expect(page.url()).toBe(new URL('/console/applications', logtoConsoleUrl).href); + }); + it('delete the initial application', async () => { await expect(page).toClick('table tbody tr td div[class$=item] a[class$=title]', { text: initialApp.name, diff --git a/packages/phrases/src/locales/de/translation/admin-console/guide.ts b/packages/phrases/src/locales/de/translation/admin-console/guide.ts index 51a165804..29b6309f4 100644 --- a/packages/phrases/src/locales/de/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/de/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Native', MachineToMachine: 'Maschinen-zu-Maschinen', Protected: 'Geschützte App', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Filter-Framework', diff --git a/packages/phrases/src/locales/en/translation/admin-console/guide.ts b/packages/phrases/src/locales/en/translation/admin-console/guide.ts index 41b61d957..dcc17620b 100644 --- a/packages/phrases/src/locales/en/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/en/translation/admin-console/guide.ts @@ -8,6 +8,7 @@ const guide = { Native: 'Native', MachineToMachine: 'Machine-to-machine', Protected: 'Protected app', + ThirdParty: 'Third-party app', }, filter: { title: 'Filter framework', diff --git a/packages/phrases/src/locales/es/translation/admin-console/guide.ts b/packages/phrases/src/locales/es/translation/admin-console/guide.ts index 0fa217026..0d6c9cdcd 100644 --- a/packages/phrases/src/locales/es/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/es/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Nativa', MachineToMachine: 'Máquina a máquina', Protected: 'Aplicación protegida', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Filtrar framework', diff --git a/packages/phrases/src/locales/fr/translation/admin-console/guide.ts b/packages/phrases/src/locales/fr/translation/admin-console/guide.ts index d2a76d52f..ca18ac851 100644 --- a/packages/phrases/src/locales/fr/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/fr/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Natif', MachineToMachine: 'Machine-à-machine', Protected: 'Application protégée', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Filtrer le framework', diff --git a/packages/phrases/src/locales/it/translation/admin-console/guide.ts b/packages/phrases/src/locales/it/translation/admin-console/guide.ts index 3c1e5759e..54530f36d 100644 --- a/packages/phrases/src/locales/it/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/it/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Nativo', MachineToMachine: 'Dalla macchina alla macchina', Protected: 'App protetta', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Filtra framework', diff --git a/packages/phrases/src/locales/ja/translation/admin-console/guide.ts b/packages/phrases/src/locales/ja/translation/admin-console/guide.ts index 85f14b6fe..26c45b9cf 100644 --- a/packages/phrases/src/locales/ja/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/ja/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'ネイティブ', MachineToMachine: 'Machine-to-machine', Protected: '保護されたアプリ', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'フレームワークを絞り込む', diff --git a/packages/phrases/src/locales/ko/translation/admin-console/guide.ts b/packages/phrases/src/locales/ko/translation/admin-console/guide.ts index 565c8df76..e42dc71c0 100644 --- a/packages/phrases/src/locales/ko/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/ko/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: '네이티브', MachineToMachine: '기계 간 통신', Protected: '보호된 앱', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: '프레임워크 필터', diff --git a/packages/phrases/src/locales/pl-pl/translation/admin-console/guide.ts b/packages/phrases/src/locales/pl-pl/translation/admin-console/guide.ts index 2b4d3110f..887b90ec0 100644 --- a/packages/phrases/src/locales/pl-pl/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/pl-pl/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Natywna', MachineToMachine: 'Maszyna-do-maszyny', Protected: 'Aplikacja chroniona', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Filtr Framework', diff --git a/packages/phrases/src/locales/pt-br/translation/admin-console/guide.ts b/packages/phrases/src/locales/pt-br/translation/admin-console/guide.ts index 49b981df7..f48fe8cda 100644 --- a/packages/phrases/src/locales/pt-br/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/pt-br/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Nativo', MachineToMachine: 'Máquina a máquina', Protected: 'Aplicativo protegido', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Filtrar framework', diff --git a/packages/phrases/src/locales/pt-pt/translation/admin-console/guide.ts b/packages/phrases/src/locales/pt-pt/translation/admin-console/guide.ts index 4a781068e..b5d1e91a7 100644 --- a/packages/phrases/src/locales/pt-pt/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/pt-pt/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Nativo', MachineToMachine: 'Máquina-a-máquina', Protected: 'Aplicação protegida', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Filtrar framework', diff --git a/packages/phrases/src/locales/ru/translation/admin-console/guide.ts b/packages/phrases/src/locales/ru/translation/admin-console/guide.ts index ac2a84197..fe74c9555 100644 --- a/packages/phrases/src/locales/ru/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/ru/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Нативное', MachineToMachine: 'Машина к машине', Protected: 'Защищенное приложение', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: 'Фильтры фреймворков', diff --git a/packages/phrases/src/locales/tr-tr/translation/admin-console/guide.ts b/packages/phrases/src/locales/tr-tr/translation/admin-console/guide.ts index 5d91075df..c5fbf05f4 100644 --- a/packages/phrases/src/locales/tr-tr/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/tr-tr/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: 'Doğal', MachineToMachine: 'Makineden makineye', Protected: 'Korunan uygulama', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: "Framework'ü filtrele", diff --git a/packages/phrases/src/locales/zh-cn/translation/admin-console/guide.ts b/packages/phrases/src/locales/zh-cn/translation/admin-console/guide.ts index b585bad87..615c32b13 100644 --- a/packages/phrases/src/locales/zh-cn/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/zh-cn/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: '原生应用', MachineToMachine: 'Machine-to-machine', Protected: '受保护的应用', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: '筛选框架', diff --git a/packages/phrases/src/locales/zh-hk/translation/admin-console/guide.ts b/packages/phrases/src/locales/zh-hk/translation/admin-console/guide.ts index 60548e520..ec7aea2f1 100644 --- a/packages/phrases/src/locales/zh-hk/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/zh-hk/translation/admin-console/guide.ts @@ -8,6 +8,8 @@ const guide = { Native: '原生應用', MachineToMachine: '機器對機器', Protected: '受保護的應用程式', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: '篩選框架', diff --git a/packages/phrases/src/locales/zh-tw/translation/admin-console/guide.ts b/packages/phrases/src/locales/zh-tw/translation/admin-console/guide.ts index c3338458e..979bd879c 100644 --- a/packages/phrases/src/locales/zh-tw/translation/admin-console/guide.ts +++ b/packages/phrases/src/locales/zh-tw/translation/admin-console/guide.ts @@ -6,8 +6,11 @@ const guide = { Traditional: '傳統網頁應用', SPA: '單頁應用', Native: '原生應用', + /** UNTRANSLATED */ MachineToMachine: 'Machine-to-machine', Protected: '受保護的應用', + /** UNTRANSLATED */ + ThirdParty: 'Third-party app', }, filter: { title: '篩選框架',