diff --git a/front/.eslintrc.cjs b/front/.eslintrc.cjs index 76125b7..514cad1 100644 --- a/front/.eslintrc.cjs +++ b/front/.eslintrc.cjs @@ -32,5 +32,16 @@ module.exports = { } ], 'jsx-quotes': [2, 'prefer-single'], + 'comma-dangle': 'off', + '@typescript-eslint/comma-dangle': ['warn', { + 'arrays': 'always-multiline', + 'objects': 'always-multiline', + 'imports': 'always-multiline', + 'exports': 'always-multiline', + 'functions': 'only-multiline', + 'enums': 'always-multiline', + 'generics': 'always-multiline', + 'tuples': 'always-multiline', + }], }, } diff --git a/front/package-lock.json b/front/package-lock.json index 096b2cd..a60e9b8 100644 --- a/front/package-lock.json +++ b/front/package-lock.json @@ -8,8 +8,8 @@ "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", "react": "^18.2.0", "react-bootstrap": "^2.8.0", @@ -20,6 +20,9 @@ "react-router-dom": "^6.14.1" }, "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", @@ -28,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" } @@ -817,6 +821,22 @@ "node": "^12.22.0 || ^14.17.0 || >=16.0.0" } }, + "node_modules/@faker-js/faker": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/@faker-js/faker/-/faker-8.0.2.tgz", + "integrity": "sha512-Uo3pGspElQW91PCvKSIAXoEgAUlRnH29sX2/p89kg7sP1m2PzCufHINd0FhTXQf6DYGiUlVncdSPa2F9wxed2A==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/fakerjs" + } + ], + "engines": { + "node": "^14.17.0 || ^16.13.0 || >=18.0.0", + "npm": ">=6.14.13" + } + }, "node_modules/@humanwhocodes/config-array": { "version": "0.11.10", "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.10.tgz", @@ -1041,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", @@ -1053,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", @@ -2361,6 +2389,11 @@ "node": ">=6" } }, + "node_modules/jwt-decode": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/jwt-decode/-/jwt-decode-3.1.2.tgz", + "integrity": "sha512-UfpWE/VZn0iP50d8cz9NrZLM9lSWhcJ+0Gt/nm4by88UL+J1SiKN8/5dkjMmbEzwL2CAe+67GsegCbIKtbp75A==" + }, "node_modules/leaflet": { "version": "1.9.4", "resolved": "https://registry.npmjs.org/leaflet/-/leaflet-1.9.4.tgz", @@ -2394,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 5420348..cc9df15 100644 --- a/front/package.json +++ b/front/package.json @@ -12,8 +12,8 @@ "addFetchApiRoute": "bash utils/addFetchApiRoute.sh" }, "dependencies": { - "@types/leaflet": "^1.9.3", "bootstrap": "^5.3.0", + "jwt-decode": "^3.1.2", "leaflet": "^1.9.4", "react": "^18.2.0", "react-bootstrap": "^2.8.0", @@ -24,6 +24,9 @@ "react-router-dom": "^6.14.1" }, "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", @@ -32,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/prototype.html b/front/prototype.html new file mode 100644 index 0000000..0eceeca --- /dev/null +++ b/front/prototype.html @@ -0,0 +1,315 @@ + + + + + + + Document + + + + +
+
+

Поэзия

+
+
"Fury said to
+
a mouse, That
+
he met
+
in the
+
house,
+
'Let us
+
both go
+
to law:
+
I will
+
prosecute
+
you.
+
Come, I'll
+
take no
+
denial;
+
We must
+
have a
+
trial:
+
For
+
really
+
this
+
morning
+
I've
+
nothing
+
to do.'
+
Said the
+
mouse to
+
the cur,
+
'Such a
+
trial,
+
dear sir,
+
With no
+
jury or
+
judge,
+
would be
+
wasting
+
our breath.'
+
'I'll be
+
judge,
+
I'll be
+
jury,'
+
Said
+
cunning
+
old Fury;
+
'I'll try
+
the whole
+
cause,
+
and
+
condemn
+
you
+
to
+
death.' "
+
+ +
+ + + + + + + + \ No newline at end of file diff --git a/front/src/App.css b/front/src/App.css index 9866f71..f28b5cb 100644 --- a/front/src/App.css +++ b/front/src/App.css @@ -1,16 +1,9 @@ -body { - height: 100vh; - overflow: hidden; - color: white; - font-family: sans-serif; -} - -.modal-content, .modal-content .form-select { - background-color: rgb(17, 17, 17) !important; +:root { + --bs-body-bg: rgb(17, 17, 17) !important; } /* В связи со сложившейся политической обстановкой */ .leaflet-attribution-flag { position: absolute; right: -100px; -} +} \ No newline at end of file diff --git a/front/src/api/announcement/index.ts b/front/src/api/announcement/index.ts index e69de29..48ea148 100644 --- a/front/src/api/announcement/index.ts +++ b/front/src/api/announcement/index.ts @@ -0,0 +1,12 @@ +import { Announcement, AnnouncementResponse } from './types' + +const processAnnouncement = (data: AnnouncementResponse): Announcement => ({ + ...data, + lat: data.latitude, + lng: data.longtitude, + bestBy: data.best_by, + bookedBy: data.booked_by, + userId: data.user_id, +}) + +export { processAnnouncement } diff --git a/front/src/api/announcement/types.ts b/front/src/api/announcement/types.ts index ea04b23..449fc6f 100644 --- a/front/src/api/announcement/types.ts +++ b/front/src/api/announcement/types.ts @@ -6,7 +6,7 @@ type AnnouncementResponse = { user_id: number, name: string, category: Category, - best_by: number, + best_by: string, address: string, longtitude: number, latitude: number, @@ -14,7 +14,7 @@ type AnnouncementResponse = { src: string | null, metro: string, trashId: number | null, - booked_by: number + booked_by: number, } const isAnnouncementResponse = (obj: unknown): obj is AnnouncementResponse => ( @@ -23,7 +23,7 @@ const isAnnouncementResponse = (obj: unknown): obj is AnnouncementResponse => ( 'user_id': 'number', 'name': 'string', 'category': isCategory, - 'best_by': 'number', + 'best_by': 'string', 'address': 'string', 'longtitude': 'number', 'latitude': 'number', @@ -31,7 +31,7 @@ const isAnnouncementResponse = (obj: unknown): obj is AnnouncementResponse => ( 'src': 'string?', 'metro': 'string', 'trashId': 'number?', - 'booked_by': 'number' + 'booked_by': 'number', }) ) @@ -48,7 +48,7 @@ type Announcement = { src: string | null, metro: string, trashId: number | null, - bookedBy: number + bookedBy: number, } export type { diff --git a/front/src/api/announcements/index.ts b/front/src/api/announcements/index.ts index 2e7c3d7..22889e7 100644 --- a/front/src/api/announcements/index.ts +++ b/front/src/api/announcements/index.ts @@ -1,25 +1,17 @@ import { API_URL } from '../../config' -import { FiltersType, URLEncodeFilters } from '../../utils/filters' +import { FiltersType, URLEncodeFilters, convertFilterNames } from '../../utils/filters' +import { processAnnouncement } from '../announcement' import { Announcement } from '../announcement/types' import { AnnouncementsResponse } from './types' const initialAnnouncements: Announcement[] = [] const composeAnnouncementsURL = (filters: FiltersType) => ( - API_URL + '/announcements?' + new URLSearchParams(URLEncodeFilters(filters)).toString() + API_URL + '/announcements?' + new URLSearchParams(convertFilterNames(URLEncodeFilters(filters))).toString() ) -const processAnnouncements = (data: AnnouncementsResponse): Announcement[] => { - const annList = data.list_of_announcements - - return annList.map(ann => ({ - ...ann, - lat: ann.latitude, - lng: ann.longtitude, - bestBy: ann.best_by, - bookedBy: ann.booked_by, - userId: ann.user_id - })) -} +const processAnnouncements = (data: AnnouncementsResponse): Announcement[] => ( + data.map(processAnnouncement) +) export { initialAnnouncements, composeAnnouncementsURL, processAnnouncements } diff --git a/front/src/api/announcements/types.ts b/front/src/api/announcements/types.ts index fefd5f2..37e86cc 100644 --- a/front/src/api/announcements/types.ts +++ b/front/src/api/announcements/types.ts @@ -1,16 +1,10 @@ -import { isArrayOf, isObject } from '../../utils/types' +import { isArrayOf } from '../../utils/types' import { AnnouncementResponse, isAnnouncementResponse } from '../announcement/types' -type AnnouncementsResponse = { - list_of_announcements: AnnouncementResponse[], - Success: boolean -} +type AnnouncementsResponse = AnnouncementResponse[] const isAnnouncementsResponse = (obj: unknown): obj is AnnouncementsResponse => ( - isObject(obj, { - 'list_of_announcements': obj => isArrayOf(obj, isAnnouncementResponse), - 'Success': 'boolean' - }) + isArrayOf(obj, isAnnouncementResponse) ) export type { diff --git a/front/src/api/book/index.ts b/front/src/api/book/index.ts new file mode 100644 index 0000000..22fd424 --- /dev/null +++ b/front/src/api/book/index.ts @@ -0,0 +1,12 @@ +import { API_URL } from '../../config' +import { Book, BookResponse } from './types' + +const composeBookURL = () => ( + API_URL + '/book?' +) + +const processBook = (data: BookResponse): Book => { + return data.Success +} + +export { composeBookURL, processBook } diff --git a/front/src/api/book/types.ts b/front/src/api/book/types.ts new file mode 100644 index 0000000..c2226e6 --- /dev/null +++ b/front/src/api/book/types.ts @@ -0,0 +1,17 @@ +import { isObject } from '../../utils/types' + +type BookResponse = { + Success: boolean, +} + +const isBookResponse = (obj: unknown): obj is BookResponse => ( + isObject(obj, { + 'Success': 'boolean', + }) +) + +type Book = boolean + +export type { BookResponse, Book } + +export { isBookResponse } 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/osmAddress/index.ts b/front/src/api/osmAddress/index.ts index ec7ef9e..7933676 100644 --- a/front/src/api/osmAddress/index.ts +++ b/front/src/api/osmAddress/index.ts @@ -1,4 +1,5 @@ import { LatLng } from 'leaflet' + import { OsmAddressResponse } from './types' const initialOsmAddress = '' diff --git a/front/src/api/osmAddress/types.ts b/front/src/api/osmAddress/types.ts index 88ba31b..2013e52 100644 --- a/front/src/api/osmAddress/types.ts +++ b/front/src/api/osmAddress/types.ts @@ -1,7 +1,7 @@ import { isObject } from '../../utils/types' type OsmAddressResponse = { - display_name: string + display_name: string, } const isOsmAddressResponse = (obj: unknown): obj is OsmAddressResponse => ( diff --git a/front/src/api/poetry/index.ts b/front/src/api/poetry/index.ts new file mode 100644 index 0000000..36e60b3 --- /dev/null +++ b/front/src/api/poetry/index.ts @@ -0,0 +1,18 @@ +import { API_URL } from '../../config' +import { PoetryResponse, Poetry } from './types' + +const initialPoetry: Poetry = { + title: '', + text: '', + author: '', +} + +const composePoetryURL = () => ( + API_URL + '/poetry?' +) + +const processPoetry = (data: PoetryResponse): Poetry => { + return data +} + +export { initialPoetry, composePoetryURL, processPoetry } diff --git a/front/src/api/poetry/types.ts b/front/src/api/poetry/types.ts new file mode 100644 index 0000000..d3855dc --- /dev/null +++ b/front/src/api/poetry/types.ts @@ -0,0 +1,23 @@ +import { isObject } from '../../utils/types' + +type PoetryResponse = { + title: string, + text: string, + author: string, +} + +const isPoetryResponse = (obj: unknown): obj is PoetryResponse => ( + isObject(obj, { + 'title': 'string', + 'text': 'string', + 'author': 'string', + }) +) + +type Poetry = PoetryResponse + +const isPoetry = isPoetryResponse + +export type { PoetryResponse, Poetry } + +export { isPoetryResponse, isPoetry } diff --git a/front/src/api/putAnnouncement/types.ts b/front/src/api/putAnnouncement/types.ts index 420f8a4..9339351 100644 --- a/front/src/api/putAnnouncement/types.ts +++ b/front/src/api/putAnnouncement/types.ts @@ -1,12 +1,12 @@ import { isObject } from '../../utils/types' type PutAnnouncementResponse = { - Answer: boolean + Answer: boolean, } const isPutAnnouncementResponse = (obj: unknown): obj is PutAnnouncementResponse => ( isObject(obj, { - 'Answer': 'boolean' + 'Answer': 'boolean', }) ) diff --git a/front/src/api/removeAnnouncement/index.ts b/front/src/api/removeAnnouncement/index.ts new file mode 100644 index 0000000..dd52f7a --- /dev/null +++ b/front/src/api/removeAnnouncement/index.ts @@ -0,0 +1,16 @@ +import { API_URL } from '../../config' +import { RemoveAnnouncement, RemoveAnnouncementResponse } from './types' + +const composeRemoveAnnouncementURL = () => ( + API_URL + '/announcement?' +) + +function processRemoveAnnouncement(data: RemoveAnnouncementResponse): RemoveAnnouncement { + if (!data.Answer) { + throw new Error('Не удалось закрыть объявление') + } + + return data.Answer +} + +export { composeRemoveAnnouncementURL, processRemoveAnnouncement } diff --git a/front/src/api/removeAnnouncement/types.ts b/front/src/api/removeAnnouncement/types.ts new file mode 100644 index 0000000..016379a --- /dev/null +++ b/front/src/api/removeAnnouncement/types.ts @@ -0,0 +1,17 @@ +import { isObject } from '../../utils/types' + +type RemoveAnnouncementResponse = { + Answer: boolean, +} + +const isRemoveAnnouncementResponse = (obj: unknown): obj is RemoveAnnouncementResponse => ( + isObject(obj, { + 'Answer': 'boolean', + }) +) + +type RemoveAnnouncement = boolean + +export type { RemoveAnnouncementResponse, RemoveAnnouncement } + +export { isRemoveAnnouncementResponse } 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/signup/index.ts b/front/src/api/signup/index.ts new file mode 100644 index 0000000..003518d --- /dev/null +++ b/front/src/api/signup/index.ts @@ -0,0 +1,22 @@ +import { API_URL } from '../../config' +import { SignUp, SignUpResponse } from './types' + +const composeSignUpURL = () => ( + API_URL + '/signup?' +) + +const composeSignUpBody = (formData: FormData) => { + formData.append('nickname', formData.get('username') ?? '') + + return formData +} + +const processSignUp = (data: SignUpResponse): SignUp => { + if (!data.Success) { + throw new Error(data.Message) + } + + return true +} + +export { composeSignUpURL, composeSignUpBody, processSignUp } diff --git a/front/src/api/signup/types.ts b/front/src/api/signup/types.ts new file mode 100644 index 0000000..02dd310 --- /dev/null +++ b/front/src/api/signup/types.ts @@ -0,0 +1,23 @@ +import { isConst, isObject } from '../../utils/types' + +type SignUpResponse = { + Success: true, +} | { + Success: false, + Message: string, +} + +const isSignUpResponse = (obj: unknown): obj is SignUpResponse => ( + isObject(obj, { + 'Success': isConst(true), + }) || isObject(obj, { + 'Success': isConst(false), + 'Message': 'string', + }) +) + +type SignUp = boolean + +export type { SignUpResponse, SignUp } + +export { isSignUpResponse } diff --git a/front/src/api/token/index.ts b/front/src/api/token/index.ts new file mode 100644 index 0000000..2c14666 --- /dev/null +++ b/front/src/api/token/index.ts @@ -0,0 +1,12 @@ +import { API_URL } from '../../config' +import { Token, TokenResponse } from './types' + +const composeTokenURL = () => ( + API_URL + '/token?' +) + +const processToken = (data: TokenResponse): Token => { + return data.access_token +} + +export { composeTokenURL, processToken } diff --git a/front/src/api/token/types.ts b/front/src/api/token/types.ts new file mode 100644 index 0000000..343f511 --- /dev/null +++ b/front/src/api/token/types.ts @@ -0,0 +1,17 @@ +import { isObject } from '../../utils/types' + +type TokenResponse = { + access_token: string, +} + +const isTokenResponse = (obj: unknown): obj is TokenResponse => ( + isObject(obj, { + 'access_token': 'string', + }) +) + +type Token = string + +export type { TokenResponse, Token } + +export { isTokenResponse } diff --git a/front/src/api/trashbox/index.ts b/front/src/api/trashbox/index.ts index 0632a48..311987a 100644 --- a/front/src/api/trashbox/index.ts +++ b/front/src/api/trashbox/index.ts @@ -6,11 +6,12 @@ import { Trashbox, TrashboxResponse } from './types' const composeTrashboxURL = (position: LatLng) => ( API_URL + '/trashbox?' + new URLSearchParams({ lat: position.lat.toString(), - lng: position.lng.toString() + lng: position.lng.toString(), }).toString() ) -const processTrashbox = (data: TrashboxResponse): Trashbox[] => +const processTrashbox = (data: TrashboxResponse): Trashbox[] => ( data +) export { composeTrashboxURL, processTrashbox } diff --git a/front/src/api/trashbox/types.ts b/front/src/api/trashbox/types.ts index b2a17c0..7458354 100644 --- a/front/src/api/trashbox/types.ts +++ b/front/src/api/trashbox/types.ts @@ -1,18 +1,20 @@ import { isArrayOf, isObject, isString } from '../../utils/types' type Trashbox = { + Name: string, Lat: number, Lng: number, Address: string, - Categories: string[] + Categories: string[], } const isTrashbox = (obj: unknown): obj is Trashbox => ( isObject(obj, { + 'Name': 'string', 'Lat': 'number', 'Lng': 'number', 'Address': 'string', - 'Categories': obj => isArrayOf(obj, isString) + 'Categories': obj => isArrayOf(obj, isString), }) ) diff --git a/front/src/api/user/index.ts b/front/src/api/user/index.ts new file mode 100644 index 0000000..9f489b4 --- /dev/null +++ b/front/src/api/user/index.ts @@ -0,0 +1,27 @@ +import { API_URL } from '../../config' +import { UserResponse, User } from './types' + +import { faker } from '@faker-js/faker/locale/ru' + + +const initialUser: User = import.meta.env.DEV ? { // Temporary, until api is realized + id: Math.random() * 100, + name: faker.person.firstName() + ' ' + faker.person.lastName(), + regDate: faker.date.anytime().getTime(), + points: Math.round(Math.random() * 1000), +} : { + id: 1, + name: 'Вася пупкин', + regDate: 0, + points: 100, +} + +const composeUserURL = () => ( + API_URL + '/users/me?' +) + +const processUser = (data: UserResponse): User => { + return data +} + +export { initialUser, composeUserURL, processUser } diff --git a/front/src/api/user/types.ts b/front/src/api/user/types.ts new file mode 100644 index 0000000..ba96548 --- /dev/null +++ b/front/src/api/user/types.ts @@ -0,0 +1,31 @@ +import { isObject } from '../../utils/types' + +type User = { + id: number, + name: string, + regDate: number, + points: number, +} + +const isUser = (obj: unknown): obj is User => ( + isObject(obj, { + 'id': 'number', + 'name': 'string', + 'regDate': 'number', + 'points': 'number', + }) +) + +type UserResponse = User + +// const isUserResponse = (obj: unknown): obj is UserResponse => ( +// isObject(obj, { + +// }) +// ) + +const isUserResponse = isUser + +export type { UserResponse, User } + +export { isUserResponse, isUser } 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/assets/addIcon.svg b/front/src/assets/addIcon.svg index a6ae317..3925975 100644 --- a/front/src/assets/addIcon.svg +++ b/front/src/assets/addIcon.svg @@ -1 +1,3 @@ - \ No newline at end of file + + + diff --git a/front/src/assets/backArrow.svg b/front/src/assets/backArrow.svg new file mode 100644 index 0000000..b831a4a --- /dev/null +++ b/front/src/assets/backArrow.svg @@ -0,0 +1,4 @@ + + + diff --git a/front/src/assets/filterIcon.svg b/front/src/assets/filterIcon.svg index cd0e442..a9443e5 100644 --- a/front/src/assets/filterIcon.svg +++ b/front/src/assets/filterIcon.svg @@ -1 +1,3 @@ - \ No newline at end of file + + + \ No newline at end of file diff --git a/front/src/assets/handIcon.svg b/front/src/assets/handIcon.svg index d535741..673ab9a 100644 --- a/front/src/assets/handIcon.svg +++ b/front/src/assets/handIcon.svg @@ -1 +1,3 @@ - \ No newline at end of file + + + diff --git a/front/src/assets/handStars.svg b/front/src/assets/handStars.svg new file mode 100644 index 0000000..d172f1b --- /dev/null +++ b/front/src/assets/handStars.svg @@ -0,0 +1,7 @@ + + + + + + + diff --git a/front/src/assets/metro.ts b/front/src/assets/metro.ts index 76168a5..940713e 100644 --- a/front/src/assets/metro.ts +++ b/front/src/assets/metro.ts @@ -21,7 +21,7 @@ const stations: Record> = { 'Кировский завод', 'Автово', 'Ленинский проспект', - 'Проспект Ветеранов' + 'Проспект Ветеранов', ]), blue: new Set([ 'Парнас', @@ -41,7 +41,7 @@ const stations: Record> = { 'Парк Победы', 'Московская', 'Звёздная', - 'Купчино' + 'Купчино', ]), green: new Set([ 'Приморская', @@ -54,7 +54,7 @@ const stations: Record> = { 'Ломоносовская', 'Пролетарская', 'Обухово', - 'Рыбацкое' + 'Рыбацкое', ]), orange: new Set([ 'Спасская', @@ -64,7 +64,7 @@ const stations: Record> = { 'Новочеркасская', 'Ладожская', 'Проспект Большевиков', - 'Улица Дыбенко' + 'Улица Дыбенко', ]), violet: new Set([ 'Комендантский проспект', @@ -81,7 +81,7 @@ const stations: Record> = { 'Международная', 'Проспект славы', 'Дунайскай', - 'Шушары' + 'Шушары', ]), } diff --git a/front/src/assets/puff.svg b/front/src/assets/puff.svg index d598440..b50002d 100644 --- a/front/src/assets/puff.svg +++ b/front/src/assets/puff.svg @@ -10,4 +10,4 @@ - \ No newline at end of file + diff --git a/front/src/assets/rightAngle.svg b/front/src/assets/rightAngle.svg new file mode 100644 index 0000000..2d6f9e5 --- /dev/null +++ b/front/src/assets/rightAngle.svg @@ -0,0 +1,4 @@ + + + + diff --git a/front/src/assets/signOut.svg b/front/src/assets/signOut.svg new file mode 100644 index 0000000..0bb8512 --- /dev/null +++ b/front/src/assets/signOut.svg @@ -0,0 +1,4 @@ + + + + diff --git a/front/src/assets/userCategories.ts b/front/src/assets/userCategories.ts new file mode 100644 index 0000000..ae4b485 --- /dev/null +++ b/front/src/assets/userCategories.ts @@ -0,0 +1,35 @@ +import { Announcement } from '../api/announcement/types' +import { getId } from '../utils/auth' +import { FiltersType } from '../utils/filters' + +const userCategories = ['givingOut', 'needDispose'] as const + +type UserCategory = typeof userCategories[number] + +const UserCategoriesNames: Record = { + givingOut: 'Раздача', + needDispose: 'Нужно утилизировать', +} + +const userCategoriesInfos: Record string> = { + givingOut: (ann: Announcement) => ( + `Годен до ${new Date(ann.bestBy).toLocaleDateString('ru')}` + ), + needDispose: (ann: Announcement) => ( + `Были заинтересованы: ${ann.bookedBy} чел.` + ), +} + +const composeUserCategoriesFilters: Record FiltersType> = { + givingOut: () => ({ + userId: getId(), + expired: false, + }), + needDispose: () => ({ + userId: getId(), + expired: true, + }), +} + +export type { UserCategory } +export { userCategories, UserCategoriesNames, userCategoriesInfos, composeUserCategoriesFilters } diff --git a/front/src/assets/userIcon.svg b/front/src/assets/userIcon.svg index e2a4590..2139409 100644 --- a/front/src/assets/userIcon.svg +++ b/front/src/assets/userIcon.svg @@ -1 +1,3 @@ - \ No newline at end of file + + + diff --git a/front/src/components/AnnouncementDetails.tsx b/front/src/components/AnnouncementDetails.tsx index 352b138..d691cac 100644 --- a/front/src/components/AnnouncementDetails.tsx +++ b/front/src/components/AnnouncementDetails.tsx @@ -1,16 +1,21 @@ 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 } from '../hooks/api' +import { useBook, useRemoveAnnouncement } from '../hooks/api' import { Announcement } from '../api/announcement/types' import { iconItem } from '../utils/markerIcons' -import { CSSProperties } from 'react' +import { useId } from '../hooks' +import SelectDisposalTrashbox from './SelectDisposalTrashbox' type AnnouncementDetailsProps = { close: () => void, - announcement: Announcement + refresh: () => void, + announcement: Announcement, } const styles = { @@ -19,17 +24,93 @@ const styles = { alignItems: 'center', justifyContent: 'center', } as CSSProperties, + map: { + width: '100%', + minHeight: 300, + } as CSSProperties, } -function AnnouncementDetails({ close, announcement: { id, name, category, bestBy, description, lat, lng, address, metro } }: AnnouncementDetailsProps) { - const { handleBook, status: bookStatus } = useBook(id) +const View = ({ + announcement: { name, category, bestBy, description, lat, lng, address, metro, userId }, +}: { 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 { handleRemove, removeButton } = useRemoveAnnouncement(closeRefresh) + + const myId = useId() + + return ( + <> +

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

+ {(myId === userId) ? ( + <> + + + setDisposeShow(true)} announcement={announcement} /> + setDisposeShow(false)} style={{ zIndex: 100000 }}> + + + Утилизация + + + + ) } diff --git a/front/src/components/AuthForm.tsx b/front/src/components/AuthForm.tsx index bd5f129..3f1dd89 100644 --- a/front/src/components/AuthForm.tsx +++ b/front/src/components/AuthForm.tsx @@ -1,54 +1,77 @@ -import { FormEventHandler } from 'react' -import { Button, Form } from 'react-bootstrap' +import { FormEventHandler, useCallback } from 'react' +import { Button, ButtonGroup, Form } from 'react-bootstrap' + +import { useSignIn, useSignUp } from '../hooks/api' +import { composeSignUpBody } from '../api/signup' type AuthFormProps = { - register: boolean - handleAuth: FormEventHandler, - loading: boolean, - error: string + goBack: () => void, } -function AuthForm ({ handleAuth, register, loading, error }: AuthFormProps) { - const buttonText = loading ? 'Загрузка...' : (error || (register ? 'Зарегистрироваться' : 'Войти')) +const AuthForm = ({ goBack }: AuthFormProps) => { + const { handleSignUp, signUpButton } = useSignUp() + + const { handleSignIn, signInButton } = useSignIn() + + const handleAuth: FormEventHandler = useCallback((e) => { + e.preventDefault() + e.stopPropagation() + + const formData = new FormData(e.currentTarget) + + const register = (e.nativeEvent as SubmitEvent).submitter?.id === 'register' + + void (async () => { + const accountCreated = register ? ( + await handleSignUp(composeSignUpBody(formData)) + ) : true + + if (accountCreated) { + if (await handleSignIn(formData)) { + goBack() + } + } + + })() + }, [goBack, handleSignUp, handleSignIn]) return (
- - Почта - + + Как меня называть + - {register && <> - - Имя - - - - - Фамилия - - - } - - Пароль - + И я могу доказать, что это я + - {register && - - - - - Я согласен с условиями обработки персональных данных - - - - } + + + + + Я согласен с условиями обработки персональных данных + + + - + + + ) +} + +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/SelectDisposalTrashbox.tsx b/front/src/components/SelectDisposalTrashbox.tsx new file mode 100644 index 0000000..1bf4405 --- /dev/null +++ b/front/src/components/SelectDisposalTrashbox.tsx @@ -0,0 +1,119 @@ +import { Button, Modal } from 'react-bootstrap' +import { MapContainer, TileLayer } from 'react-leaflet' +import { CSSProperties, useState } from 'react' +import { LatLng } from 'leaflet' + +import { useDispose, useTrashboxes } from '../hooks/api' +import { UseFetchReturn, gotError, gotResponse } from '../hooks/useFetch' +import TrashboxMarkers from './TrashboxMarkers' +import { Category } from '../assets/category' +import { Trashbox } from '../api/trashbox/types' + +type SelectDisposalTrashboxProps = { + annId: number, + category: Category, + address: LatLng, + closeRefresh: () => void, +} + +type SelectedTrashbox = { + index: number, + category: string, +} + +const styles = { + map: { + width: '100%', + height: 400, + } as CSSProperties, +} + +function SelectDisposalTrashbox({ annId, category, address, closeRefresh }: SelectDisposalTrashboxProps) { + const trashboxes = useTrashboxes(address, category) + + const [selectedTrashbox, setSelectedTrashbox] = useState({ index: -1, category: '' }) + + const { handleDispose, disposeButton } = useDispose(closeRefresh) + + return ( + <> + +
+ {gotResponse(trashboxes) + ? ( + gotError(trashboxes) ? ( +

{trashboxes.error}

+ ) : ( + + + + + ) + ) : ( +
+

Загрузка...

+
+ ) + + } +
+ +
+ + + } +
    + {useMemo(() => announcements.map((ann, i) => ( +
  • + + {ann.src?.endsWith('mp4') ? ( +
  • + )), [announcements, category])} +
+ {showScrollButtons.right && + + } + +} + +export default StoriesPreview \ No newline at end of file diff --git a/front/src/components/TrashboxMarkers.tsx b/front/src/components/TrashboxMarkers.tsx index a1a625a..52a18bc 100644 --- a/front/src/components/TrashboxMarkers.tsx +++ b/front/src/components/TrashboxMarkers.tsx @@ -5,30 +5,38 @@ import { iconTrash } from '../utils/markerIcons' type TrashboxMarkersProps = { trashboxes: Trashbox[], - selectTrashbox: ({ index, category }: { index: number, category: string }) => void + selectTrashbox: ({ index, category }: { + index: number, + category: string, + }) => void, } -function TrashboxMarkers({ trashboxes, selectTrashbox }: TrashboxMarkersProps) { - return ( - <>{trashboxes.map((trashbox, index) => ( - - -

{trashbox.Address}

-

Тип мусора: <> - {trashbox.Categories.map((category, j) => - - selectTrashbox({ index, category })}> - {category} - - {(j < trashbox.Categories.length - 1) ? ', ' : ''} - - )} -

-

{trashbox.Lat.toFixed(4)}, {trashbox.Lng.toFixed(4)}

-
-
- ))} - ) -} +const TrashboxMarkers = ({ trashboxes, selectTrashbox }: TrashboxMarkersProps) => ( + <>{trashboxes.map((trashbox, index) => ( + + +

{trashbox.Name}

+

{trashbox.Address}

+

Тип мусора:{' '} + {trashbox.Categories.map((category, j) => + + { + e.preventDefault() + e.stopPropagation() + selectTrashbox({ index, category }) + }}> + {category} + + {(j < trashbox.Categories.length - 1) ? ', ' : ''} + + )} +

+

+ {trashbox.Lat.toFixed(4)}, {trashbox.Lng.toFixed(4)} +

+
+
+ ))} +) export default TrashboxMarkers diff --git a/front/src/components/index.ts b/front/src/components/index.ts index aa92c4b..b124f0c 100644 --- a/front/src/components/index.ts +++ b/front/src/components/index.ts @@ -7,3 +7,11 @@ export { default as TrashboxMarkers } from './TrashboxMarkers' export { default as WithToken } from './WithToken' export { default as ClickHandler } from './ClickHandler' export { default as AuthForm } from './AuthForm' +export { default as BackHeader } from './BackHeader' +export { default as CategoryPreview } from './CategoryPreview' +export { default as StoriesPreview } from './StoriesPreview' +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 9023aee..7e2d8f9 100644 --- a/front/src/hooks/api/index.ts +++ b/front/src/hooks/api/index.ts @@ -1,6 +1,13 @@ export { default as useAnnouncements } from './useAnnouncements' export { default as useBook } from './useBook' -export { default as useAuth } from './useAuth' export { default as useTrashboxes } from './useTrashboxes' export { default as useAddAnnouncement } from './useAddAnnouncement' export { default as useOsmAddresses } from './useOsmAddress' +export { default as useUser } from './useUser' +export { default as useRemoveAnnouncement } from './useRemoveAnnouncement' +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/useAddAnnouncement.ts b/front/src/hooks/api/useAddAnnouncement.ts index 5c60488..1b5a16d 100644 --- a/front/src/hooks/api/useAddAnnouncement.ts +++ b/front/src/hooks/api/useAddAnnouncement.ts @@ -1,31 +1,26 @@ -import { useCallback } from 'react' - -import { useSend } from '..' +import { useSendWithButton } from '..' import { composePutAnnouncementURL, processPutAnnouncement } from '../../api/putAnnouncement' import { isPutAnnouncementResponse } from '../../api/putAnnouncement/types' -import useSendButtonCaption from '../useSendButtonCaption' -const useAddAnnouncement = () => { - const { doSend, loading, error } = useSend( +function useAddAnnouncement() { + const { doSend, button } = useSendWithButton( + 'Опубликовать', + 'Опубликовано', + true, composePutAnnouncementURL(), 'PUT', true, isPutAnnouncementResponse, - processPutAnnouncement, + processPutAnnouncement ) - const { update, ...button } = useSendButtonCaption('Опубликовать', loading, error, 'Опубликовано') - - const doSendWithButton = useCallback(async (formData: FormData) => { - const data = await doSend({}, { - body: formData + function handleAdd(formData: FormData) { + void doSend({}, { + body: formData, }) - update(data) + } - return data - }, [doSend, update]) - - return { doSend: doSendWithButton, button } + return { handleAdd, addButton: button } } export default useAddAnnouncement diff --git a/front/src/hooks/api/useAnnouncements.ts b/front/src/hooks/api/useAnnouncements.ts index 1e8dbc2..1828fc6 100644 --- a/front/src/hooks/api/useAnnouncements.ts +++ b/front/src/hooks/api/useAnnouncements.ts @@ -1,7 +1,6 @@ import { useFetch } from '../' import { FiltersType } from '../../utils/filters' import { composeAnnouncementsURL, initialAnnouncements, processAnnouncements } from '../../api/announcements' - import { isAnnouncementsResponse } from '../../api/announcements/types' const useAnnouncements = (filters: FiltersType) => ( diff --git a/front/src/hooks/api/useAuth.ts b/front/src/hooks/api/useAuth.ts deleted file mode 100644 index afcebd7..0000000 --- a/front/src/hooks/api/useAuth.ts +++ /dev/null @@ -1,117 +0,0 @@ -import { useState } from 'react' - -import { API_URL } from '../../config' -import { isConst, isObject } from '../../utils/types' -import { handleHTTPErrors } from '../../utils' - -interface AuthData { - email: string, - password: string, -} - -// interface LoginData extends AuthData { } - -// interface SignUpData extends AuthData { -// name: string, -// surname: string -// } - -type SignUpResponse = { - Success: true -} | { - Success: false, - Message: string -} - -const isSignUpResponse = (obj: unknown): obj is SignUpResponse => ( - isObject(obj, { - 'Success': isConst(true) - }) || - isObject(obj, { - 'Success': isConst(false), - 'Message': 'string' - }) -) - -interface LogInResponse { - access_token: string, - token_type: 'bearer' -} - -const isLogInResponse = (obj: unknown): obj is LogInResponse => ( - isObject(obj, { - 'access_token': 'string', - 'token_type': isConst('bearer') - }) -) - -function useAuth() { - const [loading, setLoading] = useState(false) - const [error, setError] = useState('') - - async function doAuth(data: AuthData, newAccount: boolean) { - setLoading(true) - - if (newAccount) { - try { - const res = await fetch(API_URL + '/signup', { - method: 'POST', - body: JSON.stringify(data), - headers: { - 'Content-Type': 'application/json' - } - }) - - handleHTTPErrors(res) - - const signupData: unknown = await res.json() - - if (!isSignUpResponse(signupData)) { - throw new Error('Malformed server response') - } - - if (signupData.Success === false) { - throw new Error(signupData.Message) - } - - } catch (err) { - setError(err instanceof Error ? err.message : err as string) - setLoading(false) - return null - } - } - - try { - const res = await fetch(API_URL + '/auth/token' + new URLSearchParams({ - username: data.email, - password: data.password - }).toString(), { - method: 'POST', - headers: { - 'Content-Type': 'application/x-www-form-urlencoded' - } - }) - - const logInData: unknown = await res.json() - - if (!isLogInResponse(logInData)) { - throw new Error('Malformed server response') - } - - const token = logInData.access_token - - setError('') - setLoading(false) - - return token - } catch (err) { - setError(err instanceof Error ? err.message : err as string) - setLoading(false) - return null - } - } - - return { doAuth, loading, error } -} - -export default useAuth diff --git a/front/src/hooks/api/useBook.ts b/front/src/hooks/api/useBook.ts index 71fe6ed..5a1e091 100644 --- a/front/src/hooks/api/useBook.ts +++ b/front/src/hooks/api/useBook.ts @@ -1,74 +1,32 @@ -import { useState } from 'react' -import { useNavigate } from 'react-router-dom' +import { useCallback } from 'react' -import { getToken } from '../../utils/auth' -import { API_URL } from '../../config' -import { isObject } from '../../utils/types' -import { handleHTTPErrors } from '../../utils' +import { useSendWithButton } from '..' +import { composeBookURL, processBook } from '../../api/book' +import { isBookResponse } from '../../api/book/types' -type BookResponse = { - Success: boolean -} +function useBook() { + const { doSend, button } = useSendWithButton('Забронировать', + 'Забронировано', + true, + composeBookURL(), + 'POST', + true, + isBookResponse, + processBook, + ) -const isBookResponse = (obj: unknown): obj is BookResponse => ( - isObject(obj, { - 'Success': 'boolean' - }) -) + const handleBook = useCallback((id: number) => { + void doSend({}, { + body: JSON.stringify({ + id, + }), + headers: { + 'Content-Type': 'application/json', + }, + }) + }, [doSend]) -type BookStatus = '' | 'Загрузка...' | 'Забронировано' | 'Ошибка бронирования' - -function useBook(id: number) { - const navigate = useNavigate() - - const [status, setStatus] = useState('') - - const handleBook = async () => { - const token = getToken() - - if (token) { - setStatus('Загрузка...') - - try { - - const res = await fetch(API_URL + '/book', { - method: 'POST', - body: JSON.stringify({ - id: id - }), - headers: { - 'Authorization': 'Bearer ' + token, - 'Content-Type': 'application/json' - } - }) - - handleHTTPErrors(res) - - const data: unknown = await res.json() - - if (!isBookResponse(data)) { - throw new Error('Malformed server response') - } - - if (data.Success === true) { - setStatus('Забронировано') - } else { - throw new Error('Server refused to book') - } - } - catch (err) { - setStatus('Ошибка бронирования') - - if (import.meta.env.DEV) { - console.log(err) - } - } - } else { - return navigate('/login') - } - } - - return { handleBook, status } + return { handleBook, bookButton: button } } export default useBook diff --git a/front/src/hooks/api/useDispose.ts b/front/src/hooks/api/useDispose.ts new file mode 100644 index 0000000..957e20b --- /dev/null +++ b/front/src/hooks/api/useDispose.ts @@ -0,0 +1,35 @@ +import { useCallback } from 'react' + +import { useSendWithButton } from '..' +import { composeDisposeBody, composeDisposeURL, processDispose } from '../../api/dispose' +import { DisposeParams, isDisposeResponse } from '../../api/dispose/types' + +const useDispose = (resolve: () => void) => { + const { doSend, button } = useSendWithButton( + 'Выбор сделан', + 'Зачтено', + true, + composeDisposeURL(), + 'POST', + true, + isDisposeResponse, + processDispose, + ) + + const doSendWithClose = useCallback(async (...args: DisposeParams) => { + const res = await doSend({}, { + body: composeDisposeBody(...args), + headers: { + 'Content-Type': 'application/json', + }, + }) + + if (res) { + resolve() + } + }, [doSend, resolve]) + + return { handleDispose: doSendWithClose, disposeButton: button } +} + +export default useDispose diff --git a/front/src/hooks/api/useOsmAddress.ts b/front/src/hooks/api/useOsmAddress.ts index 2c91a37..015cf6e 100644 --- a/front/src/hooks/api/useOsmAddress.ts +++ b/front/src/hooks/api/useOsmAddress.ts @@ -11,7 +11,7 @@ const useOsmAddresses = (addressPosition: LatLng) => ( false, isOsmAddressResponse, processOsmAddress, - '' + '', ) ) diff --git a/front/src/hooks/api/usePoetry.ts b/front/src/hooks/api/usePoetry.ts new file mode 100644 index 0000000..6547f63 --- /dev/null +++ b/front/src/hooks/api/usePoetry.ts @@ -0,0 +1,122 @@ +import { Poetry } from '../../api/poetry/types' +import { UseFetchReturn } from '../useFetch' + +const testPoetry: Poetry = { + title: 'The Mouse\'s Tale', + text: `
"Fury said to
+
a mouse, That
+
he met
+
in the
+
house,
+
'Let us
+
both go
+
to law:
+
I will
+
prosecute
+
you.
+
Come, I'll
+
take no
+
denial;
+
We must
+
have a
+
trial:
+
For
+
really
+
this
+
morning
+
I've
+
nothing
+
to do.'
+
Said the
+
mouse to
+
the cur,
+
'Such a
+
trial,
+
dear sir,
+
With no
+
jury or
+
judge,
+
would be
+
wasting
+
our breath.'
+
'I'll be
+
judge,
+
I'll be
+
jury,'
+
Said
+
cunning
+
old Fury;
+
'I'll try
+
the whole
+
cause,
+
and
+
condemn
+
you
+
to
+
death.' "
`, + author: 'Lewis Carroll', +} + +function usePoetry(): UseFetchReturn { + return ( + // useFetch( + // composePoetryURL(), + // 'GET', + // false, + // isPoetryResponse, + // processPoetry, + // initialPoetry, + // ) + + { + data: testPoetry, + loading: false, + error: null, + refetch: () => { return }, + } + + // { + // data: undefined, + // loading: false, + // error: 'хе-хе', + // refetch: () => { return }, + // } + + // { + // data: undefined, + // loading: true, + // error: null, + // refetch: () => { return }, + // } + ) +} + +export default usePoetry diff --git a/front/src/hooks/api/useRemoveAnnouncement.ts b/front/src/hooks/api/useRemoveAnnouncement.ts new file mode 100644 index 0000000..ed9a92a --- /dev/null +++ b/front/src/hooks/api/useRemoveAnnouncement.ts @@ -0,0 +1,37 @@ +import { useCallback } from 'react' + +import { useSendWithButton } from '..' +import { composeRemoveAnnouncementURL, processRemoveAnnouncement } from '../../api/removeAnnouncement' +import { isRemoveAnnouncementResponse } from '../../api/removeAnnouncement/types' + +const useRemoveAnnouncement = (resolve: () => void) => { + const { doSend, button } = useSendWithButton( + 'Закрыть объявление', + 'Закрыто', + true, + composeRemoveAnnouncementURL(), + 'DELETE', + true, + isRemoveAnnouncementResponse, + processRemoveAnnouncement, + ) + + const doSendWithClose = useCallback(async (id: number) => { + const res = await doSend({}, { + body: JSON.stringify({ + id, + }), + headers: { + 'Content-Type': 'application/json', + }, + }) + + if (res) { + resolve() + } + }, [doSend, resolve]) + + return { handleRemove: doSendWithClose, removeButton: button } +} + +export default useRemoveAnnouncement 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/useSignIn.ts b/front/src/hooks/api/useSignIn.ts new file mode 100644 index 0000000..7cf7e32 --- /dev/null +++ b/front/src/hooks/api/useSignIn.ts @@ -0,0 +1,35 @@ +import { useSendWithButton } from '..' +import { composeTokenURL, processToken } from '../../api/token' +import { isTokenResponse } from '../../api/token/types' +import { setToken } from '../../utils/auth' + +function useSignIn() { + const { doSend, button } = useSendWithButton( + 'Мы уже знакомы', + 'Войдено', + false, + composeTokenURL(), + 'POST', + false, + isTokenResponse, + processToken, + ) + + async function handleSignIn(formData: FormData) { + const token = await doSend({}, { + body: formData, + }) + + if (token !== undefined) { + setToken(token) + + return true + } + + return false + } + + return { handleSignIn, signInButton: button } +} + +export default useSignIn diff --git a/front/src/hooks/api/useSignUp.ts b/front/src/hooks/api/useSignUp.ts new file mode 100644 index 0000000..2bb95b7 --- /dev/null +++ b/front/src/hooks/api/useSignUp.ts @@ -0,0 +1,28 @@ +import { useSendWithButton } from '..' +import { composeSignUpURL, processSignUp } from '../../api/signup' +import { isSignUpResponse } from '../../api/signup/types' + +function useSignUp() { + const { doSend, button } = useSendWithButton( + 'Я здесь впервые', + 'Зарегистрирован', + false, + composeSignUpURL(), + 'POST', + false, + isSignUpResponse, + processSignUp, + ) + + async function handleSignUp(formData: FormData) { + const res = await doSend({}, { + body: formData, + }) + + return res ?? false + } + + return { handleSignUp, signUpButton: button } +} + +export default useSignUp diff --git a/front/src/hooks/api/useTrashboxes.ts b/front/src/hooks/api/useTrashboxes.ts index 826062b..7e28027 100644 --- a/front/src/hooks/api/useTrashboxes.ts +++ b/front/src/hooks/api/useTrashboxes.ts @@ -1,18 +1,41 @@ import { LatLng } from 'leaflet' +import { sampleSize, random } from 'lodash' -import { useFetch } from '../' -import { composeTrashboxURL, processTrashbox } from '../../api/trashbox' -import { isTrashboxResponse } from '../../api/trashbox/types' +import { Trashbox } from '../../api/trashbox/types' +import { UseFetchReturn } from '../useFetch' -const useTrashboxes = (position: LatLng) => ( - useFetch( - composeTrashboxURL(position), - 'GET', - true, - isTrashboxResponse, - processTrashbox, - [] - ) +import { faker } from '@faker-js/faker/locale/ru' +import { Category, categories } from '../../assets/category' +import { useCallback, useMemo } from 'react' + +function genMockTrashbox(pos: LatLng): Trashbox { + const loc = faker.location.nearbyGPSCoordinate({ origin: [pos.lat, pos.lng], radius: 1 }) + + return { + Name: faker.company.name(), + Address: faker.location.streetAddress(), + Categories: faker.lorem.words({ max: 3, min: 1 }).split(' '), + Lat: loc[0], + Lng: loc[1], + } +} + +const useTrashboxes = (position: LatLng, category: Category): UseFetchReturn => ( + // useFetch( + // composeTrashboxURL(position, category), + // 'GET', + // true, + // isTrashboxResponse, + // processTrashbox, + // [], + // ) + + { + data: useMemo(() => new Array(3).fill(3).map(() => genMockTrashbox(position)), [position]), + loading: false, + error: null, + refetch: () => { return }, + } ) export default useTrashboxes diff --git a/front/src/hooks/api/useUser.ts b/front/src/hooks/api/useUser.ts new file mode 100644 index 0000000..758ba8f --- /dev/null +++ b/front/src/hooks/api/useUser.ts @@ -0,0 +1,23 @@ +import { initialUser } from '../../api/user' +import { User } from '../../api/user/types' +import { UseFetchReturn } from '../useFetch' + +const useUser = (): UseFetchReturn => ( + // useFetch( + // composeUserURL(), + // 'GET', + // true, + // isUserResponse, + // processUser, + // initialUser + // ) + + { + data: initialUser, + loading: false, + error: null, + refetch: () => { return }, + } +) + +export default useUser 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/hooks/index.ts b/front/src/hooks/index.ts index e1c5fa4..60cc24c 100644 --- a/front/src/hooks/index.ts +++ b/front/src/hooks/index.ts @@ -1,3 +1,8 @@ export { default as useStoryDimensions } from './useStoryDimensions' export { default as useSend } from './useSend' export { default as useFetch } from './useFetch' +export { default as useStoryIndex } from './useStoryIndex' +export { default as useFilters } from './useFilters' +export { default as useSendWithButton } from './useSendWithButton' +export { default as useSendButtonCaption } from './useSendButtonCaption' +export { default as useId } from './useId' diff --git a/front/src/hooks/useFetch.ts b/front/src/hooks/useFetch.ts index 41e9ab6..82a5106 100644 --- a/front/src/hooks/useFetch.ts +++ b/front/src/hooks/useFetch.ts @@ -1,75 +1,107 @@ import { useEffect, useState } from 'react' -import { SetState } from '../utils/types' import useSend from './useSend' type UseFetchShared = { - loading: boolean, abort?: () => void, + refetch: () => void, } type UseFetchSucced = { - error: null, data: T, + loading: false, + error: null, +} & UseFetchShared + +type UseFetchLoading = { + data: undefined, + loading: true, + error: null, } & UseFetchShared type UseFetchErrored = { + data: undefined, + loading: false, error: string, - data: undefined } & UseFetchShared -const gotError = (res: UseFetchErrored | UseFetchSucced): res is UseFetchErrored => ( +type UseFetchReturn = UseFetchSucced | UseFetchLoading | UseFetchErrored + +const gotError = (res: UseFetchReturn): res is UseFetchErrored => ( typeof res.error === 'string' ) -const fallbackError = (res: UseFetchSucced | UseFetchErrored) => ( - gotError(res) ? res.error : res.data -) - -type UseFetchReturn = ({ - error: null, - data: T -} | { - error: string, - data: undefined -}) & { - loading: boolean, - setData: SetState - abort?: (() => void) +function fallbackError(res: UseFetchSucced | UseFetchErrored): T | string +function fallbackError(res: UseFetchReturn): T | string | undefined +function fallbackError(res: UseFetchReturn): T | string | undefined { + return ( + gotError(res) ? res.error : res.data + ) } -function useFetch( +const gotResponse = (res: UseFetchReturn): res is UseFetchSucced | UseFetchErrored => ( + !res.loading +) + +function useFetch>( url: string, method: RequestInit['method'], needAuth: boolean, guardResponse: (data: unknown) => data is R, processResponse: (data: R) => T, initialData?: T, - params?: Omit + params?: Omit, ): UseFetchReturn { const [data, setData] = useState(initialData) - const { doSend, loading, error } = useSend(url, method, needAuth, guardResponse, processResponse, params) + const { doSend, loading, error } = useSend( + url, + method, + needAuth, + guardResponse, + processResponse, + true, + params, + ) - useEffect(() => { + function refetch() { doSend().then( data => { if (data !== undefined) setData(data) } ).catch( // must never occur err => import.meta.env.DEV && console.error('Failed to do fetch request', err) ) - }, [doSend]) + } + + useEffect(refetch, [doSend]) + + if (loading === true) { + return { + data: undefined, + loading, + error: null, + refetch, + } + } + + if (error !== null) { + return { + data: undefined, + loading, + error, + refetch, + } + } return { - ...( - error === null ? ({ - data: data!, error: null - }) : ({ data: undefined, error }) - ), + data: data!, loading, - setData + error, + refetch, } } +export type { UseFetchReturn } + export default useFetch -export { gotError, fallbackError } +export { gotError, gotResponse, fallbackError } diff --git a/front/src/hooks/useFilters.ts b/front/src/hooks/useFilters.ts new file mode 100644 index 0000000..c41a305 --- /dev/null +++ b/front/src/hooks/useFilters.ts @@ -0,0 +1,48 @@ +import { useEffect, useState } from 'react' +import { useSearchParams } from 'react-router-dom' + +import { FiltersType, URLDecoreFilters, URLEncodeFilters, defaultFilters, excludeFilters } from '../utils/filters' +import { SetState } from '../utils/types' + +function useFilters(initialFilters: FiltersType = defaultFilters): [FiltersType, SetState] { + const [searchParams, setSearchParams] = useSearchParams() + + const [filters, setFilters] = useState(initialFilters) + + const appendFiltersSearchParams = (filters: FiltersType, replace = false) => ( + setSearchParams(params => ({ + ...excludeFilters(Object.fromEntries(params)), + ...URLEncodeFilters(filters), + }), { replace }) + ) + + useEffect(() => { + const urlFilters = URLDecoreFilters(searchParams) + + setFilters(prev => ({ + ...prev, + ...urlFilters, + })) + + appendFiltersSearchParams({ + ...URLEncodeFilters(initialFilters), + ...URLEncodeFilters(urlFilters), + }, true) + // searchParams have actual query string at first render + // eslint-disable-next-line react-hooks/exhaustive-deps + }, []) + + const withQuery = (f: SetState) => ( + (nextInit: (FiltersType | ((prev: FiltersType) => FiltersType))) => { + const newFilters = (typeof nextInit === 'function') ? nextInit(filters) : nextInit + + appendFiltersSearchParams(newFilters) + + f(nextInit) + } + ) + + return [filters, withQuery(setFilters)] +} + +export default useFilters diff --git a/front/src/hooks/useId.ts b/front/src/hooks/useId.ts new file mode 100644 index 0000000..c7b28c1 --- /dev/null +++ b/front/src/hooks/useId.ts @@ -0,0 +1,17 @@ +import { useNavigate } from 'react-router-dom' + +import { getId } from '../utils/auth' + +function useId(require = false) { + const navigate = useNavigate() + + const id = getId() + + if (require && id < 0) { + navigate('/login') + } + + return id +} + +export default useId diff --git a/front/src/hooks/useSend.ts b/front/src/hooks/useSend.ts index 6928512..1742ac5 100644 --- a/front/src/hooks/useSend.ts +++ b/front/src/hooks/useSend.ts @@ -4,15 +4,16 @@ import { useNavigate } from 'react-router-dom' import { getToken } from '../utils/auth' import { handleHTTPErrors, isAborted } from '../utils' -function useSend( +function useSend>( url: string, method: RequestInit['method'], needAuth: boolean, guardResponse: (data: unknown) => data is R, processResponse: (data: R) => T, - defaultParams?: Omit + startWithLoading = false, + defaultParams?: Omit, ) { - const [loading, setLoading] = useState(false) + const [loading, setLoading] = useState(startWithLoading) const [error, setError] = useState(null) const navigate = useNavigate() @@ -35,7 +36,7 @@ function useSend( const headers = new Headers({ ...defaultParams?.headers, - ...params?.headers + ...params?.headers, }) if (needAuth) { @@ -47,7 +48,7 @@ function useSend( return undefined } - headers.append('Auth', `Bearer ${token}`) + headers.append('Authorization', `Bearer ${token}`) } try { @@ -73,10 +74,14 @@ function useSend( } catch (err) { if (err instanceof Error && !isAborted(err)) { - setError('Ошибка сети') + if (err instanceof TypeError) { + setError('Ошибка сети') + } else { + setError(err.message) + } if (import.meta.env.DEV) { - console.log(url, params, err) + console.error(url, params, err) } } @@ -88,7 +93,7 @@ function useSend( return { doSend, loading, error, - abort: abortControllerRef.current?.abort.bind(abortControllerRef.current) + abort: abortControllerRef.current?.abort.bind(abortControllerRef.current), } } diff --git a/front/src/hooks/useSendButtonCaption.ts b/front/src/hooks/useSendButtonCaption.ts index cb209b2..4ab2f81 100644 --- a/front/src/hooks/useSendButtonCaption.ts +++ b/front/src/hooks/useSendButtonCaption.ts @@ -5,7 +5,7 @@ function useSendButtonCaption( loading: boolean, error: string | null, result = initial, - singular = true + singular = true, ) { const [caption, setCaption] = useState(initial) const [disabled, setDisabled] = useState(false) diff --git a/front/src/hooks/useSendWithButton.ts b/front/src/hooks/useSendWithButton.ts new file mode 100644 index 0000000..8730664 --- /dev/null +++ b/front/src/hooks/useSendWithButton.ts @@ -0,0 +1,27 @@ +import { useCallback } from 'react' + +import { useSend } from '.' +import useSendButtonCaption from './useSendButtonCaption' + +function useSendWithButton>( + initial: string, + result: string, + singular?: boolean, + ...useSendArgs: Parameters> +) { + const { doSend, loading, error } = useSend(...useSendArgs) + + const { update, ...button } = useSendButtonCaption(initial, loading, error, result, singular) + + const doSendWithButton = useCallback(async (...args: Parameters) => { + const data = await doSend(...args) + + update(data) + + return data + }, [doSend, update]) + + return { doSend: doSendWithButton, button } +} + +export default useSendWithButton diff --git a/front/src/hooks/useStoryDimensions.ts b/front/src/hooks/useStoryDimensions.ts index 7d69655..7e684d0 100644 --- a/front/src/hooks/useStoryDimensions.ts +++ b/front/src/hooks/useStoryDimensions.ts @@ -3,7 +3,7 @@ import { useState, useEffect } from 'react'; const getWindowDimensions = () => ( { width: window.innerWidth, - height: window.innerHeight + height: window.innerHeight, } ) @@ -12,20 +12,22 @@ function useStoryDimensions(maxRatio = 16 / 9) { useEffect(() => { function handleResize() { - setWindowDimensions(getWindowDimensions()); + setWindowDimensions(getWindowDimensions()) } - + window.addEventListener('resize', handleResize); - return () => window.removeEventListener('resize', handleResize); + return () => window.removeEventListener('resize', handleResize) }, []); - const height = windowDimensions.height - 56 + const bottomBarHeight = 56 + + const height = windowDimensions.height - bottomBarHeight const ratio = Math.max(maxRatio, height / windowDimensions.width) return { height: height, - width: Math.round(height / ratio) + width: Math.round(height / ratio), } } diff --git a/front/src/hooks/useStoryIndex.ts b/front/src/hooks/useStoryIndex.ts new file mode 100644 index 0000000..84c1fe6 --- /dev/null +++ b/front/src/hooks/useStoryIndex.ts @@ -0,0 +1,56 @@ +import { useEffect, useState } from 'react' +import { useSearchParams } from 'react-router-dom' + +import { SetState } from '../utils/types' + +function useStoryIndex(annLength: number | undefined) { + const [index, setIndex] = useState(0) + + const [searchParams, setSearchParams] = useSearchParams() + + const withReset = (f: SetState) => (...args: Parameters>) => { + setIndex(0) + setSearchParams(prev => ({ + ...Object.fromEntries(prev), + storyIndex: '0', + }), { replace: true }) + f(...args) + } + + useEffect(() => { + setIndex(annLength ? + Number.parseInt(searchParams.get('storyIndex') || '0') : + 0) + // searchParams have actual query string at first render + // eslint-disable-next-line react-hooks/exhaustive-deps + }, [annLength]) + + const increment = () => setIndex(prev => { + const newIndex = (prev + 1) % (annLength || 1) + setSearchParams(prev => ({ + ...Object.fromEntries(prev), + storyIndex: newIndex.toString(), + }), { replace: true }) + + return newIndex + }) + + const decrement = () => setIndex(prev => { + const newIndex = prev > 0 ? (prev - 1) : 0 + setSearchParams(prev => ({ + ...Object.fromEntries(prev), + storyIndex: newIndex.toString(), + }), { replace: true }) + + return newIndex + }) + + return { + n: index, + withReset, + increment, + decrement, + } +} + +export default useStoryIndex diff --git a/front/src/main.tsx b/front/src/main.tsx index 20632c1..79cbf0e 100644 --- a/front/src/main.tsx +++ b/front/src/main.tsx @@ -2,6 +2,7 @@ import React from 'react' import ReactDOM from 'react-dom/client' import App from './App.tsx' + import './index.css' import 'bootstrap/dist/css/bootstrap.min.css' diff --git a/front/src/pages/AddPage.tsx b/front/src/pages/AddPage.tsx index ac4c292..61c180d 100644 --- a/front/src/pages/AddPage.tsx +++ b/front/src/pages/AddPage.tsx @@ -1,22 +1,18 @@ -import { CSSProperties, FormEventHandler, useState } from 'react' -import { Form, Button, Card } from 'react-bootstrap' +import { CSSProperties, FormEventHandler, useEffect, useState } from 'react' +import { Form, Button } from 'react-bootstrap' import { MapContainer, TileLayer } from 'react-leaflet' import { latLng } from 'leaflet' +import { useNavigate } from 'react-router-dom' -import { ClickHandler, LocationMarker, TrashboxMarkers } from '../components' -import { useAddAnnouncement, useTrashboxes } from '../hooks/api' +import { ClickHandler, LocationMarker } from '../components' +import { useAddAnnouncement } from '../hooks/api' import { categories, categoryNames } from '../assets/category' import { stations, lines, lineNames } from '../assets/metro' -import { fallbackError, gotError } from '../hooks/useFetch' +import { fallbackError, gotResponse } from '../hooks/useFetch' import { useOsmAddresses } from '../hooks/api' +import CardLayout from '../components/CardLayout' const styles = { - modal: { - height: 'calc(100vh - 3rem)', - } as CSSProperties, - body: { - overflowY: 'auto', - } as CSSProperties, map: { width: '100%', height: 400, @@ -26,12 +22,11 @@ const styles = { function AddPage() { const [addressPosition, setAddressPosition] = useState(latLng(59.972, 30.3227)) - const trashboxes = useTrashboxes(addressPosition) - const [selectedTrashbox, setSelectedTrashbox] = useState({ index: -1, category: '' }) - const address = useOsmAddresses(addressPosition) - const { doSend, button } = useAddAnnouncement() + const { handleAdd, addButton } = useAddAnnouncement() + + const navigate = useNavigate() const handleSubmit: FormEventHandler = (event) => { event.preventDefault() @@ -44,142 +39,104 @@ function AddPage() { formData.append('address', address.data || '') // if address.error formData.set('bestBy', new Date((formData.get('bestBy') as number | null) || 0).getTime().toString()) - void doSend(formData) + handleAdd(formData) } + useEffect(() => { + if (addButton.children === 'Опубликовано') { + navigate('/') + } + }, [addButton.children, navigate]) + return ( - - -
- - Заголовок объявления - - + + + + Заголовок объявления + + - - Категория - - - {categories.map(category => - - )} - - - - - Срок годности - - - - - Адрес выдачи -
- - - - - -
-

Адрес: {fallbackError(address)}

-
- - - Описание - - - - - Иллюстрация (фото или видео) - - - - - Станция метро - - - - {lines.map( - line => - - {Array.from(stations[line]).map(metro => - - )} - - )} - - - - - Пункт сбора мусора -
- {trashboxes.loading - ? ( -
-

Загрузка...

-
- ) : ( - gotError(trashboxes) ? ( -

{trashboxes.error}

- ) : ( - - - - - ) - ) - } -
- {!gotError(trashboxes) && selectedTrashbox.index > -1 ? ( -

Выбран пункт сбора мусора на { - trashboxes.data[selectedTrashbox.index].Address - } с категорией {selectedTrashbox.category}

- ) : ( -

Выберите пунк сбора мусора и категорию

+ + Категория + + + {categories.map(category => + )} - + +
-