Migrace z pořadových indexů na unikátní klíče
This commit is contained in:
		
							parent
							
								
									774cb4f9d2
								
							
						
					
					
						commit
						02de6691a8
					
				| @ -14,7 +14,7 @@ import './App.scss'; | ||||
| import { SelectSearchOption } from 'react-select-search'; | ||||
| import { faCircleCheck, faNoteSticky, faTrashCan } from '@fortawesome/free-regular-svg-icons'; | ||||
| import { useSettings } from './context/settings'; | ||||
| import { ClientData, Restaurants, Food, Order, Locations, PizzaOrder, PizzaDayState, FoodChoices, DayMenu, DepartureTime } from './types'; | ||||
| import { ClientData, Restaurants, Food, Order, Locations, PizzaOrder, PizzaDayState, FoodChoices, DayMenu, DepartureTime, LocationKey } from './types'; | ||||
| import Footer from './components/Footer'; | ||||
| import { faChainBroken, faChevronLeft, faChevronRight, faGear, faSatelliteDish, faSearch } from '@fortawesome/free-solid-svg-icons'; | ||||
| import Loader from './components/Loader'; | ||||
| @ -141,11 +141,8 @@ function App() { | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     if (choiceRef?.current?.value && choiceRef.current.value !== "") { | ||||
|       // TODO: wtf, cos pil, když jsi tohle psal?
 | ||||
|       const key = choiceRef?.current?.value; | ||||
|       const locationIndex = Object.keys(Locations).indexOf(key as unknown as Locations); | ||||
|       const locationsKey = Object.keys(Locations)[locationIndex]; | ||||
|       const restaurantKey = Object.keys(Restaurants).indexOf(locationsKey); | ||||
|       const locationKey = choiceRef.current.value as LocationKey; | ||||
|       const restaurantKey = Object.keys(Restaurants).indexOf(locationKey); | ||||
|       if (restaurantKey > -1 && food) { | ||||
|         const restaurant = Object.values(Restaurants)[restaurantKey]; | ||||
|         setFoodChoiceList(food[restaurant]?.food); | ||||
| @ -194,9 +191,9 @@ function App() { | ||||
|   }, [auth?.login, easterEgg?.duration, easterEgg?.url, eggImage]); | ||||
| 
 | ||||
|   const doAddChoice = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||
|     const index = Object.keys(Locations).indexOf(event.target.value as unknown as Locations); | ||||
|     const locationKey = event.target.value as LocationKey; | ||||
|     if (auth?.login) { | ||||
|       await errorHandler(() => addChoice(index, undefined, dayIndex)); | ||||
|       await errorHandler(() => addChoice(locationKey, undefined, dayIndex)); | ||||
|       if (foodChoiceRef.current?.value) { | ||||
|         foodChoiceRef.current.value = ""; | ||||
|       } | ||||
| @ -211,17 +208,16 @@ function App() { | ||||
| 
 | ||||
|   const doAddFoodChoice = async (event: React.ChangeEvent<HTMLSelectElement>) => { | ||||
|     if (event.target.value && foodChoiceList?.length && choiceRef.current?.value) { | ||||
|       const restaurantKey = choiceRef.current.value; | ||||
|       const locationKey = choiceRef.current.value as LocationKey; | ||||
|       if (auth?.login) { | ||||
|         const locationIndex = Object.keys(Locations).indexOf(restaurantKey as unknown as Locations); | ||||
|         await errorHandler(() => addChoice(locationIndex, Number(event.target.value), dayIndex)); | ||||
|         await errorHandler(() => addChoice(locationKey, Number(event.target.value), dayIndex)); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const doRemoveChoices = async (locationKey: string) => { | ||||
|   const doRemoveChoices = async (locationKey: LocationKey) => { | ||||
|     if (auth?.login) { | ||||
|       await errorHandler(() => removeChoices(Number(locationKey), dayIndex)); | ||||
|       await errorHandler(() => removeChoices(locationKey, dayIndex)); | ||||
|       // Vyresetujeme výběr, aby bylo jasné pro který případně vybíráme jídlo
 | ||||
|       if (choiceRef?.current?.value) { | ||||
|         choiceRef.current.value = ""; | ||||
| @ -232,9 +228,9 @@ function App() { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   const doRemoveFoodChoice = async (locationKey: string, foodIndex: number) => { | ||||
|   const doRemoveFoodChoice = async (locationKey: LocationKey, foodIndex: number) => { | ||||
|     if (auth?.login) { | ||||
|       await errorHandler(() => removeChoice(Number(locationKey), foodIndex, dayIndex)); | ||||
|       await errorHandler(() => removeChoice(locationKey, foodIndex, dayIndex)); | ||||
|       if (choiceRef?.current?.value) { | ||||
|         choiceRef.current.value = ""; | ||||
|       } | ||||
| @ -430,11 +426,8 @@ function App() { | ||||
|                   <option></option> | ||||
|                   {Object.entries(Locations) | ||||
|                     .filter(entry => { | ||||
|                       // TODO: wtf, cos pil, když jsi tohle psal? v2
 | ||||
|                       const key = entry[0]; | ||||
|                       const locationIndex = Object.keys(Locations).indexOf(key as unknown as Locations); | ||||
|                       const locationsKey = Object.keys(Locations)[locationIndex]; | ||||
|                       const restaurantKey = Object.keys(Restaurants).indexOf(locationsKey); | ||||
|                       const locationKey = entry[0] as LocationKey; | ||||
|                       const restaurantKey = Object.keys(Restaurants).indexOf(locationKey); | ||||
|                       const v = Object.values(Restaurants)[restaurantKey]; | ||||
|                       return v == null || !food[v]?.closed; | ||||
|                     }) | ||||
| @ -459,11 +452,16 @@ function App() { | ||||
|               {Object.keys(data.choices).length > 0 ? | ||||
|                 <Table bordered className='mt-5'> | ||||
|                   <tbody> | ||||
|                     {Object.keys(data.choices).map((locationKey: string) => { | ||||
|                       const locationName = Object.values(Locations)[Number(locationKey)]; | ||||
|                       const locationLoginList = Object.entries(data.choices[Number(locationKey)]); | ||||
|                     {Object.keys(data.choices).map(key => { | ||||
|                       const locationKey = key as LocationKey; | ||||
|                       const locationName = Locations[locationKey]; | ||||
|                       const loginObject = data.choices[locationKey]; | ||||
|                       if (!loginObject) { | ||||
|                         return; | ||||
|                       } | ||||
|                       const locationLoginList = Object.entries(loginObject); | ||||
|                       return ( | ||||
|                         <tr key={locationKey}> | ||||
|                         <tr key={key}> | ||||
|                           <td>{locationName}</td> | ||||
|                           <td className='p-0'> | ||||
|                             <Table> | ||||
| @ -485,20 +483,20 @@ function App() { | ||||
|                                         setNoteModalOpen(true); | ||||
|                                       }} title='Upravit poznámku' className='action-icon' icon={faNoteSticky} />} | ||||
|                                       {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||
|                                         doRemoveChoices(locationKey); | ||||
|                                         doRemoveChoices(key as LocationKey); | ||||
|                                       }} title={`Odstranit volbu ${locationName}, včetně případných zvolených jídel`} className='action-icon' icon={faTrashCan} />} | ||||
|                                     </td> | ||||
|                                     {userChoices?.length && food ? <td> | ||||
|                                       <ul> | ||||
|                                         {userChoices?.map(foodIndex => { | ||||
|                                           const locationsKey = Object.keys(Locations)[Number(locationKey)] | ||||
|                                           const restaurantKey = Object.keys(Restaurants).indexOf(locationsKey); | ||||
|                                           // TODO narovnat, tohle je zbytečně složité
 | ||||
|                                           const restaurantKey = Object.keys(Restaurants).indexOf(key); | ||||
|                                           const restaurant = Object.values(Restaurants)[restaurantKey]; | ||||
|                                           const foodName = food[restaurant]?.food[foodIndex].name; | ||||
|                                           return <li key={foodIndex}> | ||||
|                                             {foodName} | ||||
|                                             {login === auth.login && canChangeChoice && <FontAwesomeIcon onClick={() => { | ||||
|                                               doRemoveFoodChoice(locationKey, foodIndex); | ||||
|                                               doRemoveFoodChoice(key as LocationKey, foodIndex); | ||||
|                                             }} title={`Odstranit ${foodName}`} className='action-icon' icon={faTrashCan} />} | ||||
|                                           </li> | ||||
|                                         })} | ||||
|  | ||||
| @ -1,18 +1,18 @@ | ||||
| import { AddChoiceRequest, ChangeDepartureTimeRequest, RemoveChoiceRequest, RemoveChoicesRequest, UpdateNoteRequest } from "../types"; | ||||
| import { AddChoiceRequest, ChangeDepartureTimeRequest, LocationKey, RemoveChoiceRequest, RemoveChoicesRequest, UpdateNoteRequest } from "../types"; | ||||
| import { api } from "./Api"; | ||||
| 
 | ||||
| const FOOD_API_PREFIX = '/api/food'; | ||||
| 
 | ||||
| export const addChoice = async (locationIndex: number, foodIndex?: number, dayIndex?: number) => { | ||||
|     return await api.post<AddChoiceRequest, void>(`${FOOD_API_PREFIX}/addChoice`, { locationIndex, foodIndex, dayIndex }); | ||||
| export const addChoice = async (locationKey: LocationKey, foodIndex?: number, dayIndex?: number) => { | ||||
|     return await api.post<AddChoiceRequest, void>(`${FOOD_API_PREFIX}/addChoice`, { locationKey, foodIndex, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const removeChoices = async (locationIndex: number, dayIndex?: number) => { | ||||
|     return await api.post<RemoveChoicesRequest, void>(`${FOOD_API_PREFIX}/removeChoices`, { locationIndex, dayIndex }); | ||||
| export const removeChoices = async (locationKey: LocationKey, dayIndex?: number) => { | ||||
|     return await api.post<RemoveChoicesRequest, void>(`${FOOD_API_PREFIX}/removeChoices`, { locationKey, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const removeChoice = async (locationIndex: number, foodIndex: number, dayIndex?: number) => { | ||||
|     return await api.post<RemoveChoiceRequest, void>(`${FOOD_API_PREFIX}/removeChoice`, { locationIndex, foodIndex, dayIndex }); | ||||
| export const removeChoice = async (locationKey: LocationKey, foodIndex: number, dayIndex?: number) => { | ||||
|     return await api.post<RemoveChoiceRequest, void>(`${FOOD_API_PREFIX}/removeChoice`, { locationKey, foodIndex, dayIndex }); | ||||
| } | ||||
| 
 | ||||
| export const updateNote = async (note?: string, dayIndex?: number) => { | ||||
|  | ||||
| @ -33,7 +33,6 @@ const router = express.Router(); | ||||
| router.post("/addChoice", async (req: Request<{}, any, AddChoiceRequest>, res, next) => { | ||||
|     const login = getLogin(parseToken(req)); | ||||
|     const trusted = getTrusted(parseToken(req)); | ||||
|     if (req.body.locationIndex > -1) { | ||||
|     let date = undefined; | ||||
|     if (req.body.dayIndex != null) { | ||||
|         let dayIndex; | ||||
| @ -45,12 +44,10 @@ router.post("/addChoice", async (req: Request<{}, any, AddChoiceRequest>, res, n | ||||
|         date = getDateForWeekIndex(dayIndex); | ||||
|     } | ||||
|     try { | ||||
|             const data = await addChoice(login, trusted, req.body.locationIndex, req.body.foodIndex, date); | ||||
|         const data = await addChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         return res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
|     } | ||||
|     return res.status(400); // TODO přidat popis chyby
 | ||||
| }); | ||||
| 
 | ||||
| router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesRequest>, res, next) => { | ||||
| @ -67,7 +64,7 @@ router.post("/removeChoices", async (req: Request<{}, any, RemoveChoicesRequest> | ||||
|         date = getDateForWeekIndex(dayIndex); | ||||
|     } | ||||
|     try { | ||||
|         const data = await removeChoices(login, trusted, req.body.locationIndex, date); | ||||
|         const data = await removeChoices(login, trusted, req.body.locationKey, date); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
| @ -87,7 +84,7 @@ router.post("/removeChoice", async (req: Request<{}, any, RemoveChoiceRequest>, | ||||
|         date = getDateForWeekIndex(dayIndex); | ||||
|     } | ||||
|     try { | ||||
|         const data = await removeChoice(login, trusted, req.body.locationIndex, req.body.foodIndex, date); | ||||
|         const data = await removeChoice(login, trusted, req.body.locationKey, req.body.foodIndex, date); | ||||
|         getWebsocket().emit("message", await addVolatileData(data)); | ||||
|         res.status(200).json(data); | ||||
|     } catch (e: any) { next(e) } | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| import { InsufficientPermissions, formatDate, getDayOfWeekIndex, getFirstWorkDayOfWeek, getHumanDate, getHumanTime, getIsWeekend, getLastWorkDayOfWeek, getWeekNumber } from "./utils"; | ||||
| import { ClientData, Locations, Restaurants, DayMenu, DepartureTime, DayData, WeekMenu } from "../../types"; | ||||
| import { InsufficientPermissions, formatDate, getDayOfWeekIndex, getFirstWorkDayOfWeek, getHumanDate, getIsWeekend, getWeekNumber } from "./utils"; | ||||
| import { ClientData, Restaurants, DayMenu, DepartureTime, DayData, WeekMenu, LocationKey } from "../../types"; | ||||
| import getStorage from "./storage"; | ||||
| import { getMenuSladovnicka, getMenuTechTower, getMenuUMotliku } from "./restaurants"; | ||||
| import { getTodayMock } from "./mock"; | ||||
| @ -192,19 +192,19 @@ export async function initIfNeeded(date?: Date) { | ||||
|  *  | ||||
|  * @param login login uživatele | ||||
|  * @param trusted příznak, zda se jedná o ověřeného uživatele | ||||
|  * @param locationIndex vybrané "umístění" | ||||
|  * @param locationKey vybrané "umístění" | ||||
|  * @param date datum, ke kterému se volba vztahuje | ||||
|  * @returns  | ||||
|  */ | ||||
| export async function removeChoices(login: string, trusted: boolean, locationIndex: number, date?: Date) { | ||||
| export async function removeChoices(login: string, trusted: boolean, locationKey: LocationKey, date?: Date) { | ||||
|     const selectedDay = formatDate(date ?? getToday()); | ||||
|     let data: DayData = await storage.getData(selectedDay); | ||||
|     validateTrusted(data, login, trusted); | ||||
|     if (locationIndex in data.choices) { | ||||
|         if (login in data.choices[locationIndex]) { | ||||
|             delete data.choices[locationIndex][login] | ||||
|             if (Object.keys(data.choices[locationIndex]).length === 0) { | ||||
|                 delete data.choices[locationIndex] | ||||
|     if (locationKey in data.choices) { | ||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||
|             delete data.choices[locationKey][login] | ||||
|             if (Object.keys(data.choices[locationKey]).length === 0) { | ||||
|                 delete data.choices[locationKey] | ||||
|             } | ||||
|             await storage.setData(selectedDay, data); | ||||
|         } | ||||
| @ -218,20 +218,20 @@ export async function removeChoices(login: string, trusted: boolean, locationInd | ||||
|  *  | ||||
|  * @param login login uživatele | ||||
|  * @param trusted příznak, zda se jedná o ověřeného uživatele | ||||
|  * @param locationIndex vybrané "umístění" | ||||
|  * @param locationKey vybrané "umístění" | ||||
|  * @param foodIndex index jídla v jídelním lístku daného umístění, pokud existuje | ||||
|  * @param date datum, ke kterému se volba vztahuje | ||||
|  * @returns  | ||||
|  */ | ||||
| export async function removeChoice(login: string, trusted: boolean, locationIndex: number, foodIndex: number, date?: Date) { | ||||
| export async function removeChoice(login: string, trusted: boolean, locationKey: LocationKey, foodIndex: number, date?: Date) { | ||||
|     const selectedDay = formatDate(date ?? getToday()); | ||||
|     let data: DayData = await storage.getData(selectedDay); | ||||
|     validateTrusted(data, login, trusted); | ||||
|     if (locationIndex in data.choices) { | ||||
|         if (login in data.choices[locationIndex]) { | ||||
|             const index = data.choices[locationIndex][login].options.indexOf(foodIndex); | ||||
|     if (locationKey in data.choices) { | ||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||
|             const index = data.choices[locationKey][login].options.indexOf(foodIndex); | ||||
|             if (index > -1) { | ||||
|                 data.choices[locationIndex][login].options.splice(index, 1) | ||||
|                 data.choices[locationKey][login].options.splice(index, 1) | ||||
|                 await storage.setData(selectedDay, data); | ||||
|             } | ||||
|         } | ||||
| @ -247,10 +247,11 @@ export async function removeChoice(login: string, trusted: boolean, locationInde | ||||
| async function removeChoiceIfPresent(login: string, date: string) { | ||||
|     let data: DayData = await storage.getData(date); | ||||
|     for (const key of Object.keys(data.choices)) { | ||||
|         if (login in data.choices[key]) { | ||||
|             delete data.choices[key][login]; | ||||
|             if (Object.keys(data.choices[key]).length === 0) { | ||||
|                 delete data.choices[key]; | ||||
|         const locationKey = key as LocationKey; | ||||
|         if (data.choices[locationKey] && login in data.choices[locationKey]) { | ||||
|             delete data.choices[locationKey][login]; | ||||
|             if (Object.keys(data.choices[locationKey]).length === 0) { | ||||
|                 delete data.choices[locationKey]; | ||||
|             } | ||||
|             await storage.setData(date, data); | ||||
|         } | ||||
| @ -285,13 +286,13 @@ function validateTrusted(data: ClientData, login: string, trusted: boolean) { | ||||
|  *  | ||||
|  * @param login login uživatele | ||||
|  * @param trusted příznak, zda se jedná o ověřeného uživatele | ||||
|  * @param locationIndex vybrané "umístění" | ||||
|  * @param locationKey vybrané "umístění" | ||||
|  * @param foodIndex volitelný index jídla v daném umístění | ||||
|  * @param trusted příznak, zda se jedná o ověřeného uživatele | ||||
|  * @param date datum, ke kterému se volba vztahuje | ||||
|  * @returns aktuální data | ||||
|  */ | ||||
| export async function addChoice(login: string, trusted: boolean, locationIndex: number, foodIndex?: number, date?: Date) { | ||||
| export async function addChoice(login: string, trusted: boolean, locationKey: LocationKey, foodIndex?: number, date?: Date) { | ||||
|     const usedDate = date ?? getToday(); | ||||
|     await initIfNeeded(usedDate); | ||||
|     const selectedDate = formatDate(usedDate); | ||||
| @ -301,17 +302,21 @@ export async function addChoice(login: string, trusted: boolean, locationIndex: | ||||
|     if (foodIndex == null) { | ||||
|         data = await removeChoiceIfPresent(login, selectedDate); | ||||
|     } | ||||
|     if (!(locationIndex in data.choices)) { | ||||
|         data.choices[locationIndex] = {}; | ||||
|     // TODO vytáhnout inicializaci "prázdné struktury" do vlastní funkce
 | ||||
|     if (!(data.choices[locationKey])) { | ||||
|         data.choices[locationKey] = {} | ||||
|     } | ||||
|     if (!(login in data.choices[locationIndex])) { | ||||
|         data.choices[locationIndex][login] = { | ||||
|     if (!(login in data.choices[locationKey])) { | ||||
|         if (!data.choices[locationKey]) { | ||||
|             data.choices[locationKey] = {} | ||||
|         } | ||||
|         data.choices[locationKey][login] = { | ||||
|             trusted, | ||||
|             options: [] | ||||
|         }; | ||||
|     } | ||||
|     if (foodIndex != null && !data.choices[locationIndex][login].options.includes(foodIndex)) { | ||||
|         data.choices[locationIndex][login].options.push(foodIndex); | ||||
|     if (foodIndex != null && !data.choices[locationKey][login].options.includes(foodIndex)) { | ||||
|         data.choices[locationKey][login].options.push(foodIndex); | ||||
|     } | ||||
|     await storage.setData(selectedDate, data); | ||||
|     return data; | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { Choices } from "../../types"; | ||||
| import { Choices, LocationKey } from "../../types"; | ||||
| 
 | ||||
| /** Vrátí datum v ISO formátu. */ | ||||
| export function formatDate(date: Date) { | ||||
| @ -110,21 +110,21 @@ export const checkBodyParams = (req: any, paramNames: string[]) => { | ||||
| // TODO umístit do samostatného souboru
 | ||||
| export class InsufficientPermissions extends Error { } | ||||
| 
 | ||||
| export const getUsersByLocation = (data: Choices, login: string): string[] => { | ||||
| export const getUsersByLocation = (choices: Choices, login: string): string[] => { | ||||
|     const result: string[] = []; | ||||
| 
 | ||||
|     for (const location in data) { | ||||
|         if (data.hasOwnProperty(location)) { | ||||
|             if (data[location][login]) { | ||||
|                 for (const username in data[location]) { | ||||
|                     if (data[location].hasOwnProperty(username)) { | ||||
|     for (const location of Object.entries(choices)) { | ||||
|         const locationKey = location[0] as LocationKey; | ||||
|         const locationValue = location[1]; | ||||
|         if (locationValue[login]) { | ||||
|             for (const username in choices[locationKey]) { | ||||
|                 if (choices[locationKey].hasOwnProperty(username)) { | ||||
|                     result.push(username); | ||||
|                 } | ||||
|             } | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|     } | ||||
| 
 | ||||
|     return result; | ||||
| } | ||||
| @ -1,20 +1,20 @@ | ||||
| import { FeatureRequest, PizzaOrder } from "./Types"; | ||||
| import { FeatureRequest, LocationKey, PizzaOrder } from "./Types"; | ||||
| 
 | ||||
| export type ILocationKey = { | ||||
|     locationKey: LocationKey, | ||||
| } | ||||
| 
 | ||||
| export type IDayIndex = { | ||||
|     dayIndex?: number, | ||||
| } | ||||
| 
 | ||||
| export type AddChoiceRequest = IDayIndex & { | ||||
|     locationIndex: number, | ||||
| export type AddChoiceRequest = IDayIndex & ILocationKey & { | ||||
|     foodIndex?: number, | ||||
| } | ||||
| 
 | ||||
| export type RemoveChoicesRequest = IDayIndex & { | ||||
|     locationIndex: number, | ||||
| } | ||||
| export type RemoveChoicesRequest = IDayIndex & ILocationKey; | ||||
| 
 | ||||
| export type RemoveChoiceRequest = IDayIndex & { | ||||
|     locationIndex: number, | ||||
| export type RemoveChoiceRequest = IDayIndex & ILocationKey & { | ||||
|     foodIndex: number, | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -12,10 +12,11 @@ export type FoodChoices = { | ||||
|     note?: string, | ||||
| } | ||||
| 
 | ||||
| // TODO okomentovat / rozdělit
 | ||||
| export type Choices = { | ||||
|     [location: string]: { | ||||
|     [location in LocationKey]?: { | ||||
|         [login: string]: FoodChoices | ||||
|     }, | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /** Velikost konkrétní pizzy */ | ||||
| @ -119,6 +120,9 @@ export enum Locations { | ||||
|     ROZHODUJI = 'Rozhoduji se', | ||||
| } | ||||
| 
 | ||||
| // TODO totéž
 | ||||
| export type LocationKey = keyof typeof Locations; | ||||
| 
 | ||||
| export enum UdalostEnum { | ||||
|     ZAHAJENA_PIZZA = "Zahájen pizza day", | ||||
|     OBJEDNANA_PIZZA = "Objednána pizza", | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user