Compare commits
23 Commits
d67df0c2e1
...
feature/we
| Author | SHA1 | Date | |
|---|---|---|---|
| cc38f2e66c | |||
| d4c5db61ea | |||
| 55cdea17ea | |||
| 3725c7f569 | |||
| 2659c6a5b8 | |||
| 1bb3f43979 | |||
| 7e539f550b | |||
| fbf6b0dc9d | |||
| a997cdb198 | |||
| bf45dcdbfc | |||
| 83ccdef790 | |||
| 51d1870198 | |||
| 193f53c029 | |||
| 4bda233b63 | |||
| d758dbffa6 | |||
| 6af95bb449 | |||
| e3469763ce | |||
| 7f8a327329 | |||
| 53b8ce7095 | |||
| 94f512e0e4 | |||
| 60c6840db4 | |||
| 248eea6f85 | |||
| 7f8b90c15e |
14
.env
14
.env
@@ -1,8 +1,8 @@
|
|||||||
# VITE_API_URL='https://wn.st.unprism.ru'
|
VITE_API_URL='https://wn.st.unprism.ru'
|
||||||
# VITE_REACT_APP ='https://wn.st.unprism.ru/'
|
VITE_REACT_APP ='https://wn.st.unprism.ru/'
|
||||||
# VITE_KRBL_MEDIA='https://wn.st.unprism.ru/media/'
|
VITE_KRBL_MEDIA='https://wn.st.unprism.ru/media/'
|
||||||
# VITE_NEED_AUTH='true'
|
|
||||||
VITE_API_URL='https://wn.krbl.ru'
|
|
||||||
VITE_REACT_APP ='https://wn.krbl.ru/'
|
|
||||||
VITE_KRBL_MEDIA='https://wn.krbl.ru/media/'
|
|
||||||
VITE_NEED_AUTH='true'
|
VITE_NEED_AUTH='true'
|
||||||
|
# VITE_API_URL='https://wn.krbl.ru'
|
||||||
|
# VITE_REACT_APP ='https://wn.krbl.ru/'
|
||||||
|
# VITE_KRBL_MEDIA='https://wn.krbl.ru/media/'
|
||||||
|
# VITE_NEED_AUTH='true'
|
||||||
|
|||||||
4
Subtract.svg
Normal file
4
Subtract.svg
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
<svg width="24" height="21" viewBox="0 0 24 21" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M11.3888 2.85031C11.7638 2.66434 12.1989 2.66434 12.5739 2.85031L21.0037 7.06792C21.4462 7.29108 21.7245 7.73709 21.7245 8.22802V20.145H21.7464C21.7462 20.6134 21.3641 20.9999 20.8843 21H15.7689C15.3788 21 15.0562 20.6874 15.0562 20.2932V13.807C15.0562 12.1333 13.6909 10.7792 12.0033 10.7792C10.3159 10.7794 8.95116 12.1334 8.95115 13.807V20.2932C8.95115 20.68 8.63599 21 8.23847 21H3.10101C2.62861 21 2.23837 20.6209 2.23818 20.145V8.22802C2.23819 7.73714 2.51576 7.2911 2.95818 7.06792L11.3888 2.85031Z" fill="white"/>
|
||||||
|
<path d="M10.7962 0.268599C11.5012 -0.088374 12.3413 -0.0882526 13.0463 0.261335L23.6142 5.54243C23.9592 5.71352 24.1018 6.1379 23.9218 6.48751C23.7943 6.73297 23.5464 6.87397 23.2839 6.87397C23.179 6.87392 23.0663 6.85189 22.9689 6.79987L12.401 1.51877C12.101 1.37 11.7408 1.37 11.4408 1.51877L1.03036 6.79261C0.677932 6.97088 0.250655 6.82952 0.0781632 6.48751C-0.101845 6.14534 0.0407786 5.72096 0.385795 5.54243L10.7962 0.268599Z" fill="white"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.1 KiB |
4
package-lock.json
generated
4
package-lock.json
generated
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "white-nights",
|
"name": "white-nights",
|
||||||
"version": "1.0.6",
|
"version": "1.0.8",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "white-nights",
|
"name": "white-nights",
|
||||||
"version": "1.0.6",
|
"version": "1.0.8",
|
||||||
"license": "UNLICENSED",
|
"license": "UNLICENSED",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@emotion/react": "^11.14.0",
|
"@emotion/react": "^11.14.0",
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "white-nights",
|
"name": "white-nights",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "1.0.6",
|
"version": "1.0.8",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"license": "UNLICENSED",
|
"license": "UNLICENSED",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
@@ -1,18 +1,25 @@
|
|||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
|
|
||||||
import { Router } from "./router";
|
import { Router } from "./router";
|
||||||
import { CustomTheme } from "@shared";
|
import { CustomTheme, languageStore } from "@shared";
|
||||||
import { ThemeProvider } from "@mui/material/styles";
|
import { ThemeProvider } from "@mui/material/styles";
|
||||||
import { ToastContainer } from "react-toastify";
|
import { ToastContainer } from "react-toastify";
|
||||||
import { GlobalErrorBoundary } from "./GlobalErrorBoundary";
|
import { GlobalErrorBoundary } from "./GlobalErrorBoundary";
|
||||||
import { TestingModeBanner } from "@widgets";
|
import { TestingModeBanner } from "@widgets";
|
||||||
|
import { observer } from "mobx-react-lite";
|
||||||
|
|
||||||
export const App: React.FC = () => (
|
export const App: React.FC = observer(() => {
|
||||||
<GlobalErrorBoundary>
|
React.useEffect(() => {
|
||||||
<ThemeProvider theme={CustomTheme.Light}>
|
document.documentElement.lang = languageStore.language;
|
||||||
<TestingModeBanner />
|
}, [languageStore.language]);
|
||||||
<ToastContainer />
|
|
||||||
<Router />
|
return (
|
||||||
</ThemeProvider>
|
<GlobalErrorBoundary>
|
||||||
</GlobalErrorBoundary>
|
<ThemeProvider theme={CustomTheme.Light}>
|
||||||
);
|
<TestingModeBanner />
|
||||||
|
<ToastContainer />
|
||||||
|
<Router />
|
||||||
|
</ThemeProvider>
|
||||||
|
</GlobalErrorBoundary>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|||||||
@@ -92,6 +92,9 @@ const ProtectedRoute = ({ children }: { children: React.ReactNode }) => {
|
|||||||
requiredPermissions.length > 0 &&
|
requiredPermissions.length > 0 &&
|
||||||
!requiredPermissions.every((permission) => authStore.canAccess(permission))
|
!requiredPermissions.every((permission) => authStore.canAccess(permission))
|
||||||
) {
|
) {
|
||||||
|
if (location.pathname === "/devices" && authStore.hasRole("devices_maintenance_rw")) {
|
||||||
|
return <>{children}</>;
|
||||||
|
}
|
||||||
return <Navigate to="/" replace />;
|
return <Navigate to="/" replace />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
4
src/client/src/App.d.ts
vendored
Normal file
4
src/client/src/App.d.ts
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
import React from "react";
|
||||||
|
|
||||||
|
declare const App: React.FC;
|
||||||
|
export default App;
|
||||||
@@ -21,7 +21,57 @@ import {
|
|||||||
GetCityResponse,
|
GetCityResponse,
|
||||||
GetSightArticleResponse,
|
GetSightArticleResponse,
|
||||||
} from "./types";
|
} from "./types";
|
||||||
|
// @ts-ignore
|
||||||
import { orderStationsByRoute } from "../../utils/routeStationsUtils";
|
import { orderStationsByRoute } from "../../utils/routeStationsUtils";
|
||||||
|
import { resamplePath } from "../../utils/animationUtils";
|
||||||
|
import { colorStore } from "../../stores/ColorStore";
|
||||||
|
|
||||||
|
function hexToRgbString(hex: string): string | null {
|
||||||
|
const clean = hex.trim().replace(/^#/, "");
|
||||||
|
const full = clean.length === 3 ? clean.split("").map((c) => c + c).join("") : clean;
|
||||||
|
if (full.length !== 6) return null;
|
||||||
|
const r = parseInt(full.slice(0, 2), 16);
|
||||||
|
const g = parseInt(full.slice(2, 4), 16);
|
||||||
|
const b = parseInt(full.slice(4, 6), 16);
|
||||||
|
return `${r}, ${g}, ${b}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function darkenHex(hex: string, amount: number): string {
|
||||||
|
const rgb = hexToRgbString(hex);
|
||||||
|
if (!rgb) return hex;
|
||||||
|
const [r, g, b] = rgb.split(",").map(Number);
|
||||||
|
const factor = 1 - amount;
|
||||||
|
const dr = Math.round(r * factor);
|
||||||
|
const dg = Math.round(g * factor);
|
||||||
|
const db = Math.round(b * factor);
|
||||||
|
return `#${dr.toString(16).padStart(2, "0")}${dg.toString(16).padStart(2, "0")}${db.toString(16).padStart(2, "0")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function lightenRgbString(hex: string, amount: number): string | null {
|
||||||
|
const rgb = hexToRgbString(hex);
|
||||||
|
if (!rgb) return null;
|
||||||
|
const [r, g, b] = rgb.split(",").map(Number);
|
||||||
|
const lr = Math.round(r + (255 - r) * amount);
|
||||||
|
const lg = Math.round(g + (255 - g) * amount);
|
||||||
|
const lb = Math.round(b + (255 - b) * amount);
|
||||||
|
return `${lr}, ${lg}, ${lb}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function applyCarrierColors(carrier: { main_color?: string; left_color?: string; right_color?: string }) {
|
||||||
|
const mainColor = carrier.main_color || "#006f3a";
|
||||||
|
const leftColor = carrier.left_color || "#006f3a";
|
||||||
|
const rightColor = carrier.right_color || "#006f3a";
|
||||||
|
const mainDark = darkenHex(mainColor, 0.3);
|
||||||
|
|
||||||
|
document.documentElement.style.setProperty("--carrier-main", mainColor);
|
||||||
|
document.documentElement.style.setProperty("--carrier-main-rgb", hexToRgbString(mainColor) ?? "0, 111, 58");
|
||||||
|
document.documentElement.style.setProperty("--carrier-main-dark", mainDark);
|
||||||
|
document.documentElement.style.setProperty("--carrier-left", leftColor);
|
||||||
|
document.documentElement.style.setProperty("--carrier-left-rgb", hexToRgbString(leftColor) ?? "0, 111, 58");
|
||||||
|
document.documentElement.style.setProperty("--carrier-right", rightColor);
|
||||||
|
document.documentElement.style.setProperty("--carrier-right-rgb", hexToRgbString(rightColor) ?? "0, 111, 58");
|
||||||
|
document.documentElement.style.setProperty("--carrier-right-menu-rgb", lightenRgbString(rightColor, 0.38) ?? "179, 165, 152");
|
||||||
|
}
|
||||||
|
|
||||||
class ApiStore {
|
class ApiStore {
|
||||||
isLoading = true;
|
isLoading = true;
|
||||||
@@ -54,13 +104,14 @@ class ApiStore {
|
|||||||
carrier: GetCarrierResponse | null = null;
|
carrier: GetCarrierResponse | null = null;
|
||||||
city: GetCityResponse | null = null;
|
city: GetCityResponse | null = null;
|
||||||
|
|
||||||
private positionIndex = 0;
|
positionIndex = 0;
|
||||||
private positionInterval: ReturnType<typeof setInterval> | null = null;
|
private positionInterval: ReturnType<typeof setInterval> | null = null;
|
||||||
|
|
||||||
simulationSpeed = 1;
|
simulationSpeed = 1;
|
||||||
simulationDirection: 1 | -1 = 1;
|
simulationDirection: 1 | -1 = 1;
|
||||||
simulationPaused = false;
|
simulationPaused = false;
|
||||||
simulationInstantMove = false;
|
simulationInstantMove = false;
|
||||||
|
showHitboxes = false;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
makeAutoObservable(this);
|
makeAutoObservable(this);
|
||||||
@@ -87,12 +138,35 @@ class ApiStore {
|
|||||||
};
|
};
|
||||||
|
|
||||||
getRoute = async () => {
|
getRoute = async () => {
|
||||||
this.route = await getRoute(this.routeId!);
|
const route = await getRoute(this.routeId!);
|
||||||
|
if (route.path && route.path.length > 1) {
|
||||||
|
// Рассчитываем общую дистанцию для выбора адекватного шага ресемплинга
|
||||||
|
let totalDist = 0;
|
||||||
|
for (let i = 0; i < route.path.length - 1; i++) {
|
||||||
|
const p1 = route.path[i];
|
||||||
|
const p2 = route.path[i + 1];
|
||||||
|
totalDist += Math.sqrt(
|
||||||
|
Math.pow(p2[0] - p1[0], 2) + Math.pow(p2[1] - p1[1], 2)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// Хотим иметь примерно 2000 точек для равномерности и плавности
|
||||||
|
const segmentLength = totalDist / 2000;
|
||||||
|
if (segmentLength > 0) {
|
||||||
|
route.path = resamplePath(route.path as [number, number][], segmentLength);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
runInAction(() => {
|
||||||
|
this.route = route;
|
||||||
|
});
|
||||||
this.updateOrderedRouteStations();
|
this.updateOrderedRouteStations();
|
||||||
};
|
};
|
||||||
|
|
||||||
getCarrier = async () => {
|
getCarrier = async () => {
|
||||||
this.carrier = await getCarrier(this.route!.carrier_id!);
|
this.carrier = await getCarrier(this.route!.carrier_id!);
|
||||||
|
applyCarrierColors(this.carrier);
|
||||||
|
if (this.carrier.main_color) {
|
||||||
|
colorStore.setMainColor(this.carrier.main_color);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
getCity = async () => {
|
getCity = async () => {
|
||||||
@@ -170,6 +244,10 @@ class ApiStore {
|
|||||||
this.simulationInstantMove = !this.simulationInstantMove;
|
this.simulationInstantMove = !this.simulationInstantMove;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
toggleShowHitboxes = () => {
|
||||||
|
this.showHitboxes = !this.showHitboxes;
|
||||||
|
};
|
||||||
|
|
||||||
startPositionSimulation = () => {
|
startPositionSimulation = () => {
|
||||||
if (this.positionInterval) return;
|
if (this.positionInterval) return;
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ export type GetContextResponse = {
|
|||||||
};
|
};
|
||||||
endStopId: string;
|
endStopId: string;
|
||||||
nearestSightId: string;
|
nearestSightId: string;
|
||||||
|
nearestStationId?: string | null;
|
||||||
rawCoordinates: {
|
rawCoordinates: {
|
||||||
latitude: number;
|
latitude: number;
|
||||||
longitude: number;
|
longitude: number;
|
||||||
@@ -54,6 +55,7 @@ export type GetRouteResponse = {
|
|||||||
center_latitude: number;
|
center_latitude: number;
|
||||||
center_longitude: number;
|
center_longitude: number;
|
||||||
governor_appeal: number;
|
governor_appeal: number;
|
||||||
|
button_text?: string;
|
||||||
id: number;
|
id: number;
|
||||||
path: [number, number][];
|
path: [number, number][];
|
||||||
rotate: number;
|
rotate: number;
|
||||||
@@ -77,6 +79,9 @@ export type GetCarrierResponse = {
|
|||||||
logo: string;
|
logo: string;
|
||||||
short_name: string;
|
short_name: string;
|
||||||
slogan: string;
|
slogan: string;
|
||||||
|
main_color?: string;
|
||||||
|
left_color?: string;
|
||||||
|
right_color?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type GetCityResponse = {
|
export type GetCityResponse = {
|
||||||
@@ -105,6 +110,7 @@ export type GetRouteSightsResponse = {
|
|||||||
icon?: string;
|
icon?: string;
|
||||||
alt_icon?: string;
|
alt_icon?: string;
|
||||||
is_default_icon?: boolean;
|
is_default_icon?: boolean;
|
||||||
|
short_name?: string;
|
||||||
}[];
|
}[];
|
||||||
|
|
||||||
export type GetRouteStationsResponse = {
|
export type GetRouteStationsResponse = {
|
||||||
|
|||||||
9
src/client/src/api/apiConfig.d.ts
vendored
Normal file
9
src/client/src/api/apiConfig.d.ts
vendored
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
import { AxiosInstance } from "axios";
|
||||||
|
|
||||||
|
export declare const apiBaseURL: string;
|
||||||
|
export declare const geoBaseURL: string;
|
||||||
|
export declare const weatherBaseURL: string;
|
||||||
|
export declare const getMediaUrl: (id: string) => string;
|
||||||
|
export declare const apiInstance: AxiosInstance;
|
||||||
|
export declare const geoInstance: AxiosInstance;
|
||||||
|
export declare const weatherInstance: AxiosInstance;
|
||||||
8
src/client/src/assets/Constants.d.ts
vendored
Normal file
8
src/client/src/assets/Constants.d.ts
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
export declare const UP_SCALE: number;
|
||||||
|
export declare const PATH_WIDTH: number;
|
||||||
|
export declare const STATION_RADIUS: number;
|
||||||
|
export declare const STATION_OUTLINE_WIDTH: number;
|
||||||
|
export declare const SIGHT_SIZE: number;
|
||||||
|
export declare const SCALE_FACTOR: number;
|
||||||
|
export declare const BACKGROUND_COLOR: number;
|
||||||
|
export declare const PATH_COLOR: number;
|
||||||
4
src/client/src/assets/icons/subtract-home.svg
Normal file
4
src/client/src/assets/icons/subtract-home.svg
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
<svg width="24" height="21" viewBox="0 0 24 21" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M11.3888 2.85031C11.7638 2.66434 12.1989 2.66434 12.5739 2.85031L21.0037 7.06792C21.4462 7.29108 21.7245 7.73709 21.7245 8.22802V20.145H21.7464C21.7462 20.6134 21.3641 20.9999 20.8843 21H15.7689C15.3788 21 15.0562 20.6874 15.0562 20.2932V13.807C15.0562 12.1333 13.6909 10.7792 12.0033 10.7792C10.3159 10.7794 8.95116 12.1334 8.95115 13.807V20.2932C8.95115 20.68 8.63599 21 8.23847 21H3.10101C2.62861 21 2.23837 20.6209 2.23818 20.145V8.22802C2.23819 7.73714 2.51576 7.2911 2.95818 7.06792L11.3888 2.85031Z" fill="white"/>
|
||||||
|
<path d="M10.7962 0.268599C11.5012 -0.088374 12.3413 -0.0882526 13.0463 0.261335L23.6142 5.54243C23.9592 5.71352 24.1018 6.1379 23.9218 6.48751C23.7943 6.73297 23.5464 6.87397 23.2839 6.87397C23.179 6.87392 23.0663 6.85189 22.9689 6.79987L12.401 1.51877C12.101 1.37 11.7408 1.37 11.4408 1.51877L1.03036 6.79261C0.677932 6.97088 0.250655 6.82952 0.0781632 6.48751C-0.101845 6.14534 0.0407786 5.72096 0.385795 5.54243L10.7962 0.268599Z" fill="white"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.1 KiB |
3
src/client/src/assets/icons/three-view-pan.svg
Normal file
3
src/client/src/assets/icons/three-view-pan.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="15" height="14" viewBox="0 0 15 14" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M13.9941 5.96193C13.028 4.86487 12.0189 3.81 11.0097 2.73404C10.5374 2.2277 9.78591 2.14332 9.18473 2.50197C8.86267 2.69185 8.64796 2.98721 8.51914 3.32476C7.51002 2.33319 6.47943 1.42601 5.47031 0.434434C4.69737 -0.325069 3.47355 -0.0508043 3.10855 0.961866C3.10855 1.00406 3.08707 1.00406 3.0656 1.02516C2.65766 0.750893 2.18531 0.708699 1.69148 0.919672C1.11178 1.17284 0.811185 1.61588 0.789714 2.2488C0.789714 2.56526 0.897071 2.86062 1.06884 3.13489C0.746777 3.24038 0.489128 3.40915 0.295893 3.66232C-0.154991 4.25305 -0.09058 5.05474 0.489127 5.60327C2.16384 7.24886 3.83854 8.89445 5.53472 10.5611C5.57767 10.6033 5.62061 10.6455 5.68502 10.7088C5.66355 10.7088 5.64208 10.7088 5.62061 10.7088C4.63296 10.8565 3.62384 11.0253 2.63619 11.173C1.88472 11.2784 1.3909 11.6793 1.17619 12.3966C0.918542 13.2405 1.43383 14 2.29266 14C4.71884 14 7.12355 14 9.54973 14C10.7091 14 11.6968 13.5992 12.5127 12.7975C12.9421 12.3755 13.3715 11.9536 13.8009 11.5316C15.3253 9.99151 15.4112 7.56532 13.9727 5.94083L13.9941 5.96193ZM13.2212 11.0675C12.8133 11.4683 12.4053 11.8692 11.9974 12.27C11.3103 12.924 10.4944 13.2616 9.52826 13.2616C8.36885 13.2616 7.18796 13.2616 6.02855 13.2616C4.86913 13.2616 3.60236 13.2616 2.37854 13.2616C1.99207 13.2616 1.82031 13.0506 1.92766 12.692C2.01354 12.2911 2.3356 12.0169 2.76501 11.9536C4.01031 11.7637 5.27708 11.5738 6.52237 11.3839C6.69414 11.3628 6.84443 11.2995 6.88737 11.1308C6.95178 10.9409 6.88737 10.7932 6.73708 10.6666C4.84767 8.81006 2.95825 6.9535 1.06884 5.09694C0.83266 4.86487 0.746777 4.6117 0.85413 4.31634C1.00442 3.87329 1.56266 3.72561 1.94913 3.99988C2.01354 4.04207 2.05648 4.10536 2.1209 4.14756C3.34472 5.35011 4.59001 6.57375 5.81384 7.77629C5.92119 7.88178 6.02855 7.96617 6.17884 7.94507C6.47943 7.94507 6.62972 7.60752 6.47943 7.37545C6.43649 7.31215 6.37208 7.24886 6.32914 7.20667C4.84767 5.72986 3.34472 4.27414 1.84177 2.81843C1.64854 2.62855 1.54119 2.39648 1.6056 2.12222C1.73443 1.61588 2.3356 1.42601 2.74354 1.76356C2.78648 1.80576 2.82943 1.84795 2.89384 1.89015C4.35384 3.32476 5.83532 4.78048 7.29532 6.2151C7.33826 6.25729 7.35973 6.29948 7.40267 6.32058C7.57444 6.46826 7.81061 6.48936 7.9609 6.32058C8.1112 6.1729 8.1112 5.94083 7.9609 5.77205C7.72473 5.53998 7.25237 5.07584 7.25237 5.07584L6.24325 4.08427C6.24325 4.08427 5.49178 3.34586 5.12678 2.98721C4.76178 2.62855 4.41825 2.291 4.05325 1.93234C3.88149 1.74247 3.77413 1.55259 3.83855 1.27833C3.9459 0.814185 4.50413 0.62431 4.89061 0.919672C4.95502 0.961866 4.99796 1.02516 5.0409 1.06735C6.1359 2.14332 7.25238 3.11379 8.34738 4.18975C8.54061 4.37963 8.71238 4.56951 8.90561 4.73828C9.09885 4.92816 9.33502 4.94926 9.50679 4.80158C9.67855 4.65389 9.65708 4.42182 9.46385 4.21085C9.18473 3.89439 9.2062 3.47245 9.50679 3.21928C9.80738 2.96611 10.2368 2.98721 10.5159 3.28257C11.4606 4.29524 12.4053 5.32901 13.35 6.34168C14.5953 7.713 14.5524 9.73834 13.2427 11.0464L13.2212 11.0675Z" fill="white"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 3.0 KiB |
3
src/client/src/assets/icons/three-view-rotate.svg
Normal file
3
src/client/src/assets/icons/three-view-rotate.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="15" height="12" viewBox="0 0 15 12" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M13.993 3.76793C13.0258 2.64439 12.0155 1.56407 11.0053 0.462135C10.5324 -0.0564207 9.78007 -0.142847 9.17822 0.224464C8.8558 0.418922 8.64085 0.721413 8.51188 1.06712C7.69509 0.505348 6.57737 0.786232 6.19046 1.82334C6.19046 1.82334 6.19046 1.82334 6.19046 1.88816C5.78206 1.62889 5.33068 1.60728 4.85779 1.82334C4.40641 2.0178 4.14847 2.3419 4.0195 2.75242C3.6326 2.36351 2.60085 1.348 2.47189 1.21836C2.0205 0.786232 1.46164 0.699806 0.902776 0.959084C0.773808 1.0023 0.687829 1.06712 0.580356 1.15354C0.365409 1.30479 0.236441 1.49925 0.128968 1.73692C0.128968 1.78013 0.107473 1.82334 0.0859786 1.86656C0 1.9962 0 2.14744 0 2.29869C0 2.53636 0.064484 2.75242 0.150463 2.96849C0.150463 2.96849 0.343915 3.27098 0.472883 3.40062C2.14947 5.08592 3.82605 6.77123 5.52413 8.47814C5.56712 8.52135 5.61011 8.56457 5.67459 8.62939C5.6531 8.62939 5.6316 8.62939 5.61011 8.62939C4.62135 8.78063 3.6111 8.95348 2.62235 9.10473C1.87004 9.21276 1.37566 9.62329 1.16071 10.3579C0.902776 11.2222 1.41865 12 2.27843 12C4.70733 12 7.11473 12 9.54363 12C10.7043 12 11.6931 11.5895 12.5099 10.7684C12.9398 10.3363 13.3697 9.90417 13.7996 9.47204C15.3257 7.89477 15.4117 5.41002 13.9715 3.74632L13.993 3.76793ZM13.2192 8.9967C12.8108 9.40722 12.4024 9.81774 11.994 10.2283C11.3062 10.8981 10.4894 11.2438 9.52213 11.2438C8.36142 11.2438 7.17922 11.2438 6.0185 11.2438C4.85779 11.2438 3.58961 11.2438 2.36441 11.2438C1.97751 11.2438 1.80555 11.0277 1.91302 10.6604C1.999 10.2499 2.32142 9.96899 2.75132 9.90417C3.99801 9.70971 5.26619 9.51525 6.51288 9.3208C6.68484 9.29919 6.8353 9.23437 6.87829 9.06152C6.94278 8.86706 6.87829 8.71581 6.72783 8.58617C4.8363 6.6848 2.94477 4.78343 1.05324 2.88206C0.92427 2.75242 0.838292 2.60118 0.795302 2.44993C0.795302 2.44993 0.795302 2.40672 0.795302 2.38511C0.795302 2.38511 0.795302 2.3419 0.795302 2.32029C0.795302 2.27708 0.795302 2.23387 0.795302 2.16905C0.795302 2.16905 0.795302 2.14744 0.795302 2.12583C0.795302 2.10423 0.795302 2.08262 0.816797 2.06101C0.816797 2.03941 0.838292 1.9962 0.859786 1.97459C1.07473 1.60728 1.56911 1.49925 1.93452 1.80174C1.97751 1.84495 2.0205 1.88816 2.08498 1.93138C3.05224 2.90367 4.0195 3.87596 4.98676 4.84825C5.2447 5.10753 5.52413 5.38841 5.78206 5.64769C5.88954 5.75572 5.99701 5.84215 6.14747 5.82054C6.4484 5.82054 6.59886 5.47484 6.4484 5.23717C6.40541 5.17235 6.34092 5.10753 6.29794 5.06432C5.84655 4.61058 5.37366 4.13524 4.92228 3.6815C4.77181 3.50865 4.70733 3.29258 4.77181 3.05491C4.90078 2.53636 5.50263 2.3419 5.91103 2.6876C5.95402 2.73082 5.99701 2.77403 6.06149 2.81724C6.38391 3.14134 6.68484 3.44383 7.00726 3.76793C7.09324 3.85435 7.17922 3.94078 7.28669 4.04881C7.32968 4.09202 7.35117 4.13524 7.39416 4.15684C7.56612 4.30809 7.80256 4.3297 7.95302 4.15684C8.10349 4.0056 8.10349 3.76793 7.95302 3.59508C7.80256 3.44383 7.6521 3.27098 7.48014 3.11973C7.35117 2.99009 7.2437 2.88206 7.11473 2.75242C6.8568 2.42833 6.92128 2.0178 7.15772 1.78013C7.45865 1.47764 7.86705 1.49925 8.23245 1.86656C8.4689 2.10423 8.68384 2.32029 8.92028 2.55796C9.11374 2.75242 9.35018 2.77403 9.52213 2.62278C9.69409 2.47154 9.6726 2.23387 9.47915 2.0178C9.19971 1.6937 9.22121 1.26158 9.52213 1.0023C9.82306 0.743019 10.253 0.764626 10.5324 1.06712C11.4781 2.10423 12.4239 3.16295 13.3697 4.20006C14.6164 5.60448 14.5734 7.6787 13.2622 9.0183L13.2192 8.9967Z" fill="white"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 3.4 KiB |
3
src/client/src/assets/icons/three-view-zoom.svg
Normal file
3
src/client/src/assets/icons/three-view-zoom.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="15" height="13" viewBox="0 0 15 13" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M5.67979 9.7487C5.63689 9.68658 5.57254 9.64516 5.52964 9.60374C3.85647 7.98845 2.16185 6.37316 0.488679 4.73716C-0.0904948 4.17802 -0.154849 3.41179 0.295619 2.83194C0.488677 2.58343 0.746089 2.41776 1.06785 2.31422C0.896245 2.045 0.767539 1.77579 0.78899 1.44444C0.78899 0.823178 1.11075 0.388292 1.68993 0.139785C2.2691 -0.108722 2.80537 -0.0258861 3.25584 0.388291C3.87792 0.96814 4.47854 1.5687 5.07917 2.14855C5.44383 2.5006 5.8085 2.85265 6.19461 3.22541C6.58073 2.23138 7.69618 1.94146 8.51131 2.5006C8.64001 2.14855 8.85452 1.87933 9.17629 1.69295C9.77691 1.3409 10.5277 1.42374 10.9996 1.92075C12.0078 2.9769 13.016 4.01235 13.9813 5.08921C15.4185 6.68379 15.3112 9.06531 13.8097 10.5771C13.3807 10.9912 12.9516 11.4054 12.5226 11.8196C11.7075 12.5858 10.7207 12.9793 9.5624 13C7.13845 13 4.73595 13 2.312 13C1.45397 13 0.917694 12.2752 1.19656 11.4261C1.41106 10.722 1.92589 10.3493 2.65522 10.225C3.64196 10.08 4.65015 9.91438 5.63689 9.76941C5.63689 9.76941 5.65834 9.76941 5.70124 9.76941L5.67979 9.7487ZM6.023 12.2545C7.18135 12.2545 8.36115 12.2545 9.5195 12.2545C10.4848 12.2545 11.2999 11.9231 11.9864 11.2812C12.3939 10.8877 12.8015 10.4942 13.2091 10.1008C14.5176 8.81681 14.5819 6.82875 13.3163 5.48268C12.3725 4.46794 11.4286 3.47391 10.4848 2.47989C10.2059 2.18996 9.77691 2.14855 9.4766 2.41776C9.17629 2.68698 9.15484 3.10115 9.4337 3.39108C9.62675 3.59817 9.62675 3.80526 9.4766 3.97093C9.30499 4.11589 9.09048 4.09518 8.87597 3.9088C8.64001 3.70171 8.4255 3.47391 8.18955 3.24612C7.84633 2.91477 7.41731 2.87336 7.117 3.16328C6.85959 3.41179 6.79524 3.86738 7.18135 4.19873C7.43877 4.40581 7.65327 4.65432 7.88923 4.88212C8.06084 5.04779 8.06084 5.27559 7.88923 5.42055C7.73908 5.56551 7.50312 5.56551 7.33151 5.42055C7.28861 5.37913 7.24571 5.33771 7.22426 5.317C5.7656 3.9088 4.28548 2.47989 2.82682 1.07168C2.78392 1.03027 2.74102 0.988849 2.67667 0.947431C2.2691 0.616089 1.66848 0.802469 1.53977 1.29948C1.47542 1.5687 1.58267 1.7965 1.77573 1.98288C3.27729 3.41179 4.7574 4.86141 6.25897 6.29032C6.32332 6.35245 6.36622 6.39387 6.40912 6.45599C6.55928 6.7045 6.40912 6.99442 6.10881 7.01513C5.95865 7.01513 5.8514 6.9323 5.74415 6.84946C4.52145 5.66906 3.27729 4.46794 2.05459 3.28753C1.99024 3.22541 1.94734 3.18399 1.88299 3.14257C1.49687 2.87336 0.960597 3.01832 0.78899 3.45321C0.681736 3.76384 0.78899 4.01235 1.0035 4.21943C2.89118 6.04182 4.77885 7.8642 6.66653 9.68658C6.79524 9.81083 6.88104 9.95579 6.81669 10.1422C6.75234 10.3078 6.62363 10.37 6.45202 10.3907C5.20787 10.5771 3.94227 10.7634 2.69812 10.9498C2.2691 11.0119 1.96879 11.2812 1.86153 11.6746C1.77573 12.0474 1.92589 12.2338 2.312 12.2338C3.5347 12.2338 4.73595 12.2338 5.95865 12.2338L6.023 12.2545Z" fill="white"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.8 KiB |
@@ -37,14 +37,14 @@ const Fullscreen3DModal = ({ isOpen, onClose, fileUrl }) => {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="fullscreen-3d-actions">
|
<div className="fullscreen-3d-actions">
|
||||||
<button title="Увеличить масштаб" disabled={scale >= 1}>
|
<button disabled={scale >= 1}>
|
||||||
<img src={scale_plus} alt="Увеличить" />
|
<img src={scale_plus} alt="Увеличить" />
|
||||||
</button>
|
</button>
|
||||||
<button title="Уменьшить масштаб" disabled={scale <= 0.1}>
|
<button disabled={scale <= 0.1}>
|
||||||
<img src={scale_minus} alt="Уменьшить" />
|
<img src={scale_minus} alt="Уменьшить" />
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button onPointerUp={onClose} title="Закрыть">
|
<button onPointerUp={onClose}>
|
||||||
<img src={closeIcon} alt="Закрыть" />
|
<img src={closeIcon} alt="Закрыть" />
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -411,13 +411,13 @@ const ListOfSights = observer(() => {
|
|||||||
}, [currentSelectedSight]);
|
}, [currentSelectedSight]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="right-widget">
|
<div className="right-widget" lang={selectedLanguageRight}>
|
||||||
{currentSelectedSight && (
|
{currentSelectedSight && (
|
||||||
<SightFrame
|
<SightFrame
|
||||||
key={currentSelectedSight.id}
|
key={currentSelectedSight.id}
|
||||||
media={sightFrameMedia}
|
media={sightFrameMedia}
|
||||||
sight_id={currentSelectedSight.id}
|
sight_id={currentSelectedSight.id}
|
||||||
sight_name={currentSelectedSight.short_name || currentSelectedSight.name}
|
sight_name={currentSelectedSight.name}
|
||||||
selectedLanguageRight={selectedLanguageRight}
|
selectedLanguageRight={selectedLanguageRight}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import React from "react";
|
import React, { useState, useEffect, useRef } from "react";
|
||||||
|
|
||||||
const LANGUAGES = {
|
const LANGUAGES = {
|
||||||
EN: "en",
|
EN: "en",
|
||||||
@@ -18,6 +18,26 @@ const ListHeader = function ListHeader({
|
|||||||
isTransferWidgetOpen,
|
isTransferWidgetOpen,
|
||||||
onBackToNearest,
|
onBackToNearest,
|
||||||
}) {
|
}) {
|
||||||
|
const [isIdle, setIsIdle] = useState(false);
|
||||||
|
const timerRef = useRef(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const resetTimer = () => {
|
||||||
|
setIsIdle(false);
|
||||||
|
clearTimeout(timerRef.current);
|
||||||
|
timerRef.current = setTimeout(() => setIsIdle(true), 15000);
|
||||||
|
};
|
||||||
|
|
||||||
|
const events = ["pointermove", "pointerdown", "touchstart", "keydown"];
|
||||||
|
events.forEach((e) => window.addEventListener(e, resetTimer));
|
||||||
|
resetTimer();
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
clearTimeout(timerRef.current);
|
||||||
|
events.forEach((e) => window.removeEventListener(e, resetTimer));
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
const getTitle = () => {
|
const getTitle = () => {
|
||||||
return selectedLanguageRight === LANGUAGES.RU
|
return selectedLanguageRight === LANGUAGES.RU
|
||||||
? "Достопримечательности"
|
? "Достопримечательности"
|
||||||
@@ -41,14 +61,11 @@ const ListHeader = function ListHeader({
|
|||||||
>
|
>
|
||||||
<svg
|
<svg
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
width="21"
|
width="20"
|
||||||
height="12"
|
height="20"
|
||||||
viewBox="0 0 21 12"
|
viewBox="0 0 21 12"
|
||||||
fill="none"
|
fill="none"
|
||||||
style={{
|
className={`chevron-svg${isOpen ? " is-open" : ""}${isIdle ? " is-idle" : ""}`}
|
||||||
transform: isOpen ? "rotate(180deg)" : "rotate(0deg)",
|
|
||||||
transition: "transform 0.15s ease-in-out",
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<g clipPath="url(#clip0_658_91932)">
|
<g clipPath="url(#clip0_658_91932)">
|
||||||
<path
|
<path
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ const SightComponent = function SightComponent({
|
|||||||
aria-label={`Выбрать достопримечательность ${title}`}
|
aria-label={`Выбрать достопримечательность ${title}`}
|
||||||
>
|
>
|
||||||
<div className="sight-image">{renderThumbnail()}</div>
|
<div className="sight-image">{renderThumbnail()}</div>
|
||||||
<div className="sight-title" title={title}>
|
<div className="sight-title">
|
||||||
{title}
|
{title}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import React, { useState, useEffect, useRef, useMemo } from "react";
|
import React, { useState, useEffect, useRef, useMemo, useLayoutEffect, useCallback } from "react";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { useGeolocationStore } from "../../stores";
|
import { useGeolocationStore } from "../../stores";
|
||||||
@@ -16,6 +16,10 @@ import { ThreeViewErrorBoundary } from "../ThreeViewErrorBoundary";
|
|||||||
import { apiStore } from "../../api/ApiStore/store";
|
import { apiStore } from "../../api/ApiStore/store";
|
||||||
import { ReactMarkdownComponent } from "../ReactMarkdown";
|
import { ReactMarkdownComponent } from "../ReactMarkdown";
|
||||||
import { TouchableLayout } from "../TouchableLayout";
|
import { TouchableLayout } from "../TouchableLayout";
|
||||||
|
import rotate3DIcon from "../../assets/icons/three-view-rotate.svg";
|
||||||
|
import zoom3DIcon from "../../assets/icons/three-view-zoom.svg";
|
||||||
|
import pan3DIcon from "../../assets/icons/three-view-pan.svg";
|
||||||
|
import subtractHomeIcon from "../../assets/icons/subtract-home.svg";
|
||||||
|
|
||||||
const Watermark = ({ path }) => {
|
const Watermark = ({ path }) => {
|
||||||
if (!path) return null;
|
if (!path) return null;
|
||||||
@@ -39,8 +43,95 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
const [threeViewResetKey, setThreeViewResetKey] = useState(0);
|
const [threeViewResetKey, setThreeViewResetKey] = useState(0);
|
||||||
const threeViewControlRef = useRef(null);
|
const threeViewControlRef = useRef(null);
|
||||||
const mediaCache = useRef({});
|
const mediaCache = useRef({});
|
||||||
|
const idleTimerRef = useRef(null);
|
||||||
|
|
||||||
const textWrapperRef = useRef(null);
|
const textWrapperRef = useRef(null);
|
||||||
|
const menuRef = useRef(null);
|
||||||
|
const [menuNeedsScroll, setMenuNeedsScroll] = useState(false);
|
||||||
|
const [canScrollLeft, setCanScrollLeft] = useState(false);
|
||||||
|
const [canScrollRight, setCanScrollRight] = useState(false);
|
||||||
|
|
||||||
|
const updateScrollState = useCallback(() => {
|
||||||
|
const menu = menuRef.current;
|
||||||
|
if (!menu) return;
|
||||||
|
setCanScrollLeft(menu.scrollLeft > 2);
|
||||||
|
setCanScrollRight(menu.scrollLeft + menu.clientWidth < menu.scrollWidth - 2);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const menu = menuRef.current;
|
||||||
|
if (!menu || !menuNeedsScroll) {
|
||||||
|
setCanScrollLeft(false);
|
||||||
|
setCanScrollRight(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
updateScrollState();
|
||||||
|
menu.addEventListener('scroll', updateScrollState);
|
||||||
|
return () => menu.removeEventListener('scroll', updateScrollState);
|
||||||
|
}, [menuNeedsScroll, updateScrollState]);
|
||||||
|
|
||||||
|
useLayoutEffect(() => {
|
||||||
|
const menu = menuRef.current;
|
||||||
|
if (!menu) return;
|
||||||
|
const children = Array.from(menu.querySelectorAll('.sight-frame-menu-point'));
|
||||||
|
if (children.length < 2) {
|
||||||
|
setMenuNeedsScroll(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const style = getComputedStyle(menu);
|
||||||
|
const availableWidth = menu.clientWidth - parseFloat(style.paddingLeft) - parseFloat(style.paddingRight);
|
||||||
|
const totalChildrenWidth = children.reduce((sum, el) => sum + el.offsetWidth, 0);
|
||||||
|
const evenGap = (availableWidth - totalChildrenWidth) / (children.length + 1);
|
||||||
|
setMenuNeedsScroll(evenGap < 10);
|
||||||
|
}, [articleSections, selectedSection]);
|
||||||
|
|
||||||
|
// Автозакрытие fullscreen 3D при бездействии (60 сек)
|
||||||
|
useEffect(() => {
|
||||||
|
if (!isFullscreen3D) {
|
||||||
|
if (idleTimerRef.current) {
|
||||||
|
clearInterval(idleTimerRef.current);
|
||||||
|
idleTimerRef.current = null;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let idleSeconds = 0;
|
||||||
|
|
||||||
|
const checkIdle = () => {
|
||||||
|
idleSeconds += 1;
|
||||||
|
if (idleSeconds >= 60) {
|
||||||
|
setIsFullscreen3D(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
idleTimerRef.current = setInterval(checkIdle, 1000);
|
||||||
|
|
||||||
|
const resetIdle = () => {
|
||||||
|
idleSeconds = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
const events = [
|
||||||
|
"mousedown",
|
||||||
|
"mousemove",
|
||||||
|
"keypress",
|
||||||
|
"scroll",
|
||||||
|
"touchstart",
|
||||||
|
"click",
|
||||||
|
];
|
||||||
|
events.forEach((event) => {
|
||||||
|
window.addEventListener(event, resetIdle, { passive: true });
|
||||||
|
});
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
if (idleTimerRef.current) {
|
||||||
|
clearInterval(idleTimerRef.current);
|
||||||
|
idleTimerRef.current = null;
|
||||||
|
}
|
||||||
|
events.forEach((event) => {
|
||||||
|
window.removeEventListener(event, resetIdle);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}, [isFullscreen3D]);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
routeSights,
|
routeSights,
|
||||||
@@ -162,7 +253,10 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
const introSection = {
|
const introSection = {
|
||||||
id: media?.id || "intro-title",
|
id: media?.id || "intro-title",
|
||||||
heading:
|
heading:
|
||||||
sight?.short_name || sight?.name || sight_name || "Название достопримечательности",
|
sight?.short_name ||
|
||||||
|
sight?.name ||
|
||||||
|
sight_name ||
|
||||||
|
"Название достопримечательности",
|
||||||
body: "",
|
body: "",
|
||||||
};
|
};
|
||||||
const allSections = [introSection, ...rightArticles];
|
const allSections = [introSection, ...rightArticles];
|
||||||
@@ -240,9 +334,7 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
alt=""
|
alt=""
|
||||||
className={className}
|
className={className}
|
||||||
onError={(e) => {
|
onError={(e) => {
|
||||||
console.warn(
|
console.warn(`Failed to load image: ${currentMediaData.path}`);
|
||||||
`Failed to load image: ${currentMediaData.path}`,
|
|
||||||
);
|
|
||||||
e.target.style.display = "none";
|
e.target.style.display = "none";
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
@@ -257,9 +349,7 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
playsInline
|
playsInline
|
||||||
className={className}
|
className={className}
|
||||||
onError={(e) => {
|
onError={(e) => {
|
||||||
console.warn(
|
console.warn(`Failed to load video: ${currentMediaData.path}`);
|
||||||
`Failed to load video: ${currentMediaData.path}`,
|
|
||||||
);
|
|
||||||
e.target.style.display = "none";
|
e.target.style.display = "none";
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@@ -309,27 +399,20 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="three-d-control-btn"
|
className="three-d-control-btn"
|
||||||
title="Уменьшить"
|
onPointerUp={() => threeViewControlRef.current?.zoomOut?.()}
|
||||||
onPointerUp={() =>
|
|
||||||
threeViewControlRef.current?.zoomOut?.()
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<MinusIcon />
|
<MinusIcon />
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="three-d-control-btn"
|
className="three-d-control-btn"
|
||||||
title="Увеличить"
|
onPointerUp={() => threeViewControlRef.current?.zoomIn?.()}
|
||||||
onPointerUp={() =>
|
|
||||||
threeViewControlRef.current?.zoomIn?.()
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<PlusIcon />
|
<PlusIcon />
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="three-d-control-btn"
|
className="three-d-control-btn"
|
||||||
title={isFullscreen3D ? "Свернуть" : "Развернуть"}
|
|
||||||
onPointerUp={() => {
|
onPointerUp={() => {
|
||||||
if (isFullscreen3D) {
|
if (isFullscreen3D) {
|
||||||
setIsFullscreen3D(false);
|
setIsFullscreen3D(false);
|
||||||
@@ -344,11 +427,6 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
</div>
|
</div>
|
||||||
<button
|
<button
|
||||||
className={"fullscreen-3d-button"}
|
className={"fullscreen-3d-button"}
|
||||||
title={
|
|
||||||
isFullscreen3D
|
|
||||||
? "Закрыть полноэкранный режим"
|
|
||||||
: "Открыть в полноэкранном режиме"
|
|
||||||
}
|
|
||||||
aria-hidden="true"
|
aria-hidden="true"
|
||||||
>
|
>
|
||||||
<svg
|
<svg
|
||||||
@@ -364,6 +442,100 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
/>
|
/>
|
||||||
</svg>
|
</svg>
|
||||||
</button>
|
</button>
|
||||||
|
{isFullscreen3D && (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: 94,
|
||||||
|
right: 10,
|
||||||
|
zIndex: 10,
|
||||||
|
pointerEvents: "none",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="cluster-sights-list"
|
||||||
|
style={{
|
||||||
|
background: `linear-gradient(to bottom right, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0) 100%), rgba(var(--carrier-right-rgb, 0, 111, 58), 0.4)`,
|
||||||
|
backdropFilter: "blur(10px)",
|
||||||
|
borderRadius: "8px",
|
||||||
|
width: 200,
|
||||||
|
boxShadow:
|
||||||
|
"0 0 0 1px rgba(255, 255, 255, 0.3) inset, 4px 4px 12px 0 rgba(255, 255, 255, 0.12) inset",
|
||||||
|
display: "flex",
|
||||||
|
flexDirection: "column",
|
||||||
|
padding: "8px 13px",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{[
|
||||||
|
{
|
||||||
|
label: "Вращать",
|
||||||
|
icon: (
|
||||||
|
<img
|
||||||
|
src={rotate3DIcon}
|
||||||
|
alt=""
|
||||||
|
width="14"
|
||||||
|
height="14"
|
||||||
|
/>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Приблизить / Отдалить",
|
||||||
|
icon: (
|
||||||
|
<img src={zoom3DIcon} alt="" width="14" height="14" />
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Переместить",
|
||||||
|
icon: (
|
||||||
|
<img src={pan3DIcon} alt="" width="14" height="14" />
|
||||||
|
),
|
||||||
|
},
|
||||||
|
].map((item, index, arr) => (
|
||||||
|
<div
|
||||||
|
key={index}
|
||||||
|
style={{
|
||||||
|
display: "flex",
|
||||||
|
alignItems: "center",
|
||||||
|
height: "30px",
|
||||||
|
userSelect: "none",
|
||||||
|
touchAction: "none",
|
||||||
|
padding: "0 4px",
|
||||||
|
borderBottom:
|
||||||
|
index < arr.length - 1
|
||||||
|
? "1px solid rgba(255, 255, 255, 0.1)"
|
||||||
|
: "none",
|
||||||
|
transition: "background-color 0.2s",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
style={{
|
||||||
|
display: "block",
|
||||||
|
marginRight: "8px",
|
||||||
|
flexShrink: 0,
|
||||||
|
lineHeight: 0,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{item.icon}
|
||||||
|
</span>
|
||||||
|
<span
|
||||||
|
style={{
|
||||||
|
color: "white",
|
||||||
|
fontSize: "12px",
|
||||||
|
lineHeight: "1.5",
|
||||||
|
fontWeight: "400",
|
||||||
|
overflow: "hidden",
|
||||||
|
textOverflow: "ellipsis",
|
||||||
|
whiteSpace: "nowrap",
|
||||||
|
flex: 1,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{item.label}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
default:
|
default:
|
||||||
@@ -428,6 +600,16 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
const processedSightName = useMemo(() => {
|
const processedSightName = useMemo(() => {
|
||||||
if (!sight_name) return sight_name;
|
if (!sight_name) return sight_name;
|
||||||
|
|
||||||
|
// Handle \n line breaks (только в правом виджете)
|
||||||
|
if (sight_name.includes("\n")) {
|
||||||
|
return sight_name.split("\n").map((line, i) => (
|
||||||
|
<React.Fragment key={i}>
|
||||||
|
{i > 0 && <br />}
|
||||||
|
{line}
|
||||||
|
</React.Fragment>
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
const namePattern =
|
const namePattern =
|
||||||
/([А-Яа-яA-Za-z0-9]\. [А-Яа-яA-Za-z0-9]\. [А-Яа-яA-Za-z0-9]+)/g;
|
/([А-Яа-яA-Za-z0-9]\. [А-Яа-яA-Za-z0-9]\. [А-Яа-яA-Za-z0-9]+)/g;
|
||||||
|
|
||||||
@@ -454,7 +636,7 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
|
|
||||||
const titleLineHeight = useMemo(() => {
|
const titleLineHeight = useMemo(() => {
|
||||||
if (!sight_name) return "120%";
|
if (!sight_name) return "120%";
|
||||||
const textLength = sight_name.length;
|
const textLength = sight_name.replace(/\n/g, "").length;
|
||||||
const calculatedLineHeight = Math.max(
|
const calculatedLineHeight = Math.max(
|
||||||
100,
|
100,
|
||||||
Math.min(120, 120 - (textLength / 10) * 1),
|
Math.min(120, 120 - (textLength / 10) * 1),
|
||||||
@@ -510,7 +692,9 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
overflowWrap: "break-word",
|
overflowWrap: "break-word",
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{selectedSection === 0 ? processedSightName : sight_name}
|
{selectedSection === 0
|
||||||
|
? processedSightName
|
||||||
|
: sightData?.short_name || sight_name}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
@@ -528,60 +712,43 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div className="sight-frame-menu">
|
<div className="sight-frame-menu-wrapper">
|
||||||
{selectedSection !== 0 && (
|
<div className="sight-frame-menu-fade left" style={{ opacity: canScrollLeft ? 1 : 0 }} />
|
||||||
<div
|
<div className="sight-frame-menu-fade right" style={{ opacity: canScrollRight ? 1 : 0 }} />
|
||||||
style={{
|
<div
|
||||||
position: "absolute",
|
className="sight-frame-menu"
|
||||||
left: "10px",
|
ref={menuRef}
|
||||||
marginTop: "-4.5px",
|
style={menuNeedsScroll ? { justifyContent: 'space-between' } : undefined}
|
||||||
zIndex: 1,
|
>
|
||||||
paddingLeft: "15px",
|
<div
|
||||||
paddingRight: "7.5px",
|
style={{
|
||||||
paddingTop: "4.5px",
|
position: "absolute",
|
||||||
paddingBottom: "4.5px",
|
left: "10px",
|
||||||
cursor: "pointer",
|
marginTop: "-4.5px",
|
||||||
}}
|
zIndex: 1,
|
||||||
onPointerUp={() => setSelectedSection(0)}
|
paddingLeft: "15px",
|
||||||
>
|
paddingRight: "7.5px",
|
||||||
<svg
|
paddingTop: "4.5px",
|
||||||
width="20"
|
paddingBottom: "4.5px",
|
||||||
height="25"
|
cursor: "pointer",
|
||||||
viewBox="0 0 20 25"
|
opacity: selectedSection !== 0 ? 1 : 0,
|
||||||
fill="none"
|
transform: selectedSection !== 0 ? "scale(1)" : "scale(0.5)",
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
transition: "opacity 0.3s ease, transform 0.3s ease",
|
||||||
style={{ display: "block" }}
|
pointerEvents: selectedSection !== 0 ? "auto" : "none",
|
||||||
>
|
}}
|
||||||
<defs>
|
onPointerUp={() => {
|
||||||
<linearGradient
|
setSelectedSection(0);
|
||||||
id="sightFrameGradient3"
|
setIsFullscreen3D(false);
|
||||||
x1="0%"
|
}}
|
||||||
y1="0%"
|
>
|
||||||
x2="100%"
|
<img
|
||||||
y2="100%"
|
src={subtractHomeIcon}
|
||||||
gradientUnits="userSpaceOnUse"
|
alt=""
|
||||||
>
|
width="24"
|
||||||
<stop offset="0%" stopColor="rgba(255, 255, 255, 0.2)" />
|
height="21"
|
||||||
<stop offset="100%" stopColor="rgba(255, 255, 255, 0)" />
|
style={{ display: "block" }}
|
||||||
</linearGradient>
|
/>
|
||||||
<clipPath id="clip0_662_97446">
|
</div>
|
||||||
<rect
|
|
||||||
width="20"
|
|
||||||
height="25"
|
|
||||||
fill="white"
|
|
||||||
transform="translate(12.5 0.5) rotate(90)"
|
|
||||||
/>
|
|
||||||
</clipPath>
|
|
||||||
</defs>
|
|
||||||
<g clipPath="url(#clip0_662_97446)">
|
|
||||||
<path
|
|
||||||
d="M4.03158 11.0738C4.21879 11.2087 4.34702 11.2766 4.44531 11.3747C6.93048 13.8687 9.41098 16.3665 11.8962 18.8605C12.3408 19.3067 12.5186 19.8207 12.3361 20.4339C12.0281 21.4658 10.7776 21.8393 9.95295 21.1498C9.86309 21.0743 9.78165 20.9894 9.69928 20.9064C6.85279 18.0446 4.00537 15.1817 1.15982 12.3189C0.280876 11.4341 0.281813 10.6456 1.16169 9.75982C4.04 6.86305 6.91457 3.96155 9.80786 1.07986C10.0597 0.828952 10.4144 0.619547 10.7561 0.537482C11.4019 0.382786 12.015 0.72142 12.3193 1.28644C12.6263 1.85334 12.5392 2.56079 12.0806 3.05129C11.7286 3.4286 11.3561 3.78704 10.991 4.15209C8.79601 6.35557 6.60194 8.55904 4.40599 10.7606C4.32362 10.8436 4.22721 10.9106 4.03158 11.0729L4.03158 11.0738Z"
|
|
||||||
fill="url(#sightFrameGradient3)"
|
|
||||||
/>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
{contentError ? (
|
{contentError ? (
|
||||||
<p className="error-message">{contentError}</p>
|
<p className="error-message">{contentError}</p>
|
||||||
) : (
|
) : (
|
||||||
@@ -589,7 +756,10 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
articleSections.length > 1 &&
|
articleSections.length > 1 &&
|
||||||
articleSections.slice(1).map((section, index) => (
|
articleSections.slice(1).map((section, index) => (
|
||||||
<div
|
<div
|
||||||
onPointerUp={() => setSelectedSection(index + 1)}
|
onPointerUp={() => {
|
||||||
|
setSelectedSection(index + 1);
|
||||||
|
setIsFullscreen3D(false);
|
||||||
|
}}
|
||||||
key={section.id || section.heading || index}
|
key={section.id || section.heading || index}
|
||||||
className={`sight-frame-menu-point ${
|
className={`sight-frame-menu-point ${
|
||||||
index + 1 === selectedSection ? "active" : ""
|
index + 1 === selectedSection ? "active" : ""
|
||||||
@@ -602,6 +772,7 @@ const SightFrame = observer(({ media, sight_id, sight_name }) => {
|
|||||||
))
|
))
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -95,21 +95,36 @@ const TransferWidget = observer(function TransferWidget({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const getTransferLabel = () => {
|
const getTransferLabel = () => {
|
||||||
if (selectedLanguageRight === "ru") {
|
if (!stationName) {
|
||||||
return stationName
|
if (selectedLanguageRight === "en") return "Nearest station not found";
|
||||||
? `Пересадки остановки ${stationName}:`
|
if (selectedLanguageRight === "zh") return "最近的站点未找到";
|
||||||
: "Ближайшая остановка не обнаружена";
|
return "Ближайшая остановка не обнаружена";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (selectedLanguageRight === "en") {
|
if (selectedLanguageRight === "en") {
|
||||||
return stationName
|
return (
|
||||||
? `Available transfers at station ${stationName}`
|
<>
|
||||||
: "Nearest station not found";
|
Transfer at stop<br />
|
||||||
|
«{stationName}»:
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return stationName
|
if (selectedLanguageRight === "zh") {
|
||||||
? `在车站可用的换乘:${stationName}`
|
return (
|
||||||
: "最近的站点未找到";
|
<>
|
||||||
|
换乘站<br />
|
||||||
|
«{stationName}»:
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
Пересадка на остановке<br />
|
||||||
|
«{stationName}»:
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const getNoTransfersMessage = () => {
|
const getNoTransfersMessage = () => {
|
||||||
|
|||||||
13
src/client/src/components/OverlayScrollbarsWrapper.d.ts
vendored
Normal file
13
src/client/src/components/OverlayScrollbarsWrapper.d.ts
vendored
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
import React from "react";
|
||||||
|
|
||||||
|
export declare const OverlayScrollbarsWrapper: React.ForwardRefExoticComponent<
|
||||||
|
React.PropsWithChildren<{
|
||||||
|
className?: string;
|
||||||
|
onScroll?: (event: Event) => void;
|
||||||
|
overflowX?: string;
|
||||||
|
overflowY?: string;
|
||||||
|
scrollbarVisibility?: string;
|
||||||
|
[key: string]: any;
|
||||||
|
}> &
|
||||||
|
React.RefAttributes<HTMLElement>
|
||||||
|
>;
|
||||||
@@ -71,7 +71,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.react-markdown-container blockquote {
|
.react-markdown-container blockquote {
|
||||||
border-left: 4px solid #006F3A;
|
border-left: 4px solid var(--carrier-main, #006F3A);
|
||||||
padding-left: 16px;
|
padding-left: 16px;
|
||||||
margin-top: 16px;
|
margin-top: 16px;
|
||||||
margin-bottom: 16px;
|
margin-bottom: 16px;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import React from "react";
|
import "react";
|
||||||
import ReactMarkdown from "react-markdown";
|
import ReactMarkdown from "react-markdown";
|
||||||
import rehypeRaw from "rehype-raw";
|
import rehypeRaw from "rehype-raw";
|
||||||
import "./ReactMarkdown.css";
|
import "./ReactMarkdown.css";
|
||||||
|
|||||||
@@ -6,22 +6,30 @@ export const SimulationSettings = observer(() => {
|
|||||||
const [open, setOpen] = useState(false);
|
const [open, setOpen] = useState(false);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div style={{ position: "absolute", top: 12, right: 12, zIndex: 10010 }}>
|
<div style={{ position: "fixed", top: 12, right: 12, zIndex: 2147483646 }}>
|
||||||
<button
|
<button
|
||||||
onClick={() => setOpen(!open)}
|
onClick={() => setOpen(!open)}
|
||||||
style={{
|
style={{
|
||||||
width: 36, height: 36, borderRadius: 6,
|
width: 36, height: 36, borderRadius: 6,
|
||||||
border: "1px solid rgba(255,255,255,0.25)",
|
border: "1px solid rgba(255,255,255,0.25)",
|
||||||
background: open ? "rgba(255,255,255,0.15)" : "rgba(0,0,0,0.5)",
|
background: open ? "rgba(255,255,255,0.15)" : "rgba(0,0,0,0.5)",
|
||||||
color: "white", cursor: "pointer", fontSize: 18,
|
color: "white", cursor: "pointer",
|
||||||
|
display: "flex", alignItems: "center", justifyContent: "center",
|
||||||
|
padding: 0,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
⚙
|
<svg width="22" height="22" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path
|
||||||
|
d="M12 15.5A3.5 3.5 0 0 1 8.5 12 3.5 3.5 0 0 1 12 8.5a3.5 3.5 0 0 1 3.5 3.5 3.5 3.5 0 0 1-3.5 3.5m7.43-2.92c.04-.34.07-.69.07-1.08s-.03-.73-.07-1.08l2.33-1.82c.21-.16.27-.46.13-.7l-2.21-3.83a.55.55 0 0 0-.68-.22l-2.75 1.1a8.1 8.1 0 0 0-1.86-1.08l-.42-2.93A.545.545 0 0 0 14 2h-4c-.27 0-.5.2-.54.46l-.42 2.93c-.68.28-1.3.65-1.86 1.08L4.43 5.37a.543.543 0 0 0-.68.22L1.54 9.42c-.14.24-.08.54.13.7l2.33 1.82c-.04.35-.07.7-.07 1.08s.03.73.07 1.08L1.67 15.92c-.21.16-.27.46-.13.7l2.21 3.83c.14.24.43.31.68.22l2.75-1.1c.56.43 1.18.8 1.86 1.08l.42 2.93c.04.26.27.46.54.46h4c.27 0 .5-.2.54-.46l.42-2.93c.68-.28 1.3-.65 1.86-1.08l2.75 1.1c.25.09.54.02.68-.22l2.21-3.83c.14-.24.08-.54-.13-.7l-2.33-1.82Z"
|
||||||
|
fill="white"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
{open && (
|
{open && (
|
||||||
<div style={{
|
<div style={{
|
||||||
marginTop: 6, background: "rgba(20,20,20,0.9)",
|
position: "absolute", top: 42, right: 0,
|
||||||
|
background: "rgba(20,20,20,0.9)",
|
||||||
border: "1px solid rgba(255,255,255,0.15)", borderRadius: 8,
|
border: "1px solid rgba(255,255,255,0.15)", borderRadius: 8,
|
||||||
padding: "10px 12px", minWidth: 200, fontSize: 13,
|
padding: "10px 12px", minWidth: 200, fontSize: 13,
|
||||||
}}>
|
}}>
|
||||||
@@ -72,6 +80,15 @@ export const SimulationSettings = observer(() => {
|
|||||||
onClick={apiStore.toggleSimulationInstantMove}
|
onClick={apiStore.toggleSimulationInstantMove}
|
||||||
/>
|
/>
|
||||||
</Row>
|
</Row>
|
||||||
|
|
||||||
|
{/* Хитбоксы */}
|
||||||
|
<Row>
|
||||||
|
<span>Хитбоксы</span>
|
||||||
|
<Toggle
|
||||||
|
on={apiStore.showHitboxes}
|
||||||
|
onClick={apiStore.toggleShowHitboxes}
|
||||||
|
/>
|
||||||
|
</Row>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -17,7 +17,7 @@
|
|||||||
114deg,
|
114deg,
|
||||||
rgba(255, 255, 255, 0.1) 8.71%,
|
rgba(255, 255, 255, 0.1) 8.71%,
|
||||||
rgba(255, 255, 255, 0.05) 69.69%
|
rgba(255, 255, 255, 0.05) 69.69%
|
||||||
), #006F3A;
|
), var(--carrier-main, #006F3A);
|
||||||
border-radius: 12px;
|
border-radius: 12px;
|
||||||
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
|
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
|
||||||
color: white;
|
color: white;
|
||||||
|
|||||||
@@ -19,7 +19,11 @@ function useThumbSync(scrollableRef: React.RefObject<HTMLDivElement | null>) {
|
|||||||
height: 60,
|
height: 60,
|
||||||
top: 0,
|
top: 0,
|
||||||
hasScroll: false,
|
hasScroll: false,
|
||||||
|
isAtTop: true,
|
||||||
|
isAtBottom: false,
|
||||||
});
|
});
|
||||||
|
const [visible, setVisible] = useState(false);
|
||||||
|
const hideTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
||||||
const rafRef = useRef<number | null>(null);
|
const rafRef = useRef<number | null>(null);
|
||||||
|
|
||||||
const update = useCallback(() => {
|
const update = useCallback(() => {
|
||||||
@@ -31,8 +35,11 @@ function useThumbSync(scrollableRef: React.RefObject<HTMLDivElement | null>) {
|
|||||||
const st = el.scrollTop;
|
const st = el.scrollTop;
|
||||||
const th = ch;
|
const th = ch;
|
||||||
|
|
||||||
|
const isAtTop = st <= 0;
|
||||||
|
const isAtBottom = st + ch >= sh - 1;
|
||||||
|
|
||||||
if (sh <= ch) {
|
if (sh <= ch) {
|
||||||
setState({ height: th, top: 0, hasScroll: false });
|
setState((prev) => ({ ...prev, hasScroll: false, isAtTop: true, isAtBottom: true }));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -41,7 +48,7 @@ function useThumbSync(scrollableRef: React.RefObject<HTMLDivElement | null>) {
|
|||||||
const scrollRange = sh - ch;
|
const scrollRange = sh - ch;
|
||||||
const top = range <= 0 ? 0 : (st / scrollRange) * range;
|
const top = range <= 0 ? 0 : (st / scrollRange) * range;
|
||||||
|
|
||||||
setState({ height: thumbHeight, top, hasScroll: true });
|
setState({ height: thumbHeight, top, hasScroll: true, isAtTop, isAtBottom });
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -68,7 +75,24 @@ function useThumbSync(scrollableRef: React.RefObject<HTMLDivElement | null>) {
|
|||||||
};
|
};
|
||||||
}, [update]);
|
}, [update]);
|
||||||
|
|
||||||
return state;
|
useEffect(() => {
|
||||||
|
if (state.hasScroll) {
|
||||||
|
if (hideTimerRef.current) {
|
||||||
|
clearTimeout(hideTimerRef.current);
|
||||||
|
hideTimerRef.current = null;
|
||||||
|
}
|
||||||
|
setVisible(true);
|
||||||
|
} else {
|
||||||
|
hideTimerRef.current = setTimeout(() => {
|
||||||
|
setVisible(false);
|
||||||
|
}, 200);
|
||||||
|
}
|
||||||
|
return () => {
|
||||||
|
if (hideTimerRef.current) clearTimeout(hideTimerRef.current);
|
||||||
|
};
|
||||||
|
}, [state.hasScroll]);
|
||||||
|
|
||||||
|
return { ...state, visible };
|
||||||
}
|
}
|
||||||
|
|
||||||
export const TouchableLayout = forwardRef<HTMLDivElement, TouchableLayoutProps>(
|
export const TouchableLayout = forwardRef<HTMLDivElement, TouchableLayoutProps>(
|
||||||
@@ -234,9 +258,12 @@ export const TouchableLayout = forwardRef<HTMLDivElement, TouchableLayoutProps>(
|
|||||||
};
|
};
|
||||||
}, [thumb.hasScroll]);
|
}, [thumb.hasScroll]);
|
||||||
|
|
||||||
const containerClassName = className
|
const containerClassName = [
|
||||||
? `scrollable-container ${className}`
|
"scrollable-container",
|
||||||
: "scrollable-container";
|
className,
|
||||||
|
thumb.isAtTop ? "is-at-top" : "",
|
||||||
|
thumb.isAtBottom ? "is-at-bottom" : "",
|
||||||
|
].filter(Boolean).join(" ");
|
||||||
|
|
||||||
const viewportStyle: React.CSSProperties = maxHeight
|
const viewportStyle: React.CSSProperties = maxHeight
|
||||||
? {
|
? {
|
||||||
@@ -251,15 +278,19 @@ export const TouchableLayout = forwardRef<HTMLDivElement, TouchableLayoutProps>(
|
|||||||
<div ref={scrollableRef} className="scrollable">
|
<div ref={scrollableRef} className="scrollable">
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
{thumb.hasScroll && (
|
<div
|
||||||
<div ref={trackRef} className="custom-scrollbar-track">
|
ref={trackRef}
|
||||||
|
className="custom-scrollbar-track"
|
||||||
|
style={{ opacity: thumb.hasScroll ? 1 : 0 }}
|
||||||
|
>
|
||||||
|
{thumb.visible && (
|
||||||
<div
|
<div
|
||||||
ref={thumbRef}
|
ref={thumbRef}
|
||||||
className="custom-scrollbar-thumb"
|
className="custom-scrollbar-thumb"
|
||||||
style={{ height: thumb.height, top: thumb.top }}
|
style={{ height: thumb.height, top: thumb.top }}
|
||||||
/>
|
/>
|
||||||
</div>
|
)}
|
||||||
)}
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,12 +1,8 @@
|
|||||||
import { FederatedPointerEvent, FederatedWheelEvent } from "pixi.js";
|
import { FederatedPointerEvent, FederatedWheelEvent } from "pixi.js";
|
||||||
import { ReactNode, useEffect, useState, useRef, useCallback } from "react";
|
import { ReactNode, useEffect, useState, useRef } from "react";
|
||||||
import { useTransform } from "./transformContext";
|
import { useTransform } from "./transformContext";
|
||||||
import { BACKGROUND_COLOR, SCALE_FACTOR } from "../../assets/Constants";
|
import { BACKGROUND_COLOR, SCALE_FACTOR } from "../../assets/Constants";
|
||||||
import { useApplication } from "@pixi/react";
|
import { useApplication } from "@pixi/react";
|
||||||
import { useGeolocation } from "../../context/GeolocationContext";
|
|
||||||
import ContentAPI from "../../api/content/content.api";
|
|
||||||
import React from "react";
|
|
||||||
import { useGeolocationStore } from "../../stores/hooks/useGeolocationStore";
|
|
||||||
import { useCameraAnimationStore } from "../../stores";
|
import { useCameraAnimationStore } from "../../stores";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import debounce from "lodash/debounce";
|
import debounce from "lodash/debounce";
|
||||||
@@ -25,7 +21,6 @@ export const InfiniteCanvas = observer(
|
|||||||
setIsAutoMode,
|
setIsAutoMode,
|
||||||
userActivityTimestamp,
|
userActivityTimestamp,
|
||||||
updateUserActivity,
|
updateUserActivity,
|
||||||
autoModeStartTimestamp,
|
|
||||||
setAutoModeStartTimestamp,
|
setAutoModeStartTimestamp,
|
||||||
} = useTransform();
|
} = useTransform();
|
||||||
const [loaded, setLoaded] = useState(false);
|
const [loaded, setLoaded] = useState(false);
|
||||||
@@ -49,17 +44,14 @@ export const InfiniteCanvas = observer(
|
|||||||
position: { x: number; y: number };
|
position: { x: number; y: number };
|
||||||
} | null>(null);
|
} | null>(null);
|
||||||
// Keep these for backward compatibility, but we'll use pinchStartData for calculations
|
// Keep these for backward compatibility, but we'll use pinchStartData for calculations
|
||||||
const [initialPinchDistance, setInitialPinchDistance] = useState<
|
const [, setInitialPinchDistance] = useState<number | null>(null);
|
||||||
number | null
|
const [, setInitialPinchMidpoint] = useState<{
|
||||||
>(null);
|
|
||||||
const [initialPinchMidpoint, setInitialPinchMidpoint] = useState<{
|
|
||||||
x: number;
|
x: number;
|
||||||
y: number;
|
y: number;
|
||||||
} | null>(null);
|
} | null>(null);
|
||||||
|
|
||||||
const [scaleMin, setScaleMin] = useState(0.1); // Default min scale
|
const [scaleMin, setScaleMin] = useState(0.1); // Default min scale
|
||||||
const [scaleMax, setScaleMax] = useState(3); // Default max scale
|
const [scaleMax, setScaleMax] = useState(3); // Default max scale
|
||||||
const store = useGeolocationStore();
|
|
||||||
const cameraAnimationStore = useCameraAnimationStore();
|
const cameraAnimationStore = useCameraAnimationStore();
|
||||||
|
|
||||||
// Add debounced version of syncState to reduce jittering
|
// Add debounced version of syncState to reduce jittering
|
||||||
@@ -269,13 +261,14 @@ export const InfiniteCanvas = observer(
|
|||||||
setInitialPinchMidpoint(null);
|
setInitialPinchMidpoint(null);
|
||||||
pinchStartData.current = null;
|
pinchStartData.current = null;
|
||||||
|
|
||||||
if (isDragging) {
|
if (isDragging) {
|
||||||
const newPosition = {
|
const newPosition = {
|
||||||
x: startPosition.x - startMousePosition.x + e.globalX,
|
x: startPosition.x - startMousePosition.x + e.globalX,
|
||||||
y: startPosition.y - startMousePosition.y + e.globalY,
|
y: startPosition.y - startMousePosition.y + e.globalY,
|
||||||
};
|
};
|
||||||
setPosition(newPosition);
|
setPosition(newPosition);
|
||||||
syncStateDebounced(newPosition, scale);
|
syncStateDebounced(newPosition, scale);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,31 +1,20 @@
|
|||||||
import React, {
|
import {
|
||||||
useCallback,
|
useCallback,
|
||||||
useEffect,
|
useEffect,
|
||||||
useRef,
|
|
||||||
useState,
|
useState,
|
||||||
useMemo,
|
useMemo,
|
||||||
} from "react";
|
} from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { Application, extend } from "@pixi/react";
|
import { extend } from "@pixi/react";
|
||||||
import { Container, Graphics, Sprite, Text } from "pixi.js";
|
import { Container, Graphics, Sprite, Text } from "pixi.js";
|
||||||
import { MapDataProvider, useMapData } from "./MapDataContext";
|
import { MapDataProvider, useMapData } from "./MapDataContext";
|
||||||
import { TransformProvider, useTransform } from "./transformContext";
|
import { TransformProvider, useTransform } from "./transformContext";
|
||||||
import { InfiniteCanvas } from "./InfiniteCanvas";
|
// @ts-ignore
|
||||||
import { TravelPath } from "./TravelPath";
|
|
||||||
import { Station } from "./Station";
|
|
||||||
import { SightsLayer } from "./Sight";
|
|
||||||
import Loader from "../Loader";
|
import Loader from "../Loader";
|
||||||
import {
|
import { UP_SCALE } from "./Constants";
|
||||||
BACKGROUND_COLOR,
|
|
||||||
BUS_COLOR,
|
|
||||||
STATION_OUTLINE_WIDTH,
|
|
||||||
STATION_RADIUS,
|
|
||||||
UP_SCALE,
|
|
||||||
} from "./Constants";
|
|
||||||
import "../../styles/MapLayer.css";
|
import "../../styles/MapLayer.css";
|
||||||
import { useGeolocationStore, useCameraAnimationStore } from "../../stores";
|
import { useCameraAnimationStore } from "../../stores";
|
||||||
import { coordinatesToLocal } from "./utils";
|
import { coordinatesToLocal } from "./utils";
|
||||||
import { TramIcon } from "./TramIcon";
|
|
||||||
import { SCALE_FACTOR } from "../../assets/Constants";
|
import { SCALE_FACTOR } from "../../assets/Constants";
|
||||||
import { apiStore } from "../../api/ApiStore/store";
|
import { apiStore } from "../../api/ApiStore/store";
|
||||||
import WebGLMap from "./WebGLMap";
|
import WebGLMap from "./WebGLMap";
|
||||||
@@ -42,7 +31,7 @@ export function Map() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function rotatePoint(x, y, originX, originY, angle) {
|
function rotatePoint(x: number, y: number, originX: number, originY: number, angle: number) {
|
||||||
const cos = Math.cos(angle);
|
const cos = Math.cos(angle);
|
||||||
const sin = Math.sin(angle);
|
const sin = Math.sin(angle);
|
||||||
const dx = x - originX;
|
const dx = x - originX;
|
||||||
@@ -53,8 +42,6 @@ function rotatePoint(x, y, originX, originY, angle) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const RouteMap = observer(() => {
|
const RouteMap = observer(() => {
|
||||||
const store = useGeolocationStore();
|
|
||||||
const { contextData } = store;
|
|
||||||
const {
|
const {
|
||||||
routeData,
|
routeData,
|
||||||
stationData,
|
stationData,
|
||||||
@@ -77,7 +64,6 @@ const RouteMap = observer(() => {
|
|||||||
scale,
|
scale,
|
||||||
} = useTransform();
|
} = useTransform();
|
||||||
const cameraAnimationStore = useCameraAnimationStore();
|
const cameraAnimationStore = useCameraAnimationStore();
|
||||||
const parentRef = useRef(null);
|
|
||||||
|
|
||||||
const [rotationAngle, setRotationAngle] = useState(0);
|
const [rotationAngle, setRotationAngle] = useState(0);
|
||||||
|
|
||||||
@@ -143,7 +129,7 @@ const RouteMap = observer(() => {
|
|||||||
const rotationOriginY = 0;
|
const rotationOriginY = 0;
|
||||||
|
|
||||||
const transformGeoToMapLocal = useCallback(
|
const transformGeoToMapLocal = useCallback(
|
||||||
(latitude, longitude) => {
|
(latitude: number, longitude: number) => {
|
||||||
if (centerLat === undefined || centerLon === undefined) {
|
if (centerLat === undefined || centerLon === undefined) {
|
||||||
return { x: 0, y: 0 };
|
return { x: 0, y: 0 };
|
||||||
}
|
}
|
||||||
@@ -239,99 +225,6 @@ const RouteMap = observer(() => {
|
|||||||
transformedStations,
|
transformedStations,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const drawActualBusPos = useCallback(
|
|
||||||
(g: Graphics) => {
|
|
||||||
g.clear();
|
|
||||||
if (transformedCurrentCoordinates) {
|
|
||||||
g.circle(
|
|
||||||
transformedCurrentCoordinates.x,
|
|
||||||
transformedCurrentCoordinates.y,
|
|
||||||
STATION_RADIUS / scale < 10
|
|
||||||
? 10
|
|
||||||
: STATION_RADIUS / scale > 20
|
|
||||||
? 20
|
|
||||||
: STATION_RADIUS / scale
|
|
||||||
);
|
|
||||||
g.fill({ color: BUS_COLOR });
|
|
||||||
g.stroke({ color: BACKGROUND_COLOR, width: STATION_OUTLINE_WIDTH });
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[transformedCurrentCoordinates, scale]
|
|
||||||
);
|
|
||||||
|
|
||||||
const scaledPoints = useMemo(() => {
|
|
||||||
if (!routeData?.path) return [];
|
|
||||||
|
|
||||||
return routeData.path.map(([latitude, longitude]) => {
|
|
||||||
const { x, y } = transformGeoToMapLocal(latitude, longitude);
|
|
||||||
return rotatePoint(x, y, rotationOriginX, rotationOriginY, rotationAngle);
|
|
||||||
});
|
|
||||||
}, [
|
|
||||||
routeData?.path,
|
|
||||||
transformGeoToMapLocal,
|
|
||||||
rotationOriginX,
|
|
||||||
rotationOriginY,
|
|
||||||
rotationAngle,
|
|
||||||
]);
|
|
||||||
|
|
||||||
const transformedStationsEn = useMemo(() => {
|
|
||||||
if (!stationDataEn) return [];
|
|
||||||
|
|
||||||
return stationDataEn.map((station) => {
|
|
||||||
const { x, y } = transformGeoToMapLocal(
|
|
||||||
station.latitude,
|
|
||||||
station.longitude
|
|
||||||
);
|
|
||||||
const rotatedCoords = rotatePoint(
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
rotationOriginX,
|
|
||||||
rotationOriginY,
|
|
||||||
rotationAngle
|
|
||||||
);
|
|
||||||
return {
|
|
||||||
...station,
|
|
||||||
longitude: rotatedCoords.x,
|
|
||||||
latitude: rotatedCoords.y,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
}, [
|
|
||||||
stationDataEn,
|
|
||||||
transformGeoToMapLocal,
|
|
||||||
rotationOriginX,
|
|
||||||
rotationOriginY,
|
|
||||||
rotationAngle,
|
|
||||||
]);
|
|
||||||
|
|
||||||
const transformedStationsZh = useMemo(() => {
|
|
||||||
if (!stationDataZh) return [];
|
|
||||||
|
|
||||||
return stationDataZh.map((station) => {
|
|
||||||
const { x, y } = transformGeoToMapLocal(
|
|
||||||
station.latitude,
|
|
||||||
station.longitude
|
|
||||||
);
|
|
||||||
const rotatedCoords = rotatePoint(
|
|
||||||
x,
|
|
||||||
y,
|
|
||||||
rotationOriginX,
|
|
||||||
rotationOriginY,
|
|
||||||
rotationAngle
|
|
||||||
);
|
|
||||||
return {
|
|
||||||
...station,
|
|
||||||
longitude: rotatedCoords.x,
|
|
||||||
latitude: rotatedCoords.y,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
}, [
|
|
||||||
stationDataZh,
|
|
||||||
transformGeoToMapLocal,
|
|
||||||
rotationOriginX,
|
|
||||||
rotationOriginY,
|
|
||||||
rotationAngle,
|
|
||||||
]);
|
|
||||||
|
|
||||||
if (
|
if (
|
||||||
!routeData ||
|
!routeData ||
|
||||||
!stationData ||
|
!stationData ||
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
// SightsLayer.tsx
|
// SightsLayer.tsx
|
||||||
import React from "react";
|
|
||||||
import { Graphics, Assets, Texture, TextStyle } from "pixi.js";
|
import { Graphics, Assets, Texture, TextStyle } from "pixi.js";
|
||||||
import { useCallback, useEffect, useState, useMemo } from "react";
|
import { useCallback, useEffect, useState, useMemo } from "react";
|
||||||
import { useTransform } from "./transformContext";
|
import { useTransform } from "./transformContext";
|
||||||
@@ -9,7 +8,6 @@ import { useGeolocationStore } from "../../stores"; // Импортируем us
|
|||||||
|
|
||||||
const BASE_ICON_SIZE = 30;
|
const BASE_ICON_SIZE = 30;
|
||||||
const CLUSTER_RADIUS_BASE = 10;
|
const CLUSTER_RADIUS_BASE = 10;
|
||||||
const CLUSTER_COLOR = 0x1a73e8;
|
|
||||||
|
|
||||||
type Cluster = {
|
type Cluster = {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -150,7 +148,7 @@ function SingleSight({
|
|||||||
readonly sight: SightData;
|
readonly sight: SightData;
|
||||||
onSightClick: (sightId: string) => void;
|
onSightClick: (sightId: string) => void;
|
||||||
}) {
|
}) {
|
||||||
const { scale } = useTransform();
|
useTransform();
|
||||||
const [texture, setTexture] = useState<Texture>(Texture.EMPTY);
|
const [texture, setTexture] = useState<Texture>(Texture.EMPTY);
|
||||||
const store = useGeolocationStore();
|
const store = useGeolocationStore();
|
||||||
const { setIsGovernorWidgetOpen } = store;
|
const { setIsGovernorWidgetOpen } = store;
|
||||||
@@ -197,7 +195,7 @@ function SightCluster({
|
|||||||
}) {
|
}) {
|
||||||
const store = useGeolocationStore();
|
const store = useGeolocationStore();
|
||||||
const { setIsGovernorWidgetOpen } = store;
|
const { setIsGovernorWidgetOpen } = store;
|
||||||
const { scale } = useTransform();
|
useTransform();
|
||||||
const radius = CLUSTER_RADIUS_BASE;
|
const radius = CLUSTER_RADIUS_BASE;
|
||||||
const [texture, setTexture] = useState<Texture>(Texture.EMPTY);
|
const [texture, setTexture] = useState<Texture>(Texture.EMPTY);
|
||||||
const fontSize = 14;
|
const fontSize = 14;
|
||||||
@@ -334,7 +332,7 @@ export function SightsLayer({
|
|||||||
sights,
|
sights,
|
||||||
pathPoints,
|
pathPoints,
|
||||||
}: Readonly<SightsLayerProps>) {
|
}: Readonly<SightsLayerProps>) {
|
||||||
const { scale } = useTransform();
|
useTransform();
|
||||||
const distanceThreshold = BASE_ICON_SIZE * 3;
|
const distanceThreshold = BASE_ICON_SIZE * 3;
|
||||||
|
|
||||||
const store = useGeolocationStore(); // Получаем доступ к MobX хранилищу
|
const store = useGeolocationStore(); // Получаем доступ к MobX хранилищу
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import React from "react";
|
import { Texture, Assets } from "pixi.js";
|
||||||
import { Texture, Assets, Graphics } from "pixi.js";
|
|
||||||
import { useEffect, useState, useMemo, useRef } from "react";
|
import { useEffect, useState, useMemo, useRef } from "react";
|
||||||
import { useTransform } from "./transformContext";
|
import { useTransform } from "./transformContext";
|
||||||
import { lerp, lerpAngle } from "../../utils/animationUtils";
|
import { lerp, lerpAngle } from "../../utils/animationUtils";
|
||||||
@@ -11,8 +10,6 @@ const basePath = new URL(
|
|||||||
const tramPath = new URL("../../assets/tramPosition/Tram.svg", import.meta.url)
|
const tramPath = new URL("../../assets/tramPosition/Tram.svg", import.meta.url)
|
||||||
.href;
|
.href;
|
||||||
|
|
||||||
// Константы анимации (как в HTML файле)
|
|
||||||
const ANIMATION_DURATION = 1200; // 1.2 секунды
|
|
||||||
const LERP_SPEED = 0.1; // Скорость интерполяции (10% каждый кадр)
|
const LERP_SPEED = 0.1; // Скорость интерполяции (10% каждый кадр)
|
||||||
|
|
||||||
// Функция для проверки расстояния до ближайшей точки маршрута
|
// Функция для проверки расстояния до ближайшей точки маршрута
|
||||||
@@ -101,7 +98,7 @@ const getDistanceToStations = (
|
|||||||
offset_x?: number;
|
offset_x?: number;
|
||||||
offset_y?: number;
|
offset_y?: number;
|
||||||
}[],
|
}[],
|
||||||
debug: boolean = false
|
_debug: boolean = false
|
||||||
) => {
|
) => {
|
||||||
if (!stations || stations.length === 0) {
|
if (!stations || stations.length === 0) {
|
||||||
return Infinity;
|
return Infinity;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import React, { useEffect, useState, useRef } from "react";
|
import React, { useEffect, useState, useRef } from "react";
|
||||||
import { lerp, lerpAngle } from "../../utils/animationUtils";
|
import { lerpAngle } from "../../utils/animationUtils";
|
||||||
import tramBase from "../../assets/tramPosition/Tram Base.svg";
|
import tramBase from "../../assets/tramPosition/Tram Base.svg";
|
||||||
import tramSvg from "../../assets/tramPosition/Tram_Second.svg";
|
import tramSvg from "../../assets/tramPosition/Tram_Second.svg";
|
||||||
import { getMediaUrl } from "../../api/apiConfig";
|
import { getMediaUrl } from "../../api/apiConfig";
|
||||||
|
|||||||
@@ -17,9 +17,6 @@ import {
|
|||||||
UNPASSED_STATION_COLOR,
|
UNPASSED_STATION_COLOR,
|
||||||
BUS_COLOR,
|
BUS_COLOR,
|
||||||
BASE_ICON_SIZE,
|
BASE_ICON_SIZE,
|
||||||
CLUSTER_RADIUS_BASE,
|
|
||||||
CLUSTER_COLOR,
|
|
||||||
ACTIVE_STATION_COLOR,
|
|
||||||
} from "./Constants";
|
} from "./Constants";
|
||||||
import { SCALE_FACTOR } from "../../assets/Constants";
|
import { SCALE_FACTOR } from "../../assets/Constants";
|
||||||
import { apiStore } from "../../api/ApiStore/store";
|
import { apiStore } from "../../api/ApiStore/store";
|
||||||
@@ -40,7 +37,7 @@ const YELLOW_ICON_FILTER =
|
|||||||
const clamp = (value: number, min: number, max: number) =>
|
const clamp = (value: number, min: number, max: number) =>
|
||||||
Math.min(max, Math.max(min, value));
|
Math.min(max, Math.max(min, value));
|
||||||
|
|
||||||
const debugWebglLog = (...args: unknown[]) => {
|
const debugWebglLog = (..._args: unknown[]) => {
|
||||||
if (!DEBUG_WEBGL_ROUTE_MAP) return;
|
if (!DEBUG_WEBGL_ROUTE_MAP) return;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -158,6 +155,19 @@ const useSightClustering = (
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const hasCustomIcon =
|
||||||
|
sight.is_default_icon === false && !isMediaIdEmpty(sight.icon ?? null);
|
||||||
|
|
||||||
|
if (hasCustomIcon) {
|
||||||
|
sight.visited = true;
|
||||||
|
clusteredResult.push({
|
||||||
|
type: "point",
|
||||||
|
id: String(sight.id),
|
||||||
|
data: sight,
|
||||||
|
});
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
const clusterSights: SightData[] = [];
|
const clusterSights: SightData[] = [];
|
||||||
const queue = [sight];
|
const queue = [sight];
|
||||||
sight.visited = true;
|
sight.visited = true;
|
||||||
@@ -167,8 +177,12 @@ const useSightClustering = (
|
|||||||
clusterSights.push(current);
|
clusterSights.push(current);
|
||||||
|
|
||||||
for (const potentialNeighbor of unclusteredSights) {
|
for (const potentialNeighbor of unclusteredSights) {
|
||||||
|
const neighborHasCustomIcon =
|
||||||
|
potentialNeighbor.is_default_icon === false &&
|
||||||
|
!isMediaIdEmpty(potentialNeighbor.icon ?? null);
|
||||||
if (
|
if (
|
||||||
!potentialNeighbor.visited &&
|
!potentialNeighbor.visited &&
|
||||||
|
!neighborHasCustomIcon &&
|
||||||
clusterSights.length < 4 &&
|
clusterSights.length < 4 &&
|
||||||
getDistance(current, potentialNeighbor) < distanceThreshold
|
getDistance(current, potentialNeighbor) < distanceThreshold
|
||||||
) {
|
) {
|
||||||
@@ -178,6 +192,10 @@ const useSightClustering = (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for (const leftover of queue) {
|
||||||
|
leftover.visited = false;
|
||||||
|
}
|
||||||
|
|
||||||
if (clusterSights.length > 1) {
|
if (clusterSights.length > 1) {
|
||||||
let furthestSight: SightData | null = null;
|
let furthestSight: SightData | null = null;
|
||||||
let maxDistanceToPath = -1;
|
let maxDistanceToPath = -1;
|
||||||
@@ -384,12 +402,14 @@ export const WebGLMap = observer(() => {
|
|||||||
return livePercent;
|
return livePercent;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (sight?.is_default_icon === false) {
|
||||||
sight != null &&
|
if (
|
||||||
typeof sight.icon_size === "number" &&
|
typeof sight.icon_size === "number" &&
|
||||||
Number.isFinite(sight.icon_size)
|
Number.isFinite(sight.icon_size)
|
||||||
) {
|
) {
|
||||||
return sight.icon_size;
|
return sight.icon_size;
|
||||||
|
}
|
||||||
|
return 100;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
@@ -424,7 +444,7 @@ export const WebGLMap = observer(() => {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const clampPosition = useCallback(
|
const clampPosition = useCallback(
|
||||||
(pos: { x: number; y: number }, currentScale: number) => {
|
(pos: { x: number; y: number }, _currentScale: number) => {
|
||||||
return pos;
|
return pos;
|
||||||
},
|
},
|
||||||
[],
|
[],
|
||||||
@@ -799,7 +819,8 @@ export const WebGLMap = observer(() => {
|
|||||||
const textBlockPositionX = rx + labelOffsetX;
|
const textBlockPositionX = rx + labelOffsetX;
|
||||||
const textBlockPositionY = ry + labelOffsetY;
|
const textBlockPositionY = ry + labelOffsetY;
|
||||||
|
|
||||||
const approximateTextWidth = st.name.length * fontSize * 0.6;
|
const normalizedName = st.name.replace(/\\n|\n/g, "");
|
||||||
|
const approximateTextWidth = normalizedName.length * fontSize * 0.6;
|
||||||
const textWidthInMapCoords = approximateTextWidth / scale;
|
const textWidthInMapCoords = approximateTextWidth / scale;
|
||||||
|
|
||||||
let anchorXOffset = 0;
|
let anchorXOffset = 0;
|
||||||
@@ -829,8 +850,8 @@ export const WebGLMap = observer(() => {
|
|||||||
result.push({
|
result.push({
|
||||||
x: sx,
|
x: sx,
|
||||||
y: sy,
|
y: sy,
|
||||||
name: st.name,
|
name: normalizedName,
|
||||||
sub,
|
sub: sub ? sub.replace(/\\n|\n/g, "") : sub,
|
||||||
anchorX: anchorX,
|
anchorX: anchorX,
|
||||||
anchorY: anchorY,
|
anchorY: anchorY,
|
||||||
distance: distanceInPixels,
|
distance: distanceInPixels,
|
||||||
@@ -880,6 +901,43 @@ export const WebGLMap = observer(() => {
|
|||||||
rotationAngle,
|
rotationAngle,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
// Сегментный индекс каждой упорядоченной станции на routePath (canvas-пространство)
|
||||||
|
const orderedStationSegs = useMemo(() => {
|
||||||
|
if (!orderedRouteStations || !stationData || routePath.length < 4)
|
||||||
|
return [] as number[];
|
||||||
|
return (orderedRouteStations as any[]).map((ordStation) => {
|
||||||
|
const stIdx = stationData.findIndex(
|
||||||
|
(s: any) => String(s.id) === String(ordStation.id),
|
||||||
|
);
|
||||||
|
if (stIdx < 0) return -1;
|
||||||
|
const sx = stationPoints[stIdx * 2];
|
||||||
|
const sy = stationPoints[stIdx * 2 + 1];
|
||||||
|
if (sx === undefined || sy === undefined) return -1;
|
||||||
|
let best = -1,
|
||||||
|
bestD = Infinity;
|
||||||
|
for (let i = 0; i < routePath.length - 2; i += 2) {
|
||||||
|
const p1x = routePath[i],
|
||||||
|
p1y = routePath[i + 1];
|
||||||
|
const p2x = routePath[i + 2],
|
||||||
|
p2y = routePath[i + 3];
|
||||||
|
const dx = p2x - p1x,
|
||||||
|
dy = p2y - p1y;
|
||||||
|
const len2 = dx * dx + dy * dy;
|
||||||
|
if (!len2) continue;
|
||||||
|
const t = ((sx - p1x) * dx + (sy - p1y) * dy) / len2;
|
||||||
|
const cl = Math.max(0, Math.min(1, t));
|
||||||
|
const px = p1x + cl * dx,
|
||||||
|
py = p1y + cl * dy;
|
||||||
|
const d = Math.hypot(sx - px, sy - py);
|
||||||
|
if (d < bestD) {
|
||||||
|
bestD = d;
|
||||||
|
best = i / 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return best;
|
||||||
|
});
|
||||||
|
}, [orderedRouteStations, stationData, stationPoints, routePath]);
|
||||||
|
|
||||||
const sightPoints = useMemo(() => {
|
const sightPoints = useMemo(() => {
|
||||||
if (!sightData || !routeData) return new Float32Array();
|
if (!sightData || !routeData) return new Float32Array();
|
||||||
const centerLat = routeData.center_latitude;
|
const centerLat = routeData.center_latitude;
|
||||||
@@ -1097,6 +1155,8 @@ export const WebGLMap = observer(() => {
|
|||||||
};
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const prevPositionIndexRef = useRef<number>(-1);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const centerLat = routeData?.center_latitude;
|
const centerLat = routeData?.center_latitude;
|
||||||
const centerLon = routeData?.center_longitude;
|
const centerLon = routeData?.center_longitude;
|
||||||
@@ -1114,7 +1174,16 @@ export const WebGLMap = observer(() => {
|
|||||||
const rx = x * cos - y * sin;
|
const rx = x * cos - y * sin;
|
||||||
const ry = x * sin + y * cos;
|
const ry = x * sin + y * cos;
|
||||||
|
|
||||||
if (apiStore.simulationInstantMove) {
|
const curIdx = apiStore.positionIndex;
|
||||||
|
const prevIdx = prevPositionIndexRef.current;
|
||||||
|
const pathLen = apiStore.route?.path?.length ?? 0;
|
||||||
|
const isWrap =
|
||||||
|
prevIdx >= 0 &&
|
||||||
|
pathLen > 0 &&
|
||||||
|
Math.abs(curIdx - prevIdx) > pathLen / 4;
|
||||||
|
prevPositionIndexRef.current = curIdx;
|
||||||
|
|
||||||
|
if (apiStore.simulationInstantMove || isWrap) {
|
||||||
setYellowDotImmediate(rx, ry);
|
setYellowDotImmediate(rx, ry);
|
||||||
} else {
|
} else {
|
||||||
animateYellowDotTo(rx, ry);
|
animateYellowDotTo(rx, ry);
|
||||||
@@ -1165,8 +1234,8 @@ export const WebGLMap = observer(() => {
|
|||||||
gl.enableVertexAttribArray(attribs.a_pos);
|
gl.enableVertexAttribArray(attribs.a_pos);
|
||||||
gl.vertexAttribPointer(attribs.a_pos, 2, gl.FLOAT, false, 0, 0);
|
gl.vertexAttribPointer(attribs.a_pos, 2, gl.FLOAT, false, 0, 0);
|
||||||
|
|
||||||
const vcount = routePath.length / 2;
|
|
||||||
let tramSegIndex = getCurrentSegIndex();
|
let tramSegIndex = getCurrentSegIndex();
|
||||||
|
const simulationDirection = apiStore.simulationDirection;
|
||||||
|
|
||||||
const dpr = Math.max(1, window.devicePixelRatio || 1);
|
const dpr = Math.max(1, window.devicePixelRatio || 1);
|
||||||
const desiredRouteWidthCss = 7;
|
const desiredRouteWidthCss = 7;
|
||||||
@@ -1264,23 +1333,69 @@ export const WebGLMap = observer(() => {
|
|||||||
const r1 = ((PATH_COLOR >> 16) & 0xff) / 255;
|
const r1 = ((PATH_COLOR >> 16) & 0xff) / 255;
|
||||||
const g1 = ((PATH_COLOR >> 8) & 0xff) / 255;
|
const g1 = ((PATH_COLOR >> 8) & 0xff) / 255;
|
||||||
const b1 = (PATH_COLOR & 0xff) / 255;
|
const b1 = (PATH_COLOR & 0xff) / 255;
|
||||||
gl.uniform4f(uniforms.u_color, r1, g1, b1, 1);
|
const r2 = ((UNPASSED_STATION_COLOR >> 16) & 0xff) / 255;
|
||||||
|
const g2 = ((UNPASSED_STATION_COLOR >> 8) & 0xff) / 255;
|
||||||
|
const b2 = (UNPASSED_STATION_COLOR & 0xff) / 255;
|
||||||
|
|
||||||
if (tramSegIndex >= 0) {
|
const animatedPos = animatedYellowDotPosition;
|
||||||
const animatedPos = animatedYellowDotPosition;
|
if (
|
||||||
if (
|
tramSegIndex >= 0 &&
|
||||||
animatedPos &&
|
animatedPos &&
|
||||||
animatedPos.x !== undefined &&
|
animatedPos.x !== undefined &&
|
||||||
animatedPos.y !== undefined
|
animatedPos.y !== undefined
|
||||||
) {
|
) {
|
||||||
|
if (simulationDirection === 1) {
|
||||||
|
// Вперёд: закрашено от начала до трамвая
|
||||||
|
gl.uniform4f(uniforms.u_color, r1, g1, b1, 1);
|
||||||
const passedPoints: number[] = [];
|
const passedPoints: number[] = [];
|
||||||
|
|
||||||
for (let i = 0; i <= tramSegIndex; i++) {
|
for (let i = 0; i <= tramSegIndex; i++) {
|
||||||
passedPoints.push(routePath[i * 2], routePath[i * 2 + 1]);
|
passedPoints.push(routePath[i * 2], routePath[i * 2 + 1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
passedPoints.push(animatedPos.x, animatedPos.y);
|
passedPoints.push(animatedPos.x, animatedPos.y);
|
||||||
|
if (passedPoints.length >= 4) {
|
||||||
|
const thickLineVertices = generateThickLine(
|
||||||
|
new Float32Array(passedPoints),
|
||||||
|
lineWidth,
|
||||||
|
);
|
||||||
|
gl.bufferData(gl.ARRAY_BUFFER, thickLineVertices, gl.STATIC_DRAW);
|
||||||
|
gl.drawArrays(gl.TRIANGLES, 0, thickLineVertices.length / 2);
|
||||||
|
}
|
||||||
|
gl.uniform4f(uniforms.u_color, r2, g2, b2, 1);
|
||||||
|
const unpassedPoints: number[] = [];
|
||||||
|
unpassedPoints.push(animatedPos.x, animatedPos.y);
|
||||||
|
for (let i = tramSegIndex + 1; i < vertexCount; i++) {
|
||||||
|
unpassedPoints.push(routePath[i * 2], routePath[i * 2 + 1]);
|
||||||
|
}
|
||||||
|
if (unpassedPoints.length >= 4) {
|
||||||
|
const thickLineVertices = generateThickLine(
|
||||||
|
new Float32Array(unpassedPoints),
|
||||||
|
lineWidth,
|
||||||
|
);
|
||||||
|
gl.bufferData(gl.ARRAY_BUFFER, thickLineVertices, gl.STATIC_DRAW);
|
||||||
|
gl.drawArrays(gl.TRIANGLES, 0, thickLineVertices.length / 2);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Назад: закрашено от трамвая до конца
|
||||||
|
gl.uniform4f(uniforms.u_color, r2, g2, b2, 1);
|
||||||
|
const unpassedPoints: number[] = [];
|
||||||
|
for (let i = 0; i <= tramSegIndex; i++) {
|
||||||
|
unpassedPoints.push(routePath[i * 2], routePath[i * 2 + 1]);
|
||||||
|
}
|
||||||
|
unpassedPoints.push(animatedPos.x, animatedPos.y);
|
||||||
|
if (unpassedPoints.length >= 4) {
|
||||||
|
const thickLineVertices = generateThickLine(
|
||||||
|
new Float32Array(unpassedPoints),
|
||||||
|
lineWidth,
|
||||||
|
);
|
||||||
|
gl.bufferData(gl.ARRAY_BUFFER, thickLineVertices, gl.STATIC_DRAW);
|
||||||
|
gl.drawArrays(gl.TRIANGLES, 0, thickLineVertices.length / 2);
|
||||||
|
}
|
||||||
|
gl.uniform4f(uniforms.u_color, r1, g1, b1, 1);
|
||||||
|
const passedPoints: number[] = [];
|
||||||
|
passedPoints.push(animatedPos.x, animatedPos.y);
|
||||||
|
for (let i = tramSegIndex + 1; i < vertexCount; i++) {
|
||||||
|
passedPoints.push(routePath[i * 2], routePath[i * 2 + 1]);
|
||||||
|
}
|
||||||
if (passedPoints.length >= 4) {
|
if (passedPoints.length >= 4) {
|
||||||
const thickLineVertices = generateThickLine(
|
const thickLineVertices = generateThickLine(
|
||||||
new Float32Array(passedPoints),
|
new Float32Array(passedPoints),
|
||||||
@@ -1290,30 +1405,16 @@ export const WebGLMap = observer(() => {
|
|||||||
gl.drawArrays(gl.TRIANGLES, 0, thickLineVertices.length / 2);
|
gl.drawArrays(gl.TRIANGLES, 0, thickLineVertices.length / 2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
|
// Позиция трамвая неизвестна — рисуем весь маршрут серым
|
||||||
const r2 = ((UNPASSED_STATION_COLOR >> 16) & 0xff) / 255;
|
gl.uniform4f(uniforms.u_color, r2, g2, b2, 1);
|
||||||
const g2 = ((UNPASSED_STATION_COLOR >> 8) & 0xff) / 255;
|
const allPoints: number[] = [];
|
||||||
const b2 = (UNPASSED_STATION_COLOR & 0xff) / 255;
|
for (let i = 0; i < vertexCount; i++) {
|
||||||
gl.uniform4f(uniforms.u_color, r2, g2, b2, 1);
|
allPoints.push(routePath[i * 2], routePath[i * 2 + 1]);
|
||||||
|
|
||||||
const animatedPos = animatedYellowDotPosition;
|
|
||||||
if (
|
|
||||||
animatedPos &&
|
|
||||||
animatedPos.x !== undefined &&
|
|
||||||
animatedPos.y !== undefined
|
|
||||||
) {
|
|
||||||
const unpassedPoints: number[] = [];
|
|
||||||
|
|
||||||
unpassedPoints.push(animatedPos.x, animatedPos.y);
|
|
||||||
|
|
||||||
for (let i = tramSegIndex + 1; i < vertexCount; i++) {
|
|
||||||
unpassedPoints.push(routePath[i * 2], routePath[i * 2 + 1]);
|
|
||||||
}
|
}
|
||||||
|
if (allPoints.length >= 4) {
|
||||||
if (unpassedPoints.length >= 4) {
|
|
||||||
const thickLineVertices = generateThickLine(
|
const thickLineVertices = generateThickLine(
|
||||||
new Float32Array(unpassedPoints),
|
new Float32Array(allPoints),
|
||||||
lineWidth,
|
lineWidth,
|
||||||
);
|
);
|
||||||
gl.bufferData(gl.ARRAY_BUFFER, thickLineVertices, gl.STATIC_DRAW);
|
gl.bufferData(gl.ARRAY_BUFFER, thickLineVertices, gl.STATIC_DRAW);
|
||||||
@@ -1348,92 +1449,72 @@ export const WebGLMap = observer(() => {
|
|||||||
|
|
||||||
gl.uniform1f(u_pointSize, pointInnerSizePx);
|
gl.uniform1f(u_pointSize, pointInnerSizePx);
|
||||||
|
|
||||||
let currentStationIndexInOrdered = -1;
|
|
||||||
if (currentStationId && orderedRouteStations) {
|
|
||||||
currentStationIndexInOrdered = orderedRouteStations.findIndex(
|
|
||||||
(station: any) => String(station.id) === String(currentStationId),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
if (
|
||||||
currentStationIndexInOrdered >= 0 &&
|
tramSegIndex >= 0 &&
|
||||||
orderedRouteStations &&
|
orderedRouteStations &&
|
||||||
stationData
|
stationData &&
|
||||||
|
orderedStationSegs.length > 0
|
||||||
) {
|
) {
|
||||||
const passedStations: number[] = [];
|
const passedPts1: number[] = [];
|
||||||
|
const unpassedPts1: number[] = [];
|
||||||
for (let i = 0; i < currentStationIndexInOrdered; i++) {
|
for (let i = 0; i < orderedRouteStations.length; i++) {
|
||||||
const orderedStation = orderedRouteStations[i];
|
const orderedStation = (orderedRouteStations as any[])[i];
|
||||||
if (orderedStation) {
|
const stationSeg = orderedStationSegs[i] ?? -1;
|
||||||
const stationIndexInData = stationData.findIndex(
|
if (!orderedStation || stationSeg < 0) continue;
|
||||||
(station: any) =>
|
const isPassed =
|
||||||
String(station.id) === String(orderedStation.id),
|
simulationDirection === 1
|
||||||
);
|
? stationSeg < tramSegIndex
|
||||||
if (stationIndexInData >= 0) {
|
: stationSeg > tramSegIndex;
|
||||||
passedStations.push(
|
const stIdx = stationData.findIndex(
|
||||||
stationPoints[stationIndexInData * 2] as number,
|
(s: any) => String(s.id) === String(orderedStation.id),
|
||||||
stationPoints[stationIndexInData * 2 + 1] as number,
|
);
|
||||||
);
|
if (stIdx < 0) continue;
|
||||||
}
|
const sx = stationPoints[stIdx * 2] as number;
|
||||||
|
const sy = stationPoints[stIdx * 2 + 1] as number;
|
||||||
|
if (isPassed) {
|
||||||
|
passedPts1.push(sx, sy);
|
||||||
|
} else {
|
||||||
|
unpassedPts1.push(sx, sy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (passedStations.length > 0) {
|
if (passedPts1.length > 0) {
|
||||||
const r_passed = ((PATH_COLOR >> 16) & 0xff) / 255;
|
gl.uniform4f(
|
||||||
const g_passed = ((PATH_COLOR >> 8) & 0xff) / 255;
|
u_color_pts,
|
||||||
const b_passed = (PATH_COLOR & 0xff) / 255;
|
((PATH_COLOR >> 16) & 0xff) / 255,
|
||||||
gl.uniform4f(u_color_pts, r_passed, g_passed, b_passed, 1);
|
((PATH_COLOR >> 8) & 0xff) / 255,
|
||||||
|
(PATH_COLOR & 0xff) / 255,
|
||||||
|
1,
|
||||||
|
);
|
||||||
gl.bufferData(
|
gl.bufferData(
|
||||||
gl.ARRAY_BUFFER,
|
gl.ARRAY_BUFFER,
|
||||||
new Float32Array(passedStations),
|
new Float32Array(passedPts1),
|
||||||
gl.STATIC_DRAW,
|
gl.STATIC_DRAW,
|
||||||
);
|
);
|
||||||
gl.drawArrays(gl.POINTS, 0, passedStations.length / 2);
|
gl.drawArrays(gl.POINTS, 0, passedPts1.length / 2);
|
||||||
}
|
}
|
||||||
}
|
if (unpassedPts1.length > 0) {
|
||||||
|
gl.uniform4f(
|
||||||
if (
|
u_color_pts,
|
||||||
currentStationIndexInOrdered >= 0 &&
|
((UNPASSED_STATION_COLOR >> 16) & 0xff) / 255,
|
||||||
orderedRouteStations &&
|
((UNPASSED_STATION_COLOR >> 8) & 0xff) / 255,
|
||||||
stationData
|
(UNPASSED_STATION_COLOR & 0xff) / 255,
|
||||||
) {
|
1,
|
||||||
const unpassedStations: number[] = [];
|
);
|
||||||
|
|
||||||
for (
|
|
||||||
let i = currentStationIndexInOrdered + 1;
|
|
||||||
i < orderedRouteStations.length;
|
|
||||||
i++
|
|
||||||
) {
|
|
||||||
const orderedStation = orderedRouteStations[i];
|
|
||||||
if (orderedStation) {
|
|
||||||
const stationIndexInData = stationData.findIndex(
|
|
||||||
(station: any) =>
|
|
||||||
String(station.id) === String(orderedStation.id),
|
|
||||||
);
|
|
||||||
if (stationIndexInData >= 0) {
|
|
||||||
unpassedStations.push(
|
|
||||||
stationPoints[stationIndexInData * 2] as number,
|
|
||||||
stationPoints[stationIndexInData * 2 + 1] as number,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (unpassedStations.length > 0) {
|
|
||||||
const r_unpassed = ((UNPASSED_STATION_COLOR >> 16) & 0xff) / 255;
|
|
||||||
const g_unpassed = ((UNPASSED_STATION_COLOR >> 8) & 0xff) / 255;
|
|
||||||
const b_unpassed = (UNPASSED_STATION_COLOR & 0xff) / 255;
|
|
||||||
gl.uniform4f(u_color_pts, r_unpassed, g_unpassed, b_unpassed, 1);
|
|
||||||
gl.bufferData(
|
gl.bufferData(
|
||||||
gl.ARRAY_BUFFER,
|
gl.ARRAY_BUFFER,
|
||||||
new Float32Array(unpassedStations),
|
new Float32Array(unpassedPts1),
|
||||||
gl.STATIC_DRAW,
|
gl.STATIC_DRAW,
|
||||||
);
|
);
|
||||||
gl.drawArrays(gl.POINTS, 0, unpassedStations.length / 2);
|
gl.drawArrays(gl.POINTS, 0, unpassedPts1.length / 2);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const r_unpassed = ((UNPASSED_STATION_COLOR >> 16) & 0xff) / 255;
|
gl.uniform4f(
|
||||||
const g_unpassed = ((UNPASSED_STATION_COLOR >> 8) & 0xff) / 255;
|
u_color_pts,
|
||||||
const b_unpassed = (UNPASSED_STATION_COLOR & 0xff) / 255;
|
((UNPASSED_STATION_COLOR >> 16) & 0xff) / 255,
|
||||||
gl.uniform4f(u_color_pts, r_unpassed, g_unpassed, b_unpassed, 1);
|
((UNPASSED_STATION_COLOR >> 8) & 0xff) / 255,
|
||||||
|
(UNPASSED_STATION_COLOR & 0xff) / 255,
|
||||||
|
1,
|
||||||
|
);
|
||||||
gl.bufferData(gl.ARRAY_BUFFER, stationPoints, gl.STATIC_DRAW);
|
gl.bufferData(gl.ARRAY_BUFFER, stationPoints, gl.STATIC_DRAW);
|
||||||
gl.drawArrays(gl.POINTS, 0, stationPoints.length / 2);
|
gl.drawArrays(gl.POINTS, 0, stationPoints.length / 2);
|
||||||
}
|
}
|
||||||
@@ -1452,53 +1533,6 @@ export const WebGLMap = observer(() => {
|
|||||||
|
|
||||||
const toPointsArray = (arr: number[]) => new Float32Array(arr);
|
const toPointsArray = (arr: number[]) => new Float32Array(arr);
|
||||||
|
|
||||||
const pathPts: { x: number; y: number }[] = [];
|
|
||||||
for (let i = 0; i < routePath.length; i += 2)
|
|
||||||
pathPts.push({ x: routePath[i], y: routePath[i + 1] });
|
|
||||||
const getSeg = (px: number, py: number) => {
|
|
||||||
if (pathPts.length < 2) return -1;
|
|
||||||
let best = -1,
|
|
||||||
bestD = Infinity;
|
|
||||||
for (let i = 0; i < pathPts.length - 1; i++) {
|
|
||||||
const p1 = pathPts[i],
|
|
||||||
p2 = pathPts[i + 1];
|
|
||||||
const dx = p2.x - p1.x,
|
|
||||||
dy = p2.y - p1.y;
|
|
||||||
const len2 = dx * dx + dy * dy;
|
|
||||||
if (!len2) continue;
|
|
||||||
const t = ((px - p1.x) * dx + (py - p1.y) * dy) / len2;
|
|
||||||
const tt = Math.max(0, Math.min(1, t));
|
|
||||||
const cx = p1.x + tt * dx,
|
|
||||||
cy = p1.y + tt * dy;
|
|
||||||
const d = Math.hypot(px - cx, py - cy);
|
|
||||||
if (d < bestD) {
|
|
||||||
bestD = d;
|
|
||||||
best = i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return best;
|
|
||||||
};
|
|
||||||
|
|
||||||
let tramSegForStations = -1;
|
|
||||||
{
|
|
||||||
const cLat = routeData?.center_latitude,
|
|
||||||
cLon = routeData?.center_longitude;
|
|
||||||
const tram = apiStore?.context?.currentCoordinates as any;
|
|
||||||
if (tram && cLat !== undefined && cLon !== undefined) {
|
|
||||||
const loc = coordinatesToLocal(
|
|
||||||
tram.latitude - cLat,
|
|
||||||
tram.longitude - cLon,
|
|
||||||
);
|
|
||||||
const wx = loc.x * UP_SCALE,
|
|
||||||
wy = loc.y * UP_SCALE;
|
|
||||||
const cosR = Math.cos(rotationAngle),
|
|
||||||
sinR = Math.sin(rotationAngle);
|
|
||||||
const tx = wx * cosR - wy * sinR,
|
|
||||||
ty = wx * sinR + wy * cosR;
|
|
||||||
tramSegForStations = getSeg(tx, ty);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let activeStationIndex = -1;
|
let activeStationIndex = -1;
|
||||||
const tramCoords = apiStore?.context?.currentCoordinates;
|
const tramCoords = apiStore?.context?.currentCoordinates;
|
||||||
if (
|
if (
|
||||||
@@ -1551,37 +1585,26 @@ export const WebGLMap = observer(() => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let currentStationIndexInOrdered = -1;
|
|
||||||
if (currentStationId && orderedRouteStations) {
|
|
||||||
currentStationIndexInOrdered = orderedRouteStations.findIndex(
|
|
||||||
(station: any) => String(station.id) === String(currentStationId),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const passedStationIds = new Set<string>();
|
const passedStationIds = new Set<string>();
|
||||||
const unpassedStationIds = new Set<string>();
|
const unpassedStationIds = new Set<string>();
|
||||||
|
|
||||||
if (currentStationIndexInOrdered >= 0 && orderedRouteStations) {
|
if (
|
||||||
for (let i = 0; i < currentStationIndexInOrdered; i++) {
|
tramSegIndex >= 0 &&
|
||||||
const station = orderedRouteStations[i];
|
orderedRouteStations &&
|
||||||
if (station) {
|
orderedStationSegs.length === orderedRouteStations.length
|
||||||
passedStationIds.add(String(station.id));
|
) {
|
||||||
}
|
for (let i = 0; i < orderedRouteStations.length; i++) {
|
||||||
}
|
const station = (orderedRouteStations as any[])[i];
|
||||||
|
const seg = orderedStationSegs[i] ?? -1;
|
||||||
for (
|
if (!station || seg < 0) continue;
|
||||||
let i = currentStationIndexInOrdered;
|
const isPassed =
|
||||||
i < orderedRouteStations.length;
|
simulationDirection === 1 ? seg < tramSegIndex : seg > tramSegIndex;
|
||||||
i++
|
if (isPassed) passedStationIds.add(String(station.id));
|
||||||
) {
|
else unpassedStationIds.add(String(station.id));
|
||||||
const station = orderedRouteStations[i];
|
|
||||||
if (station) {
|
|
||||||
unpassedStationIds.add(String(station.id));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (orderedRouteStations) {
|
if (orderedRouteStations) {
|
||||||
orderedRouteStations.forEach((station: any) => {
|
(orderedRouteStations as any[]).forEach((station) => {
|
||||||
unpassedStationIds.add(String(station.id));
|
unpassedStationIds.add(String(station.id));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -1718,12 +1741,27 @@ export const WebGLMap = observer(() => {
|
|||||||
const cos = Math.cos(rotationAngle);
|
const cos = Math.cos(rotationAngle);
|
||||||
const sin = Math.sin(rotationAngle);
|
const sin = Math.sin(rotationAngle);
|
||||||
|
|
||||||
const startStationData = stationData.find(
|
const startStationData = orderedRouteStations?.[0]
|
||||||
(station) => station.id.toString() === apiStore.context?.startStopId,
|
? stationData.find(
|
||||||
);
|
(station: any) =>
|
||||||
const endStationData = stationData.find(
|
station.id.toString() === String(orderedRouteStations[0].id),
|
||||||
(station) => station.id.toString() === apiStore.context?.endStopId,
|
)
|
||||||
);
|
: stationData.find(
|
||||||
|
(station: any) =>
|
||||||
|
station.id.toString() === apiStore.context?.startStopId,
|
||||||
|
);
|
||||||
|
const endStationData = orderedRouteStations?.length
|
||||||
|
? stationData.find(
|
||||||
|
(station: any) =>
|
||||||
|
station.id.toString() ===
|
||||||
|
String(
|
||||||
|
orderedRouteStations[orderedRouteStations.length - 1].id,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
: stationData.find(
|
||||||
|
(station: any) =>
|
||||||
|
station.id.toString() === apiStore.context?.endStopId,
|
||||||
|
);
|
||||||
|
|
||||||
const terminalStations: number[] = [];
|
const terminalStations: number[] = [];
|
||||||
|
|
||||||
@@ -1823,7 +1861,13 @@ export const WebGLMap = observer(() => {
|
|||||||
}
|
}
|
||||||
return best;
|
return best;
|
||||||
})();
|
})();
|
||||||
return tramSegIndex !== -1 && seg !== -1 && seg < tramSegIndex;
|
return (
|
||||||
|
tramSegIndex !== -1 &&
|
||||||
|
seg !== -1 &&
|
||||||
|
(simulationDirection === 1
|
||||||
|
? seg < tramSegIndex
|
||||||
|
: seg > tramSegIndex)
|
||||||
|
);
|
||||||
})()
|
})()
|
||||||
: false;
|
: false;
|
||||||
|
|
||||||
@@ -1856,7 +1900,13 @@ export const WebGLMap = observer(() => {
|
|||||||
}
|
}
|
||||||
return best;
|
return best;
|
||||||
})();
|
})();
|
||||||
return tramSegIndex !== -1 && seg !== -1 && seg < tramSegIndex;
|
return (
|
||||||
|
tramSegIndex !== -1 &&
|
||||||
|
seg !== -1 &&
|
||||||
|
(simulationDirection === 1
|
||||||
|
? seg < tramSegIndex
|
||||||
|
: seg > tramSegIndex)
|
||||||
|
);
|
||||||
})()
|
})()
|
||||||
: false;
|
: false;
|
||||||
|
|
||||||
@@ -1882,20 +1932,24 @@ export const WebGLMap = observer(() => {
|
|||||||
const b_unpassed = (UNPASSED_STATION_COLOR & 0xff) / 255;
|
const b_unpassed = (UNPASSED_STATION_COLOR & 0xff) / 255;
|
||||||
|
|
||||||
if (startStationData && endStationData) {
|
if (startStationData && endStationData) {
|
||||||
gl.uniform4f(u_color_pts, r_passed, g_passed, b_passed, 1.0);
|
const startIsPassed =
|
||||||
|
simulationDirection === 1 ? true : isStartPassed;
|
||||||
|
const endIsPassed = simulationDirection === -1 ? true : isEndPassed;
|
||||||
|
gl.uniform4f(
|
||||||
|
u_color_pts,
|
||||||
|
startIsPassed ? r_passed : r_unpassed,
|
||||||
|
startIsPassed ? g_passed : g_unpassed,
|
||||||
|
startIsPassed ? b_passed : b_unpassed,
|
||||||
|
1.0,
|
||||||
|
);
|
||||||
gl.drawArrays(gl.POINTS, 0, 1);
|
gl.drawArrays(gl.POINTS, 0, 1);
|
||||||
|
gl.uniform4f(
|
||||||
if (isEndPassed) {
|
u_color_pts,
|
||||||
gl.uniform4f(u_color_pts, r_passed, g_passed, b_passed, 1.0);
|
endIsPassed ? r_passed : r_unpassed,
|
||||||
} else {
|
endIsPassed ? g_passed : g_unpassed,
|
||||||
gl.uniform4f(
|
endIsPassed ? b_passed : b_unpassed,
|
||||||
u_color_pts,
|
1.0,
|
||||||
r_unpassed,
|
);
|
||||||
g_unpassed,
|
|
||||||
b_unpassed,
|
|
||||||
1.0,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
gl.drawArrays(gl.POINTS, 1, 1);
|
gl.drawArrays(gl.POINTS, 1, 1);
|
||||||
} else {
|
} else {
|
||||||
const isStartStation = startStationData !== undefined;
|
const isStartStation = startStationData !== undefined;
|
||||||
@@ -1935,6 +1989,8 @@ export const WebGLMap = observer(() => {
|
|||||||
nearestStationId,
|
nearestStationId,
|
||||||
currentStationId,
|
currentStationId,
|
||||||
orderedRouteStations,
|
orderedRouteStations,
|
||||||
|
orderedStationSegs,
|
||||||
|
apiStore.simulationDirection,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -2331,11 +2387,10 @@ export const WebGLMap = observer(() => {
|
|||||||
? { right: 0, transform: "none" }
|
? { right: 0, transform: "none" }
|
||||||
: { left: "50%", transform: "translateX(-50%)" };
|
: { left: "50%", transform: "translateX(-50%)" };
|
||||||
|
|
||||||
const apiBaseUrl = apiBaseURL;
|
|
||||||
const isMediaIdEmptyResult = isMediaIdEmpty(station?.icon);
|
const isMediaIdEmptyResult = isMediaIdEmpty(station?.icon);
|
||||||
const iconSrc = isMediaIdEmptyResult
|
const iconSrc = isMediaIdEmptyResult
|
||||||
? null
|
? null
|
||||||
: `${apiBaseUrl}/media/${station?.icon}/download`;
|
: buildMediaDownloadUrl(mediaBaseUrl, station!.icon!, mediaToken);
|
||||||
const iconSizePx = Math.round(primaryFontSize * 1.2);
|
const iconSizePx = Math.round(primaryFontSize * 1.2);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -2481,6 +2536,11 @@ export const WebGLMap = observer(() => {
|
|||||||
cursor: "pointer",
|
cursor: "pointer",
|
||||||
userSelect: "none",
|
userSelect: "none",
|
||||||
touchAction: "none",
|
touchAction: "none",
|
||||||
|
...(apiStore.showHitboxes && {
|
||||||
|
outline: "2px solid rgba(0,255,0,0.8)",
|
||||||
|
outlineOffset: "2px",
|
||||||
|
backgroundColor: "rgba(0,255,0,0.08)",
|
||||||
|
}),
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
@@ -2513,14 +2573,8 @@ export const WebGLMap = observer(() => {
|
|||||||
const rx = cluster.longitude;
|
const rx = cluster.longitude;
|
||||||
const ry = cluster.latitude;
|
const ry = cluster.latitude;
|
||||||
|
|
||||||
const iconSizePercent = resolveSightIconSizePercent();
|
|
||||||
const iconSize =
|
|
||||||
SIGHT_ICON_BASE_SIZE * clamp(iconSizePercent / 100, 0.1, 10);
|
|
||||||
|
|
||||||
const screenX = (rx * scale + position.x) / dpr;
|
const screenX = (rx * scale + position.x) / dpr;
|
||||||
const screenY = (ry * scale + position.y) / dpr;
|
const screenY = (ry * scale + position.y) / dpr;
|
||||||
const iconLeft = screenX - iconSize / 2;
|
|
||||||
const iconTop = screenY - iconSize / 2;
|
|
||||||
|
|
||||||
const isExpanded = activeClusterId === cluster.id;
|
const isExpanded = activeClusterId === cluster.id;
|
||||||
const selectedSightInCluster = cluster.sights.find(
|
const selectedSightInCluster = cluster.sights.find(
|
||||||
@@ -2532,6 +2586,20 @@ export const WebGLMap = observer(() => {
|
|||||||
const selectedIsCustomInCluster =
|
const selectedIsCustomInCluster =
|
||||||
selectedSightInCluster?.is_default_icon === false &&
|
selectedSightInCluster?.is_default_icon === false &&
|
||||||
selectedHasIconInCluster;
|
selectedHasIconInCluster;
|
||||||
|
|
||||||
|
const iconSizePercent = resolveSightIconSizePercent(
|
||||||
|
cluster.sights[0],
|
||||||
|
);
|
||||||
|
const clusterCustomScaleFactor = selectedIsCustomInCluster
|
||||||
|
? scale / Math.max(customSightIconBaseScaleRef.current ?? 1, 1e-6)
|
||||||
|
: 1;
|
||||||
|
const iconSize =
|
||||||
|
SIGHT_ICON_BASE_SIZE *
|
||||||
|
clamp(iconSizePercent / 100, 0.1, 10) *
|
||||||
|
clusterCustomScaleFactor;
|
||||||
|
|
||||||
|
const iconLeft = screenX - iconSize;
|
||||||
|
const iconTop = screenY - iconSize;
|
||||||
const hasSelectedAltIconInCluster =
|
const hasSelectedAltIconInCluster =
|
||||||
selectedSightInCluster != null &&
|
selectedSightInCluster != null &&
|
||||||
selectedIsCustomInCluster &&
|
selectedIsCustomInCluster &&
|
||||||
@@ -2589,7 +2657,7 @@ export const WebGLMap = observer(() => {
|
|||||||
)
|
)
|
||||||
: false;
|
: false;
|
||||||
|
|
||||||
const badgeColor = "#006F3A";
|
const badgeColor = "var(--carrier-main, #006F3A)";
|
||||||
const listPanelWidth = 200;
|
const listPanelWidth = 200;
|
||||||
const listItemHeight = 30;
|
const listItemHeight = 30;
|
||||||
const listMaxHeight = 250;
|
const listMaxHeight = 250;
|
||||||
@@ -2605,29 +2673,38 @@ export const WebGLMap = observer(() => {
|
|||||||
onTouchEnd={handleClusterClick}
|
onTouchEnd={handleClusterClick}
|
||||||
style={{
|
style={{
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
left: iconLeft - CLUSTER_RADIUS_BASE - 10,
|
left: iconLeft,
|
||||||
top: iconTop - CLUSTER_RADIUS_BASE - 10,
|
top: iconTop,
|
||||||
width: iconSize + CLUSTER_RADIUS_BASE * 2 + 20,
|
width: iconSize,
|
||||||
height: iconSize + CLUSTER_RADIUS_BASE * 2 + 20,
|
height: iconSize,
|
||||||
display: "flex",
|
|
||||||
alignItems: "center",
|
|
||||||
justifyContent: "center",
|
|
||||||
pointerEvents: "auto",
|
pointerEvents: "auto",
|
||||||
cursor: "pointer",
|
cursor: "pointer",
|
||||||
userSelect: "none",
|
userSelect: "none",
|
||||||
touchAction: "none",
|
touchAction: "none",
|
||||||
zIndex: 10,
|
zIndex: 10,
|
||||||
|
...(apiStore.showHitboxes && {
|
||||||
|
outline: "2px solid rgba(255,165,0,0.8)",
|
||||||
|
outlineOffset: "2px",
|
||||||
|
backgroundColor: "rgba(255,165,0,0.08)",
|
||||||
|
}),
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div style={{ position: "relative" }}>
|
<div
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
width: iconSize,
|
||||||
|
height: iconSize,
|
||||||
|
flexShrink: 0,
|
||||||
|
}}
|
||||||
|
>
|
||||||
<img
|
<img
|
||||||
src={clusterIconUrl}
|
src={clusterIconUrl}
|
||||||
alt=""
|
alt=""
|
||||||
width={iconSize}
|
|
||||||
height={iconSize}
|
|
||||||
style={{
|
style={{
|
||||||
display: "block",
|
display: "block",
|
||||||
pointerEvents: "none",
|
pointerEvents: "none",
|
||||||
|
width: "100%",
|
||||||
|
height: "100%",
|
||||||
filter: hasSelectedSight
|
filter: hasSelectedSight
|
||||||
? hasSelectedAltIconInCluster
|
? hasSelectedAltIconInCluster
|
||||||
? "none"
|
? "none"
|
||||||
@@ -2640,6 +2717,7 @@ export const WebGLMap = observer(() => {
|
|||||||
position: "absolute",
|
position: "absolute",
|
||||||
top: -6,
|
top: -6,
|
||||||
right: -6,
|
right: -6,
|
||||||
|
zIndex: 1,
|
||||||
width: 15,
|
width: 15,
|
||||||
height: 15,
|
height: 15,
|
||||||
borderRadius: "10px",
|
borderRadius: "10px",
|
||||||
@@ -2662,16 +2740,20 @@ export const WebGLMap = observer(() => {
|
|||||||
<div
|
<div
|
||||||
data-expanded-cluster={cluster.id}
|
data-expanded-cluster={cluster.id}
|
||||||
onTouchStart={handleCircleInteraction}
|
onTouchStart={handleCircleInteraction}
|
||||||
onTouchMove={handleCircleInteraction}
|
|
||||||
onMouseMove={handleCircleInteraction}
|
onMouseMove={handleCircleInteraction}
|
||||||
style={{
|
style={{
|
||||||
position: "absolute",
|
position: "absolute",
|
||||||
left: screenX - iconSize / 2,
|
left: screenX - iconSize,
|
||||||
top: screenY - iconSize / 2,
|
top: screenY - iconSize,
|
||||||
display: "flex",
|
display: "flex",
|
||||||
alignItems: "flex-start",
|
alignItems: "flex-start",
|
||||||
pointerEvents: "auto",
|
pointerEvents: "auto",
|
||||||
zIndex: 100000000000000,
|
zIndex: 100000000000000,
|
||||||
|
...(apiStore.showHitboxes && {
|
||||||
|
outline: "2px solid rgba(0,180,255,0.8)",
|
||||||
|
outlineOffset: "2px",
|
||||||
|
backgroundColor: "rgba(0,180,255,0.08)",
|
||||||
|
}),
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
@@ -2723,15 +2805,14 @@ export const WebGLMap = observer(() => {
|
|||||||
<div
|
<div
|
||||||
className="cluster-sights-list"
|
className="cluster-sights-list"
|
||||||
style={{
|
style={{
|
||||||
background: `linear-gradient(to bottom right, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0) 100%), rgba(0, 111, 58, 0.4)`,
|
background: `linear-gradient(to bottom right, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0) 100%), rgba(var(--carrier-main-rgb, 0, 111, 58), 0.4)`,
|
||||||
backdropFilter: "blur(10px)",
|
backdropFilter: "blur(10px)",
|
||||||
borderRadius: "8px",
|
borderRadius: "8px",
|
||||||
width: listPanelWidth,
|
width: listPanelWidth,
|
||||||
maxHeight: hasMoreThanTwo ? listMaxHeight : undefined,
|
maxHeight: hasMoreThanTwo ? listMaxHeight : undefined,
|
||||||
boxShadow:
|
boxShadow:
|
||||||
"0 0 0 1px rgba(255, 255, 255, 0.3) inset, 4px 4px 12px 0 rgba(255, 255, 255, 0.12) inset",
|
"inset 0 0 0 1px rgba(255, 255, 255, 0.3), inset 4px 4px 12px 0 rgba(255, 255, 255, 0.12)",
|
||||||
display: "flex",
|
display: "flex",
|
||||||
|
|
||||||
flexDirection: "column",
|
flexDirection: "column",
|
||||||
}}
|
}}
|
||||||
onClick={(e) => e.stopPropagation()}
|
onClick={(e) => e.stopPropagation()}
|
||||||
@@ -2874,7 +2955,6 @@ export const WebGLMap = observer(() => {
|
|||||||
whiteSpace: "nowrap",
|
whiteSpace: "nowrap",
|
||||||
flex: 1,
|
flex: 1,
|
||||||
}}
|
}}
|
||||||
title={sightName}
|
|
||||||
>
|
>
|
||||||
{sightName}
|
{sightName}
|
||||||
</span>
|
</span>
|
||||||
|
|||||||
@@ -4,9 +4,7 @@ import React, {
|
|||||||
useContext,
|
useContext,
|
||||||
useState,
|
useState,
|
||||||
useCallback,
|
useCallback,
|
||||||
useRef,
|
|
||||||
} from "react";
|
} from "react";
|
||||||
import { UP_SCALE } from "./Constants";
|
|
||||||
|
|
||||||
const TransformContext = createContext<{
|
const TransformContext = createContext<{
|
||||||
position: { x: number; y: number };
|
position: { x: number; y: number };
|
||||||
|
|||||||
@@ -1,20 +1,16 @@
|
|||||||
|
import subtractHomeIcon from "../../assets/icons/subtract-home.svg";
|
||||||
|
|
||||||
function BackButtonSVG({ onPointerUp }) {
|
function BackButtonSVG({ onPointerUp }) {
|
||||||
return (
|
return (
|
||||||
<svg
|
<img
|
||||||
|
src={subtractHomeIcon}
|
||||||
|
alt=""
|
||||||
|
width="24"
|
||||||
|
height="21"
|
||||||
onPointerUp={onPointerUp}
|
onPointerUp={onPointerUp}
|
||||||
className="sight-frame-get-back"
|
className="sight-frame-get-back"
|
||||||
width="13"
|
style={{ cursor: "pointer", display: "block" }}
|
||||||
height="22"
|
/>
|
||||||
viewBox="0 0 13 22"
|
|
||||||
fill="none"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
style={{ cursor: "pointer", transform: "rotate(-90deg)" }}
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M4.03042 11.0738C4.21763 11.2087 4.34586 11.2766 4.44415 11.3747C6.92933 13.8687 9.40982 16.3665 11.895 18.8605C12.3396 19.3066 12.5175 19.8207 12.3349 20.4339C12.027 21.4658 10.7764 21.8393 9.95179 21.1498C9.86193 21.0743 9.78049 20.9894 9.69812 20.9064C6.85163 18.0446 4.00421 15.1817 1.15866 12.3189C0.279718 11.4341 0.280654 10.6456 1.16053 9.75982C4.03884 6.86305 6.91341 3.96155 9.8067 1.07986C10.0585 0.828952 10.4133 0.619547 10.7549 0.537482C11.4008 0.382786 12.0139 0.72142 12.3181 1.28644C12.6251 1.85334 12.5381 2.56079 12.0794 3.05129C11.7275 3.4286 11.3549 3.78704 10.9899 4.15209C8.79485 6.35557 6.60078 8.55904 4.40483 10.7606C4.32246 10.8436 4.22605 10.9106 4.03042 11.0729L4.03042 11.0738Z"
|
|
||||||
fill="white"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,8 @@ import { useGeolocationStore } from "../../stores";
|
|||||||
import "../../styles/LeftWidget.css";
|
import "../../styles/LeftWidget.css";
|
||||||
import { apiStore } from "../../api/ApiStore/store";
|
import { apiStore } from "../../api/ApiStore/store";
|
||||||
import { apiBaseURL } from "../../api/apiConfig";
|
import { apiBaseURL } from "../../api/apiConfig";
|
||||||
|
import { ReactMarkdownComponent } from "../ReactMarkdown";
|
||||||
|
import { TouchableLayout } from "../TouchableLayout";
|
||||||
|
|
||||||
const LeftWidget = observer(
|
const LeftWidget = observer(
|
||||||
({ selectedSightId, onClose, isVisible, sightTop }) => {
|
({ selectedSightId, onClose, isVisible, sightTop }) => {
|
||||||
@@ -15,8 +17,7 @@ const LeftWidget = observer(
|
|||||||
const [isImageLoaded, setIsImageLoaded] = useState(false);
|
const [isImageLoaded, setIsImageLoaded] = useState(false);
|
||||||
const [widgetHeight, setWidgetHeight] = useState(0);
|
const [widgetHeight, setWidgetHeight] = useState(0);
|
||||||
|
|
||||||
const textRef = useRef(null);
|
const layoutRef = useRef(null);
|
||||||
const activeTouch = useRef(null);
|
|
||||||
const widgetRef = useRef(null);
|
const widgetRef = useRef(null);
|
||||||
|
|
||||||
const store = useGeolocationStore();
|
const store = useGeolocationStore();
|
||||||
@@ -37,64 +38,10 @@ const LeftWidget = observer(
|
|||||||
}, [selectedSightData, isImageLoaded, isVisible, isLoading, error]);
|
}, [selectedSightData, isImageLoaded, isVisible, isLoading, error]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const scrollContainer = textRef.current;
|
if (layoutRef.current) {
|
||||||
if (!scrollContainer) return;
|
const scrollable = layoutRef.current.querySelector(".scrollable");
|
||||||
|
if (scrollable) scrollable.scrollTop = 0;
|
||||||
const handleTouchStart = (e) => {
|
}
|
||||||
e.stopPropagation();
|
|
||||||
if (e.touches.length === 1) {
|
|
||||||
activeTouch.current = {
|
|
||||||
identifier: e.touches[0].identifier,
|
|
||||||
lastY: e.touches[0].clientY,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleTouchMove = (e) => {
|
|
||||||
e.preventDefault();
|
|
||||||
if (activeTouch.current) {
|
|
||||||
for (const touch of e.changedTouches) {
|
|
||||||
if (touch.identifier === activeTouch.current.identifier) {
|
|
||||||
const deltaY = touch.clientY - activeTouch.current.lastY;
|
|
||||||
scrollContainer.scrollTop -= deltaY;
|
|
||||||
activeTouch.current.lastY = touch.clientY;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleTouchEnd = (e) => {
|
|
||||||
for (const touch of e.changedTouches) {
|
|
||||||
if (
|
|
||||||
activeTouch.current &&
|
|
||||||
touch.identifier === activeTouch.current.identifier
|
|
||||||
) {
|
|
||||||
activeTouch.current = null;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
scrollContainer.addEventListener("touchstart", handleTouchStart, {
|
|
||||||
passive: true,
|
|
||||||
});
|
|
||||||
scrollContainer.addEventListener("touchmove", handleTouchMove, {
|
|
||||||
passive: false,
|
|
||||||
});
|
|
||||||
scrollContainer.addEventListener("touchend", handleTouchEnd, {
|
|
||||||
passive: true,
|
|
||||||
});
|
|
||||||
scrollContainer.addEventListener("touchcancel", handleTouchEnd, {
|
|
||||||
passive: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
scrollContainer.removeEventListener("touchstart", handleTouchStart);
|
|
||||||
scrollContainer.removeEventListener("touchmove", handleTouchMove);
|
|
||||||
scrollContainer.removeEventListener("touchend", handleTouchEnd);
|
|
||||||
scrollContainer.removeEventListener("touchcancel", handleTouchEnd);
|
|
||||||
};
|
|
||||||
}, [selectedSightData]);
|
}, [selectedSightData]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -120,8 +67,10 @@ const LeftWidget = observer(
|
|||||||
selectedLanguage === "ru"
|
selectedLanguage === "ru"
|
||||||
? routeSights.find((sight) => sight.id === selectedSightId)
|
? routeSights.find((sight) => sight.id === selectedSightId)
|
||||||
: selectedLanguage === "en"
|
: selectedLanguage === "en"
|
||||||
? routeSightsEn.find((sight) => sight.id === selectedSightId)
|
? routeSightsEn.find((sight) => sight.id === selectedSightId)
|
||||||
: routeSightsZh.find((sight) => sight.id === selectedSightId);
|
: routeSightsZh.find((sight) => sight.id === selectedSightId);
|
||||||
|
|
||||||
|
if (!sight) return;
|
||||||
|
|
||||||
const leftArticle = sight.left_article;
|
const leftArticle = sight.left_article;
|
||||||
|
|
||||||
@@ -129,18 +78,20 @@ const LeftWidget = observer(
|
|||||||
selectedLanguage === "ru"
|
selectedLanguage === "ru"
|
||||||
? sightArticles.get(leftArticle + "_" + selectedLanguage)
|
? sightArticles.get(leftArticle + "_" + selectedLanguage)
|
||||||
: selectedLanguage === "en"
|
: selectedLanguage === "en"
|
||||||
? sightArticlesEn.get(leftArticle + "_" + selectedLanguage)
|
? sightArticlesEn.get(leftArticle + "_" + selectedLanguage)
|
||||||
: sightArticlesZh.get(leftArticle + "_" + selectedLanguage);
|
: sightArticlesZh.get(leftArticle + "_" + selectedLanguage);
|
||||||
|
|
||||||
|
if (!leftArticleData?.media?.length) return;
|
||||||
|
|
||||||
const media = await ContentAPI.getMediaPreview(
|
const media = await ContentAPI.getMediaPreview(
|
||||||
leftArticleData.media[0].id,
|
leftArticleData.media[0].id,
|
||||||
selectedLanguage
|
selectedLanguage,
|
||||||
);
|
);
|
||||||
|
|
||||||
const response = {
|
const response = {
|
||||||
mediaPath: media.path,
|
mediaPath: media.path,
|
||||||
mediaType: media.type,
|
mediaType: media.type,
|
||||||
title: sight.short_name || sight.name || leftArticleData.heading,
|
title: leftArticleData.heading,
|
||||||
text: leftArticleData.body,
|
text: leftArticleData.body,
|
||||||
address: sight.address,
|
address: sight.address,
|
||||||
};
|
};
|
||||||
@@ -178,7 +129,7 @@ const LeftWidget = observer(
|
|||||||
setIsImageLoaded(false);
|
setIsImageLoaded(false);
|
||||||
console.error(
|
console.error(
|
||||||
"Ошибка загрузки изображения для достопримечательности:",
|
"Ошибка загрузки изображения для достопримечательности:",
|
||||||
selectedSightId
|
selectedSightId,
|
||||||
);
|
);
|
||||||
if (isVisible) {
|
if (isVisible) {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
@@ -208,7 +159,7 @@ const LeftWidget = observer(
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div ref={widgetRef} style={widgetTransformStyle} className="left-widget">
|
<div ref={widgetRef} style={widgetTransformStyle} className="left-widget" lang={selectedLanguage}>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
<div>Загрузка информации...</div>
|
<div>Загрузка информации...</div>
|
||||||
) : error ? (
|
) : error ? (
|
||||||
@@ -234,9 +185,11 @@ const LeftWidget = observer(
|
|||||||
<div className="left-widget-address">
|
<div className="left-widget-address">
|
||||||
{selectedSightData.address}
|
{selectedSightData.address}
|
||||||
</div>
|
</div>
|
||||||
<div ref={textRef} className="left-widget-text">
|
<TouchableLayout ref={layoutRef} className="left-widget-text-scroll">
|
||||||
{selectedSightData.text}
|
<div className="left-widget-text">
|
||||||
</div>
|
<ReactMarkdownComponent value={selectedSightData.text} />
|
||||||
|
</div>
|
||||||
|
</TouchableLayout>
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
) : (isVisible || selectedSightData) && !isLoading ? (
|
) : (isVisible || selectedSightData) && !isLoading ? (
|
||||||
@@ -244,13 +197,13 @@ const LeftWidget = observer(
|
|||||||
{selectedLanguage === "ru"
|
{selectedLanguage === "ru"
|
||||||
? "Выберите достопримечательность для просмотра деталей."
|
? "Выберите достопримечательность для просмотра деталей."
|
||||||
: selectedLanguage === "zh"
|
: selectedLanguage === "zh"
|
||||||
? "选择一个地标来查看详细信息。"
|
? "选择一个地标来查看详细信息。"
|
||||||
: "Select a landmark to view details."}
|
: "Select a landmark to view details."}
|
||||||
</div>
|
</div>
|
||||||
) : null}
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
export default LeftWidget;
|
export default LeftWidget;
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import "../../styles/SideMenu.css";
|
|||||||
import AppealWidget from "../widgets/AppealWidget";
|
import AppealWidget from "../widgets/AppealWidget";
|
||||||
import { useEffect, useState, useCallback, useRef } from "react";
|
import { useEffect, useState, useCallback, useRef } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import gouvermentImage from "../../assets/images/test-image.png";
|
|
||||||
import sideMenuPhoto from "/side-menu-photo.png";
|
import sideMenuPhoto from "/side-menu-photo.png";
|
||||||
import RouteWidget from "../widgets/RouteWidget";
|
import RouteWidget from "../widgets/RouteWidget";
|
||||||
import ContentAPI from "../../api/content/content.api";
|
import ContentAPI from "../../api/content/content.api";
|
||||||
@@ -13,6 +12,7 @@ import StationsList from "./StationsList";
|
|||||||
import LeftWidget from "./LeftWidget";
|
import LeftWidget from "./LeftWidget";
|
||||||
import { apiStore } from "../../api/ApiStore/store";
|
import { apiStore } from "../../api/ApiStore/store";
|
||||||
import { getMediaUrl } from "../../api/apiConfig";
|
import { getMediaUrl } from "../../api/apiConfig";
|
||||||
|
import defaultCrest from "../../assets/images/Герб.png";
|
||||||
|
|
||||||
const SideMenu = observer(({ onMenuToggle }) => {
|
const SideMenu = observer(({ onMenuToggle }) => {
|
||||||
const {
|
const {
|
||||||
@@ -263,15 +263,20 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const isMenuOpenRef = useRef(isMenuOpen);
|
||||||
|
const handleMenuToggleRef = useRef(handleMenuToggle);
|
||||||
|
useEffect(() => { isMenuOpenRef.current = isMenuOpen; }, [isMenuOpen]);
|
||||||
|
useEffect(() => { handleMenuToggleRef.current = handleMenuToggle; });
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// Автоматическое закрытие сайд-меню после 45 секунд бездействия
|
// Автоматическое закрытие сайд-меню после 60 секунд бездействия
|
||||||
let idleSeconds = 0;
|
let idleSeconds = 0;
|
||||||
|
|
||||||
const checkIdle = () => {
|
const checkIdle = () => {
|
||||||
idleSeconds += 1;
|
idleSeconds += 1;
|
||||||
|
|
||||||
if (idleSeconds >= 45 && isMenuOpen) {
|
if (idleSeconds >= 60 && isMenuOpenRef.current) {
|
||||||
handleMenuToggle(false);
|
handleMenuToggleRef.current(false);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -300,7 +305,7 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
window.removeEventListener(event, resetIdle);
|
window.removeEventListener(event, resetIdle);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
}, [isMenuOpen, handleMenuToggle]);
|
}, []);
|
||||||
|
|
||||||
// Закрываем и открываем список достопримечательностей при изменении сортировки
|
// Закрываем и открываем список достопримечательностей при изменении сортировки
|
||||||
const prevSortingByRef = useRef(sortingBy);
|
const prevSortingByRef = useRef(sortingBy);
|
||||||
@@ -357,7 +362,7 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
pointerEvents: "auto",
|
pointerEvents: "auto",
|
||||||
background:
|
background:
|
||||||
isSightsOpen || isStationOpen
|
isSightsOpen || isStationOpen
|
||||||
? `linear-gradient(to bottom right, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0) 100%), rgba(76, 175, 75, 0.4)`
|
? `linear-gradient(to bottom right, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0) 100%), rgba(var(--carrier-left-rgb, 76, 175, 75), 0.4)`
|
||||||
: undefined,
|
: undefined,
|
||||||
backdropFilter:
|
backdropFilter:
|
||||||
isSightsOpen || isStationOpen ? "blur(10px)" : undefined,
|
isSightsOpen || isStationOpen ? "blur(10px)" : undefined,
|
||||||
@@ -369,13 +374,11 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
"background 0.3s ease, backdrop-filter 0.3s ease, box-shadow 0.3s ease",
|
"background 0.3s ease, backdrop-filter 0.3s ease, box-shadow 0.3s ease",
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{designData?.creastPath && (
|
<img
|
||||||
<img
|
className="side-menu-crest"
|
||||||
className="side-menu-crest"
|
src={designData?.creastPath || defaultCrest}
|
||||||
src={designData?.creastPath}
|
alt="Герб"
|
||||||
alt="Герб"
|
/>
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{carrier?.slogan && (
|
{carrier?.slogan && (
|
||||||
<div className="side-menu-label">{carrier.slogan}</div>
|
<div className="side-menu-label">{carrier.slogan}</div>
|
||||||
)}
|
)}
|
||||||
@@ -444,14 +447,16 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
}}
|
}}
|
||||||
className="appeal-button"
|
className="appeal-button"
|
||||||
>
|
>
|
||||||
{selectedLanguage == "ru"
|
{route?.button_text
|
||||||
? "Обращение губернатора"
|
? route.button_text
|
||||||
: selectedLanguage == "zh"
|
: selectedLanguage == "ru"
|
||||||
? "州长致辞"
|
? "Обращение губернатора"
|
||||||
: "Governor's appeal"}
|
: selectedLanguage == "zh"
|
||||||
|
? "州长致辞"
|
||||||
|
: "Governor's appeal"}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
<div className="side-menu-buttons">
|
<div className="side-menu-buttons" style={{ marginTop: route?.governor_appeal > 0 ? '40px' : '260px' }}>
|
||||||
<div
|
<div
|
||||||
onPointerUp={() => {
|
onPointerUp={() => {
|
||||||
if (!isSightsOpen) {
|
if (!isSightsOpen) {
|
||||||
@@ -493,7 +498,7 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
}, 300);
|
}, 300);
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
className={`side-menu-button side-menu-button--sights ${
|
className={`side-menu-button ${
|
||||||
isSightsOpen ? "side-menu-button--active" : ""
|
isSightsOpen ? "side-menu-button--active" : ""
|
||||||
}`}
|
}`}
|
||||||
>
|
>
|
||||||
@@ -502,7 +507,7 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
: selectedLanguage == "zh"
|
: selectedLanguage == "zh"
|
||||||
? "景点"
|
? "景点"
|
||||||
: "Attractions"}
|
: "Attractions"}
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
onPointerUp={() => {
|
onPointerUp={() => {
|
||||||
if (!isStationOpen) {
|
if (!isStationOpen) {
|
||||||
@@ -550,7 +555,7 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
: selectedLanguage == "zh"
|
: selectedLanguage == "zh"
|
||||||
? "车站"
|
? "车站"
|
||||||
: "Stations"}
|
: "Stations"}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="side-menu-tag">
|
<div className="side-menu-tag">
|
||||||
{/* {selectedLanguage == "ru"
|
{/* {selectedLanguage == "ru"
|
||||||
@@ -579,11 +584,17 @@ const SideMenu = observer(({ onMenuToggle }) => {
|
|||||||
<RouteWidget />
|
<RouteWidget />
|
||||||
|
|
||||||
<AppealWidget
|
<AppealWidget
|
||||||
widgetImgPath={gouvermentImage}
|
widgetImgPath={(() => {
|
||||||
|
const m = sightArticles.get(route?.governor_appeal + "_ru")?.media;
|
||||||
|
const mediaId = Array.isArray(m) ? m[0]?.id : m?.id;
|
||||||
|
return mediaId ? getMediaUrl(mediaId) : undefined;
|
||||||
|
})()}
|
||||||
|
isOpen={isWidgetOpen}
|
||||||
style={{
|
style={{
|
||||||
transform: isWidgetOpen ? "translateX(0)" : "translateX(-200%)",
|
transform: isWidgetOpen ? "translateX(0)" : "translateX(-200%)",
|
||||||
transition: "transform 0.5s ease",
|
transition: "transform 0.5s ease",
|
||||||
zIndex: -1,
|
zIndex: -1,
|
||||||
|
pointerEvents: isWidgetOpen ? "auto" : "none",
|
||||||
}}
|
}}
|
||||||
widgetLabel={
|
widgetLabel={
|
||||||
selectedLanguage == "ru"
|
selectedLanguage == "ru"
|
||||||
|
|||||||
@@ -64,7 +64,6 @@ const StationItem = ({
|
|||||||
className="side-menu-sight"
|
className="side-menu-sight"
|
||||||
onPointerDown={(e) => handlePointerDown(e, station.id)}
|
onPointerDown={(e) => handlePointerDown(e, station.id)}
|
||||||
onPointerUp={(e) => handlePointerUp(e, station.id, handleStationClick)}
|
onPointerUp={(e) => handlePointerUp(e, station.id, handleStationClick)}
|
||||||
title={station.name}
|
|
||||||
>
|
>
|
||||||
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
||||||
{station.name}
|
{station.name}
|
||||||
|
|||||||
@@ -84,6 +84,12 @@ const SightItem = ({
|
|||||||
return () => window.removeEventListener("resize", checkWidth);
|
return () => window.removeEventListener("resize", checkWidth);
|
||||||
}, [sightName]);
|
}, [sightName]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (localSelectedSightId !== sight.id) {
|
||||||
|
setIsExpanded(false);
|
||||||
|
}
|
||||||
|
}, [localSelectedSightId, sight.id]);
|
||||||
|
|
||||||
const handleClick = (e) => {
|
const handleClick = (e) => {
|
||||||
const newExpanded = !isExpanded;
|
const newExpanded = !isExpanded;
|
||||||
setIsExpanded(newExpanded);
|
setIsExpanded(newExpanded);
|
||||||
@@ -96,16 +102,19 @@ const SightItem = ({
|
|||||||
const stations = sightStationsCache.get(cacheKey) || [];
|
const stations = sightStationsCache.get(cacheKey) || [];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div
|
||||||
|
className={
|
||||||
|
localSelectedSightId === sight.id
|
||||||
|
? "side-menu-sight-selected-wrapper"
|
||||||
|
: ""
|
||||||
|
}
|
||||||
|
>
|
||||||
<div
|
<div
|
||||||
ref={containerRef}
|
ref={containerRef}
|
||||||
id={`sight-${sight.id}`}
|
id={`sight-${sight.id}`}
|
||||||
onPointerDown={(e) => handlePointerDown(e, sight.id)}
|
onPointerDown={(e) => handlePointerDown(e, sight.id)}
|
||||||
onPointerUp={(e) => handlePointerUp(e, sight.id, handleClick)}
|
onPointerUp={(e) => handlePointerUp(e, sight.id, handleClick)}
|
||||||
className={`side-menu-sight pointer ${
|
className="side-menu-sight pointer"
|
||||||
localSelectedSightId === sight.id ? "selected" : ""
|
|
||||||
}`}
|
|
||||||
title={sightName}
|
|
||||||
>
|
>
|
||||||
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
||||||
{sightName}
|
{sightName}
|
||||||
|
|||||||
@@ -70,7 +70,6 @@ const SightItem = ({
|
|||||||
className={`side-menu-sight pointer ${
|
className={`side-menu-sight pointer ${
|
||||||
localSelectedSightId === sight.id ? "selected" : ""
|
localSelectedSightId === sight.id ? "selected" : ""
|
||||||
}`}
|
}`}
|
||||||
title={sightName}
|
|
||||||
>
|
>
|
||||||
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
||||||
{sightName}
|
{sightName}
|
||||||
|
|||||||
@@ -11,6 +11,42 @@ import { apiStore } from "../../api/ApiStore/store";
|
|||||||
import { useClickDetection } from "../../hooks/useClickDetection";
|
import { useClickDetection } from "../../hooks/useClickDetection";
|
||||||
import { TouchableLayout } from "../TouchableLayout";
|
import { TouchableLayout } from "../TouchableLayout";
|
||||||
|
|
||||||
|
const SightTransferItem = ({ name, style, onPointerUp }) => {
|
||||||
|
const containerRef = useRef(null);
|
||||||
|
const textRef = useRef(null);
|
||||||
|
const [shouldAnimate, setShouldAnimate] = useState(false);
|
||||||
|
|
||||||
|
useLayoutEffect(() => {
|
||||||
|
const checkWidth = () => {
|
||||||
|
if (containerRef.current && textRef.current) {
|
||||||
|
const containerWidth = containerRef.current.offsetWidth;
|
||||||
|
const textWidth = textRef.current.scrollWidth;
|
||||||
|
const shouldAnimateValue = textWidth > containerWidth;
|
||||||
|
setShouldAnimate(shouldAnimateValue);
|
||||||
|
if (shouldAnimateValue) {
|
||||||
|
containerRef.current.style.setProperty("--container-width", `${containerWidth}px`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
checkWidth();
|
||||||
|
window.addEventListener("resize", checkWidth);
|
||||||
|
return () => window.removeEventListener("resize", checkWidth);
|
||||||
|
}, [name]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
ref={containerRef}
|
||||||
|
className="side-menu-sight-transfer pointer"
|
||||||
|
style={style}
|
||||||
|
onPointerUp={onPointerUp}
|
||||||
|
>
|
||||||
|
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
||||||
|
{name}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
const StationItem = ({
|
const StationItem = ({
|
||||||
station,
|
station,
|
||||||
handlePointerDown,
|
handlePointerDown,
|
||||||
@@ -75,7 +111,13 @@ const StationItem = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div
|
||||||
|
className={
|
||||||
|
selectedStationId === station.id
|
||||||
|
? "side-menu-sight-selected-wrapper"
|
||||||
|
: ""
|
||||||
|
}
|
||||||
|
>
|
||||||
<div
|
<div
|
||||||
ref={containerRef}
|
ref={containerRef}
|
||||||
className="side-menu-sight"
|
className="side-menu-sight"
|
||||||
@@ -88,7 +130,6 @@ const StationItem = ({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
title={station.name}
|
|
||||||
>
|
>
|
||||||
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
<span ref={textRef} className={shouldAnimate ? "marquee-text" : ""}>
|
||||||
{station.name}
|
{station.name}
|
||||||
@@ -101,9 +142,9 @@ const StationItem = ({
|
|||||||
>
|
>
|
||||||
{sights.length > 0 ? (
|
{sights.length > 0 ? (
|
||||||
sights.map((sight, index) => (
|
sights.map((sight, index) => (
|
||||||
<div
|
<SightTransferItem
|
||||||
key={sight.id}
|
key={sight.id}
|
||||||
className="side-menu-sight-transfer pointer"
|
name={getSightName(sight)}
|
||||||
style={{
|
style={{
|
||||||
borderBottom:
|
borderBottom:
|
||||||
index < sights.length - 1
|
index < sights.length - 1
|
||||||
@@ -115,19 +156,11 @@ const StationItem = ({
|
|||||||
onPointerUp={(e) => {
|
onPointerUp={(e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
if (onSightClick) {
|
if (onSightClick) {
|
||||||
// Вычисляем позицию элемента для правильного позиционирования левого виджета
|
const elementRect = e.currentTarget.getBoundingClientRect();
|
||||||
const element = e.currentTarget;
|
onSightClick(sight.id, elementRect.top);
|
||||||
const elementRect = element.getBoundingClientRect();
|
|
||||||
|
|
||||||
// Используем позицию элемента относительно viewport (elementRect.top)
|
|
||||||
// чтобы верхняя граница виджета совпадала с верхней границей элемента
|
|
||||||
const elementTop = elementRect.top;
|
|
||||||
onSightClick(sight.id, elementTop);
|
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
/>
|
||||||
{getSightName(sight)}
|
|
||||||
</div>
|
|
||||||
))
|
))
|
||||||
) : (
|
) : (
|
||||||
<div className="side-menu-sight-transfer-empty">
|
<div className="side-menu-sight-transfer-empty">
|
||||||
|
|||||||
@@ -1,13 +1,50 @@
|
|||||||
import '../../styles/AppealWidget.css'
|
import { useRef, useEffect } from "react";
|
||||||
|
import "../../styles/AppealWidget.css";
|
||||||
|
import { TouchableLayout } from "../TouchableLayout";
|
||||||
|
import { ReactMarkdownComponent } from "../ReactMarkdown";
|
||||||
|
|
||||||
function AppealWidget({widgetImgPath, widgetLabel, widgetText, style}) {
|
function AppealWidget({
|
||||||
return (
|
widgetImgPath,
|
||||||
<div style={style} className='dynamic-widget'>
|
widgetLabel,
|
||||||
<img className='dynamic-widget-image' src={widgetImgPath} />
|
widgetText,
|
||||||
<div className='dynamic-widget-label'>{widgetLabel}</div>
|
style,
|
||||||
<div className='dynamic-widget-text'>{widgetText}</div>
|
isOpen,
|
||||||
|
}) {
|
||||||
|
const stopProp = (e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
};
|
||||||
|
const layoutRef = useRef(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isOpen && layoutRef.current) {
|
||||||
|
const scrollable = layoutRef.current.querySelector(".scrollable");
|
||||||
|
if (scrollable) scrollable.scrollTop = 0;
|
||||||
|
}
|
||||||
|
}, [isOpen]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
style={style}
|
||||||
|
className="dynamic-widget"
|
||||||
|
onPointerDown={stopProp}
|
||||||
|
onPointerMove={stopProp}
|
||||||
|
onPointerUp={stopProp}
|
||||||
|
>
|
||||||
|
{widgetImgPath && (
|
||||||
|
<img className="dynamic-widget-image" src={widgetImgPath} />
|
||||||
|
)}
|
||||||
|
<div className="dynamic-widget-label">{widgetLabel}</div>
|
||||||
|
<TouchableLayout
|
||||||
|
ref={layoutRef}
|
||||||
|
className="dynamic-widget-text-scroll"
|
||||||
|
>
|
||||||
|
<div className="dynamic-widget-text">
|
||||||
|
<ReactMarkdownComponent value={widgetText} />
|
||||||
</div>
|
</div>
|
||||||
);
|
</TouchableLayout>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default AppealWidget
|
export default AppealWidget;
|
||||||
|
|||||||
@@ -64,6 +64,13 @@ const RouteWidget = observer(() => {
|
|||||||
}, [context?.endStopId, isLoading, selectedLanguage]);
|
}, [context?.endStopId, isLoading, selectedLanguage]);
|
||||||
|
|
||||||
const shouldAnimate = (text, maxLength) => text?.length > maxLength;
|
const shouldAnimate = (text, maxLength) => text?.length > maxLength;
|
||||||
|
const getNumberSizeClass = (text) => {
|
||||||
|
const length = text?.length || 0;
|
||||||
|
if (length <= 2) return "";
|
||||||
|
if (length === 3) return "route-widget-number--3";
|
||||||
|
if (length === 4) return "route-widget-number--4";
|
||||||
|
return "route-widget-number--5";
|
||||||
|
};
|
||||||
const getLabelSizeClass = (text) => {
|
const getLabelSizeClass = (text) => {
|
||||||
const length = text?.length || 0;
|
const length = text?.length || 0;
|
||||||
|
|
||||||
@@ -77,7 +84,7 @@ const RouteWidget = observer(() => {
|
|||||||
const routeZhSubtitle = `${startStationZh?.name} - ${endStationZh?.name}`;
|
const routeZhSubtitle = `${startStationZh?.name} - ${endStationZh?.name}`;
|
||||||
return (
|
return (
|
||||||
<div className="route-widget">
|
<div className="route-widget">
|
||||||
<div className="route-widget-number">
|
<div className={`route-widget-number ${getNumberSizeClass(route?.route_sys_number || context?.routeNumber)}`}>
|
||||||
{route?.route_sys_number || context?.routeNumber || ""}
|
{route?.route_sys_number || context?.routeNumber || ""}
|
||||||
</div>
|
</div>
|
||||||
<div className="route-widget-content">
|
<div className="route-widget-content">
|
||||||
@@ -85,16 +92,14 @@ const RouteWidget = observer(() => {
|
|||||||
className={`route-widget-label ${
|
className={`route-widget-label ${
|
||||||
shouldAnimate(startStation?.name, 18) ? "marquee" : ""
|
shouldAnimate(startStation?.name, 18) ? "marquee" : ""
|
||||||
} ${getLabelSizeClass(startStation?.name)}`}
|
} ${getLabelSizeClass(startStation?.name)}`}
|
||||||
title={startStation?.name}
|
>
|
||||||
>
|
|
||||||
{startStation?.name}
|
{startStation?.name}
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
className={`route-widget-label ${
|
className={`route-widget-label ${
|
||||||
shouldAnimate(endStation?.name, 18) ? "marquee" : ""
|
shouldAnimate(endStation?.name, 18) ? "marquee" : ""
|
||||||
} ${getLabelSizeClass(endStation?.name)}`}
|
} ${getLabelSizeClass(endStation?.name)}`}
|
||||||
title={endStation?.name}
|
>
|
||||||
>
|
|
||||||
{endStation?.name}
|
{endStation?.name}
|
||||||
</div>
|
</div>
|
||||||
{(selectedLanguage === "en" || selectedLanguage === "ru") && (
|
{(selectedLanguage === "en" || selectedLanguage === "ru") && (
|
||||||
@@ -102,8 +107,7 @@ const RouteWidget = observer(() => {
|
|||||||
className={`route-widget-subtitle ${
|
className={`route-widget-subtitle ${
|
||||||
shouldAnimate(routeEnSubtitle, 50) ? "marquee" : ""
|
shouldAnimate(routeEnSubtitle, 50) ? "marquee" : ""
|
||||||
}`}
|
}`}
|
||||||
title={routeEnSubtitle}
|
>
|
||||||
>
|
|
||||||
{routeEnSubtitle}
|
{routeEnSubtitle}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
@@ -112,8 +116,7 @@ const RouteWidget = observer(() => {
|
|||||||
className={`route-widget-subtitle ${
|
className={`route-widget-subtitle ${
|
||||||
shouldAnimate(routeZhSubtitle, 50) ? "marquee" : ""
|
shouldAnimate(routeZhSubtitle, 50) ? "marquee" : ""
|
||||||
}`}
|
}`}
|
||||||
title={routeZhSubtitle}
|
>
|
||||||
>
|
|
||||||
{routeZhSubtitle}
|
{routeZhSubtitle}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { Canvas, useThree } from "@react-three/fiber";
|
import { Canvas, useThree, useFrame } from "@react-three/fiber";
|
||||||
import { OrbitControls, Stage, useGLTF } from "@react-three/drei";
|
import { OrbitControls, Center, useGLTF } from "@react-three/drei";
|
||||||
import React, { useEffect, Suspense } from "react";
|
import React, { useEffect, useRef, Suspense, useCallback } from "react";
|
||||||
import { BACKGROUND_COLOR } from "../../assets/Constants";
|
import { BACKGROUND_COLOR } from "../../assets/Constants";
|
||||||
import * as THREE from "three";
|
import * as THREE from "three";
|
||||||
import type { OrbitControls as OrbitControlsImpl } from "three-stdlib";
|
import type { OrbitControls as OrbitControlsImpl } from "three-stdlib";
|
||||||
@@ -23,6 +23,7 @@ interface ThreeViewProps {
|
|||||||
const ZOOM_FACTOR = 1.2;
|
const ZOOM_FACTOR = 1.2;
|
||||||
const MIN_DISTANCE = 1;
|
const MIN_DISTANCE = 1;
|
||||||
const MAX_DISTANCE = 100;
|
const MAX_DISTANCE = 100;
|
||||||
|
const CAMERA_FOV = 40;
|
||||||
|
|
||||||
const TouchController = () => {
|
const TouchController = () => {
|
||||||
const { camera, controls, gl } = useThree();
|
const { camera, controls, gl } = useThree();
|
||||||
@@ -197,6 +198,47 @@ const AutoResize = () => {
|
|||||||
return null;
|
return null;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const FitCamera = ({
|
||||||
|
groupRef,
|
||||||
|
onReady,
|
||||||
|
}: {
|
||||||
|
groupRef: React.RefObject<THREE.Group>;
|
||||||
|
onReady: () => void;
|
||||||
|
}) => {
|
||||||
|
const { camera, controls } = useThree();
|
||||||
|
const fitted = useRef(false);
|
||||||
|
|
||||||
|
useFrame(() => {
|
||||||
|
if (fitted.current) return;
|
||||||
|
const group = groupRef.current;
|
||||||
|
if (!group || group.children.length === 0) return;
|
||||||
|
|
||||||
|
const box = new THREE.Box3().setFromObject(group);
|
||||||
|
const sphere = new THREE.Sphere();
|
||||||
|
box.getBoundingSphere(sphere);
|
||||||
|
|
||||||
|
if (sphere.radius === 0) return;
|
||||||
|
|
||||||
|
const fov = THREE.MathUtils.degToRad(CAMERA_FOV);
|
||||||
|
const dist = sphere.radius / Math.sin(fov / 2);
|
||||||
|
|
||||||
|
camera.position.set(0, 0, dist);
|
||||||
|
camera.lookAt(0, 0, 0);
|
||||||
|
camera.updateProjectionMatrix();
|
||||||
|
|
||||||
|
if (controls) {
|
||||||
|
const orbit = controls as unknown as OrbitControlsImpl;
|
||||||
|
orbit.target.set(0, 0, 0);
|
||||||
|
orbit.update();
|
||||||
|
}
|
||||||
|
|
||||||
|
fitted.current = true;
|
||||||
|
onReady();
|
||||||
|
});
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
const Model = ({
|
const Model = ({
|
||||||
fileUrl,
|
fileUrl,
|
||||||
onLoad,
|
onLoad,
|
||||||
@@ -231,24 +273,39 @@ export const ThreeView: React.FC<ThreeViewProps> = ({
|
|||||||
height = "100%",
|
height = "100%",
|
||||||
onLoad,
|
onLoad,
|
||||||
onError,
|
onError,
|
||||||
onAspectRatioCalculated,
|
|
||||||
controlRef,
|
controlRef,
|
||||||
}) => {
|
}) => {
|
||||||
|
const [isReady, setIsReady] = React.useState(false);
|
||||||
|
const groupRef = useRef<THREE.Group>(null!);
|
||||||
|
|
||||||
|
const handleReady = useCallback(() => {
|
||||||
|
setIsReady(true);
|
||||||
|
onLoad?.();
|
||||||
|
}, [onLoad]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div style={{ width, height, position: "relative", overflow: "hidden" }}>
|
<div style={{ width, height, position: "relative", overflow: "hidden" }}>
|
||||||
|
{!isReady && (
|
||||||
|
<div style={{
|
||||||
|
position: "absolute", inset: 0,
|
||||||
|
backgroundColor: `#${BACKGROUND_COLOR.toString(16).padStart(6, "0")}`,
|
||||||
|
zIndex: 1,
|
||||||
|
}} />
|
||||||
|
)}
|
||||||
<Canvas
|
<Canvas
|
||||||
gl={{
|
gl={{
|
||||||
antialias: true,
|
antialias: true,
|
||||||
toneMappingExposure: 1.5,
|
toneMappingExposure: 1.5,
|
||||||
outputColorSpace: THREE.SRGBColorSpace,
|
outputColorSpace: THREE.SRGBColorSpace,
|
||||||
}}
|
}}
|
||||||
camera={{ position: [0, 0, 5], fov: 40 }}
|
camera={{ position: [0, 0, 50], fov: CAMERA_FOV }}
|
||||||
style={{ width: "100%", height: "100%" }}
|
style={{ width: "100%", height: "100%" }}
|
||||||
onError={(e) => onError?.(e.message)}
|
onError={(e: any) => onError?.(e.message)}
|
||||||
>
|
>
|
||||||
<AutoResize />
|
<AutoResize />
|
||||||
<TouchController />
|
<TouchController />
|
||||||
{controlRef && <ZoomController controlRef={controlRef} />}
|
{controlRef && <ZoomController controlRef={controlRef} />}
|
||||||
|
<FitCamera groupRef={groupRef} onReady={handleReady} />
|
||||||
<color attach="background" args={[BACKGROUND_COLOR]} />
|
<color attach="background" args={[BACKGROUND_COLOR]} />
|
||||||
<ambientLight intensity={0.8} />
|
<ambientLight intensity={0.8} />
|
||||||
<directionalLight position={[30, 30, 30]} intensity={1.2} />
|
<directionalLight position={[30, 30, 30]} intensity={1.2} />
|
||||||
@@ -266,23 +323,18 @@ export const ThreeView: React.FC<ThreeViewProps> = ({
|
|||||||
<pointLight position={[0, 30, 0]} intensity={0.6} />
|
<pointLight position={[0, 30, 0]} intensity={0.6} />
|
||||||
|
|
||||||
<Suspense fallback={null}>
|
<Suspense fallback={null}>
|
||||||
<Stage
|
<Center precise>
|
||||||
environment={null}
|
<group ref={groupRef}>
|
||||||
intensity={1}
|
<Model fileUrl={fileUrl} />
|
||||||
contactShadow={false}
|
</group>
|
||||||
shadows={false}
|
</Center>
|
||||||
adjustCamera={true}
|
|
||||||
center={{ precise: true }}
|
|
||||||
>
|
|
||||||
<Model fileUrl={fileUrl} onLoad={onLoad} />
|
|
||||||
</Stage>
|
|
||||||
</Suspense>
|
</Suspense>
|
||||||
|
|
||||||
<OrbitControls
|
<OrbitControls
|
||||||
makeDefault
|
makeDefault
|
||||||
enableZoom={true}
|
enableZoom={true}
|
||||||
enablePan={true}
|
enablePan={true}
|
||||||
target={[50, 50, 50]}
|
target={[0, 0, 0]}
|
||||||
minDistance={1}
|
minDistance={1}
|
||||||
maxDistance={100}
|
maxDistance={100}
|
||||||
enableDamping={true}
|
enableDamping={true}
|
||||||
|
|||||||
@@ -14,9 +14,9 @@
|
|||||||
.side-menu-sights-block {
|
.side-menu-sights-block {
|
||||||
height: calc(60%);
|
height: calc(60%);
|
||||||
overflow-y: scroll;
|
overflow-y: scroll;
|
||||||
margin-left: 20px;
|
margin-left: 0;
|
||||||
margin-top: 8px;
|
margin-top: 8px;
|
||||||
margin-right: 5px;
|
padding-right: 5px;
|
||||||
touch-action: none; /* Отключаем стандартные действия */
|
touch-action: none; /* Отключаем стандартные действия */
|
||||||
overscroll-behavior: contain; /* Предотвращаем прокрутку родительских элементов */
|
overscroll-behavior: contain; /* Предотвращаем прокрутку родительских элементов */
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,33 +52,6 @@ class CameraAnimationStore {
|
|||||||
return t < 0.5 ? 4 * t * t * t : 1 - Math.pow(-2 * t + 2, 3) / 2;
|
return t < 0.5 ? 4 * t * t * t : 1 - Math.pow(-2 * t + 2, 3) / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
private calculateDistance(p1: CameraPosition, p2: CameraPosition): number {
|
|
||||||
return Math.sqrt(Math.pow(p1.x - p2.x, 2) + Math.pow(p1.y - p2.y, 2));
|
|
||||||
}
|
|
||||||
|
|
||||||
private isNearStation(
|
|
||||||
tramPos: CameraPosition,
|
|
||||||
stations: Station[]
|
|
||||||
): { isNear: boolean; distance: number } {
|
|
||||||
if (!stations || stations.length === 0)
|
|
||||||
return { isNear: false, distance: Infinity };
|
|
||||||
const threshold = 300; // Порог в координатах карты
|
|
||||||
let minDistance = Infinity;
|
|
||||||
|
|
||||||
for (const station of stations) {
|
|
||||||
const distance = this.calculateDistance(tramPos, {
|
|
||||||
x: station.longitude,
|
|
||||||
y: station.latitude,
|
|
||||||
});
|
|
||||||
minDistance = Math.min(minDistance, distance);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
isNear: minDistance < threshold,
|
|
||||||
distance: minDistance,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public setUpdateCallback(
|
public setUpdateCallback(
|
||||||
callback: ((pos: CameraPosition, zoom: number) => void) | null
|
callback: ((pos: CameraPosition, zoom: number) => void) | null
|
||||||
) {
|
) {
|
||||||
@@ -140,7 +113,7 @@ class CameraAnimationStore {
|
|||||||
public followTram(
|
public followTram(
|
||||||
tramMapPos: CameraPosition,
|
tramMapPos: CameraPosition,
|
||||||
screenCenter: CameraPosition,
|
screenCenter: CameraPosition,
|
||||||
stations: Station[] = []
|
_stations: Station[] = []
|
||||||
) {
|
) {
|
||||||
// Анимация начинается с текущего зума и плавно переходит к максимальному зуму
|
// Анимация начинается с текущего зума и плавно переходит к максимальному зуму
|
||||||
// для плавного приближения к желтой точке при слежении
|
// для плавного приближения к желтой точке при слежении
|
||||||
|
|||||||
@@ -1,21 +1,45 @@
|
|||||||
import { makeAutoObservable, runInAction } from "mobx";
|
import { makeAutoObservable, runInAction } from "mobx";
|
||||||
|
|
||||||
const COLOR_WHITE = { h: 151, s: 0, l: 100 };
|
|
||||||
const COLOR_GREEN = { h: 151, s: 100, l: 22 };
|
|
||||||
|
|
||||||
const TRANSITION_DURATION = 60000;
|
const TRANSITION_DURATION = 60000;
|
||||||
const TICK_INTERVAL = 100;
|
const TICK_INTERVAL = 100;
|
||||||
const TICK_STEP = TICK_INTERVAL / TRANSITION_DURATION;
|
const TICK_STEP = TICK_INTERVAL / TRANSITION_DURATION;
|
||||||
|
|
||||||
|
function hexToRgb(hex: string): { r: number; g: number; b: number } | null {
|
||||||
|
const clean = hex.trim().replace(/^#/, "");
|
||||||
|
const full = clean.length === 3 ? clean.split("").map((c) => c + c).join("") : clean;
|
||||||
|
if (full.length !== 6) return null;
|
||||||
|
return {
|
||||||
|
r: parseInt(full.slice(0, 2), 16),
|
||||||
|
g: parseInt(full.slice(2, 4), 16),
|
||||||
|
b: parseInt(full.slice(4, 6), 16),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function interpolateRgb(
|
||||||
|
from: { r: number; g: number; b: number },
|
||||||
|
to: { r: number; g: number; b: number },
|
||||||
|
t: number
|
||||||
|
): string {
|
||||||
|
const r = Math.round(from.r + (to.r - from.r) * t);
|
||||||
|
const g = Math.round(from.g + (to.g - from.g) * t);
|
||||||
|
const b = Math.round(from.b + (to.b - from.b) * t);
|
||||||
|
return `rgb(${r}, ${g}, ${b})`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const WHITE = { r: 255, g: 255, b: 255 };
|
||||||
|
const DEFAULT_MAIN = { r: 0, g: 111, b: 58 };
|
||||||
|
|
||||||
interface ColorStore {
|
interface ColorStore {
|
||||||
currentColor: string;
|
currentColor: string;
|
||||||
setCurrentColor: (color: string) => void;
|
setCurrentColor: (color: string) => void;
|
||||||
|
setMainColor: (hex: string) => void;
|
||||||
startColorAnimation: () => void;
|
startColorAnimation: () => void;
|
||||||
stopColorAnimation: () => void;
|
stopColorAnimation: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
class ColorStore implements ColorStore {
|
class ColorStore implements ColorStore {
|
||||||
currentColor: string = "#fff";
|
currentColor: string = "#fff";
|
||||||
|
private mainColor: { r: number; g: number; b: number } = DEFAULT_MAIN;
|
||||||
private progress: number = 0;
|
private progress: number = 0;
|
||||||
private direction: number = 1;
|
private direction: number = 1;
|
||||||
private tickInterval: ReturnType<typeof setInterval> | null = null;
|
private tickInterval: ReturnType<typeof setInterval> | null = null;
|
||||||
@@ -28,12 +52,12 @@ class ColorStore implements ColorStore {
|
|||||||
this.currentColor = color;
|
this.currentColor = color;
|
||||||
};
|
};
|
||||||
|
|
||||||
private interpolateColor(progress: number): string {
|
setMainColor = (hex: string) => {
|
||||||
const h = Math.round(COLOR_WHITE.h + (COLOR_GREEN.h - COLOR_WHITE.h) * progress);
|
const parsed = hexToRgb(hex);
|
||||||
const s = Math.round(COLOR_WHITE.s + (COLOR_GREEN.s - COLOR_WHITE.s) * progress);
|
if (parsed) {
|
||||||
const l = Math.round(COLOR_WHITE.l + (COLOR_GREEN.l - COLOR_WHITE.l) * progress);
|
this.mainColor = parsed;
|
||||||
return `hsl(${h}, ${s}%, ${l}%)`;
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
startColorAnimation = () => {
|
startColorAnimation = () => {
|
||||||
if (this.tickInterval) return;
|
if (this.tickInterval) return;
|
||||||
@@ -50,7 +74,7 @@ class ColorStore implements ColorStore {
|
|||||||
this.direction = 1;
|
this.direction = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.currentColor = this.interpolateColor(this.progress);
|
this.currentColor = interpolateRgb(WHITE, this.mainColor, this.progress);
|
||||||
});
|
});
|
||||||
}, TICK_INTERVAL);
|
}, TICK_INTERVAL);
|
||||||
};
|
};
|
||||||
@@ -68,4 +92,4 @@ class ColorStore implements ColorStore {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const colorStore = new ColorStore();
|
export const colorStore = new ColorStore();
|
||||||
export { ColorStore };
|
export { ColorStore };
|
||||||
|
|||||||
@@ -5,37 +5,124 @@
|
|||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
width: 420px;
|
width: 420px;
|
||||||
|
max-height: calc(100vh - 150px - 98px);
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
background: linear-gradient(
|
background:
|
||||||
|
linear-gradient(
|
||||||
114deg,
|
114deg,
|
||||||
rgba(255, 255, 255, 0) 8.71%,
|
rgba(255, 255, 255, 0) 8.71%,
|
||||||
rgba(255, 255, 255, 0.16) 69.69%
|
rgba(255, 255, 255, 0.16) 69.69%
|
||||||
),
|
),
|
||||||
#006F3A;
|
var(--carrier-left, #006f3a);
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
|
overflow: hidden;
|
||||||
|
touch-action: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.dynamic-widget-image {
|
.dynamic-widget-image {
|
||||||
border-radius-top-left: 10px;
|
padding-top: 2px;
|
||||||
border-radius-top-right: 10px;
|
margin-left: 2px;
|
||||||
padding-top: 4px;
|
margin-right: 2px;
|
||||||
margin-left: 4px;
|
width: 416px;
|
||||||
margin-right: 4px;
|
border-radius: 10px 10px 0 0;
|
||||||
width: 412px;
|
object-fit: cover;
|
||||||
}
|
}
|
||||||
|
|
||||||
.dynamic-widget-label {
|
.dynamic-widget-label {
|
||||||
width: 380px;
|
width: 100%;
|
||||||
margin-top: 29px;
|
padding: 10px 20px;
|
||||||
|
box-sizing: border-box;
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
|
color: #fff;
|
||||||
|
border-bottom: 1px solid var(--Glass-stroke, rgba(255, 255, 255, 0.8));
|
||||||
|
background:
|
||||||
|
linear-gradient(
|
||||||
|
180deg,
|
||||||
|
rgba(255, 255, 255, 0.2) 0%,
|
||||||
|
rgba(255, 255, 255, 0) 100%
|
||||||
|
),
|
||||||
|
rgba(var(--carrier-right-menu-rgb, 179, 165, 152), 0.4);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text-scroll.scrollable-container {
|
||||||
|
flex: 1;
|
||||||
|
min-height: 0;
|
||||||
|
align-self: stretch;
|
||||||
|
margin: 15px 20px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text-scroll .scrollable-viewport {
|
||||||
|
flex: 1;
|
||||||
|
min-height: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text-scroll .scrollable {
|
||||||
|
flex: 1;
|
||||||
|
min-height: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
.dynamic-widget-text {
|
.dynamic-widget-text {
|
||||||
margin-top: 16px;
|
|
||||||
margin-bottom: 25px;
|
|
||||||
width: 380px;
|
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
line-height: 150%;
|
line-height: 135%;
|
||||||
|
padding-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container {
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 135%;
|
||||||
|
font-weight: 300;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container p {
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 135%;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container p:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container h1,
|
||||||
|
.dynamic-widget-text .react-markdown-container h2,
|
||||||
|
.dynamic-widget-text .react-markdown-container h3,
|
||||||
|
.dynamic-widget-text .react-markdown-container h4,
|
||||||
|
.dynamic-widget-text .react-markdown-container h5,
|
||||||
|
.dynamic-widget-text .react-markdown-container h6 {
|
||||||
|
font-size: 18px;
|
||||||
|
margin-top: 10px;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container ul,
|
||||||
|
.dynamic-widget-text .react-markdown-container ol {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
padding-left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container li {
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container blockquote {
|
||||||
|
margin-top: 8px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
padding-left: 12px;
|
||||||
|
border-left: 3px solid rgba(255, 255, 255, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container img {
|
||||||
|
max-width: 100%;
|
||||||
|
border-radius: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamic-widget-text .react-markdown-container a {
|
||||||
|
color: rgba(255, 255, 255, 0.9);
|
||||||
|
text-decoration: underline;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,7 +14,7 @@
|
|||||||
rgba(255, 255, 255, 0) 8.71%,
|
rgba(255, 255, 255, 0) 8.71%,
|
||||||
rgba(255, 255, 255, 0.16) 69.69%
|
rgba(255, 255, 255, 0.16) 69.69%
|
||||||
),
|
),
|
||||||
#006F3A;
|
var(--carrier-left, #006F3A);
|
||||||
will-change: transform, opacity;
|
will-change: transform, opacity;
|
||||||
backface-visibility: hidden;
|
backface-visibility: hidden;
|
||||||
}
|
}
|
||||||
@@ -67,17 +67,78 @@
|
|||||||
line-height: 150%;
|
line-height: 150%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.left-widget-text {
|
.left-widget-text-scroll.scrollable-container {
|
||||||
margin-top: 15px;
|
margin-top: 15px;
|
||||||
|
overflow: hidden;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text-scroll .scrollable-viewport {
|
||||||
|
max-height: 200px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text {
|
||||||
color: #fff;
|
color: #fff;
|
||||||
font-family: "Roboto";
|
font-family: "Roboto";
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
line-height: 135%;
|
line-height: 135%;
|
||||||
max-height: 200px; /* Пример ограничения высоты */
|
padding-right: 3px;
|
||||||
overflow-y: auto;
|
}
|
||||||
touch-action: none;
|
|
||||||
overscroll-behavior: contain;
|
.left-widget-text .react-markdown-container {
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 135%;
|
||||||
|
font-weight: 300;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container p {
|
||||||
|
font-size: 16px;
|
||||||
|
line-height: 135%;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container p:last-child {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container h1,
|
||||||
|
.left-widget-text .react-markdown-container h2,
|
||||||
|
.left-widget-text .react-markdown-container h3,
|
||||||
|
.left-widget-text .react-markdown-container h4,
|
||||||
|
.left-widget-text .react-markdown-container h5,
|
||||||
|
.left-widget-text .react-markdown-container h6 {
|
||||||
|
font-size: 18px;
|
||||||
|
margin-top: 10px;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container ul,
|
||||||
|
.left-widget-text .react-markdown-container ol {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
padding-left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container li {
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container blockquote {
|
||||||
|
margin-top: 8px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
padding-left: 12px;
|
||||||
|
border-left: 3px solid rgba(255, 255, 255, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container img {
|
||||||
|
max-width: 100%;
|
||||||
|
border-radius: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left-widget-text .react-markdown-container a {
|
||||||
|
color: rgba(255, 255, 255, 0.9);
|
||||||
|
text-decoration: underline;
|
||||||
}
|
}
|
||||||
|
|
||||||
.left-widget-image {
|
.left-widget-image {
|
||||||
@@ -93,6 +154,16 @@
|
|||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
padding-bottom: 6px;
|
padding-bottom: 6px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.side-menu-sight-transfer span {
|
||||||
|
display: inline-block;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.side-menu-sight-transfer span.marquee-text {
|
||||||
|
animation: side-menu-marquee 14s linear infinite;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Анимация для списка пересадок */
|
/* Анимация для списка пересадок */
|
||||||
|
|||||||
@@ -1,3 +1,45 @@
|
|||||||
|
@property --fade-top {
|
||||||
|
syntax: "<length>";
|
||||||
|
inherits: false;
|
||||||
|
initial-value: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@property --fade-bottom {
|
||||||
|
syntax: "<length>";
|
||||||
|
inherits: false;
|
||||||
|
initial-value: 45px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes pulse-chevron {
|
||||||
|
0% {
|
||||||
|
transform: rotate(var(--r, 0deg)) translateY(0px) scale(1);
|
||||||
|
}
|
||||||
|
40% {
|
||||||
|
transform: rotate(var(--r, 0deg)) translateY(-4px) scale(1.12);
|
||||||
|
}
|
||||||
|
60% {
|
||||||
|
transform: rotate(var(--r, 0deg)) translateY(-5px) scale(1.14);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: rotate(var(--r, 0deg)) translateY(0px) scale(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.chevron-svg {
|
||||||
|
font-size: 20px;
|
||||||
|
animation: pulse-chevron 1.2s ease-in-out infinite;
|
||||||
|
animation-play-state: paused;
|
||||||
|
will-change: transform;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chevron-svg.is-idle {
|
||||||
|
animation-play-state: running;
|
||||||
|
}
|
||||||
|
|
||||||
|
.chevron-svg.is-open {
|
||||||
|
--r: 180deg;
|
||||||
|
}
|
||||||
|
|
||||||
.right-widget {
|
.right-widget {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
right: 32px;
|
right: 32px;
|
||||||
@@ -17,7 +59,7 @@
|
|||||||
rgba(255, 255, 255, 0) 8.71%,
|
rgba(255, 255, 255, 0) 8.71%,
|
||||||
rgba(255, 255, 255, 0.16) 69.69%
|
rgba(255, 255, 255, 0.16) 69.69%
|
||||||
),
|
),
|
||||||
#006f3a;
|
var(--carrier-right, #806c59);
|
||||||
color: white;
|
color: white;
|
||||||
|
|
||||||
max-height: 68px;
|
max-height: 68px;
|
||||||
@@ -63,7 +105,11 @@
|
|||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
width: 128px;
|
width: 128px;
|
||||||
|
|
||||||
background-color: #0e8953;
|
background-color: color-mix(
|
||||||
|
in srgb,
|
||||||
|
var(--carrier-right, #806c59) 80%,
|
||||||
|
black
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-of-sights-title {
|
.list-of-sights-title {
|
||||||
@@ -90,6 +136,27 @@
|
|||||||
backface-visibility: hidden;
|
backface-visibility: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.list-of-sights-content .scrollable {
|
||||||
|
--fade-top: 0px;
|
||||||
|
--fade-bottom: 45px;
|
||||||
|
mask-image: linear-gradient(
|
||||||
|
to bottom,
|
||||||
|
transparent 0px,
|
||||||
|
black var(--fade-top),
|
||||||
|
black calc(100% - var(--fade-bottom)),
|
||||||
|
transparent 100%
|
||||||
|
);
|
||||||
|
transition: --fade-top 0.5s ease, --fade-bottom 0.5s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-of-sights-content:not(.is-at-top) .scrollable {
|
||||||
|
--fade-top: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-of-sights-content.is-at-bottom .scrollable {
|
||||||
|
--fade-bottom: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
.list-of-sights-grid {
|
.list-of-sights-grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(3, 1fr);
|
grid-template-columns: repeat(3, 1fr);
|
||||||
@@ -103,6 +170,11 @@
|
|||||||
pointer-events: auto;
|
pointer-events: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.list-of-sights-content .custom-scrollbar-track {
|
||||||
|
margin-bottom: 10px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
.sight-component {
|
.sight-component {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@@ -194,7 +266,7 @@
|
|||||||
rgba(255, 255, 255, 0) 8.71%,
|
rgba(255, 255, 255, 0) 8.71%,
|
||||||
rgba(255, 255, 255, 0.16) 69.69%
|
rgba(255, 255, 255, 0.16) 69.69%
|
||||||
),
|
),
|
||||||
#006f3a;
|
var(--carrier-right, #806c59);
|
||||||
max-height: calc(100vh - 128px);
|
max-height: calc(100vh - 128px);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -237,7 +309,7 @@
|
|||||||
rgba(255, 255, 255, 0.22) 0%,
|
rgba(255, 255, 255, 0.22) 0%,
|
||||||
rgba(255, 255, 255, 0.04) 100%
|
rgba(255, 255, 255, 0.04) 100%
|
||||||
),
|
),
|
||||||
rgba(0, 111, 58, 0.72);
|
rgba(var(--carrier-right-rgb, 128, 108, 89), 0.72);
|
||||||
box-shadow: 4px 4px 12px 0px rgba(255, 255, 255, 0.12) inset;
|
box-shadow: 4px 4px 12px 0px rgba(255, 255, 255, 0.12) inset;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
color: white;
|
color: white;
|
||||||
@@ -246,6 +318,16 @@
|
|||||||
min-width: 0;
|
min-width: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.sight-frame-title:not(.intro-title) {
|
||||||
|
background:
|
||||||
|
linear-gradient(
|
||||||
|
180deg,
|
||||||
|
rgba(255, 255, 255, 0.2) 0%,
|
||||||
|
rgba(255, 255, 255, 0) 100%
|
||||||
|
),
|
||||||
|
rgba(var(--carrier-right-menu-rgb, 179, 165, 152), 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
.sight-frame-title p {
|
.sight-frame-title p {
|
||||||
word-wrap: break-word;
|
word-wrap: break-word;
|
||||||
overflow-wrap: break-word;
|
overflow-wrap: break-word;
|
||||||
@@ -304,7 +386,7 @@
|
|||||||
background: linear-gradient(
|
background: linear-gradient(
|
||||||
to right,
|
to right,
|
||||||
transparent 35%,
|
transparent 35%,
|
||||||
#0e8953 50%,
|
color-mix(in srgb, var(--carrier-right, #806c59) 80%, black) 50%,
|
||||||
transparent 65%
|
transparent 65%
|
||||||
);
|
);
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
@@ -326,13 +408,51 @@
|
|||||||
margin-bottom: 0;
|
margin-bottom: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sight-frame-menu {
|
.sight-frame-menu-wrapper {
|
||||||
position: relative;
|
position: relative;
|
||||||
padding: 7px;
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sight-frame-menu-fade {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
width: 120px;
|
||||||
|
z-index: 3;
|
||||||
|
pointer-events: none;
|
||||||
|
transition: opacity 0.4s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sight-frame-menu-fade.left {
|
||||||
|
left: 0;
|
||||||
|
background: linear-gradient(
|
||||||
|
to right,
|
||||||
|
rgba(var(--carrier-right-menu-rgb, 179, 165, 152), 0.95),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
|
border-radius: 0 0 0 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sight-frame-menu-fade.right {
|
||||||
|
right: 0;
|
||||||
|
background: linear-gradient(
|
||||||
|
to left,
|
||||||
|
rgba(var(--carrier-right-menu-rgb, 179, 165, 152), 0.95),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
|
border-radius: 0 0 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sight-frame-menu {
|
||||||
|
z-index: 10000;
|
||||||
|
position: relative;
|
||||||
|
padding: 7px 60px;
|
||||||
|
width: 100%;
|
||||||
|
height: 60px;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-around;
|
justify-content: space-evenly;
|
||||||
border-radius: 0px 0px 10px 10px;
|
border-radius: 0px 0px 10px 10px;
|
||||||
border-top: 1px solid rgba(255, 255, 255, 0.8);
|
border-top: 1px solid rgba(255, 255, 255, 0.8);
|
||||||
background:
|
background:
|
||||||
@@ -341,11 +461,22 @@
|
|||||||
rgba(255, 255, 255, 0.2) 0%,
|
rgba(255, 255, 255, 0.2) 0%,
|
||||||
rgba(255, 255, 255, 0) 100%
|
rgba(255, 255, 255, 0) 100%
|
||||||
),
|
),
|
||||||
rgba(0, 111, 58, 0.4);
|
rgba(var(--carrier-right-menu-rgb, 179, 165, 152), 0.4);
|
||||||
box-shadow: 4px 4px 12px 0px rgba(255, 255, 255, 0.12) inset;
|
box-shadow: 4px 4px 12px 0px rgba(255, 255, 255, 0.12) inset;
|
||||||
backdrop-filter: blur(10px);
|
backdrop-filter: blur(10px);
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
|
overflow-x: auto;
|
||||||
|
overflow-y: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sight-frame-menu::-webkit-scrollbar {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sight-frame-menu {
|
||||||
|
-ms-overflow-style: none;
|
||||||
|
scrollbar-width: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sight-frame-menu-point {
|
.sight-frame-menu-point {
|
||||||
@@ -356,16 +487,17 @@
|
|||||||
font-style: normal;
|
font-style: normal;
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
padding: 8px 12px;
|
padding: 8px 12px;
|
||||||
|
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
|
flex-shrink: 0;
|
||||||
|
border-bottom: 2px solid transparent;
|
||||||
transition:
|
transition:
|
||||||
background-color 0.1s ease,
|
background-color 0.1s ease,
|
||||||
color 0.1s ease;
|
color 0.1s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sight-frame-menu-point.active {
|
.sight-frame-menu-point.active {
|
||||||
border-bottom: 2px solid #fff;
|
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
|
border-bottom-color: #fff;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sight-frame-text-wrapper::-webkit-scrollbar-track {
|
.sight-frame-text-wrapper::-webkit-scrollbar-track {
|
||||||
@@ -519,7 +651,8 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.alphabet {
|
.alphabet {
|
||||||
width: 100px;
|
width: 40px;
|
||||||
|
flex-shrink: 0;
|
||||||
margin-right: 10px;
|
margin-right: 10px;
|
||||||
padding-top: 24px;
|
padding-top: 24px;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -579,8 +712,9 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.alphabet-position {
|
.alphabet-position {
|
||||||
display: inline-flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.transfer-button-container {
|
.transfer-button-container {
|
||||||
@@ -607,14 +741,14 @@
|
|||||||
|
|
||||||
position: absolute;
|
position: absolute;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
border: 1px solid #006f3a;
|
border: 1px solid var(--carrier-main, #006f3a);
|
||||||
background:
|
background:
|
||||||
linear-gradient(
|
linear-gradient(
|
||||||
180deg,
|
180deg,
|
||||||
rgba(255, 255, 255, 0.2) 0%,
|
rgba(255, 255, 255, 0.2) 0%,
|
||||||
rgba(255, 255, 255, 0) 100%
|
rgba(255, 255, 255, 0) 100%
|
||||||
),
|
),
|
||||||
rgba(0, 111, 58, 0.4);
|
rgba(var(--carrier-main-rgb, 0, 111, 58), 0.4);
|
||||||
|
|
||||||
box-shadow: 4px 4px 12px 0px rgba(255, 255, 255, 0.12) inset;
|
box-shadow: 4px 4px 12px 0px rgba(255, 255, 255, 0.12) inset;
|
||||||
backdrop-filter: blur(10px);
|
backdrop-filter: blur(10px);
|
||||||
@@ -747,7 +881,7 @@
|
|||||||
border-radius: 32px;
|
border-radius: 32px;
|
||||||
right: 20px;
|
right: 20px;
|
||||||
bottom: 20px;
|
bottom: 20px;
|
||||||
background: #006f3a;
|
background: var(--carrier-right, #806c59);
|
||||||
z-index: 9999;
|
z-index: 9999;
|
||||||
display: flex;
|
display: flex;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,10 +26,12 @@
|
|||||||
position: fixed;
|
position: fixed;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
box-shadow: 0 0 0 1px rgba(255, 255, 255, 0.3) inset,
|
box-shadow:
|
||||||
|
0 0 0 1px rgba(255, 255, 255, 0.3) inset,
|
||||||
/* Внутренняя рамка */ 4px 4px 12px 0 rgba(255, 255, 255, 0.12) inset; /* Ваш существующий внутренний shadow */
|
/* Внутренняя рамка */ 4px 4px 12px 0 rgba(255, 255, 255, 0.12) inset; /* Ваш существующий внутренний shadow */
|
||||||
padding: 1px; /* Чтобы контент не прилипал к рамке */
|
padding: 1px; /* Чтобы контент не прилипал к рамке */
|
||||||
background: linear-gradient(
|
background:
|
||||||
|
linear-gradient(
|
||||||
to bottom right,
|
to bottom right,
|
||||||
rgba(255, 255, 255, 0.2) 0%,
|
rgba(255, 255, 255, 0.2) 0%,
|
||||||
rgba(255, 255, 255, 0) 100%
|
rgba(255, 255, 255, 0) 100%
|
||||||
@@ -50,7 +52,8 @@
|
|||||||
height: 96px;
|
height: 96px;
|
||||||
background-color: #fcd500;
|
background-color: #fcd500;
|
||||||
color: black;
|
color: black;
|
||||||
border-radius: 10px;
|
border-top-left-radius: 10px;
|
||||||
|
border-bottom-left-radius: 10px;
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -59,6 +62,18 @@
|
|||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.route-widget-number--3 {
|
||||||
|
font-size: 50px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-number--4 {
|
||||||
|
font-size: 38px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-number--5 {
|
||||||
|
font-size: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
.route-widget-content {
|
.route-widget-content {
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
width: 257px;
|
width: 257px;
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
rgba(255, 255, 255, 0.2) 0%,
|
rgba(255, 255, 255, 0.2) 0%,
|
||||||
rgba(255, 255, 255, 0) 100%
|
rgba(255, 255, 255, 0) 100%
|
||||||
),
|
),
|
||||||
#006f3a;
|
var(--carrier-left, #006f3a);
|
||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-label {
|
.side-menu-label {
|
||||||
@@ -35,11 +35,13 @@
|
|||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
margin-top: 120px;
|
margin-top: 120px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
|
width: 220px;
|
||||||
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-buttons {
|
.side-menu-buttons {
|
||||||
width: 220px;
|
width: 220px;
|
||||||
margin-top: 260px;
|
margin-top: 40px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-button {
|
.side-menu-button {
|
||||||
@@ -51,10 +53,6 @@
|
|||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-button--sights {
|
|
||||||
background-color: #fcd500;
|
|
||||||
}
|
|
||||||
|
|
||||||
.side-menu-button--active {
|
.side-menu-button--active {
|
||||||
background-color: #fcd500;
|
background-color: #fcd500;
|
||||||
color: #000;
|
color: #000;
|
||||||
@@ -138,10 +136,10 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
3.33% {
|
3.33% {
|
||||||
fill: rgb(76, 175, 75);
|
fill: var(--carrier-left, rgb(76, 175, 75));
|
||||||
}
|
}
|
||||||
50% {
|
50% {
|
||||||
fill: rgb(76, 175, 75);
|
fill: var(--carrier-left, rgb(76, 175, 75));
|
||||||
}
|
}
|
||||||
53.33% {
|
53.33% {
|
||||||
fill: #ffffff;
|
fill: #ffffff;
|
||||||
@@ -191,7 +189,7 @@
|
|||||||
top: -2px;
|
top: -2px;
|
||||||
width: 100px;
|
width: 100px;
|
||||||
height: 7px;
|
height: 7px;
|
||||||
background-color: #0e8953;
|
background-color: color-mix(in srgb, var(--carrier-left, #006f3a) 80%, black);
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -207,7 +205,7 @@
|
|||||||
rgba(255, 255, 255, 0) 8.71%,
|
rgba(255, 255, 255, 0) 8.71%,
|
||||||
rgba(255, 255, 255, 0.16) 69.69%
|
rgba(255, 255, 255, 0.16) 69.69%
|
||||||
),
|
),
|
||||||
#006f3a;
|
var(--carrier-left, #006f3a);
|
||||||
position: absolute;
|
position: absolute;
|
||||||
width: 288px;
|
width: 288px;
|
||||||
transform: translateY(100%);
|
transform: translateY(100%);
|
||||||
@@ -215,6 +213,8 @@
|
|||||||
transition:
|
transition:
|
||||||
transform 0.3s ease-out,
|
transform 0.3s ease-out,
|
||||||
opacity 0.3s ease-out;
|
opacity 0.3s ease-out;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-sights.slide-in {
|
.side-menu-sights.slide-in {
|
||||||
@@ -227,23 +227,23 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-sights-block {
|
.side-menu-sights-block {
|
||||||
height: calc(100% - 20px);
|
flex: 1;
|
||||||
margin-left: 20px;
|
min-height: 0;
|
||||||
margin-top: 8px;
|
margin-top: 8px;
|
||||||
touch-action: none;
|
touch-action: none;
|
||||||
overscroll-behavior: contain;
|
overscroll-behavior: contain;
|
||||||
width: auto;
|
width: 100%;
|
||||||
max-width: calc(100% - 20px);
|
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-sight {
|
.side-menu-sight {
|
||||||
padding-bottom: 2px;
|
padding-bottom: 2px;
|
||||||
margin-right: 20px;
|
|
||||||
margin-bottom: 6px;
|
margin-bottom: 6px;
|
||||||
margin-top: 6px;
|
margin-top: 6px;
|
||||||
border-bottom: 1px solid #0e8953;
|
border-bottom: 1px solid
|
||||||
|
color-mix(in srgb, var(--carrier-left, #006f3a) 80%, black);
|
||||||
font-family: "Roboto";
|
font-family: "Roboto";
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
font-weight: 300;
|
font-weight: 300;
|
||||||
@@ -253,6 +253,12 @@
|
|||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.side-menu-sight-selected-wrapper {
|
||||||
|
background: rgba(0, 0, 0, 0.2);
|
||||||
|
margin-left: -20px;
|
||||||
|
padding-left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
.side-menu-sight > span {
|
.side-menu-sight > span {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
|
|||||||
@@ -43,6 +43,7 @@
|
|||||||
position: relative;
|
position: relative;
|
||||||
background: rgba(255, 255, 255, 0.2);
|
background: rgba(255, 255, 255, 0.2);
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
|
transition: opacity 0.2s ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
.custom-scrollbar-thumb {
|
.custom-scrollbar-thumb {
|
||||||
@@ -62,11 +63,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-sights-block .scrollable-viewport {
|
.side-menu-sights-block .scrollable-viewport {
|
||||||
height: calc(92%);
|
height: calc(98%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.side-menu-sights-block .scrollable {
|
.side-menu-sights-block .scrollable {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
padding-left: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.list-of-sights-content .scrollable-viewport {
|
.list-of-sights-content .scrollable-viewport {
|
||||||
|
|||||||
@@ -134,6 +134,58 @@ export class PositionAnimator {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Передискретизация пути для обеспечения равномерного расстояния между точками
|
||||||
|
* @param path - массив [lat, lon] или [x, y]
|
||||||
|
* @param segmentLength - желаемое расстояние между точками (в единицах координат)
|
||||||
|
* @returns новый массив точек
|
||||||
|
*/
|
||||||
|
export const resamplePath = <T extends number[]>(path: T[], segmentLength: number): T[] => {
|
||||||
|
if (path.length < 2) return path;
|
||||||
|
|
||||||
|
const newPath: T[] = [path[0]];
|
||||||
|
let leftover = 0;
|
||||||
|
|
||||||
|
for (let i = 0; i < path.length - 1; i++) {
|
||||||
|
const p1 = path[i];
|
||||||
|
const p2 = path[i + 1];
|
||||||
|
const dx = p2[0] - p1[0];
|
||||||
|
const dy = p2[1] - p1[1];
|
||||||
|
const dist = Math.sqrt(dx * dx + dy * dy);
|
||||||
|
|
||||||
|
if (dist === 0) continue;
|
||||||
|
|
||||||
|
let currentDist = segmentLength - leftover;
|
||||||
|
while (currentDist <= dist) {
|
||||||
|
const t = currentDist / dist;
|
||||||
|
const point = new Array(p1.length) as T;
|
||||||
|
for (let j = 0; j < p1.length; j++) {
|
||||||
|
point[j] = p1[j] + (p2[j] - p1[j]) * t;
|
||||||
|
}
|
||||||
|
newPath.push(point);
|
||||||
|
currentDist += segmentLength;
|
||||||
|
}
|
||||||
|
leftover = dist - (currentDist - segmentLength);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Добавляем последнюю точку, если она существенно отличается от последней добавленной
|
||||||
|
const lastP = path[path.length - 1];
|
||||||
|
const lastNewP = newPath[newPath.length - 1];
|
||||||
|
let isDifferent = false;
|
||||||
|
for (let j = 0; j < lastP.length; j++) {
|
||||||
|
if (Math.abs(lastP[j] - lastNewP[j]) > 0.0000001) {
|
||||||
|
isDifferent = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isDifferent) {
|
||||||
|
newPath.push(lastP);
|
||||||
|
}
|
||||||
|
|
||||||
|
return newPath;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Класс для анимации по полярным координатам
|
* Класс для анимации по полярным координатам
|
||||||
* Основано на логике анимации из HTML файла (a.html)
|
* Основано на логике анимации из HTML файла (a.html)
|
||||||
|
|||||||
@@ -17,6 +17,11 @@ const isItemVisible = (item: (typeof NAVIGATION_ITEMS.primary)[number]): boolean
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Пользователь с ролью ТО всегда видит раздел устройств
|
||||||
|
if (item.path === "/devices" && authStore.hasRole("devices_maintenance_rw")) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
const routePermissions = item.path ? ROUTE_REQUIRED_RESOURCES[item.path] ?? [] : [];
|
const routePermissions = item.path ? ROUTE_REQUIRED_RESOURCES[item.path] ?? [] : [];
|
||||||
const canAccessRoute = routePermissions.every((permission) =>
|
const canAccessRoute = routePermissions.every((permission) =>
|
||||||
authStore.canAccess(permission),
|
authStore.canAccess(permission),
|
||||||
|
|||||||
@@ -1,12 +1,17 @@
|
|||||||
import React from "react";
|
import React, { useEffect } from "react";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { ArrowLeft } from "lucide-react";
|
import { ArrowLeft } from "lucide-react";
|
||||||
import { LanguageSwitcher } from "@widgets";
|
import { LanguageSwitcher } from "@widgets";
|
||||||
import { articlesStore } from "@shared";
|
import { articlesStore, selectedCityStore } from "@shared";
|
||||||
|
|
||||||
const ArticleCreatePage: React.FC = () => {
|
const ArticleCreatePage: React.FC = () => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const { articleData } = articlesStore;
|
const { articleData } = articlesStore;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import React, { useEffect } from "react";
|
|||||||
import { useNavigate, useParams } from "react-router-dom";
|
import { useNavigate, useParams } from "react-router-dom";
|
||||||
import { ArrowLeft } from "lucide-react";
|
import { ArrowLeft } from "lucide-react";
|
||||||
import { LanguageSwitcher } from "@widgets";
|
import { LanguageSwitcher } from "@widgets";
|
||||||
import { articlesStore, languageStore } from "@shared";
|
import { articlesStore, languageStore, selectedCityStore } from "@shared";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
|
|
||||||
const ArticleEditPage: React.FC = observer(() => {
|
const ArticleEditPage: React.FC = observer(() => {
|
||||||
@@ -11,6 +11,11 @@ const ArticleEditPage: React.FC = observer(() => {
|
|||||||
|
|
||||||
const { articleData, getArticle } = articlesStore;
|
const { articleData, getArticle } = articlesStore;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// Устанавливаем русский язык при загрузке страницы
|
// Устанавливаем русский язык при загрузке страницы
|
||||||
languageStore.setLanguage("ru");
|
languageStore.setLanguage("ru");
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
||||||
import { ruRU } from "@mui/x-data-grid/locales";
|
import { ruRU } from "@mui/x-data-grid/locales";
|
||||||
import { authStore, articlesStore, languageStore, SearchInput } from "@shared";
|
import { authStore, articlesStore, languageStore, SearchInput, selectedCityStore } from "@shared";
|
||||||
import { useEffect, useState, useMemo } from "react";
|
import { useEffect, useState, useMemo } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { Trash2, Eye, Minus } from "lucide-react";
|
import { Trash2, Eye, Minus } from "lucide-react";
|
||||||
@@ -31,7 +31,7 @@ export const ArticleListPage = observer(() => {
|
|||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
};
|
};
|
||||||
fetchArticles();
|
fetchArticles();
|
||||||
}, [language]);
|
}, [language, selectedCityStore.cityVersion]);
|
||||||
|
|
||||||
const columns: GridColDef[] = [
|
const columns: GridColDef[] = [
|
||||||
{
|
{
|
||||||
@@ -55,11 +55,12 @@ export const ArticleListPage = observer(() => {
|
|||||||
sortable: false,
|
sortable: false,
|
||||||
renderCell: (params: GridRenderCellParams) => (
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
<button onClick={() => navigate(`/article/${params.row.id}`)}>
|
<button title="Просмотр" onClick={() => navigate(`/article/${params.row.id}`)}>
|
||||||
<Eye size={20} className="text-green-500" />
|
<Eye size={20} className="text-green-500" />
|
||||||
</button>
|
</button>
|
||||||
{canWriteArticles && (
|
{canWriteArticles && (
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -75,14 +76,16 @@ export const ArticleListPage = observer(() => {
|
|||||||
|
|
||||||
const rows = useMemo(() => {
|
const rows = useMemo(() => {
|
||||||
const query = searchQuery.trim().toLowerCase();
|
const query = searchQuery.trim().toLowerCase();
|
||||||
|
const cityId = selectedCityStore.selectedCityId;
|
||||||
return articleList[language].data
|
return articleList[language].data
|
||||||
.filter((article) => !query || (article.heading ?? "").toLowerCase().includes(query))
|
.filter((article) => !query || (article.heading ?? "").toLowerCase().includes(query))
|
||||||
|
.filter((article) => !cityId || article.city_id === cityId)
|
||||||
.map((article) => ({
|
.map((article) => ({
|
||||||
id: article.id,
|
id: article.id,
|
||||||
heading: article.heading,
|
heading: article.heading,
|
||||||
body: article.body,
|
body: article.body,
|
||||||
}));
|
}));
|
||||||
}, [articleList[language].data, searchQuery]);
|
}, [articleList[language].data, searchQuery, selectedCityStore.selectedCityId]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -105,7 +108,9 @@ export const ArticleListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<div className="w-full">
|
<div className="w-full">
|
||||||
<DataGrid
|
<DataGrid
|
||||||
@@ -113,6 +118,7 @@ export const ArticleListPage = observer(() => {
|
|||||||
columns={columns}
|
columns={columns}
|
||||||
checkboxSelection={canWriteArticles}
|
checkboxSelection={canWriteArticles}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ import {
|
|||||||
languageStore,
|
languageStore,
|
||||||
isMediaIdEmpty,
|
isMediaIdEmpty,
|
||||||
useSelectedCity,
|
useSelectedCity,
|
||||||
|
selectedCityStore,
|
||||||
SelectMediaDialog,
|
SelectMediaDialog,
|
||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
PreviewMediaDialog,
|
PreviewMediaDialog,
|
||||||
@@ -27,6 +28,56 @@ import {
|
|||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { ImageUploadCard, LanguageSwitcher } from "@widgets";
|
import { ImageUploadCard, LanguageSwitcher } from "@widgets";
|
||||||
|
|
||||||
|
type ColorFields = { main_color: string; left_color: string; right_color: string };
|
||||||
|
|
||||||
|
const colorFields = (data: ColorFields) => ({
|
||||||
|
main_color: data.main_color,
|
||||||
|
left_color: data.left_color,
|
||||||
|
right_color: data.right_color,
|
||||||
|
});
|
||||||
|
|
||||||
|
const ColorPickerField = ({
|
||||||
|
label,
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
}: {
|
||||||
|
label: string;
|
||||||
|
value: string;
|
||||||
|
onChange: (val: string) => void;
|
||||||
|
}) => (
|
||||||
|
<div className="flex items-center gap-3 w-full">
|
||||||
|
<div
|
||||||
|
className="w-10 h-10 rounded border border-gray-300 flex-shrink-0 cursor-pointer overflow-hidden relative"
|
||||||
|
style={{ backgroundColor: value || "#ffffff" }}
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
type="color"
|
||||||
|
value={value || "#ffffff"}
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
className="absolute inset-0 w-full h-full opacity-0 cursor-pointer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<TextField
|
||||||
|
fullWidth
|
||||||
|
label={label}
|
||||||
|
value={value}
|
||||||
|
placeholder="#000000"
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
InputProps={{
|
||||||
|
endAdornment: value ? (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => onChange("")}
|
||||||
|
className="text-gray-400 hover:text-gray-600 text-xs px-1"
|
||||||
|
>
|
||||||
|
✕
|
||||||
|
</button>
|
||||||
|
) : undefined,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
export const CarrierCreatePage = observer(() => {
|
export const CarrierCreatePage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const { createCarrierData, setCreateCarrierData } = carrierStore;
|
const { createCarrierData, setCreateCarrierData } = carrierStore;
|
||||||
@@ -43,6 +94,11 @@ export const CarrierCreatePage = observer(() => {
|
|||||||
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
||||||
>(null);
|
>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const fetchCities = async () => {
|
const fetchCities = async () => {
|
||||||
if (!authStore.me) {
|
if (!authStore.me) {
|
||||||
@@ -220,6 +276,69 @@ export const CarrierCreatePage = observer(() => {
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<div className="w-full flex flex-col gap-6">
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<ColorPickerField
|
||||||
|
label="Основной цвет"
|
||||||
|
value={createCarrierData.main_color}
|
||||||
|
onChange={(val) =>
|
||||||
|
setCreateCarrierData(
|
||||||
|
createCarrierData[language].full_name,
|
||||||
|
createCarrierData[language].short_name,
|
||||||
|
createCarrierData.city_id,
|
||||||
|
createCarrierData[language].slogan,
|
||||||
|
selectedMediaId || "",
|
||||||
|
language,
|
||||||
|
{ ...colorFields(createCarrierData), main_color: val }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-gray-500 pl-1">
|
||||||
|
Используется в: виджет обращений, значки на карте, скопление достопримечательностей на карте, информационный виджет
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<ColorPickerField
|
||||||
|
label="Левый цвет"
|
||||||
|
value={createCarrierData.left_color}
|
||||||
|
onChange={(val) =>
|
||||||
|
setCreateCarrierData(
|
||||||
|
createCarrierData[language].full_name,
|
||||||
|
createCarrierData[language].short_name,
|
||||||
|
createCarrierData.city_id,
|
||||||
|
createCarrierData[language].slogan,
|
||||||
|
selectedMediaId || "",
|
||||||
|
language,
|
||||||
|
{ ...colorFields(createCarrierData), left_color: val }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-gray-500 pl-1">
|
||||||
|
Используется в: боковое меню, левый виджет достопримечательности
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<ColorPickerField
|
||||||
|
label="Правый цвет"
|
||||||
|
value={createCarrierData.right_color}
|
||||||
|
onChange={(val) =>
|
||||||
|
setCreateCarrierData(
|
||||||
|
createCarrierData[language].full_name,
|
||||||
|
createCarrierData[language].short_name,
|
||||||
|
createCarrierData.city_id,
|
||||||
|
createCarrierData[language].slogan,
|
||||||
|
selectedMediaId || "",
|
||||||
|
language,
|
||||||
|
{ ...colorFields(createCarrierData), right_color: val }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-gray-500 pl-1">
|
||||||
|
Используется в: список достопримечательностей, страница достопримечательности
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
||||||
<ImageUploadCard
|
<ImageUploadCard
|
||||||
title="Логотип перевозчика"
|
title="Логотип перевозчика"
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import {
|
|||||||
languageStore,
|
languageStore,
|
||||||
isMediaIdEmpty,
|
isMediaIdEmpty,
|
||||||
LoadingSpinner,
|
LoadingSpinner,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { ImageUploadCard, LanguageSwitcher, DeleteModal } from "@widgets";
|
import { ImageUploadCard, LanguageSwitcher, DeleteModal } from "@widgets";
|
||||||
@@ -30,6 +31,60 @@ import {
|
|||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
|
|
||||||
|
type ColorFields = {
|
||||||
|
main_color: string;
|
||||||
|
left_color: string;
|
||||||
|
right_color: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const colorFields = (data: ColorFields) => ({
|
||||||
|
main_color: data.main_color,
|
||||||
|
left_color: data.left_color,
|
||||||
|
right_color: data.right_color,
|
||||||
|
});
|
||||||
|
|
||||||
|
const ColorPickerField = ({
|
||||||
|
label,
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
}: {
|
||||||
|
label: string;
|
||||||
|
value: string;
|
||||||
|
onChange: (val: string) => void;
|
||||||
|
}) => (
|
||||||
|
<div className="flex items-center gap-3 w-full">
|
||||||
|
<div
|
||||||
|
className="w-10 h-10 rounded border border-gray-300 flex-shrink-0 cursor-pointer overflow-hidden relative"
|
||||||
|
style={{ backgroundColor: value || "#ffffff" }}
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
type="color"
|
||||||
|
value={value || "#ffffff"}
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
className="absolute inset-0 w-full h-full opacity-0 cursor-pointer"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<TextField
|
||||||
|
fullWidth
|
||||||
|
label={label}
|
||||||
|
value={value}
|
||||||
|
placeholder="#000000"
|
||||||
|
onChange={(e) => onChange(e.target.value)}
|
||||||
|
InputProps={{
|
||||||
|
endAdornment: value ? (
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={() => onChange("")}
|
||||||
|
className="text-gray-400 hover:text-gray-600 text-xs px-1"
|
||||||
|
>
|
||||||
|
✕
|
||||||
|
</button>
|
||||||
|
) : undefined,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
export const CarrierEditPage = observer(() => {
|
export const CarrierEditPage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const { id } = useParams();
|
const { id } = useParams();
|
||||||
@@ -49,6 +104,11 @@ export const CarrierEditPage = observer(() => {
|
|||||||
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
||||||
>(null);
|
>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
(async () => {
|
(async () => {
|
||||||
if (!id) {
|
if (!id) {
|
||||||
@@ -68,13 +128,19 @@ export const CarrierEditPage = observer(() => {
|
|||||||
const carrierData = await getCarrier(Number(id));
|
const carrierData = await getCarrier(Number(id));
|
||||||
|
|
||||||
if (carrierData) {
|
if (carrierData) {
|
||||||
|
const colors = {
|
||||||
|
main_color: carrierData.ru?.main_color || "",
|
||||||
|
left_color: carrierData.ru?.left_color || "",
|
||||||
|
right_color: carrierData.ru?.right_color || "",
|
||||||
|
};
|
||||||
setEditCarrierData(
|
setEditCarrierData(
|
||||||
carrierData.ru?.full_name || "",
|
carrierData.ru?.full_name || "",
|
||||||
carrierData.ru?.short_name || "",
|
carrierData.ru?.short_name || "",
|
||||||
carrierData.ru?.city_id || 0,
|
carrierData.ru?.city_id || 0,
|
||||||
carrierData.ru?.slogan || "",
|
carrierData.ru?.slogan || "",
|
||||||
carrierData.ru?.logo || "",
|
carrierData.ru?.logo || "",
|
||||||
"ru"
|
"ru",
|
||||||
|
colors,
|
||||||
);
|
);
|
||||||
setEditCarrierData(
|
setEditCarrierData(
|
||||||
carrierData.en?.full_name || "",
|
carrierData.en?.full_name || "",
|
||||||
@@ -82,7 +148,7 @@ export const CarrierEditPage = observer(() => {
|
|||||||
carrierData.en?.city_id || 0,
|
carrierData.en?.city_id || 0,
|
||||||
carrierData.en?.slogan || "",
|
carrierData.en?.slogan || "",
|
||||||
carrierData.en?.logo || "",
|
carrierData.en?.logo || "",
|
||||||
"en"
|
"en",
|
||||||
);
|
);
|
||||||
setEditCarrierData(
|
setEditCarrierData(
|
||||||
carrierData.zh?.full_name || "",
|
carrierData.zh?.full_name || "",
|
||||||
@@ -90,7 +156,7 @@ export const CarrierEditPage = observer(() => {
|
|||||||
carrierData.zh?.city_id || 0,
|
carrierData.zh?.city_id || 0,
|
||||||
carrierData.zh?.slogan || "",
|
carrierData.zh?.slogan || "",
|
||||||
carrierData.zh?.logo || "",
|
carrierData.zh?.logo || "",
|
||||||
"zh"
|
"zh",
|
||||||
);
|
);
|
||||||
setInitialCityName(carrierData.ru?.city || "");
|
setInitialCityName(carrierData.ru?.city || "");
|
||||||
}
|
}
|
||||||
@@ -129,7 +195,7 @@ export const CarrierEditPage = observer(() => {
|
|||||||
editCarrierData.city_id,
|
editCarrierData.city_id,
|
||||||
editCarrierData[language].slogan,
|
editCarrierData[language].slogan,
|
||||||
media.id,
|
media.id,
|
||||||
language
|
language,
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -211,7 +277,7 @@ export const CarrierEditPage = observer(() => {
|
|||||||
Number(e.target.value),
|
Number(e.target.value),
|
||||||
editCarrierData[language].slogan,
|
editCarrierData[language].slogan,
|
||||||
editCarrierData.logo,
|
editCarrierData.logo,
|
||||||
language
|
language,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -235,7 +301,7 @@ export const CarrierEditPage = observer(() => {
|
|||||||
editCarrierData.city_id,
|
editCarrierData.city_id,
|
||||||
editCarrierData[language].slogan,
|
editCarrierData[language].slogan,
|
||||||
editCarrierData.logo,
|
editCarrierData.logo,
|
||||||
language
|
language,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -252,7 +318,7 @@ export const CarrierEditPage = observer(() => {
|
|||||||
editCarrierData.city_id,
|
editCarrierData.city_id,
|
||||||
editCarrierData[language].slogan,
|
editCarrierData[language].slogan,
|
||||||
editCarrierData.logo,
|
editCarrierData.logo,
|
||||||
language
|
language,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -268,11 +334,77 @@ export const CarrierEditPage = observer(() => {
|
|||||||
editCarrierData.city_id,
|
editCarrierData.city_id,
|
||||||
e.target.value,
|
e.target.value,
|
||||||
editCarrierData.logo,
|
editCarrierData.logo,
|
||||||
language
|
language,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<div className="w-full flex flex-col gap-6">
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<ColorPickerField
|
||||||
|
label="Основной цвет"
|
||||||
|
value={editCarrierData.main_color}
|
||||||
|
onChange={(val) =>
|
||||||
|
setEditCarrierData(
|
||||||
|
editCarrierData[language].full_name,
|
||||||
|
editCarrierData[language].short_name,
|
||||||
|
editCarrierData.city_id,
|
||||||
|
editCarrierData[language].slogan,
|
||||||
|
editCarrierData.logo,
|
||||||
|
language,
|
||||||
|
{ ...colorFields(editCarrierData), main_color: val },
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-gray-500 pl-1">
|
||||||
|
Используется в: значки на карте, скопление достопримечательностей
|
||||||
|
на карте, информационный виджет
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<ColorPickerField
|
||||||
|
label="Левый цвет"
|
||||||
|
value={editCarrierData.left_color}
|
||||||
|
onChange={(val) =>
|
||||||
|
setEditCarrierData(
|
||||||
|
editCarrierData[language].full_name,
|
||||||
|
editCarrierData[language].short_name,
|
||||||
|
editCarrierData.city_id,
|
||||||
|
editCarrierData[language].slogan,
|
||||||
|
editCarrierData.logo,
|
||||||
|
language,
|
||||||
|
{ ...colorFields(editCarrierData), left_color: val },
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-gray-500 pl-1">
|
||||||
|
Используется в: виджет обращений, боковое меню (фон, список
|
||||||
|
остановок), левый виджет достопримечательности
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<ColorPickerField
|
||||||
|
label="Правый цвет"
|
||||||
|
value={editCarrierData.right_color}
|
||||||
|
onChange={(val) =>
|
||||||
|
setEditCarrierData(
|
||||||
|
editCarrierData[language].full_name,
|
||||||
|
editCarrierData[language].short_name,
|
||||||
|
editCarrierData.city_id,
|
||||||
|
editCarrierData[language].slogan,
|
||||||
|
editCarrierData.logo,
|
||||||
|
language,
|
||||||
|
{ ...colorFields(editCarrierData), right_color: val },
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<p className="text-xs text-gray-500 pl-1">
|
||||||
|
Используется в: список достопримечательностей (фон, карточки),
|
||||||
|
правый виджет достопримечательности
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
||||||
<ImageUploadCard
|
<ImageUploadCard
|
||||||
title="Логотип перевозчика"
|
title="Логотип перевозчика"
|
||||||
@@ -346,7 +478,7 @@ export const CarrierEditPage = observer(() => {
|
|||||||
editCarrierData.city_id,
|
editCarrierData.city_id,
|
||||||
editCarrierData[language].slogan,
|
editCarrierData[language].slogan,
|
||||||
"",
|
"",
|
||||||
language
|
language,
|
||||||
);
|
);
|
||||||
setIsDeleteLogoModalOpen(false);
|
setIsDeleteLogoModalOpen(false);
|
||||||
}}
|
}}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
||||||
import { ruRU } from "@mui/x-data-grid/locales";
|
import { ruRU } from "@mui/x-data-grid/locales";
|
||||||
import { authStore, carrierStore, cityStore, languageStore, SearchInput } from "@shared";
|
import { authStore, carrierStore, cityStore, languageStore, selectedCityStore, SearchInput } from "@shared";
|
||||||
import { useEffect, useState, useMemo } from "react";
|
import { useEffect, useState, useMemo } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { Pencil, Trash2, Minus } from "lucide-react";
|
import { Pencil, Trash2, Minus } from "lucide-react";
|
||||||
@@ -39,7 +39,7 @@ export const CarrierListPage = observer(() => {
|
|||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
};
|
};
|
||||||
fetchData();
|
fetchData();
|
||||||
}, [language]);
|
}, [language, selectedCityStore.cityVersion]);
|
||||||
|
|
||||||
const columns: GridColDef[] = [
|
const columns: GridColDef[] = [
|
||||||
{
|
{
|
||||||
@@ -98,10 +98,11 @@ export const CarrierListPage = observer(() => {
|
|||||||
sortable: false,
|
sortable: false,
|
||||||
renderCell: (params: GridRenderCellParams) => (
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
<button onClick={() => navigate(`/carrier/${params.row.id}/edit`)}>
|
<button title="Редактировать" onClick={() => navigate(`/carrier/${params.row.id}/edit`)}>
|
||||||
<Pencil size={20} className="text-blue-500" />
|
<Pencil size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -161,13 +162,21 @@ export const CarrierListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={rows}
|
rows={rows}
|
||||||
columns={columns}
|
columns={columns}
|
||||||
checkboxSelection={canWriteCarriers}
|
checkboxSelection={canWriteCarriers}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
|
onRowDoubleClick={(params) => {
|
||||||
|
if (canWriteCarriers) {
|
||||||
|
navigate(`/carrier/${params.id}/edit`);
|
||||||
|
}
|
||||||
|
}}
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
|
|||||||
@@ -17,18 +17,27 @@ import {
|
|||||||
countryStore,
|
countryStore,
|
||||||
languageStore,
|
languageStore,
|
||||||
mediaStore,
|
mediaStore,
|
||||||
|
snapshotStore,
|
||||||
isMediaIdEmpty,
|
isMediaIdEmpty,
|
||||||
SelectMediaDialog,
|
SelectMediaDialog,
|
||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
PreviewMediaDialog,
|
PreviewMediaDialog,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { LanguageSwitcher, ImageUploadCard } from "@widgets";
|
import { LanguageSwitcher, ImageUploadCard } from "@widgets";
|
||||||
|
|
||||||
export const CityCreatePage = observer(() => {
|
export const CityCreatePage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
const { createCityData, setCreateCityData } = cityStore;
|
const { createCityData, setCreateCityData, setCreateCityWeatherCode } =
|
||||||
|
cityStore;
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [isSelectMediaOpen, setIsSelectMediaOpen] = useState(false);
|
const [isSelectMediaOpen, setIsSelectMediaOpen] = useState(false);
|
||||||
const [isUploadMediaOpen, setIsUploadMediaOpen] = useState(false);
|
const [isUploadMediaOpen, setIsUploadMediaOpen] = useState(false);
|
||||||
@@ -52,7 +61,13 @@ export const CityCreatePage = observer(() => {
|
|||||||
const handleCreate = async () => {
|
const handleCreate = async () => {
|
||||||
try {
|
try {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
const ruCityName = createCityData.ru.name.trim();
|
||||||
await cityStore.createCity();
|
await cityStore.createCity();
|
||||||
|
try {
|
||||||
|
await snapshotStore.createEmptySnapshot(`${ruCityName}_Пустой_Экспорт`);
|
||||||
|
} catch (e) {
|
||||||
|
console.warn("Failed to create empty snapshot for city:", e);
|
||||||
|
}
|
||||||
toast.success("Город успешно создан");
|
toast.success("Город успешно создан");
|
||||||
navigate("/city");
|
navigate("/city");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -72,7 +87,7 @@ export const CityCreatePage = observer(() => {
|
|||||||
createCityData[language].name,
|
createCityData[language].name,
|
||||||
createCityData.country_code,
|
createCityData.country_code,
|
||||||
media.id,
|
media.id,
|
||||||
language
|
language,
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -111,7 +126,7 @@ export const CityCreatePage = observer(() => {
|
|||||||
e.target.value,
|
e.target.value,
|
||||||
createCityData.country_code,
|
createCityData.country_code,
|
||||||
createCityData.arms,
|
createCityData.arms,
|
||||||
language
|
language,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
@@ -127,7 +142,7 @@ export const CityCreatePage = observer(() => {
|
|||||||
createCityData[language].name,
|
createCityData[language].name,
|
||||||
e.target.value,
|
e.target.value,
|
||||||
createCityData.arms,
|
createCityData.arms,
|
||||||
language
|
language,
|
||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@@ -139,6 +154,14 @@ export const CityCreatePage = observer(() => {
|
|||||||
</Select>
|
</Select>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
|
||||||
|
<TextField
|
||||||
|
fullWidth
|
||||||
|
label="Код города для погоды"
|
||||||
|
value={createCityData.weather_city_code ?? 0}
|
||||||
|
onChange={(e) => setCreateCityWeatherCode(Number(e.target.value))}
|
||||||
|
helperText="ID города брать с ресурса openweathermap.org"
|
||||||
|
/>
|
||||||
|
|
||||||
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
||||||
<ImageUploadCard
|
<ImageUploadCard
|
||||||
title="Герб города"
|
title="Герб города"
|
||||||
@@ -153,7 +176,7 @@ export const CityCreatePage = observer(() => {
|
|||||||
createCityData[language].name,
|
createCityData[language].name,
|
||||||
createCityData.country_code,
|
createCityData.country_code,
|
||||||
"",
|
"",
|
||||||
language
|
language,
|
||||||
);
|
);
|
||||||
setActiveMenuType(null);
|
setActiveMenuType(null);
|
||||||
}}
|
}}
|
||||||
|
|||||||
@@ -23,12 +23,19 @@ import {
|
|||||||
SelectMediaDialog,
|
SelectMediaDialog,
|
||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
PreviewMediaDialog,
|
PreviewMediaDialog,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { LanguageSwitcher, ImageUploadCard } from "@widgets";
|
import { LanguageSwitcher, ImageUploadCard } from "@widgets";
|
||||||
|
|
||||||
export const CityEditPage = observer(() => {
|
export const CityEditPage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [isLoadingData, setIsLoadingData] = useState(true);
|
const [isLoadingData, setIsLoadingData] = useState(true);
|
||||||
const [isSelectMediaOpen, setIsSelectMediaOpen] = useState(false);
|
const [isSelectMediaOpen, setIsSelectMediaOpen] = useState(false);
|
||||||
@@ -40,7 +47,13 @@ export const CityEditPage = observer(() => {
|
|||||||
>(null);
|
>(null);
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
const { id } = useParams();
|
const { id } = useParams();
|
||||||
const { editCityData, editCity, getCity, setEditCityData } = cityStore;
|
const {
|
||||||
|
editCityData,
|
||||||
|
editCity,
|
||||||
|
getCity,
|
||||||
|
setEditCityData,
|
||||||
|
setEditCityWeatherCode,
|
||||||
|
} = cityStore;
|
||||||
const { getCountries } = countryStore;
|
const { getCountries } = countryStore;
|
||||||
const { getMedia, getOneMedia } = mediaStore;
|
const { getMedia, getOneMedia } = mediaStore;
|
||||||
|
|
||||||
@@ -74,6 +87,7 @@ export const CityEditPage = observer(() => {
|
|||||||
setEditCityData(ruData.name, ruData.country_code, ruData.arms, "ru");
|
setEditCityData(ruData.name, ruData.country_code, ruData.arms, "ru");
|
||||||
setEditCityData(enData.name, enData.country_code, enData.arms, "en");
|
setEditCityData(enData.name, enData.country_code, enData.arms, "en");
|
||||||
setEditCityData(zhData.name, zhData.country_code, zhData.arms, "zh");
|
setEditCityData(zhData.name, zhData.country_code, zhData.arms, "zh");
|
||||||
|
setEditCityWeatherCode(ruData.weather_city_code ?? 0);
|
||||||
|
|
||||||
await getOneMedia(ruData.arms as string);
|
await getOneMedia(ruData.arms as string);
|
||||||
|
|
||||||
@@ -107,7 +121,7 @@ export const CityEditPage = observer(() => {
|
|||||||
: null;
|
: null;
|
||||||
const effectiveArmsUrl = isMediaIdEmpty(editCityData.arms)
|
const effectiveArmsUrl = isMediaIdEmpty(editCityData.arms)
|
||||||
? null
|
? null
|
||||||
: selectedMedia?.id ?? editCityData.arms;
|
: (selectedMedia?.id ?? editCityData.arms);
|
||||||
|
|
||||||
if (isLoadingData) {
|
if (isLoadingData) {
|
||||||
return (
|
return (
|
||||||
@@ -179,6 +193,14 @@ export const CityEditPage = observer(() => {
|
|||||||
</Select>
|
</Select>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
|
||||||
|
<TextField
|
||||||
|
fullWidth
|
||||||
|
label="Код города для погоды"
|
||||||
|
value={editCityData.weather_city_code ?? 0}
|
||||||
|
onChange={(e) => setEditCityWeatherCode(Number(e.target.value))}
|
||||||
|
helperText="ID города брать с ресурса openweathermap.org"
|
||||||
|
/>
|
||||||
|
|
||||||
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
||||||
<ImageUploadCard
|
<ImageUploadCard
|
||||||
title="Герб города"
|
title="Герб города"
|
||||||
|
|||||||
@@ -114,10 +114,11 @@ export const CityListPage = observer(() => {
|
|||||||
sortable: false,
|
sortable: false,
|
||||||
renderCell: (params: GridRenderCellParams) => (
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
<button onClick={() => navigate(`/city/${params.row.id}/edit`)}>
|
<button title="Редактировать" onClick={() => navigate(`/city/${params.row.id}/edit`)}>
|
||||||
<Pencil size={20} className="text-blue-500" />
|
<Pencil size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
@@ -139,7 +140,10 @@ export const CityListPage = observer(() => {
|
|||||||
<div className="flex justify-between items-center mb-10">
|
<div className="flex justify-between items-center mb-10">
|
||||||
<h1 className="text-2xl">Города</h1>
|
<h1 className="text-2xl">Города</h1>
|
||||||
{canWriteCities && (
|
{canWriteCities && (
|
||||||
<CreateButton label="Создать город" path="/city/create" />
|
<CreateButton
|
||||||
|
label="Создать город"
|
||||||
|
path="/city/create"
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -155,13 +159,21 @@ export const CityListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={filteredRows}
|
rows={filteredRows}
|
||||||
columns={columns}
|
columns={columns}
|
||||||
checkboxSelection={canWriteCities}
|
checkboxSelection={canWriteCities}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
|
onRowDoubleClick={(params) => {
|
||||||
|
if (canWriteCities) {
|
||||||
|
navigate(`/city/${params.id}/edit`);
|
||||||
|
}
|
||||||
|
}}
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
@@ -194,7 +206,11 @@ export const CityListPage = observer(() => {
|
|||||||
slots={{
|
slots={{
|
||||||
noRowsOverlay: () => (
|
noRowsOverlay: () => (
|
||||||
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
||||||
{isLoading ? <CircularProgress size={20} /> : "Нет городов"}
|
{isLoading ? (
|
||||||
|
<CircularProgress size={20} />
|
||||||
|
) : (
|
||||||
|
"Нет городов"
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
),
|
),
|
||||||
}}
|
}}
|
||||||
|
|||||||
@@ -15,11 +15,18 @@ import {
|
|||||||
RU_COUNTRIES,
|
RU_COUNTRIES,
|
||||||
EN_COUNTRIES,
|
EN_COUNTRIES,
|
||||||
ZH_COUNTRIES,
|
ZH_COUNTRIES,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { useState } from "react";
|
import { useState, useEffect } from "react";
|
||||||
|
|
||||||
export const CountryAddPage = observer(() => {
|
export const CountryAddPage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const { createCountryData, setCountryData, createCountry } = countryStore;
|
const { createCountryData, setCountryData, createCountry } = countryStore;
|
||||||
|
|
||||||
|
|||||||
@@ -4,12 +4,18 @@ import { ArrowLeft, Save } from "lucide-react";
|
|||||||
import { Loader2 } from "lucide-react";
|
import { Loader2 } from "lucide-react";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import { countryStore, languageStore } from "@shared";
|
import { countryStore, languageStore, selectedCityStore } from "@shared";
|
||||||
import { useState } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { LanguageSwitcher } from "@widgets";
|
import { LanguageSwitcher } from "@widgets";
|
||||||
|
|
||||||
export const CountryCreatePage = observer(() => {
|
export const CountryCreatePage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
const { createCountryData, setCountryData, createCountry } = countryStore;
|
const { createCountryData, setCountryData, createCountry } = countryStore;
|
||||||
|
|||||||
@@ -4,12 +4,18 @@ import { ArrowLeft, Save } from "lucide-react";
|
|||||||
import { Loader2 } from "lucide-react";
|
import { Loader2 } from "lucide-react";
|
||||||
import { useNavigate, useParams } from "react-router-dom";
|
import { useNavigate, useParams } from "react-router-dom";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import { countryStore, languageStore, LoadingSpinner } from "@shared";
|
import { countryStore, languageStore, LoadingSpinner, selectedCityStore } from "@shared";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { LanguageSwitcher } from "@widgets";
|
import { LanguageSwitcher } from "@widgets";
|
||||||
|
|
||||||
export const CountryEditPage = observer(() => {
|
export const CountryEditPage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [isLoadingData, setIsLoadingData] = useState(true);
|
const [isLoadingData, setIsLoadingData] = useState(true);
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
|
|||||||
@@ -92,7 +92,10 @@ export const CountryListPage = observer(() => {
|
|||||||
<div className="flex justify-between items-center mb-10">
|
<div className="flex justify-between items-center mb-10">
|
||||||
<h1 className="text-2xl">Страны</h1>
|
<h1 className="text-2xl">Страны</h1>
|
||||||
{canWriteCountries && (
|
{canWriteCountries && (
|
||||||
<CreateButton label="Добавить страну" path="/country/add" />
|
<CreateButton
|
||||||
|
label="Добавить страну"
|
||||||
|
path="/country/add"
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -108,13 +111,16 @@ export const CountryListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={rows}
|
rows={rows}
|
||||||
columns={columns}
|
columns={columns}
|
||||||
checkboxSelection={canWriteCountries}
|
checkboxSelection={canWriteCountries}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
@@ -147,7 +153,11 @@ export const CountryListPage = observer(() => {
|
|||||||
slots={{
|
slots={{
|
||||||
noRowsOverlay: () => (
|
noRowsOverlay: () => (
|
||||||
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
||||||
{isLoading ? <CircularProgress size={20} /> : "Нет стран"}
|
{isLoading ? (
|
||||||
|
<CircularProgress size={20} />
|
||||||
|
) : (
|
||||||
|
"Нет стран"
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
),
|
),
|
||||||
}}
|
}}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import {
|
|||||||
cityStore,
|
cityStore,
|
||||||
createSightStore,
|
createSightStore,
|
||||||
languageStore,
|
languageStore,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import {
|
import {
|
||||||
CreateInformationTab,
|
CreateInformationTab,
|
||||||
@@ -14,6 +15,7 @@ import {
|
|||||||
} from "@widgets";
|
} from "@widgets";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
|
import { runInAction } from "mobx";
|
||||||
|
|
||||||
function a11yProps(index: number) {
|
function a11yProps(index: number) {
|
||||||
return {
|
return {
|
||||||
@@ -28,7 +30,12 @@ export const CreateSightPage = observer(() => {
|
|||||||
const [value, setValue] = useState(0);
|
const [value, setValue] = useState(0);
|
||||||
const { getCities } = cityStore;
|
const { getCities } = cityStore;
|
||||||
const { getArticles } = articlesStore;
|
const { getArticles } = articlesStore;
|
||||||
const { needLeaveAgree } = createSightStore;
|
const needLeave = createSightStore.needLeaveAgree;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const handleChange = (_: React.SyntheticEvent, newValue: number) => {
|
const handleChange = (_: React.SyntheticEvent, newValue: number) => {
|
||||||
setValue(newValue);
|
setValue(newValue);
|
||||||
@@ -36,9 +43,15 @@ export const CreateSightPage = observer(() => {
|
|||||||
|
|
||||||
let blocker = useBlocker(
|
let blocker = useBlocker(
|
||||||
({ currentLocation, nextLocation }) =>
|
({ currentLocation, nextLocation }) =>
|
||||||
needLeaveAgree && currentLocation.pathname !== nextLocation.pathname
|
needLeave && currentLocation.pathname !== nextLocation.pathname,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (blocker.state === "blocked" && !needLeave) {
|
||||||
|
blocker.proceed();
|
||||||
|
}
|
||||||
|
}, [blocker.state, needLeave]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const fetchData = async () => {
|
const fetchData = async () => {
|
||||||
if (!authStore.me) {
|
if (!authStore.me) {
|
||||||
@@ -50,6 +63,14 @@ export const CreateSightPage = observer(() => {
|
|||||||
await authStore.fetchMeCities().catch(() => undefined);
|
await authStore.fetchMeCities().catch(() => undefined);
|
||||||
}
|
}
|
||||||
await getArticles(languageStore.language);
|
await getArticles(languageStore.language);
|
||||||
|
|
||||||
|
const { selectedCityId, selectedCity } = selectedCityStore;
|
||||||
|
if (selectedCityId && selectedCity && !createSightStore.sight.city_id) {
|
||||||
|
runInAction(() => {
|
||||||
|
createSightStore.sight.city_id = selectedCityId;
|
||||||
|
createSightStore.sight.city = selectedCity.name;
|
||||||
|
});
|
||||||
|
}
|
||||||
};
|
};
|
||||||
fetchData();
|
fetchData();
|
||||||
}, []);
|
}, []);
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import {
|
|||||||
cityStore,
|
cityStore,
|
||||||
editSightStore,
|
editSightStore,
|
||||||
LoadingSpinner,
|
LoadingSpinner,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { useBlocker, useParams } from "react-router-dom";
|
import { useBlocker, useParams } from "react-router-dom";
|
||||||
|
|
||||||
@@ -25,6 +26,11 @@ export const EditSightPage = observer(() => {
|
|||||||
const { sight, getSightInfo, needLeaveAgree, getRightArticles } = editSightStore;
|
const { sight, getSightInfo, needLeaveAgree, getRightArticles } = editSightStore;
|
||||||
const { getArticles } = articlesStore;
|
const { getArticles } = articlesStore;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const { id } = useParams();
|
const { id } = useParams();
|
||||||
const { getCities } = cityStore;
|
const { getCities } = cityStore;
|
||||||
|
|
||||||
|
|||||||
@@ -7,10 +7,10 @@ import {
|
|||||||
FormControl,
|
FormControl,
|
||||||
InputLabel,
|
InputLabel,
|
||||||
} from "@mui/material";
|
} from "@mui/material";
|
||||||
import { mediaStore, MEDIA_TYPE_LABELS } from "@shared";
|
import { mediaStore, MEDIA_TYPE_LABELS, selectedCityStore } from "@shared";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { ArrowLeft, Loader2, Save } from "lucide-react";
|
import { ArrowLeft, Loader2, Save } from "lucide-react";
|
||||||
import { useState } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
@@ -20,10 +20,15 @@ export const MediaCreatePage = observer(() => {
|
|||||||
const [type, setType] = useState("");
|
const [type, setType] = useState("");
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const handleCreate = async () => {
|
const handleCreate = async () => {
|
||||||
try {
|
try {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
await mediaStore.createMedia(name, type);
|
await mediaStore.createMedia(name, type, selectedCityStore.selectedCityId);
|
||||||
toast.success("Медиа успешно создано");
|
toast.success("Медиа успешно создано");
|
||||||
navigate("/media");
|
navigate("/media");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ import {
|
|||||||
MEDIA_TYPE_LABELS,
|
MEDIA_TYPE_LABELS,
|
||||||
languageStore,
|
languageStore,
|
||||||
LoadingSpinner,
|
LoadingSpinner,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { MediaViewer } from "@widgets";
|
import { MediaViewer } from "@widgets";
|
||||||
|
|
||||||
@@ -42,6 +43,11 @@ export const MediaEditPage = observer(() => {
|
|||||||
const [mediaType, setMediaType] = useState(media?.media_type ?? 1);
|
const [mediaType, setMediaType] = useState(media?.media_type ?? 1);
|
||||||
const [availableMediaTypes, setAvailableMediaTypes] = useState<number[]>([]);
|
const [availableMediaTypes, setAvailableMediaTypes] = useState<number[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (id) {
|
if (id) {
|
||||||
mediaStore.getOneMedia(id);
|
mediaStore.getOneMedia(id);
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
||||||
import { ruRU } from "@mui/x-data-grid/locales";
|
import { ruRU } from "@mui/x-data-grid/locales";
|
||||||
import { authStore, languageStore, MEDIA_TYPE_LABELS, mediaStore, SearchInput } from "@shared";
|
import { authStore, languageStore, MEDIA_TYPE_LABELS, mediaStore, SearchInput, selectedCityStore } from "@shared";
|
||||||
import { useEffect, useState, useMemo } from "react";
|
import { useEffect, useState, useMemo } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { Eye, Trash2, Minus } from "lucide-react";
|
import { Eye, Trash2, Minus } from "lucide-react";
|
||||||
@@ -78,11 +78,12 @@ export const MediaListPage = observer(() => {
|
|||||||
sortable: false,
|
sortable: false,
|
||||||
renderCell: (params: GridRenderCellParams) => (
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
<button onClick={() => navigate(`/media/${params.row.id}`)}>
|
<button title="Просмотр" onClick={() => navigate(`/media/${params.row.id}`)}>
|
||||||
<Eye size={20} className="text-green-500" />
|
<Eye size={20} className="text-green-500" />
|
||||||
</button>
|
</button>
|
||||||
{canWriteMedia && (
|
{canWriteMedia && (
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -98,19 +99,23 @@ export const MediaListPage = observer(() => {
|
|||||||
|
|
||||||
const rows = useMemo(() => {
|
const rows = useMemo(() => {
|
||||||
const query = searchQuery.trim().toLowerCase();
|
const query = searchQuery.trim().toLowerCase();
|
||||||
|
const cityId = selectedCityStore.selectedCityId;
|
||||||
return media
|
return media
|
||||||
.filter((item) => !query || (item.media_name ?? "").toLowerCase().includes(query))
|
.filter((item) => !query || (item.media_name ?? "").toLowerCase().includes(query))
|
||||||
|
.filter((item) => !cityId || item.city_id === cityId)
|
||||||
.map((item) => ({
|
.map((item) => ({
|
||||||
id: item.id,
|
id: item.id,
|
||||||
media_name: item.media_name,
|
media_name: item.media_name,
|
||||||
media_type: item.media_type,
|
media_type: item.media_type,
|
||||||
}));
|
}));
|
||||||
}, [media, searchQuery]);
|
}, [media, searchQuery, selectedCityStore.selectedCityId]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="w-full">
|
<div className="w-full">
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
{canWriteMedia && ids.length > 0 && (
|
{canWriteMedia && ids.length > 0 && (
|
||||||
<div className="flex justify-end mb-5 duration-300">
|
<div className="flex justify-end mb-5 duration-300">
|
||||||
@@ -129,6 +134,7 @@ export const MediaListPage = observer(() => {
|
|||||||
columns={columns}
|
columns={columns}
|
||||||
checkboxSelection={canWriteMedia}
|
checkboxSelection={canWriteMedia}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import React, { useEffect } from "react";
|
import { useEffect } from "react";
|
||||||
import { useParams } from "react-router-dom";
|
import { useParams } from "react-router-dom";
|
||||||
import { apiStore } from "../../../client/src/api/ApiStore/store";
|
import { apiStore } from "../../../client/src/api/ApiStore/store";
|
||||||
import App from "../../../client/src/App";
|
import App from "../../../client/src/App";
|
||||||
|
|||||||
@@ -39,11 +39,18 @@ import type { Route } from "@shared";
|
|||||||
|
|
||||||
export const RouteCreatePage = observer(() => {
|
export const RouteCreatePage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [carrier, setCarrier] = useState<string>("");
|
const [carrier, setCarrier] = useState<string>("");
|
||||||
const [routeNumber, setRouteNumber] = useState("");
|
const [routeNumber, setRouteNumber] = useState("");
|
||||||
const [routeCoords, setRouteCoords] = useState("");
|
const [routeCoords, setRouteCoords] = useState("");
|
||||||
const [govRouteNumber, setGovRouteNumber] = useState("");
|
const [govRouteNumber, setGovRouteNumber] = useState("");
|
||||||
const [governorAppeal, setGovernorAppeal] = useState<string>("");
|
const [governorAppeal, setGovernorAppeal] = useState<string>("");
|
||||||
|
const [buttonText, setButtonText] = useState("");
|
||||||
const [direction, setDirection] = useState("backward");
|
const [direction, setDirection] = useState("backward");
|
||||||
const [scaleMin, setScaleMin] = useState("10");
|
const [scaleMin, setScaleMin] = useState("10");
|
||||||
const [scaleMax, setScaleMax] = useState("100");
|
const [scaleMax, setScaleMax] = useState("100");
|
||||||
@@ -51,7 +58,7 @@ export const RouteCreatePage = observer(() => {
|
|||||||
const [turn, setTurn] = useState("");
|
const [turn, setTurn] = useState("");
|
||||||
const [centerLat, setCenterLat] = useState("");
|
const [centerLat, setCenterLat] = useState("");
|
||||||
const [centerLng, setCenterLng] = useState("");
|
const [centerLng, setCenterLng] = useState("");
|
||||||
const [videoTimer, setVideoTimer] = useState(60);
|
const [videoTimer, setVideoTimer] = useState(420);
|
||||||
const [videoPreview, setVideoPreview] = useState<string>("");
|
const [videoPreview, setVideoPreview] = useState<string>("");
|
||||||
const [icon, setIcon] = useState<string>("");
|
const [icon, setIcon] = useState<string>("");
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
@@ -286,9 +293,13 @@ export const RouteCreatePage = observer(() => {
|
|||||||
newRoute.governor_appeal = governor_appeal;
|
newRoute.governor_appeal = governor_appeal;
|
||||||
}
|
}
|
||||||
|
|
||||||
await routeStore.createRoute(newRoute);
|
if (buttonText.trim()) {
|
||||||
|
newRoute.button_text = buttonText.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
const newId = await routeStore.createRoute(newRoute);
|
||||||
toast.success("Маршрут успешно создан");
|
toast.success("Маршрут успешно создан");
|
||||||
navigate(-1);
|
navigate(`/route/${newId}/edit`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
toast.error("Произошла ошибка при создании маршрута");
|
toast.error("Произошла ошибка при создании маршрута");
|
||||||
@@ -401,6 +412,18 @@ export const RouteCreatePage = observer(() => {
|
|||||||
onChange={(e) => setGovRouteNumber(e.target.value)}
|
onChange={(e) => setGovRouteNumber(e.target.value)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<Typography variant="subtitle1" sx={{ fontWeight: 600 }}>
|
||||||
|
Текст кнопки обращения
|
||||||
|
</Typography>
|
||||||
|
<TextField
|
||||||
|
value={buttonText}
|
||||||
|
onChange={(e) => setButtonText(e.target.value)}
|
||||||
|
placeholder="Обращение губернатора"
|
||||||
|
fullWidth
|
||||||
|
size="small"
|
||||||
|
helperText="Если пусто, будет использован текст по умолчанию"
|
||||||
|
/>
|
||||||
|
|
||||||
<Typography variant="subtitle1" sx={{ fontWeight: 600 }}>
|
<Typography variant="subtitle1" sx={{ fontWeight: 600 }}>
|
||||||
Обращение к пассажирам
|
Обращение к пассажирам
|
||||||
</Typography>
|
</Typography>
|
||||||
@@ -555,7 +578,7 @@ export const RouteCreatePage = observer(() => {
|
|||||||
/>
|
/>
|
||||||
<TextField
|
<TextField
|
||||||
className="w-full"
|
className="w-full"
|
||||||
label="Таймер видео (сек)"
|
label="Таймер видео заставки (сек)"
|
||||||
type="number"
|
type="number"
|
||||||
value={videoTimer}
|
value={videoTimer}
|
||||||
onChange={(e) => {
|
onChange={(e) => {
|
||||||
|
|||||||
@@ -36,11 +36,18 @@ import {
|
|||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
PreviewMediaDialog,
|
PreviewMediaDialog,
|
||||||
LoadingSpinner,
|
LoadingSpinner,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { LinkedItems } from "../LinekedStations";
|
import { LinkedItems } from "../LinekedStations";
|
||||||
|
|
||||||
export const RouteEditPage = observer(() => {
|
export const RouteEditPage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const { id } = useParams();
|
const { id } = useParams();
|
||||||
const { editRouteData, copyRouteAction } = routeStore;
|
const { editRouteData, copyRouteAction } = routeStore;
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
@@ -548,9 +555,9 @@ export const RouteEditPage = observer(() => {
|
|||||||
/>
|
/>
|
||||||
<TextField
|
<TextField
|
||||||
className="w-full"
|
className="w-full"
|
||||||
label="Таймер видео (сек)"
|
label="Таймер видео заставки (сек)"
|
||||||
type="number"
|
type="number"
|
||||||
value={editRouteData.video_timer ?? 60}
|
value={editRouteData.video_timer ?? 420}
|
||||||
onChange={(e) => {
|
onChange={(e) => {
|
||||||
const val = Math.max(1, Math.round(Number(e.target.value)));
|
const val = Math.max(1, Math.round(Number(e.target.value)));
|
||||||
if (Number.isFinite(val)) {
|
if (Number.isFinite(val)) {
|
||||||
@@ -559,6 +566,22 @@ export const RouteEditPage = observer(() => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<Typography variant="subtitle1" sx={{ fontWeight: 600 }}>
|
||||||
|
Текст кнопки обращения
|
||||||
|
</Typography>
|
||||||
|
<TextField
|
||||||
|
value={editRouteData.button_text || ""}
|
||||||
|
onChange={(e) =>
|
||||||
|
routeStore.setEditRouteData({
|
||||||
|
button_text: e.target.value,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
placeholder="Обращение губернатора"
|
||||||
|
fullWidth
|
||||||
|
size="small"
|
||||||
|
helperText="Если пусто, будет использован текст по умолчанию"
|
||||||
|
/>
|
||||||
|
|
||||||
<Typography variant="subtitle1" sx={{ fontWeight: 600 }}>
|
<Typography variant="subtitle1" sx={{ fontWeight: 600 }}>
|
||||||
Обращение к пассажирам
|
Обращение к пассажирам
|
||||||
</Typography>
|
</Typography>
|
||||||
|
|||||||
@@ -6,10 +6,10 @@ import { observer } from "mobx-react-lite";
|
|||||||
import { Map, Pencil, Trash2, Minus, Monitor } from "lucide-react";
|
import { Map, Pencil, Trash2, Minus, Monitor } from "lucide-react";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { CreateButton, DeleteModal, LanguageSwitcher } from "@widgets";
|
import { CreateButton, DeleteModal, LanguageSwitcher } from "@widgets";
|
||||||
import { Box, CircularProgress } from "@mui/material";
|
import { Box, CircularProgress, Tooltip } from "@mui/material";
|
||||||
|
|
||||||
export const RouteListPage = observer(() => {
|
export const RouteListPage = observer(() => {
|
||||||
const { routes, getRoutes, deleteRoute } = routeStore;
|
const { routes, getRoutes, deleteRoute, sightCounts, stationCounts, countsLoading, loadCounts } = routeStore;
|
||||||
const { carriers, getCarriers } = carrierStore;
|
const { carriers, getCarriers } = carrierStore;
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const [isDeleteModalOpen, setIsDeleteModalOpen] = useState(false);
|
const [isDeleteModalOpen, setIsDeleteModalOpen] = useState(false);
|
||||||
@@ -38,9 +38,12 @@ export const RouteListPage = observer(() => {
|
|||||||
await getCarriers("zh");
|
await getCarriers("zh");
|
||||||
await getRoutes();
|
await getRoutes();
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
|
|
||||||
|
const routeIds = routeStore.routes.data.map((r) => r.id);
|
||||||
|
loadCounts(routeIds);
|
||||||
};
|
};
|
||||||
fetchData();
|
fetchData();
|
||||||
}, [language]);
|
}, [language, selectedCityStore.cityVersion]);
|
||||||
|
|
||||||
const columns: GridColDef[] = [
|
const columns: GridColDef[] = [
|
||||||
{
|
{
|
||||||
@@ -128,6 +131,42 @@ export const RouteListPage = observer(() => {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
field: "sightCount",
|
||||||
|
headerName: "Достопримечательности",
|
||||||
|
width: 180,
|
||||||
|
align: "center" as const,
|
||||||
|
headerAlign: "center" as const,
|
||||||
|
sortable: true,
|
||||||
|
renderHeader: (params: any) => (
|
||||||
|
<Tooltip title="Отображает количество привязанных достопримечательностей">
|
||||||
|
<span>{params.colDef.headerName}</span>
|
||||||
|
</Tooltip>
|
||||||
|
),
|
||||||
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
|
<div className="w-full h-full flex items-center justify-center">
|
||||||
|
{params.value === null ? <CircularProgress size={14} /> : params.value}
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: "stationCount",
|
||||||
|
headerName: "Остановки",
|
||||||
|
width: 120,
|
||||||
|
align: "center" as const,
|
||||||
|
headerAlign: "center" as const,
|
||||||
|
sortable: true,
|
||||||
|
renderHeader: (params: any) => (
|
||||||
|
<Tooltip title="Отображает количество привязанных остановок">
|
||||||
|
<span>{params.colDef.headerName}</span>
|
||||||
|
</Tooltip>
|
||||||
|
),
|
||||||
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
|
<div className="w-full h-full flex items-center justify-center">
|
||||||
|
{params.value === null ? <CircularProgress size={14} /> : params.value}
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
},
|
||||||
...(canShowActionsColumn ? [{
|
...(canShowActionsColumn ? [{
|
||||||
field: "actions",
|
field: "actions",
|
||||||
headerName: "Действия",
|
headerName: "Действия",
|
||||||
@@ -139,22 +178,23 @@ export const RouteListPage = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
{canWriteRoutes && (
|
{canWriteRoutes && (
|
||||||
<button onClick={() => navigate(`/route/${params.row.id}/edit`)}>
|
<button title="Редактировать" onClick={() => navigate(`/route/${params.row.id}/edit`)}>
|
||||||
<Pencil size={20} className="text-blue-500" />
|
<Pencil size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
{canShowRoutePreview && (
|
{canShowRoutePreview && (
|
||||||
<button onClick={() => navigate(`/route-preview/${params.row.id}`)}>
|
<button title="Предпросмотр на карте" onClick={() => navigate(`/route-preview/${params.row.id}`)}>
|
||||||
<Map size={20} className="text-purple-500" />
|
<Map size={20} className="text-purple-500" />
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
{canShowRoutePreview && (
|
{canShowRoutePreview && (
|
||||||
<button onClick={() => window.open(`/demo/${params.row.id}`, "_blank")}>
|
<button title="Демо" onClick={() => window.open(`/demo/${params.row.id}`, "_blank")}>
|
||||||
<Monitor size={20} className="text-green-500" />
|
<Monitor size={20} className="text-green-500" />
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
{canWriteRoutes && (
|
{canWriteRoutes && (
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -171,6 +211,9 @@ export const RouteListPage = observer(() => {
|
|||||||
|
|
||||||
const rows = useMemo(() => {
|
const rows = useMemo(() => {
|
||||||
const { selectedCityId } = selectedCityStore;
|
const { selectedCityId } = selectedCityStore;
|
||||||
|
if (!selectedCityId) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
const query = searchQuery.trim().toLowerCase();
|
const query = searchQuery.trim().toLowerCase();
|
||||||
let filtered = routes.data;
|
let filtered = routes.data;
|
||||||
if (selectedCityId) {
|
if (selectedCityId) {
|
||||||
@@ -195,8 +238,10 @@ export const RouteListPage = observer(() => {
|
|||||||
route_sys_number: route.route_sys_number,
|
route_sys_number: route.route_sys_number,
|
||||||
route_direction: route.route_direction ? "Прямой" : "Обратный",
|
route_direction: route.route_direction ? "Прямой" : "Обратный",
|
||||||
route_name: route.route_name,
|
route_name: route.route_name,
|
||||||
|
sightCount: sightCounts.has(route.id) ? sightCounts.get(route.id) : null,
|
||||||
|
stationCount: stationCounts.has(route.id) ? stationCounts.get(route.id) : null,
|
||||||
}));
|
}));
|
||||||
}, [routes.data, carriers["ru"].data, selectedCityStore.selectedCityId, searchQuery]);
|
}, [routes.data, carriers["ru"].data, selectedCityStore.selectedCityId, searchQuery, sightCounts.size, stationCounts.size, countsLoading]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -206,7 +251,10 @@ export const RouteListPage = observer(() => {
|
|||||||
<div className="flex justify-between items-center mb-10">
|
<div className="flex justify-between items-center mb-10">
|
||||||
<h1 className="text-2xl">Маршруты</h1>
|
<h1 className="text-2xl">Маршруты</h1>
|
||||||
{canWriteRoutes && (
|
{canWriteRoutes && (
|
||||||
<CreateButton label="Создать маршрут" path="/route/create" />
|
<CreateButton
|
||||||
|
label="Создать маршрут"
|
||||||
|
path="/route/create"
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -222,7 +270,9 @@ export const RouteListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@@ -230,6 +280,7 @@ export const RouteListPage = observer(() => {
|
|||||||
onRowDoubleClick={(params) => canWriteRoutes && navigate(`/route/${params.row.id}/edit`)}
|
onRowDoubleClick={(params) => canWriteRoutes && navigate(`/route/${params.row.id}/edit`)}
|
||||||
checkboxSelection={canWriteRoutes}
|
checkboxSelection={canWriteRoutes}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
@@ -262,7 +313,13 @@ export const RouteListPage = observer(() => {
|
|||||||
slots={{
|
slots={{
|
||||||
noRowsOverlay: () => (
|
noRowsOverlay: () => (
|
||||||
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
||||||
{isLoading ? <CircularProgress size={20} /> : "Нет маршрутов"}
|
{isLoading ? (
|
||||||
|
<CircularProgress size={20} />
|
||||||
|
) : !selectedCityStore.selectedCityId ? (
|
||||||
|
"Выберите город"
|
||||||
|
) : (
|
||||||
|
"Нет маршрутов"
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
),
|
),
|
||||||
}}
|
}}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Box, Stack, Typography, Button } from "@mui/material";
|
import { Button } from "@mui/material";
|
||||||
import { useNavigate, useNavigationType } from "react-router";
|
import { useNavigate, useNavigationType } from "react-router";
|
||||||
import { MediaViewer } from "@widgets";
|
import { MediaViewer } from "@widgets";
|
||||||
import { useMapData } from "./MapDataContext";
|
import { useMapData } from "./MapDataContext";
|
||||||
@@ -15,22 +15,24 @@ type LeftSidebarProps = {
|
|||||||
|
|
||||||
export const LeftSidebar = observer(({ open, onToggle }: LeftSidebarProps) => {
|
export const LeftSidebar = observer(({ open, onToggle }: LeftSidebarProps) => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const navigationType = useNavigationType(); // PUSH, POP, REPLACE
|
const navigationType = useNavigationType();
|
||||||
const { routeData } = useMapData();
|
const { routeData } = useMapData();
|
||||||
const [carrierThumbnail, setCarrierThumbnail] = useState<string | null>(null);
|
|
||||||
const [carrierLogo, setCarrierLogo] = useState<string | null>(null);
|
const [carrierLogo, setCarrierLogo] = useState<string | null>(null);
|
||||||
|
const [carrierSlogan, setCarrierSlogan] = useState<string | null>(null);
|
||||||
|
const [carrierShortName, setCarrierShortName] = useState<string | null>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
async function fetchCarrierThumbnail() {
|
async function fetchCarrierData() {
|
||||||
if (routeData?.carrier_id) {
|
if (routeData?.carrier_id) {
|
||||||
const { city_id, logo } = (
|
const carrier = (
|
||||||
await authInstance.get(`/carrier/${routeData.carrier_id}`)
|
await authInstance.get(`/carrier/${routeData.carrier_id}`)
|
||||||
).data;
|
).data;
|
||||||
const { arms } = (await authInstance.get(`/city/${city_id}`)).data;
|
setCarrierLogo(carrier.logo);
|
||||||
setCarrierThumbnail(arms);
|
setCarrierSlogan(carrier.slogan ?? null);
|
||||||
setCarrierLogo(logo);
|
setCarrierShortName(carrier.short_name ?? null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fetchCarrierThumbnail();
|
fetchCarrierData();
|
||||||
}, [routeData?.carrier_id]);
|
}, [routeData?.carrier_id]);
|
||||||
|
|
||||||
const handleBack = () => {
|
const handleBack = () => {
|
||||||
@@ -42,131 +44,190 @@ export const LeftSidebar = observer(({ open, onToggle }: LeftSidebarProps) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box
|
<div
|
||||||
sx={{
|
style={{
|
||||||
position: "relative",
|
position: "relative",
|
||||||
|
width: 288,
|
||||||
height: "100%",
|
height: "100%",
|
||||||
color: "#fff",
|
color: "#fff",
|
||||||
transition: "padding 0.3s ease",
|
|
||||||
p: open ? 2 : 0,
|
|
||||||
display: "flex",
|
|
||||||
flexDirection: "column",
|
|
||||||
alignItems: "stretch",
|
|
||||||
justifyContent: "flex-start",
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Stack
|
{/* Кнопка назад — вне основного меню */}
|
||||||
direction="column"
|
<div
|
||||||
height="100%"
|
style={{
|
||||||
width="100%"
|
padding: "12px 12px 0",
|
||||||
spacing={4}
|
opacity: open ? 1 : 0,
|
||||||
alignItems="stretch"
|
pointerEvents: open ? "auto" : "none",
|
||||||
justifyContent="space-between"
|
transition: "opacity 0.25s ease",
|
||||||
sx={{
|
}}
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
onClick={handleBack}
|
||||||
|
variant="contained"
|
||||||
|
sx={{
|
||||||
|
backgroundColor: "#222",
|
||||||
|
color: "#fff",
|
||||||
|
borderRadius: 1.5,
|
||||||
|
px: 2,
|
||||||
|
py: 1,
|
||||||
|
"&:hover": { backgroundColor: "#2d2d2d" },
|
||||||
|
}}
|
||||||
|
fullWidth
|
||||||
|
startIcon={<ArrowBackIcon />}
|
||||||
|
>
|
||||||
|
Назад
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Основное меню — повторяет .side-menu */}
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
boxSizing: "border-box",
|
||||||
|
paddingTop: 46,
|
||||||
|
display: "flex",
|
||||||
|
flexDirection: "column",
|
||||||
|
alignItems: "center",
|
||||||
|
height: "calc(100% - 56px)",
|
||||||
|
position: "relative",
|
||||||
opacity: open ? 1 : 0,
|
opacity: open ? 1 : 0,
|
||||||
transition: "opacity 0.25s ease",
|
transition: "opacity 0.25s ease",
|
||||||
pointerEvents: open ? "auto" : "none",
|
pointerEvents: open ? "auto" : "none",
|
||||||
display: open ? "flex" : "none",
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div>
|
{/* Герб — .side-menu-crest */}
|
||||||
<Button
|
<div
|
||||||
onClick={handleBack}
|
style={{
|
||||||
variant="contained"
|
width: 170,
|
||||||
color="primary"
|
height: 170,
|
||||||
sx={{
|
alignSelf: "flex-start",
|
||||||
backgroundColor: "#222",
|
marginLeft: 20,
|
||||||
color: "#fff",
|
backgroundColor: "rgba(255,255,255,0.15)",
|
||||||
borderRadius: 1.5,
|
borderRadius: 8,
|
||||||
px: 2,
|
display: "flex",
|
||||||
py: 1,
|
alignItems: "center",
|
||||||
marginBottom: 10,
|
justifyContent: "center",
|
||||||
"&:hover": {
|
color: "rgba(255,255,255,0.5)",
|
||||||
backgroundColor: "#2d2d2d",
|
fontSize: 14,
|
||||||
},
|
fontWeight: 500,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Герб
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Слоган — .side-menu-label */}
|
||||||
|
{carrierSlogan && (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
marginTop: 10,
|
||||||
|
textAlign: "left",
|
||||||
|
fontSize: 15,
|
||||||
|
padding: "0 20px",
|
||||||
|
alignSelf: "flex-start",
|
||||||
|
fontWeight: 400,
|
||||||
|
lineHeight: "150%",
|
||||||
}}
|
}}
|
||||||
fullWidth
|
|
||||||
startIcon={<ArrowBackIcon />}
|
|
||||||
>
|
>
|
||||||
Назад
|
{carrierSlogan}
|
||||||
</Button>
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
<Stack
|
{/* Кнопки — .side-menu-buttons */}
|
||||||
direction="column"
|
<div
|
||||||
alignItems="center"
|
style={{
|
||||||
justifyContent="center"
|
width: 220,
|
||||||
spacing={3}
|
marginTop: routeData?.governor_appeal || 0 > 0 ? 40 : 260,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
backgroundColor: "#fff",
|
||||||
|
color: "#000",
|
||||||
|
textAlign: "center",
|
||||||
|
padding: "8px 16px",
|
||||||
|
marginBottom: 16,
|
||||||
|
borderRadius: 10,
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<div
|
Достопримечательности
|
||||||
style={{
|
</div>
|
||||||
maxWidth: 150,
|
<div
|
||||||
display: "flex",
|
style={{
|
||||||
flexDirection: "column",
|
backgroundColor: "#fff",
|
||||||
alignItems: "center",
|
color: "#000",
|
||||||
gap: 10,
|
textAlign: "center",
|
||||||
}}
|
padding: "8px 16px",
|
||||||
>
|
marginBottom: 16,
|
||||||
{carrierThumbnail && !isMediaIdEmpty(carrierThumbnail) && (
|
borderRadius: 10,
|
||||||
<MediaViewer
|
}}
|
||||||
media={{
|
>
|
||||||
id: carrierThumbnail,
|
Остановки
|
||||||
media_type: 1, // Тип "Фото" для логотипа
|
|
||||||
filename: "route_thumbnail",
|
|
||||||
}}
|
|
||||||
fullWidth
|
|
||||||
fullHeight
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<Typography sx={{ color: "#fff" }} textAlign="center">
|
|
||||||
При поддержке Правительства
|
|
||||||
</Typography>
|
|
||||||
</div>
|
|
||||||
</Stack>
|
|
||||||
|
|
||||||
<div className="flex flex-col items-center justify-center gap-2 mt-10">
|
|
||||||
<button className="bg-[#fcd500] text-black px-4 py-2 rounded-md w-full font-medium my-10">
|
|
||||||
Обращение губернатора
|
|
||||||
</button>
|
|
||||||
<button className="bg-white text-black px-4 py-2 rounded-md w-full font-medium mx-5">
|
|
||||||
Достопримечательности
|
|
||||||
</button>
|
|
||||||
<button className="bg-white text-black px-4 py-2 rounded-md w-full font-medium mx-5">
|
|
||||||
Остановки
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Stack
|
{/* Нижняя секция — .side-menu-bottom-section */}
|
||||||
direction="column"
|
<div
|
||||||
alignItems="center"
|
style={{
|
||||||
maxHeight={150}
|
position: "absolute",
|
||||||
justifyContent="center"
|
bottom: 0,
|
||||||
flexGrow={1}
|
left: 0,
|
||||||
|
width: "100%",
|
||||||
|
display: "flex",
|
||||||
|
flexDirection: "column",
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
{carrierLogo && !isMediaIdEmpty(carrierLogo) && (
|
{/* .side-menu-carrier-block */}
|
||||||
<MediaViewer
|
<div style={{ padding: "0 20px" }}>
|
||||||
media={{
|
{carrierLogo && !isMediaIdEmpty(carrierLogo) && (
|
||||||
id: carrierLogo,
|
<div style={{ width: 170 }}>
|
||||||
media_type: 1, // Тип "Фото" для логотипа
|
<MediaViewer
|
||||||
filename: "route_thumbnail_logo",
|
media={{
|
||||||
}}
|
id: carrierLogo,
|
||||||
fullHeight
|
media_type: 1,
|
||||||
/>
|
filename: "carrier_logo",
|
||||||
)}
|
}}
|
||||||
</Stack>
|
fullWidth
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{carrierShortName && (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
marginTop: 4,
|
||||||
|
textAlign: "left",
|
||||||
|
fontSize: 16,
|
||||||
|
fontWeight: 700,
|
||||||
|
lineHeight: "150%",
|
||||||
|
color: "#fff",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{carrierShortName}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
<Typography
|
{/* .side-menu-bottom-photo */}
|
||||||
variant="h6"
|
<img
|
||||||
textAlign="center"
|
src="/side-menu-photo.png"
|
||||||
sx={{ color: "#fff", marginTop: "auto" }}
|
alt=""
|
||||||
>
|
style={{
|
||||||
#ВсемПоПути
|
width: "288px",
|
||||||
</Typography>
|
marginTop: 32,
|
||||||
</Stack>
|
display: "block",
|
||||||
|
pointerEvents: "none",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="absolute bottom-[20px] -right-[520px] z-10">
|
<div
|
||||||
|
className="absolute bottom-[20px] z-10"
|
||||||
|
style={{
|
||||||
|
right: open ? -520 : -312,
|
||||||
|
transition: "right 0.3s ease",
|
||||||
|
}}
|
||||||
|
>
|
||||||
<LanguageSelector onBack={onToggle} isSidebarOpen={open} />
|
<LanguageSelector onBack={onToggle} isSidebarOpen={open} />
|
||||||
</div>
|
</div>
|
||||||
</Box>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -141,6 +141,7 @@ export function RightSidebar() {
|
|||||||
bgcolor="primary.main"
|
bgcolor="primary.main"
|
||||||
border="1px solid #e0e0e0"
|
border="1px solid #e0e0e0"
|
||||||
borderRadius={2}
|
borderRadius={2}
|
||||||
|
zIndex={2}
|
||||||
>
|
>
|
||||||
<Typography variant="h6" sx={{ mb: 2, color: "#fff" }} textAlign="center">
|
<Typography variant="h6" sx={{ mb: 2, color: "#fff" }} textAlign="center">
|
||||||
Настройка маршрута
|
Настройка маршрута
|
||||||
|
|||||||
@@ -448,6 +448,22 @@ const StationLabel = observer(
|
|||||||
anchor={dynamicAnchor}
|
anchor={dynamicAnchor}
|
||||||
zIndex={isHovered || isControlHovered ? 1000 : 0}
|
zIndex={isHovered || isControlHovered ? 1000 : 0}
|
||||||
>
|
>
|
||||||
|
{ruLabelWidth > 0 && (
|
||||||
|
<pixiGraphics
|
||||||
|
draw={(g: Graphics) => {
|
||||||
|
g.clear();
|
||||||
|
const hasSecondLabel = !!(station.name && language !== "ru" && ruLabel);
|
||||||
|
const pad = 10 / scale;
|
||||||
|
const w = ruLabelWidth + pad * 2;
|
||||||
|
const top = -compensatedRuFontSize / 2 - pad;
|
||||||
|
const bottom = hasSecondLabel
|
||||||
|
? compensatedRuFontSize * 1.1 + compensatedNameFontSize / 2 + pad
|
||||||
|
: compensatedRuFontSize / 2 + pad;
|
||||||
|
g.rect(-w / 2, top, w, bottom - top);
|
||||||
|
g.fill({ color: 0x000000, alpha: 0.001 });
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
{ruLabel && (
|
{ruLabel && (
|
||||||
<pixiText
|
<pixiText
|
||||||
ref={ruLabelRef}
|
ref={ruLabelRef}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { Stack, Typography, Box, IconButton } from "@mui/material";
|
|||||||
import { Close } from "@mui/icons-material";
|
import { Close } from "@mui/icons-material";
|
||||||
import { Landmark } from "lucide-react";
|
import { Landmark } from "lucide-react";
|
||||||
import { useMapData } from "./MapDataContext";
|
import { useMapData } from "./MapDataContext";
|
||||||
|
import { RouteWidget } from "./webgl-prototype/RouteWidget";
|
||||||
|
|
||||||
export function Widgets() {
|
export function Widgets() {
|
||||||
const { selectedSight, setSelectedSight } = useMapData();
|
const { selectedSight, setSelectedSight } = useMapData();
|
||||||
@@ -13,22 +14,11 @@ export function Widgets() {
|
|||||||
position="absolute"
|
position="absolute"
|
||||||
top={32}
|
top={32}
|
||||||
left={32}
|
left={32}
|
||||||
|
zIndex={2}
|
||||||
sx={{ pointerEvents: "none" }}
|
sx={{ pointerEvents: "none" }}
|
||||||
>
|
>
|
||||||
<Stack
|
{/* Виджет маршрута */}
|
||||||
bgcolor="primary.main"
|
<RouteWidget />
|
||||||
width={361}
|
|
||||||
height={96}
|
|
||||||
p={2}
|
|
||||||
m={2}
|
|
||||||
borderRadius={2}
|
|
||||||
alignItems="center"
|
|
||||||
justifyContent="center"
|
|
||||||
>
|
|
||||||
<Typography variant="h6" sx={{ color: "#fff" }}>
|
|
||||||
Остановка
|
|
||||||
</Typography>
|
|
||||||
</Stack>
|
|
||||||
|
|
||||||
{/* Виджет выбранной достопримечательности (заменяет виджет погоды) */}
|
{/* Виджет выбранной достопримечательности (заменяет виджет погоды) */}
|
||||||
<Stack
|
<Stack
|
||||||
|
|||||||
@@ -54,7 +54,8 @@ export const RoutePreview = () => {
|
|||||||
<Box
|
<Box
|
||||||
sx={{
|
sx={{
|
||||||
position: "relative",
|
position: "relative",
|
||||||
width: isLeftSidebarOpen ? 300 : 0,
|
zIndex: 20,
|
||||||
|
width: isLeftSidebarOpen ? 288 : 0,
|
||||||
transition: "width 0.3s ease",
|
transition: "width 0.3s ease",
|
||||||
overflow: "visible",
|
overflow: "visible",
|
||||||
height: "100%",
|
height: "100%",
|
||||||
@@ -145,14 +146,14 @@ export const RouteMap = observer(() => {
|
|||||||
) {
|
) {
|
||||||
const coordinates = coordinatesToLocal(
|
const coordinates = coordinatesToLocal(
|
||||||
originalRouteData?.center_latitude,
|
originalRouteData?.center_latitude,
|
||||||
originalRouteData?.center_longitude
|
originalRouteData?.center_longitude,
|
||||||
);
|
);
|
||||||
|
|
||||||
setTransform(
|
setTransform(
|
||||||
coordinates.x,
|
coordinates.x,
|
||||||
coordinates.y,
|
coordinates.y,
|
||||||
originalRouteData?.rotate,
|
originalRouteData?.rotate,
|
||||||
originalRouteData?.scale_min
|
originalRouteData?.scale_min,
|
||||||
);
|
);
|
||||||
setIsSetup(true);
|
setIsSetup(true);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ export interface RouteData {
|
|||||||
icon_size?: number;
|
icon_size?: number;
|
||||||
font_size: number;
|
font_size: number;
|
||||||
governor_appeal: number;
|
governor_appeal: number;
|
||||||
|
button_text?: string;
|
||||||
id: number;
|
id: number;
|
||||||
path: [number, number][];
|
path: [number, number][];
|
||||||
rotate: number;
|
rotate: number;
|
||||||
|
|||||||
@@ -0,0 +1,120 @@
|
|||||||
|
.route-widget-label.marquee {
|
||||||
|
display: inline-block;
|
||||||
|
animation: marquee 14s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-subtitle.marquee {
|
||||||
|
display: inline-block;
|
||||||
|
animation: marquee 14s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes marquee {
|
||||||
|
0% {
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
50% {
|
||||||
|
transform: translateX(-50%);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget {
|
||||||
|
width: 361px;
|
||||||
|
height: 96px;
|
||||||
|
position: fixed;
|
||||||
|
display: inline-flex;
|
||||||
|
border-radius: 10px;
|
||||||
|
box-shadow:
|
||||||
|
0 0 0 1px rgba(255, 255, 255, 0.3) inset,
|
||||||
|
/* Внутренняя рамка */ 4px 4px 12px 0 rgba(255, 255, 255, 0.12) inset; /* Ваш существующий внутренний shadow */
|
||||||
|
padding: 1px; /* Чтобы контент не прилипал к рамке */
|
||||||
|
background:
|
||||||
|
linear-gradient(
|
||||||
|
to bottom right,
|
||||||
|
rgba(255, 255, 255, 0.2) 0%,
|
||||||
|
rgba(255, 255, 255, 0) 100%
|
||||||
|
),
|
||||||
|
rgba(179, 165, 152, 0.4);
|
||||||
|
backdrop-filter: blur(10px);
|
||||||
|
pointer-events: auto;
|
||||||
|
z-index: 10000001;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-number {
|
||||||
|
position: absolute;
|
||||||
|
width: fit-content;
|
||||||
|
left: 0px;
|
||||||
|
top: 0px;
|
||||||
|
min-width: 94px;
|
||||||
|
max-width: 100px;
|
||||||
|
height: 96px;
|
||||||
|
background-color: #fcd500;
|
||||||
|
color: black;
|
||||||
|
border-top-left-radius: 10px;
|
||||||
|
border-bottom-left-radius: 10px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 70px;
|
||||||
|
padding: 14px;
|
||||||
|
font-weight: 900;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-number--3 {
|
||||||
|
font-size: 50px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-number--4 {
|
||||||
|
font-size: 38px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-number--5 {
|
||||||
|
font-size: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-content {
|
||||||
|
overflow: hidden;
|
||||||
|
width: 257px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
margin-top: 13px;
|
||||||
|
margin-left: 109px;
|
||||||
|
margin-right: 9px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-label {
|
||||||
|
white-space: nowrap;
|
||||||
|
font-size: 24px;
|
||||||
|
margin: 1px 0;
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 700;
|
||||||
|
line-height: 24px;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-label--medium {
|
||||||
|
font-size: 22px;
|
||||||
|
line-height: 22px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-label--small {
|
||||||
|
font-size: 20px;
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-label--xsmall {
|
||||||
|
font-size: 18px;
|
||||||
|
line-height: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.route-widget-subtitle {
|
||||||
|
white-space: nowrap;
|
||||||
|
color: #cbcbcb;
|
||||||
|
font-size: 12px;
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 400;
|
||||||
|
line-height: normal;
|
||||||
|
margin-top: 4px;
|
||||||
|
}
|
||||||
@@ -0,0 +1,80 @@
|
|||||||
|
import styles from "./RouteWidget.module.css";
|
||||||
|
import { useMapData } from "../MapDataContext";
|
||||||
|
import { languageStore } from "@shared";
|
||||||
|
import { observer } from "mobx-react-lite";
|
||||||
|
|
||||||
|
const shouldAnimate = (text: string | undefined, maxLength: number) =>
|
||||||
|
(text?.length ?? 0) > maxLength;
|
||||||
|
|
||||||
|
const getNumberSizeClass = (text: string | undefined) => {
|
||||||
|
const length = text?.length ?? 0;
|
||||||
|
if (length <= 2) return "";
|
||||||
|
if (length === 3) return styles["route-widget-number--3"];
|
||||||
|
if (length === 4) return styles["route-widget-number--4"];
|
||||||
|
return styles["route-widget-number--5"];
|
||||||
|
};
|
||||||
|
|
||||||
|
const getLabelSizeClass = (text: string | undefined) => {
|
||||||
|
const length = text?.length ?? 0;
|
||||||
|
if (length <= 40) return "";
|
||||||
|
if (length <= 60) return styles["route-widget-label--medium"];
|
||||||
|
if (length <= 80) return styles["route-widget-label--small"];
|
||||||
|
return styles["route-widget-label--xsmall"];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const RouteWidget = observer(() => {
|
||||||
|
const { routeData, stationData } = useMapData();
|
||||||
|
const { language } = languageStore;
|
||||||
|
|
||||||
|
const stations = stationData?.[language] ?? stationData?.["ru"] ?? [];
|
||||||
|
const stationsRu = stationData?.["ru"] ?? [];
|
||||||
|
|
||||||
|
const startStation = stations[0];
|
||||||
|
const endStation = stations[stations.length - 1];
|
||||||
|
|
||||||
|
const startStationRu = stationsRu[0];
|
||||||
|
const endStationRu = stationsRu[stationsRu.length - 1];
|
||||||
|
|
||||||
|
const enSubtitle = `${startStationRu?.name ?? ""} - ${endStationRu?.name ?? ""}`;
|
||||||
|
const zhSubtitle = `${startStation?.name ?? ""} - ${endStation?.name ?? ""}`;
|
||||||
|
const subtitle = language === "zh" ? zhSubtitle : enSubtitle;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={styles["route-widget"]} style={{ position: "relative" }}>
|
||||||
|
<div className={[styles["route-widget-number"], getNumberSizeClass(routeData?.route_sys_number)].join(" ")}>
|
||||||
|
{routeData?.route_sys_number || ""}
|
||||||
|
</div>
|
||||||
|
<div className={styles["route-widget-content"]}>
|
||||||
|
<div
|
||||||
|
className={[
|
||||||
|
styles["route-widget-label"],
|
||||||
|
shouldAnimate(startStation?.name, 18) ? styles["marquee"] : "",
|
||||||
|
getLabelSizeClass(startStation?.name),
|
||||||
|
].join(" ")}
|
||||||
|
title={startStation?.name}
|
||||||
|
>
|
||||||
|
{startStation?.name}
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className={[
|
||||||
|
styles["route-widget-label"],
|
||||||
|
shouldAnimate(endStation?.name, 18) ? styles["marquee"] : "",
|
||||||
|
getLabelSizeClass(endStation?.name),
|
||||||
|
].join(" ")}
|
||||||
|
title={endStation?.name}
|
||||||
|
>
|
||||||
|
{endStation?.name}
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className={[
|
||||||
|
styles["route-widget-subtitle"],
|
||||||
|
shouldAnimate(subtitle, 50) ? styles["marquee"] : "",
|
||||||
|
].join(" ")}
|
||||||
|
title={subtitle}
|
||||||
|
>
|
||||||
|
{subtitle}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
});
|
||||||
@@ -1910,7 +1910,8 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
|
|
||||||
const stationIconSizePercent =
|
const stationIconSizePercent =
|
||||||
liveStationIconSizes.get(station.id) ??
|
liveStationIconSizes.get(station.id) ??
|
||||||
(typeof station.icon_size === "number" && Number.isFinite(station.icon_size)
|
(typeof station.icon_size === "number" &&
|
||||||
|
Number.isFinite(station.icon_size)
|
||||||
? station.icon_size
|
? station.icon_size
|
||||||
: 100);
|
: 100);
|
||||||
const iconSizePx = Math.max(
|
const iconSizePx = Math.max(
|
||||||
@@ -2277,6 +2278,7 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
position: "absolute",
|
position: "absolute",
|
||||||
inset: 0,
|
inset: 0,
|
||||||
pointerEvents: "none",
|
pointerEvents: "none",
|
||||||
|
zIndex: 1,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{stationData.ru.map((station, index) => {
|
{stationData.ru.map((station, index) => {
|
||||||
@@ -2325,9 +2327,6 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
const stationScreenY =
|
const stationScreenY =
|
||||||
rotatedY * camera.scale + camera.translation.y;
|
rotatedY * camera.scale + camera.translation.y;
|
||||||
|
|
||||||
const labelX = stationScreenX + offsetX;
|
|
||||||
const labelY = stationScreenY + offsetY;
|
|
||||||
|
|
||||||
const backendAlign = station.align;
|
const backendAlign = station.align;
|
||||||
|
|
||||||
const anchor = getAnchorFromOffset(backendAlign ?? 2);
|
const anchor = getAnchorFromOffset(backendAlign ?? 2);
|
||||||
@@ -2337,8 +2336,6 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
|
|
||||||
const dpr = Math.max(1, window.devicePixelRatio || 1);
|
const dpr = Math.max(1, window.devicePixelRatio || 1);
|
||||||
|
|
||||||
const cssX = labelX / dpr;
|
|
||||||
const cssY = labelY / dpr;
|
|
||||||
const rotationCss = `${rotationAngle}rad`;
|
const rotationCss = `${rotationAngle}rad`;
|
||||||
const counterRotationCss = `${-rotationAngle}rad`;
|
const counterRotationCss = `${-rotationAngle}rad`;
|
||||||
|
|
||||||
@@ -2357,6 +2354,13 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
const scaleFactor = 1 + (zoomClampedScale - 1) * 0.4;
|
const scaleFactor = 1 + (zoomClampedScale - 1) * 0.4;
|
||||||
|
|
||||||
const primaryFontSize = 16 * fontScale * scaleFactor;
|
const primaryFontSize = 16 * fontScale * scaleFactor;
|
||||||
|
|
||||||
|
const mainLabelHeight = primaryFontSize * 1.2;
|
||||||
|
const labelX = stationScreenX + offsetX;
|
||||||
|
const labelY = stationScreenY + offsetY + mainLabelHeight / 2;
|
||||||
|
|
||||||
|
const cssX = labelX / dpr;
|
||||||
|
const cssY = labelY / dpr;
|
||||||
const secondaryFontSize = 13 * fontScale * scaleFactor;
|
const secondaryFontSize = 13 * fontScale * scaleFactor;
|
||||||
|
|
||||||
const secondaryMarginTop = 5 * fontScale * scaleFactor;
|
const secondaryMarginTop = 5 * fontScale * scaleFactor;
|
||||||
@@ -2402,13 +2406,7 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
hoveredStationIconId === station.id ||
|
hoveredStationIconId === station.id ||
|
||||||
resizingStationIconId === station.id;
|
resizingStationIconId === station.id;
|
||||||
|
|
||||||
const secondaryLineHeight = 1.2;
|
const secondaryLineHeight = 1.2 * scaleFactor;
|
||||||
const secondaryHeight = showSecondary
|
|
||||||
? secondaryFontSize * secondaryLineHeight
|
|
||||||
: 0;
|
|
||||||
const menuPaddingTop = showSecondary
|
|
||||||
? Math.max(0, secondaryHeight - secondaryMarginTop) + 3
|
|
||||||
: 3;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div key={station.id}>
|
<div key={station.id}>
|
||||||
@@ -2438,7 +2436,7 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
color: "#fff",
|
color: "#fff",
|
||||||
fontFamily: "Roboto, sans-serif",
|
fontFamily: "Roboto, sans-serif",
|
||||||
textAlign: "left",
|
textAlign: "left",
|
||||||
pointerEvents: "auto",
|
pointerEvents: "none",
|
||||||
cursor: "grab",
|
cursor: "grab",
|
||||||
userSelect: "none",
|
userSelect: "none",
|
||||||
touchAction: "none",
|
touchAction: "none",
|
||||||
@@ -2446,14 +2444,16 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
style={{
|
style={{
|
||||||
pointerEvents: "auto",
|
display: "inline-block",
|
||||||
|
pointerEvents: "none",
|
||||||
transformOrigin: "left center",
|
transformOrigin: "left center",
|
||||||
transform: `rotate(${rotationCss})`,
|
transform: `rotate(${rotationCss})`,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
style={{
|
style={{
|
||||||
pointerEvents: "auto",
|
display: "inline-block",
|
||||||
|
pointerEvents: "none",
|
||||||
transformOrigin: "left center",
|
transformOrigin: "left center",
|
||||||
transform: `rotate(${counterRotationCss})`,
|
transform: `rotate(${counterRotationCss})`,
|
||||||
}}
|
}}
|
||||||
@@ -2547,6 +2547,7 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
) : null}
|
) : null}
|
||||||
<div
|
<div
|
||||||
style={{
|
style={{
|
||||||
|
position: "relative",
|
||||||
fontWeight: 700,
|
fontWeight: 700,
|
||||||
fontSize: primaryFontSize,
|
fontSize: primaryFontSize,
|
||||||
textShadow: "0 0 4px rgba(0,0,0,0.6)",
|
textShadow: "0 0 4px rgba(0,0,0,0.6)",
|
||||||
@@ -2555,26 +2556,26 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{station.name}
|
{station.name}
|
||||||
|
{showSecondary ? (
|
||||||
|
<div
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
top: "100%",
|
||||||
|
marginTop: -1 * secondaryMarginTop,
|
||||||
|
fontWeight: 400,
|
||||||
|
fontSize: secondaryFontSize,
|
||||||
|
lineHeight: secondaryLineHeight,
|
||||||
|
color: "#CBCBCB",
|
||||||
|
textShadow: "0 0 3px rgba(0,0,0,0.4)",
|
||||||
|
whiteSpace: "nowrap",
|
||||||
|
...secondaryPositionStyle,
|
||||||
|
pointerEvents: "none",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{secondaryStation?.name}
|
||||||
|
</div>
|
||||||
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
{showSecondary ? (
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
position: "absolute",
|
|
||||||
top: "100%",
|
|
||||||
marginTop: -1 * secondaryMarginTop,
|
|
||||||
fontWeight: 400,
|
|
||||||
fontSize: secondaryFontSize,
|
|
||||||
lineHeight: secondaryLineHeight,
|
|
||||||
color: "#CBCBCB",
|
|
||||||
textShadow: "0 0 3px rgba(0,0,0,0.4)",
|
|
||||||
whiteSpace: "nowrap",
|
|
||||||
...secondaryPositionStyle,
|
|
||||||
pointerEvents: "none",
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{secondaryStation?.name}
|
|
||||||
</div>
|
|
||||||
) : null}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -2585,9 +2586,9 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
top: "100%",
|
top: "100%",
|
||||||
left: "50%",
|
left: "50%",
|
||||||
transform: "translateX(-50%)",
|
transform: "translateX(-50%)",
|
||||||
paddingTop: menuPaddingTop,
|
paddingTop: "8px",
|
||||||
pointerEvents: "auto",
|
pointerEvents: "auto",
|
||||||
zIndex: 10,
|
zIndex: 1000000,
|
||||||
cursor: "default",
|
cursor: "default",
|
||||||
}}
|
}}
|
||||||
onPointerDown={(e) => e.stopPropagation()}
|
onPointerDown={(e) => e.stopPropagation()}
|
||||||
@@ -2706,13 +2707,14 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
? camera.scale /
|
? camera.scale /
|
||||||
Math.max(customSightIconBaseScaleRef.current ?? 1, 1e-6)
|
Math.max(customSightIconBaseScaleRef.current ?? 1, 1e-6)
|
||||||
: 1;
|
: 1;
|
||||||
const sightIconSizePercent = sight.is_default_icon === false
|
const sightIconSizePercent =
|
||||||
? (liveSightIconSizes.get(sight.id) ??
|
sight.is_default_icon === false
|
||||||
(typeof sight.icon_size === "number" &&
|
? (liveSightIconSizes.get(sight.id) ??
|
||||||
Number.isFinite(sight.icon_size)
|
(typeof sight.icon_size === "number" &&
|
||||||
? sight.icon_size
|
Number.isFinite(sight.icon_size)
|
||||||
: 100))
|
? sight.icon_size
|
||||||
: (routeData?.icon_size ?? originalRouteData?.icon_size ?? 100);
|
: 100))
|
||||||
|
: (routeData?.icon_size ?? originalRouteData?.icon_size ?? 100);
|
||||||
const iconSize =
|
const iconSize =
|
||||||
30 *
|
30 *
|
||||||
clamp(sightIconSizePercent / 100, 0.1, 10) *
|
clamp(sightIconSizePercent / 100, 0.1, 10) *
|
||||||
@@ -2723,7 +2725,10 @@ export const WebGLRouteMapPrototype = observer(() => {
|
|||||||
resizingSightIconId === sight.id);
|
resizingSightIconId === sight.id);
|
||||||
const iconLeft = cssX - iconSize;
|
const iconLeft = cssX - iconSize;
|
||||||
const iconTop = cssY - iconSize;
|
const iconTop = cssY - iconSize;
|
||||||
const sightZoomClampedScale = Math.min(Math.max(camera.scale, 1), 3);
|
const sightZoomClampedScale = Math.min(
|
||||||
|
Math.max(camera.scale, 1),
|
||||||
|
3,
|
||||||
|
);
|
||||||
const sightScaleFactor = 1 + (sightZoomClampedScale - 1) * 0.4;
|
const sightScaleFactor = 1 + (sightZoomClampedScale - 1) * 0.4;
|
||||||
const labelHeight = 24 * sightScaleFactor;
|
const labelHeight = 24 * sightScaleFactor;
|
||||||
const labelPadding = 6 * sightScaleFactor;
|
const labelPadding = 6 * sightScaleFactor;
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ export const SightListPage = observer(() => {
|
|||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
};
|
};
|
||||||
fetchSights();
|
fetchSights();
|
||||||
}, [language]);
|
}, [language, selectedCityStore.cityVersion]);
|
||||||
|
|
||||||
const columns: GridColDef[] = [
|
const columns: GridColDef[] = [
|
||||||
{
|
{
|
||||||
@@ -105,10 +105,11 @@ export const SightListPage = observer(() => {
|
|||||||
sortable: false,
|
sortable: false,
|
||||||
renderCell: (params: GridRenderCellParams) => (
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
<button onClick={() => navigate(`/sight/${params.row.id}/edit`)}>
|
<button title="Редактировать" onClick={() => navigate(`/sight/${params.row.id}/edit`)}>
|
||||||
<Pencil size={20} className="text-blue-500" />
|
<Pencil size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -121,8 +122,11 @@ export const SightListPage = observer(() => {
|
|||||||
}] : []),
|
}] : []),
|
||||||
];
|
];
|
||||||
|
|
||||||
|
const { selectedCityId } = selectedCityStore;
|
||||||
const filteredSights = useMemo(() => {
|
const filteredSights = useMemo(() => {
|
||||||
const { selectedCityId } = selectedCityStore;
|
if (!selectedCityId) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
const allowedCityIds = canReadCities
|
const allowedCityIds = canReadCities
|
||||||
? null
|
? null
|
||||||
: authStore.meCities["ru"].map((c) => c.city_id);
|
: authStore.meCities["ru"].map((c) => c.city_id);
|
||||||
@@ -131,12 +135,12 @@ export const SightListPage = observer(() => {
|
|||||||
if (allowedCityIds && !allowedCityIds.includes(sight.city_id)) {
|
if (allowedCityIds && !allowedCityIds.includes(sight.city_id)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (selectedCityId && sight.city_id !== selectedCityId) {
|
if (sight.city_id !== selectedCityId) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
});
|
});
|
||||||
}, [sights, selectedCityStore.selectedCityId, canReadCities, authStore.meCities]);
|
}, [sights, selectedCityId, canReadCities, authStore.meCities]);
|
||||||
|
|
||||||
const query = searchQuery.trim().toLowerCase();
|
const query = searchQuery.trim().toLowerCase();
|
||||||
const rows = filteredSights
|
const rows = filteredSights
|
||||||
@@ -177,7 +181,9 @@ export const SightListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@@ -185,6 +191,7 @@ export const SightListPage = observer(() => {
|
|||||||
onRowDoubleClick={(params) => canWriteSights && navigate(`/sight/${params.row.id}/edit`)}
|
onRowDoubleClick={(params) => canWriteSights && navigate(`/sight/${params.row.id}/edit`)}
|
||||||
checkboxSelection={canWriteSights}
|
checkboxSelection={canWriteSights}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
@@ -215,6 +222,8 @@ export const SightListPage = observer(() => {
|
|||||||
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
<CircularProgress size={20} />
|
<CircularProgress size={20} />
|
||||||
|
) : !selectedCityId ? (
|
||||||
|
"Выберите город"
|
||||||
) : (
|
) : (
|
||||||
"Нет достопримечательностей"
|
"Нет достопримечательностей"
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,17 +1,155 @@
|
|||||||
import { Button, TextField } from "@mui/material";
|
import {
|
||||||
import { snapshotStore } from "@shared";
|
Button,
|
||||||
|
TextField,
|
||||||
|
Dialog,
|
||||||
|
DialogTitle,
|
||||||
|
DialogContent,
|
||||||
|
DialogActions,
|
||||||
|
} from "@mui/material";
|
||||||
|
import { snapshotStore, authStore, routeStore, selectedCityStore, cityStore, carrierStore } from "@shared";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { ArrowLeft, Loader2, Save } from "lucide-react";
|
import { ArrowLeft, Loader2, Save } from "lucide-react";
|
||||||
import { useState } from "react";
|
import { useState, useEffect, useMemo } from "react";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import { runInAction } from "mobx";
|
import { runInAction } from "mobx";
|
||||||
|
|
||||||
|
function escapeRegex(s: string) {
|
||||||
|
return s.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildExportNameRegex(cityNames: string[]): RegExp {
|
||||||
|
if (!cityNames.length) return /.+/;
|
||||||
|
const pattern = cityNames.map(escapeRegex).join("|");
|
||||||
|
return new RegExp(`^(${pattern})_.+$`);
|
||||||
|
}
|
||||||
|
|
||||||
export const SnapshotCreatePage = observer(() => {
|
export const SnapshotCreatePage = observer(() => {
|
||||||
const { createSnapshot, getSnapshotStatus, getStorageInfo, snapshotStatus } = snapshotStore;
|
const { createSnapshot, getSnapshotStatus, getStorageInfo, snapshotStatus } = snapshotStore;
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [name, setName] = useState("");
|
const [name, setName] = useState("");
|
||||||
|
const [nameError, setNameError] = useState<string | null>(null);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [duplicateWarningOpen, setDuplicateWarningOpen] = useState(false);
|
||||||
|
const [duplicateRouteNumbers, setDuplicateRouteNumbers] = useState<string[]>([]);
|
||||||
|
|
||||||
|
const exportNameRegex = useMemo(() => {
|
||||||
|
const names = cityStore.cities["ru"].data.map((c) => c.name.trim());
|
||||||
|
return buildExportNameRegex(names);
|
||||||
|
}, [cityStore.cities["ru"].data.length]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!cityStore.cities["ru"].loaded) {
|
||||||
|
cityStore.getCities("ru");
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const canReadRoutes = authStore.canRead("routes");
|
||||||
|
|
||||||
|
const startExport = async () => {
|
||||||
|
try {
|
||||||
|
setIsLoading(true);
|
||||||
|
const id = await createSnapshot(name);
|
||||||
|
|
||||||
|
await getSnapshotStatus(id);
|
||||||
|
|
||||||
|
while (snapshotStore.snapshotStatus?.Status != "done") {
|
||||||
|
await new Promise((resolve) => setTimeout(resolve, 1000));
|
||||||
|
await getSnapshotStatus(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (snapshotStore.snapshotStatus?.Status === "done") {
|
||||||
|
toast.success("Экспорт медиа успешно создан");
|
||||||
|
|
||||||
|
runInAction(() => {
|
||||||
|
snapshotStore.snapshotStatus = null;
|
||||||
|
});
|
||||||
|
|
||||||
|
await getStorageInfo();
|
||||||
|
navigate(-1);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
toast.error("Ошибка при создании экспорта медиа");
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSave = async () => {
|
||||||
|
if (!canReadRoutes) {
|
||||||
|
await startExport();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
runInAction(() => {
|
||||||
|
routeStore.routes.loaded = false;
|
||||||
|
});
|
||||||
|
await routeStore.getRoutes();
|
||||||
|
await carrierStore.getCarriers("ru");
|
||||||
|
|
||||||
|
const routes = routeStore.routes.data;
|
||||||
|
const carriers = carrierStore.carriers.ru.data;
|
||||||
|
const carrierCityMap = new Map<number, number>();
|
||||||
|
for (const c of carriers) {
|
||||||
|
carrierCityMap.set(c.id, c.city_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
const duplicateMessages: string[] = [];
|
||||||
|
|
||||||
|
const directionKey = new Map<string, number>();
|
||||||
|
for (const route of routes) {
|
||||||
|
const num = (route.route_sys_number ?? "").trim();
|
||||||
|
if (!num) continue;
|
||||||
|
const cityId = carrierCityMap.get(route.carrier_id) ?? 0;
|
||||||
|
const key = `${num}|${route.route_direction}|${cityId}`;
|
||||||
|
directionKey.set(key, (directionKey.get(key) ?? 0) + 1);
|
||||||
|
}
|
||||||
|
for (const [key, count] of directionKey) {
|
||||||
|
if (count > 1) {
|
||||||
|
const [num, dir] = key.split("|");
|
||||||
|
const dirLabel = dir === "true" ? "прямой" : "обратный";
|
||||||
|
duplicateMessages.push(
|
||||||
|
`Дублируется маршрут №${num} (${dirLabel})`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const cityPerNumber = new Map<string, Set<number>>();
|
||||||
|
for (const route of routes) {
|
||||||
|
const num = (route.route_sys_number ?? "").trim();
|
||||||
|
if (!num) continue;
|
||||||
|
const cityId = carrierCityMap.get(route.carrier_id) ?? 0;
|
||||||
|
if (!cityPerNumber.has(num)) {
|
||||||
|
cityPerNumber.set(num, new Set());
|
||||||
|
}
|
||||||
|
cityPerNumber.get(num)!.add(cityId);
|
||||||
|
}
|
||||||
|
for (const [num, cities] of cityPerNumber) {
|
||||||
|
if (cities.size > 1) {
|
||||||
|
duplicateMessages.push(
|
||||||
|
`Маршрут №${num} присутствует в нескольких городах`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (duplicateMessages.length > 0) {
|
||||||
|
setDuplicateRouteNumbers(duplicateMessages);
|
||||||
|
setDuplicateWarningOpen(true);
|
||||||
|
} else {
|
||||||
|
await startExport();
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
await startExport();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="w-full h-[400px] flex justify-center items-center">
|
<div className="w-full h-[400px] flex justify-center items-center">
|
||||||
@@ -32,7 +170,19 @@ export const SnapshotCreatePage = observer(() => {
|
|||||||
label="Название"
|
label="Название"
|
||||||
required
|
required
|
||||||
value={name}
|
value={name}
|
||||||
onChange={(e) => setName(e.target.value)}
|
error={!!nameError}
|
||||||
|
helperText={nameError ?? " "}
|
||||||
|
onChange={(e) => {
|
||||||
|
const val = e.target.value;
|
||||||
|
setName(val);
|
||||||
|
const trimmed = val.trim();
|
||||||
|
const hasFullFormat = trimmed.includes("_") && trimmed.split("_").slice(1).join("_").length > 0;
|
||||||
|
if (hasFullFormat && !exportNameRegex.test(trimmed)) {
|
||||||
|
setNameError("Название должно начинаться с названия существующего города");
|
||||||
|
} else {
|
||||||
|
setNameError(null);
|
||||||
|
}
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
@@ -40,36 +190,8 @@ export const SnapshotCreatePage = observer(() => {
|
|||||||
color="primary"
|
color="primary"
|
||||||
className="w-min flex gap-2 items-center"
|
className="w-min flex gap-2 items-center"
|
||||||
startIcon={<Save size={20} />}
|
startIcon={<Save size={20} />}
|
||||||
onClick={async () => {
|
onClick={handleSave}
|
||||||
try {
|
disabled={isLoading || !exportNameRegex.test(name.trim())}
|
||||||
setIsLoading(true);
|
|
||||||
const id = await createSnapshot(name);
|
|
||||||
|
|
||||||
await getSnapshotStatus(id);
|
|
||||||
|
|
||||||
while (snapshotStore.snapshotStatus?.Status != "done") {
|
|
||||||
await new Promise((resolve) => setTimeout(resolve, 1000));
|
|
||||||
await getSnapshotStatus(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (snapshotStore.snapshotStatus?.Status === "done") {
|
|
||||||
toast.success("Экспорт медиа успешно создан");
|
|
||||||
|
|
||||||
runInAction(() => {
|
|
||||||
snapshotStore.snapshotStatus = null;
|
|
||||||
});
|
|
||||||
|
|
||||||
await getStorageInfo();
|
|
||||||
navigate(-1);
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error(error);
|
|
||||||
toast.error("Ошибка при создании экспорта медиа");
|
|
||||||
} finally {
|
|
||||||
setIsLoading(false);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
disabled={isLoading || !name.trim()}
|
|
||||||
>
|
>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
@@ -87,6 +209,45 @@ export const SnapshotCreatePage = observer(() => {
|
|||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<Dialog
|
||||||
|
open={duplicateWarningOpen}
|
||||||
|
onClose={() => !isLoading && setDuplicateWarningOpen(false)}
|
||||||
|
maxWidth="sm"
|
||||||
|
fullWidth
|
||||||
|
>
|
||||||
|
<DialogTitle>Найдены повторяющиеся маршруты</DialogTitle>
|
||||||
|
<DialogContent>
|
||||||
|
<p className="mb-3">
|
||||||
|
Обнаружены маршруты с одинаковыми номерами трассы. Это может привести к
|
||||||
|
некорректным данным в экспорте.
|
||||||
|
</p>
|
||||||
|
<ul className="list-disc pl-5">
|
||||||
|
{duplicateRouteNumbers.map((msg, i) => (
|
||||||
|
<li key={i}>{msg}</li>
|
||||||
|
))}
|
||||||
|
</ul>
|
||||||
|
</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button
|
||||||
|
onClick={() => setDuplicateWarningOpen(false)}
|
||||||
|
disabled={isLoading}
|
||||||
|
>
|
||||||
|
Отмена
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
sx={{ backgroundColor: "#795548", "&:hover": { backgroundColor: "#5D4037" } }}
|
||||||
|
disabled={isLoading}
|
||||||
|
onClick={async () => {
|
||||||
|
setDuplicateWarningOpen(false);
|
||||||
|
await startExport();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Продолжить экспорт
|
||||||
|
</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,14 +1,24 @@
|
|||||||
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
||||||
import { ruRU } from "@mui/x-data-grid/locales";
|
import { ruRU } from "@mui/x-data-grid/locales";
|
||||||
import { authStore, languageStore, snapshotStore, SearchInput } from "@shared";
|
import { authStore, languageStore, snapshotStore, cityStore, vehicleStore, SearchInput } from "@shared";
|
||||||
import { useEffect, useState, useMemo } from "react";
|
import { useEffect, useState, useMemo } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { DatabaseBackup, Trash2 } from "lucide-react";
|
import { DatabaseBackup, Trash2 } from "lucide-react";
|
||||||
import { CreateButton, DeleteModal, SnapshotRestore } from "@widgets";
|
import { CreateButton, DeleteModal, SnapshotRestore } from "@widgets";
|
||||||
import { Alert, Box, CircularProgress } from "@mui/material";
|
import { Alert, Box, Button, CircularProgress, Dialog, DialogActions, DialogContent, DialogTitle, TextField, Typography } from "@mui/material";
|
||||||
|
|
||||||
const LOW_STORAGE_THRESHOLD_GB = 10;
|
const LOW_STORAGE_THRESHOLD_GB = 10;
|
||||||
|
|
||||||
|
function escapeRegex(s: string) {
|
||||||
|
return s.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildExportNameRegex(cityNames: string[]): RegExp {
|
||||||
|
if (!cityNames.length) return /.+/;
|
||||||
|
const pattern = cityNames.map(escapeRegex).join("|");
|
||||||
|
return new RegExp(`^(${pattern})_.+$`);
|
||||||
|
}
|
||||||
|
|
||||||
const SEGMENT_COLORS = [
|
const SEGMENT_COLORS = [
|
||||||
"#FF3B30",
|
"#FF3B30",
|
||||||
"#FF9500",
|
"#FF9500",
|
||||||
@@ -30,18 +40,31 @@ export const SnapshotListPage = observer(() => {
|
|||||||
restoreSnapshot,
|
restoreSnapshot,
|
||||||
storageInfo,
|
storageInfo,
|
||||||
getStorageInfo,
|
getStorageInfo,
|
||||||
|
createEmptySnapshot,
|
||||||
} = snapshotStore;
|
} = snapshotStore;
|
||||||
const canWriteDevices = authStore.canWrite("devices");
|
const canWriteDevices = authStore.canWrite("devices");
|
||||||
|
const canReadDevices = authStore.canRead("devices");
|
||||||
const canCreateSnapshot =
|
const canCreateSnapshot =
|
||||||
authStore.hasRole("snapshot_create") && canWriteDevices;
|
authStore.hasRole("snapshot_create") && canWriteDevices;
|
||||||
const canManageSnapshots = authStore.canWrite("snapshot") && canWriteDevices;
|
const canManageSnapshots = authStore.canWrite("snapshot") && canWriteDevices;
|
||||||
|
|
||||||
const [isDeleteModalOpen, setIsDeleteModalOpen] = useState(false);
|
const [isDeleteModalOpen, setIsDeleteModalOpen] = useState(false);
|
||||||
|
const [isSnapshotOnDeviceWarning, setIsSnapshotOnDeviceWarning] = useState(false);
|
||||||
|
const [devicesWithSnapshot, setDevicesWithSnapshot] = useState<string[]>([]);
|
||||||
const [rowId, setRowId] = useState<string | null>(null);
|
const [rowId, setRowId] = useState<string | null>(null);
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
const [isRestoreModalOpen, setIsRestoreModalOpen] = useState(false);
|
const [isRestoreModalOpen, setIsRestoreModalOpen] = useState(false);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [searchQuery, setSearchQuery] = useState("");
|
const [searchQuery, setSearchQuery] = useState("");
|
||||||
|
const [isEmptySnapshotModalOpen, setIsEmptySnapshotModalOpen] = useState(false);
|
||||||
|
const [emptySnapshotName, setEmptySnapshotName] = useState("");
|
||||||
|
const [emptySnapshotNameError, setEmptySnapshotNameError] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const exportNameRegex = useMemo(() => {
|
||||||
|
const names = cityStore.cities["ru"].data.map((c) => c.name.trim());
|
||||||
|
return buildExportNameRegex(names);
|
||||||
|
}, [cityStore.cities["ru"].data.length]);
|
||||||
|
const [isCreatingEmpty, setIsCreatingEmpty] = useState(false);
|
||||||
const [paginationModel, setPaginationModel] = useState({
|
const [paginationModel, setPaginationModel] = useState({
|
||||||
page: 0,
|
page: 0,
|
||||||
pageSize: 50,
|
pageSize: 50,
|
||||||
@@ -57,7 +80,11 @@ export const SnapshotListPage = observer(() => {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const fetchSnapshots = async () => {
|
const fetchSnapshots = async () => {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
await Promise.all([getSnapshots(), getStorageInfo()]);
|
const promises: Promise<void>[] = [getSnapshots(), getStorageInfo()];
|
||||||
|
if (canReadDevices && !vehicleStore.vehicles.loaded) {
|
||||||
|
promises.push(vehicleStore.getVehicles());
|
||||||
|
}
|
||||||
|
await Promise.all(promises);
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
};
|
};
|
||||||
fetchSnapshots();
|
fetchSnapshots();
|
||||||
@@ -72,6 +99,26 @@ export const SnapshotListPage = observer(() => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const columns: GridColDef[] = [
|
const columns: GridColDef[] = [
|
||||||
|
{
|
||||||
|
field: "color",
|
||||||
|
headerName: "",
|
||||||
|
width: 28,
|
||||||
|
sortable: false,
|
||||||
|
disableColumnMenu: true,
|
||||||
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
|
<div className="flex items-center justify-center h-full w-full">
|
||||||
|
<span
|
||||||
|
style={{
|
||||||
|
display: "inline-block",
|
||||||
|
width: 12,
|
||||||
|
height: 12,
|
||||||
|
backgroundColor: params.value,
|
||||||
|
borderRadius: "50%",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
},
|
||||||
{
|
{
|
||||||
field: "name",
|
field: "name",
|
||||||
headerName: "Название",
|
headerName: "Название",
|
||||||
@@ -113,6 +160,7 @@ export const SnapshotListPage = observer(() => {
|
|||||||
renderCell: (params: GridRenderCellParams) => (
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
<button
|
<button
|
||||||
|
title="Восстановить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsRestoreModalOpen(true);
|
setIsRestoreModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -121,9 +169,22 @@ export const SnapshotListPage = observer(() => {
|
|||||||
<DatabaseBackup size={20} className="text-blue-500" />
|
<DatabaseBackup size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
|
const snapshotId = params.row.id;
|
||||||
|
if (canReadDevices) {
|
||||||
|
const devicesUsing = vehicleStore.vehicles.data
|
||||||
|
.filter(v => v.vehicle.current_snapshot_uuid === snapshotId)
|
||||||
|
.map(v => v.vehicle.tail_number || v.vehicle.uuid || `ID ${v.vehicle.id}`);
|
||||||
|
if (devicesUsing.length > 0) {
|
||||||
|
setDevicesWithSnapshot(devicesUsing);
|
||||||
|
setIsSnapshotOnDeviceWarning(true);
|
||||||
|
setRowId(snapshotId);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(snapshotId);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Trash2 size={20} className="text-red-500" />
|
<Trash2 size={20} className="text-red-500" />
|
||||||
@@ -146,12 +207,13 @@ export const SnapshotListPage = observer(() => {
|
|||||||
.toLowerCase()
|
.toLowerCase()
|
||||||
.includes(query),
|
.includes(query),
|
||||||
)
|
)
|
||||||
.map((snapshot) => ({
|
.map((snapshot, index) => ({
|
||||||
id: snapshot.ID,
|
id: snapshot.ID,
|
||||||
name: snapshot.Name,
|
name: snapshot.Name,
|
||||||
parent: snapshots.find((s) => s.ID === snapshot.ParentID)?.Name,
|
parent: snapshots.find((s) => s.ID === snapshot.ParentID)?.Name,
|
||||||
created_at: formatCreationTime(snapshot.CreationTime),
|
created_at: formatCreationTime(snapshot.CreationTime),
|
||||||
occupied_disk_space_gb: snapshot.occupied_disk_space_gb,
|
occupied_disk_space_gb: snapshot.occupied_disk_space_gb,
|
||||||
|
color: SEGMENT_COLORS[index % SEGMENT_COLORS.length],
|
||||||
}));
|
}));
|
||||||
}, [snapshots, searchQuery]);
|
}, [snapshots, searchQuery]);
|
||||||
|
|
||||||
@@ -167,13 +229,28 @@ export const SnapshotListPage = observer(() => {
|
|||||||
<div className="flex justify-between items-center mb-10">
|
<div className="flex justify-between items-center mb-10">
|
||||||
<h1 className="text-2xl ">Экспорт Медиа</h1>
|
<h1 className="text-2xl ">Экспорт Медиа</h1>
|
||||||
|
|
||||||
{canCreateSnapshot && (
|
<div className="flex gap-3">
|
||||||
<CreateButton
|
{canCreateSnapshot && (
|
||||||
label="Создать экспорт медиа"
|
<Button
|
||||||
path="/snapshot/create"
|
variant="outlined"
|
||||||
disabled={isLowStorage}
|
disabled={isLowStorage}
|
||||||
/>
|
onClick={() => {
|
||||||
)}
|
setEmptySnapshotName("");
|
||||||
|
setEmptySnapshotNameError(null);
|
||||||
|
setIsEmptySnapshotModalOpen(true);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Создать пустой экспорт
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
{canCreateSnapshot && (
|
||||||
|
<CreateButton
|
||||||
|
label="Создать экспорт медиа"
|
||||||
|
path="/snapshot/create"
|
||||||
|
disabled={isLowStorage}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{usedGB != null && totalGB != null && (
|
{usedGB != null && totalGB != null && (
|
||||||
<div className="bg-white rounded-2xl p-5 mb-6 shadow-sm border border-gray-100">
|
<div className="bg-white rounded-2xl p-5 mb-6 shadow-sm border border-gray-100">
|
||||||
@@ -185,7 +262,7 @@ export const SnapshotListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex w-full h-3 rounded-lg overflow-hidden bg-gray-100">
|
<div className="flex w-full h-3 rounded-lg overflow-hidden bg-gray-100">
|
||||||
{rows.map((row, i) => {
|
{rows.map((row) => {
|
||||||
const pct =
|
const pct =
|
||||||
row.occupied_disk_space_gb != null && totalGB > 0
|
row.occupied_disk_space_gb != null && totalGB > 0
|
||||||
? (row.occupied_disk_space_gb / totalGB) * 100
|
? (row.occupied_disk_space_gb / totalGB) * 100
|
||||||
@@ -196,8 +273,7 @@ export const SnapshotListPage = observer(() => {
|
|||||||
key={row.id}
|
key={row.id}
|
||||||
style={{
|
style={{
|
||||||
width: `${pct}%`,
|
width: `${pct}%`,
|
||||||
backgroundColor:
|
backgroundColor: row.color,
|
||||||
SEGMENT_COLORS[i % SEGMENT_COLORS.length],
|
|
||||||
}}
|
}}
|
||||||
title={`${row.name}: ${row.occupied_disk_space_gb?.toFixed(1)} ГБ`}
|
title={`${row.name}: ${row.occupied_disk_space_gb?.toFixed(1)} ГБ`}
|
||||||
/>
|
/>
|
||||||
@@ -215,7 +291,7 @@ export const SnapshotListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-wrap gap-x-5 gap-y-1 mt-3">
|
<div className="flex flex-wrap gap-x-5 gap-y-1 mt-3">
|
||||||
{rows.map((row, i) => {
|
{rows.map((row) => {
|
||||||
if (row.occupied_disk_space_gb == null || row.occupied_disk_space_gb <= 0)
|
if (row.occupied_disk_space_gb == null || row.occupied_disk_space_gb <= 0)
|
||||||
return null;
|
return null;
|
||||||
return (
|
return (
|
||||||
@@ -225,10 +301,7 @@ export const SnapshotListPage = observer(() => {
|
|||||||
>
|
>
|
||||||
<span
|
<span
|
||||||
className="inline-block w-2.5 h-2.5 rounded-full"
|
className="inline-block w-2.5 h-2.5 rounded-full"
|
||||||
style={{
|
style={{ backgroundColor: row.color }}
|
||||||
backgroundColor:
|
|
||||||
SEGMENT_COLORS[i % SEGMENT_COLORS.length],
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
{row.name}
|
{row.name}
|
||||||
</div>
|
</div>
|
||||||
@@ -262,7 +335,9 @@ export const SnapshotListPage = observer(() => {
|
|||||||
</Alert>
|
</Alert>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@@ -301,6 +376,88 @@ export const SnapshotListPage = observer(() => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<Dialog
|
||||||
|
open={isEmptySnapshotModalOpen}
|
||||||
|
onClose={() => setIsEmptySnapshotModalOpen(false)}
|
||||||
|
fullWidth
|
||||||
|
maxWidth="xs"
|
||||||
|
>
|
||||||
|
<DialogTitle>Создать пустой экспорт</DialogTitle>
|
||||||
|
<DialogContent>
|
||||||
|
<TextField
|
||||||
|
autoFocus
|
||||||
|
fullWidth
|
||||||
|
label="Название"
|
||||||
|
value={emptySnapshotName}
|
||||||
|
error={!!emptySnapshotNameError}
|
||||||
|
helperText={emptySnapshotNameError ?? " "}
|
||||||
|
onChange={(e) => {
|
||||||
|
const val = e.target.value;
|
||||||
|
setEmptySnapshotName(val);
|
||||||
|
const trimmed = val.trim();
|
||||||
|
const hasFullFormat = trimmed.includes("_") && trimmed.split("_").slice(1).join("_").length > 0;
|
||||||
|
if (hasFullFormat && !exportNameRegex.test(trimmed)) {
|
||||||
|
setEmptySnapshotNameError("Название должно начинаться с названия существующего города");
|
||||||
|
} else {
|
||||||
|
setEmptySnapshotNameError(null);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button onClick={() => setIsEmptySnapshotModalOpen(false)}>
|
||||||
|
Отмена
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
disabled={!exportNameRegex.test(emptySnapshotName.trim()) || isCreatingEmpty}
|
||||||
|
onClick={async () => {
|
||||||
|
setIsCreatingEmpty(true);
|
||||||
|
try {
|
||||||
|
await createEmptySnapshot(emptySnapshotName);
|
||||||
|
await getSnapshots();
|
||||||
|
setIsEmptySnapshotModalOpen(false);
|
||||||
|
} finally {
|
||||||
|
setIsCreatingEmpty(false);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{isCreatingEmpty ? <CircularProgress size={20} /> : "Создать"}
|
||||||
|
</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
|
|
||||||
|
<Dialog
|
||||||
|
open={isSnapshotOnDeviceWarning}
|
||||||
|
onClose={() => setIsSnapshotOnDeviceWarning(false)}
|
||||||
|
fullWidth
|
||||||
|
maxWidth="xs"
|
||||||
|
>
|
||||||
|
<DialogTitle>Удаление невозможно</DialogTitle>
|
||||||
|
<DialogContent>
|
||||||
|
<Alert severity="warning" sx={{ mt: 1 }}>
|
||||||
|
Этот экспорт загружен на устройства. Удалите или замените экспорт на
|
||||||
|
устройствах перед удалением.
|
||||||
|
</Alert>
|
||||||
|
<Box sx={{ mt: 2 }}>
|
||||||
|
<Typography variant="body2" fontWeight={600} gutterBottom>
|
||||||
|
Устройства:
|
||||||
|
</Typography>
|
||||||
|
{devicesWithSnapshot.map((name, i) => (
|
||||||
|
<Typography key={i} variant="body2">
|
||||||
|
• {name}
|
||||||
|
</Typography>
|
||||||
|
))}
|
||||||
|
</Box>
|
||||||
|
</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button onClick={() => setIsSnapshotOnDeviceWarning(false)}>
|
||||||
|
Закрыть
|
||||||
|
</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
|
|
||||||
<SnapshotRestore
|
<SnapshotRestore
|
||||||
open={isRestoreModalOpen}
|
open={isRestoreModalOpen}
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ import {
|
|||||||
mediaStore,
|
mediaStore,
|
||||||
isMediaIdEmpty,
|
isMediaIdEmpty,
|
||||||
useSelectedCity,
|
useSelectedCity,
|
||||||
|
selectedCityStore,
|
||||||
SelectMediaDialog,
|
SelectMediaDialog,
|
||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
PreviewMediaDialog,
|
PreviewMediaDialog,
|
||||||
@@ -32,6 +33,12 @@ import {
|
|||||||
|
|
||||||
export const StationCreatePage = observer(() => {
|
export const StationCreatePage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
const {
|
const {
|
||||||
@@ -68,9 +75,9 @@ export const StationCreatePage = observer(() => {
|
|||||||
const executeCreate = async () => {
|
const executeCreate = async () => {
|
||||||
try {
|
try {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
await createStation();
|
const data = await createStation();
|
||||||
toast.success("Остановка успешно создана");
|
toast.success("Остановка успешно создана");
|
||||||
navigate("/station");
|
navigate(`/station/${data.id}/edit`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error creating station:", error);
|
console.error("Error creating station:", error);
|
||||||
toast.error("Ошибка при создании остановки");
|
toast.error("Ошибка при создании остановки");
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ import {
|
|||||||
mediaStore,
|
mediaStore,
|
||||||
isMediaIdEmpty,
|
isMediaIdEmpty,
|
||||||
LoadingSpinner,
|
LoadingSpinner,
|
||||||
|
selectedCityStore,
|
||||||
SelectMediaDialog,
|
SelectMediaDialog,
|
||||||
PreviewMediaDialog,
|
PreviewMediaDialog,
|
||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
@@ -34,6 +35,12 @@ import { LinkedSights } from "../LinkedSights";
|
|||||||
|
|
||||||
export const StationEditPage = observer(() => {
|
export const StationEditPage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [isLoadingData, setIsLoadingData] = useState(true);
|
const [isLoadingData, setIsLoadingData] = useState(true);
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
|
|||||||
@@ -49,7 +49,7 @@ export const StationListPage = observer(() => {
|
|||||||
loadSightCounts(stationIds);
|
loadSightCounts(stationIds);
|
||||||
};
|
};
|
||||||
fetchStations();
|
fetchStations();
|
||||||
}, [language]);
|
}, [language, selectedCityStore.cityVersion]);
|
||||||
|
|
||||||
const columns: GridColDef[] = [
|
const columns: GridColDef[] = [
|
||||||
{
|
{
|
||||||
@@ -86,13 +86,13 @@ export const StationListPage = observer(() => {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
field: "sightCount",
|
field: "sightCount",
|
||||||
headerName: "Достопримечательности",
|
headerName: "Привязки",
|
||||||
width: 180,
|
width: 180,
|
||||||
align: "center" as const,
|
align: "center" as const,
|
||||||
headerAlign: "center" as const,
|
headerAlign: "center" as const,
|
||||||
sortable: true,
|
sortable: true,
|
||||||
renderHeader: (params) => (
|
renderHeader: (params) => (
|
||||||
<Tooltip title="Количество привязанных достопримечательностей">
|
<Tooltip title="Отображает количество привязок">
|
||||||
<span>{params.colDef.headerName}</span>
|
<span>{params.colDef.headerName}</span>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
),
|
),
|
||||||
@@ -114,7 +114,7 @@ export const StationListPage = observer(() => {
|
|||||||
headerAlign: "center" as const,
|
headerAlign: "center" as const,
|
||||||
sortable: true,
|
sortable: true,
|
||||||
renderHeader: (params) => (
|
renderHeader: (params) => (
|
||||||
<Tooltip title="Подтверждение добавленных пересадок">
|
<Tooltip title="Отображает подтверждение добавленных пересадок">
|
||||||
<span>{params.colDef.headerName}</span>
|
<span>{params.colDef.headerName}</span>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
),
|
),
|
||||||
@@ -141,7 +141,7 @@ export const StationListPage = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
{canWriteStations && (
|
{canWriteStations && (
|
||||||
<button onClick={() => navigate(`/station/${params.row.id}/edit`)}>
|
<button title="Редактировать" onClick={() => navigate(`/station/${params.row.id}/edit`)}>
|
||||||
<Pencil size={20} className="text-blue-500" />
|
<Pencil size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
@@ -151,13 +151,14 @@ export const StationListPage = observer(() => {
|
|||||||
setSelectedStationId(params.row.id);
|
setSelectedStationId(params.row.id);
|
||||||
setIsTransfersModalOpen(true);
|
setIsTransfersModalOpen(true);
|
||||||
}}
|
}}
|
||||||
title="Редактировать пересадки"
|
title="Управление пересадками"
|
||||||
>
|
>
|
||||||
<Route size={20} className="text-purple-500" />
|
<Route size={20} className="text-purple-500" />
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
{canWriteStations && (
|
{canWriteStations && (
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -174,9 +175,12 @@ export const StationListPage = observer(() => {
|
|||||||
|
|
||||||
const rows = useMemo(() => {
|
const rows = useMemo(() => {
|
||||||
const { selectedCityId } = selectedCityStore;
|
const { selectedCityId } = selectedCityStore;
|
||||||
|
if (!selectedCityId) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
const query = searchQuery.trim().toLowerCase();
|
const query = searchQuery.trim().toLowerCase();
|
||||||
return stationLists[language].data
|
return stationLists[language].data
|
||||||
.filter((station: any) => !selectedCityId || station.city_id === selectedCityId)
|
.filter((station: any) => station.city_id === selectedCityId)
|
||||||
.filter(
|
.filter(
|
||||||
(station: any) =>
|
(station: any) =>
|
||||||
!query ||
|
!query ||
|
||||||
@@ -202,7 +206,10 @@ export const StationListPage = observer(() => {
|
|||||||
<div className="flex justify-between items-center mb-10">
|
<div className="flex justify-between items-center mb-10">
|
||||||
<h1 className="text-2xl">Остановки</h1>
|
<h1 className="text-2xl">Остановки</h1>
|
||||||
{canWriteStations && (
|
{canWriteStations && (
|
||||||
<CreateButton label="Создать остановку" path="/station/create" />
|
<CreateButton
|
||||||
|
label="Создать остановку"
|
||||||
|
path="/station/create"
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -218,7 +225,9 @@ export const StationListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@@ -226,6 +235,7 @@ export const StationListPage = observer(() => {
|
|||||||
onRowDoubleClick={(params) => canWriteStations && navigate(`/station/${params.row.id}/edit`)}
|
onRowDoubleClick={(params) => canWriteStations && navigate(`/station/${params.row.id}/edit`)}
|
||||||
checkboxSelection={canWriteStations}
|
checkboxSelection={canWriteStations}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
@@ -276,7 +286,13 @@ export const StationListPage = observer(() => {
|
|||||||
slots={{
|
slots={{
|
||||||
noRowsOverlay: () => (
|
noRowsOverlay: () => (
|
||||||
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
||||||
{isLoading ? <CircularProgress size={20} /> : "Нет остановок"}
|
{isLoading ? (
|
||||||
|
<CircularProgress size={20} />
|
||||||
|
) : !selectedCityStore.selectedCityId ? (
|
||||||
|
"Выберите город"
|
||||||
|
) : (
|
||||||
|
"Нет остановок"
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
),
|
),
|
||||||
}}
|
}}
|
||||||
|
|||||||
@@ -1,4 +1,11 @@
|
|||||||
import { Button, Paper, TextField } from "@mui/material";
|
import {
|
||||||
|
Button,
|
||||||
|
Paper,
|
||||||
|
TextField,
|
||||||
|
Typography,
|
||||||
|
Box,
|
||||||
|
Divider,
|
||||||
|
} from "@mui/material";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { ArrowLeft, Loader2, Save } from "lucide-react";
|
import { ArrowLeft, Loader2, Save } from "lucide-react";
|
||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
@@ -10,9 +17,10 @@ import {
|
|||||||
SelectMediaDialog,
|
SelectMediaDialog,
|
||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
PreviewMediaDialog,
|
PreviewMediaDialog,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { ImageUploadCard } from "@widgets";
|
import { ImageUploadCard, PermissionsTable, RolesHintTable, ROLE_RESOURCES } from "@widgets";
|
||||||
|
|
||||||
export const UserCreatePage = observer(() => {
|
export const UserCreatePage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
@@ -26,13 +34,38 @@ export const UserCreatePage = observer(() => {
|
|||||||
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
||||||
>(null);
|
>(null);
|
||||||
|
|
||||||
|
const [localRoles, setLocalRoles] = useState<string[]>(
|
||||||
|
createUserData.roles ?? ["articles_ro", "articles_rw", "media_ro", "media_rw"]
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
mediaStore.getMedia();
|
mediaStore.getMedia();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const allRw = ROLE_RESOURCES.every(({ key }) => localRoles.includes(`${key}_rw`));
|
||||||
|
const isAdmin = allRw && !localRoles.includes("devices_maintenance_rw");
|
||||||
|
if (isAdmin !== createUserData.is_admin) {
|
||||||
|
setCreateUserData(
|
||||||
|
createUserData.name || "",
|
||||||
|
createUserData.email || "",
|
||||||
|
createUserData.password || "",
|
||||||
|
isAdmin,
|
||||||
|
createUserData.icon
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}, [localRoles]);
|
||||||
|
|
||||||
const handleCreate = async () => {
|
const handleCreate = async () => {
|
||||||
try {
|
try {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
// Убеждаемся, что роли в сторе обновлены перед созданием
|
||||||
|
userStore.createUserData.roles = localRoles;
|
||||||
await createUser();
|
await createUser();
|
||||||
toast.success("Пользователь успешно создан");
|
toast.success("Пользователь успешно создан");
|
||||||
navigate("/user");
|
navigate("/user");
|
||||||
@@ -67,18 +100,15 @@ export const UserCreatePage = observer(() => {
|
|||||||
: selectedMedia?.id ?? createUserData.icon ?? null;
|
: selectedMedia?.id ?? createUserData.icon ?? null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Paper className="w-full h-full p-3 flex flex-col gap-10">
|
<Paper className="w-full p-6 flex flex-col gap-8">
|
||||||
<div className="flex items-center gap-4">
|
<button className="flex items-center gap-2 self-start" onClick={() => navigate(-1)}>
|
||||||
<button
|
<ArrowLeft size={20} />
|
||||||
className="flex items-center gap-2"
|
Назад
|
||||||
onClick={() => navigate(-1)}
|
</button>
|
||||||
>
|
|
||||||
<ArrowLeft size={20} />
|
<section className="flex flex-col gap-6">
|
||||||
Назад
|
<Typography variant="h6">Основные данные</Typography>
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="flex flex-col gap-10 w-full items-end">
|
|
||||||
<TextField
|
<TextField
|
||||||
fullWidth
|
fullWidth
|
||||||
label="Имя"
|
label="Имя"
|
||||||
@@ -116,6 +146,7 @@ export const UserCreatePage = observer(() => {
|
|||||||
label="Пароль"
|
label="Пароль"
|
||||||
value={createUserData.password || ""}
|
value={createUserData.password || ""}
|
||||||
required
|
required
|
||||||
|
type="password"
|
||||||
onChange={(e) =>
|
onChange={(e) =>
|
||||||
setCreateUserData(
|
setCreateUserData(
|
||||||
createUserData.name || "",
|
createUserData.name || "",
|
||||||
@@ -127,7 +158,7 @@ export const UserCreatePage = observer(() => {
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="w-full flex flex-col gap-4 max-w-[300px] mx-auto">
|
<div className="w-full flex flex-col gap-4 max-w-[300px]">
|
||||||
<ImageUploadCard
|
<ImageUploadCard
|
||||||
title="Аватар"
|
title="Аватар"
|
||||||
imageKey="thumbnail"
|
imageKey="thumbnail"
|
||||||
@@ -156,23 +187,72 @@ export const UserCreatePage = observer(() => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
<Button
|
<Divider />
|
||||||
variant="contained"
|
|
||||||
className="w-min flex gap-2 items-center"
|
<section className="flex flex-col gap-4">
|
||||||
startIcon={<Save size={20} />}
|
<Typography variant="h6">Права доступа</Typography>
|
||||||
onClick={handleCreate}
|
|
||||||
disabled={
|
<Box sx={{ display: "flex", gap: 1 }}>
|
||||||
isLoading || !createUserData.name || !createUserData.password
|
<Button
|
||||||
}
|
variant="outlined"
|
||||||
>
|
size="small"
|
||||||
{isLoading ? (
|
onClick={() => {
|
||||||
<Loader2 size={20} className="animate-spin" />
|
setCreateUserData(
|
||||||
) : (
|
createUserData.name || "",
|
||||||
"Создать"
|
createUserData.email || "",
|
||||||
)}
|
createUserData.password || "",
|
||||||
</Button>
|
true,
|
||||||
</div>
|
createUserData.icon
|
||||||
|
);
|
||||||
|
const next: string[] = [];
|
||||||
|
for (const { key } of ROLE_RESOURCES) {
|
||||||
|
next.push(`${key}_rw`);
|
||||||
|
}
|
||||||
|
next.push("snapshot_create");
|
||||||
|
setLocalRoles(next);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Полный доступ (admin)
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="outlined"
|
||||||
|
size="small"
|
||||||
|
onClick={() => {
|
||||||
|
setCreateUserData(
|
||||||
|
createUserData.name || "",
|
||||||
|
createUserData.email || "",
|
||||||
|
createUserData.password || "",
|
||||||
|
false,
|
||||||
|
createUserData.icon
|
||||||
|
);
|
||||||
|
setLocalRoles(["devices_ro", "vehicles_ro", "devices_maintenance_rw"]);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Администратор ТО
|
||||||
|
</Button>
|
||||||
|
</Box>
|
||||||
|
|
||||||
|
<PermissionsTable localRoles={localRoles} setLocalRoles={setLocalRoles} />
|
||||||
|
<RolesHintTable />
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="contained"
|
||||||
|
className="self-end w-min flex gap-2 items-center"
|
||||||
|
startIcon={<Save size={20} />}
|
||||||
|
onClick={handleCreate}
|
||||||
|
disabled={
|
||||||
|
isLoading || !createUserData.name || !createUserData.password || !createUserData.email
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{isLoading ? (
|
||||||
|
<Loader2 size={20} className="animate-spin" />
|
||||||
|
) : (
|
||||||
|
"Создать"
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
|
||||||
<SelectMediaDialog
|
<SelectMediaDialog
|
||||||
open={isSelectMediaOpen}
|
open={isSelectMediaOpen}
|
||||||
|
|||||||
@@ -1,18 +1,9 @@
|
|||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
FormControlLabel,
|
|
||||||
Checkbox,
|
|
||||||
Paper,
|
Paper,
|
||||||
TextField,
|
TextField,
|
||||||
Box,
|
Box,
|
||||||
Typography,
|
Typography,
|
||||||
Table,
|
|
||||||
TableBody,
|
|
||||||
TableCell,
|
|
||||||
TableHead,
|
|
||||||
TableRow,
|
|
||||||
Radio,
|
|
||||||
RadioGroup,
|
|
||||||
Divider,
|
Divider,
|
||||||
} from "@mui/material";
|
} from "@mui/material";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
@@ -31,45 +22,12 @@ import {
|
|||||||
authStore,
|
authStore,
|
||||||
cityStore,
|
cityStore,
|
||||||
MultiSelect,
|
MultiSelect,
|
||||||
|
selectedCityStore,
|
||||||
type User,
|
type User,
|
||||||
type UserCity,
|
type UserCity,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { ImageUploadCard, DeleteModal } from "@widgets";
|
import { ImageUploadCard, DeleteModal, PermissionsTable, RolesHintTable, ROLE_RESOURCES } from "@widgets";
|
||||||
|
|
||||||
const ROLE_RESOURCES = [
|
|
||||||
{ key: "snapshot", label: "Экспорт" },
|
|
||||||
{ key: "devices", label: "Устройства" },
|
|
||||||
{ key: "vehicles", label: "Транспорт" },
|
|
||||||
{ key: "users", label: "Пользователи" },
|
|
||||||
{ key: "sights", label: "Достопримечательности" },
|
|
||||||
{ key: "stations", label: "Остановки" },
|
|
||||||
{ key: "routes", label: "Маршруты" },
|
|
||||||
{ key: "countries", label: "Страны" },
|
|
||||||
{ key: "cities", label: "Города" },
|
|
||||||
{ key: "carriers", label: "Перевозчики" },
|
|
||||||
] as const;
|
|
||||||
|
|
||||||
type PermissionLevel = "none" | "ro" | "rw";
|
|
||||||
|
|
||||||
function getPermissionLevel(roles: string[], resource: string): PermissionLevel {
|
|
||||||
if (roles.includes(`${resource}_rw`)) return "rw";
|
|
||||||
if (roles.includes(`${resource}_ro`)) return "ro";
|
|
||||||
return "none";
|
|
||||||
}
|
|
||||||
|
|
||||||
function applyPermissionChange(
|
|
||||||
roles: string[],
|
|
||||||
resource: string,
|
|
||||||
level: PermissionLevel,
|
|
||||||
): string[] {
|
|
||||||
const filtered = roles.filter(
|
|
||||||
(r) => r !== `${resource}_ro` && r !== `${resource}_rw`,
|
|
||||||
);
|
|
||||||
if (level === "ro") return [...filtered, `${resource}_ro`];
|
|
||||||
if (level === "rw") return [...filtered, `${resource}_rw`];
|
|
||||||
return filtered;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const UserEditPage = observer(() => {
|
export const UserEditPage = observer(() => {
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
@@ -93,10 +51,29 @@ export const UserEditPage = observer(() => {
|
|||||||
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
"thumbnail" | "watermark_lu" | "watermark_rd" | "image" | null
|
||||||
>(null);
|
>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
languageStore.setLanguage("ru");
|
languageStore.setLanguage("ru");
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const allRw = ROLE_RESOURCES.every(({ key }) => localRoles.includes(`${key}_rw`));
|
||||||
|
const isAdmin = allRw && !localRoles.includes("devices_maintenance_rw");
|
||||||
|
if (isAdmin !== editUserData.is_admin) {
|
||||||
|
setEditUserData(
|
||||||
|
editUserData.name || "",
|
||||||
|
editUserData.email || "",
|
||||||
|
editUserData.password || "",
|
||||||
|
isAdmin,
|
||||||
|
editUserData.icon || ""
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}, [localRoles]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
(async () => {
|
(async () => {
|
||||||
if (id) {
|
if (id) {
|
||||||
@@ -311,155 +288,36 @@ export const UserEditPage = observer(() => {
|
|||||||
<section className="flex flex-col gap-4">
|
<section className="flex flex-col gap-4">
|
||||||
<Typography variant="h6">Права доступа</Typography>
|
<Typography variant="h6">Права доступа</Typography>
|
||||||
|
|
||||||
<FormControlLabel
|
<Box sx={{ display: "flex", gap: 1 }}>
|
||||||
control={
|
<Button
|
||||||
<Checkbox
|
variant="outlined"
|
||||||
checked={localRoles.includes("admin")}
|
size="small"
|
||||||
onChange={(e) => {
|
onClick={() => {
|
||||||
if (e.target.checked) {
|
setEditUserData(editUserData.name || "", editUserData.email || "", editUserData.password || "", true, editUserData.icon || "");
|
||||||
setLocalRoles((prev) => {
|
const next: string[] = [];
|
||||||
let next = prev.filter((r) => r !== "admin");
|
for (const { key } of ROLE_RESOURCES) {
|
||||||
for (const { key } of ROLE_RESOURCES) {
|
next.push(`${key}_rw`);
|
||||||
next = next.filter((r) => r !== `${key}_ro` && r !== `${key}_rw`);
|
}
|
||||||
next.push(`${key}_rw`);
|
next.push("snapshot_create");
|
||||||
}
|
setLocalRoles(next);
|
||||||
if (!next.includes("snapshot_create")) {
|
}}
|
||||||
next.push("snapshot_create");
|
>
|
||||||
}
|
Полный доступ (admin)
|
||||||
next.push("admin");
|
</Button>
|
||||||
return next;
|
<Button
|
||||||
});
|
variant="outlined"
|
||||||
} else {
|
size="small"
|
||||||
setLocalRoles((prev) => prev.filter((r) => r !== "admin"));
|
onClick={() => {
|
||||||
}
|
setEditUserData(editUserData.name || "", editUserData.email || "", editUserData.password || "", false, editUserData.icon || "");
|
||||||
}}
|
setLocalRoles(["devices_ro", "vehicles_ro", "devices_maintenance_rw"]);
|
||||||
/>
|
}}
|
||||||
}
|
>
|
||||||
label="Полный доступ (admin)"
|
Администратор ТО
|
||||||
/>
|
</Button>
|
||||||
|
|
||||||
<Box sx={{ border: "1px solid", borderColor: "divider", borderRadius: 1 }}>
|
|
||||||
<Table size="small">
|
|
||||||
<TableHead>
|
|
||||||
<TableRow sx={{ bgcolor: "action.hover" }}>
|
|
||||||
<TableCell sx={{ fontWeight: 600, width: 220 }}>Ресурс</TableCell>
|
|
||||||
<TableCell align="center" sx={{ fontWeight: 600 }}>Нет доступа</TableCell>
|
|
||||||
<TableCell align="center" sx={{ fontWeight: 600 }}>Чтение</TableCell>
|
|
||||||
<TableCell align="center" sx={{ fontWeight: 600 }}>Чтение/Запись</TableCell>
|
|
||||||
<TableCell align="center" sx={{ fontWeight: 600 }}>
|
|
||||||
Создание (snapshot_create)
|
|
||||||
</TableCell>
|
|
||||||
</TableRow>
|
|
||||||
</TableHead>
|
|
||||||
<TableBody>
|
|
||||||
{ROLE_RESOURCES.map(({ key, label }) => {
|
|
||||||
const level = getPermissionLevel(localRoles, key);
|
|
||||||
const isSnapshotResource = key === "snapshot";
|
|
||||||
|
|
||||||
const handleChange = (val: string) => {
|
|
||||||
setLocalRoles((prev) => {
|
|
||||||
let updated = applyPermissionChange(prev, key, val as PermissionLevel);
|
|
||||||
|
|
||||||
if (key === "devices") {
|
|
||||||
updated = applyPermissionChange(
|
|
||||||
updated,
|
|
||||||
"vehicles",
|
|
||||||
val as PermissionLevel,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const allRw = ROLE_RESOURCES.every(({ key: k }) =>
|
|
||||||
updated.includes(`${k}_rw`),
|
|
||||||
);
|
|
||||||
if (allRw && !updated.includes("admin")) {
|
|
||||||
const next = [...updated];
|
|
||||||
if (!next.includes("snapshot_create")) {
|
|
||||||
next.push("snapshot_create");
|
|
||||||
}
|
|
||||||
next.push("admin");
|
|
||||||
return next;
|
|
||||||
}
|
|
||||||
if (!allRw) {
|
|
||||||
return updated.filter((r) => r !== "admin");
|
|
||||||
}
|
|
||||||
return updated;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleSnapshotCreateChange = (checked: boolean) => {
|
|
||||||
if (!isSnapshotResource) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setLocalRoles((prev) => {
|
|
||||||
const withoutSnapshotCreate = prev.filter(
|
|
||||||
(role) => role !== "snapshot_create"
|
|
||||||
);
|
|
||||||
return checked
|
|
||||||
? [...withoutSnapshotCreate, "snapshot_create"]
|
|
||||||
: withoutSnapshotCreate;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<TableRow key={key} hover>
|
|
||||||
<TableCell>{label}</TableCell>
|
|
||||||
<TableCell align="center" padding="checkbox">
|
|
||||||
<RadioGroup
|
|
||||||
row
|
|
||||||
value={level}
|
|
||||||
onChange={(e) => handleChange(e.target.value)}
|
|
||||||
sx={{ justifyContent: "center", flexWrap: "nowrap" }}
|
|
||||||
>
|
|
||||||
<Radio value="none" size="small" />
|
|
||||||
</RadioGroup>
|
|
||||||
</TableCell>
|
|
||||||
<TableCell align="center" padding="checkbox">
|
|
||||||
{isSnapshotResource ? (
|
|
||||||
<Typography variant="body2" color="text.secondary">
|
|
||||||
-
|
|
||||||
</Typography>
|
|
||||||
) : (
|
|
||||||
<RadioGroup
|
|
||||||
row
|
|
||||||
value={level}
|
|
||||||
onChange={(e) => handleChange(e.target.value)}
|
|
||||||
sx={{ justifyContent: "center", flexWrap: "nowrap" }}
|
|
||||||
>
|
|
||||||
<Radio value="ro" size="small" />
|
|
||||||
</RadioGroup>
|
|
||||||
)}
|
|
||||||
</TableCell>
|
|
||||||
<TableCell align="center" padding="checkbox">
|
|
||||||
<RadioGroup
|
|
||||||
row
|
|
||||||
value={level}
|
|
||||||
onChange={(e) => handleChange(e.target.value)}
|
|
||||||
sx={{ justifyContent: "center", flexWrap: "nowrap" }}
|
|
||||||
>
|
|
||||||
<Radio value="rw" size="small" />
|
|
||||||
</RadioGroup>
|
|
||||||
</TableCell>
|
|
||||||
<TableCell align="center" padding="checkbox">
|
|
||||||
{isSnapshotResource ? (
|
|
||||||
<Checkbox
|
|
||||||
checked={localRoles.includes("snapshot_create")}
|
|
||||||
onChange={(e) =>
|
|
||||||
handleSnapshotCreateChange(e.target.checked)
|
|
||||||
}
|
|
||||||
size="small"
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<Typography variant="body2" color="text.secondary">
|
|
||||||
-
|
|
||||||
</Typography>
|
|
||||||
)}
|
|
||||||
</TableCell>
|
|
||||||
</TableRow>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</TableBody>
|
|
||||||
</Table>
|
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
|
<PermissionsTable localRoles={localRoles} setLocalRoles={setLocalRoles} />
|
||||||
|
<RolesHintTable />
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<Divider />
|
<Divider />
|
||||||
|
|||||||
@@ -92,10 +92,11 @@ export const UserListPage = observer(() => {
|
|||||||
sortable: false,
|
sortable: false,
|
||||||
renderCell: (params: GridRenderCellParams) => (
|
renderCell: (params: GridRenderCellParams) => (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
<button onClick={() => navigate(`/user/${params.row.id}/edit`)}>
|
<button title="Редактировать" onClick={() => navigate(`/user/${params.row.id}/edit`)}>
|
||||||
<Pencil size={20} className="text-blue-500" />
|
<Pencil size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -146,13 +147,21 @@ export const UserListPage = observer(() => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
<DataGrid
|
<DataGrid
|
||||||
rows={rows}
|
rows={rows}
|
||||||
columns={columns}
|
columns={columns}
|
||||||
checkboxSelection={canWriteUsers}
|
checkboxSelection={canWriteUsers}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
|
onRowDoubleClick={(params) => {
|
||||||
|
if (canWriteUsers) {
|
||||||
|
navigate(`/user/${params.row.id}/edit`);
|
||||||
|
}
|
||||||
|
}}
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
|
|||||||
@@ -12,6 +12,8 @@ import {
|
|||||||
VEHICLE_TYPES,
|
VEHICLE_TYPES,
|
||||||
carrierStore,
|
carrierStore,
|
||||||
languageStore,
|
languageStore,
|
||||||
|
cityStore,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { ArrowLeft, Save } from "lucide-react";
|
import { ArrowLeft, Save } from "lucide-react";
|
||||||
@@ -26,11 +28,18 @@ export const VehicleCreatePage = observer(() => {
|
|||||||
const [type, setType] = useState("");
|
const [type, setType] = useState("");
|
||||||
const [carrierId, setCarrierId] = useState<number | null>(null);
|
const [carrierId, setCarrierId] = useState<number | null>(null);
|
||||||
const [model, setModel] = useState("");
|
const [model, setModel] = useState("");
|
||||||
|
const [cityId, setCityId] = useState<number | null>(selectedCityStore.selectedCityId);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const { language } = languageStore;
|
const { language } = languageStore;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
carrierStore.getCarriers(language);
|
carrierStore.getCarriers(language);
|
||||||
|
cityStore.getCities("ru");
|
||||||
}, [language]);
|
}, [language]);
|
||||||
|
|
||||||
const handleCreate = async () => {
|
const handleCreate = async () => {
|
||||||
@@ -43,6 +52,7 @@ export const VehicleCreatePage = observer(() => {
|
|||||||
?.full_name as string,
|
?.full_name as string,
|
||||||
carrierId!,
|
carrierId!,
|
||||||
model || undefined,
|
model || undefined,
|
||||||
|
cityId ?? undefined,
|
||||||
);
|
);
|
||||||
toast.success("Транспорт успешно создан");
|
toast.success("Транспорт успешно создан");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -73,12 +83,11 @@ export const VehicleCreatePage = observer(() => {
|
|||||||
onChange={(e) => setTailNumber(e.target.value)}
|
onChange={(e) => setTailNumber(e.target.value)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<FormControl fullWidth>
|
<FormControl fullWidth required>
|
||||||
<InputLabel>Тип</InputLabel>
|
<InputLabel>Тип</InputLabel>
|
||||||
<Select
|
<Select
|
||||||
value={type}
|
value={type}
|
||||||
label="Тип"
|
label="Тип"
|
||||||
required
|
|
||||||
onChange={(e) => setType(e.target.value)}
|
onChange={(e) => setType(e.target.value)}
|
||||||
>
|
>
|
||||||
{VEHICLE_TYPES.map((type) => (
|
{VEHICLE_TYPES.map((type) => (
|
||||||
@@ -89,12 +98,11 @@ export const VehicleCreatePage = observer(() => {
|
|||||||
</Select>
|
</Select>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
|
||||||
<FormControl fullWidth>
|
<FormControl fullWidth required>
|
||||||
<InputLabel>Перевозчик</InputLabel>
|
<InputLabel>Перевозчик</InputLabel>
|
||||||
<Select
|
<Select
|
||||||
value={carrierId || ""}
|
value={carrierId || ""}
|
||||||
label="Перевозчик"
|
label="Перевозчик"
|
||||||
required
|
|
||||||
onChange={(e) => setCarrierId(e.target.value as number)}
|
onChange={(e) => setCarrierId(e.target.value as number)}
|
||||||
>
|
>
|
||||||
{carrierStore.carriers[language].data?.map((carrier) => (
|
{carrierStore.carriers[language].data?.map((carrier) => (
|
||||||
@@ -113,12 +121,27 @@ export const VehicleCreatePage = observer(() => {
|
|||||||
placeholder="Произвольное название модели"
|
placeholder="Произвольное название модели"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<FormControl fullWidth required>
|
||||||
|
<InputLabel>Город</InputLabel>
|
||||||
|
<Select
|
||||||
|
value={cityId ?? ""}
|
||||||
|
label="Город"
|
||||||
|
onChange={(e) => setCityId(e.target.value ? Number(e.target.value) : null)}
|
||||||
|
>
|
||||||
|
{cityStore.cities.ru.data.map((city) => (
|
||||||
|
<MenuItem key={city.id} value={city.id}>
|
||||||
|
{city.name}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</Select>
|
||||||
|
</FormControl>
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
variant="contained"
|
variant="contained"
|
||||||
className="w-min flex gap-2 items-center"
|
className="w-min flex gap-2 items-center"
|
||||||
startIcon={<Save size={20} />}
|
startIcon={<Save size={20} />}
|
||||||
onClick={handleCreate}
|
onClick={handleCreate}
|
||||||
disabled={isLoading || !tailNumber || !type || !carrierId}
|
disabled={isLoading || !tailNumber || !type || !carrierId || !cityId}
|
||||||
>
|
>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
<Loader2 size={20} className="animate-spin" />
|
<Loader2 size={20} className="animate-spin" />
|
||||||
|
|||||||
@@ -20,6 +20,8 @@ import {
|
|||||||
VEHICLE_TYPES,
|
VEHICLE_TYPES,
|
||||||
vehicleStore,
|
vehicleStore,
|
||||||
LoadingSpinner,
|
LoadingSpinner,
|
||||||
|
cityStore,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
@@ -38,6 +40,11 @@ export const VehicleEditPage = observer(() => {
|
|||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [isLoadingData, setIsLoadingData] = useState(true);
|
const [isLoadingData, setIsLoadingData] = useState(true);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
selectedCityStore.setIsLocked(true);
|
||||||
|
return () => selectedCityStore.setIsLocked(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// Устанавливаем русский язык при загрузке страницы
|
// Устанавливаем русский язык при загрузке страницы
|
||||||
languageStore.setLanguage("ru");
|
languageStore.setLanguage("ru");
|
||||||
@@ -54,6 +61,7 @@ export const VehicleEditPage = observer(() => {
|
|||||||
try {
|
try {
|
||||||
await getVehicle(Number(id));
|
await getVehicle(Number(id));
|
||||||
await getCarriers(language);
|
await getCarriers(language);
|
||||||
|
await cityStore.getCities("ru");
|
||||||
|
|
||||||
setEditVehicleData({
|
setEditVehicleData({
|
||||||
tail_number: vehicle[Number(id)]?.vehicle.tail_number ?? "",
|
tail_number: vehicle[Number(id)]?.vehicle.tail_number ?? "",
|
||||||
@@ -63,6 +71,7 @@ export const VehicleEditPage = observer(() => {
|
|||||||
model: vehicle[Number(id)]?.vehicle.model ?? "",
|
model: vehicle[Number(id)]?.vehicle.model ?? "",
|
||||||
snapshot_update_blocked:
|
snapshot_update_blocked:
|
||||||
vehicle[Number(id)]?.vehicle.snapshot_update_blocked ?? false,
|
vehicle[Number(id)]?.vehicle.snapshot_update_blocked ?? false,
|
||||||
|
city_id: vehicle[Number(id)]?.vehicle.city_id ?? selectedCityStore.selectedCityId ?? undefined,
|
||||||
});
|
});
|
||||||
} finally {
|
} finally {
|
||||||
setIsLoadingData(false);
|
setIsLoadingData(false);
|
||||||
@@ -125,12 +134,11 @@ export const VehicleEditPage = observer(() => {
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<FormControl fullWidth>
|
<FormControl fullWidth required>
|
||||||
<InputLabel>Тип</InputLabel>
|
<InputLabel>Тип</InputLabel>
|
||||||
<Select
|
<Select
|
||||||
value={editVehicleData.type}
|
value={editVehicleData.type}
|
||||||
label="Тип"
|
label="Тип"
|
||||||
required
|
|
||||||
onChange={(e) =>
|
onChange={(e) =>
|
||||||
setEditVehicleData({ ...editVehicleData, type: e.target.value })
|
setEditVehicleData({ ...editVehicleData, type: e.target.value })
|
||||||
}
|
}
|
||||||
@@ -143,12 +151,11 @@ export const VehicleEditPage = observer(() => {
|
|||||||
</Select>
|
</Select>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
|
||||||
<FormControl fullWidth>
|
<FormControl fullWidth required>
|
||||||
<InputLabel>Перевозчик</InputLabel>
|
<InputLabel>Перевозчик</InputLabel>
|
||||||
<Select
|
<Select
|
||||||
value={editVehicleData.carrier_id}
|
value={editVehicleData.carrier_id}
|
||||||
label="Перевозчик"
|
label="Перевозчик"
|
||||||
required
|
|
||||||
onChange={(e) =>
|
onChange={(e) =>
|
||||||
setEditVehicleData({
|
setEditVehicleData({
|
||||||
...editVehicleData,
|
...editVehicleData,
|
||||||
@@ -177,6 +184,26 @@ export const VehicleEditPage = observer(() => {
|
|||||||
placeholder="Произвольное название модели"
|
placeholder="Произвольное название модели"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<FormControl fullWidth required>
|
||||||
|
<InputLabel>Город</InputLabel>
|
||||||
|
<Select
|
||||||
|
value={editVehicleData.city_id ?? ""}
|
||||||
|
label="Город"
|
||||||
|
onChange={(e) =>
|
||||||
|
setEditVehicleData({
|
||||||
|
...editVehicleData,
|
||||||
|
city_id: e.target.value ? Number(e.target.value) : undefined,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{cityStore.cities.ru.data.map((city) => (
|
||||||
|
<MenuItem key={city.id} value={city.id}>
|
||||||
|
{city.name}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</Select>
|
||||||
|
</FormControl>
|
||||||
|
|
||||||
<FormControlLabel
|
<FormControlLabel
|
||||||
control={
|
control={
|
||||||
<Checkbox
|
<Checkbox
|
||||||
@@ -202,7 +229,8 @@ export const VehicleEditPage = observer(() => {
|
|||||||
isLoading ||
|
isLoading ||
|
||||||
!editVehicleData.tail_number ||
|
!editVehicleData.tail_number ||
|
||||||
!editVehicleData.type ||
|
!editVehicleData.type ||
|
||||||
!editVehicleData.carrier_id
|
!editVehicleData.carrier_id ||
|
||||||
|
!editVehicleData.city_id
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
import { DataGrid, GridColDef, GridRenderCellParams } from "@mui/x-data-grid";
|
||||||
import { ruRU } from "@mui/x-data-grid/locales";
|
import { ruRU } from "@mui/x-data-grid/locales";
|
||||||
import { authStore, carrierStore, languageStore, vehicleStore, SearchInput } from "@shared";
|
import { authStore, carrierStore, languageStore, vehicleStore, SearchInput, selectedCityStore, cityStore } from "@shared";
|
||||||
import { useEffect, useState, useMemo } from "react";
|
import { useEffect, useState, useMemo } from "react";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { Eye, Pencil, Trash2, Minus } from "lucide-react";
|
import { Eye, Pencil, Trash2, Minus } from "lucide-react";
|
||||||
@@ -11,7 +11,7 @@ import { Box, CircularProgress } from "@mui/material";
|
|||||||
|
|
||||||
export const VehicleListPage = observer(() => {
|
export const VehicleListPage = observer(() => {
|
||||||
const { vehicles, getVehicles, deleteVehicle } = vehicleStore;
|
const { vehicles, getVehicles, deleteVehicle } = vehicleStore;
|
||||||
const { carriers, getCarriers } = carrierStore;
|
const { getCarriers } = carrierStore;
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const [isDeleteModalOpen, setIsDeleteModalOpen] = useState(false);
|
const [isDeleteModalOpen, setIsDeleteModalOpen] = useState(false);
|
||||||
const [isBulkDeleteModalOpen, setIsBulkDeleteModalOpen] = useState(false);
|
const [isBulkDeleteModalOpen, setIsBulkDeleteModalOpen] = useState(false);
|
||||||
@@ -31,10 +31,11 @@ export const VehicleListPage = observer(() => {
|
|||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
await getVehicles();
|
await getVehicles();
|
||||||
await getCarriers(language);
|
await getCarriers(language);
|
||||||
|
await cityStore.getCities("ru");
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
};
|
};
|
||||||
fetchData();
|
fetchData();
|
||||||
}, [language]);
|
}, [language, selectedCityStore.cityVersion]);
|
||||||
|
|
||||||
const columns: GridColDef[] = [
|
const columns: GridColDef[] = [
|
||||||
{
|
{
|
||||||
@@ -114,15 +115,16 @@ export const VehicleListPage = observer(() => {
|
|||||||
return (
|
return (
|
||||||
<div className="flex h-full gap-7 justify-center items-center">
|
<div className="flex h-full gap-7 justify-center items-center">
|
||||||
{canWrite && (
|
{canWrite && (
|
||||||
<button onClick={() => navigate(`/vehicle/${params.row.id}/edit`)}>
|
<button title="Редактировать" onClick={() => navigate(`/vehicle/${params.row.id}/edit`)}>
|
||||||
<Pencil size={20} className="text-blue-500" />
|
<Pencil size={20} className="text-blue-500" />
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
<button onClick={() => navigate(`/vehicle/${params.row.id}`)}>
|
<button title="Просмотр" onClick={() => navigate(`/vehicle/${params.row.id}`)}>
|
||||||
<Eye size={20} className="text-green-500" />
|
<Eye size={20} className="text-green-500" />
|
||||||
</button>
|
</button>
|
||||||
{canWrite && (
|
{canWrite && (
|
||||||
<button
|
<button
|
||||||
|
title="Удалить"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setIsDeleteModalOpen(true);
|
setIsDeleteModalOpen(true);
|
||||||
setRowId(params.row.id);
|
setRowId(params.row.id);
|
||||||
@@ -137,9 +139,13 @@ export const VehicleListPage = observer(() => {
|
|||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
const { selectedCityId } = selectedCityStore;
|
||||||
|
|
||||||
const rows = useMemo(() => {
|
const rows = useMemo(() => {
|
||||||
|
if (!selectedCityId) return [];
|
||||||
const query = searchQuery.trim().toLowerCase();
|
const query = searchQuery.trim().toLowerCase();
|
||||||
return (vehicles.data ?? [])
|
return (vehicles.data ?? [])
|
||||||
|
.filter((vehicle) => vehicle.vehicle.city_id === selectedCityId)
|
||||||
.filter(
|
.filter(
|
||||||
(vehicle) =>
|
(vehicle) =>
|
||||||
!query ||
|
!query ||
|
||||||
@@ -151,11 +157,9 @@ export const VehicleListPage = observer(() => {
|
|||||||
tail_number: vehicle.vehicle.tail_number,
|
tail_number: vehicle.vehicle.tail_number,
|
||||||
type: vehicle.vehicle.type,
|
type: vehicle.vehicle.type,
|
||||||
carrier: vehicle.vehicle.carrier,
|
carrier: vehicle.vehicle.carrier,
|
||||||
city: carriers[language].data?.find(
|
city: cityStore.cities.ru.data.find((c) => c.id === vehicle.vehicle.city_id)?.name,
|
||||||
(carrier) => carrier.id === vehicle.vehicle.carrier_id
|
|
||||||
)?.city,
|
|
||||||
}));
|
}));
|
||||||
}, [vehicles.data, carriers[language].data, searchQuery]);
|
}, [vehicles.data, selectedCityId, searchQuery]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -169,7 +173,9 @@ export const VehicleListPage = observer(() => {
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
{(rows.length > 0 || searchQuery) && (
|
||||||
|
<SearchInput value={searchQuery} onChange={setSearchQuery} />
|
||||||
|
)}
|
||||||
|
|
||||||
{canWriteVehicles && ids.length > 0 && (
|
{canWriteVehicles && ids.length > 0 && (
|
||||||
<div className="flex justify-end mb-5 duration-300">
|
<div className="flex justify-end mb-5 duration-300">
|
||||||
@@ -188,6 +194,7 @@ export const VehicleListPage = observer(() => {
|
|||||||
columns={columns}
|
columns={columns}
|
||||||
checkboxSelection={canWriteVehicles}
|
checkboxSelection={canWriteVehicles}
|
||||||
disableRowSelectionExcludeModel
|
disableRowSelectionExcludeModel
|
||||||
|
disableRowSelectionOnClick
|
||||||
loading={isLoading}
|
loading={isLoading}
|
||||||
paginationModel={paginationModel}
|
paginationModel={paginationModel}
|
||||||
onPaginationModelChange={setPaginationModel}
|
onPaginationModelChange={setPaginationModel}
|
||||||
@@ -222,6 +229,8 @@ export const VehicleListPage = observer(() => {
|
|||||||
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
<Box sx={{ mt: 5, textAlign: "center", color: "text.secondary" }}>
|
||||||
{isLoading ? (
|
{isLoading ? (
|
||||||
<CircularProgress size={20} />
|
<CircularProgress size={20} />
|
||||||
|
) : !selectedCityId ? (
|
||||||
|
"Выберите город"
|
||||||
) : (
|
) : (
|
||||||
"Нет транспортных средств"
|
"Нет транспортных средств"
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { authStore } from "@shared";
|
import { authStore, snapshotStore } from "@shared";
|
||||||
import {
|
import {
|
||||||
Power,
|
Power,
|
||||||
LucideIcon,
|
LucideIcon,
|
||||||
@@ -12,7 +12,9 @@ import {
|
|||||||
Split,
|
Split,
|
||||||
PersonStanding,
|
PersonStanding,
|
||||||
Cpu,
|
Cpu,
|
||||||
|
RefreshCcw,
|
||||||
} from "lucide-react";
|
} from "lucide-react";
|
||||||
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
import carrierIcon from "./carrier.svg";
|
import carrierIcon from "./carrier.svg";
|
||||||
|
|
||||||
@@ -165,6 +167,15 @@ export const NAVIGATION_ITEMS: {
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
secondary: [
|
secondary: [
|
||||||
|
{
|
||||||
|
id: "clear-cache",
|
||||||
|
label: "Очистить кэш",
|
||||||
|
icon: RefreshCcw,
|
||||||
|
onClick: () => {
|
||||||
|
snapshotStore.clearStoreCache();
|
||||||
|
toast.success("Кэш очищен");
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
id: "logout",
|
id: "logout",
|
||||||
label: "Выйти",
|
label: "Выйти",
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import {
|
|||||||
SelectMediaDialog,
|
SelectMediaDialog,
|
||||||
UploadMediaDialog,
|
UploadMediaDialog,
|
||||||
Language,
|
Language,
|
||||||
|
selectedCityStore,
|
||||||
} from "@shared";
|
} from "@shared";
|
||||||
import { observer } from "mobx-react-lite";
|
import { observer } from "mobx-react-lite";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
@@ -232,6 +233,7 @@ export const ArticleSelectOrCreateDialog = observer(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const cityId = selectedCityStore.selectedCityId;
|
||||||
const response = await authInstance.post("/article", {
|
const response = await authInstance.post("/article", {
|
||||||
translations: {
|
translations: {
|
||||||
heading: {
|
heading: {
|
||||||
@@ -245,6 +247,7 @@ export const ArticleSelectOrCreateDialog = observer(
|
|||||||
zh: newArticleData.zh.body || "Новый текст (ZH)",
|
zh: newArticleData.zh.body || "Новый текст (ZH)",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
...(cityId ? { city_id: cityId } : {}),
|
||||||
});
|
});
|
||||||
|
|
||||||
const { id } = response.data;
|
const { id } = response.data;
|
||||||
@@ -519,9 +522,12 @@ export const ArticleSelectOrCreateDialog = observer(
|
|||||||
languageStore.setLanguage("ru");
|
languageStore.setLanguage("ru");
|
||||||
};
|
};
|
||||||
|
|
||||||
const filteredArticles = articles[modalLanguage].filter((article) =>
|
const cityId = selectedCityStore.selectedCityId;
|
||||||
article?.service_name?.toLowerCase().includes(searchQuery.toLowerCase())
|
const filteredArticles = articles[modalLanguage].filter((article) => {
|
||||||
);
|
if (!article?.service_name?.toLowerCase().includes(searchQuery.toLowerCase())) return false;
|
||||||
|
if (cityId && article.city_id !== cityId) return false;
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
|
||||||
const [isPreviewLoading, setIsPreviewLoading] = useState(false);
|
const [isPreviewLoading, setIsPreviewLoading] = useState(false);
|
||||||
const [queuedPreviewId, setQueuedPreviewId] = useState<number | null>(null);
|
const [queuedPreviewId, setQueuedPreviewId] = useState<number | null>(null);
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user