+
{`+${countryCode}`}
-
-
+
);
};
diff --git a/packages/ui/src/components/InputFields/SmartInputField/index.test.tsx b/packages/ui/src/components/InputFields/SmartInputField/index.test.tsx
index 5456c5551..e92dd1089 100644
--- a/packages/ui/src/components/InputFields/SmartInputField/index.test.tsx
+++ b/packages/ui/src/components/InputFields/SmartInputField/index.test.tsx
@@ -1,7 +1,7 @@
import { SignInIdentifier } from '@logto/schemas';
import { Globals } from '@react-spring/web';
import { assert } from '@silverhand/essentials';
-import { fireEvent, render } from '@testing-library/react';
+import { act, fireEvent, render } from '@testing-library/react';
import { getBoundingClientRectMock } from '@/__mocks__/logto';
import { getDefaultCountryCallingCode } from '@/utils/country-code';
@@ -65,21 +65,25 @@ describe('SmartInputField Component', () => {
);
test('phone', async () => {
- const { container, queryAllByText, queryByTestId } = renderInputField({
+ const { container, getByText, queryByTestId } = renderInputField({
enabledTypes: [SignInIdentifier.Phone],
});
- const countryCode = queryAllByText(`+${defaultCountryCallingCode}`);
- expect(countryCode).toHaveLength(2);
+ const countryCode = getByText(`+${defaultCountryCallingCode}`);
+ expect(countryCode).not.toBeNull();
expect(queryByTestId('prefix')?.style.width).toBe('100px');
- const selector = container.querySelector('select');
- assert(selector, new Error('selector should not be null'));
+ act(() => {
+ fireEvent.click(countryCode);
+ });
const newCountryCode = '86';
- fireEvent.change(selector, { target: { value: newCountryCode } });
+ // Expect country code modal shown
+ const newCodeButton = getByText(`+${newCountryCode}`);
+ fireEvent.click(newCodeButton);
+
expect(onChange).toBeCalledWith({
type: SignInIdentifier.Phone,
value: '',
diff --git a/packages/ui/src/components/InputFields/SmartInputField/index.tsx b/packages/ui/src/components/InputFields/SmartInputField/index.tsx
index 35dc6bbe9..80c4eb9fd 100644
--- a/packages/ui/src/components/InputFields/SmartInputField/index.tsx
+++ b/packages/ui/src/components/InputFields/SmartInputField/index.tsx
@@ -75,8 +75,9 @@ const SmartInputField = (
{
- onCountryCodeChange(event);
+ inputRef={innerRef.current}
+ onChange={(value) => {
+ onCountryCodeChange(value);
innerRef.current?.focus();
}}
/>
diff --git a/packages/ui/src/components/InputFields/SmartInputField/use-smart-input-field.ts b/packages/ui/src/components/InputFields/SmartInputField/use-smart-input-field.ts
index ff499bfc5..e7ca1265b 100644
--- a/packages/ui/src/components/InputFields/SmartInputField/use-smart-input-field.ts
+++ b/packages/ui/src/components/InputFields/SmartInputField/use-smart-input-field.ts
@@ -96,8 +96,8 @@ const useSmartInputField = ({ _defaultType, defaultValue, enabledTypes }: Props)
[defaultType, currentType, enabledTypeSet]
);
- const onCountryCodeChange = useCallback>(
- ({ target: { value } }) => {
+ const onCountryCodeChange = useCallback(
+ (value: string) => {
if (currentType === SignInIdentifier.Phone) {
const code = value.replace(/\D/g, '');
setCountryCode(code);
diff --git a/packages/ui/src/components/NavBar/index.tsx b/packages/ui/src/components/NavBar/index.tsx
index 1720121a9..08e71431b 100644
--- a/packages/ui/src/components/NavBar/index.tsx
+++ b/packages/ui/src/components/NavBar/index.tsx
@@ -1,3 +1,4 @@
+import { useCallback } from 'react';
import { useTranslation } from 'react-i18next';
import { useNavigate } from 'react-router-dom';
@@ -10,20 +11,30 @@ import * as styles from './index.module.scss';
type Props = {
title?: string;
type?: 'back' | 'close';
+ onClose?: () => void;
};
-const NavBar = ({ title, type = 'back' }: Props) => {
+const NavBar = ({ title, type = 'back', onClose }: Props) => {
const navigate = useNavigate();
const { t } = useTranslation();
+
const isClosable = type === 'close';
- const clickHandler = () => {
+ const clickHandler = useCallback(() => {
+ if (onClose) {
+ onClose();
+
+ return;
+ }
+
if (isClosable) {
window.close();
+
+ return;
}
navigate(-1);
- };
+ }, [isClosable, navigate, onClose]);
return (
diff --git a/packages/ui/src/components/TermsLinks/index.tsx b/packages/ui/src/components/TermsLinks/index.tsx
index 3b831e20b..1f72237f3 100644
--- a/packages/ui/src/components/TermsLinks/index.tsx
+++ b/packages/ui/src/components/TermsLinks/index.tsx
@@ -5,14 +5,13 @@ import TextLink from '@/components/TextLink';
import * as styles from './index.module.scss';
type Props = {
- className?: string;
// eslint-disable-next-line react/boolean-prop-naming
inline?: boolean;
termsOfUseUrl?: string;
privacyPolicyUrl?: string;
};
-const TermsLinks = ({ className, inline, termsOfUseUrl, privacyPolicyUrl }: Props) => {
+const TermsLinks = ({ inline, termsOfUseUrl, privacyPolicyUrl }: Props) => {
const { t } = useTranslation();
return (
diff --git a/packages/ui/src/components/TextLink/index.tsx b/packages/ui/src/components/TextLink/index.tsx
index 8c8299033..b0a488adf 100644
--- a/packages/ui/src/components/TextLink/index.tsx
+++ b/packages/ui/src/components/TextLink/index.tsx
@@ -1,10 +1,14 @@
import classNames from 'classnames';
+import { useMemo } from 'react';
import type { ReactNode, AnchorHTMLAttributes } from 'react';
import type { TFuncKey } from 'react-i18next';
import { useTranslation } from 'react-i18next';
import type { LinkProps } from 'react-router-dom';
import { Link } from 'react-router-dom';
+import { useIframeModal } from '@/Providers/IframeModalProvider';
+import usePlatform from '@/hooks/use-platform';
+
import * as styles from './index.module.scss';
export type Props = AnchorHTMLAttributes
& {
@@ -17,6 +21,34 @@ export type Props = AnchorHTMLAttributes & {
const TextLink = ({ className, children, text, icon, type = 'primary', to, ...rest }: Props) => {
const { t } = useTranslation();
+ const { isMobile } = usePlatform();
+ const { setModalState } = useIframeModal();
+
+ // By default the behavior of opening a new window is not supported in WkWebView, or in android webview.
+ // Hijack the hyperlink props and open the link in an iframe modal instead.
+ const hyperLinkProps = useMemo(() => {
+ const { href, target, onClick, ...others } = rest;
+
+ // Keep the original behavior if the link is not external.
+ if (!href || target !== '_blank') {
+ return rest;
+ }
+
+ return {
+ href,
+ target,
+ onClick: (event: React.MouseEvent) => {
+ if (isMobile) {
+ const title = text && t(text);
+ event.preventDefault();
+ setModalState({ href, title: typeof title === 'string' ? title : undefined });
+ }
+
+ onClick?.(event);
+ },
+ ...others,
+ };
+ }, [isMobile, rest, setModalState, t, text]);
if (to) {
return (
@@ -28,7 +60,11 @@ const TextLink = ({ className, children, text, icon, type = 'primary', to, ...re
}
return (
-
+
{icon}
{children ?? (text ? t(text) : '')}
diff --git a/packages/ui/src/hooks/use-debounce.ts b/packages/ui/src/hooks/use-debounce.ts
new file mode 100644
index 000000000..963bda7d6
--- /dev/null
+++ b/packages/ui/src/hooks/use-debounce.ts
@@ -0,0 +1,44 @@
+/**
+ * Original Reference: https://github.com/juliencrn/usehooks-ts/blob/master/src/useDebounce/useDebounce.ts
+ *
+ * The MIT License (MIT)
+ *
+ * Copyright (c) 2020 Julien CARON
+ *Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to deal
+ * in the Software without restriction, including without limitation the rights
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+ * copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in all
+ * copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+ * SOFTWARE.
+ */
+
+import { useEffect, useState } from 'react';
+
+function useDebounce(value: T, delay?: number): T {
+ const [debouncedValue, setDebouncedValue] = useState(value);
+
+ useEffect(() => {
+ const timer = setTimeout(() => {
+ setDebouncedValue(value);
+ }, delay ?? 500);
+
+ return () => {
+ clearTimeout(timer);
+ };
+ }, [value, delay]);
+
+ return debouncedValue;
+}
+
+export default useDebounce;
diff --git a/packages/ui/src/pages/ForgotPassword/index.test.tsx b/packages/ui/src/pages/ForgotPassword/index.test.tsx
index feeb9715f..4033d8427 100644
--- a/packages/ui/src/pages/ForgotPassword/index.test.tsx
+++ b/packages/ui/src/pages/ForgotPassword/index.test.tsx
@@ -81,7 +81,7 @@ describe('ForgotPassword', () => {
test.each(stateCases)('render the forgot password page with state %o', async (state) => {
mockUseLocation.mockImplementation(() => ({ state }));
- const { queryByText, queryAllByText, container, queryByTestId } = renderPage(settings);
+ const { queryByText, container, queryByTestId } = renderPage(settings);
const inputField = container.querySelector('input[name="identifier"]');
const countryCodeSelectorPrefix = queryByTestId('prefix');
@@ -95,7 +95,7 @@ describe('ForgotPassword', () => {
if (state.identifier === SignInIdentifier.Phone && settings.phone) {
expect(inputField.getAttribute('value')).toBe(phone);
expect(countryCodeSelectorPrefix?.style.width).toBe('100px');
- expect(queryAllByText(`+${countryCode}`)).toHaveLength(2);
+ expect(queryByText(`+${countryCode}`)).not.toBeNull();
} else if (state.identifier === SignInIdentifier.Phone) {
// Phone Number not enabled
expect(inputField.getAttribute('value')).toBe('');
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index 800e7beab..a1d3175b2 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -925,6 +925,7 @@ importers:
react-router-dom: ^6.2.2
react-string-replace: ^1.0.0
react-timer-hook: ^3.0.5
+ react-top-loading-bar: ^2.3.1
stylelint: ^15.0.0
superstruct: ^0.16.0
ts-jest: ^29.0.5
@@ -989,6 +990,7 @@ importers:
react-router-dom: 6.2.2_biqbaboplfbrettd7655fr4n2y
react-string-replace: 1.0.0
react-timer-hook: 3.0.5_biqbaboplfbrettd7655fr4n2y
+ react-top-loading-bar: 2.3.1_react@18.2.0
stylelint: 15.0.0
superstruct: 0.16.0
ts-jest: 29.0.5_cdjgginuefokmzmklysahvrmme
@@ -12389,6 +12391,15 @@ packages:
react-dom: 18.2.0_react@18.2.0
dev: true
+ /react-top-loading-bar/2.3.1_react@18.2.0:
+ resolution: {integrity: sha512-rQk2Nm+TOBrM1C4E3e6KwT65iXyRSgBHjCkr2FNja1S51WaPulRA5nKj/xazuQ3x89wDDdGsrqkqy0RBIfd0xg==}
+ engines: {node: '>=10'}
+ peerDependencies:
+ react: ^16 || ^17 || ^18 || ^18.0.0
+ dependencies:
+ react: 18.2.0
+ dev: true
+
/react-transition-group/2.9.0_biqbaboplfbrettd7655fr4n2y:
resolution: {integrity: sha512-+HzNTCHpeQyl4MJ/bdE0u6XRMe9+XG/+aL4mCxVN4DnPBQ0/5bfHWPDuOZUzYdMj94daZaZdCCc1Dzt9R/xSSg==}
peerDependencies: