Compare commits
21 Commits
Author | SHA1 | Date |
---|---|---|
|
73229cbe19 | |
|
c89584dba0 | |
|
83965dd675 | |
|
a96a49d649 | |
|
f8c797abf3 | |
|
6b8cad1081 | |
|
ccc3e2cb79 | |
|
9ecb7d6981 | |
|
fc743dbb7d | |
|
28789e9054 | |
|
4e849b47a2 | |
|
33bf78ecc3 | |
|
42a4ae0c6c | |
|
fdb464ae68 | |
|
a6bba53dd2 | |
|
eff29677dc | |
|
2da77f7347 | |
|
87b14e8716 | |
|
52fba3a879 | |
|
61de5c81e7 | |
|
6a9bed479a |
6
.env
6
.env
|
@ -1,9 +1,11 @@
|
||||||
NEXT_PUBLIC_SERVER_BASE_URL=https://api.bbuddy.expert/api
|
NEXT_PUBLIC_SERVER_BASE_URL=https://api.bbuddy.expert/api
|
||||||
NEXT_PUBLIC_AGORA_APPID=ed90c9dc42634e5687d4e2e0766b363f
|
NEXT_PUBLIC_AGORA_APPID=ed90c9dc42634e5687d4e2e0766b363f
|
||||||
|
NEXT_PUBLIC_GOOGLE_CLIENT_ID=909563069647-03rivr8k1jmirf382bcfehegamthcfg4.apps.googleusercontent.com
|
||||||
NEXT_PUBLIC_STRIPE_PUBLISHABLE_KEY=pk_test_51LVB3LK5pVGxNPeKk4gedt5NW4cb8k7BVXvgOMPTK4x1nnbGTD8BCqDqgInboT6N72YwrTl4tOsVz8rAjbUadX1m00y4Aq5qE8
|
NEXT_PUBLIC_STRIPE_PUBLISHABLE_KEY=pk_test_51LVB3LK5pVGxNPeKk4gedt5NW4cb8k7BVXvgOMPTK4x1nnbGTD8BCqDqgInboT6N72YwrTl4tOsVz8rAjbUadX1m00y4Aq5qE8
|
||||||
STRIPE_SECRET_KEY=sk_test_51LVB3LK5pVGxNPeK6j0wCsPqYMoGfcuwf1LpwGEBsr1dUx4NngukyjYL2oMZer5EOlW3lqnVEPjNDruN0OkUohIf00fWFUHN5O
|
NEXT_PUBLIC_STRIPE_SECRET_KEY=sk_test_51LVB3LK5pVGxNPeK6j0wCsPqYMoGfcuwf1LpwGEBsr1dUx4NngukyjYL2oMZer5EOlW3lqnVEPjNDruN0OkUohIf00fWFUHN5O
|
||||||
STRIPE_PAYMENT_DESCRIPTION='BBuddy services'
|
NEXT_PUBLIC_STRIPE_PAYMENT_DESCRIPTION='BBuddy services'
|
||||||
|
|
||||||
NEXT_PUBLIC_CONTENTFUL_SPACE_ID = voxpxjq7y7vf
|
NEXT_PUBLIC_CONTENTFUL_SPACE_ID = voxpxjq7y7vf
|
||||||
NEXT_PUBLIC_CONTENTFUL_ACCESS_TOKEN = s99GWKfpDKkNwiEJ3pN7US_tmqsGvDlaex-sOJwpzuc
|
NEXT_PUBLIC_CONTENTFUL_ACCESS_TOKEN = s99GWKfpDKkNwiEJ3pN7US_tmqsGvDlaex-sOJwpzuc
|
||||||
NEXT_PUBLIC_CONTENTFUL_PREVIEW_ACCESS_TOKEN = Z9WOKpLDbKNj7xVOmT_VXYNLH0AZwISFvQsq0PQlHfE
|
NEXT_PUBLIC_CONTENTFUL_PREVIEW_ACCESS_TOKEN = Z9WOKpLDbKNj7xVOmT_VXYNLH0AZwISFvQsq0PQlHfE
|
||||||
|
|
||||||
|
|
|
@ -38,3 +38,5 @@ yarn-error.log*
|
||||||
# typescript
|
# typescript
|
||||||
*.tsbuildinfo
|
*.tsbuildinfo
|
||||||
next-env.d.ts
|
next-env.d.ts
|
||||||
|
|
||||||
|
certificates
|
|
@ -1,5 +1,6 @@
|
||||||
// @ts-check
|
// @ts-check
|
||||||
const withNextIntl = require('next-intl/plugin')();
|
const createNextIntlPlugin = require('next-intl/plugin');
|
||||||
|
const withNextIntl = createNextIntlPlugin();
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
const json = require('./package.json');
|
const json = require('./package.json');
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "bbuddy-ui",
|
"name": "bbuddy-ui",
|
||||||
"version": "0.2.5",
|
"version": "0.3.5",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev -p 4200",
|
"dev": "next dev -p 4200",
|
||||||
|
@ -13,6 +13,8 @@
|
||||||
"@ant-design/icons": "^5.2.6",
|
"@ant-design/icons": "^5.2.6",
|
||||||
"@ant-design/nextjs-registry": "^1.0.0",
|
"@ant-design/nextjs-registry": "^1.0.0",
|
||||||
"@contentful/rich-text-react-renderer": "^15.22.9",
|
"@contentful/rich-text-react-renderer": "^15.22.9",
|
||||||
|
"@microsoft/signalr": "^8.0.7",
|
||||||
|
"@react-oauth/google": "^0.12.1",
|
||||||
"@stripe/react-stripe-js": "^2.7.3",
|
"@stripe/react-stripe-js": "^2.7.3",
|
||||||
"@stripe/stripe-js": "^4.1.0",
|
"@stripe/stripe-js": "^4.1.0",
|
||||||
"agora-rtc-react": "2.1.0",
|
"agora-rtc-react": "2.1.0",
|
||||||
|
@ -27,7 +29,9 @@
|
||||||
"next": "14.0.3",
|
"next": "14.0.3",
|
||||||
"next-intl": "^3.3.1",
|
"next-intl": "^3.3.1",
|
||||||
"react": "^18",
|
"react": "^18",
|
||||||
|
"react-apple-login": "^1.1.6",
|
||||||
"react-dom": "^18",
|
"react-dom": "^18",
|
||||||
|
"react-signalr": "^0.2.24",
|
||||||
"react-slick": "^0.29.0",
|
"react-slick": "^0.29.0",
|
||||||
"react-stripe-js": "^1.1.5",
|
"react-stripe-js": "^1.1.5",
|
||||||
"slick-carousel": "^1.8.1",
|
"slick-carousel": "^1.8.1",
|
||||||
|
|
|
@ -12,3 +12,39 @@ export const getRegister = (locale: string): Promise<{ jwtToken: string }> => ap
|
||||||
method: 'post',
|
method: 'post',
|
||||||
locale
|
locale
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const getRegisterByGoogle = (locale: string, accesstoken: string): Promise<{ jwtToken: string }> => apiRequest({
|
||||||
|
url: '/auth/registerexternal',
|
||||||
|
method: 'post',
|
||||||
|
data: {
|
||||||
|
platform: 0,
|
||||||
|
provider: 4,
|
||||||
|
accesstoken
|
||||||
|
},
|
||||||
|
locale
|
||||||
|
});
|
||||||
|
|
||||||
|
export const getLoginByGoogle = (locale: string, accesstoken: string): Promise<{ jwtToken: string }> => apiRequest({
|
||||||
|
url: '/auth/tryloginexternal',
|
||||||
|
method: 'post',
|
||||||
|
data: {
|
||||||
|
platform: 0,
|
||||||
|
provider: 4,
|
||||||
|
accesstoken
|
||||||
|
},
|
||||||
|
locale
|
||||||
|
});
|
||||||
|
|
||||||
|
export const getRegisterByApple = (locale: string, code: string): Promise<{ jwtToken: string }> => apiRequest({
|
||||||
|
url: '/auth/registerexternalappleweb',
|
||||||
|
method: 'post',
|
||||||
|
data: { code },
|
||||||
|
locale
|
||||||
|
});
|
||||||
|
|
||||||
|
export const getLoginByApple = (locale: string, code: string): Promise<{ jwtToken: string }> => apiRequest({
|
||||||
|
url: '/auth/tryloginexternalappleweb',
|
||||||
|
method: 'post',
|
||||||
|
data: { code },
|
||||||
|
locale
|
||||||
|
});
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
import {apiRequest} from "../helpers";
|
||||||
|
|
||||||
|
|
||||||
|
export const getChatList = (locale: string, token: string): Promise<any> => apiRequest({
|
||||||
|
url: '/chat/chatList',
|
||||||
|
method: 'get',
|
||||||
|
locale,
|
||||||
|
token
|
||||||
|
});
|
||||||
|
|
||||||
|
export const getChatMessages = (locale: string, token: string, group: number): Promise<any> => apiRequest({
|
||||||
|
url: '/chat/chat_messages/'+group,
|
||||||
|
method: 'get',
|
||||||
|
locale,
|
||||||
|
token
|
||||||
|
});
|
|
@ -23,22 +23,22 @@ export const apiRequest = async <T = any, K = any>(
|
||||||
|
|
||||||
return response.data;
|
return response.data;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
// const {
|
const {
|
||||||
// response: {
|
response: {
|
||||||
// status: responseCode = null,
|
status: responseCode = null,
|
||||||
// statusText = '',
|
statusText = '',
|
||||||
// data: { message = '', status: errorKey = '' } = {},
|
data,
|
||||||
// } = {},
|
} = {},
|
||||||
// code: statusCode = '',
|
code: statusCode = '',
|
||||||
// } = err as AxiosError;
|
} = err as AxiosError;
|
||||||
//
|
|
||||||
// throw new Error(
|
throw new Error(
|
||||||
// JSON.stringify({
|
JSON.stringify({
|
||||||
// statusCode,
|
statusCode,
|
||||||
// statusMessage: message || statusText,
|
statusMessage: statusText,
|
||||||
// responseCode,
|
responseCode,
|
||||||
// errorKey,
|
details: data
|
||||||
// }),
|
}),
|
||||||
// );
|
);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -23,20 +23,20 @@ export const useSessionTracking = (locale: string, sessionId: number) => {
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
return () => {
|
return () => {
|
||||||
window.clearInterval(timer.current);
|
window?.clearInterval(timer.current);
|
||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const start = () => {
|
const start = () => {
|
||||||
window.clearInterval(timer.current);
|
window?.clearInterval(timer.current);
|
||||||
|
|
||||||
timer.current = window.setInterval(() => {
|
timer.current = window?.setInterval(() => {
|
||||||
fetchData();
|
fetchData();
|
||||||
}, DURATION);
|
}, DURATION);
|
||||||
};
|
};
|
||||||
|
|
||||||
const stop = () => {
|
const stop = () => {
|
||||||
window.clearInterval(timer.current);
|
window?.clearInterval(timer.current);
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
|
@ -13,7 +13,6 @@ export async function createCheckoutSession(
|
||||||
const ui_mode = data.get(
|
const ui_mode = data.get(
|
||||||
"uiMode",
|
"uiMode",
|
||||||
) as Stripe.Checkout.SessionCreateParams.UiMode;
|
) as Stripe.Checkout.SessionCreateParams.UiMode;
|
||||||
console.log('DATA', data)
|
|
||||||
const origin: string = headers().get("origin") as string;
|
const origin: string = headers().get("origin") as string;
|
||||||
|
|
||||||
const checkoutSession: Stripe.Checkout.Session =
|
const checkoutSession: Stripe.Checkout.Session =
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
|
|
||||||
export default function Directions({ params: { locale } }: { params: { locale: string } }) {
|
export default function Directions() {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="main-popular">
|
<div className="main-popular">
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { Experts } from '../../../../components/Experts/Experts';
|
import { Experts } from '../../../../components/Experts/Experts';
|
||||||
|
|
||||||
export default function ExpertsPage({ params: { locale } }: { params: { locale: string } }) {
|
export default function ExpertsPage({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const t = useTranslations('Experts');
|
const t = useTranslations('Experts');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
// import { useTranslations } from 'next-intl';
|
// import { useTranslations } from 'next-intl';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import { getTranslations, unstable_setRequestLocale } from 'next-intl/server';
|
import { getTranslations } from 'next-intl/server';
|
||||||
import { i18nText } from '../../../../i18nKeys';
|
import { i18nText } from '../../../../i18nKeys';
|
||||||
import { fetchBlogPosts } from '../../../../lib/contentful/blogPosts';
|
import { fetchBlogPosts } from '../../../../lib/contentful/blogPosts';
|
||||||
|
|
||||||
export default async function News({params: {locale}}: { params: { locale: string } }) {
|
export default async function News({params: {locale}}: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const t = await getTranslations('Main');
|
const t = await getTranslations('Main');
|
||||||
const { data, total } = await fetchBlogPosts({preview: false, sticky: true})
|
const { data, total } = await fetchBlogPosts({preview: false, sticky: true})
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Link } from '../../../../../../navigation';
|
import { Link } from '../../../../../../i18n/routing';
|
||||||
import { CustomSelect } from '../../../../../../components/view/CustomSelect';
|
import { CustomSelect } from '../../../../../../components/view/CustomSelect';
|
||||||
|
|
||||||
export default function AddOffer() {
|
export default function AddOffer() {
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { Link } from '../../../../../../navigation';
|
import { Link } from '../../../../../../i18n/routing';
|
||||||
import { CustomSelect } from '../../../../../../components/view/CustomSelect';
|
import { CustomSelect } from '../../../../../../components/view/CustomSelect';
|
||||||
|
|
||||||
export default function NewTopic() {
|
export default function NewTopic() {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { i18nText } from '../../../../../i18nKeys';
|
import { i18nText } from '../../../../../i18nKeys';
|
||||||
|
|
||||||
|
@ -10,7 +10,7 @@ export const metadata: Metadata = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function Information({ params: { locale } }: { params: { locale: string } }) {
|
export default function Information({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const t = useTranslations('Account.LegalInformation');
|
const t = useTranslations('Account.LegalInformation');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
|
'use client'
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
import { Link } from '../../../../../../i18n/routing';
|
||||||
import { Link } from '../../../../../../navigation';
|
|
||||||
import { i18nText } from '../../../../../../i18nKeys';
|
import { i18nText } from '../../../../../../i18nKeys';
|
||||||
|
import {ChatMessages} from "../../../../../../components/Chat/ChatMessages";
|
||||||
|
/*
|
||||||
export function generateStaticParams({
|
export function generateStaticParams({
|
||||||
params: { locale },
|
params: { locale },
|
||||||
}: { params: { locale: string } }) {
|
}: { params: { locale: string } }) {
|
||||||
|
@ -15,56 +16,22 @@ export function generateStaticParams({
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
export default function Message({ params }: { params: { locale: string, textId: string } }) {
|
export default function Message({ params: { locale, textId } }: { params: { locale: string, textId: string } }) {
|
||||||
unstable_setRequestLocale(params.locale);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ol className="breadcrumb">
|
<ol className="breadcrumb">
|
||||||
<li className="breadcrumb-item">
|
<li className="breadcrumb-item">
|
||||||
<Link href={'/account/messages' as any}>
|
<Link href={'/account/messages' as any}>
|
||||||
{i18nText('accountMenu.messages', params.locale)}
|
{i18nText('accountMenu.messages', locale)}
|
||||||
</Link>
|
</Link>
|
||||||
</li>
|
</li>
|
||||||
<li className="breadcrumb-item active" aria-current="page">{`Person ${params.textId}`}</li>
|
|
||||||
</ol>
|
</ol>
|
||||||
|
|
||||||
<div className="b-message">
|
<ChatMessages locale={locale} groupId={parseInt(textId)} />
|
||||||
<div className="b-message__inner">
|
|
||||||
<div className="b-message__list b-message__list--me">
|
|
||||||
<div className="b-message__item ">
|
|
||||||
<div className="b-message__avatar">
|
|
||||||
<img src="/images/person.png" className="" alt="" />
|
|
||||||
</div>
|
|
||||||
<div className="b-message__text">
|
|
||||||
🤩 It all for you!
|
|
||||||
|
|
||||||
<span className="date">07.09.2022</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className="b-message__list">
|
|
||||||
<div className="b-message__item">
|
|
||||||
<div className="b-message__avatar">
|
|
||||||
<img src="/images/person.png" className="" alt="" />
|
|
||||||
</div>
|
|
||||||
<div className="b-message__text">
|
|
||||||
🤩 It all for you!
|
|
||||||
<span className="date">07.09.2022</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<form className="b-message__form" action="">
|
|
||||||
<textarea placeholder="Type your message here" name="" id="" />
|
|
||||||
<label className="b-message__upload-file">
|
|
||||||
<input type="file" required />
|
|
||||||
</label>
|
|
||||||
<div className="b-message__microphone" />
|
|
||||||
<button className="b-message__btn" type="submit" />
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,12 +1,10 @@
|
||||||
|
'use client'
|
||||||
import React, { Suspense } from 'react';
|
import React, { Suspense } from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
|
||||||
import { Link } from '../../../../../navigation';
|
|
||||||
import { CustomInput } from '../../../../../components/view/CustomInput';
|
import { CustomInput } from '../../../../../components/view/CustomInput';
|
||||||
import { i18nText } from '../../../../../i18nKeys';
|
import { i18nText } from '../../../../../i18nKeys';
|
||||||
|
import {ChatList} from "../../../../../components/Chat/ChatList";
|
||||||
|
|
||||||
export default function Messages({ params: { locale } }: { params: { locale: string } }) {
|
export default function Messages({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ol className="breadcrumb">
|
<ol className="breadcrumb">
|
||||||
|
@ -15,74 +13,7 @@ export default function Messages({ params: { locale } }: { params: { locale: str
|
||||||
<Suspense>
|
<Suspense>
|
||||||
<CustomInput placeholder={i18nText('name', locale)} />
|
<CustomInput placeholder={i18nText('name', locale)} />
|
||||||
</Suspense>
|
</Suspense>
|
||||||
<div className="messages-session">
|
<ChatList locale={locale}/>
|
||||||
<Link
|
|
||||||
className="card-profile"
|
|
||||||
href={'messages/1' as any}
|
|
||||||
>
|
|
||||||
<div className="card-profile__header">
|
|
||||||
<div className="card-profile__header__portrait">
|
|
||||||
<img src="/images/person.png" className="" alt="" />
|
|
||||||
</div>
|
|
||||||
<div className="card-profile__header__inner">
|
|
||||||
<div style={{ width: '100%' }}>
|
|
||||||
<div className="card-profile__header__name">
|
|
||||||
David
|
|
||||||
<span className="count">14</span>
|
|
||||||
</div>
|
|
||||||
<div className="card-profile__header__title">
|
|
||||||
Lorem ipsum dolor sit at, consecte...
|
|
||||||
</div>
|
|
||||||
<div className="card-profile__header__date ">
|
|
||||||
25 may
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</Link>
|
|
||||||
<Link
|
|
||||||
className="card-profile"
|
|
||||||
href={'messages/2' as any}
|
|
||||||
>
|
|
||||||
<div className="card-profile__header">
|
|
||||||
<div className="card-profile__header__portrait">
|
|
||||||
<img src="/images/person.png" className="" alt="" />
|
|
||||||
</div>
|
|
||||||
<div className="card-profile__header__inner">
|
|
||||||
<div style={{ width: '100%' }}>
|
|
||||||
<div className="card-profile__header__name">David</div>
|
|
||||||
<div className="card-profile__header__title">
|
|
||||||
Lorem ipsum dolor sit at, consecte...
|
|
||||||
</div>
|
|
||||||
<div className="card-profile__header__date ">
|
|
||||||
25 may
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</Link>
|
|
||||||
<Link
|
|
||||||
className="card-profile"
|
|
||||||
href={'messages/3' as any}
|
|
||||||
>
|
|
||||||
<div className="card-profile__header">
|
|
||||||
<div className="card-profile__header__portrait">
|
|
||||||
<img src="/images/person.png" className="" alt="" />
|
|
||||||
</div>
|
|
||||||
<div className="card-profile__header__inner">
|
|
||||||
<div style={{ width: '100%' }}>
|
|
||||||
<div className="card-profile__header__name">David</div>
|
|
||||||
<div className="card-profile__header__title">
|
|
||||||
Lorem ipsum dolor sit at, consecte...
|
|
||||||
</div>
|
|
||||||
<div className="card-profile__header__date ">
|
|
||||||
25 may
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
import 'dayjs/locale/ru';
|
import 'dayjs/locale/ru';
|
||||||
import 'dayjs/locale/en';
|
import 'dayjs/locale/en';
|
||||||
|
@ -16,7 +16,7 @@ export const metadata: Metadata = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function Notifications({ params: { locale } }: { params: { locale: string } }) {
|
export default function Notifications({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const date = dayjs('2022-05-22').locale(locale);
|
const date = dayjs('2022-05-22').locale(locale);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { Link } from '../../../../../../navigation';
|
import { Link } from '../../../../../../i18n/routing';
|
||||||
import { i18nText } from '../../../../../../i18nKeys/';
|
import { i18nText } from '../../../../../../i18nKeys/';
|
||||||
|
|
||||||
export default function ChangePassword({ params: { locale } }: { params: { locale: string } }) {
|
export default function ChangePassword({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
import React, { Suspense } from 'react';
|
import React, { Suspense } from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { ProfileSettings } from '../../../../../components/Account';
|
import { ProfileSettings } from '../../../../../components/Account';
|
||||||
import { i18nText } from '../../../../../i18nKeys';
|
import { i18nText } from '../../../../../i18nKeys';
|
||||||
|
|
||||||
export default function Settings({ params: { locale } }: { params: { locale: string } }) {
|
export default function Settings({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { i18nText } from '../../../../../i18nKeys';
|
import { i18nText } from '../../../../../i18nKeys';
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ export const metadata: Metadata = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function Support({ params: { locale } }: { params: { locale: string } }) {
|
export default function Support({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import React, { Suspense } from 'react';
|
import React, { Suspense } from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { notFound } from 'next/navigation';
|
import { notFound } from 'next/navigation';
|
||||||
import { AccountMenu, RoomDetails, RoomsTabs } from '../../../../../../components/Account';
|
import { AccountMenu, RoomDetails, RoomsTabs } from '../../../../../../components/Account';
|
||||||
import { RoomsType } from '../../../../../../types/rooms';
|
import { RoomsType } from '../../../../../../types/rooms';
|
||||||
|
@ -13,7 +13,7 @@ export async function generateStaticParams({
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function RoomsDetailItem({ params: { locale, slug } }: { params: { locale: string, slug?: string[] } }) {
|
export default function RoomsDetailItem({ params: { locale, slug } }: { params: { locale: string, slug?: string[] } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const roomType: string = slug?.length > 0 && slug[0] || '';
|
const roomType: string = slug?.length > 0 && slug[0] || '';
|
||||||
const roomId: number | null = slug?.length > 1 && Number(slug[1]) || null;
|
const roomId: number | null = slug?.length > 1 && Number(slug[1]) || null;
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import React, { Suspense } from 'react';
|
import React, { Suspense } from 'react';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { notFound } from 'next/navigation';
|
import { notFound } from 'next/navigation';
|
||||||
import { AccountMenu, SessionDetails, SessionsTabs } from '../../../../../../components/Account';
|
import { AccountMenu, SessionDetails, SessionsTabs } from '../../../../../../components/Account';
|
||||||
import { SessionType } from '../../../../../../types/sessions';
|
import { SessionType } from '../../../../../../types/sessions';
|
||||||
|
@ -13,7 +13,7 @@ export async function generateStaticParams({
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function SessionDetailItem({ params: { locale, slug } }: { params: { locale: string, slug?: string[] } }) {
|
export default function SessionDetailItem({ params: { locale, slug } }: { params: { locale: string, slug?: string[] } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const sessionType: string = slug?.length > 0 && slug[0] || '';
|
const sessionType: string = slug?.length > 0 && slug[0] || '';
|
||||||
const sessionId: number | null = slug?.length > 1 && Number(slug[1]) || null;
|
const sessionId: number | null = slug?.length > 1 && Number(slug[1]) || null;
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { GeneralTopSection } from '../../../components/Page';
|
import { GeneralTopSection } from '../../../components/Page';
|
||||||
|
|
||||||
|
@ -9,8 +9,8 @@ export const metadata: Metadata = {
|
||||||
description: 'Bbuddy desc Take the lead with BB'
|
description: 'Bbuddy desc Take the lead with BB'
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function BbClientPage({ params: { locale } }: { params: { locale: string } }) {
|
export default function BbClientPage() {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const t = useTranslations('BbClient');
|
const t = useTranslations('BbClient');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { GeneralTopSection } from '../../../components/Page';
|
import { GeneralTopSection } from '../../../components/Page';
|
||||||
import { ScreenCarousel } from '../../../components/Page/ScreenCarousel';
|
import { ScreenCarousel } from '../../../components/Page/ScreenCarousel';
|
||||||
|
@ -10,8 +10,8 @@ export const metadata: Metadata = {
|
||||||
description: 'Bbuddy desc Become a BB expert'
|
description: 'Bbuddy desc Become a BB expert'
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function BbExpertPage({ params: { locale } }: { params: { locale: string } }) {
|
export default function BbExpertPage() {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const t = useTranslations('BbExpert');
|
const t = useTranslations('BbExpert');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -50,7 +50,6 @@ function renderWidget (widget: Widget, index: number) {
|
||||||
|
|
||||||
export default async function BlogItem({params}: { params: BlogPostPageParams }) {
|
export default async function BlogItem({params}: { params: BlogPostPageParams }) {
|
||||||
const item = await fetchBlogPost({slug: params.slug, preview: draftMode().isEnabled })
|
const item = await fetchBlogPost({slug: params.slug, preview: draftMode().isEnabled })
|
||||||
console.log('BLOG POST')
|
|
||||||
console.log(Util.inspect(item, {showHidden: false, depth: null, colors: true}))
|
console.log(Util.inspect(item, {showHidden: false, depth: null, colors: true}))
|
||||||
if (!item) notFound();
|
if (!item) notFound();
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { draftMode } from 'next/headers'
|
import { draftMode } from 'next/headers'
|
||||||
import {unstable_setRequestLocale} from "next-intl/server";
|
// import {unstable_setRequestLocale} from "next-intl/server";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import {fetchBlogPosts} from "../../../../../lib/contentful/blogPosts";
|
import {fetchBlogPosts} from "../../../../../lib/contentful/blogPosts";
|
||||||
import {fetchBlogPostCategories} from "../../../../../lib/contentful/blogPostsCategories";
|
import {fetchBlogPostCategories} from "../../../../../lib/contentful/blogPostsCategories";
|
||||||
|
@ -20,9 +20,9 @@ interface BlogPostPageProps {
|
||||||
params: BlogPostPageParams
|
params: BlogPostPageParams
|
||||||
}
|
}
|
||||||
|
|
||||||
export default async function Blog({params, searchParams}: { params: BlogPostPageParams, searhParams?: {page: number} }) {
|
export default async function Blog({params, searchParams}: { params: BlogPostPageParams, searchParams?: {page: number} }) {
|
||||||
unstable_setRequestLocale(params.locale);
|
// unstable_setRequestLocale(params.locale);
|
||||||
const page = searchParams.page || undefined
|
const page = searchParams?.page || undefined
|
||||||
return (
|
return (
|
||||||
<BlogPosts basePath={'/'+params.locale+'/blog/'} locale={params.locale} currentCat={params.slug} page={page}/>
|
<BlogPosts basePath={'/'+params.locale+'/blog/'} locale={params.locale} currentCat={params.slug} page={page}/>
|
||||||
);
|
);
|
||||||
|
|
|
@ -2,7 +2,7 @@ import React from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import * as Util from "node:util";
|
import * as Util from "node:util";
|
||||||
import {fetchBlogPosts} from "../../../lib/contentful/blogPosts";
|
import {fetchBlogPosts} from "../../../lib/contentful/blogPosts";
|
||||||
import {unstable_setRequestLocale} from "next-intl/server";
|
// import {unstable_setRequestLocale} from "next-intl/server";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import {fetchBlogPostCategories} from "../../../lib/contentful/blogPostsCategories";
|
import {fetchBlogPostCategories} from "../../../lib/contentful/blogPostsCategories";
|
||||||
import {CustomPagination} from "../../../components/view/CustomPagination";
|
import {CustomPagination} from "../../../components/view/CustomPagination";
|
||||||
|
@ -24,10 +24,10 @@ export async function generateStaticParams(): Promise<BlogPostPageParams[]> {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
export default async function Blog({ params: { locale }, searchParams }: { params: { locale: string }, searhParams?: {page: number} }) {
|
export default async function Blog({ params: { locale }, searchParams }: { params: { locale: string }, searchParams?: {page: number} }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const pageSize = DEFAULT_PAGE_SIZE
|
const pageSize = DEFAULT_PAGE_SIZE
|
||||||
const page = searchParams.page || undefined
|
const page = searchParams?.page || undefined
|
||||||
// BlogPosts('/'+locale+'/blog/', locale, pageSize)
|
// BlogPosts('/'+locale+'/blog/', locale, pageSize)
|
||||||
return (
|
return (
|
||||||
|
|
||||||
|
@ -36,7 +36,6 @@ export default async function Blog({ params: { locale }, searchParams }: { param
|
||||||
locale={locale}
|
locale={locale}
|
||||||
pageSize={pageSize}
|
pageSize={pageSize}
|
||||||
page={page}
|
page={page}
|
||||||
>
|
/>
|
||||||
</BlogPosts>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import React, { Suspense } from 'react';
|
import React, { Suspense } from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { notFound } from 'next/navigation';
|
import { notFound } from 'next/navigation';
|
||||||
import { getExpertById, getExpertsList } from '../../../../actions/experts';
|
import { getExpertById, getExpertsList } from '../../../../actions/experts';
|
||||||
import {
|
import {
|
||||||
|
@ -20,7 +20,7 @@ export const metadata: Metadata = {
|
||||||
export async function generateStaticParams({
|
export async function generateStaticParams({
|
||||||
params: { locale },
|
params: { locale },
|
||||||
}: { params: { locale: string } }) {
|
}: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const result: { locale: string, expertId: string }[] = [];
|
const result: { locale: string, expertId: string }[] = [];
|
||||||
const experts = await getExpertsList(locale, { themesTagIds: [] });
|
const experts = await getExpertsList(locale, { themesTagIds: [] });
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import type { Metadata } from 'next';
|
import type { Metadata } from 'next';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
// import { unstable_setRequestLocale } from 'next-intl/server';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { Experts } from '../../../components/Experts/Experts';
|
import { Experts } from '../../../components/Experts/Experts';
|
||||||
|
|
||||||
|
@ -10,7 +10,7 @@ export const metadata: Metadata = {
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function ExpertsPage({ params: { locale } }: { params: { locale: string } }) {
|
export default function ExpertsPage({ params: { locale } }: { params: { locale: string } }) {
|
||||||
unstable_setRequestLocale(locale);
|
// unstable_setRequestLocale(locale);
|
||||||
const t = useTranslations('Experts');
|
const t = useTranslations('Experts');
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -1,12 +1,15 @@
|
||||||
import React, { ReactNode, Suspense } from 'react';
|
import React, { ReactNode, Suspense } from 'react';
|
||||||
import { Metadata } from 'next';
|
import { Metadata } from 'next';
|
||||||
import { unstable_setRequestLocale } from 'next-intl/server';
|
import { NextIntlClientProvider } from 'next-intl';
|
||||||
|
import { getMessages } from 'next-intl/server';
|
||||||
import { notFound } from 'next/navigation';
|
import { notFound } from 'next/navigation';
|
||||||
import { ConfigProvider } from 'antd';
|
import { ConfigProvider } from 'antd';
|
||||||
import { AntdRegistry } from '@ant-design/nextjs-registry';
|
import { AntdRegistry } from '@ant-design/nextjs-registry';
|
||||||
|
import { GoogleOAuthProvider } from '@react-oauth/google';
|
||||||
import theme from '../../constants/theme';
|
import theme from '../../constants/theme';
|
||||||
import { ALLOWED_LOCALES } from '../../constants/locale';
|
import { ALLOWED_LOCALES } from '../../constants/locale';
|
||||||
import { Header, Footer, AppConfig } from '../../components/Page';
|
import { Header, Footer, AppConfig } from '../../components/Page';
|
||||||
|
import { routing } from '../../i18n/routing';
|
||||||
|
|
||||||
type LayoutProps = {
|
type LayoutProps = {
|
||||||
children: ReactNode;
|
children: ReactNode;
|
||||||
|
@ -21,25 +24,31 @@ export const metadata: Metadata = {
|
||||||
title: 'Bbuddy'
|
title: 'Bbuddy'
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function LocaleLayout({ children, params: { locale } }: LayoutProps) {
|
export default async function LocaleLayout({ children, params: { locale } }: LayoutProps) {
|
||||||
if (!ALLOWED_LOCALES.includes(locale as any)) notFound();
|
if (!routing.locales.includes(locale as any)) {
|
||||||
|
notFound();
|
||||||
|
}
|
||||||
|
|
||||||
unstable_setRequestLocale(locale);
|
const messages = await getMessages();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AntdRegistry>
|
<NextIntlClientProvider messages={messages}>
|
||||||
<ConfigProvider theme={theme}>
|
<AntdRegistry>
|
||||||
<div className="b-wrapper">
|
<GoogleOAuthProvider clientId={process.env.NEXT_PUBLIC_GOOGLE_CLIENT_ID || ''}>
|
||||||
<Suspense fallback={null}>
|
<ConfigProvider theme={theme}>
|
||||||
<AppConfig />
|
<div className="b-wrapper">
|
||||||
</Suspense>
|
<Suspense fallback={null}>
|
||||||
<div className="b-content">
|
<AppConfig />
|
||||||
<Header locale={locale} />
|
</Suspense>
|
||||||
{children}
|
<div className="b-content">
|
||||||
</div>
|
<Header locale={locale} />
|
||||||
<Footer locale={locale} />
|
{children}
|
||||||
</div>
|
</div>
|
||||||
</ConfigProvider>
|
<Footer locale={locale} />
|
||||||
</AntdRegistry>
|
</div>
|
||||||
|
</ConfigProvider>
|
||||||
|
</GoogleOAuthProvider>
|
||||||
|
</AntdRegistry>
|
||||||
|
</NextIntlClientProvider>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,50 @@
|
||||||
|
'use client'
|
||||||
|
|
||||||
|
import React, { useEffect } from 'react';
|
||||||
|
import { notification } from 'antd';
|
||||||
|
import { useSearchParams } from 'next/navigation';
|
||||||
|
import { CustomSpin } from '../../../../components/view/CustomSpin';
|
||||||
|
import { getLoginByApple } from '../../../../actions/auth';
|
||||||
|
import { getUserData } from '../../../../actions/profile';
|
||||||
|
import { AUTH_TOKEN_KEY, AUTH_USER } from '../../../../constants/common';
|
||||||
|
import { useLocalStorage } from '../../../../hooks/useLocalStorage';
|
||||||
|
import { useRouter } from '../../../../i18n/routing';
|
||||||
|
|
||||||
|
export default function AppleLoginPage({ params: { locale } }: { params: { locale: string } }) {
|
||||||
|
const params = useSearchParams();
|
||||||
|
const router = useRouter();
|
||||||
|
const [, setToken] = useLocalStorage(AUTH_TOKEN_KEY, '');
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const code = params.get('code');
|
||||||
|
if (code) {
|
||||||
|
getLoginByApple(locale, code)
|
||||||
|
.then((data) => {
|
||||||
|
if (data.jwtToken) {
|
||||||
|
getUserData(locale, data.jwtToken)
|
||||||
|
.then((profile) => {
|
||||||
|
localStorage.setItem(AUTH_USER, JSON.stringify(profile));
|
||||||
|
setToken(data.jwtToken);
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
notification.error({
|
||||||
|
message: 'Error',
|
||||||
|
description: 'Access denied'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
const err = error?.message ? JSON.parse(error.message) : {};
|
||||||
|
notification.error({
|
||||||
|
message: 'Error',
|
||||||
|
description: err?.details?.errMessage || undefined
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
router.push('/');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [params]);
|
||||||
|
|
||||||
|
return <CustomSpin />;
|
||||||
|
}
|
|
@ -0,0 +1,45 @@
|
||||||
|
'use client'
|
||||||
|
|
||||||
|
import React, { useEffect } from 'react';
|
||||||
|
import { notification } from 'antd';
|
||||||
|
import { useSearchParams } from 'next/navigation';
|
||||||
|
import { CustomSpin } from '../../../../components/view/CustomSpin';
|
||||||
|
import { getRegisterByApple } from '../../../../actions/auth';
|
||||||
|
import { getUserData } from '../../../../actions/profile';
|
||||||
|
import { AUTH_TOKEN_KEY, AUTH_USER } from '../../../../constants/common';
|
||||||
|
import { useLocalStorage } from "../../../../hooks/useLocalStorage";
|
||||||
|
import { useRouter } from '../../../../i18n/routing';
|
||||||
|
|
||||||
|
export default function AppleRegisterPage({ params: { locale } }: { params: { locale: string } }) {
|
||||||
|
const params = useSearchParams();
|
||||||
|
const router = useRouter();
|
||||||
|
const [, setToken] = useLocalStorage(AUTH_TOKEN_KEY, '');
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const code = params.get('code');
|
||||||
|
if (code) {
|
||||||
|
getRegisterByApple(locale, code)
|
||||||
|
.then((data) => {
|
||||||
|
if (data.jwtToken) {
|
||||||
|
getUserData(locale, data.jwtToken)
|
||||||
|
.then((profile) => {
|
||||||
|
localStorage.setItem(AUTH_USER, JSON.stringify(profile));
|
||||||
|
setToken(data.jwtToken);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
const err = error?.message ? JSON.parse(error.message) : {};
|
||||||
|
notification.error({
|
||||||
|
message: 'Error',
|
||||||
|
description: err?.details?.errMessage || undefined
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
router.push('/');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [params]);
|
||||||
|
|
||||||
|
return <CustomSpin />;
|
||||||
|
}
|
|
@ -10,7 +10,10 @@ type RootLayoutProps = {
|
||||||
|
|
||||||
export const metadata: Metadata = {
|
export const metadata: Metadata = {
|
||||||
title: 'Bbuddy',
|
title: 'Bbuddy',
|
||||||
description: 'Bbuddy'
|
description: 'Bbuddy',
|
||||||
|
verification: {
|
||||||
|
google: 'UqmM7WbpuMetvvkMeyVRGKiSvXHEGyaaRuYbWxM-njs'
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function RootLayout({ children, params: { locale } }: RootLayoutProps) {
|
export default function RootLayout({ children, params: { locale } }: RootLayoutProps) {
|
||||||
|
|
|
@ -1,27 +0,0 @@
|
||||||
import { fetchBlogPosts } from '../lib/contentful/blogPosts';
|
|
||||||
|
|
||||||
export default async function sitemap() {
|
|
||||||
const paths = [
|
|
||||||
{
|
|
||||||
url: process.env.NEXT_PUBLIC_HOST,
|
|
||||||
lastModified: new Date(),
|
|
||||||
changeFrequency: "monthly",
|
|
||||||
priority: 1
|
|
||||||
}
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
const blogPosts = await fetchBlogPosts({ preview: false })
|
|
||||||
|
|
||||||
blogPosts.data.forEach((item) => {
|
|
||||||
|
|
||||||
paths.push({
|
|
||||||
url: `${process.env.NEXT_PUBLIC_HOST}${item.slug}`,
|
|
||||||
lastModified: item.createdAt.split('T')[0],
|
|
||||||
changeFrequency: 'daily',
|
|
||||||
priority: '1.0'
|
|
||||||
})
|
|
||||||
})
|
|
||||||
|
|
||||||
return paths
|
|
||||||
}
|
|
|
@ -1,24 +1,47 @@
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
import { Button } from 'antd';
|
import { Button } from 'antd';
|
||||||
import { useSelectedLayoutSegment, usePathname } from 'next/navigation';
|
import { useSelectedLayoutSegment, usePathname } from 'next/navigation';
|
||||||
import { Link } from '../../navigation';
|
import { Link } from '../../i18n/routing';
|
||||||
import { AUTH_TOKEN_KEY, AUTH_USER } from '../../constants/common';
|
import { AUTH_TOKEN_KEY, AUTH_USER } from '../../constants/common';
|
||||||
import { deleteStorageKey } from '../../hooks/useLocalStorage';
|
import {deleteStorageKey, useLocalStorage} from '../../hooks/useLocalStorage';
|
||||||
import { i18nText } from '../../i18nKeys';
|
import { i18nText } from '../../i18nKeys';
|
||||||
import { getMenuConfig } from '../../utils/account';
|
import { getMenuConfig } from '../../utils/account';
|
||||||
|
import { getChatList } from '../../actions/chat/groups';
|
||||||
|
|
||||||
export const AccountMenu = ({ locale }: { locale: string }) => {
|
export const AccountMenu = ({ locale }: { locale: string }) => {
|
||||||
const selectedLayoutSegment = useSelectedLayoutSegment();
|
const selectedLayoutSegment = useSelectedLayoutSegment();
|
||||||
const pathname = selectedLayoutSegment || '';
|
const pathname = selectedLayoutSegment || '';
|
||||||
const paths = usePathname();
|
const paths = usePathname();
|
||||||
const menu: { path: string, title: string, count?: number }[] = getMenuConfig(locale);
|
const menu: { path: string, title: string, count?: number }[] = getMenuConfig(locale);
|
||||||
|
const [counts, setCounts] = useState<any>({});
|
||||||
|
|
||||||
const onLogout = () => {
|
const onLogout = () => {
|
||||||
deleteStorageKey(AUTH_TOKEN_KEY);
|
deleteStorageKey(AUTH_TOKEN_KEY);
|
||||||
deleteStorageKey(AUTH_USER);
|
deleteStorageKey(AUTH_USER);
|
||||||
window?.location?.replace(`/${paths.split('/')[1]}/`);
|
window?.location?.replace(`/${paths.split('/')[1]}/`);
|
||||||
};
|
};
|
||||||
|
const [jwt] = useLocalStorage(AUTH_TOKEN_KEY, '');
|
||||||
|
let init = false
|
||||||
|
useEffect(() => {
|
||||||
|
if (jwt && locale && !init) {
|
||||||
|
init = true;
|
||||||
|
getChatList(locale, jwt).then((payload: any)=> {
|
||||||
|
if (payload?.directs) {
|
||||||
|
let summ = 0;
|
||||||
|
payload?.directs.forEach((el: any) => {
|
||||||
|
summ = summ + el.newMessagesCount
|
||||||
|
})
|
||||||
|
setCounts({'messages': summ})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}, [jwt, locale])
|
||||||
|
|
||||||
|
const getterCount = (path: string, count: number)=> {
|
||||||
|
return counts[path]? counts[path] : count
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ul className="list-sidebar">
|
<ul className="list-sidebar">
|
||||||
|
@ -26,8 +49,8 @@ export const AccountMenu = ({ locale }: { locale: string }) => {
|
||||||
<li key={path} className="list-sidebar__item">
|
<li key={path} className="list-sidebar__item">
|
||||||
<Link href={`/account/${path}` as any} className={path === pathname ? 'active' : ''}>
|
<Link href={`/account/${path}` as any} className={path === pathname ? 'active' : ''}>
|
||||||
{title}
|
{title}
|
||||||
{count ? (
|
{getterCount(path, count) ? (
|
||||||
<span className="count">{count}</span>
|
<span className="count">{getterCount(path, count)}</span>
|
||||||
) : null}
|
) : null}
|
||||||
</Link>
|
</Link>
|
||||||
</li>
|
</li>
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { Form, message, Upload } from 'antd';
|
||||||
import type { UploadFile } from 'antd';
|
import type { UploadFile } from 'antd';
|
||||||
import ImgCrop from 'antd-img-crop';
|
import ImgCrop from 'antd-img-crop';
|
||||||
import { CameraOutlined, DeleteOutlined } from '@ant-design/icons';
|
import { CameraOutlined, DeleteOutlined } from '@ant-design/icons';
|
||||||
import { useRouter } from '../../navigation';
|
import { useRouter } from '../../i18n/routing';
|
||||||
import { i18nText } from '../../i18nKeys';
|
import { i18nText } from '../../i18nKeys';
|
||||||
import { ProfileRequest } from '../../types/profile';
|
import { ProfileRequest } from '../../types/profile';
|
||||||
import { validateImage } from '../../utils/account';
|
import { validateImage } from '../../utils/account';
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { EditRoomForm } from './EditRoomForm';
|
||||||
import debounce from 'lodash/debounce';
|
import debounce from 'lodash/debounce';
|
||||||
import { createRoom } from '../../../actions/rooms';
|
import { createRoom } from '../../../actions/rooms';
|
||||||
import { Loader } from '../../view/Loader';
|
import { Loader } from '../../view/Loader';
|
||||||
import { useRouter } from '../../../navigation';
|
import { useRouter } from '../../../i18n/routing';
|
||||||
import { RoomsType } from '../../../types/rooms';
|
import { RoomsType } from '../../../types/rooms';
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { Button, notification, Tag } from 'antd';
|
||||||
import { DeleteOutlined, LeftOutlined } from '@ant-design/icons';
|
import { DeleteOutlined, LeftOutlined } from '@ant-design/icons';
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
import Image from 'next/image';
|
import Image from 'next/image';
|
||||||
import { useRouter } from '../../../navigation';
|
import { useRouter } from '../../../i18n/routing';
|
||||||
import { Report, Room, RoomsType } from '../../../types/rooms';
|
import { Report, Room, RoomsType } from '../../../types/rooms';
|
||||||
import { i18nText } from '../../../i18nKeys';
|
import { i18nText } from '../../../i18nKeys';
|
||||||
import { LinkButton } from '../../view/LinkButton';
|
import { LinkButton } from '../../view/LinkButton';
|
||||||
|
|
|
@ -14,7 +14,7 @@ import { getRecentRooms, getUpcomingRooms } from '../../../actions/rooms';
|
||||||
import { Loader } from '../../view/Loader';
|
import { Loader } from '../../view/Loader';
|
||||||
import { useLocalStorage } from '../../../hooks/useLocalStorage';
|
import { useLocalStorage } from '../../../hooks/useLocalStorage';
|
||||||
import { AUTH_TOKEN_KEY } from '../../../constants/common';
|
import { AUTH_TOKEN_KEY } from '../../../constants/common';
|
||||||
import { usePathname, useRouter } from '../../../navigation';
|
import { usePathname, useRouter } from '../../../i18n/routing';
|
||||||
import { i18nText } from '../../../i18nKeys';
|
import { i18nText } from '../../../i18nKeys';
|
||||||
import { CreateRoom } from './CreateRoom';
|
import { CreateRoom } from './CreateRoom';
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { Button, Empty, notification, Tag } from 'antd';
|
||||||
import { LeftOutlined, PlusOutlined, RightOutlined } from '@ant-design/icons';
|
import { LeftOutlined, PlusOutlined, RightOutlined } from '@ant-design/icons';
|
||||||
import Image from 'next/image';
|
import Image from 'next/image';
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
import { Link, useRouter } from '../../../navigation';
|
import { Link, useRouter } from '../../../i18n/routing';
|
||||||
import { i18nText } from '../../../i18nKeys';
|
import { i18nText } from '../../../i18nKeys';
|
||||||
import { getDuration, getPrice } from '../../../utils/expert';
|
import { getDuration, getPrice } from '../../../utils/expert';
|
||||||
import { PublicUser, Session, SessionState, SessionType } from '../../../types/sessions';
|
import { PublicUser, Session, SessionState, SessionType } from '../../../types/sessions';
|
||||||
|
|
|
@ -14,7 +14,7 @@ import { useLocalStorage } from '../../../hooks/useLocalStorage';
|
||||||
import { AUTH_TOKEN_KEY, AUTH_USER } from '../../../constants/common';
|
import { AUTH_TOKEN_KEY, AUTH_USER } from '../../../constants/common';
|
||||||
import { getRecentSessions, getRequestedSessions, getUpcomingSessions } from '../../../actions/sessions';
|
import { getRecentSessions, getRequestedSessions, getUpcomingSessions } from '../../../actions/sessions';
|
||||||
import { Session, Sessions, SessionType } from '../../../types/sessions';
|
import { Session, Sessions, SessionType } from '../../../types/sessions';
|
||||||
import { useRouter, usePathname } from '../../../navigation';
|
import { useRouter, usePathname } from '../../../i18n/routing';
|
||||||
import { i18nText } from '../../../i18nKeys';
|
import { i18nText } from '../../../i18nKeys';
|
||||||
|
|
||||||
type SessionsTabsProps = {
|
type SessionsTabsProps = {
|
||||||
|
|
|
@ -10,7 +10,7 @@ type PostsProps = {
|
||||||
basePath: string;
|
basePath: string;
|
||||||
locale: string;
|
locale: string;
|
||||||
pageSize?: number;
|
pageSize?: number;
|
||||||
currentCat: string;
|
currentCat?: string;
|
||||||
page?: number
|
page?: number
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,76 @@
|
||||||
|
'use client'
|
||||||
|
import React, {useEffect, useState} from 'react';
|
||||||
|
import {AUTH_TOKEN_KEY} from '../../constants/common';
|
||||||
|
import {getChatList, getChatMessages} from "../../actions/chat/groups";
|
||||||
|
import {useLocalStorage} from "../../hooks/useLocalStorage";
|
||||||
|
import { Link } from "../../i18n/routing";
|
||||||
|
import dayjs from "dayjs";
|
||||||
|
import relativeTime from "dayjs/plugin/relativeTime";
|
||||||
|
import {message} from "antd";
|
||||||
|
import {Loader} from "../view/Loader";
|
||||||
|
|
||||||
|
dayjs.extend(relativeTime);
|
||||||
|
|
||||||
|
type CompProps = {
|
||||||
|
locale: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ChatList = ({ locale }: CompProps) => {
|
||||||
|
const [jwt] = useLocalStorage(AUTH_TOKEN_KEY, '');
|
||||||
|
const [chats, setСhats] = useState<any | undefined>();
|
||||||
|
const [loading, setLoading] = useState<boolean>(false);
|
||||||
|
useEffect(() => {
|
||||||
|
if (jwt) {
|
||||||
|
setLoading(true);
|
||||||
|
Promise.all([
|
||||||
|
getChatList(locale, jwt),
|
||||||
|
])
|
||||||
|
.then(([_groups]) => {
|
||||||
|
setСhats(_groups)
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
console.log(e)
|
||||||
|
message.error('Не удалось загрузить данные');
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
setLoading(false);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},[jwt])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="messages-session">
|
||||||
|
<Loader isLoading={loading}>
|
||||||
|
{chats?.directs.map((item: any, i: number) => (
|
||||||
|
<Link
|
||||||
|
key={'chat'+i}
|
||||||
|
className="card-profile"
|
||||||
|
href={'messages/'+item.group.id as any}
|
||||||
|
>
|
||||||
|
<div className="card-profile__header">
|
||||||
|
<div className="card-profile__header__portrait">
|
||||||
|
<img src={item.faceImageUrl} className="" alt="" />
|
||||||
|
</div>
|
||||||
|
<div className="card-profile__header__inner">
|
||||||
|
<div style={{ width: '100%' }}>
|
||||||
|
<div className="card-profile__header__name">
|
||||||
|
{item.firstName}
|
||||||
|
{item.newMessagesCount && (
|
||||||
|
<span className="count">{item.newMessagesCount}</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="card-profile__header__title">
|
||||||
|
{item?.lastMessage?.text}
|
||||||
|
</div>
|
||||||
|
<div className="card-profile__header__date ">
|
||||||
|
{dayjs(item?.lastMessage?.sentAt).fromNow()}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Link>
|
||||||
|
))}
|
||||||
|
</Loader>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
|
@ -0,0 +1,203 @@
|
||||||
|
'use client'
|
||||||
|
import React, {useEffect, useState} from 'react';
|
||||||
|
import {AUTH_TOKEN_KEY} from '../../constants/common';
|
||||||
|
import {getChatList, getChatMessages} from "../../actions/chat/groups";
|
||||||
|
import {useLocalStorage} from "../../hooks/useLocalStorage";
|
||||||
|
import dayjs from "dayjs";
|
||||||
|
import relativeTime from "dayjs/plugin/relativeTime";
|
||||||
|
|
||||||
|
import {message} from "antd";
|
||||||
|
import {Loader} from "../view/Loader";
|
||||||
|
import SignalrConnection from "../../lib/signalr-connection";
|
||||||
|
import {CheckOutlined} from "@ant-design/icons";
|
||||||
|
|
||||||
|
dayjs.extend(relativeTime);
|
||||||
|
|
||||||
|
type CompProps = {
|
||||||
|
locale: string;
|
||||||
|
groupId: number
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ChatMessages = ({ locale, groupId }: CompProps) => {
|
||||||
|
const [jwt] = useLocalStorage(AUTH_TOKEN_KEY, '');
|
||||||
|
const { newMessage, joinChat, readMessages, addListener } = SignalrConnection({ jwt }) || {};
|
||||||
|
//const messages = await getChatMessages(locale, jwt, groupId)
|
||||||
|
const [loading, setLoading] = useState<boolean>(false);
|
||||||
|
const [text, setText] = useState('');
|
||||||
|
const [me, setMe] = useState<any | undefined>();
|
||||||
|
const [notMe, setNotMe] = useState<any | undefined>();
|
||||||
|
const [messages, setMessages] = useState<any[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (jwt) {
|
||||||
|
setLoading(true);
|
||||||
|
Promise.all([
|
||||||
|
getChatList(locale, jwt),
|
||||||
|
getChatMessages(locale, jwt, groupId),
|
||||||
|
])
|
||||||
|
.then(([_groups, _messages]) => {
|
||||||
|
//
|
||||||
|
const _group = _groups.directs.find( (el: any) => el.group.id === groupId)
|
||||||
|
if (_group.group.members[0].userId === parseInt(_group.userId)){
|
||||||
|
setMe(_group.group.members[0].user)
|
||||||
|
setNotMe(_group.group.members[1].user)
|
||||||
|
} else {
|
||||||
|
setMe(_group.group.members[1].user)
|
||||||
|
setNotMe(_group.group.members[0].user)
|
||||||
|
}
|
||||||
|
setMessages(_messages.messages);
|
||||||
|
joinChat(groupId)
|
||||||
|
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
console.log(e)
|
||||||
|
message.error('Не удалось загрузить данные');
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
setLoading(false);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},[jwt])
|
||||||
|
|
||||||
|
|
||||||
|
const onConnected = (flag: boolean) =>{
|
||||||
|
if (flag && joinChat) {
|
||||||
|
joinChat(groupId)
|
||||||
|
readUreaded()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const readUreaded = () => {
|
||||||
|
const msgs = [] as number[];
|
||||||
|
messages.forEach((message: any) => {
|
||||||
|
if (!message.seen && message.sentByMe === false){
|
||||||
|
msgs.push(message.id)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
readMessages && readMessages(msgs);
|
||||||
|
}
|
||||||
|
|
||||||
|
const onReceiveMessage = (payload: any) => {
|
||||||
|
const _messages = [... messages];
|
||||||
|
let flag = false;
|
||||||
|
const msg = {
|
||||||
|
data : payload.data,
|
||||||
|
dataType: null,
|
||||||
|
id: payload.id,
|
||||||
|
receiverId:null,
|
||||||
|
seen: false,
|
||||||
|
senderId: payload.creatorId,
|
||||||
|
sentAt: payload.createdUtc+'.000Z',
|
||||||
|
sentByMe: payload.creatorId === me.id,
|
||||||
|
text: payload.content,
|
||||||
|
type:"text"
|
||||||
|
}
|
||||||
|
_messages.forEach((item: any, i) => {
|
||||||
|
if (item.id === msg.id){
|
||||||
|
_messages[i] = msg
|
||||||
|
flag = true
|
||||||
|
}
|
||||||
|
})
|
||||||
|
//console.log(payload, flag)
|
||||||
|
if (flag){
|
||||||
|
setMessages([..._messages]);
|
||||||
|
} else {
|
||||||
|
setMessages([msg, ..._messages]);
|
||||||
|
}
|
||||||
|
readMessages && readMessages([msg.id]);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
const onOpponentRead = (payload: any) => {
|
||||||
|
console.log('onOpponentRead', payload)
|
||||||
|
const _messages = [... messages] as any[];
|
||||||
|
_messages.forEach((item: any, i) => {
|
||||||
|
if (item.id === payload.messageId){
|
||||||
|
_messages[i].seen = true
|
||||||
|
|
||||||
|
}
|
||||||
|
})
|
||||||
|
setMessages([..._messages])
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (addListener) {
|
||||||
|
addListener('onConnected', onConnected);
|
||||||
|
addListener('ReceiveMessage', onReceiveMessage);
|
||||||
|
addListener('MessageWasRead', onOpponentRead);
|
||||||
|
}
|
||||||
|
}, [messages, me, setMessages]);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
const handleSendMessages = () => {
|
||||||
|
newMessage && newMessage({
|
||||||
|
GroupId: groupId.toString(),
|
||||||
|
CreatorId: me.id,
|
||||||
|
Content: text
|
||||||
|
})
|
||||||
|
setText('')
|
||||||
|
}
|
||||||
|
|
||||||
|
const onEnterPress = (e: any) => {
|
||||||
|
if(e.keyCode == 13 && e.shiftKey == false) {
|
||||||
|
e.preventDefault();
|
||||||
|
handleSendMessages();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleChangeMessage = (e: any) =>{
|
||||||
|
const { value } = e.target;
|
||||||
|
e.preventDefault();
|
||||||
|
setText(value);
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const messageRender = (message: any, i:number) => {
|
||||||
|
const item = message.sentByMe ? me : notMe
|
||||||
|
const date = dayjs(message.sentAt).fromNow()
|
||||||
|
const imgSrc = item?.faceImage?.descriptor ? 'http://static.bbuddy.expert/' + item.faceImage.descriptor.split(':')[1] : ''
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={message.sentByMe ? 'b-message__list b-message__list--me' : 'b-message__list'}
|
||||||
|
key={'message'+i}
|
||||||
|
>
|
||||||
|
<div className="b-message__item ">
|
||||||
|
<div className="b-message__avatar">
|
||||||
|
{imgSrc && (<img src={imgSrc} className="" alt=""/>)}
|
||||||
|
</div>
|
||||||
|
<div className="b-message__text" style={{minWidth: '150px'}}>
|
||||||
|
{message.text}
|
||||||
|
<span className="date">
|
||||||
|
<span className="checks" style={{color: message.seen ? 'green' : 'blue'}}>
|
||||||
|
<CheckOutlined />
|
||||||
|
{ message?.seen && (<CheckOutlined style={{marginLeft: '-10px'}} />)}
|
||||||
|
</span>
|
||||||
|
{date}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return (
|
||||||
|
|
||||||
|
<div className="b-message">
|
||||||
|
<Loader isLoading={loading}>
|
||||||
|
<div className="b-message__inner">
|
||||||
|
{messages.map((el, i)=> (messageRender(el, i)))}
|
||||||
|
</div>
|
||||||
|
</Loader>
|
||||||
|
<div className="b-message__form">
|
||||||
|
<textarea placeholder="Type your message here" onKeyDown={onEnterPress}
|
||||||
|
onChange={handleChangeMessage} value={text}/>
|
||||||
|
<button className="b-message__btn" type="submit" onClick={handleSendMessages}/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
)
|
||||||
|
}
|
|
@ -3,7 +3,7 @@
|
||||||
import React, { useCallback, useEffect, useState } from 'react';
|
import React, { useCallback, useEffect, useState } from 'react';
|
||||||
import { useSearchParams } from 'next/navigation';
|
import { useSearchParams } from 'next/navigation';
|
||||||
import debounce from 'lodash/debounce';
|
import debounce from 'lodash/debounce';
|
||||||
import { useRouter } from '../../navigation';
|
import { useRouter } from '../../i18n/routing';
|
||||||
import { AdditionalFilter } from '../../types/experts';
|
import { AdditionalFilter } from '../../types/experts';
|
||||||
import { getObjectByFilter, getObjectByAdditionalFilter, getSearchParamsString } from '../../utils/filter';
|
import { getObjectByFilter, getObjectByAdditionalFilter, getSearchParamsString } from '../../utils/filter';
|
||||||
import { CustomInput } from '../view/CustomInput';
|
import { CustomInput } from '../view/CustomInput';
|
||||||
|
|
|
@ -15,6 +15,9 @@ import { getStorageValue } from '../../hooks/useLocalStorage';
|
||||||
import { AUTH_TOKEN_KEY, SESSION_DATA } from '../../constants/common';
|
import { AUTH_TOKEN_KEY, SESSION_DATA } from '../../constants/common';
|
||||||
import { ScheduleModal } from '../Modals/ScheduleModal';
|
import { ScheduleModal } from '../Modals/ScheduleModal';
|
||||||
import { ScheduleModalResult } from '../Modals/ScheduleModalResult';
|
import { ScheduleModalResult } from '../Modals/ScheduleModalResult';
|
||||||
|
import SignalrConnection from '../../lib/signalr-connection';
|
||||||
|
import { useRouter } from '../../i18n/routing';
|
||||||
|
import { useLocalStorage } from '../../hooks/useLocalStorage';
|
||||||
|
|
||||||
type ExpertDetailsProps = {
|
type ExpertDetailsProps = {
|
||||||
expert: ExpertDetails;
|
expert: ExpertDetails;
|
||||||
|
@ -32,8 +35,29 @@ export const ExpertCard: FC<ExpertDetailsProps> = ({ expert, locale, expertId })
|
||||||
const { publicCoachDetails } = expert || {};
|
const { publicCoachDetails } = expert || {};
|
||||||
const [showSchedulerModal, setShowSchedulerModal] = useState<boolean>(false);
|
const [showSchedulerModal, setShowSchedulerModal] = useState<boolean>(false);
|
||||||
const [mode, setMode] = useState<'data' | 'time' | 'pay' | 'finish'>('data');
|
const [mode, setMode] = useState<'data' | 'time' | 'pay' | 'finish'>('data');
|
||||||
const { publicCoachDetails: { tags = [], sessionCost = 0, sessionDuration = 0, coachLanguages = [] } } = expert || {};
|
|
||||||
const isRus = locale === Locale.ru;
|
const isRus = locale === Locale.ru;
|
||||||
|
const { publicCoachDetails: { tags = [], sessionCost = 0, sessionDuration = 0, coachLanguages = [] , id, botUserId} } = expert || {};
|
||||||
|
const [jwt] = useLocalStorage(AUTH_TOKEN_KEY, '');
|
||||||
|
const { joinChatPerson, closeConnection } = SignalrConnection({ jwt }) || {};
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
document?.addEventListener('show_pay_form', handleShowPayForm);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
// if (closeConnection) closeConnection();
|
||||||
|
document?.removeEventListener('show_pay_form', handleShowPayForm);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleJoinChat = (id?: number) => {
|
||||||
|
if (id && joinChatPerson) {
|
||||||
|
joinChatPerson(id).then((res: any) => {
|
||||||
|
router.push(`/account/messages/${res.id}` as string);
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const checkSession = (data?: SignupSessionData) => {
|
const checkSession = (data?: SignupSessionData) => {
|
||||||
if (data?.startAtUtc && data?.tagId) {
|
if (data?.startAtUtc && data?.tagId) {
|
||||||
|
@ -44,7 +68,7 @@ export const ExpertCard: FC<ExpertDetailsProps> = ({ expert, locale, expertId })
|
||||||
} else {
|
} else {
|
||||||
setShowSchedulerModal(false);
|
setShowSchedulerModal(false);
|
||||||
const showAuth = new Event('show_auth_enter');
|
const showAuth = new Event('show_auth_enter');
|
||||||
document.dispatchEvent(showAuth);
|
document?.dispatchEvent(showAuth);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -54,13 +78,6 @@ export const ExpertCard: FC<ExpertDetailsProps> = ({ expert, locale, expertId })
|
||||||
setMode('pay');
|
setMode('pay');
|
||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
document.addEventListener('show_pay_form', handleShowPayForm);
|
|
||||||
return () => {
|
|
||||||
document.removeEventListener('show_pay_form', handleShowPayForm);
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const onSchedulerHandle = () => {
|
const onSchedulerHandle = () => {
|
||||||
setMode('data');
|
setMode('data');
|
||||||
setShowSchedulerModal(true);
|
setShowSchedulerModal(true);
|
||||||
|
@ -86,18 +103,26 @@ export const ExpertCard: FC<ExpertDetailsProps> = ({ expert, locale, expertId })
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="expert-card__wrap-btn">
|
{jwt && (
|
||||||
<Button className="btn-apply" onClick={onSchedulerHandle}>
|
<div className="expert-card__wrap-btn">
|
||||||
<img src="/images/calendar-outline.svg" className="" alt="" />
|
<Button className="btn-apply" onClick={() => handleJoinChat(id)}>
|
||||||
{i18nText('schedule', locale)}
|
{i18nText('chat.join', locale)}
|
||||||
</Button>
|
</Button>
|
||||||
{/*
|
<Button
|
||||||
<a href="#" className="btn-video">
|
className={`btn-apply${!botUserId ? ' btn-disabled' : ''}`}
|
||||||
<img src="/images/videocam-outline.svg" className="" alt=""/>
|
disabled={!botUserId}
|
||||||
Video
|
onClick={botUserId ? () => handleJoinChat(botUserId) : undefined}
|
||||||
</a>
|
>
|
||||||
*/}
|
{i18nText('chat.joinAI', locale)}
|
||||||
</div>
|
</Button>
|
||||||
|
{/*
|
||||||
|
<a href="#" className="btn-video">
|
||||||
|
<img src="/images/videocam-outline.svg" className="" alt=""/>
|
||||||
|
Video
|
||||||
|
</a>
|
||||||
|
*/}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div className="expert-info">
|
<div className="expert-info">
|
||||||
{/* <h2 className="title-h2">{}</h2> */}
|
{/* <h2 className="title-h2">{}</h2> */}
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { List, Tag } from 'antd';
|
||||||
import { RightOutlined } from '@ant-design/icons';
|
import { RightOutlined } from '@ant-design/icons';
|
||||||
import isEqual from 'lodash/isEqual';
|
import isEqual from 'lodash/isEqual';
|
||||||
import Image from 'next/image';
|
import Image from 'next/image';
|
||||||
import { Link, useRouter } from '../../navigation';
|
import { Link, useRouter } from '../../i18n/routing';
|
||||||
import { ExpertsData, Filter, GeneralFilter } from '../../types/experts';
|
import { ExpertsData, Filter, GeneralFilter } from '../../types/experts';
|
||||||
import { getObjectByFilter, getObjectByAdditionalFilter, getSearchParamsString } from '../../utils/filter';
|
import { getObjectByFilter, getObjectByAdditionalFilter, getSearchParamsString } from '../../utils/filter';
|
||||||
import { getDuration, getPrice } from '../../utils/expert';
|
import { getDuration, getPrice } from '../../utils/expert';
|
||||||
|
|
|
@ -4,7 +4,7 @@ import React, { useCallback, useEffect, useState } from 'react';
|
||||||
import { Button, Collapse, List } from 'antd';
|
import { Button, Collapse, List } from 'antd';
|
||||||
import type { CollapseProps } from 'antd';
|
import type { CollapseProps } from 'antd';
|
||||||
import { useSearchParams } from 'next/navigation';
|
import { useSearchParams } from 'next/navigation';
|
||||||
import { useRouter } from '../../navigation';
|
import { useRouter } from '../../i18n/routing';
|
||||||
import { Filter } from '../../types/experts';
|
import { Filter } from '../../types/experts';
|
||||||
import { Languages, SearchData, Tag } from '../../types/tags';
|
import { Languages, SearchData, Tag } from '../../types/tags';
|
||||||
import { getObjectByFilter, getObjectByAdditionalFilter, getSearchParamsString } from '../../utils/filter';
|
import { getObjectByFilter, getObjectByAdditionalFilter, getSearchParamsString } from '../../utils/filter';
|
||||||
|
|
|
@ -1,12 +1,16 @@
|
||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import React, { Dispatch, FC, SetStateAction, useEffect } from 'react';
|
import React, { Dispatch, FC, SetStateAction, useEffect } from 'react';
|
||||||
import { usePathname } from 'next/navigation';
|
import { usePathname, useSearchParams } from 'next/navigation';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import { Modal, Form } from 'antd';
|
import { Modal, Form, notification } from 'antd';
|
||||||
import { CloseOutlined } from '@ant-design/icons';
|
import { CloseOutlined } from '@ant-design/icons';
|
||||||
import { RegisterContent, ResetContent, FinishContent, EnterContent } from './authModalContent';
|
import { RegisterContent, ResetContent, FinishContent, EnterContent } from './authModalContent';
|
||||||
import { i18nText } from '../../i18nKeys';
|
import { i18nText } from '../../i18nKeys';
|
||||||
|
import { useRouter } from '../../i18n/routing';
|
||||||
|
import { AUTH_USER} from '../../constants/common';
|
||||||
|
import { getRegisterByApple, getLoginByApple } from '../../actions/auth';
|
||||||
|
import { getUserData } from '../../actions/profile';
|
||||||
|
|
||||||
type AuthModalProps = {
|
type AuthModalProps = {
|
||||||
open: boolean;
|
open: boolean;
|
||||||
|
@ -27,6 +31,47 @@ export const AuthModal: FC<AuthModalProps> = ({
|
||||||
}) => {
|
}) => {
|
||||||
const [form] = Form.useForm<{ login: string, password: string, confirmPassword: string }>();
|
const [form] = Form.useForm<{ login: string, password: string, confirmPassword: string }>();
|
||||||
const paths = usePathname().split('/');
|
const paths = usePathname().split('/');
|
||||||
|
const params = useSearchParams();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const onUpdateToken = (token: string) => {
|
||||||
|
if (updateToken && typeof updateToken !== 'string') {
|
||||||
|
updateToken(token);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const code = params.get('code');
|
||||||
|
const type = params.get('state');
|
||||||
|
if (code && type) {
|
||||||
|
const appleFunc = type === 'bbregister' ? getRegisterByApple : getLoginByApple;
|
||||||
|
appleFunc(locale, code)
|
||||||
|
.then((data) => {
|
||||||
|
if (data.jwtToken) {
|
||||||
|
getUserData(locale, data.jwtToken)
|
||||||
|
.then((profile) => {
|
||||||
|
localStorage.setItem(AUTH_USER, JSON.stringify(profile));
|
||||||
|
onUpdateToken(data.jwtToken);
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
notification.error({
|
||||||
|
message: 'Error',
|
||||||
|
description: 'Access denied'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
const err = error?.message ? JSON.parse(error.message) : {};
|
||||||
|
notification.error({
|
||||||
|
message: 'Error',
|
||||||
|
description: err?.details?.errMessage || undefined
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
router.push('/');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [params]);
|
||||||
|
|
||||||
const onAfterClose = () => {
|
const onAfterClose = () => {
|
||||||
form.resetFields();
|
form.resetFields();
|
||||||
|
@ -38,12 +83,6 @@ export const AuthModal: FC<AuthModalProps> = ({
|
||||||
}
|
}
|
||||||
}, [mode]);
|
}, [mode]);
|
||||||
|
|
||||||
const onUpdateToken = (token: string) => {
|
|
||||||
if (updateToken && typeof updateToken !== 'string') {
|
|
||||||
updateToken(token);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
className="b-modal"
|
className="b-modal"
|
||||||
|
|
|
@ -76,7 +76,6 @@ export const ScheduleModal: FC<ScheduleModalProps> = ({
|
||||||
getSchedulerSession(parseData as SignupSessionData, locale || 'en', jwt)
|
getSchedulerSession(parseData as SignupSessionData, locale || 'en', jwt)
|
||||||
.then((session) => {
|
.then((session) => {
|
||||||
setSessionId(session?.sessionId);
|
setSessionId(session?.sessionId);
|
||||||
console.log(session?.sessionId);
|
|
||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
console.log(err);
|
console.log(err);
|
||||||
|
|
|
@ -3,7 +3,8 @@
|
||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { Modal, Result } from 'antd';
|
import { Modal, Result } from 'antd';
|
||||||
import { CloseOutlined } from '@ant-design/icons';
|
import { CloseOutlined } from '@ant-design/icons';
|
||||||
import { useSearchParams, useRouter } from 'next/navigation';
|
import { useSearchParams } from 'next/navigation';
|
||||||
|
import { useRouter } from '../../i18n/routing';
|
||||||
import { Stripe } from 'stripe';
|
import { Stripe } from 'stripe';
|
||||||
import { getStripePaymentStatus } from '../../actions/stripe';
|
import { getStripePaymentStatus } from '../../actions/stripe';
|
||||||
import { sessionPaymentConfirm } from '../../actions/sessions';
|
import { sessionPaymentConfirm } from '../../actions/sessions';
|
||||||
|
|
|
@ -1,12 +1,11 @@
|
||||||
import React, { FC, useState } from 'react';
|
import React, { FC, useState } from 'react';
|
||||||
import { Form, FormInstance, notification } from 'antd';
|
import { Form, FormInstance, notification } from 'antd';
|
||||||
import Image from 'next/image';
|
import Image from 'next/image';
|
||||||
import { Social } from '../../../types/social';
|
import { useGoogleLogin } from '@react-oauth/google';
|
||||||
|
import AppleLogin from 'react-apple-login';
|
||||||
import { AUTH_USER } from '../../../constants/common';
|
import { AUTH_USER } from '../../../constants/common';
|
||||||
import { SocialConfig } from '../../../constants/social';
|
import { getAuth, getLoginByGoogle } from '../../../actions/auth';
|
||||||
import { useOauthWindow } from '../../../hooks/useOauthWindow';
|
import { getUserData } from '../../../actions/profile';
|
||||||
import { getAuth } from '../../../actions/auth';
|
|
||||||
import {getPersonalData, getUserData} from '../../../actions/profile';
|
|
||||||
import { CustomInput } from '../../view/CustomInput';
|
import { CustomInput } from '../../view/CustomInput';
|
||||||
import { CustomInputPassword } from '../../view/CustomInputPassword';
|
import { CustomInputPassword } from '../../view/CustomInputPassword';
|
||||||
import { FilledButton } from '../../view/FilledButton';
|
import { FilledButton } from '../../view/FilledButton';
|
||||||
|
@ -30,7 +29,6 @@ export const EnterContent: FC<EnterProps> = ({
|
||||||
handleCancel
|
handleCancel
|
||||||
}) => {
|
}) => {
|
||||||
const [isLoading, setIsLoading] = useState<boolean>(false);
|
const [isLoading, setIsLoading] = useState<boolean>(false);
|
||||||
const { openOauthWindow } = useOauthWindow();
|
|
||||||
|
|
||||||
const onLogin = () => {
|
const onLogin = () => {
|
||||||
form.validateFields().then(() => {
|
form.validateFields().then(() => {
|
||||||
|
@ -59,47 +57,44 @@ export const EnterContent: FC<EnterProps> = ({
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const onSocialEnter = (type: Social) => {
|
|
||||||
const url = SocialConfig[type].oauthUrl;
|
|
||||||
|
|
||||||
if (!url) return;
|
const onGoogleLogin = useGoogleLogin({
|
||||||
|
onError: (err) => {
|
||||||
openOauthWindow(url, type, async (event: MessageEvent) => {
|
notification.error({
|
||||||
const { data: socialData } = event
|
message: err.error,
|
||||||
|
description: err.error_description
|
||||||
// примерная схема последующей обработки
|
});
|
||||||
|
},
|
||||||
// const socialErrors: string[] = [];
|
onSuccess: (tokenResponse) => {
|
||||||
// try {
|
setIsLoading(true);
|
||||||
// // отправляем запрос на бэк с данными из соц сети
|
getLoginByGoogle(locale, tokenResponse.access_token)
|
||||||
// const { data: { jwtToken } } = await query(socialData);
|
.then((data) => {
|
||||||
// // обновляем токен
|
if (data.jwtToken) {
|
||||||
// updateToken(jwtToken);
|
getUserData(locale, data.jwtToken)
|
||||||
// // получаем данные о пользователе
|
.then((profile) => {
|
||||||
// await getAuthUser()
|
localStorage.setItem(AUTH_USER, JSON.stringify(profile));
|
||||||
// } catch (error: any) {
|
updateToken(data.jwtToken);
|
||||||
// if (error.httpStatus === 449) {
|
handleCancel();
|
||||||
// // ошибка, когда отсутствует e-mail
|
})
|
||||||
//
|
} else {
|
||||||
// // какие-то дальнейшие действия после получения ошибки, например, закрываем окно и открываем модалку регистрации
|
notification.error({
|
||||||
// handleCancel();
|
message: 'Error',
|
||||||
// openSocialEmailRequestModal(socialData);
|
description: 'Access denied'
|
||||||
// } else if (error.httpStatus === 409) {
|
});
|
||||||
// // ошибка, когда по переданному email уже существует аккаунт
|
}
|
||||||
//
|
})
|
||||||
// // какие-то дальнейшие действия после получения ошибки, например, закрываем окно и открываем модалку с вводом пароля
|
.catch((error) => {
|
||||||
// handleCancel();
|
const err = error?.message ? JSON.parse(error.message) : {};
|
||||||
// openSocialPasswordModal(socialData);
|
notification.error({
|
||||||
// } else {
|
message: 'Error',
|
||||||
// // в остальных случаях записываем ошибку в массив ошибок
|
description: err?.details?.errMessage || undefined
|
||||||
// socialErrors.push(error.toString());
|
});
|
||||||
// }
|
})
|
||||||
// }
|
.finally(() => {
|
||||||
//
|
setIsLoading(false);
|
||||||
// // если все успешно, закрываем окно
|
});
|
||||||
// handleCancel();
|
}
|
||||||
})
|
});
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
@ -155,21 +150,24 @@ export const EnterContent: FC<EnterProps> = ({
|
||||||
{`${i18nText('forgotPass', locale)}?`}
|
{`${i18nText('forgotPass', locale)}?`}
|
||||||
</LinkButton>
|
</LinkButton>
|
||||||
<span>{i18nText('or', locale)}</span>
|
<span>{i18nText('or', locale)}</span>
|
||||||
<OutlinedButton
|
<AppleLogin
|
||||||
icon={<Image src="/images/facebook-logo.png" height={20} width={20} alt="" />}
|
clientId="bbuddy.expert"
|
||||||
onClick={() => onSocialEnter(Social.FACEBOOK)}
|
redirectURI="https://bbuddy.expert"
|
||||||
>
|
state="bblogin"
|
||||||
{i18nText('facebook', locale)}
|
responseType="code"
|
||||||
</OutlinedButton>
|
responseMode="query"
|
||||||
<OutlinedButton
|
render={({ onClick }) => (
|
||||||
icon={<Image src="/images/apple-logo.png" height={22} width={22} alt="" />}
|
<OutlinedButton
|
||||||
onClick={() => onSocialEnter(Social.APPLE)}
|
icon={<Image src="/images/apple-logo.png" height={22} width={22} alt="" />}
|
||||||
>
|
onClick={onClick}
|
||||||
{i18nText('apple', locale)}
|
>
|
||||||
</OutlinedButton>
|
{i18nText('apple', locale)}
|
||||||
|
</OutlinedButton>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
<OutlinedButton
|
<OutlinedButton
|
||||||
icon={<Image src="/images/google-logo.png" height={20} width={20} alt="" />}
|
icon={<Image src="/images/google-logo.png" height={20} width={20} alt="" />}
|
||||||
onClick={() => onSocialEnter(Social.GOOGLE)}
|
onClick={onGoogleLogin}
|
||||||
>
|
>
|
||||||
{i18nText('google', locale)}
|
{i18nText('google', locale)}
|
||||||
</OutlinedButton>
|
</OutlinedButton>
|
||||||
|
|
|
@ -1,12 +1,11 @@
|
||||||
import React, { FC, useState } from 'react';
|
import React, { FC, useState } from 'react';
|
||||||
import { Form, FormInstance, notification } from 'antd';
|
import { Form, FormInstance, notification } from 'antd';
|
||||||
import Image from 'next/image';
|
import Image from 'next/image';
|
||||||
import { Social } from '../../../types/social';
|
import { useGoogleLogin } from '@react-oauth/google';
|
||||||
|
import AppleLogin from 'react-apple-login';
|
||||||
import { AUTH_USER } from '../../../constants/common';
|
import { AUTH_USER } from '../../../constants/common';
|
||||||
import { SocialConfig } from '../../../constants/social';
|
import { getRegister, getRegisterByGoogle } from '../../../actions/auth';
|
||||||
import { getRegister } from '../../../actions/auth';
|
import { getUserData, setPersonData } from '../../../actions/profile';
|
||||||
import { setPersonData } from '../../../actions/profile';
|
|
||||||
import { useOauthWindow } from '../../../hooks/useOauthWindow';
|
|
||||||
import { CustomInput } from '../../view/CustomInput';
|
import { CustomInput } from '../../view/CustomInput';
|
||||||
import { CustomInputPassword } from '../../view/CustomInputPassword';
|
import { CustomInputPassword } from '../../view/CustomInputPassword';
|
||||||
import { FilledButton } from '../../view/FilledButton';
|
import { FilledButton } from '../../view/FilledButton';
|
||||||
|
@ -29,7 +28,6 @@ export const RegisterContent: FC<RegisterProps> = ({
|
||||||
handleCancel
|
handleCancel
|
||||||
}) => {
|
}) => {
|
||||||
const [isLoading, setIsLoading] = useState<boolean>(false);
|
const [isLoading, setIsLoading] = useState<boolean>(false);
|
||||||
const { openOauthWindow } = useOauthWindow();
|
|
||||||
|
|
||||||
const onRegister = () => {
|
const onRegister = () => {
|
||||||
form.validateFields().then(() => {
|
form.validateFields().then(() => {
|
||||||
|
@ -64,47 +62,38 @@ export const RegisterContent: FC<RegisterProps> = ({
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const onSocialRegister = (type: Social) => {
|
const onGoogleLogin = useGoogleLogin({
|
||||||
const url = SocialConfig[type].oauthUrl;
|
onError: (err) => {
|
||||||
|
notification.error({
|
||||||
if (!url) return;
|
message: err.error,
|
||||||
|
description: err.error_description
|
||||||
openOauthWindow(url, type, async (event: MessageEvent) => {
|
});
|
||||||
const { data: socialData } = event
|
},
|
||||||
|
onSuccess: (tokenResponse) => {
|
||||||
// примерная схема последующей обработки
|
setIsLoading(true);
|
||||||
|
getRegisterByGoogle(locale, tokenResponse.access_token)
|
||||||
// const socialErrors: string[] = [];
|
.then((data) => {
|
||||||
// try {
|
if (data.jwtToken) {
|
||||||
// // отправляем запрос на бэк с данными из соц сети
|
getUserData(locale, data.jwtToken)
|
||||||
// const { data: { jwtToken } } = await query(socialData);
|
.then((profile) => {
|
||||||
// // обновляем токен
|
localStorage.setItem(AUTH_USER, JSON.stringify(profile));
|
||||||
// updateToken(jwtToken);
|
updateToken(data.jwtToken);
|
||||||
// // получаем данные о пользователе
|
handleCancel();
|
||||||
// await getAuthUser()
|
})
|
||||||
// } catch (error: any) {
|
}
|
||||||
// if (error.httpStatus === 449) {
|
})
|
||||||
// // ошибка, когда отсутствует e-mail
|
.catch((error) => {
|
||||||
//
|
const err = error?.message ? JSON.parse(error.message) : {};
|
||||||
// // какие-то дальнейшие действия после получения ошибки, например, закрываем окно и открываем модалку регистрации
|
notification.error({
|
||||||
// handleCancel();
|
message: 'Error',
|
||||||
// openSocialEmailRequestModal(socialData);
|
description: err?.details?.errMessage || undefined
|
||||||
// } else if (error.httpStatus === 409) {
|
});
|
||||||
// // ошибка, когда по переданному email уже существует аккаунт
|
})
|
||||||
//
|
.finally(() => {
|
||||||
// // какие-то дальнейшие действия после получения ошибки, например, закрываем окно и открываем модалку с вводом пароля
|
setIsLoading(false);
|
||||||
// handleCancel();
|
});
|
||||||
// openSocialPasswordModal(socialData);
|
}
|
||||||
// } else {
|
});
|
||||||
// // в остальных случаях записываем ошибку в массив ошибок
|
|
||||||
// socialErrors.push(error.toString());
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// // если все успешно, закрываем окно
|
|
||||||
// handleCancel();
|
|
||||||
})
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
@ -177,21 +166,24 @@ export const RegisterContent: FC<RegisterProps> = ({
|
||||||
</FilledButton>
|
</FilledButton>
|
||||||
<OutlinedButton onClick={() => updateMode('enter')}>{i18nText('enter', locale)}</OutlinedButton>
|
<OutlinedButton onClick={() => updateMode('enter')}>{i18nText('enter', locale)}</OutlinedButton>
|
||||||
<span>{i18nText('or', locale)}</span>
|
<span>{i18nText('or', locale)}</span>
|
||||||
<OutlinedButton
|
<AppleLogin
|
||||||
icon={<Image src="/images/facebook-logo.png" height={20} width={20} alt="" />}
|
clientId="bbuddy.expert"
|
||||||
onClick={() => onSocialRegister(Social.FACEBOOK)}
|
redirectURI="https://bbuddy.expert"
|
||||||
>
|
state="bbregister"
|
||||||
{i18nText('facebook', locale)}
|
responseType="code"
|
||||||
</OutlinedButton>
|
responseMode="query"
|
||||||
<OutlinedButton
|
render={({ onClick }) => (
|
||||||
icon={<Image src="/images/apple-logo.png" height={22} width={22} alt="" />}
|
<OutlinedButton
|
||||||
onClick={() => onSocialRegister(Social.APPLE)}
|
icon={<Image src="/images/apple-logo.png" height={22} width={22} alt="" />}
|
||||||
>
|
onClick={onClick}
|
||||||
{i18nText('apple', locale)}
|
>
|
||||||
</OutlinedButton>
|
{i18nText('apple', locale)}
|
||||||
|
</OutlinedButton>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
<OutlinedButton
|
<OutlinedButton
|
||||||
icon={<Image src="/images/google-logo.png" height={20} width={20} alt="" />}
|
icon={<Image src="/images/google-logo.png" height={20} width={20} alt="" />}
|
||||||
onClick={() => onSocialRegister(Social.GOOGLE)}
|
onClick={onGoogleLogin}
|
||||||
>
|
>
|
||||||
{i18nText('google', locale)}
|
{i18nText('google', locale)}
|
||||||
</OutlinedButton>
|
</OutlinedButton>
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { Link as IntlLink } from '../../../navigation';
|
import { Link as IntlLink } from '../../../i18n/routing';
|
||||||
import { i18nText } from '../../../i18nKeys';
|
import { i18nText } from '../../../i18nKeys';
|
||||||
|
|
||||||
export const Footer = ({ locale }: { locale: string }) => {
|
export const Footer = ({ locale }: { locale: string }) => {
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
'use client'
|
'use client'
|
||||||
|
|
||||||
import React, { FC, useState, useEffect } from 'react';
|
import React, { useState, useEffect } from 'react';
|
||||||
import { Button } from 'antd';
|
import { Button } from 'antd';
|
||||||
import { useSelectedLayoutSegment } from 'next/navigation';
|
import { useSelectedLayoutSegment } from 'next/navigation';
|
||||||
import { Link } from '../../../navigation';
|
import { Link } from '../../../i18n/routing';
|
||||||
import { AUTH_TOKEN_KEY } from '../../../constants/common';
|
import { AUTH_TOKEN_KEY } from '../../../constants/common';
|
||||||
import { useLocalStorage } from '../../../hooks/useLocalStorage';
|
import { useLocalStorage } from '../../../hooks/useLocalStorage';
|
||||||
import { AuthModal } from '../../Modals/AuthModal';
|
import { AuthModal } from '../../Modals/AuthModal';
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useSelectedLayoutSegment } from 'next/navigation';
|
import { useSelectedLayoutSegment } from 'next/navigation';
|
||||||
import dynamic from 'next/dynamic';
|
import dynamic from 'next/dynamic';
|
||||||
import { Link } from '../../../navigation';
|
import { Link } from '../../../i18n/routing';
|
||||||
|
|
||||||
type HeaderMenuProps = {
|
type HeaderMenuProps = {
|
||||||
locale: string;
|
locale: string;
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
import React, { FC, useState } from 'react';
|
import React, { FC, useState } from 'react';
|
||||||
import dynamic from 'next/dynamic';
|
import dynamic from 'next/dynamic';
|
||||||
import { useSelectedLayoutSegment } from 'next/navigation';
|
import { useSelectedLayoutSegment } from 'next/navigation';
|
||||||
import { Link } from '../../../navigation';
|
import { Link } from '../../../i18n/routing';
|
||||||
|
|
||||||
type HeaderMenuMobileProps = {
|
type HeaderMenuMobileProps = {
|
||||||
locale: string;
|
locale: string;
|
||||||
|
|
|
@ -4,7 +4,7 @@ import React, { useTransition, useState } from 'react';
|
||||||
import { Dropdown } from 'antd';
|
import { Dropdown } from 'antd';
|
||||||
import type { MenuProps } from 'antd';
|
import type { MenuProps } from 'antd';
|
||||||
import { CaretDownOutlined, CaretUpOutlined } from '@ant-design/icons';
|
import { CaretDownOutlined, CaretUpOutlined } from '@ant-design/icons';
|
||||||
import { useRouter, usePathname } from '../../../navigation';
|
import { useRouter, usePathname } from '../../../i18n/routing';
|
||||||
import { LOCALES } from '../../../constants/locale';
|
import { LOCALES } from '../../../constants/locale';
|
||||||
import { Locale } from '../../../types/locale';
|
import { Locale } from '../../../types/locale';
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@ import { HeaderMenu } from './HeaderMenu';
|
||||||
import { LanguageSwitcher } from './LanguageSwitcher';
|
import { LanguageSwitcher } from './LanguageSwitcher';
|
||||||
import { HeaderMobileMenu } from './HeaderMobileMenu';
|
import { HeaderMobileMenu } from './HeaderMobileMenu';
|
||||||
import { HEAD_ROUTES } from '../../../constants/routes';
|
import { HEAD_ROUTES } from '../../../constants/routes';
|
||||||
import { Link } from '../../../navigation';
|
import { Link } from '../../../i18n/routing';
|
||||||
import { i18nText } from '../../../i18nKeys';
|
import { i18nText } from '../../../i18nKeys';
|
||||||
|
|
||||||
type HeaderProps = {
|
type HeaderProps = {
|
||||||
|
|
|
@ -81,7 +81,7 @@ export const CheckoutForm: FC<PaymentFormProps> = ({ amount, sessionId, locale }
|
||||||
elements,
|
elements,
|
||||||
clientSecret,
|
clientSecret,
|
||||||
confirmParams: {
|
confirmParams: {
|
||||||
return_url: window.location.href,
|
return_url: window?.location?.href || '',
|
||||||
payment_method_data: {
|
payment_method_data: {
|
||||||
allow_redisplay: 'limited',
|
allow_redisplay: 'limited',
|
||||||
// billing_details: {
|
// billing_details: {
|
||||||
|
|
|
@ -2,7 +2,6 @@ import { Locale } from '../types/locale';
|
||||||
|
|
||||||
export const DEFAULT_LOCALE = Locale.en;
|
export const DEFAULT_LOCALE = Locale.en;
|
||||||
export const ALLOWED_LOCALES = [Locale.en, Locale.ru, Locale.de, Locale.it, Locale.es, Locale.fr] as const;
|
export const ALLOWED_LOCALES = [Locale.en, Locale.ru, Locale.de, Locale.it, Locale.es, Locale.fr] as const;
|
||||||
export const LOCALE_PREFIX = undefined;
|
|
||||||
|
|
||||||
export const LOCALES = {
|
export const LOCALES = {
|
||||||
[Locale.en]: 'En',
|
[Locale.en]: 'En',
|
||||||
|
|
|
@ -2,14 +2,14 @@ import { useState, useEffect } from 'react';
|
||||||
|
|
||||||
export function getStorageValue (key: string, defaultValue: any) {
|
export function getStorageValue (key: string, defaultValue: any) {
|
||||||
if (typeof window !== 'undefined') {
|
if (typeof window !== 'undefined') {
|
||||||
const saved = localStorage.getItem(key);
|
const saved = localStorage?.getItem(key);
|
||||||
return saved || defaultValue;
|
return saved || defaultValue;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export function deleteStorageKey (key: string) {
|
export function deleteStorageKey (key: string) {
|
||||||
if (typeof window !== 'undefined') {
|
if (typeof window !== 'undefined') {
|
||||||
localStorage.removeItem(key);
|
localStorage?.removeItem(key);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -19,7 +19,7 @@ export const useLocalStorage = (key: string, defaultValue: any) => {
|
||||||
});
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
localStorage.setItem(key, value);
|
localStorage?.setItem(key, value);
|
||||||
}, [key, value]);
|
}, [key, value]);
|
||||||
|
|
||||||
return [value, setValue];
|
return [value, setValue];
|
||||||
|
|
|
@ -14,21 +14,21 @@ export const useOauthWindow = () => {
|
||||||
|
|
||||||
if (data.messageType === 'oAuth') {
|
if (data.messageType === 'oAuth') {
|
||||||
messageHandler(event);
|
messageHandler(event);
|
||||||
window.removeEventListener('message', handler);
|
window?.removeEventListener('message', handler);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
window.removeEventListener('message', handler);
|
window?.removeEventListener('message', handler);
|
||||||
|
|
||||||
if (!oauthWindow || oauthWindow.closed) {
|
if (!oauthWindow || oauthWindow.closed) {
|
||||||
// окно ещё не существует, либо было закрыто
|
// окно ещё не существует, либо было закрыто
|
||||||
oauthWindow = window.open(url, name, params)!;
|
oauthWindow = window?.open(url, name, params)!;
|
||||||
} else {
|
} else {
|
||||||
// окно уже существует
|
// окно уже существует
|
||||||
oauthWindow!.focus();
|
oauthWindow!.focus();
|
||||||
}
|
}
|
||||||
|
|
||||||
window.addEventListener('message', handler);
|
window?.addEventListener('message', handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|
10
src/i18n.ts
10
src/i18n.ts
|
@ -1,10 +0,0 @@
|
||||||
import { getRequestConfig } from 'next-intl/server';
|
|
||||||
import { Locale } from './types/locale';
|
|
||||||
|
|
||||||
export default getRequestConfig(async ({ locale }) => ({
|
|
||||||
messages: (
|
|
||||||
await (locale === Locale.en
|
|
||||||
? import('../messages/en.json')
|
|
||||||
: import(`../messages/${locale}.json`))
|
|
||||||
).default
|
|
||||||
}));
|
|
|
@ -0,0 +1,15 @@
|
||||||
|
import { getRequestConfig } from 'next-intl/server';
|
||||||
|
import { routing } from './routing';
|
||||||
|
|
||||||
|
export default getRequestConfig(async ({ requestLocale }) => {
|
||||||
|
let locale = await requestLocale;
|
||||||
|
|
||||||
|
if (!locale || !routing.locales.includes(locale as any)) {
|
||||||
|
locale = routing.defaultLocale;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
locale,
|
||||||
|
messages: (await import(`../../messages/${locale}.json`)).default
|
||||||
|
};
|
||||||
|
});
|
|
@ -0,0 +1,10 @@
|
||||||
|
import { defineRouting } from 'next-intl/routing';
|
||||||
|
import { createNavigation } from 'next-intl/navigation';
|
||||||
|
import { ALLOWED_LOCALES, DEFAULT_LOCALE } from '../constants/locale';
|
||||||
|
|
||||||
|
export const routing = defineRouting({
|
||||||
|
locales: ALLOWED_LOCALES,
|
||||||
|
defaultLocale: DEFAULT_LOCALE
|
||||||
|
});
|
||||||
|
|
||||||
|
export const { Link, redirect, usePathname, useRouter, getPathname } = createNavigation(routing);
|
|
@ -185,6 +185,10 @@ export default {
|
||||||
sessionWishes: 'Schreiben Sie Ihre Wünsche zur Sitzung',
|
sessionWishes: 'Schreiben Sie Ihre Wünsche zur Sitzung',
|
||||||
successPayment: 'Erfolgreiche Zahlung',
|
successPayment: 'Erfolgreiche Zahlung',
|
||||||
errorPayment: 'Zahlungsfehler',
|
errorPayment: 'Zahlungsfehler',
|
||||||
|
chat: {
|
||||||
|
join: 'Chat starten',
|
||||||
|
joinAI: 'Start AI chat'
|
||||||
|
},
|
||||||
errors: {
|
errors: {
|
||||||
invalidEmail: 'Die E-Mail-Adresse ist ungültig',
|
invalidEmail: 'Die E-Mail-Adresse ist ungültig',
|
||||||
emptyEmail: 'Bitte geben Sie Ihre E-Mail ein',
|
emptyEmail: 'Bitte geben Sie Ihre E-Mail ein',
|
||||||
|
|
|
@ -185,6 +185,10 @@ export default {
|
||||||
sessionWishes: 'Write your wishes about the session',
|
sessionWishes: 'Write your wishes about the session',
|
||||||
successPayment: 'Successful Payment',
|
successPayment: 'Successful Payment',
|
||||||
errorPayment: 'Payment Error',
|
errorPayment: 'Payment Error',
|
||||||
|
chat: {
|
||||||
|
join: 'Start chat',
|
||||||
|
joinAI: 'Start AI chat'
|
||||||
|
},
|
||||||
errors: {
|
errors: {
|
||||||
invalidEmail: 'The email address is not valid',
|
invalidEmail: 'The email address is not valid',
|
||||||
emptyEmail: 'Please enter your E-mail',
|
emptyEmail: 'Please enter your E-mail',
|
||||||
|
|
|
@ -185,6 +185,10 @@ export default {
|
||||||
sessionWishes: 'Escribe tus deseos sobre la sesión',
|
sessionWishes: 'Escribe tus deseos sobre la sesión',
|
||||||
successPayment: 'Pago Exitoso',
|
successPayment: 'Pago Exitoso',
|
||||||
errorPayment: 'Error de Pago',
|
errorPayment: 'Error de Pago',
|
||||||
|
chat: {
|
||||||
|
join: 'Empezar un chat',
|
||||||
|
joinAI: 'Start AI chat'
|
||||||
|
},
|
||||||
errors: {
|
errors: {
|
||||||
invalidEmail: 'La dirección de correo electrónico no es válida',
|
invalidEmail: 'La dirección de correo electrónico no es válida',
|
||||||
emptyEmail: 'Introduce tu correo electrónico',
|
emptyEmail: 'Introduce tu correo electrónico',
|
||||||
|
|
|
@ -185,6 +185,10 @@ export default {
|
||||||
sessionWishes: 'Écrivez vos souhaits concernant la session',
|
sessionWishes: 'Écrivez vos souhaits concernant la session',
|
||||||
successPayment: 'Paiement Réussi',
|
successPayment: 'Paiement Réussi',
|
||||||
errorPayment: 'Erreur de Paiement',
|
errorPayment: 'Erreur de Paiement',
|
||||||
|
chat: {
|
||||||
|
join: 'Commencer la discussion',
|
||||||
|
joinAI: 'Start AI chat'
|
||||||
|
},
|
||||||
errors: {
|
errors: {
|
||||||
invalidEmail: 'L\'adresse e-mail n\'est pas valide',
|
invalidEmail: 'L\'adresse e-mail n\'est pas valide',
|
||||||
emptyEmail: 'Veuillez saisir votre e-mail',
|
emptyEmail: 'Veuillez saisir votre e-mail',
|
||||||
|
|
|
@ -185,6 +185,10 @@ export default {
|
||||||
sessionWishes: 'Scrivi i tuoi desideri riguardo alla sessione',
|
sessionWishes: 'Scrivi i tuoi desideri riguardo alla sessione',
|
||||||
successPayment: 'Pagamento Riuscito',
|
successPayment: 'Pagamento Riuscito',
|
||||||
errorPayment: 'Errore di Pagamento',
|
errorPayment: 'Errore di Pagamento',
|
||||||
|
chat: {
|
||||||
|
join: 'Avvia chat',
|
||||||
|
joinAI: 'Start AI chat'
|
||||||
|
},
|
||||||
errors: {
|
errors: {
|
||||||
invalidEmail: 'L\'indirizzo e-mail non è valido',
|
invalidEmail: 'L\'indirizzo e-mail non è valido',
|
||||||
emptyEmail: 'Inserisci l\'e-mail',
|
emptyEmail: 'Inserisci l\'e-mail',
|
||||||
|
|
|
@ -185,6 +185,10 @@ export default {
|
||||||
sessionWishes: 'Напишите свои пожелания по поводу сессии',
|
sessionWishes: 'Напишите свои пожелания по поводу сессии',
|
||||||
successPayment: 'Успешная оплата',
|
successPayment: 'Успешная оплата',
|
||||||
errorPayment: 'Ошибка оплаты',
|
errorPayment: 'Ошибка оплаты',
|
||||||
|
chat: {
|
||||||
|
join: 'Начать чат',
|
||||||
|
joinAI: 'Начать чат с ИИ'
|
||||||
|
},
|
||||||
errors: {
|
errors: {
|
||||||
invalidEmail: 'Адрес электронной почты недействителен',
|
invalidEmail: 'Адрес электронной почты недействителен',
|
||||||
emptyEmail: 'Пожалуйста, введите ваш E-mail',
|
emptyEmail: 'Пожалуйста, введите ваш E-mail',
|
||||||
|
|
|
@ -20,6 +20,27 @@ export const onSuccessRequestCallback = (config: InternalAxiosRequestConfig) =>
|
||||||
return newConfig;
|
return newConfig;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const onSuccessRequestJwtCallback = (config: InternalAxiosRequestConfig) => {
|
||||||
|
const newConfig = { ...config };
|
||||||
|
|
||||||
|
if (typeof window !== 'undefined') {
|
||||||
|
var jwt = localStorage.getItem('bbuddy_token_test');
|
||||||
|
|
||||||
|
if(jwt) {
|
||||||
|
newConfig.headers.set('Authorization', `Bearer ${jwt}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return newConfig;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const onSuccessResponseJwtCallback = (response: AxiosResponse) => {
|
||||||
|
var header = response.headers['x-new-token'];
|
||||||
|
if(header) {
|
||||||
|
localStorage.setItem('bbuddy_token_test', header);
|
||||||
|
}
|
||||||
|
return response;
|
||||||
|
};
|
||||||
|
|
||||||
export const onSuccessResponseCallback = (response: AxiosResponse) => response;
|
export const onSuccessResponseCallback = (response: AxiosResponse) => response;
|
||||||
|
|
||||||
export const onErrorResponseCallback = (error: any) => Promise.reject(error);
|
export const onErrorResponseCallback = (error: any) => Promise.reject(error);
|
||||||
|
@ -34,3 +55,6 @@ apiClient.interceptors.response.use(
|
||||||
onSuccessResponseCallback,
|
onSuccessResponseCallback,
|
||||||
onErrorResponseCallback
|
onErrorResponseCallback
|
||||||
);
|
);
|
||||||
|
|
||||||
|
apiClient.interceptors.response.use(onSuccessResponseJwtCallback);
|
||||||
|
apiClient.interceptors.request.use(onSuccessRequestJwtCallback);
|
||||||
|
|
|
@ -0,0 +1,87 @@
|
||||||
|
import { HubConnection, HubConnectionBuilder, LogLevel } from '@microsoft/signalr';
|
||||||
|
import { IHttpConnectionOptions } from "@microsoft/signalr/src/IHttpConnectionOptions";
|
||||||
|
import { BASE_URL } from '../constants/common';
|
||||||
|
import { IChatMessage } from '../types/chat';
|
||||||
|
|
||||||
|
const chatMessageMethodName = 'ReceiveMessage';
|
||||||
|
const chatUserStatusChangeMethodName = 'chatUserStatusChange';
|
||||||
|
const chatSeenMethodName = 'MessageWasRead';
|
||||||
|
const chatTypingMethodName = 'directTyping';
|
||||||
|
const sendChatTextMethodName = 'SendMessage';
|
||||||
|
const sendChatTypingMethodName = 'DirectIsTyping';
|
||||||
|
const serverError = 'Error';
|
||||||
|
|
||||||
|
const sendChatSeenMethodName = 'ConfirmMessageRead';
|
||||||
|
const sendChatMessagesSeenMethodName = 'ConfirmMessagesRead';
|
||||||
|
|
||||||
|
|
||||||
|
const joinChatMethodName = 'JoinChat';
|
||||||
|
const leaveChatMethodName = 'LeaveChat';
|
||||||
|
|
||||||
|
const joinChatsMethodName = 'JoinChats';
|
||||||
|
const leaveChatsMethodName = 'LeaveChats';
|
||||||
|
const chatsReceiveMessageMethodName = 'ChatsMessageCreated';
|
||||||
|
|
||||||
|
class SignalConnector {
|
||||||
|
private connection: HubConnection;
|
||||||
|
private events = {} as any;
|
||||||
|
static instance?: SignalConnector;
|
||||||
|
constructor({ jwt }: { jwt?: string}) {
|
||||||
|
console.log('here')
|
||||||
|
const options = {
|
||||||
|
accessTokenFactory: () => jwt
|
||||||
|
} as IHttpConnectionOptions;
|
||||||
|
this.connection = new HubConnectionBuilder()
|
||||||
|
.withUrl(`${BASE_URL}/hubs/chat`, options)
|
||||||
|
.withAutomaticReconnect()
|
||||||
|
.configureLogging(LogLevel.Debug)
|
||||||
|
.build();
|
||||||
|
this.connection.start().then(()=>{
|
||||||
|
this.events?.onConnected(true)
|
||||||
|
for (const k in this.events) {
|
||||||
|
if (k != 'onConnected'){
|
||||||
|
this.connection.on(k, (payload: any) => {
|
||||||
|
this.events[k](payload);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}).catch(err => console.log('SignalR Error', err));
|
||||||
|
}
|
||||||
|
public addListener = (name: string, func: any)=> {
|
||||||
|
this.events[name] = func;
|
||||||
|
}
|
||||||
|
|
||||||
|
public closeConnection = () => {
|
||||||
|
this.connection.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
public newMessage = (message: IChatMessage) => {
|
||||||
|
this.connection.invoke(sendChatTextMethodName, message).then(x => console.log('NewMsg',x))
|
||||||
|
}
|
||||||
|
public joinChat = (groupId: number) => {
|
||||||
|
this.connection.invoke(joinChatMethodName, groupId, null).then(x => console.log(joinChatMethodName, x))
|
||||||
|
}
|
||||||
|
|
||||||
|
public joinChatPerson = (accId: number) => {
|
||||||
|
return this.connection.invoke(joinChatMethodName, 0, accId)
|
||||||
|
}
|
||||||
|
|
||||||
|
public readMessages = (messagesId: number[]) => {
|
||||||
|
this.connection.invoke(sendChatMessagesSeenMethodName, messagesId).then(x => console.log(sendChatMessagesSeenMethodName, x))
|
||||||
|
}
|
||||||
|
|
||||||
|
public static getInstance({ jwt }: { jwt?: string }): SignalConnector | undefined {
|
||||||
|
if (!SignalConnector.instance) {
|
||||||
|
if (jwt) {
|
||||||
|
SignalConnector.instance = new SignalConnector({ jwt });
|
||||||
|
return SignalConnector.instance;
|
||||||
|
} else {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return SignalConnector.instance;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default SignalConnector.getInstance;
|
|
@ -2,7 +2,7 @@ import "server-only";
|
||||||
|
|
||||||
import Stripe from "stripe";
|
import Stripe from "stripe";
|
||||||
|
|
||||||
export const stripe = new Stripe(process.env.STRIPE_SECRET_KEY as string, {
|
export const stripe = new Stripe(process.env.NEXT_PUBLIC_STRIPE_SECRET_KEY as string, {
|
||||||
apiVersion: "2024-06-20",
|
apiVersion: "2024-06-20",
|
||||||
appInfo: {
|
appInfo: {
|
||||||
name: "bbuddy-ui",
|
name: "bbuddy-ui",
|
||||||
|
|
|
@ -1,11 +1,7 @@
|
||||||
import createMiddleware from 'next-intl/middleware';
|
import createMiddleware from 'next-intl/middleware';
|
||||||
import { ALLOWED_LOCALES, DEFAULT_LOCALE, LOCALE_PREFIX } from './constants/locale';
|
import { routing } from './i18n/routing';
|
||||||
|
|
||||||
export default createMiddleware({
|
export default createMiddleware(routing);
|
||||||
locales: ALLOWED_LOCALES,
|
|
||||||
localePrefix: LOCALE_PREFIX,
|
|
||||||
defaultLocale: DEFAULT_LOCALE
|
|
||||||
});
|
|
||||||
|
|
||||||
export const config = {
|
export const config = {
|
||||||
matcher: ['/', '/(en|ru|de|it|es|fr)/:path*']
|
matcher: ['/', '/(en|ru|de|it|es|fr)/:path*']
|
||||||
|
|
|
@ -1,17 +0,0 @@
|
||||||
import { createLocalizedPathnamesNavigation, Pathnames } from 'next-intl/navigation';
|
|
||||||
import { ALLOWED_LOCALES, LOCALE_PREFIX } from './constants/locale';
|
|
||||||
|
|
||||||
export const { Link, redirect, usePathname, useRouter } =
|
|
||||||
createLocalizedPathnamesNavigation({
|
|
||||||
locales: ALLOWED_LOCALES,
|
|
||||||
pathnames: {
|
|
||||||
'/': '/',
|
|
||||||
// '/experts': '/experts',
|
|
||||||
// '/experts/[expertId]': '/experts/[expertId]',
|
|
||||||
// '/news': '/news',
|
|
||||||
// '/privacy-policy': '/privacy-policy',
|
|
||||||
// '/[userId]': '/[userId]',
|
|
||||||
// '/[userId]/[...slug]': '/[userId]/[...slug]',
|
|
||||||
} satisfies Pathnames<typeof ALLOWED_LOCALES>,
|
|
||||||
localePrefix: LOCALE_PREFIX
|
|
||||||
});
|
|
|
@ -1,5 +1,5 @@
|
||||||
.b-slider {
|
.b-slider {
|
||||||
margin-top: -105px;
|
//margin-top: -105px;
|
||||||
|
|
||||||
h2 {
|
h2 {
|
||||||
margin: 40px 0 20px;
|
margin: 40px 0 20px;
|
||||||
|
|
|
@ -563,6 +563,10 @@ a {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.btn-disabled {
|
||||||
|
opacity: .4 !important;
|
||||||
|
}
|
||||||
|
|
||||||
.btn-back {
|
.btn-back {
|
||||||
user-select: none;
|
user-select: none;
|
||||||
outline: none;
|
outline: none;
|
||||||
|
@ -823,6 +827,7 @@ a {
|
||||||
flex: 0 0 100%;
|
flex: 0 0 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 16px;
|
gap: 16px;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
.btn-apply,
|
.btn-apply,
|
||||||
.btn-video {
|
.btn-video {
|
||||||
|
|
|
@ -13,6 +13,8 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
&__inner {
|
&__inner {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column-reverse;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
height: 0;
|
height: 0;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
@ -94,7 +96,8 @@
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|
||||||
textarea {
|
textarea {
|
||||||
width: calc(100% - 136px);
|
resize: none;
|
||||||
|
width: calc(100% - 40px);
|
||||||
height: 24px;
|
height: 24px;
|
||||||
padding: 0 8px;
|
padding: 0 8px;
|
||||||
border-radius: 0;
|
border-radius: 0;
|
||||||
|
|
|
@ -0,0 +1,12 @@
|
||||||
|
|
||||||
|
export interface IChatMessage {
|
||||||
|
GroupId: string;
|
||||||
|
CreatorId: number;
|
||||||
|
Content: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IChatJoin {
|
||||||
|
GroupId: string;
|
||||||
|
CreatorId: number;
|
||||||
|
Content: string;
|
||||||
|
}
|
|
@ -37,6 +37,7 @@ export type ThemeGroup = {
|
||||||
|
|
||||||
export interface ExpertItem {
|
export interface ExpertItem {
|
||||||
id: number;
|
id: number;
|
||||||
|
botUserId?: number;
|
||||||
name: string;
|
name: string;
|
||||||
surname?: string;
|
surname?: string;
|
||||||
faceImageUrl?: string;
|
faceImageUrl?: string;
|
||||||
|
|
|
@ -30,6 +30,7 @@ export type ProfileRequest = {
|
||||||
faceImage?: any;
|
faceImage?: any;
|
||||||
isFaceImageKeepExisting?: boolean;
|
isFaceImageKeepExisting?: boolean;
|
||||||
phone?: string;
|
phone?: string;
|
||||||
|
role?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type PayInfo = {
|
export type PayInfo = {
|
||||||
|
|
|
@ -5,8 +5,7 @@ import { i18nText } from '../i18nKeys';
|
||||||
const ROUTES = ['sessions', 'rooms', 'notifications', 'support', 'information', 'settings', 'messages', 'expert-profile'];
|
const ROUTES = ['sessions', 'rooms', 'notifications', 'support', 'information', 'settings', 'messages', 'expert-profile'];
|
||||||
const COUNTS: Record<string, number> = {
|
const COUNTS: Record<string, number> = {
|
||||||
sessions: 12,
|
sessions: 12,
|
||||||
notifications: 5,
|
notifications: 5
|
||||||
messages: 113
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getMenuConfig = (locale: string) => ROUTES.map((path) => ({
|
export const getMenuConfig = (locale: string) => ROUTES.map((path) => ({
|
||||||
|
|
Loading…
Reference in New Issue