diff --git a/front/src/api/sendRate/index.ts b/front/src/api/sendRate/index.ts new file mode 100644 index 0000000..89dd394 --- /dev/null +++ b/front/src/api/sendRate/index.ts @@ -0,0 +1,12 @@ +import { API_URL } from '../../config' +import { SendRateResponse, SendRate } from './types' + +const composeSendRateURL = () => ( + API_URL + '/user/rating?' +) + +const processSendRate = (data: SendRateResponse): SendRate => { + return data.Success +} + +export { composeSendRateURL, processSendRate } diff --git a/front/src/api/sendRate/types.ts b/front/src/api/sendRate/types.ts new file mode 100644 index 0000000..156f429 --- /dev/null +++ b/front/src/api/sendRate/types.ts @@ -0,0 +1,17 @@ +import { isObject } from '../../utils/types' + +type SendRateResponse = { + Success: boolean +} + +const isSendRateResponse = (obj: unknown): obj is SendRateResponse => ( + isObject(obj, { + 'Success': 'boolean', + }) +) + +type SendRate = boolean + +export type { SendRateResponse, SendRate } + +export { isSendRateResponse } diff --git a/front/src/api/userRating/index.ts b/front/src/api/userRating/index.ts new file mode 100644 index 0000000..8ceedaa --- /dev/null +++ b/front/src/api/userRating/index.ts @@ -0,0 +1,14 @@ +import { API_URL } from '../../config' +import { UserRatingResponse, UserRating } from './types' + +const initialUserRating: UserRating = 0 + +const composeUserRatingURL = (userId: number) => ( + API_URL + '/user/rating?' + (new URLSearchParams({ user_id: userId.toString() })).toString() +) + +const processUserRating = (data: UserRatingResponse): UserRating => { + return data.rating +} + +export { initialUserRating, composeUserRatingURL, processUserRating } diff --git a/front/src/api/userRating/types.ts b/front/src/api/userRating/types.ts new file mode 100644 index 0000000..7a3cf1e --- /dev/null +++ b/front/src/api/userRating/types.ts @@ -0,0 +1,17 @@ +import { isObject } from '../../utils/types' + +type UserRatingResponse = { + rating: number +} + +const isUserRatingResponse = (obj: unknown): obj is UserRatingResponse => ( + isObject(obj, { + 'rating': 'number', + }) +) + +type UserRating = number + +export type { UserRatingResponse, UserRating } + +export { isUserRatingResponse } diff --git a/front/src/components/AnnouncementDetails.tsx b/front/src/components/AnnouncementDetails.tsx index f0c0814..d691cac 100644 --- a/front/src/components/AnnouncementDetails.tsx +++ b/front/src/components/AnnouncementDetails.tsx @@ -1,15 +1,16 @@ import { Modal, Button } from 'react-bootstrap' import { MapContainer, Marker, Popup, TileLayer } from 'react-leaflet' import { CSSProperties, useState } from 'react' +import { LatLng } from 'leaflet' import LineDot from './LineDot' +import Rating from './Rating' import { categoryNames } from '../assets/category' -import { useBook, useDispose, useRemoveAnnouncement } from '../hooks/api' +import { useBook, useRemoveAnnouncement } from '../hooks/api' import { Announcement } from '../api/announcement/types' import { iconItem } from '../utils/markerIcons' import { useId } from '../hooks' import SelectDisposalTrashbox from './SelectDisposalTrashbox' -import { LatLng } from 'leaflet' type AnnouncementDetailsProps = { close: () => void, @@ -30,7 +31,7 @@ const styles = { } const View = ({ - announcement: { name, category, bestBy, description, lat, lng, address, metro }, + announcement: { name, category, bestBy, description, lat, lng, address, metro, userId }, }: { announcement: Announcement }) => ( <>

{name}

@@ -39,7 +40,9 @@ const View = ({ {/* dot */} Годен до {new Date(bestBy).toLocaleString('ru-RU')} -

{description}

+

{description}

+ + void, + sendMyRate: () => void, +} + +function Star({ filled, selected, setMyRate, sendMyRate }: StarProps) { + return ( + + ) +} + +type RatingProps = { + userId: number, + className: string | undefined, +} + +function Rating({ userId, className }: RatingProps) { + const rating = useUserRating(userId) + + const [myRate, setMyRate] = useState(0) + + const { doSendRate } = useSendRate() + + async function sendMyRate() { + const res = await doSendRate(myRate) + + if (res) { + rating.refetch() + } + } + + return ( +

+ Рейтинг пользователя:{' '} + {gotResponse(rating) ? ( + gotError(rating) ? ( + {rating.error} + ) : ( + setMyRate(0)} + > + {...Array(5).fill(5).map( + (_, i) => + setMyRate(i + 1)} + sendMyRate={() => void sendMyRate()} + /> + )} + + ) + ) : ( + Загрузка... + ) + } +

+ + ) +} + +export default Rating diff --git a/front/src/components/index.ts b/front/src/components/index.ts index 57f2928..b124f0c 100644 --- a/front/src/components/index.ts +++ b/front/src/components/index.ts @@ -14,3 +14,4 @@ export { default as Points } from './Points' export { default as SignOut } from './SignOut' export { default as Poetry } from './Poetry' export { default as SelectDisposalTrashbox } from './SelectDisposalTrashbox' +export { default as Rating } from './Rating' diff --git a/front/src/hooks/api/index.ts b/front/src/hooks/api/index.ts index 93062d8..7e2d8f9 100644 --- a/front/src/hooks/api/index.ts +++ b/front/src/hooks/api/index.ts @@ -9,3 +9,5 @@ export { default as useSignIn } from './useSignIn' export { default as useSignUp } from './useSignUp' export { default as usePoetry } from './usePoetry' export { default as useDispose } from './useDispose' +export { default as useSendRate } from './useSendRate' +export { default as useUserRating } from './useUserRating' diff --git a/front/src/hooks/api/useSendRate.ts b/front/src/hooks/api/useSendRate.ts new file mode 100644 index 0000000..d3cfc03 --- /dev/null +++ b/front/src/hooks/api/useSendRate.ts @@ -0,0 +1,33 @@ +import { useSend } from '..' +import { composeSendRateURL, processSendRate } from '../../api/sendRate' +import { isSendRateResponse } from '../../api/sendRate/types' + +function useSendRate() { + const { doSend, ...rest } = useSend( + composeSendRateURL(), + 'POST', + true, + isSendRateResponse, + processSendRate, + ) + + const doSendRate = (rate: number) => ( + doSend({}, { + body: JSON.stringify({ + rate, + }), + headers: { + 'Content-Type': 'application/json', + }, + + }) + ) + + + return { + doSendRate, + ...rest, + } +} + +export default useSendRate diff --git a/front/src/hooks/api/useUserRating.ts b/front/src/hooks/api/useUserRating.ts new file mode 100644 index 0000000..b14a5bf --- /dev/null +++ b/front/src/hooks/api/useUserRating.ts @@ -0,0 +1,30 @@ +import { composeUserRatingURL, initialUserRating, processUserRating } from '../../api/userRating' +import { UserRating, isUserRatingResponse } from '../../api/userRating/types' +import useFetch, { UseFetchReturn } from '../useFetch' + +const useUserRating = (userId: number): UseFetchReturn => ( + // useFetch( + // composeUserRatingURL(userId), + // 'GET', + // false, + // isUserRatingResponse, + // processUserRating, + // initialUserRating + // ) + + { + data: 3, + loading: false, + error: null, + refetch: () => { return }, + } + + // { + // data: undefined, + // loading: true, + // error: null, + // refetch: () => { return }, + // } +) + +export default useUserRating diff --git a/front/src/styles/Rating.module.css b/front/src/styles/Rating.module.css new file mode 100644 index 0000000..48a57c7 --- /dev/null +++ b/front/src/styles/Rating.module.css @@ -0,0 +1,19 @@ +.star { + -webkit-text-stroke: 2px var(--bs-body-color); + font-size: 1.5rem; + color: var(--bs-modal-bg); + background: none; + border: none; + padding: 0 3px; + transition: 0.15s ease-in-out; +} + +.starFilled { + color: var(--bs-body-color); +} + +.starSelected { + color: var(--bs-success); + -webkit-text-stroke: 2px var(--bs-success); + transform: scale(1.3); +} \ No newline at end of file