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) ? (
+ <>
+
+