remove admin
This commit is contained in:
@@ -18,8 +18,6 @@ import { PolitikaCookiePage } from '@pages/politika-cookie'
|
||||
import { SoglasiePage } from '@pages/soglasie-personalnyh-dannyh'
|
||||
import { ReestryPage } from '@pages/reestr-pd-rkn'
|
||||
import { TransactionsPage } from '@pages/transactions'
|
||||
import { AdminPage } from '@pages/admin'
|
||||
import { AdminOrganizationPage } from '@pages/admin-organization'
|
||||
import { WalletLayout } from '@widgets/wallet-layout'
|
||||
import { ROUTES } from '@shared/config/routes'
|
||||
import { ScrollToTop } from './ScrollToTop'
|
||||
@@ -40,10 +38,6 @@ export function RouterProvider() {
|
||||
<Route path={ROUTES.REGISTER_TEST} element={<RegisterTestPage />} />
|
||||
<Route path={ROUTES.CONVERTER_TEST} element={<ConverterTestPage />} />
|
||||
|
||||
{/* Admin panel — own auth gate, independent of the user auth system */}
|
||||
<Route path={ROUTES.ADMIN} element={<AdminPage />} />
|
||||
<Route path={ROUTES.ADMIN_ORGANIZATION} element={<AdminOrganizationPage />} />
|
||||
|
||||
<Route element={<GuestRoute />}>
|
||||
<Route path={ROUTES.LOGIN} element={<LoginPage />} />
|
||||
<Route path={ROUTES.REGISTER} element={<RegisterPage />} />
|
||||
|
||||
@@ -1,219 +0,0 @@
|
||||
import type {
|
||||
AdminLoginRequest,
|
||||
AdminLoginResponse,
|
||||
AdminMeResponse,
|
||||
AdminRefreshResponse,
|
||||
CreateOrganizationRequest,
|
||||
CreateWalletsResponse,
|
||||
WalletResponse,
|
||||
DocumentResponse,
|
||||
DocumentTypeSlug,
|
||||
Organization,
|
||||
OrganizationListResponse,
|
||||
PurchaseRequestListResponse,
|
||||
UpdateOrganizationRequest,
|
||||
} from '../model/types'
|
||||
|
||||
const ADMIN_API_URL = 'https://app.admin.elcsa.ru'
|
||||
|
||||
// In-memory admin access token — deliberately separate from the user `tokenStore`
|
||||
// so the two independent auth systems never collide. No CSRF on the admin API.
|
||||
let adminToken: string | null = null
|
||||
|
||||
export const adminTokenStore = {
|
||||
get: () => adminToken,
|
||||
set: (token: string) => { adminToken = token },
|
||||
clear: () => { adminToken = null },
|
||||
}
|
||||
|
||||
// The refresh token is body-based (sent in the `/v1/auth/refresh` request body),
|
||||
// so it must persist across reloads to restore a session. localStorage-backed.
|
||||
const ADMIN_REFRESH_KEY = 'admin_refresh_token'
|
||||
|
||||
export const adminRefreshStore = {
|
||||
get: (): string | null => {
|
||||
try { return localStorage.getItem(ADMIN_REFRESH_KEY) } catch { return null }
|
||||
},
|
||||
set: (token: string) => {
|
||||
try { localStorage.setItem(ADMIN_REFRESH_KEY, token) } catch { /* ignore */ }
|
||||
},
|
||||
clear: () => {
|
||||
try { localStorage.removeItem(ADMIN_REFRESH_KEY) } catch { /* ignore */ }
|
||||
},
|
||||
}
|
||||
|
||||
async function doAdminRequest<T>(
|
||||
path: string,
|
||||
options: RequestInit,
|
||||
allowRetry: boolean,
|
||||
): Promise<T> {
|
||||
const bearer = adminTokenStore.get()
|
||||
// For multipart uploads we must NOT set Content-Type — the browser adds the
|
||||
// boundary itself. Detect FormData bodies and skip the JSON header.
|
||||
const isFormData = options.body instanceof FormData
|
||||
|
||||
const res = await fetch(`${ADMIN_API_URL}${path}`, {
|
||||
...options,
|
||||
credentials: 'include',
|
||||
headers: {
|
||||
...(isFormData ? {} : { 'Content-Type': 'application/json' }),
|
||||
...(bearer ? { Authorization: `Bearer ${bearer}` } : {}),
|
||||
...options.headers,
|
||||
},
|
||||
})
|
||||
|
||||
if (res.status === 401 && allowRetry) {
|
||||
try {
|
||||
await refreshAdminToken()
|
||||
return doAdminRequest<T>(path, options, false)
|
||||
} catch {
|
||||
adminTokenStore.clear()
|
||||
throw new Error('Unauthorized')
|
||||
}
|
||||
}
|
||||
|
||||
const data = await res.json().catch(() => null)
|
||||
if (!res.ok) throw data
|
||||
return data as T
|
||||
}
|
||||
|
||||
// Body-based refresh: the API expects the refresh token in the request body and
|
||||
// returns a fresh access + refresh token pair (the refresh token rotates).
|
||||
export async function refreshAdminToken(): Promise<string> {
|
||||
const refreshToken = adminRefreshStore.get()
|
||||
if (!refreshToken) throw new Error('Unauthorized')
|
||||
|
||||
const res = await fetch(`${ADMIN_API_URL}/v1/auth/refresh`, {
|
||||
method: 'POST',
|
||||
credentials: 'include',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ refresh_token: refreshToken }),
|
||||
})
|
||||
if (!res.ok) {
|
||||
adminRefreshStore.clear()
|
||||
adminTokenStore.clear()
|
||||
throw new Error('Unauthorized')
|
||||
}
|
||||
const data = (await res.json()) as AdminRefreshResponse
|
||||
if (data.access_token) adminTokenStore.set(data.access_token)
|
||||
if (data.refresh_token) adminRefreshStore.set(data.refresh_token)
|
||||
return data.access_token
|
||||
}
|
||||
|
||||
export async function adminLogin(payload: AdminLoginRequest): Promise<AdminLoginResponse> {
|
||||
const data = await doAdminRequest<AdminLoginResponse>(
|
||||
'/v1/auth/login',
|
||||
{ method: 'POST', body: JSON.stringify(payload) },
|
||||
false,
|
||||
)
|
||||
if (data.access_token) adminTokenStore.set(data.access_token)
|
||||
if (data.refresh_token) adminRefreshStore.set(data.refresh_token)
|
||||
return data
|
||||
}
|
||||
|
||||
export function getAdminMe(): Promise<AdminMeResponse> {
|
||||
return doAdminRequest<AdminMeResponse>('/v1/auth/me', {}, true)
|
||||
}
|
||||
|
||||
export async function adminLogout(): Promise<void> {
|
||||
try {
|
||||
await doAdminRequest<unknown>('/v1/auth/logout', { method: 'POST' }, false)
|
||||
} finally {
|
||||
adminTokenStore.clear()
|
||||
adminRefreshStore.clear()
|
||||
}
|
||||
}
|
||||
|
||||
export function getOrganizations(limit = 50, offset = 0): Promise<OrganizationListResponse> {
|
||||
return doAdminRequest<OrganizationListResponse>(
|
||||
`/v1/organizations?limit=${limit}&offset=${offset}`,
|
||||
{},
|
||||
true,
|
||||
)
|
||||
}
|
||||
|
||||
export function createOrganization(payload: CreateOrganizationRequest): Promise<Organization> {
|
||||
return doAdminRequest<Organization>(
|
||||
'/v1/organizations',
|
||||
{ method: 'POST', body: JSON.stringify(payload) },
|
||||
true,
|
||||
)
|
||||
}
|
||||
|
||||
export function getOrganization(id: string): Promise<Organization> {
|
||||
return doAdminRequest<Organization>(`/v1/organizations/${id}`, {}, true)
|
||||
}
|
||||
|
||||
export function createOrganizationWallets(id: string): Promise<CreateWalletsResponse> {
|
||||
return doAdminRequest<CreateWalletsResponse>(
|
||||
`/v1/organizations/${id}/wallets/create`,
|
||||
{ method: 'POST', body: JSON.stringify({ id }) },
|
||||
true,
|
||||
)
|
||||
}
|
||||
|
||||
export function getOrganizationWallets(id: string): Promise<WalletResponse[]> {
|
||||
return doAdminRequest<WalletResponse[]>(
|
||||
`/v1/organizations/${id}/wallets`,
|
||||
{},
|
||||
true,
|
||||
)
|
||||
}
|
||||
|
||||
export function getDocuments(orgId: string): Promise<DocumentResponse[]> {
|
||||
return doAdminRequest<DocumentResponse[]>(
|
||||
`/v1/organizations/${orgId}/documents`,
|
||||
{},
|
||||
true,
|
||||
)
|
||||
}
|
||||
|
||||
// Upload/replace a single typed document. The type is part of the path and
|
||||
// the body carries only the file — PUT acts as an upsert for that slot.
|
||||
export function uploadDocument(
|
||||
orgId: string,
|
||||
type: DocumentTypeSlug,
|
||||
file: File,
|
||||
): Promise<DocumentResponse> {
|
||||
const body = new FormData()
|
||||
body.append('file', file)
|
||||
|
||||
return doAdminRequest<DocumentResponse>(
|
||||
`/v1/organizations/${orgId}/documents/${type}`,
|
||||
{ method: 'PUT', body },
|
||||
true,
|
||||
)
|
||||
}
|
||||
|
||||
export async function getPurchaseRequests(params: {
|
||||
organizationId?: string
|
||||
status?: string
|
||||
limit?: number
|
||||
offset?: number
|
||||
}): Promise<PurchaseRequestListResponse> {
|
||||
const query = new URLSearchParams()
|
||||
if (params.organizationId) query.set('organization_id', params.organizationId)
|
||||
if (params.status) query.set('status', params.status)
|
||||
query.set('limit', String(params.limit ?? 50))
|
||||
query.set('offset', String(params.offset ?? 0))
|
||||
|
||||
const data = await doAdminRequest<PurchaseRequestListResponse>(
|
||||
`/v1/purchase-requests?${query.toString()}`,
|
||||
{},
|
||||
true,
|
||||
)
|
||||
// TEMP: inspect real backend shape — especially which `status` values appear.
|
||||
console.log('[purchase-requests] list response:', data)
|
||||
return data
|
||||
}
|
||||
|
||||
export function updateOrganization(
|
||||
id: string,
|
||||
payload: UpdateOrganizationRequest,
|
||||
): Promise<Organization> {
|
||||
return doAdminRequest<Organization>(
|
||||
`/v1/organizations/${id}`,
|
||||
{ method: 'PATCH', body: JSON.stringify(payload) },
|
||||
true,
|
||||
)
|
||||
}
|
||||
@@ -1,19 +0,0 @@
|
||||
import { useQuery } from '@tanstack/react-query'
|
||||
import { getAdminMe } from '../api/adminApi'
|
||||
|
||||
export const ADMIN_AUTH_QUERY_KEY = ['admin-auth']
|
||||
|
||||
export function useAdminAuth(): { isAuthenticated: boolean; isLoading: boolean } {
|
||||
// `getAdminMe` is the real "is the session valid" check. It runs through
|
||||
// `doAdminRequest` with retry, so a 401 transparently triggers a body-based
|
||||
// refresh (using the persisted refresh token) and replays the request.
|
||||
const { data, isLoading, isError } = useQuery({
|
||||
queryKey: ADMIN_AUTH_QUERY_KEY,
|
||||
queryFn: getAdminMe,
|
||||
retry: false,
|
||||
staleTime: Infinity,
|
||||
gcTime: Infinity,
|
||||
refetchOnWindowFocus: false,
|
||||
})
|
||||
return { isAuthenticated: !!data && !isError, isLoading }
|
||||
}
|
||||
@@ -1,22 +0,0 @@
|
||||
import { useMutation, useQueryClient } from '@tanstack/react-query'
|
||||
import { adminLogin } from '../api/adminApi'
|
||||
import { ADMIN_AUTH_QUERY_KEY } from './useAdminAuth'
|
||||
|
||||
export function useAdminLogin() {
|
||||
const queryClient = useQueryClient()
|
||||
return useMutation({
|
||||
mutationFn: adminLogin,
|
||||
onSuccess: (data) => {
|
||||
// Tokens are already stored by adminLogin. Seed the gate's cache with the
|
||||
// AdminMeResponse-shaped profile (login response carries all these fields)
|
||||
// so we flip to "authenticated" without an extra /auth/me round-trip.
|
||||
queryClient.setQueryData(ADMIN_AUTH_QUERY_KEY, {
|
||||
id: data.id,
|
||||
login: data.login,
|
||||
first_name: data.first_name,
|
||||
last_name: data.last_name,
|
||||
role: data.role,
|
||||
})
|
||||
},
|
||||
})
|
||||
}
|
||||
@@ -1,14 +0,0 @@
|
||||
import { useMutation, useQueryClient } from '@tanstack/react-query'
|
||||
import { adminLogout } from '../api/adminApi'
|
||||
import { ADMIN_AUTH_QUERY_KEY } from './useAdminAuth'
|
||||
|
||||
export function useAdminLogout() {
|
||||
const queryClient = useQueryClient()
|
||||
return useMutation({
|
||||
mutationFn: adminLogout,
|
||||
onSuccess: () => {
|
||||
// Flip the gate back to "not authenticated" without triggering a /refresh refetch.
|
||||
queryClient.setQueryData(ADMIN_AUTH_QUERY_KEY, null)
|
||||
},
|
||||
})
|
||||
}
|
||||
@@ -1,13 +0,0 @@
|
||||
import { useMutation, useQueryClient } from '@tanstack/react-query'
|
||||
import { createOrganization } from '../api/adminApi'
|
||||
import { ORGANIZATIONS_QUERY_KEY } from './useOrganizations'
|
||||
|
||||
export function useCreateOrganization() {
|
||||
const queryClient = useQueryClient()
|
||||
return useMutation({
|
||||
mutationFn: createOrganization,
|
||||
onSuccess: () => {
|
||||
queryClient.invalidateQueries({ queryKey: ORGANIZATIONS_QUERY_KEY })
|
||||
},
|
||||
})
|
||||
}
|
||||
@@ -1,18 +0,0 @@
|
||||
import { useMutation, useQueryClient } from '@tanstack/react-query'
|
||||
import { createOrganizationWallets } from '../api/adminApi'
|
||||
import { ORGANIZATIONS_QUERY_KEY } from './useOrganizations'
|
||||
import { ORGANIZATION_QUERY_KEY } from './useOrganization'
|
||||
import { WALLETS_QUERY_KEY } from './useOrganizationWallets'
|
||||
|
||||
export function useCreateOrganizationWallets() {
|
||||
const queryClient = useQueryClient()
|
||||
return useMutation({
|
||||
mutationFn: (organizationId: string) => createOrganizationWallets(organizationId),
|
||||
onSuccess: (_result, organizationId) => {
|
||||
// `has_wallets` flips to true server-side — refresh list and detail view.
|
||||
queryClient.invalidateQueries({ queryKey: ORGANIZATIONS_QUERY_KEY })
|
||||
queryClient.invalidateQueries({ queryKey: ORGANIZATION_QUERY_KEY(organizationId) })
|
||||
queryClient.invalidateQueries({ queryKey: WALLETS_QUERY_KEY(organizationId) })
|
||||
},
|
||||
})
|
||||
}
|
||||
@@ -1,12 +0,0 @@
|
||||
import { useQuery } from '@tanstack/react-query'
|
||||
import { getDocuments } from '../api/adminApi'
|
||||
|
||||
export const DOCUMENTS_QUERY_KEY = (orgId: string) => ['admin-documents', orgId]
|
||||
|
||||
export function useDocuments(orgId: string | undefined) {
|
||||
return useQuery({
|
||||
queryKey: DOCUMENTS_QUERY_KEY(orgId ?? ''),
|
||||
queryFn: () => getDocuments(orgId as string),
|
||||
enabled: !!orgId,
|
||||
})
|
||||
}
|
||||
@@ -1,12 +0,0 @@
|
||||
import { useQuery } from '@tanstack/react-query'
|
||||
import { getOrganization } from '../api/adminApi'
|
||||
|
||||
export const ORGANIZATION_QUERY_KEY = (id: string) => ['admin-organization', id]
|
||||
|
||||
export function useOrganization(id: string | undefined) {
|
||||
return useQuery({
|
||||
queryKey: ORGANIZATION_QUERY_KEY(id ?? ''),
|
||||
queryFn: () => getOrganization(id as string),
|
||||
enabled: !!id,
|
||||
})
|
||||
}
|
||||
@@ -1,12 +0,0 @@
|
||||
import { useQuery } from '@tanstack/react-query'
|
||||
import { getOrganizationWallets } from '../api/adminApi'
|
||||
|
||||
export const WALLETS_QUERY_KEY = (orgId: string) => ['admin-wallets', orgId]
|
||||
|
||||
export function useOrganizationWallets(orgId: string | undefined) {
|
||||
return useQuery({
|
||||
queryKey: WALLETS_QUERY_KEY(orgId ?? ''),
|
||||
queryFn: () => getOrganizationWallets(orgId as string),
|
||||
enabled: !!orgId,
|
||||
})
|
||||
}
|
||||
@@ -1,11 +0,0 @@
|
||||
import { useQuery } from '@tanstack/react-query'
|
||||
import { getOrganizations } from '../api/adminApi'
|
||||
|
||||
export const ORGANIZATIONS_QUERY_KEY = ['admin-organizations']
|
||||
|
||||
export function useOrganizations() {
|
||||
return useQuery({
|
||||
queryKey: ORGANIZATIONS_QUERY_KEY,
|
||||
queryFn: () => getOrganizations(),
|
||||
})
|
||||
}
|
||||
@@ -1,15 +0,0 @@
|
||||
import { useQuery } from '@tanstack/react-query'
|
||||
import { getPurchaseRequests } from '../api/adminApi'
|
||||
|
||||
export const PURCHASE_REQUESTS_QUERY_KEY = (orgId: string) => [
|
||||
'admin-purchase-requests',
|
||||
orgId,
|
||||
]
|
||||
|
||||
export function usePurchaseRequests(orgId: string | undefined) {
|
||||
return useQuery({
|
||||
queryKey: PURCHASE_REQUESTS_QUERY_KEY(orgId ?? ''),
|
||||
queryFn: () => getPurchaseRequests({ organizationId: orgId }),
|
||||
enabled: !!orgId,
|
||||
})
|
||||
}
|
||||
@@ -1,16 +0,0 @@
|
||||
import { useMutation, useQueryClient } from '@tanstack/react-query'
|
||||
import { updateOrganization } from '../api/adminApi'
|
||||
import type { UpdateOrganizationRequest } from '../model/types'
|
||||
import { ORGANIZATIONS_QUERY_KEY } from './useOrganizations'
|
||||
import { ORGANIZATION_QUERY_KEY } from './useOrganization'
|
||||
|
||||
export function useUpdateOrganization(id: string) {
|
||||
const queryClient = useQueryClient()
|
||||
return useMutation({
|
||||
mutationFn: (payload: UpdateOrganizationRequest) => updateOrganization(id, payload),
|
||||
onSuccess: (data) => {
|
||||
queryClient.setQueryData(ORGANIZATION_QUERY_KEY(id), data)
|
||||
queryClient.invalidateQueries({ queryKey: ORGANIZATIONS_QUERY_KEY })
|
||||
},
|
||||
})
|
||||
}
|
||||
@@ -1,20 +0,0 @@
|
||||
import { useMutation, useQueryClient } from '@tanstack/react-query'
|
||||
import { uploadDocument } from '../api/adminApi'
|
||||
import type { DocumentTypeSlug } from '../model/types'
|
||||
import { DOCUMENTS_QUERY_KEY } from './useDocuments'
|
||||
|
||||
interface UploadArgs {
|
||||
type: DocumentTypeSlug
|
||||
file: File
|
||||
}
|
||||
|
||||
export function useUploadDocument(orgId: string) {
|
||||
const queryClient = useQueryClient()
|
||||
return useMutation({
|
||||
mutationFn: ({ type, file }: UploadArgs) =>
|
||||
uploadDocument(orgId, type, file),
|
||||
onSuccess: () => {
|
||||
queryClient.invalidateQueries({ queryKey: DOCUMENTS_QUERY_KEY(orgId) })
|
||||
},
|
||||
})
|
||||
}
|
||||
@@ -1,43 +0,0 @@
|
||||
export {
|
||||
adminLogin,
|
||||
adminLogout,
|
||||
getAdminMe,
|
||||
getOrganizations,
|
||||
getOrganization,
|
||||
createOrganization,
|
||||
createOrganizationWallets,
|
||||
getOrganizationWallets,
|
||||
updateOrganization,
|
||||
getDocuments,
|
||||
uploadDocument,
|
||||
getPurchaseRequests,
|
||||
refreshAdminToken,
|
||||
adminTokenStore,
|
||||
} from './api/adminApi'
|
||||
export type {
|
||||
AdminLoginRequest,
|
||||
AdminLoginResponse,
|
||||
AdminMeResponse,
|
||||
Organization,
|
||||
OrganizationListResponse,
|
||||
CreateOrganizationRequest,
|
||||
UpdateOrganizationRequest,
|
||||
WalletResponse,
|
||||
DocumentResponse,
|
||||
DocumentTypeSlug,
|
||||
PurchaseRequestResponse,
|
||||
PurchaseRequestListResponse,
|
||||
BankDetails,
|
||||
} from './model/types'
|
||||
export { useAdminAuth, ADMIN_AUTH_QUERY_KEY } from './hooks/useAdminAuth'
|
||||
export { useAdminLogin } from './hooks/useAdminLogin'
|
||||
export { useAdminLogout } from './hooks/useAdminLogout'
|
||||
export { useOrganizations, ORGANIZATIONS_QUERY_KEY } from './hooks/useOrganizations'
|
||||
export { useOrganization, ORGANIZATION_QUERY_KEY } from './hooks/useOrganization'
|
||||
export { useCreateOrganization } from './hooks/useCreateOrganization'
|
||||
export { useCreateOrganizationWallets } from './hooks/useCreateOrganizationWallets'
|
||||
export { useUpdateOrganization } from './hooks/useUpdateOrganization'
|
||||
export { useOrganizationWallets, WALLETS_QUERY_KEY } from './hooks/useOrganizationWallets'
|
||||
export { useDocuments, DOCUMENTS_QUERY_KEY } from './hooks/useDocuments'
|
||||
export { useUploadDocument } from './hooks/useUploadDocument'
|
||||
export { usePurchaseRequests, PURCHASE_REQUESTS_QUERY_KEY } from './hooks/usePurchaseRequests'
|
||||
@@ -1,151 +0,0 @@
|
||||
export interface AdminLoginRequest {
|
||||
login: string
|
||||
password: string
|
||||
}
|
||||
|
||||
export interface AdminLoginResponse {
|
||||
access_token: string
|
||||
refresh_token: string
|
||||
token_type: string
|
||||
id: string
|
||||
login: string
|
||||
first_name: string | null
|
||||
last_name: string | null
|
||||
role: string
|
||||
}
|
||||
|
||||
export interface AdminRefreshResponse {
|
||||
access_token: string
|
||||
refresh_token: string
|
||||
token_type?: string
|
||||
}
|
||||
|
||||
export interface AdminMeResponse {
|
||||
id: string
|
||||
login: string
|
||||
first_name: string | null
|
||||
last_name: string | null
|
||||
role: string
|
||||
}
|
||||
|
||||
export type BankDetails = Record<string, unknown>
|
||||
|
||||
export interface Organization {
|
||||
id: string
|
||||
user_id: string
|
||||
name: string
|
||||
short_name: string | null
|
||||
inn: string
|
||||
ogrn: string | null
|
||||
kpp: string | null
|
||||
legal_address: string | null
|
||||
actual_address: string | null
|
||||
bank_details: BankDetails | null
|
||||
contact_person: string | null
|
||||
contact_phone: string | null
|
||||
status: string
|
||||
kyc_verified: boolean
|
||||
kyc_verified_at: string | null
|
||||
has_wallets: boolean
|
||||
created_by: string | null
|
||||
created_at: string | null
|
||||
updated_at: string | null
|
||||
}
|
||||
|
||||
export interface OrganizationListResponse {
|
||||
items: Organization[]
|
||||
total: number
|
||||
}
|
||||
|
||||
export interface WalletResponse {
|
||||
id: string
|
||||
chain: string
|
||||
address: string
|
||||
derivation_path: string
|
||||
created_at: string | null
|
||||
}
|
||||
|
||||
export interface CreateWalletsRequest {
|
||||
id: string
|
||||
}
|
||||
|
||||
export interface CreateWalletsResponse {
|
||||
wallets: WalletResponse[]
|
||||
mnemonic: string
|
||||
}
|
||||
|
||||
// Fixed document type slugs — match the path segments of the per-type
|
||||
// PUT/GET endpoints (`/documents/{slug}`). The API no longer accepts a
|
||||
// free-text document type.
|
||||
export type DocumentTypeSlug =
|
||||
| 'charter'
|
||||
| 'inn-certificate'
|
||||
| 'ogrn-certificate'
|
||||
| 'bank-details'
|
||||
| 'kyc-representative'
|
||||
| 'power-of-attorney'
|
||||
| 'other'
|
||||
|
||||
export interface DocumentResponse {
|
||||
organization_id: string
|
||||
document_type: string
|
||||
s3_key: string | null
|
||||
file_name: string | null
|
||||
content_type: string | null
|
||||
file_size_bytes: number | null
|
||||
download_url: string | null
|
||||
}
|
||||
|
||||
// Monetary fields are strings to preserve decimal precision — do not coerce to number.
|
||||
export interface PurchaseRequestResponse {
|
||||
id: string
|
||||
organization_id: string
|
||||
status: string
|
||||
usdt_amount: string
|
||||
rub_amount: string | null
|
||||
exchange_rate: string | null
|
||||
service_fee_percent: string | null
|
||||
comment: string | null
|
||||
admin_comment: string | null
|
||||
target_wallet_chain: string | null
|
||||
target_wallet_address: string | null
|
||||
tx_hash: string | null
|
||||
assigned_to: string | null
|
||||
created_at: string | null
|
||||
updated_at: string | null
|
||||
completed_at: string | null
|
||||
}
|
||||
|
||||
export interface PurchaseRequestListResponse {
|
||||
items: PurchaseRequestResponse[]
|
||||
total: number
|
||||
}
|
||||
|
||||
export interface UpdateOrganizationRequest {
|
||||
name?: string | null
|
||||
short_name?: string | null
|
||||
ogrn?: string | null
|
||||
kpp?: string | null
|
||||
legal_address?: string | null
|
||||
actual_address?: string | null
|
||||
bank_details?: BankDetails | null
|
||||
contact_person?: string | null
|
||||
contact_phone?: string | null
|
||||
status?: string | null
|
||||
}
|
||||
|
||||
export interface CreateOrganizationRequest {
|
||||
email: string
|
||||
password: string
|
||||
name: string
|
||||
inn: string
|
||||
short_name?: string | null
|
||||
ogrn?: string | null
|
||||
kpp?: string | null
|
||||
legal_address?: string | null
|
||||
actual_address?: string | null
|
||||
bank_details?: BankDetails | null
|
||||
contact_person?: string | null
|
||||
contact_phone?: string | null
|
||||
status?: string
|
||||
}
|
||||
@@ -1 +0,0 @@
|
||||
export { AdminOrganizationPage } from './ui/AdminOrganizationPage'
|
||||
@@ -1,90 +0,0 @@
|
||||
import { useEffect, useState } from 'react'
|
||||
import { useUpdateOrganization } from '@features/admin'
|
||||
import type { Organization, UpdateOrganizationRequest } from '@features/admin'
|
||||
|
||||
interface FormState {
|
||||
name: string
|
||||
short_name: string
|
||||
ogrn: string
|
||||
kpp: string
|
||||
legal_address: string
|
||||
actual_address: string
|
||||
contact_person: string
|
||||
contact_phone: string
|
||||
status: string
|
||||
}
|
||||
|
||||
function toForm(org: Organization): FormState {
|
||||
return {
|
||||
name: org.name ?? '',
|
||||
short_name: org.short_name ?? '',
|
||||
ogrn: org.ogrn ?? '',
|
||||
kpp: org.kpp ?? '',
|
||||
legal_address: org.legal_address ?? '',
|
||||
actual_address: org.actual_address ?? '',
|
||||
contact_person: org.contact_person ?? '',
|
||||
contact_phone: org.contact_phone ?? '',
|
||||
status: org.status ?? '',
|
||||
}
|
||||
}
|
||||
|
||||
function extractErrorMessage(error: unknown): string {
|
||||
const e = error as { detail?: unknown }
|
||||
if (typeof e?.detail === 'string') return e.detail
|
||||
if (Array.isArray(e?.detail) && (e.detail[0] as { msg?: string })?.msg) {
|
||||
return (e.detail[0] as { msg: string }).msg
|
||||
}
|
||||
return 'Не удалось сохранить изменения'
|
||||
}
|
||||
|
||||
export function useOrganizationForm(
|
||||
org: Organization | undefined,
|
||||
id: string,
|
||||
onSaved?: () => void,
|
||||
) {
|
||||
const [form, setForm] = useState<FormState>(() =>
|
||||
org ? toForm(org) : {
|
||||
name: '', short_name: '', ogrn: '', kpp: '', legal_address: '',
|
||||
actual_address: '', contact_person: '', contact_phone: '', status: '',
|
||||
},
|
||||
)
|
||||
const mutation = useUpdateOrganization(id)
|
||||
|
||||
// Sync local form state once the organization loads / changes.
|
||||
useEffect(() => {
|
||||
if (org) setForm(toForm(org))
|
||||
}, [org])
|
||||
|
||||
const setField = (key: keyof FormState) => (value: string) =>
|
||||
setForm((prev) => ({ ...prev, [key]: value }))
|
||||
|
||||
const handleSubmit = (e: React.FormEvent) => {
|
||||
e.preventDefault()
|
||||
|
||||
const trimmedOrNull = (v: string) => (v.trim() ? v.trim() : null)
|
||||
|
||||
const payload: UpdateOrganizationRequest = {
|
||||
name: form.name.trim(),
|
||||
short_name: trimmedOrNull(form.short_name),
|
||||
ogrn: trimmedOrNull(form.ogrn),
|
||||
kpp: trimmedOrNull(form.kpp),
|
||||
legal_address: trimmedOrNull(form.legal_address),
|
||||
actual_address: trimmedOrNull(form.actual_address),
|
||||
contact_person: trimmedOrNull(form.contact_person),
|
||||
contact_phone: trimmedOrNull(form.contact_phone),
|
||||
status: trimmedOrNull(form.status),
|
||||
}
|
||||
|
||||
mutation.mutate(payload, { onSuccess: () => onSaved?.() })
|
||||
}
|
||||
|
||||
const error = mutation.isError ? extractErrorMessage(mutation.error) : null
|
||||
|
||||
return {
|
||||
form,
|
||||
setField,
|
||||
handleSubmit,
|
||||
isSaving: mutation.isPending,
|
||||
error,
|
||||
}
|
||||
}
|
||||
@@ -1,154 +0,0 @@
|
||||
.page {
|
||||
min-height: 100vh;
|
||||
background: var(--bg-deep);
|
||||
padding: 40px 48px;
|
||||
}
|
||||
|
||||
.header {
|
||||
max-width: 900px;
|
||||
margin: 0 auto 28px;
|
||||
}
|
||||
|
||||
.back {
|
||||
background: none;
|
||||
border: none;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.6));
|
||||
font-size: 14px;
|
||||
cursor: pointer;
|
||||
padding: 0;
|
||||
margin-bottom: 14px;
|
||||
transition: color 0.2s;
|
||||
}
|
||||
|
||||
.back:hover {
|
||||
color: var(--text-primary, #fff);
|
||||
}
|
||||
|
||||
.title {
|
||||
font-size: clamp(24px, 3.5vw, 34px);
|
||||
font-weight: 700;
|
||||
color: var(--text-primary, #fff);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.form {
|
||||
max-width: 900px;
|
||||
margin: 0 auto;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 24px;
|
||||
}
|
||||
|
||||
.tabs {
|
||||
max-width: 900px;
|
||||
margin: 0 auto 24px;
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
border-bottom: 1px solid rgba(255, 255, 255, 0.08);
|
||||
}
|
||||
|
||||
.tab {
|
||||
background: none;
|
||||
border: none;
|
||||
border-bottom: 2px solid transparent;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.6));
|
||||
font-size: 15px;
|
||||
font-weight: 600;
|
||||
padding: 12px 16px;
|
||||
cursor: pointer;
|
||||
transition: color 0.2s, border-color 0.2s;
|
||||
}
|
||||
|
||||
.tab:hover {
|
||||
color: var(--text-primary, #fff);
|
||||
}
|
||||
|
||||
.tabActive {
|
||||
color: var(--text-primary, #fff);
|
||||
border-bottom-color: var(--interactive, #4a6dff);
|
||||
}
|
||||
|
||||
.tabPanel {
|
||||
max-width: 900px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.section {
|
||||
background: rgba(255, 255, 255, 0.04);
|
||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||
border-radius: 20px;
|
||||
padding: 24px;
|
||||
}
|
||||
|
||||
.sectionTitle {
|
||||
font-size: 14px;
|
||||
letter-spacing: 1.5px;
|
||||
text-transform: uppercase;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.5));
|
||||
font-weight: 600;
|
||||
margin: 0 0 18px;
|
||||
}
|
||||
|
||||
.grid {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr;
|
||||
gap: 18px;
|
||||
}
|
||||
|
||||
.bankLabel {
|
||||
display: block;
|
||||
font-size: 12px;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.5));
|
||||
font-weight: 600;
|
||||
margin-bottom: 8px;
|
||||
}
|
||||
|
||||
.textarea {
|
||||
width: 100%;
|
||||
background: var(--glass-bg, rgba(255, 255, 255, 0.06));
|
||||
border: 1px solid var(--glass-border, rgba(255, 255, 255, 0.1));
|
||||
border-radius: 10px;
|
||||
color: var(--text-primary, #fff);
|
||||
font-family: var(--font-mono, monospace);
|
||||
font-size: 13px;
|
||||
padding: 12px 14px;
|
||||
resize: vertical;
|
||||
outline: none;
|
||||
transition: border-color 0.2s, box-shadow 0.2s;
|
||||
}
|
||||
|
||||
.textarea:focus {
|
||||
border-color: var(--interactive, #4a6dff);
|
||||
box-shadow: 0 0 0 3px rgba(74, 109, 255, 0.15);
|
||||
}
|
||||
|
||||
.state {
|
||||
max-width: 900px;
|
||||
margin: 0 auto;
|
||||
padding: 40px 16px;
|
||||
text-align: center;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.6));
|
||||
}
|
||||
|
||||
.error {
|
||||
color: #ff5a5a;
|
||||
font-size: 13px;
|
||||
margin: 0;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.actions {
|
||||
max-width: 320px;
|
||||
margin: 0 auto;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.page {
|
||||
padding: 28px 20px;
|
||||
}
|
||||
|
||||
.grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
@@ -1,150 +0,0 @@
|
||||
import { useState } from 'react'
|
||||
import { useNavigate, useParams } from 'react-router-dom'
|
||||
import { useAdminAuth, useOrganization } from '@features/admin'
|
||||
import { ROUTES } from '@shared/config/routes'
|
||||
import { FormField, Notification, PrimaryButton } from '@shared/ui'
|
||||
import { AdminLoginForm } from '@widgets/admin-login-form'
|
||||
import { OrganizationDocuments } from '@widgets/organization-documents'
|
||||
import { OrganizationPurchaseRequests } from '@widgets/organization-purchase-requests'
|
||||
import { OrganizationWallets } from '@widgets/organization-wallets'
|
||||
import { useOrganizationForm } from '../model/useOrganizationForm'
|
||||
import styles from './AdminOrganizationPage.module.css'
|
||||
|
||||
type Tab = 'info' | 'wallets' | 'documents' | 'requests'
|
||||
|
||||
const TABS: { id: Tab; label: string }[] = [
|
||||
{ id: 'info', label: 'Общая информация' },
|
||||
{ id: 'wallets', label: 'Кошельки' },
|
||||
{ id: 'documents', label: 'Документы' },
|
||||
{ id: 'requests', label: 'Заявки' },
|
||||
]
|
||||
|
||||
function formatDateTime(value: string | null): string {
|
||||
if (!value) return '—'
|
||||
const d = new Date(value)
|
||||
if (Number.isNaN(d.getTime())) return '—'
|
||||
return d.toLocaleString('ru-RU')
|
||||
}
|
||||
|
||||
export function AdminOrganizationPage() {
|
||||
const { isAuthenticated, isLoading: isAuthLoading } = useAdminAuth()
|
||||
const { organizationId } = useParams<{ organizationId: string }>()
|
||||
const navigate = useNavigate()
|
||||
const { data: org, isLoading, isError } = useOrganization(organizationId)
|
||||
const [notice, setNotice] = useState(false)
|
||||
const [activeTab, setActiveTab] = useState<Tab>('info')
|
||||
const { form, setField, handleSubmit, isSaving, error } = useOrganizationForm(
|
||||
org,
|
||||
organizationId ?? '',
|
||||
() => setNotice(true),
|
||||
)
|
||||
|
||||
if (isAuthLoading) return null
|
||||
if (!isAuthenticated) return <AdminLoginForm />
|
||||
|
||||
return (
|
||||
<div className={styles.page}>
|
||||
<header className={styles.header}>
|
||||
<button className={styles.back} type="button" onClick={() => navigate(ROUTES.ADMIN)}>
|
||||
← Назад к списку
|
||||
</button>
|
||||
<h1 className={styles.title}>{org ? org.name : 'Юридическое лицо'}</h1>
|
||||
</header>
|
||||
|
||||
{isLoading && <div className={styles.state}>Загрузка...</div>}
|
||||
{isError && <div className={styles.state}>Не удалось загрузить организацию</div>}
|
||||
|
||||
{org && (
|
||||
<div className={styles.tabs}>
|
||||
{TABS.map((tab) => (
|
||||
<button
|
||||
key={tab.id}
|
||||
type="button"
|
||||
className={`${styles.tab} ${activeTab === tab.id ? styles.tabActive : ''}`}
|
||||
onClick={() => setActiveTab(tab.id)}
|
||||
>
|
||||
{tab.label}
|
||||
</button>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{org && activeTab === 'info' && (
|
||||
<form className={styles.form} onSubmit={handleSubmit}>
|
||||
<section className={styles.section}>
|
||||
<h2 className={styles.sectionTitle}>Реквизиты</h2>
|
||||
<div className={styles.grid}>
|
||||
<FormField label="Наименование" value={form.name} onChange={setField('name')} placeholder="ООО «Ромашка»" required />
|
||||
<FormField label="Краткое наименование" value={form.short_name} onChange={setField('short_name')} placeholder="Ромашка" />
|
||||
<FormField label="ИНН" value={org.inn} readOnly icon="lock" />
|
||||
<FormField label="ОГРН" value={form.ogrn} onChange={setField('ogrn')} placeholder="1027700132195" />
|
||||
<FormField label="КПП" value={form.kpp} onChange={setField('kpp')} placeholder="770801001" />
|
||||
<FormField label="Статус" value={form.status} onChange={setField('status')} placeholder="active" />
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section className={styles.section}>
|
||||
<h2 className={styles.sectionTitle}>Адреса</h2>
|
||||
<div className={styles.grid}>
|
||||
<FormField label="Юридический адрес" value={form.legal_address} onChange={setField('legal_address')} placeholder="г. Москва, ул. Тверская, д. 1" />
|
||||
<FormField label="Фактический адрес" value={form.actual_address} onChange={setField('actual_address')} placeholder="г. Москва, ул. Тверская, д. 1" />
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section className={styles.section}>
|
||||
<h2 className={styles.sectionTitle}>Контакты</h2>
|
||||
<div className={styles.grid}>
|
||||
<FormField label="Контактное лицо" value={form.contact_person} onChange={setField('contact_person')} placeholder="Иванов Иван Иванович" />
|
||||
<FormField label="Контактный телефон" type="tel" value={form.contact_phone} onChange={setField('contact_phone')} placeholder="+7 (999) 000-00-00" />
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section className={styles.section}>
|
||||
<h2 className={styles.sectionTitle}>Системная информация</h2>
|
||||
<div className={styles.grid}>
|
||||
<FormField label="ID организации" value={org.id} readOnly icon="lock" />
|
||||
<FormField label="ID пользователя" value={org.user_id} readOnly icon="lock" />
|
||||
<FormField label="KYC" value={org.kyc_verified ? 'Подтверждён' : 'Не подтверждён'} readOnly />
|
||||
<FormField label="Дата KYC" value={formatDateTime(org.kyc_verified_at)} readOnly />
|
||||
<FormField label="Кошельки" value={org.has_wallets ? 'Есть' : 'Нет'} readOnly />
|
||||
<FormField label="Создано" value={formatDateTime(org.created_at)} readOnly />
|
||||
<FormField label="Обновлено" value={formatDateTime(org.updated_at)} readOnly />
|
||||
</div>
|
||||
</section>
|
||||
|
||||
{error && <p className={styles.error}>{error}</p>}
|
||||
|
||||
<div className={styles.actions}>
|
||||
<PrimaryButton label={isSaving ? 'Сохранение...' : 'Сохранить изменения'} disabled={isSaving} />
|
||||
</div>
|
||||
</form>
|
||||
)}
|
||||
|
||||
{org && activeTab === 'wallets' && (
|
||||
<div className={styles.tabPanel}>
|
||||
<OrganizationWallets orgId={org.id} />
|
||||
</div>
|
||||
)}
|
||||
|
||||
{org && activeTab === 'documents' && (
|
||||
<div className={styles.tabPanel}>
|
||||
<OrganizationDocuments orgId={org.id} />
|
||||
</div>
|
||||
)}
|
||||
|
||||
{org && activeTab === 'requests' && (
|
||||
<div className={styles.tabPanel}>
|
||||
<OrganizationPurchaseRequests orgId={org.id} />
|
||||
</div>
|
||||
)}
|
||||
|
||||
{notice && (
|
||||
<Notification
|
||||
status="success"
|
||||
message="Изменения сохранены"
|
||||
onClose={() => setNotice(false)}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
)
|
||||
}
|
||||
@@ -1 +0,0 @@
|
||||
export { AdminPage } from './ui/AdminPage'
|
||||
@@ -1,84 +0,0 @@
|
||||
.page {
|
||||
min-height: 100vh;
|
||||
background: var(--bg-deep);
|
||||
padding: 40px 48px;
|
||||
}
|
||||
|
||||
.header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
max-width: 1200px;
|
||||
margin: 0 auto 36px;
|
||||
}
|
||||
|
||||
.greeting {
|
||||
font-size: clamp(28px, 4vw, 40px);
|
||||
font-weight: 700;
|
||||
color: var(--text-primary, #fff);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.logout {
|
||||
background: rgba(255, 255, 255, 0.06);
|
||||
border: 1px solid var(--glass-border, rgba(255, 255, 255, 0.1));
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.7));
|
||||
border-radius: 10px;
|
||||
height: 40px;
|
||||
padding: 0 18px;
|
||||
font-size: 13px;
|
||||
font-weight: 600;
|
||||
cursor: pointer;
|
||||
transition: background 0.2s, color 0.2s, border-color 0.2s;
|
||||
}
|
||||
|
||||
.logout:hover {
|
||||
background: rgba(255, 90, 90, 0.12);
|
||||
border-color: rgba(255, 90, 90, 0.3);
|
||||
color: #ff5a5a;
|
||||
}
|
||||
|
||||
.content {
|
||||
max-width: 1200px;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.toolbar {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 16px;
|
||||
margin-bottom: 20px;
|
||||
flex-wrap: wrap;
|
||||
}
|
||||
|
||||
.sectionTitle {
|
||||
font-size: 20px;
|
||||
font-weight: 700;
|
||||
color: var(--text-primary, #fff);
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.addBtn {
|
||||
background: linear-gradient(135deg, var(--grad-edge, #4a6dff), var(--grad-center, #6f4aff));
|
||||
border: none;
|
||||
color: #fff;
|
||||
border-radius: 12px;
|
||||
height: 44px;
|
||||
padding: 0 20px;
|
||||
font-size: 14px;
|
||||
font-weight: 700;
|
||||
cursor: pointer;
|
||||
transition: filter 0.2s, box-shadow 0.2s;
|
||||
}
|
||||
|
||||
.addBtn:hover {
|
||||
filter: brightness(1.12);
|
||||
box-shadow: 0 6px 20px rgba(74, 109, 255, 0.35);
|
||||
}
|
||||
|
||||
@media (max-width: 768px) {
|
||||
.page {
|
||||
padding: 28px 20px;
|
||||
}
|
||||
}
|
||||
@@ -1,77 +0,0 @@
|
||||
import { useState } from 'react'
|
||||
import { useAdminAuth, useAdminLogout, useCreateOrganizationWallets } from '@features/admin'
|
||||
import type { Organization } from '@features/admin'
|
||||
import { Notification } from '@shared/ui'
|
||||
import { AdminLoginForm } from '@widgets/admin-login-form'
|
||||
import { LegalEntitiesTable } from '@widgets/legal-entities-table'
|
||||
import { AddLegalEntityModal } from '@widgets/add-legal-entity-modal'
|
||||
import styles from './AdminPage.module.css'
|
||||
|
||||
type NotificationState = { message: string; status: 'success' | 'error' | 'warning' }
|
||||
|
||||
export function AdminPage() {
|
||||
const { isAuthenticated, isLoading } = useAdminAuth()
|
||||
const logout = useAdminLogout()
|
||||
const createWallets = useCreateOrganizationWallets()
|
||||
const [modalOpen, setModalOpen] = useState(false)
|
||||
const [notification, setNotification] = useState<NotificationState | null>(null)
|
||||
|
||||
// After a legal entity is created we immediately provision its wallets.
|
||||
// The page stays mounted (unlike the modal), so these mutate callbacks fire reliably.
|
||||
function handleCreated(organization: Organization) {
|
||||
setNotification({ status: 'success', message: 'Юридическое лицо добавлено' })
|
||||
createWallets.mutate(organization.id, {
|
||||
onSuccess: (result) => {
|
||||
setNotification({
|
||||
status: 'success',
|
||||
message: `Кошельки созданы (${result.wallets.length})`,
|
||||
})
|
||||
},
|
||||
onError: () => {
|
||||
setNotification({
|
||||
status: 'warning',
|
||||
message: 'Юридическое лицо создано, но кошельки создать не удалось',
|
||||
})
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
if (isLoading) return null
|
||||
if (!isAuthenticated) return <AdminLoginForm />
|
||||
|
||||
return (
|
||||
<div className={styles.page}>
|
||||
<header className={styles.header}>
|
||||
<h1 className={styles.greeting}>Привет, Марк!</h1>
|
||||
<button className={styles.logout} type="button" onClick={() => logout.mutate()}>
|
||||
Выйти
|
||||
</button>
|
||||
</header>
|
||||
|
||||
<section className={styles.content}>
|
||||
<div className={styles.toolbar}>
|
||||
<h2 className={styles.sectionTitle}>Юридические лица</h2>
|
||||
<button className={styles.addBtn} type="button" onClick={() => setModalOpen(true)}>
|
||||
+ Добавить юридическое лицо
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<LegalEntitiesTable />
|
||||
</section>
|
||||
|
||||
<AddLegalEntityModal
|
||||
open={modalOpen}
|
||||
onClose={() => setModalOpen(false)}
|
||||
onCreated={handleCreated}
|
||||
/>
|
||||
|
||||
{notification && (
|
||||
<Notification
|
||||
status={notification.status}
|
||||
message={notification.message}
|
||||
onClose={() => setNotification(null)}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
)
|
||||
}
|
||||
@@ -19,9 +19,4 @@ export const ROUTES = {
|
||||
SOGLASIE_PERSONALNYH_DANNYH: '/soglasie-personalnyh-dannyh',
|
||||
REESTR_PD_RKN: '/reestr-pd-rkn',
|
||||
TRANSACTIONS: '/transactions',
|
||||
ADMIN: '/sys-c7f29a4e-d81b-4630-ops-console',
|
||||
ADMIN_ORGANIZATION: '/sys-c7f29a4e-d81b-4630-ops-console/organizations/:organizationId',
|
||||
} as const
|
||||
|
||||
export const adminOrganizationPath = (id: string) =>
|
||||
`/sys-c7f29a4e-d81b-4630-ops-console/organizations/${id}`
|
||||
|
||||
@@ -1 +0,0 @@
|
||||
export { AddLegalEntityModal } from './ui/AddLegalEntityModal'
|
||||
@@ -1,86 +0,0 @@
|
||||
import { useState } from 'react'
|
||||
import { useCreateOrganization } from '@features/admin'
|
||||
import type { CreateOrganizationRequest, Organization } from '@features/admin'
|
||||
|
||||
const INITIAL = {
|
||||
email: '',
|
||||
password: '',
|
||||
name: '',
|
||||
inn: '',
|
||||
short_name: '',
|
||||
ogrn: '',
|
||||
kpp: '',
|
||||
legal_address: '',
|
||||
actual_address: '',
|
||||
contact_person: '',
|
||||
contact_phone: '',
|
||||
status: 'active',
|
||||
bank_name: '',
|
||||
bik: '',
|
||||
account: '',
|
||||
corr_account: '',
|
||||
}
|
||||
|
||||
type FormState = typeof INITIAL
|
||||
|
||||
function extractErrorMessage(error: unknown): string {
|
||||
const e = error as { detail?: unknown }
|
||||
if (typeof e?.detail === 'string') return e.detail
|
||||
if (Array.isArray(e?.detail) && (e.detail[0] as { msg?: string })?.msg) {
|
||||
return (e.detail[0] as { msg: string }).msg
|
||||
}
|
||||
return 'Не удалось добавить юридическое лицо'
|
||||
}
|
||||
|
||||
export function useAddLegalEntityForm(onSuccess: (organization: Organization) => void) {
|
||||
const [form, setForm] = useState<FormState>(INITIAL)
|
||||
const mutation = useCreateOrganization()
|
||||
|
||||
const setField = (key: keyof FormState) => (value: string) =>
|
||||
setForm((prev) => ({ ...prev, [key]: value }))
|
||||
|
||||
const handleSubmit = (e: React.FormEvent) => {
|
||||
e.preventDefault()
|
||||
|
||||
const trimmedOrNull = (v: string) => (v.trim() ? v.trim() : null)
|
||||
|
||||
const bankEntries: Record<string, string> = {}
|
||||
if (form.bank_name.trim()) bankEntries.bank_name = form.bank_name.trim()
|
||||
if (form.bik.trim()) bankEntries.bik = form.bik.trim()
|
||||
if (form.account.trim()) bankEntries.account = form.account.trim()
|
||||
if (form.corr_account.trim()) bankEntries.corr_account = form.corr_account.trim()
|
||||
|
||||
const payload: CreateOrganizationRequest = {
|
||||
email: form.email.trim(),
|
||||
password: form.password,
|
||||
name: form.name.trim(),
|
||||
inn: form.inn.trim(),
|
||||
short_name: trimmedOrNull(form.short_name),
|
||||
ogrn: trimmedOrNull(form.ogrn),
|
||||
kpp: trimmedOrNull(form.kpp),
|
||||
legal_address: trimmedOrNull(form.legal_address),
|
||||
actual_address: trimmedOrNull(form.actual_address),
|
||||
contact_person: trimmedOrNull(form.contact_person),
|
||||
contact_phone: trimmedOrNull(form.contact_phone),
|
||||
bank_details: Object.keys(bankEntries).length ? bankEntries : null,
|
||||
status: form.status.trim() || 'active',
|
||||
}
|
||||
|
||||
mutation.mutate(payload, {
|
||||
onSuccess: (organization) => {
|
||||
setForm(INITIAL)
|
||||
onSuccess(organization)
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
const error = mutation.isError ? extractErrorMessage(mutation.error) : null
|
||||
|
||||
return {
|
||||
form,
|
||||
setField,
|
||||
handleSubmit,
|
||||
isLoading: mutation.isPending,
|
||||
error,
|
||||
}
|
||||
}
|
||||
@@ -1,101 +0,0 @@
|
||||
@keyframes dialogIn {
|
||||
from { opacity: 0; transform: scale(0.96); }
|
||||
to { opacity: 1; transform: scale(1); }
|
||||
}
|
||||
|
||||
.overlay {
|
||||
position: fixed;
|
||||
inset: 0;
|
||||
background: rgba(0, 0, 0, 0.6);
|
||||
backdrop-filter: blur(4px);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
z-index: 1000;
|
||||
padding: 16px;
|
||||
}
|
||||
|
||||
.dialog {
|
||||
background: var(--bg-mid, #151520);
|
||||
border: 1px solid var(--glass-border, rgba(255, 255, 255, 0.1));
|
||||
border-radius: 20px;
|
||||
width: 100%;
|
||||
max-width: 720px;
|
||||
max-height: 90vh;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
animation: dialogIn 0.18s ease;
|
||||
}
|
||||
|
||||
.header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
padding: 20px 24px 0;
|
||||
flex-shrink: 0;
|
||||
}
|
||||
|
||||
.title {
|
||||
font-size: 17px;
|
||||
font-weight: 700;
|
||||
color: var(--text-primary, #fff);
|
||||
}
|
||||
|
||||
.closeBtn {
|
||||
background: none;
|
||||
border: none;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.4));
|
||||
font-size: 16px;
|
||||
cursor: pointer;
|
||||
padding: 4px;
|
||||
line-height: 1;
|
||||
transition: color 0.15s;
|
||||
}
|
||||
|
||||
.closeBtn:hover {
|
||||
color: var(--text-primary, #fff);
|
||||
}
|
||||
|
||||
.body {
|
||||
padding: 16px 24px 24px;
|
||||
overflow-y: auto;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.groupLabel {
|
||||
font-size: 12px;
|
||||
letter-spacing: 1.5px;
|
||||
text-transform: uppercase;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.5));
|
||||
font-weight: 600;
|
||||
margin: 16px 0 4px;
|
||||
}
|
||||
|
||||
.groupLabel:first-child {
|
||||
margin-top: 4px;
|
||||
}
|
||||
|
||||
.grid {
|
||||
display: grid;
|
||||
grid-template-columns: 1fr 1fr;
|
||||
gap: 16px;
|
||||
}
|
||||
|
||||
.error {
|
||||
color: #ff5a5a;
|
||||
font-size: 13px;
|
||||
margin: 12px 0 0;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.actions {
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
@media (max-width: 560px) {
|
||||
.grid {
|
||||
grid-template-columns: 1fr;
|
||||
}
|
||||
}
|
||||
@@ -1,80 +0,0 @@
|
||||
import { useEffect } from 'react'
|
||||
import type { Organization } from '@features/admin'
|
||||
import { FormField, PrimaryButton } from '@shared/ui'
|
||||
import { useAddLegalEntityForm } from '../model/useAddLegalEntityForm'
|
||||
import styles from './AddLegalEntityModal.module.css'
|
||||
|
||||
interface Props {
|
||||
open: boolean
|
||||
onClose: () => void
|
||||
onCreated: (organization: Organization) => void
|
||||
}
|
||||
|
||||
export function AddLegalEntityModal({ open, onClose, onCreated }: Props) {
|
||||
const { form, setField, handleSubmit, isLoading, error } = useAddLegalEntityForm((organization) => {
|
||||
onCreated(organization)
|
||||
onClose()
|
||||
})
|
||||
|
||||
useEffect(() => {
|
||||
if (!open) return
|
||||
function onKey(e: KeyboardEvent) {
|
||||
if (e.key === 'Escape') onClose()
|
||||
}
|
||||
window.addEventListener('keydown', onKey)
|
||||
return () => window.removeEventListener('keydown', onKey)
|
||||
}, [open, onClose])
|
||||
|
||||
if (!open) return null
|
||||
|
||||
function handleOverlay(e: React.MouseEvent<HTMLDivElement>) {
|
||||
if (e.target === e.currentTarget) onClose()
|
||||
}
|
||||
|
||||
return (
|
||||
<div className={styles.overlay} onMouseDown={handleOverlay}>
|
||||
<div className={styles.dialog}>
|
||||
<div className={styles.header}>
|
||||
<span className={styles.title}>Добавить юридическое лицо</span>
|
||||
<button className={styles.closeBtn} type="button" onClick={onClose} aria-label="Закрыть">✕</button>
|
||||
</div>
|
||||
|
||||
<form className={styles.body} onSubmit={handleSubmit}>
|
||||
<p className={styles.groupLabel}>Обязательные поля</p>
|
||||
<div className={styles.grid}>
|
||||
<FormField label="Email" type="email" value={form.email} onChange={setField('email')} placeholder="org@mail.ru" required />
|
||||
<FormField label="Пароль" type="password" value={form.password} onChange={setField('password')} placeholder="Минимум 8 символов" required />
|
||||
<FormField label="Наименование" value={form.name} onChange={setField('name')} placeholder="ООО «Ромашка»" required />
|
||||
<FormField label="ИНН" value={form.inn} onChange={setField('inn')} placeholder="10–12 цифр" required />
|
||||
</div>
|
||||
|
||||
<p className={styles.groupLabel}>Дополнительные поля</p>
|
||||
<div className={styles.grid}>
|
||||
<FormField label="Краткое наименование" value={form.short_name} onChange={setField('short_name')} placeholder="Ромашка" />
|
||||
<FormField label="ОГРН" value={form.ogrn} onChange={setField('ogrn')} placeholder="—" />
|
||||
<FormField label="КПП" value={form.kpp} onChange={setField('kpp')} placeholder="—" />
|
||||
<FormField label="Статус" value={form.status} onChange={setField('status')} placeholder="active" />
|
||||
<FormField label="Юридический адрес" value={form.legal_address} onChange={setField('legal_address')} placeholder="—" />
|
||||
<FormField label="Фактический адрес" value={form.actual_address} onChange={setField('actual_address')} placeholder="—" />
|
||||
<FormField label="Контактное лицо" value={form.contact_person} onChange={setField('contact_person')} placeholder="—" />
|
||||
<FormField label="Контактный телефон" type="tel" value={form.contact_phone} onChange={setField('contact_phone')} placeholder="+7 (999) 000-00-00" />
|
||||
</div>
|
||||
|
||||
<p className={styles.groupLabel}>Банковские реквизиты</p>
|
||||
<div className={styles.grid}>
|
||||
<FormField label="Банк" value={form.bank_name} onChange={setField('bank_name')} placeholder="—" />
|
||||
<FormField label="БИК" value={form.bik} onChange={setField('bik')} placeholder="—" />
|
||||
<FormField label="Расчётный счёт" value={form.account} onChange={setField('account')} placeholder="—" />
|
||||
<FormField label="Корр. счёт" value={form.corr_account} onChange={setField('corr_account')} placeholder="—" />
|
||||
</div>
|
||||
|
||||
{error && <p className={styles.error}>{error}</p>}
|
||||
|
||||
<div className={styles.actions}>
|
||||
<PrimaryButton label={isLoading ? 'Сохранение...' : 'Сохранить'} disabled={isLoading} />
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
@@ -1 +0,0 @@
|
||||
export { AdminLoginForm } from './ui/AdminLoginForm'
|
||||
@@ -1,35 +0,0 @@
|
||||
import { useState } from 'react'
|
||||
import { useAdminLogin } from '@features/admin'
|
||||
|
||||
function extractErrorMessage(error: unknown): string {
|
||||
const e = error as { detail?: unknown }
|
||||
if (typeof e?.detail === 'string') return e.detail
|
||||
if (Array.isArray(e?.detail) && (e.detail[0] as { msg?: string })?.msg) {
|
||||
return (e.detail[0] as { msg: string }).msg
|
||||
}
|
||||
return 'Неверный логин или пароль'
|
||||
}
|
||||
|
||||
export function useAdminLoginForm() {
|
||||
const [login, setLogin] = useState('')
|
||||
const [password, setPassword] = useState('')
|
||||
const loginMutation = useAdminLogin()
|
||||
|
||||
const handleSubmit = (e: React.FormEvent) => {
|
||||
e.preventDefault()
|
||||
if (!login || !password) return
|
||||
loginMutation.mutate({ login, password })
|
||||
}
|
||||
|
||||
const error = loginMutation.isError ? extractErrorMessage(loginMutation.error) : null
|
||||
|
||||
return {
|
||||
login,
|
||||
setLogin,
|
||||
password,
|
||||
setPassword,
|
||||
isLoading: loginMutation.isPending,
|
||||
error,
|
||||
handleSubmit,
|
||||
}
|
||||
}
|
||||
@@ -1,49 +0,0 @@
|
||||
.wrap {
|
||||
min-height: 100vh;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 24px;
|
||||
background: var(--bg-deep);
|
||||
}
|
||||
|
||||
.card {
|
||||
background: var(--glass-bg, rgba(255, 255, 255, 0.06));
|
||||
border: 1px solid var(--glass-border, rgba(255, 255, 255, 0.1));
|
||||
border-radius: 24px;
|
||||
padding: 36px 32px;
|
||||
width: 100%;
|
||||
max-width: 420px;
|
||||
}
|
||||
|
||||
.title {
|
||||
text-align: center;
|
||||
font-size: 24px;
|
||||
font-weight: 700;
|
||||
color: var(--text-primary, #fff);
|
||||
margin: 0 0 4px;
|
||||
}
|
||||
|
||||
.subtitle {
|
||||
text-align: center;
|
||||
font-size: 14px;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.5));
|
||||
margin: 0 0 28px;
|
||||
}
|
||||
|
||||
.fields {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 18px;
|
||||
}
|
||||
|
||||
.error {
|
||||
color: #ff5a5a;
|
||||
font-size: 13px;
|
||||
margin: 14px 0 0;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.submit {
|
||||
margin-top: 26px;
|
||||
}
|
||||
@@ -1,40 +0,0 @@
|
||||
import { FormField, PrimaryButton } from '@shared/ui'
|
||||
import { useAdminLoginForm } from '../model/useAdminLoginForm'
|
||||
import styles from './AdminLoginForm.module.css'
|
||||
|
||||
export function AdminLoginForm() {
|
||||
const { login, setLogin, password, setPassword, isLoading, error, handleSubmit } = useAdminLoginForm()
|
||||
|
||||
return (
|
||||
<div className={styles.wrap}>
|
||||
<form className={styles.card} onSubmit={handleSubmit}>
|
||||
<h1 className={styles.title}>Панель администратора</h1>
|
||||
<p className={styles.subtitle}>Войдите, чтобы продолжить</p>
|
||||
|
||||
<div className={styles.fields}>
|
||||
<FormField
|
||||
label="Логин"
|
||||
value={login}
|
||||
onChange={setLogin}
|
||||
placeholder="Введите логин"
|
||||
required
|
||||
/>
|
||||
<FormField
|
||||
label="Пароль"
|
||||
type="password"
|
||||
value={password}
|
||||
onChange={setPassword}
|
||||
placeholder="Введите пароль"
|
||||
required
|
||||
/>
|
||||
</div>
|
||||
|
||||
{error && <p className={styles.error}>{error}</p>}
|
||||
|
||||
<div className={styles.submit}>
|
||||
<PrimaryButton label={isLoading ? 'Вход...' : 'Войти'} disabled={isLoading} />
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
@@ -1 +0,0 @@
|
||||
export { LegalEntitiesTable } from './ui/LegalEntitiesTable'
|
||||
@@ -1,93 +0,0 @@
|
||||
.tableWrap {
|
||||
background: rgba(255, 255, 255, 0.04);
|
||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||
border-radius: 24px;
|
||||
padding: 24px;
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.table {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
}
|
||||
|
||||
.table th {
|
||||
text-align: left;
|
||||
font-size: 12px;
|
||||
letter-spacing: 1.5px;
|
||||
text-transform: uppercase;
|
||||
color: var(--text-secondary);
|
||||
font-weight: 500;
|
||||
padding: 0 16px 18px;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.table td {
|
||||
padding: 16px;
|
||||
border-top: 1px solid rgba(255, 255, 255, 0.06);
|
||||
vertical-align: middle;
|
||||
transition: background 0.2s ease;
|
||||
font-size: 14px;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.table tr:hover td {
|
||||
background: rgba(255, 255, 255, 0.04);
|
||||
}
|
||||
|
||||
.row {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.name {
|
||||
font-weight: 600;
|
||||
display: block;
|
||||
}
|
||||
|
||||
.subname {
|
||||
display: block;
|
||||
font-size: 12px;
|
||||
color: var(--text-secondary);
|
||||
margin-top: 2px;
|
||||
}
|
||||
|
||||
.mono {
|
||||
font-family: var(--font-mono, monospace);
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.status {
|
||||
display: inline-block;
|
||||
padding: 4px 10px;
|
||||
border-radius: 999px;
|
||||
font-size: 12px;
|
||||
font-weight: 600;
|
||||
background: rgba(74, 109, 255, 0.12);
|
||||
color: #7c95ff;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.kyc {
|
||||
display: inline-block;
|
||||
padding: 4px 10px;
|
||||
border-radius: 999px;
|
||||
font-size: 12px;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.kycOk {
|
||||
background: rgba(0, 196, 140, 0.14);
|
||||
color: #00c48c;
|
||||
}
|
||||
|
||||
.kycNo {
|
||||
background: rgba(255, 90, 90, 0.14);
|
||||
color: #ff5a5a;
|
||||
}
|
||||
|
||||
.state {
|
||||
padding: 40px 16px;
|
||||
text-align: center;
|
||||
color: var(--text-secondary);
|
||||
font-size: 14px;
|
||||
}
|
||||
@@ -1,88 +0,0 @@
|
||||
import { useNavigate } from 'react-router-dom'
|
||||
import { useOrganizations } from '@features/admin'
|
||||
import { adminOrganizationPath } from '@shared/config/routes'
|
||||
import styles from './LegalEntitiesTable.module.css'
|
||||
|
||||
const STATUS_LABELS: Record<string, string> = {
|
||||
active: 'Активно',
|
||||
blocked: 'Заблокировано',
|
||||
inactive: 'Неактивно',
|
||||
}
|
||||
|
||||
function formatDate(value: string | null): string {
|
||||
if (!value) return '—'
|
||||
const d = new Date(value)
|
||||
if (Number.isNaN(d.getTime())) return '—'
|
||||
return d.toLocaleDateString('ru-RU')
|
||||
}
|
||||
|
||||
export function LegalEntitiesTable() {
|
||||
const { data, isLoading, isError } = useOrganizations()
|
||||
const navigate = useNavigate()
|
||||
|
||||
if (isLoading) {
|
||||
return <div className={styles.tableWrap}><div className={styles.state}>Загрузка...</div></div>
|
||||
}
|
||||
|
||||
if (isError) {
|
||||
return (
|
||||
<div className={styles.tableWrap}>
|
||||
<div className={styles.state}>Не удалось загрузить список юридических лиц</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
if (!data || data.items.length === 0) {
|
||||
return (
|
||||
<div className={styles.tableWrap}>
|
||||
<div className={styles.state}>Юридические лица ещё не добавлены</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
return (
|
||||
<div className={styles.tableWrap}>
|
||||
<table className={styles.table}>
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Название</th>
|
||||
<th>ИНН</th>
|
||||
<th>КПП</th>
|
||||
<th>Контактное лицо</th>
|
||||
<th>Телефон</th>
|
||||
<th>Статус</th>
|
||||
<th>KYC</th>
|
||||
<th>Создано</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{data.items.map((org) => (
|
||||
<tr
|
||||
key={org.id}
|
||||
className={styles.row}
|
||||
onClick={() => navigate(adminOrganizationPath(org.id))}
|
||||
>
|
||||
<td>
|
||||
<span className={styles.name}>{org.name}</span>
|
||||
{org.short_name && <span className={styles.subname}>{org.short_name}</span>}
|
||||
</td>
|
||||
<td className={styles.mono}>{org.inn}</td>
|
||||
<td className={styles.mono}>{org.kpp ?? '—'}</td>
|
||||
<td>{org.contact_person ?? '—'}</td>
|
||||
<td className={styles.mono}>{org.contact_phone ?? '—'}</td>
|
||||
<td>
|
||||
<span className={styles.status}>{STATUS_LABELS[org.status] ?? org.status}</span>
|
||||
</td>
|
||||
<td>
|
||||
<span className={`${styles.kyc} ${org.kyc_verified ? styles.kycOk : styles.kycNo}`}>
|
||||
{org.kyc_verified ? 'Да' : 'Нет'}
|
||||
</span>
|
||||
</td>
|
||||
<td>{formatDate(org.created_at)}</td>
|
||||
</tr>
|
||||
))}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
@@ -1 +0,0 @@
|
||||
export { OrganizationDocuments } from './ui/OrganizationDocuments'
|
||||
@@ -1,124 +0,0 @@
|
||||
.section {
|
||||
background: rgba(255, 255, 255, 0.04);
|
||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||
border-radius: 20px;
|
||||
padding: 24px;
|
||||
}
|
||||
|
||||
.sectionTitle {
|
||||
font-size: 14px;
|
||||
letter-spacing: 1.5px;
|
||||
text-transform: uppercase;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.5));
|
||||
font-weight: 600;
|
||||
margin: 0 0 18px;
|
||||
}
|
||||
|
||||
.slots {
|
||||
list-style: none;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.slot {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: space-between;
|
||||
gap: 16px;
|
||||
flex-wrap: wrap;
|
||||
padding: 16px 0;
|
||||
border-top: 1px solid rgba(255, 255, 255, 0.06);
|
||||
}
|
||||
|
||||
.slot:first-child {
|
||||
border-top: none;
|
||||
}
|
||||
|
||||
.slotInfo {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 4px;
|
||||
min-width: 200px;
|
||||
}
|
||||
|
||||
.slotLabel {
|
||||
font-size: 14px;
|
||||
font-weight: 600;
|
||||
color: var(--text-primary, #fff);
|
||||
}
|
||||
|
||||
.slotFile {
|
||||
font-size: 13px;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.7));
|
||||
}
|
||||
|
||||
.slotActions {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
gap: 10px;
|
||||
}
|
||||
|
||||
.hiddenInput {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.uploadBtn {
|
||||
background: var(--interactive, #4a6dff);
|
||||
border: none;
|
||||
border-radius: 10px;
|
||||
color: #fff;
|
||||
font-size: 14px;
|
||||
font-weight: 600;
|
||||
padding: 11px 20px;
|
||||
cursor: pointer;
|
||||
transition: opacity 0.2s;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.uploadBtn:disabled {
|
||||
opacity: 0.5;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.mono {
|
||||
font-family: var(--font-mono, monospace);
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.downloadBtn {
|
||||
display: inline-block;
|
||||
background: rgba(74, 109, 255, 0.12);
|
||||
border: 1px solid rgba(74, 109, 255, 0.3);
|
||||
border-radius: 8px;
|
||||
color: #7c95ff;
|
||||
font-size: 13px;
|
||||
font-weight: 600;
|
||||
padding: 7px 14px;
|
||||
cursor: pointer;
|
||||
text-decoration: none;
|
||||
transition: background 0.2s;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.downloadBtn:hover {
|
||||
background: rgba(74, 109, 255, 0.22);
|
||||
}
|
||||
|
||||
.muted {
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.4));
|
||||
}
|
||||
|
||||
.state {
|
||||
padding: 32px 16px;
|
||||
text-align: center;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.6));
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.error {
|
||||
color: #ff5a5a;
|
||||
font-size: 13px;
|
||||
margin: 0 0 12px;
|
||||
}
|
||||
@@ -1,148 +0,0 @@
|
||||
import { useRef, useState } from 'react'
|
||||
import { useDocuments, useUploadDocument } from '@features/admin'
|
||||
import type { DocumentResponse, DocumentTypeSlug } from '@features/admin'
|
||||
import styles from './OrganizationDocuments.module.css'
|
||||
|
||||
interface Props {
|
||||
orgId: string
|
||||
}
|
||||
|
||||
const DOCUMENT_SLOTS: { type: DocumentTypeSlug; label: string }[] = [
|
||||
{ type: 'charter', label: 'Устав' },
|
||||
{ type: 'inn-certificate', label: 'Свидетельство ИНН' },
|
||||
{ type: 'ogrn-certificate', label: 'Свидетельство ОГРН' },
|
||||
{ type: 'bank-details', label: 'Банковские реквизиты' },
|
||||
{ type: 'kyc-representative', label: 'Документ представителя (KYC)' },
|
||||
{ type: 'power-of-attorney', label: 'Доверенность' },
|
||||
{ type: 'other', label: 'Прочее' },
|
||||
]
|
||||
|
||||
function formatSize(bytes: number | null): string {
|
||||
if (bytes == null) return '—'
|
||||
if (bytes < 1024) return `${bytes} Б`
|
||||
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} КБ`
|
||||
return `${(bytes / (1024 * 1024)).toFixed(1)} МБ`
|
||||
}
|
||||
|
||||
function extractErrorMessage(error: unknown): string {
|
||||
const e = error as { detail?: unknown; message?: unknown }
|
||||
if (typeof e?.detail === 'string') return e.detail
|
||||
if (Array.isArray(e?.detail) && (e.detail[0] as { msg?: string })?.msg) {
|
||||
return (e.detail[0] as { msg: string }).msg
|
||||
}
|
||||
if (typeof e?.message === 'string') return e.message
|
||||
return 'Не удалось выполнить операцию'
|
||||
}
|
||||
|
||||
export function OrganizationDocuments({ orgId }: Props) {
|
||||
const { data: documents, isLoading, isError } = useDocuments(orgId)
|
||||
const upload = useUploadDocument(orgId)
|
||||
|
||||
// Which slot's file picker is currently being uploaded — drives the per-row
|
||||
// "Загрузка..." state and disables only the active row.
|
||||
const [activeType, setActiveType] = useState<DocumentTypeSlug | null>(null)
|
||||
|
||||
// Map list items onto fixed slots. `document_type` may come back in
|
||||
// underscore form (e.g. `inn_certificate`) while our slugs use hyphens.
|
||||
const byType = new Map<string, DocumentResponse>()
|
||||
for (const doc of documents ?? []) {
|
||||
byType.set(doc.document_type.replace(/_/g, '-'), doc)
|
||||
}
|
||||
|
||||
function handleSelect(type: DocumentTypeSlug, input: HTMLInputElement) {
|
||||
const file = input.files?.[0]
|
||||
// Reset so picking the same file again still fires onChange.
|
||||
input.value = ''
|
||||
if (!file) return
|
||||
setActiveType(type)
|
||||
upload.mutate(
|
||||
{ type, file },
|
||||
{ onSettled: () => setActiveType(null) },
|
||||
)
|
||||
}
|
||||
|
||||
const uploadError = upload.isError ? extractErrorMessage(upload.error) : null
|
||||
|
||||
return (
|
||||
<section className={styles.section}>
|
||||
<h2 className={styles.sectionTitle}>Документы</h2>
|
||||
|
||||
{uploadError && <p className={styles.error}>{uploadError}</p>}
|
||||
|
||||
{isLoading && <div className={styles.state}>Загрузка...</div>}
|
||||
{isError && <div className={styles.state}>Не удалось загрузить документы</div>}
|
||||
|
||||
{!isLoading && !isError && (
|
||||
<ul className={styles.slots}>
|
||||
{DOCUMENT_SLOTS.map(({ type, label }) => {
|
||||
const doc = byType.get(type)
|
||||
const isUploading = upload.isPending && activeType === type
|
||||
return (
|
||||
<li key={type} className={styles.slot}>
|
||||
<div className={styles.slotInfo}>
|
||||
<span className={styles.slotLabel}>{label}</span>
|
||||
{doc ? (
|
||||
<span className={styles.slotFile}>
|
||||
{doc.file_name ?? '—'}
|
||||
<span className={styles.mono}> · {formatSize(doc.file_size_bytes)}</span>
|
||||
</span>
|
||||
) : (
|
||||
<span className={styles.muted}>Не загружен</span>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className={styles.slotActions}>
|
||||
{doc?.download_url && (
|
||||
<a
|
||||
className={styles.downloadBtn}
|
||||
href={doc.download_url}
|
||||
target="_blank"
|
||||
rel="noopener noreferrer"
|
||||
>
|
||||
Скачать
|
||||
</a>
|
||||
)}
|
||||
<UploadButton
|
||||
label={doc ? 'Заменить' : 'Загрузить'}
|
||||
busy={isUploading}
|
||||
disabled={upload.isPending}
|
||||
onSelect={(input) => handleSelect(type, input)}
|
||||
/>
|
||||
</div>
|
||||
</li>
|
||||
)
|
||||
})}
|
||||
</ul>
|
||||
)}
|
||||
</section>
|
||||
)
|
||||
}
|
||||
|
||||
interface UploadButtonProps {
|
||||
label: string
|
||||
busy: boolean
|
||||
disabled: boolean
|
||||
onSelect: (input: HTMLInputElement) => void
|
||||
}
|
||||
|
||||
function UploadButton({ label, busy, disabled, onSelect }: UploadButtonProps) {
|
||||
const inputRef = useRef<HTMLInputElement>(null)
|
||||
return (
|
||||
<>
|
||||
<button
|
||||
type="button"
|
||||
className={styles.uploadBtn}
|
||||
disabled={disabled}
|
||||
onClick={() => inputRef.current?.click()}
|
||||
>
|
||||
{busy ? 'Загрузка...' : label}
|
||||
</button>
|
||||
<input
|
||||
ref={inputRef}
|
||||
type="file"
|
||||
className={styles.hiddenInput}
|
||||
onChange={(e) => onSelect(e.currentTarget)}
|
||||
/>
|
||||
</>
|
||||
)
|
||||
}
|
||||
@@ -1 +0,0 @@
|
||||
export { OrganizationPurchaseRequests } from './ui/OrganizationPurchaseRequests'
|
||||
@@ -1,50 +0,0 @@
|
||||
.tableWrap {
|
||||
overflow-x: auto;
|
||||
}
|
||||
|
||||
.table {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
}
|
||||
|
||||
.table th {
|
||||
text-align: left;
|
||||
font-size: 12px;
|
||||
letter-spacing: 1.5px;
|
||||
text-transform: uppercase;
|
||||
color: var(--text-secondary);
|
||||
font-weight: 500;
|
||||
padding: 0 16px 14px;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.table td {
|
||||
padding: 14px 16px;
|
||||
border-top: 1px solid rgba(255, 255, 255, 0.06);
|
||||
vertical-align: middle;
|
||||
font-size: 14px;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.mono {
|
||||
font-family: var(--font-mono, monospace);
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.status {
|
||||
display: inline-block;
|
||||
padding: 4px 10px;
|
||||
border-radius: 999px;
|
||||
font-size: 12px;
|
||||
font-weight: 600;
|
||||
background: rgba(74, 109, 255, 0.12);
|
||||
color: #7c95ff;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.state {
|
||||
padding: 32px 16px;
|
||||
text-align: center;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.6));
|
||||
font-size: 14px;
|
||||
}
|
||||
@@ -1,63 +0,0 @@
|
||||
import { usePurchaseRequests } from '@features/admin'
|
||||
import styles from './OrganizationPurchaseRequests.module.css'
|
||||
|
||||
interface Props {
|
||||
orgId: string
|
||||
}
|
||||
|
||||
function formatAmount(value: string | null, suffix: string): string {
|
||||
if (!value) return '—'
|
||||
return `${value} ${suffix}`
|
||||
}
|
||||
|
||||
function formatDate(value: string | null): string {
|
||||
if (!value) return '—'
|
||||
const d = new Date(value)
|
||||
if (Number.isNaN(d.getTime())) return '—'
|
||||
return d.toLocaleString('ru-RU')
|
||||
}
|
||||
|
||||
export function OrganizationPurchaseRequests({ orgId }: Props) {
|
||||
const { data, isLoading, isError } = usePurchaseRequests(orgId)
|
||||
|
||||
if (isLoading) {
|
||||
return <div className={styles.state}>Загрузка...</div>
|
||||
}
|
||||
|
||||
if (isError) {
|
||||
return <div className={styles.state}>Не удалось загрузить заявки</div>
|
||||
}
|
||||
|
||||
if (!data || data.items.length === 0) {
|
||||
return <div className={styles.state}>Заявок пока нет</div>
|
||||
}
|
||||
|
||||
return (
|
||||
<div className={styles.tableWrap}>
|
||||
<table className={styles.table}>
|
||||
<thead>
|
||||
<tr>
|
||||
<th>USDT</th>
|
||||
<th>Сумма ₽</th>
|
||||
<th>Курс</th>
|
||||
<th>Статус</th>
|
||||
<th>Создана</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{data.items.map((req) => (
|
||||
<tr key={req.id}>
|
||||
<td className={styles.mono}>{formatAmount(req.usdt_amount, 'USDT')}</td>
|
||||
<td className={styles.mono}>{formatAmount(req.rub_amount, '₽')}</td>
|
||||
<td className={styles.mono}>{req.exchange_rate ?? '—'}</td>
|
||||
<td>
|
||||
<span className={styles.status}>{req.status}</span>
|
||||
</td>
|
||||
<td>{formatDate(req.created_at)}</td>
|
||||
</tr>
|
||||
))}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
@@ -1 +0,0 @@
|
||||
export { OrganizationWallets } from './ui/OrganizationWallets'
|
||||
@@ -1,52 +0,0 @@
|
||||
.section {
|
||||
background: rgba(255, 255, 255, 0.04);
|
||||
border: 1px solid rgba(255, 255, 255, 0.08);
|
||||
border-radius: 20px;
|
||||
padding: 24px;
|
||||
}
|
||||
|
||||
.sectionTitle {
|
||||
font-size: 14px;
|
||||
letter-spacing: 1.5px;
|
||||
text-transform: uppercase;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.5));
|
||||
font-weight: 600;
|
||||
margin: 0 0 18px;
|
||||
}
|
||||
|
||||
.table {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
}
|
||||
|
||||
.table th {
|
||||
text-align: left;
|
||||
font-size: 12px;
|
||||
letter-spacing: 1.5px;
|
||||
text-transform: uppercase;
|
||||
color: var(--text-secondary);
|
||||
font-weight: 500;
|
||||
padding: 0 16px 14px;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.table td {
|
||||
padding: 14px 16px;
|
||||
border-top: 1px solid rgba(255, 255, 255, 0.06);
|
||||
vertical-align: middle;
|
||||
font-size: 14px;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.mono {
|
||||
font-family: var(--font-mono, monospace);
|
||||
font-size: 13px;
|
||||
word-break: break-all;
|
||||
}
|
||||
|
||||
.state {
|
||||
padding: 32px 16px;
|
||||
text-align: center;
|
||||
color: var(--text-secondary, rgba(255, 255, 255, 0.6));
|
||||
font-size: 14px;
|
||||
}
|
||||
@@ -1,53 +0,0 @@
|
||||
import { useOrganizationWallets } from '@features/admin'
|
||||
import styles from './OrganizationWallets.module.css'
|
||||
|
||||
interface Props {
|
||||
orgId: string
|
||||
}
|
||||
|
||||
function formatDate(value: string | null): string {
|
||||
if (!value) return '—'
|
||||
const d = new Date(value)
|
||||
if (Number.isNaN(d.getTime())) return '—'
|
||||
return d.toLocaleString('ru-RU')
|
||||
}
|
||||
|
||||
export function OrganizationWallets({ orgId }: Props) {
|
||||
const { data: wallets, isLoading, isError } = useOrganizationWallets(orgId)
|
||||
|
||||
return (
|
||||
<section className={styles.section}>
|
||||
<h2 className={styles.sectionTitle}>Кошельки</h2>
|
||||
|
||||
{isLoading && <div className={styles.state}>Загрузка...</div>}
|
||||
{isError && <div className={styles.state}>Не удалось загрузить кошельки</div>}
|
||||
|
||||
{wallets && wallets.length === 0 && (
|
||||
<div className={styles.state}>Кошельки ещё не созданы</div>
|
||||
)}
|
||||
|
||||
{wallets && wallets.length > 0 && (
|
||||
<table className={styles.table}>
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Сеть</th>
|
||||
<th>Адрес</th>
|
||||
<th>Derivation path</th>
|
||||
<th>Создано</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{wallets.map((wallet) => (
|
||||
<tr key={wallet.id}>
|
||||
<td>{wallet.chain}</td>
|
||||
<td className={styles.mono}>{wallet.address}</td>
|
||||
<td className={styles.mono}>{wallet.derivation_path}</td>
|
||||
<td>{formatDate(wallet.created_at)}</td>
|
||||
</tr>
|
||||
))}
|
||||
</tbody>
|
||||
</table>
|
||||
)}
|
||||
</section>
|
||||
)
|
||||
}
|
||||
Reference in New Issue
Block a user