From b93ab9794d7c7efdc2aa2f3f4d70dbf15979a37f Mon Sep 17 00:00:00 2001 From: dm1sh Date: Tue, 1 Aug 2023 18:23:56 +0300 Subject: [PATCH] Added announcement disposal: Added ann details button Added modal shown on its click Moved trashbox selection there Added trashboxes mock while testing in restricted area --- front/package-lock.json | 20 ++- front/package.json | 4 +- front/src/api/dispose/index.ts | 19 +++ front/src/api/dispose/types.ts | 23 ++++ front/src/api/trashbox/types.ts | 2 + front/src/components/AnnouncementDetails.tsx | 124 ++++++++++++------ .../src/components/SelectDisposalTrashbox.tsx | 119 +++++++++++++++++ front/src/components/TrashboxMarkers.tsx | 49 +++---- front/src/components/index.ts | 1 + front/src/hooks/api/index.ts | 1 + front/src/hooks/api/useDispose.ts | 35 +++++ front/src/hooks/api/useRemoveAnnouncement.ts | 2 +- front/src/hooks/api/useTrashboxes.ts | 47 +++++-- front/src/pages/AddPage.tsx | 54 +------- front/src/utils/dispose.ts | 9 ++ 15 files changed, 383 insertions(+), 126 deletions(-) create mode 100644 front/src/api/dispose/index.ts create mode 100644 front/src/api/dispose/types.ts create mode 100644 front/src/components/SelectDisposalTrashbox.tsx create mode 100644 front/src/hooks/api/useDispose.ts create mode 100644 front/src/utils/dispose.ts diff --git a/front/package-lock.json b/front/package-lock.json index 2b86f66..a60e9b8 100644 --- a/front/package-lock.json +++ b/front/package-lock.json @@ -8,7 +8,6 @@ "name": "front", "version": "0.0.0", "dependencies": { - "@types/leaflet": "^1.9.3", "bootstrap": "^5.3.0", "jwt-decode": "^3.1.2", "leaflet": "^1.9.4", @@ -22,6 +21,8 @@ }, "devDependencies": { "@faker-js/faker": "^8.0.2", + "@types/leaflet": "^1.9.3", + "@types/lodash": "^4.14.196", "@types/react": "^18.2.14", "@types/react-dom": "^18.2.6", "@typescript-eslint/eslint-plugin": "^5.61.0", @@ -30,6 +31,7 @@ "eslint": "^8.44.0", "eslint-plugin-react-hooks": "^4.6.0", "eslint-plugin-react-refresh": "^0.4.1", + "lodash": "^4.17.21", "typescript": "^5.0.2", "vite": "^4.4.0" } @@ -1059,7 +1061,8 @@ "node_modules/@types/geojson": { "version": "7946.0.10", "resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-7946.0.10.tgz", - "integrity": "sha512-Nmh0K3iWQJzniTuPRcJn5hxXkfB1T1pgB89SBig5PlJQU5yocazeu4jATJlaA0GYFKWMqDdvYemoSnF2pXgLVA==" + "integrity": "sha512-Nmh0K3iWQJzniTuPRcJn5hxXkfB1T1pgB89SBig5PlJQU5yocazeu4jATJlaA0GYFKWMqDdvYemoSnF2pXgLVA==", + "dev": true }, "node_modules/@types/json-schema": { "version": "7.0.12", @@ -1071,10 +1074,17 @@ "version": "1.9.3", "resolved": "https://registry.npmjs.org/@types/leaflet/-/leaflet-1.9.3.tgz", "integrity": "sha512-Caa1lYOgKVqDkDZVWkto2Z5JtVo09spEaUt2S69LiugbBpoqQu92HYFMGUbYezZbnBkyOxMNPXHSgRrRY5UyIA==", + "dev": true, "dependencies": { "@types/geojson": "*" } }, + "node_modules/@types/lodash": { + "version": "4.14.196", + "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.196.tgz", + "integrity": "sha512-22y3o88f4a94mKljsZcanlNWPzO0uBsBdzLAngf2tp533LzZcQzb6+eZPJ+vCTt+bqF2XnvT9gejTLsAcJAJyQ==", + "dev": true + }, "node_modules/@types/prop-types": { "version": "15.7.5", "resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.5.tgz", @@ -2417,6 +2427,12 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", + "dev": true + }, "node_modules/lodash.debounce": { "version": "4.0.8", "resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz", diff --git a/front/package.json b/front/package.json index 163e525..cc9df15 100644 --- a/front/package.json +++ b/front/package.json @@ -12,7 +12,6 @@ "addFetchApiRoute": "bash utils/addFetchApiRoute.sh" }, "dependencies": { - "@types/leaflet": "^1.9.3", "bootstrap": "^5.3.0", "jwt-decode": "^3.1.2", "leaflet": "^1.9.4", @@ -26,6 +25,8 @@ }, "devDependencies": { "@faker-js/faker": "^8.0.2", + "@types/leaflet": "^1.9.3", + "@types/lodash": "^4.14.196", "@types/react": "^18.2.14", "@types/react-dom": "^18.2.6", "@typescript-eslint/eslint-plugin": "^5.61.0", @@ -34,6 +35,7 @@ "eslint": "^8.44.0", "eslint-plugin-react-hooks": "^4.6.0", "eslint-plugin-react-refresh": "^0.4.1", + "lodash": "^4.17.21", "typescript": "^5.0.2", "vite": "^4.4.0" } diff --git a/front/src/api/dispose/index.ts b/front/src/api/dispose/index.ts new file mode 100644 index 0000000..de3fcf8 --- /dev/null +++ b/front/src/api/dispose/index.ts @@ -0,0 +1,19 @@ +import { API_URL } from '../../config' +import { TrashboxDispose, DisposeResponse } from './types' + +const composeDisposeURL = () => ( + API_URL + '/announcement/dispose?' +) + +const composeDisposeBody = (ann_id: number, trashbox: TrashboxDispose) => ( + JSON.stringify({ + ann_id, + trashbox, + }) +) + +const processDispose = (data: DisposeResponse): boolean => { + return data.Success +} + +export { composeDisposeURL, composeDisposeBody, processDispose } diff --git a/front/src/api/dispose/types.ts b/front/src/api/dispose/types.ts new file mode 100644 index 0000000..d16e857 --- /dev/null +++ b/front/src/api/dispose/types.ts @@ -0,0 +1,23 @@ +import { composeDisposeBody } from '.' +import { isObject } from '../../utils/types' +import { Trashbox } from '../trashbox/types' + +type TrashboxDispose = Omit & { Category: string } + +type DisposeParams = Parameters + +type DisposeAnnParams = DisposeParams extends [ann_id: number, ...args: infer P] ? P : never + +type DisposeResponse = { + Success: boolean, +} + +const isDisposeResponse = (obj: unknown): obj is DisposeResponse => ( + isObject(obj, { + 'Success': 'boolean', + }) +) + +export type { TrashboxDispose, DisposeParams, DisposeAnnParams, DisposeResponse } + +export { isDisposeResponse } diff --git a/front/src/api/trashbox/types.ts b/front/src/api/trashbox/types.ts index 9d9461b..7458354 100644 --- a/front/src/api/trashbox/types.ts +++ b/front/src/api/trashbox/types.ts @@ -1,6 +1,7 @@ import { isArrayOf, isObject, isString } from '../../utils/types' type Trashbox = { + Name: string, Lat: number, Lng: number, Address: string, @@ -9,6 +10,7 @@ type Trashbox = { const isTrashbox = (obj: unknown): obj is Trashbox => ( isObject(obj, { + 'Name': 'string', 'Lat': 'number', 'Lng': 'number', 'Address': 'string', diff --git a/front/src/components/AnnouncementDetails.tsx b/front/src/components/AnnouncementDetails.tsx index 86c62aa..f0c0814 100644 --- a/front/src/components/AnnouncementDetails.tsx +++ b/front/src/components/AnnouncementDetails.tsx @@ -1,13 +1,15 @@ import { Modal, Button } from 'react-bootstrap' import { MapContainer, Marker, Popup, TileLayer } from 'react-leaflet' -import { CSSProperties } from 'react' +import { CSSProperties, useState } from 'react' import LineDot from './LineDot' import { categoryNames } from '../assets/category' -import { useBook, useRemoveAnnouncement } from '../hooks/api' +import { useBook, useDispose, 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, @@ -27,26 +29,85 @@ const styles = { } as CSSProperties, } -function AnnouncementDetails({ close, refresh, announcement: { - id, name, category, bestBy, description, lat, lng, address, metro, bookedBy, userId, -} }: AnnouncementDetailsProps) { +const View = ({ + announcement: { name, category, bestBy, description, lat, lng, address, metro }, +}: { announcement: Announcement }) => ( + <> +

{name}

+ + {categoryNames[category]} + {/* dot */} + Годен до {new Date(bestBy).toLocaleString('ru-RU')} + +

{description}

+ + + + + + + {address} +
+ {metro} +
+
+
+ +) + +type ControlProps = { + closeRefresh: () => void, + announcement: Announcement, + showDispose: () => void +} + +function Control({ + closeRefresh, + announcement: { bookedBy, id, userId }, + showDispose +}: ControlProps) { const { handleBook, bookButton } = useBook() - const removeRefresh = () => { + const { handleRemove, removeButton } = useRemoveAnnouncement(closeRefresh) + + const myId = useId() + + return ( + <> +

Забронировали {bookedBy} чел.

+ {(myId === userId) ? ( + <> + +