utopia-ui 3.0.90 → 3.0.92
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs +812 -784
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.ts +61 -109
- package/dist/index.esm.js +812 -784
- package/dist/index.esm.js.map +1 -1
- package/dist/types/src/Components/Map/Layer.d.ts +1 -1
- package/dist/types/src/Components/Map/LayerContext.d.ts +2 -1
- package/dist/types/src/Components/Map/Subcomponents/ItemFormPopup.d.ts +5 -2
- package/dist/types/src/Components/Map/Subcomponents/ItemPopupComponents/PopupStartEndInput.d.ts +1 -2
- package/dist/types/src/Components/Map/Subcomponents/ItemViewPopup.d.ts +0 -2
- package/dist/types/src/Components/Map/hooks/useSelectPosition.d.ts +2 -2
- package/dist/types/src/Components/Map/index.d.ts +0 -9
- package/dist/types/src/index.d.ts +1 -0
- package/package.json +1 -1
package/dist/index.esm.js
CHANGED
@@ -7,7 +7,7 @@ import { DomEvent, divIcon, Point, control, marker, LatLng, LatLngBounds } from
|
|
7
7
|
import { useMap, useMapEvents, TileLayer, GeoJSON, MapContainer, Popup, Marker, Tooltip } from 'react-leaflet';
|
8
8
|
import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
|
9
9
|
import * as React from 'react';
|
10
|
-
import { createContext, useContext, useState, useCallback, useReducer, useEffect, createRef, useRef, memo,
|
10
|
+
import { createContext, useContext, useState, useCallback, useReducer, useEffect, createRef, useRef, memo, cloneElement, forwardRef, useMemo } from 'react';
|
11
11
|
import { useNavigate, useInRouterContext, BrowserRouter, useLocation, Outlet, Link, NavLink } from 'react-router-dom';
|
12
12
|
import { toast, ToastContainer } from 'react-toastify';
|
13
13
|
import MarkerClusterGroup from 'react-leaflet-cluster';
|
@@ -120,7 +120,7 @@ const useSetClusterRef = () => {
|
|
120
120
|
return setClusterRef;
|
121
121
|
};
|
122
122
|
|
123
|
-
const LayerContext = createContext({
|
123
|
+
const LayerContext$1 = createContext({
|
124
124
|
layers: [],
|
125
125
|
// eslint-disable-next-line @typescript-eslint/no-empty-function
|
126
126
|
addLayer: () => { },
|
@@ -148,13 +148,13 @@ function useLayerManager(initialLayers) {
|
|
148
148
|
}, []);
|
149
149
|
return { layers, addLayer };
|
150
150
|
}
|
151
|
-
const LayersProvider = ({ initialLayers, children }) => (jsx(LayerContext.Provider, { value: useLayerManager(initialLayers), children: children }));
|
151
|
+
const LayersProvider = ({ initialLayers, children }) => (jsx(LayerContext$1.Provider, { value: useLayerManager(initialLayers), children: children }));
|
152
152
|
const useLayers = () => {
|
153
|
-
const { layers } = useContext(LayerContext);
|
153
|
+
const { layers } = useContext(LayerContext$1);
|
154
154
|
return layers;
|
155
155
|
};
|
156
156
|
const useAddLayer = () => {
|
157
|
-
const { addLayer } = useContext(LayerContext);
|
157
|
+
const { addLayer } = useContext(LayerContext$1);
|
158
158
|
return addLayer;
|
159
159
|
};
|
160
160
|
|
@@ -409,7 +409,7 @@ const useVisibleGroupType = () => {
|
|
409
409
|
return visibleGroupTypes;
|
410
410
|
};
|
411
411
|
|
412
|
-
const ItemContext = createContext({
|
412
|
+
const ItemContext$1 = createContext({
|
413
413
|
items: [],
|
414
414
|
addItem: () => { },
|
415
415
|
updateItem: () => { },
|
@@ -509,33 +509,33 @@ function useItemsManager(initialItems) {
|
|
509
509
|
allItemsLoaded,
|
510
510
|
};
|
511
511
|
}
|
512
|
-
const ItemsProvider = ({ initialItems, children }) => (jsx(ItemContext.Provider, { value: useItemsManager(initialItems), children: children }));
|
512
|
+
const ItemsProvider = ({ initialItems, children }) => (jsx(ItemContext$1.Provider, { value: useItemsManager(initialItems), children: children }));
|
513
513
|
const useItems = () => {
|
514
|
-
const { items } = useContext(ItemContext);
|
514
|
+
const { items } = useContext(ItemContext$1);
|
515
515
|
return items;
|
516
516
|
};
|
517
517
|
const useAddItem = () => {
|
518
|
-
const { addItem } = useContext(ItemContext);
|
518
|
+
const { addItem } = useContext(ItemContext$1);
|
519
519
|
return addItem;
|
520
520
|
};
|
521
521
|
const useUpdateItem = () => {
|
522
|
-
const { updateItem } = useContext(ItemContext);
|
522
|
+
const { updateItem } = useContext(ItemContext$1);
|
523
523
|
return updateItem;
|
524
524
|
};
|
525
525
|
const useRemoveItem = () => {
|
526
|
-
const { removeItem } = useContext(ItemContext);
|
526
|
+
const { removeItem } = useContext(ItemContext$1);
|
527
527
|
return removeItem;
|
528
528
|
};
|
529
529
|
const useSetItemsApi = () => {
|
530
|
-
const { setItemsApi } = useContext(ItemContext);
|
530
|
+
const { setItemsApi } = useContext(ItemContext$1);
|
531
531
|
return setItemsApi;
|
532
532
|
};
|
533
533
|
const useSetItemsData = () => {
|
534
|
-
const { setItemsData } = useContext(ItemContext);
|
534
|
+
const { setItemsData } = useContext(ItemContext$1);
|
535
535
|
return setItemsData;
|
536
536
|
};
|
537
537
|
const useAllItemsLoaded = () => {
|
538
|
-
const { allItemsLoaded } = useContext(ItemContext);
|
538
|
+
const { allItemsLoaded } = useContext(ItemContext$1);
|
539
539
|
return allItemsLoaded;
|
540
540
|
};
|
541
541
|
|
@@ -819,6 +819,22 @@ const useSetAdminRole = () => {
|
|
819
819
|
return setAdminRole;
|
820
820
|
};
|
821
821
|
|
822
|
+
const PoupFormContext = createContext({
|
823
|
+
popupForm: {},
|
824
|
+
setPopupForm: () => {
|
825
|
+
/* empty function */
|
826
|
+
},
|
827
|
+
});
|
828
|
+
function usePopupFormManager() {
|
829
|
+
const [popupForm, setPopupForm] = useState(null);
|
830
|
+
return { popupForm, setPopupForm };
|
831
|
+
}
|
832
|
+
const PopupFormProvider = ({ children }) => (jsx(PoupFormContext.Provider, { value: usePopupFormManager(), children: children }));
|
833
|
+
const usePopupForm = () => {
|
834
|
+
const { popupForm, setPopupForm } = useContext(PoupFormContext);
|
835
|
+
return { popupForm, setPopupForm };
|
836
|
+
};
|
837
|
+
|
822
838
|
const SelectPositionContext = createContext({
|
823
839
|
selectPosition: null,
|
824
840
|
setSelectPosition: () => { },
|
@@ -842,7 +858,9 @@ function useSelectPositionManager() {
|
|
842
858
|
}, [markerClicked]);
|
843
859
|
useEffect(() => {
|
844
860
|
if (selectPosition != null) {
|
861
|
+
// selectPosition can be null, Layer or Item
|
845
862
|
if ('menuIcon' in selectPosition) {
|
863
|
+
// if selectPosition is a Layer
|
846
864
|
mapClicked &&
|
847
865
|
mapClicked.setItemFormPopup({
|
848
866
|
layer: selectPosition,
|
@@ -851,6 +869,7 @@ function useSelectPositionManager() {
|
|
851
869
|
setSelectPosition(null);
|
852
870
|
}
|
853
871
|
if ('text' in selectPosition) {
|
872
|
+
// if selectPosition is an Item
|
854
873
|
const position = mapClicked?.position.lng &&
|
855
874
|
{
|
856
875
|
type: 'Point',
|
@@ -1115,7 +1134,7 @@ const ContextWrapper = ({ children }) => {
|
|
1115
1134
|
// eslint-disable-next-line react/prop-types
|
1116
1135
|
const Wrappers = ({ children }) => {
|
1117
1136
|
const queryClient = new QueryClient();
|
1118
|
-
return (jsx(PermissionsProvider, { initialPermissions: [], children: jsx(TagsProvider, { initialTags: [], children: jsx(LayersProvider, { initialLayers: [], children: jsx(FilterProvider, { initialTags: [], children: jsx(ItemsProvider, { initialItems: [], children: jsx(SelectPositionProvider, { children: jsx(LeafletRefsProvider, { initialLeafletRefs: {}, children: jsx(QueryClientProvider, { client: queryClient, children: jsx(AppStateProvider, { children: jsx(ClusterRefProvider, { children: jsxs(QuestsProvider, { initialOpen: true, children: [jsx(ToastContainer, { position: 'top-right', autoClose: 2000, hideProgressBar: true, newestOnTop: false, closeOnClick: true, rtl: false, pauseOnFocusLoss: true, draggable: true, pauseOnHover: true, theme: 'light', closeButton: CloseButton }), children] }) }) }) }) }) }) }) }) }) }) }));
|
1137
|
+
return (jsx(PermissionsProvider, { initialPermissions: [], children: jsx(TagsProvider, { initialTags: [], children: jsx(LayersProvider, { initialLayers: [], children: jsx(FilterProvider, { initialTags: [], children: jsx(ItemsProvider, { initialItems: [], children: jsx(SelectPositionProvider, { children: jsx(LeafletRefsProvider, { initialLeafletRefs: {}, children: jsx(QueryClientProvider, { client: queryClient, children: jsx(AppStateProvider, { children: jsx(ClusterRefProvider, { children: jsx(PopupFormProvider, { children: jsxs(QuestsProvider, { initialOpen: true, children: [jsx(ToastContainer, { position: 'top-right', autoClose: 2000, hideProgressBar: true, newestOnTop: false, closeOnClick: true, rtl: false, pauseOnFocusLoss: true, draggable: true, pauseOnHover: true, theme: 'light', closeButton: CloseButton }), children] }) }) }) }) }) }) }) }) }) }) }) }));
|
1119
1138
|
};
|
1120
1139
|
|
1121
1140
|
const useTheme = (defaultTheme = 'default') => {
|
@@ -2584,7 +2603,7 @@ function fixUrls(message) {
|
|
2584
2603
|
/**
|
2585
2604
|
* @category Map
|
2586
2605
|
*/
|
2587
|
-
const TextView = ({ item, itemId, text, truncate = false, rawText, itemTextField, }) => {
|
2606
|
+
const TextView$1 = ({ item, itemId, text, truncate = false, rawText, itemTextField, }) => {
|
2588
2607
|
if (item) {
|
2589
2608
|
text = item.text;
|
2590
2609
|
itemId = item.id;
|
@@ -2660,8 +2679,6 @@ const TextView = ({ item, itemId, text, truncate = false, rawText, itemTextField
|
|
2660
2679
|
return jsx(MemoizedVideoEmbed, { url: href });
|
2661
2680
|
}
|
2662
2681
|
if (href?.startsWith('#')) {
|
2663
|
-
console.log(href.slice(1).toLowerCase());
|
2664
|
-
console.log(tags);
|
2665
2682
|
const tag = tags.find((t) => t.name.toLowerCase() === decodeURI(href).slice(1).toLowerCase());
|
2666
2683
|
if (tag)
|
2667
2684
|
return (jsx(CustomHashTagLink, { tag: tag, itemId: itemId, children: children }));
|
@@ -2729,13 +2746,13 @@ function UtopiaMapInner({ children, geo, showFilterControl = false, showGratitud
|
|
2729
2746
|
const setClusterRef = useSetClusterRef();
|
2730
2747
|
const clusterRef = useClusterRef();
|
2731
2748
|
const setMapClicked = useSetMapClicked();
|
2732
|
-
const
|
2733
|
-
useTheme(defaultTheme);
|
2749
|
+
const { setPopupForm } = usePopupForm();
|
2734
2750
|
const layers = useLayers();
|
2735
2751
|
const addVisibleLayer = useAddVisibleLayer();
|
2736
2752
|
const leafletRefs = useLeafletRefs();
|
2737
2753
|
const location = useLocation();
|
2738
2754
|
const map = useMap();
|
2755
|
+
useTheme(defaultTheme);
|
2739
2756
|
useEffect(() => {
|
2740
2757
|
layers.forEach((layer) => addVisibleLayer(layer));
|
2741
2758
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
@@ -2749,7 +2766,7 @@ function UtopiaMapInner({ children, geo, showFilterControl = false, showGratitud
|
|
2749
2766
|
if (!init.current) {
|
2750
2767
|
donationWidget &&
|
2751
2768
|
setTimeout(() => {
|
2752
|
-
toast(jsxs(Fragment, { children: [jsx(TextView, { itemId: '', rawText: '## Do you like this Map?' }), jsxs("div", { children: [jsx(TextView, { itemId: '', rawText: 'Support us building free opensource maps for communities and help us grow 🌱☀️' }), jsx("a", { href: 'https://opencollective.com/utopia-project', children: jsx("div", { className: 'tw:btn tw:btn-sm tw:float-right tw:btn-primary', children: "Donate" }) })] })] }), { autoClose: false });
|
2769
|
+
toast(jsxs(Fragment, { children: [jsx(TextView$1, { itemId: '', rawText: '## Do you like this Map?' }), jsxs("div", { children: [jsx(TextView$1, { itemId: '', rawText: 'Support us building free opensource maps for communities and help us grow 🌱☀️' }), jsx("a", { href: 'https://opencollective.com/utopia-project', children: jsx("div", { className: 'tw:btn tw:btn-sm tw:float-right tw:btn-primary', children: "Donate" }) })] })] }), { autoClose: false });
|
2753
2770
|
}, 600000);
|
2754
2771
|
init.current = true;
|
2755
2772
|
}
|
@@ -2762,7 +2779,7 @@ function UtopiaMapInner({ children, geo, showFilterControl = false, showGratitud
|
|
2762
2779
|
// eslint-disable-next-line no-console
|
2763
2780
|
console.log(e.latlng.lat + ',' + e.latlng.lng);
|
2764
2781
|
if (selectNewItemPosition) {
|
2765
|
-
setMapClicked({ position: e.latlng, setItemFormPopup });
|
2782
|
+
setMapClicked({ position: e.latlng, setItemFormPopup: setPopupForm });
|
2766
2783
|
}
|
2767
2784
|
},
|
2768
2785
|
moveend: () => { },
|
@@ -2833,13 +2850,11 @@ function UtopiaMapInner({ children, geo, showFilterControl = false, showGratitud
|
|
2833
2850
|
layer.bindPopup(feature.properties.name);
|
2834
2851
|
}
|
2835
2852
|
};
|
2836
|
-
return (jsxs("div", { className: `tw:h-full ${selectNewItemPosition != null ? 'crosshair-cursor-enabled' : ''}`, children: [jsx(Outlet, {}), jsxs(Control, { position: 'topLeft', zIndex: '1000', absolute: true, children: [jsx(SearchControl, {}), jsx(TagsControl, {})] }), jsxs(Control, { position: 'bottomLeft', zIndex: '999', absolute: true, children: [showFilterControl && jsx(FilterControl, {}), showLayerControl && jsx(LayerControl, {}), showGratitudeControl && jsx(GratitudeControl, {})] }), jsx(TileLayer, { maxZoom: 19, attribution: '\u00A9 <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors', url: 'https://tile.osmand.net/hd/{z}/{x}/{y}.png' }), jsx(MarkerClusterGroup, { ref: (r) => setClusterRef(r), showCoverageOnHover: true, chunkedLoading: true, maxClusterRadius: 50, removeOutsideVisibleBounds: false, children:
|
2837
|
-
? cloneElement(child, { setItemFormPopup, itemFormPopup, clusterRef })
|
2838
|
-
: child) }), geo && (jsx(GeoJSON, { data: geo, onEachFeature: onEachFeature, eventHandlers: {
|
2853
|
+
return (jsxs("div", { className: `tw:h-full ${selectNewItemPosition != null ? 'crosshair-cursor-enabled' : ''}`, children: [jsx(Outlet, {}), jsxs(Control, { position: 'topLeft', zIndex: '1000', absolute: true, children: [jsx(SearchControl, {}), jsx(TagsControl, {})] }), jsxs(Control, { position: 'bottomLeft', zIndex: '999', absolute: true, children: [showFilterControl && jsx(FilterControl, {}), showLayerControl && jsx(LayerControl, {}), showGratitudeControl && jsx(GratitudeControl, {})] }), jsx(TileLayer, { maxZoom: 19, attribution: '\u00A9 <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors', url: 'https://tile.osmand.net/hd/{z}/{x}/{y}.png' }), jsx(MarkerClusterGroup, { ref: (r) => setClusterRef(r), showCoverageOnHover: true, chunkedLoading: true, maxClusterRadius: 50, removeOutsideVisibleBounds: false, children: children }), geo && (jsx(GeoJSON, { data: geo, onEachFeature: onEachFeature, eventHandlers: {
|
2839
2854
|
click: (e) => {
|
2840
2855
|
if (selectNewItemPosition) {
|
2841
2856
|
e.layer.closePopup();
|
2842
|
-
setMapClicked({ position: e.latlng, setItemFormPopup });
|
2857
|
+
setMapClicked({ position: e.latlng, setItemFormPopup: setPopupForm });
|
2843
2858
|
}
|
2844
2859
|
},
|
2845
2860
|
} })), jsx(MapEventListener, {}), jsx(AddButton, { triggerAction: setSelectNewItemPosition }), selectNewItemPosition != null && (jsx(SelectPosition, { setSelectNewItemPosition: setSelectNewItemPosition }))] }));
|
@@ -2883,653 +2898,143 @@ function UtopiaMap({ height = '500px', width = '100%', center = [50.6, 9.5], zoo
|
|
2883
2898
|
return (jsx(ContextWrapper, { children: jsx(MapContainer, { style: { height, width }, center: new LatLng(center[0], center[1]), zoom: zoom, zoomControl: showZoomControl, maxZoom: 19, children: jsx(UtopiaMapInner, { geo: geo, showFilterControl: showFilterControl, showGratitudeControl: showGratitudeControl, showLayerControl: showLayerControl, donationWidget: donationWidget, showThemeControl: showThemeControl, defaultTheme: defaultTheme, children: children }) }) }));
|
2884
2899
|
}
|
2885
2900
|
|
2886
|
-
const
|
2887
|
-
|
2888
|
-
|
2889
|
-
|
2890
|
-
|
2891
|
-
|
2892
|
-
|
2893
|
-
const f = h * 6 - i;
|
2894
|
-
const p = v * (1 - s);
|
2895
|
-
const q = v * (1 - f * s);
|
2896
|
-
const t = v * (1 - (1 - f) * s);
|
2897
|
-
switch (i % 6) {
|
2898
|
-
case 0:
|
2899
|
-
r = v;
|
2900
|
-
g = t;
|
2901
|
-
b = p;
|
2902
|
-
break;
|
2903
|
-
case 1:
|
2904
|
-
r = q;
|
2905
|
-
g = v;
|
2906
|
-
b = p;
|
2907
|
-
break;
|
2908
|
-
case 2:
|
2909
|
-
r = p;
|
2910
|
-
g = v;
|
2911
|
-
b = t;
|
2912
|
-
break;
|
2913
|
-
case 3:
|
2914
|
-
r = p;
|
2915
|
-
g = q;
|
2916
|
-
b = v;
|
2917
|
-
break;
|
2918
|
-
case 4:
|
2919
|
-
r = t;
|
2920
|
-
g = p;
|
2921
|
-
b = v;
|
2922
|
-
break;
|
2923
|
-
case 5:
|
2924
|
-
r = v;
|
2925
|
-
g = p;
|
2926
|
-
b = q;
|
2927
|
-
break;
|
2928
|
-
}
|
2929
|
-
return rgbToHex(Math.round(r * 255), Math.round(g * 255), Math.round(b * 255));
|
2930
|
-
}
|
2931
|
-
const rgbToHex = (r, g, b) => '#' +
|
2932
|
-
[r, g, b]
|
2933
|
-
.map((x) => {
|
2934
|
-
const hex = x.toString(16);
|
2935
|
-
return hex.length === 1 ? `0${hex}` : hex;
|
2936
|
-
})
|
2937
|
-
.join('');
|
2901
|
+
const LayerContext = createContext({
|
2902
|
+
name: '',
|
2903
|
+
markerDefaultColor: '',
|
2904
|
+
markerDefaultColor2: '',
|
2905
|
+
markerShape: '',
|
2906
|
+
menuText: '',
|
2907
|
+
});
|
2938
2908
|
|
2939
2909
|
/**
|
2940
|
-
* @category
|
2910
|
+
* @category Map
|
2941
2911
|
*/
|
2942
|
-
|
2943
|
-
|
2944
|
-
|
2912
|
+
const Layer = ({ data, children, name = 'places', menuIcon = 'MapPinIcon', menuText = 'add new place', menuColor = '#2E7D32', markerIcon, markerShape = 'circle', markerDefaultColor = '#777', markerDefaultColor2 = 'RGBA(35, 31, 32, 0.2)', api, itemType, userProfileLayer = false, customEditLink, customEditParameter,
|
2913
|
+
// eslint-disable-next-line camelcase
|
2914
|
+
public_edit_items, listed = true, }) => {
|
2915
|
+
const setItemsApi = useSetItemsApi();
|
2916
|
+
const setItemsData = useSetItemsData();
|
2917
|
+
const addTag = useAddTag();
|
2918
|
+
const [newTagsToAdd] = useState([]);
|
2919
|
+
const [tagsReady] = useState(false);
|
2945
2920
|
useEffect(() => {
|
2946
|
-
|
2947
|
-
|
2948
|
-
|
2949
|
-
|
2950
|
-
|
2951
|
-
|
2952
|
-
|
2921
|
+
data &&
|
2922
|
+
setItemsData({
|
2923
|
+
data,
|
2924
|
+
children,
|
2925
|
+
name,
|
2926
|
+
menuIcon,
|
2927
|
+
menuText,
|
2928
|
+
menuColor,
|
2929
|
+
markerIcon,
|
2930
|
+
markerShape,
|
2931
|
+
markerDefaultColor,
|
2932
|
+
markerDefaultColor2,
|
2933
|
+
api,
|
2934
|
+
itemType,
|
2935
|
+
userProfileLayer,
|
2936
|
+
// Can we just use editCallback for all cases?
|
2937
|
+
customEditLink,
|
2938
|
+
customEditParameter,
|
2939
|
+
// eslint-disable-next-line camelcase
|
2940
|
+
public_edit_items,
|
2941
|
+
listed,
|
2942
|
+
});
|
2943
|
+
api &&
|
2944
|
+
setItemsApi({
|
2945
|
+
data,
|
2946
|
+
children,
|
2947
|
+
name,
|
2948
|
+
menuIcon,
|
2949
|
+
menuText,
|
2950
|
+
menuColor,
|
2951
|
+
markerIcon,
|
2952
|
+
markerShape,
|
2953
|
+
markerDefaultColor,
|
2954
|
+
markerDefaultColor2,
|
2955
|
+
api,
|
2956
|
+
itemType,
|
2957
|
+
userProfileLayer,
|
2958
|
+
customEditLink,
|
2959
|
+
customEditParameter,
|
2960
|
+
// eslint-disable-next-line camelcase
|
2961
|
+
public_edit_items,
|
2962
|
+
listed,
|
2963
|
+
});
|
2964
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
2965
|
+
}, [data, api]);
|
2966
|
+
useEffect(() => {
|
2967
|
+
if (tagsReady) {
|
2968
|
+
const processedTags = {};
|
2969
|
+
newTagsToAdd.map((newtag) => {
|
2970
|
+
if (!processedTags[newtag.name]) {
|
2971
|
+
processedTags[newtag.name] = true;
|
2972
|
+
addTag(newtag);
|
2973
|
+
}
|
2974
|
+
return null;
|
2975
|
+
});
|
2953
2976
|
}
|
2954
|
-
|
2955
|
-
|
2956
|
-
|
2977
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
2978
|
+
}, [tagsReady]);
|
2979
|
+
return (jsx(LayerContext.Provider, { value: {
|
2980
|
+
name,
|
2981
|
+
markerDefaultColor,
|
2982
|
+
markerDefaultColor2,
|
2983
|
+
markerShape,
|
2984
|
+
markerIcon,
|
2985
|
+
menuText,
|
2986
|
+
}, children: children }));
|
2987
|
+
};
|
2957
2988
|
|
2958
2989
|
/**
|
2959
|
-
* @
|
2990
|
+
* This Components injects Tags comming from an {@link ItemsApi | `API`}
|
2991
|
+
* ```tsx
|
2992
|
+
* <Tags api={tagsApi} />
|
2993
|
+
* ```
|
2994
|
+
* or from on {@link Tag| `Array`}
|
2995
|
+
* ```tsx
|
2996
|
+
* <Tags data={tags} />
|
2997
|
+
* ```
|
2998
|
+
* Can be child of {@link AppShell | `AppShell`}
|
2999
|
+
* ```tsx
|
3000
|
+
* <AppShell>
|
3001
|
+
* ...
|
3002
|
+
* <Tags api={tagsApi} />
|
3003
|
+
* </AppShell>
|
3004
|
+
* ```
|
3005
|
+
* Or child of {@link UtopiaMap | `UtopiaMap`}
|
3006
|
+
* ```tsx
|
3007
|
+
* <UtopiaMap>
|
3008
|
+
* ...
|
3009
|
+
* <Tags api={tagsApi} />
|
3010
|
+
* </UtopiaMap>
|
3011
|
+
* ```
|
3012
|
+
* @category Map
|
2960
3013
|
*/
|
2961
|
-
function
|
2962
|
-
const
|
3014
|
+
function Tags({ data, api }) {
|
3015
|
+
const setTagData = useSetTagData();
|
3016
|
+
const setTagApi = useSetTagApi();
|
2963
3017
|
useEffect(() => {
|
2964
|
-
|
2965
|
-
|
2966
|
-
|
2967
|
-
|
2968
|
-
|
2969
|
-
|
2970
|
-
updateFormValue(newValue);
|
2971
|
-
}
|
2972
|
-
};
|
2973
|
-
return (jsxs("div", { className: `tw:form-control ${containerStyle ?? ''}`, children: [labelTitle ? (jsx("label", { className: 'tw:label', children: jsx("span", { className: `tw:label-text tw:text-base-content ${labelStyle ?? ''}`, children: labelTitle }) })) : null, jsx("input", { required: required, pattern: pattern, type: type ?? 'text', name: dataField, value: inputValue, placeholder: placeholder ?? '', autoComplete: autocomplete, onChange: handleChange, className: `tw:input tw:input-bordered tw:w-full ${inputStyle ?? ''}` })] }));
|
2974
|
-
}
|
2975
|
-
|
2976
|
-
function ItemFormPopup(props) {
|
2977
|
-
const [spinner, setSpinner] = useState(false);
|
2978
|
-
const [popupTitle, setPopupTitle] = useState('');
|
2979
|
-
const formRef = useRef(null);
|
2980
|
-
const map = useMap();
|
2981
|
-
const addItem = useAddItem();
|
2982
|
-
const updateItem = useUpdateItem();
|
2983
|
-
const items = useItems();
|
2984
|
-
useRemoveItem();
|
2985
|
-
const tags = useTags();
|
2986
|
-
const addTag = useAddTag();
|
3018
|
+
data && setTagData(data);
|
3019
|
+
api && setTagApi(api);
|
3020
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
3021
|
+
}, [api, data]);
|
3022
|
+
const location = useLocation();
|
3023
|
+
const addFilterTag = useAddFilterTag();
|
2987
3024
|
const resetFilterTags = useResetFilterTags();
|
2988
|
-
const
|
2989
|
-
const
|
2990
|
-
|
2991
|
-
|
2992
|
-
|
2993
|
-
|
2994
|
-
|
2995
|
-
|
2996
|
-
|
2997
|
-
|
2998
|
-
|
2999
|
-
|
3000
|
-
|
3001
|
-
return;
|
3002
|
-
}
|
3003
|
-
setSpinner(true);
|
3004
|
-
formItem.text &&
|
3005
|
-
formItem.text
|
3006
|
-
.toLocaleLowerCase()
|
3007
|
-
.match(hashTagRegex)
|
3008
|
-
?.map((tag) => {
|
3009
|
-
if (!tags.find((t) => t.name.toLocaleLowerCase() === tag.slice(1).toLocaleLowerCase())) {
|
3010
|
-
addTag({ id: crypto.randomUUID(), name: tag.slice(1), color: randomColor() });
|
3011
|
-
}
|
3012
|
-
return null;
|
3013
|
-
});
|
3014
|
-
if (props.item) {
|
3015
|
-
let success = false;
|
3016
|
-
try {
|
3017
|
-
await props.layer.api?.updateItem({ ...formItem, id: props.item.id });
|
3018
|
-
success = true;
|
3019
|
-
// eslint-disable-next-line no-catch-all/no-catch-all
|
3020
|
-
}
|
3021
|
-
catch (error) {
|
3022
|
-
toast.error(error.toString());
|
3023
|
-
}
|
3024
|
-
if (success) {
|
3025
|
-
updateItem({ ...props.item, ...formItem });
|
3026
|
-
toast.success('Item updated');
|
3027
|
-
}
|
3028
|
-
setSpinner(false);
|
3029
|
-
map.closePopup();
|
3030
|
-
}
|
3031
|
-
else {
|
3032
|
-
const item = items.find((i) => i.user_created?.id === user?.id && i.layer === props.layer);
|
3033
|
-
const uuid = crypto.randomUUID();
|
3034
|
-
let success = false;
|
3035
|
-
try {
|
3036
|
-
props.layer.userProfileLayer &&
|
3037
|
-
item &&
|
3038
|
-
(await props.layer.api?.updateItem({ ...formItem, id: item.id }));
|
3039
|
-
(!props.layer.userProfileLayer || !item) &&
|
3040
|
-
(await props.layer.api?.createItem({
|
3041
|
-
...formItem,
|
3042
|
-
name,
|
3043
|
-
id: uuid,
|
3044
|
-
}));
|
3045
|
-
success = true;
|
3046
|
-
// eslint-disable-next-line no-catch-all/no-catch-all
|
3047
|
-
}
|
3048
|
-
catch (error) {
|
3049
|
-
toast.error(error.toString());
|
3050
|
-
}
|
3051
|
-
if (success) {
|
3052
|
-
if (props.layer.userProfileLayer && item)
|
3053
|
-
updateItem({ ...item, ...formItem });
|
3054
|
-
if (!props.layer.userProfileLayer || !item) {
|
3055
|
-
addItem({
|
3056
|
-
...formItem,
|
3057
|
-
name: (formItem.name ? formItem.name : user?.first_name) ?? '',
|
3058
|
-
user_created: user ?? undefined,
|
3059
|
-
id: uuid,
|
3060
|
-
layer: props.layer,
|
3061
|
-
public_edit: !user,
|
3062
|
-
});
|
3063
|
-
}
|
3064
|
-
toast.success('New item created');
|
3065
|
-
resetFilterTags();
|
3066
|
-
}
|
3067
|
-
setSpinner(false);
|
3068
|
-
map.closePopup();
|
3069
|
-
}
|
3070
|
-
props.setItemFormPopup(null);
|
3071
|
-
};
|
3072
|
-
const resetPopup = () => {
|
3073
|
-
if (formRef.current) {
|
3074
|
-
formRef.current.reset();
|
3075
|
-
}
|
3076
|
-
};
|
3077
|
-
useEffect(() => {
|
3078
|
-
resetPopup();
|
3079
|
-
}, [props.position]);
|
3080
|
-
return (jsx(Popup, { minWidth: 275, maxWidth: 275, autoPanPadding: [20, 80], eventHandlers: {
|
3081
|
-
remove: () => {
|
3082
|
-
setTimeout(function () {
|
3083
|
-
resetPopup();
|
3084
|
-
}, 100);
|
3085
|
-
},
|
3086
|
-
}, position: props.position, children: jsxs("form", { ref: formRef, onReset: resetPopup, autoComplete: 'off', onSubmit: (e) => handleSubmit(e), children: [props.item ? (jsx("div", { className: 'tw:h-3' })) : (jsx("div", { className: 'tw:flex tw:justify-center', children: jsx("b", { className: 'tw:text-xl tw:text-center tw:font-bold', children: props.layer.menuText }) })), props.children ? (Children.toArray(props.children).map((child) => isValidElement(child)
|
3087
|
-
? cloneElement(child, {
|
3088
|
-
item: props.item,
|
3089
|
-
key: props.position.toString(),
|
3090
|
-
setPopupTitle,
|
3091
|
-
})
|
3092
|
-
: '')) : (jsxs(Fragment, { children: [jsx(TextInput, { type: 'text', placeholder: 'Name', dataField: 'name', defaultValue: props.item ? props.item.name : '', inputStyle: '' }), jsx(TextAreaInput, { placeholder: 'Text', dataField: 'text', defaultValue: props.item?.text ?? '', inputStyle: 'tw:h-40 tw:mt-5' }, props.position.toString())] })), jsx("div", { className: 'tw:flex tw:justify-center', children: jsx("button", { className: spinner
|
3093
|
-
? 'tw:btn tw:btn-disabled tw:mt-5 tw:place-self-center'
|
3094
|
-
: 'tw:btn tw:mt-5 tw:place-self-center', type: 'submit', children: spinner ? jsx("span", { className: 'tw:loading tw:loading-spinner' }) : 'Save' }) })] }) }));
|
3095
|
-
}
|
3096
|
-
|
3097
|
-
// in miliseconds
|
3098
|
-
const units = [
|
3099
|
-
{ label: 'year', seconds: 31536000 },
|
3100
|
-
{ label: 'month', seconds: 2592000 },
|
3101
|
-
{ label: 'week', seconds: 604800 },
|
3102
|
-
{ label: 'day', seconds: 86400 },
|
3103
|
-
{ label: 'hour', seconds: 3600 },
|
3104
|
-
{ label: 'minute', seconds: 60 },
|
3105
|
-
{ label: 'second', seconds: 1 },
|
3106
|
-
];
|
3107
|
-
const timeAgo = (date) => {
|
3108
|
-
const time = Math.floor((new Date().valueOf() - new Date(date).valueOf()) / 1000);
|
3109
|
-
const { interval, unit } = calculateTimeDifference(time);
|
3110
|
-
const suffix = interval === 1 ? '' : 's';
|
3111
|
-
return `${interval} ${unit}${suffix} ago`;
|
3112
|
-
};
|
3113
|
-
const calculateTimeDifference = (time) => {
|
3114
|
-
for (const { label, seconds } of units) {
|
3115
|
-
const interval = Math.floor(time / seconds);
|
3116
|
-
if (interval >= 1) {
|
3117
|
-
return {
|
3118
|
-
interval,
|
3119
|
-
unit: label,
|
3120
|
-
};
|
3121
|
-
}
|
3122
|
-
}
|
3123
|
-
return {
|
3124
|
-
interval: 0,
|
3125
|
-
unit: '',
|
3126
|
-
};
|
3127
|
-
};
|
3128
|
-
|
3129
|
-
function EllipsisVerticalIcon({
|
3130
|
-
title,
|
3131
|
-
titleId,
|
3132
|
-
...props
|
3133
|
-
}, svgRef) {
|
3134
|
-
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3135
|
-
xmlns: "http://www.w3.org/2000/svg",
|
3136
|
-
viewBox: "0 0 16 16",
|
3137
|
-
fill: "currentColor",
|
3138
|
-
"aria-hidden": "true",
|
3139
|
-
"data-slot": "icon",
|
3140
|
-
ref: svgRef,
|
3141
|
-
"aria-labelledby": titleId
|
3142
|
-
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3143
|
-
id: titleId
|
3144
|
-
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3145
|
-
d: "M8 2a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM8 6.5a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM9.5 12.5a1.5 1.5 0 1 0-3 0 1.5 1.5 0 0 0 3 0Z"
|
3146
|
-
}));
|
3147
|
-
}
|
3148
|
-
const ForwardRef$9 = /*#__PURE__*/ React.forwardRef(EllipsisVerticalIcon);
|
3149
|
-
|
3150
|
-
function PencilIcon({
|
3151
|
-
title,
|
3152
|
-
titleId,
|
3153
|
-
...props
|
3154
|
-
}, svgRef) {
|
3155
|
-
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3156
|
-
xmlns: "http://www.w3.org/2000/svg",
|
3157
|
-
viewBox: "0 0 24 24",
|
3158
|
-
fill: "currentColor",
|
3159
|
-
"aria-hidden": "true",
|
3160
|
-
"data-slot": "icon",
|
3161
|
-
ref: svgRef,
|
3162
|
-
"aria-labelledby": titleId
|
3163
|
-
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3164
|
-
id: titleId
|
3165
|
-
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3166
|
-
d: "M21.731 2.269a2.625 2.625 0 0 0-3.712 0l-1.157 1.157 3.712 3.712 1.157-1.157a2.625 2.625 0 0 0 0-3.712ZM19.513 8.199l-3.712-3.712-12.15 12.15a5.25 5.25 0 0 0-1.32 2.214l-.8 2.685a.75.75 0 0 0 .933.933l2.685-.8a5.25 5.25 0 0 0 2.214-1.32L19.513 8.2Z"
|
3167
|
-
}));
|
3168
|
-
}
|
3169
|
-
const ForwardRef$8 = /*#__PURE__*/ React.forwardRef(PencilIcon);
|
3170
|
-
|
3171
|
-
function TrashIcon({
|
3172
|
-
title,
|
3173
|
-
titleId,
|
3174
|
-
...props
|
3175
|
-
}, svgRef) {
|
3176
|
-
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3177
|
-
xmlns: "http://www.w3.org/2000/svg",
|
3178
|
-
viewBox: "0 0 24 24",
|
3179
|
-
fill: "currentColor",
|
3180
|
-
"aria-hidden": "true",
|
3181
|
-
"data-slot": "icon",
|
3182
|
-
ref: svgRef,
|
3183
|
-
"aria-labelledby": titleId
|
3184
|
-
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3185
|
-
id: titleId
|
3186
|
-
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3187
|
-
fillRule: "evenodd",
|
3188
|
-
d: "M16.5 4.478v.227a48.816 48.816 0 0 1 3.878.512.75.75 0 1 1-.256 1.478l-.209-.035-1.005 13.07a3 3 0 0 1-2.991 2.77H8.084a3 3 0 0 1-2.991-2.77L4.087 6.66l-.209.035a.75.75 0 0 1-.256-1.478A48.567 48.567 0 0 1 7.5 4.705v-.227c0-1.564 1.213-2.9 2.816-2.951a52.662 52.662 0 0 1 3.369 0c1.603.051 2.815 1.387 2.815 2.951Zm-6.136-1.452a51.196 51.196 0 0 1 3.273 0C14.39 3.05 15 3.684 15 4.478v.113a49.488 49.488 0 0 0-6 0v-.113c0-.794.609-1.428 1.364-1.452Zm-.355 5.945a.75.75 0 1 0-1.5.058l.347 9a.75.75 0 1 0 1.499-.058l-.346-9Zm5.48.058a.75.75 0 1 0-1.498-.058l-.347 9a.75.75 0 0 0 1.5.058l.345-9Z",
|
3189
|
-
clipRule: "evenodd"
|
3190
|
-
}));
|
3191
|
-
}
|
3192
|
-
const ForwardRef$7 = /*#__PURE__*/ React.forwardRef(TrashIcon);
|
3193
|
-
|
3194
|
-
var TargetDotSVG = '';
|
3195
|
-
|
3196
|
-
const isClickInsideRectangle = (e, element) => {
|
3197
|
-
const r = element.getBoundingClientRect();
|
3198
|
-
return e.clientX > r.left && e.clientX < r.right && e.clientY > r.top && e.clientY < r.bottom;
|
3199
|
-
};
|
3200
|
-
const DialogModal = ({ title, isOpened, onClose, children, showCloseButton = true, closeOnClickOutside = true, className, }) => {
|
3201
|
-
const ref = useRef(null);
|
3202
|
-
useEffect(() => {
|
3203
|
-
if (isOpened) {
|
3204
|
-
ref.current?.showModal();
|
3205
|
-
ref.current?.classList.remove('tw:hidden');
|
3206
|
-
document.body.classList.add('modal-open'); // prevent bg scroll
|
3207
|
-
}
|
3208
|
-
else {
|
3209
|
-
ref.current?.close();
|
3210
|
-
ref.current?.classList.add('tw:hidden');
|
3211
|
-
document.body.classList.remove('modal-open');
|
3212
|
-
}
|
3213
|
-
}, [isOpened]);
|
3214
|
-
if (isOpened) {
|
3215
|
-
return (jsx("dialog", { className: `${className ?? ''} card tw:shadow-xl tw:absolute tw:right-0 tw:top-0 tw:bottom-0 tw:left-0 tw:m-auto tw:transition-opacity tw:duration-300 tw:p-4 tw:max-w-xl tw:bg-base-100`, ref: ref, onCancel: onClose, onClick: (e) => ref.current && !isClickInsideRectangle(e, ref.current) && closeOnClickOutside && onClose(), children: jsxs("div", { className: 'card-body tw:p-2', children: [jsx("h2", { className: 'tw:text-2xl tw:font-semibold tw:mb-2 tw:text-center', children: title }), children, showCloseButton && (jsx("button", { className: 'btn btn-sm btn-circle btn-ghost tw:absolute tw:right-2 tw:top-2', onClick: onClose, children: "\u2715" }))] }) }));
|
3216
|
-
}
|
3217
|
-
else
|
3218
|
-
return jsx(Fragment, {});
|
3219
|
-
};
|
3220
|
-
|
3221
|
-
function HeaderView({ item, api, editCallback, deleteCallback, setPositionCallback, loading, hideMenu = false, big = false, truncateSubname = true, hideSubname = false, showAddress = false, }) {
|
3222
|
-
const [modalOpen, setModalOpen] = useState(false);
|
3223
|
-
const hasUserPermission = useHasUserPermission();
|
3224
|
-
const navigate = useNavigate();
|
3225
|
-
const appState = useAppState();
|
3226
|
-
const [imageLoaded, setImageLoaded] = useState(false);
|
3227
|
-
const avatar = item.image && appState.assetsApi.url + item.image + '?width=160&heigth=160';
|
3228
|
-
const title = item.name;
|
3229
|
-
const subtitle = item.subname;
|
3230
|
-
const [address] = useState('');
|
3231
|
-
const params = new URLSearchParams(window.location.search);
|
3232
|
-
const openDeleteModal = async (event) => {
|
3233
|
-
setModalOpen(true);
|
3234
|
-
event.stopPropagation();
|
3235
|
-
};
|
3236
|
-
return (jsxs(Fragment, { children: [jsxs("div", { className: 'tw:flex tw:flex-row', children: [jsx("div", { className: 'tw:grow tw:max-w-[calc(100%-60px)] }', children: jsxs("div", { className: 'flex items-center', children: [avatar && (jsx("div", { className: 'tw:avatar', children: jsxs("div", { className: `${big ? 'tw:w-20' : 'tw:w-10'} tw:inline tw:items-center tw:justify-center overflow-hidden`, children: [jsx("img", { className: 'tw:w-full tw:h-full tw:object-cover tw:rounded-full', src: avatar, alt: item.name + ' logo', onLoad: () => setImageLoaded(true), onError: () => setImageLoaded(false), style: { display: imageLoaded ? 'block' : 'none' } }), !imageLoaded && (jsx("div", { className: 'tw:w-full tw:h-full tw:bg-gray-200 tw:rounded-full' }))] }) })), jsxs("div", { className: `${avatar ? 'tw:ml-2' : ''} tw:overflow-hidden`, children: [jsx("div", { className: `${big ? 'tw:xl:text-3xl tw:text-2xl' : 'tw:text-xl'} tw:font-semibold tw:truncate`, title: title, children: title }), showAddress && address && !hideSubname && (jsx("div", { className: `tw:text-xs tw:text-gray-500 ${truncateSubname && 'tw:truncate'}`, children: address })), subtitle && !hideSubname && (jsx("div", { className: `tw:text-xs tw:opacity-50 ${truncateSubname && 'tw:truncate'}`, children: subtitle }))] })] }) }), jsx("div", { onClick: (e) => e.stopPropagation(), className: `${big ? 'tw:mt-5' : 'tw:mt-1'}`, children: (api?.deleteItem || item.layer?.api?.updateItem) &&
|
3237
|
-
(hasUserPermission(api?.collectionName, 'delete', item) ||
|
3238
|
-
hasUserPermission(api?.collectionName, 'update', item)) &&
|
3239
|
-
!hideMenu && (jsxs("div", { className: 'tw:dropdown tw:dropdown-bottom', children: [jsx("label", { tabIndex: 0, className: 'tw:bg-base-100 tw:btn tw:m-1 tw:leading-3 tw:border-none tw:min-h-0 tw:h-6', children: jsx(ForwardRef$9, { className: 'tw:h-5 tw:w-5' }) }), jsxs("ul", { tabIndex: 0, className: 'tw:dropdown-content tw:menu tw:p-2 tw:shadow tw:bg-base-100 tw:rounded-box tw:z-1000', children: [api?.updateItem &&
|
3240
|
-
hasUserPermission(api.collectionName, 'update', item) &&
|
3241
|
-
editCallback && (jsx("li", { children: jsx("a", { className: 'tw:text-base-content! tw:cursor-pointer', onClick: (e) => item.layer?.customEditLink
|
3242
|
-
? navigate(`${item.layer.customEditLink}${item.layer.customEditParameter ? `/${item.id}${params && '?' + params}` : ''} `)
|
3243
|
-
: editCallback(e), children: jsx(ForwardRef$8, { className: 'tw:h-5 tw:w-5' }) }) })), api?.updateItem &&
|
3244
|
-
hasUserPermission(api.collectionName, 'update', item) &&
|
3245
|
-
setPositionCallback && (jsx("li", { children: jsx("a", { className: 'tw:text-base-content! tw:cursor-pointer', onClick: setPositionCallback, children: jsx(SVG, { src: TargetDotSVG, className: 'tw:w-5 tw:h-5' }) }) })), api?.deleteItem &&
|
3246
|
-
hasUserPermission(api.collectionName, 'delete', item) &&
|
3247
|
-
deleteCallback && (jsx("li", { children: jsx("a", { className: 'tw:cursor-pointer tw:text-error!', onClick: openDeleteModal, children: loading ? (jsx("span", { className: 'tw:loading tw:loading-spinner tw:loading-sm' })) : (jsx(ForwardRef$7, { className: 'tw:h-5 tw:w-5' })) }) }))] })] })) })] }), jsx(DialogModal, { isOpened: modalOpen, title: 'Are you sure?', showCloseButton: false, onClose: () => setModalOpen(false), children: jsxs("div", { onClick: (e) => e.stopPropagation(), children: [jsxs("span", { children: ["Do you want to delete ", jsx("b", { children: item.name }), "?"] }), jsx("div", { className: 'tw:grid', children: jsxs("div", { className: 'tw:flex tw:justify-between', children: [jsx("label", { className: 'tw:btn tw:mt-4 tw:btn-error', onClick: (e) => {
|
3248
|
-
deleteCallback(e);
|
3249
|
-
setModalOpen(false);
|
3250
|
-
}, children: "Yes" }), jsx("label", { className: 'tw:btn tw:mt-4', onClick: () => setModalOpen(false), children: "No" })] }) })] }) })] }));
|
3251
|
-
}
|
3252
|
-
|
3253
|
-
// eslint-disable-next-line react/display-name
|
3254
|
-
const ItemViewPopup = forwardRef((props, ref) => {
|
3255
|
-
const map = useMap();
|
3256
|
-
const [loading, setLoading] = useState(false);
|
3257
|
-
const removeItem = useRemoveItem();
|
3258
|
-
const updadateItem = useUpdateItem();
|
3259
|
-
const navigate = useNavigate();
|
3260
|
-
const setSelectPosition = useSetSelectPosition();
|
3261
|
-
const [infoExpanded, setInfoExpanded] = useState(false);
|
3262
|
-
const handleEdit = (event) => {
|
3263
|
-
event.stopPropagation();
|
3264
|
-
map.closePopup();
|
3265
|
-
props.setItemFormPopup &&
|
3266
|
-
props.setItemFormPopup({
|
3267
|
-
position: new LatLng(props.item.position?.coordinates[1], props.item.position?.coordinates[0]),
|
3268
|
-
layer: props.item.layer,
|
3269
|
-
item: props.item,
|
3270
|
-
setItemFormPopup: props.setItemFormPopup,
|
3271
|
-
});
|
3272
|
-
};
|
3273
|
-
const handleDelete = async (event) => {
|
3274
|
-
event.stopPropagation();
|
3275
|
-
setLoading(true);
|
3276
|
-
let success = false;
|
3277
|
-
try {
|
3278
|
-
!props.item.layer?.userProfileLayer &&
|
3279
|
-
(await props.item.layer?.api?.deleteItem(props.item.id));
|
3280
|
-
props.item.layer?.userProfileLayer &&
|
3281
|
-
(await props.item.layer.api?.updateItem({ id: props.item.id, position: null }));
|
3282
|
-
success = true;
|
3283
|
-
// eslint-disable-next-line no-catch-all/no-catch-all
|
3284
|
-
}
|
3285
|
-
catch (error) {
|
3286
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-argument
|
3287
|
-
toast.error(error.toString());
|
3288
|
-
}
|
3289
|
-
if (success) {
|
3290
|
-
!props.item.layer?.userProfileLayer && removeItem(props.item);
|
3291
|
-
props.item.layer?.userProfileLayer && updadateItem({ ...props.item, position: undefined });
|
3292
|
-
toast.success('Item deleted');
|
3293
|
-
}
|
3294
|
-
setLoading(false);
|
3295
|
-
map.closePopup();
|
3296
|
-
const params = new URLSearchParams(window.location.search);
|
3297
|
-
window.history.pushState({}, '', '/' + `${params ? `?${params}` : ''}`);
|
3298
|
-
navigate('/');
|
3299
|
-
};
|
3300
|
-
return (jsx(Popup, { ref: ref, maxHeight: 377, minWidth: 275, maxWidth: 275, autoPanPadding: [20, 80], children: jsxs("div", { className: 'tw:bg-base-100 tw:text-base-content', children: [jsx(HeaderView, { api: props.item.layer?.api, item: props.item, editCallback: handleEdit, deleteCallback: handleDelete, setPositionCallback: () => {
|
3301
|
-
map.closePopup();
|
3302
|
-
setSelectPosition(props.item);
|
3303
|
-
navigate('/');
|
3304
|
-
}, loading: loading }), jsx("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: props.children ? (Children.toArray(props.children).map((child) => isValidElement(child)
|
3305
|
-
? cloneElement(child, { item: props.item })
|
3306
|
-
: '')) : (jsx(TextView, { text: props.item.text, itemId: props.item.id })) }), jsxs("div", { className: 'tw:flex tw:-mb-1 tw:flex-row tw:mr-2 tw:mt-1', children: [infoExpanded ? (jsx("p", { className: 'tw:italic tw:min-h-[21px] tw:my-0! tw:opacity-50', children: `${props.item.date_updated && props.item.date_updated !== props.item.date_created ? 'updated' : 'posted'} ${props.item && props.item.user_created && props.item.user_created.first_name ? `by ${props.item.user_created.first_name}` : ''} ${props.item.date_updated ? timeAgo(props.item.date_updated) : timeAgo(props.item.date_created)}` })) : (jsx("p", { className: 'tw:my-0! tw:min-h-[21px] tw:font-bold tw:cursor-pointer tw:text-gray-500', onClick: () => setInfoExpanded(true), children: "\u24D8" })), jsx("div", { className: 'tw:grow' })] })] }) }));
|
3307
|
-
});
|
3308
|
-
|
3309
|
-
/**
|
3310
|
-
* @category Map
|
3311
|
-
*/
|
3312
|
-
const Layer = ({ data, children, name = 'places', menuIcon = 'MapPinIcon', menuText = 'add new place', menuColor = '#2E7D32', markerIcon, markerShape = 'circle', markerDefaultColor = '#777', markerDefaultColor2 = 'RGBA(35, 31, 32, 0.2)', api, itemType, userProfileLayer = false, customEditLink, customEditParameter,
|
3313
|
-
// eslint-disable-next-line camelcase
|
3314
|
-
public_edit_items, listed = true, setItemFormPopup, itemFormPopup, clusterRef, }) => {
|
3315
|
-
const filterTags = useFilterTags();
|
3316
|
-
const items = useItems();
|
3317
|
-
const setItemsApi = useSetItemsApi();
|
3318
|
-
const setItemsData = useSetItemsData();
|
3319
|
-
const getItemTags = useGetItemTags();
|
3320
|
-
const addMarker = useAddMarker();
|
3321
|
-
const addPopup = useAddPopup();
|
3322
|
-
const leafletRefs = useLeafletRefs();
|
3323
|
-
const allTagsLoaded = useAllTagsLoaded();
|
3324
|
-
const allItemsLoaded = useAllItemsLoaded();
|
3325
|
-
const setMarkerClicked = useSetMarkerClicked();
|
3326
|
-
const selectPosition = useSelectPosition();
|
3327
|
-
const tags = useTags();
|
3328
|
-
const addTag = useAddTag();
|
3329
|
-
const [newTagsToAdd, setNewTagsToAdd] = useState([]);
|
3330
|
-
const [tagsReady, setTagsReady] = useState(false);
|
3331
|
-
const isLayerVisible = useIsLayerVisible();
|
3332
|
-
const isGroupTypeVisible = useIsGroupTypeVisible();
|
3333
|
-
const visibleGroupTypes = useVisibleGroupType();
|
3334
|
-
const appState = useAppState();
|
3335
|
-
useEffect(() => {
|
3336
|
-
data &&
|
3337
|
-
setItemsData({
|
3338
|
-
data,
|
3339
|
-
children,
|
3340
|
-
name,
|
3341
|
-
menuIcon,
|
3342
|
-
menuText,
|
3343
|
-
menuColor,
|
3344
|
-
markerIcon,
|
3345
|
-
markerShape,
|
3346
|
-
markerDefaultColor,
|
3347
|
-
markerDefaultColor2,
|
3348
|
-
api,
|
3349
|
-
itemType,
|
3350
|
-
userProfileLayer,
|
3351
|
-
// Can we just use editCallback for all cases?
|
3352
|
-
customEditLink,
|
3353
|
-
customEditParameter,
|
3354
|
-
// eslint-disable-next-line camelcase
|
3355
|
-
public_edit_items,
|
3356
|
-
listed,
|
3357
|
-
setItemFormPopup,
|
3358
|
-
itemFormPopup,
|
3359
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
|
3360
|
-
clusterRef,
|
3361
|
-
});
|
3362
|
-
api &&
|
3363
|
-
setItemsApi({
|
3364
|
-
data,
|
3365
|
-
children,
|
3366
|
-
name,
|
3367
|
-
menuIcon,
|
3368
|
-
menuText,
|
3369
|
-
menuColor,
|
3370
|
-
markerIcon,
|
3371
|
-
markerShape,
|
3372
|
-
markerDefaultColor,
|
3373
|
-
markerDefaultColor2,
|
3374
|
-
api,
|
3375
|
-
itemType,
|
3376
|
-
userProfileLayer,
|
3377
|
-
customEditLink,
|
3378
|
-
customEditParameter,
|
3379
|
-
// eslint-disable-next-line camelcase
|
3380
|
-
public_edit_items,
|
3381
|
-
listed,
|
3382
|
-
setItemFormPopup,
|
3383
|
-
itemFormPopup,
|
3384
|
-
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
|
3385
|
-
clusterRef,
|
3386
|
-
});
|
3387
|
-
// eslint-disable-next-line react-hooks/exhaustive-deps
|
3388
|
-
}, [data, api]);
|
3389
|
-
useEffect(() => {
|
3390
|
-
if (tagsReady) {
|
3391
|
-
const processedTags = {};
|
3392
|
-
newTagsToAdd.map((newtag) => {
|
3393
|
-
if (!processedTags[newtag.name]) {
|
3394
|
-
processedTags[newtag.name] = true;
|
3395
|
-
addTag(newtag);
|
3396
|
-
}
|
3397
|
-
return null;
|
3398
|
-
});
|
3399
|
-
}
|
3400
|
-
// eslint-disable-next-line react-hooks/exhaustive-deps
|
3401
|
-
}, [tagsReady]);
|
3402
|
-
return (jsxs(Fragment, { children: [items &&
|
3403
|
-
items
|
3404
|
-
.filter((item) => item.layer?.name === name)
|
3405
|
-
.filter((item) => filterTags.length === 0
|
3406
|
-
? item
|
3407
|
-
: filterTags.some((tag) => getItemTags(item).some((filterTag) => filterTag.name.toLocaleLowerCase() === tag.name.toLocaleLowerCase())))
|
3408
|
-
.filter((item) => item.layer && isLayerVisible(item.layer))
|
3409
|
-
.filter((item) => (item.group_type && isGroupTypeVisible(item.group_type)) ||
|
3410
|
-
visibleGroupTypes.length === 0)
|
3411
|
-
.map((item) => {
|
3412
|
-
if (item.position?.coordinates[0] && item.position?.coordinates[1]) {
|
3413
|
-
if (item.tags) {
|
3414
|
-
item.text += '\n\n';
|
3415
|
-
item.tags.map((tag) => {
|
3416
|
-
if (!item.text?.includes(`#${encodeTag(tag)}`)) {
|
3417
|
-
item.text += `#${encodeTag(tag)}`;
|
3418
|
-
}
|
3419
|
-
return item.text;
|
3420
|
-
});
|
3421
|
-
}
|
3422
|
-
if (allTagsLoaded && allItemsLoaded) {
|
3423
|
-
item.text?.match(hashTagRegex)?.map((tag) => {
|
3424
|
-
if (!tags.find((t) => t.name.toLocaleLowerCase() === tag.slice(1).toLocaleLowerCase()) &&
|
3425
|
-
!newTagsToAdd.find((t) => t.name.toLocaleLowerCase() === tag.slice(1).toLocaleLowerCase())) {
|
3426
|
-
const newTag = {
|
3427
|
-
id: crypto.randomUUID(),
|
3428
|
-
name: tag.slice(1),
|
3429
|
-
color: randomColor(),
|
3430
|
-
};
|
3431
|
-
setNewTagsToAdd((current) => [...current, newTag]);
|
3432
|
-
}
|
3433
|
-
return null;
|
3434
|
-
});
|
3435
|
-
!tagsReady && setTagsReady(true);
|
3436
|
-
}
|
3437
|
-
const itemTags = getItemTags(item);
|
3438
|
-
const latitude = item.position.coordinates[1];
|
3439
|
-
const longitude = item.position.coordinates[0];
|
3440
|
-
let color1 = markerDefaultColor;
|
3441
|
-
let color2 = markerDefaultColor2;
|
3442
|
-
if (item.color) {
|
3443
|
-
color1 = item.color;
|
3444
|
-
}
|
3445
|
-
else if (itemTags[0]) {
|
3446
|
-
color1 = itemTags[0].color;
|
3447
|
-
}
|
3448
|
-
if (itemTags[0] && item.color) {
|
3449
|
-
color2 = itemTags[0].color;
|
3450
|
-
}
|
3451
|
-
else if (itemTags[1]) {
|
3452
|
-
color2 = itemTags[1].color;
|
3453
|
-
}
|
3454
|
-
return (jsxs(Marker, { ref: (r) => {
|
3455
|
-
if (!(item.id in leafletRefs && leafletRefs[item.id].marker === r)) {
|
3456
|
-
r && addMarker(item, r);
|
3457
|
-
}
|
3458
|
-
}, eventHandlers: {
|
3459
|
-
click: () => {
|
3460
|
-
selectPosition && setMarkerClicked(item);
|
3461
|
-
},
|
3462
|
-
}, icon: MarkerIconFactory(markerShape, color1, color2, item.markerIcon ? item.markerIcon : markerIcon, appState.assetsApi.url), position: [latitude, longitude], children: [children &&
|
3463
|
-
Children.toArray(children).some((child) => isComponentWithType(child) && child.type.__TYPE === 'ItemView') ? (Children.toArray(children).map((child) => isComponentWithType(child) && child.type.__TYPE === 'ItemView' ? (jsx(ItemViewPopup, { ref: (r) => {
|
3464
|
-
if (!(item.id in leafletRefs && leafletRefs[item.id].popup === r)) {
|
3465
|
-
r && addPopup(item, r);
|
3466
|
-
}
|
3467
|
-
}, item: item, setItemFormPopup: setItemFormPopup, children: child }, item.id + item.name)) : null)) : (jsx(Fragment, { children: jsx(ItemViewPopup, { ref: (r) => {
|
3468
|
-
if (!(item.id in leafletRefs && leafletRefs[item.id].popup === r)) {
|
3469
|
-
r && addPopup(item, r);
|
3470
|
-
}
|
3471
|
-
}, item: item, setItemFormPopup: setItemFormPopup }, item.id + item.name) })), jsx(Tooltip, { offset: [0, -38], direction: 'top', children: item.name })] }, item.id));
|
3472
|
-
}
|
3473
|
-
else
|
3474
|
-
return null;
|
3475
|
-
}), itemFormPopup &&
|
3476
|
-
itemFormPopup.layer.name === name &&
|
3477
|
-
(children &&
|
3478
|
-
Children.toArray(children).some((child) => isComponentWithType(child) && child.type.__TYPE === 'ItemForm') ? (Children.toArray(children).map((child) => isComponentWithType(child) && child.type.__TYPE === 'ItemForm' ? (jsx(ItemFormPopup, { position: itemFormPopup.position, layer: itemFormPopup.layer, setItemFormPopup: setItemFormPopup, item: itemFormPopup.item, children: child }, setItemFormPopup?.name)) : (''))) : (jsx(Fragment, { children: jsx(ItemFormPopup, { position: itemFormPopup.position, layer: itemFormPopup.layer, setItemFormPopup: setItemFormPopup, item: itemFormPopup.item }) })))] }));
|
3479
|
-
};
|
3480
|
-
function isComponentWithType(node) {
|
3481
|
-
return isValidElement(node) && typeof node.type !== 'string' && '__TYPE' in node.type;
|
3482
|
-
}
|
3483
|
-
|
3484
|
-
/**
|
3485
|
-
* This Components injects Tags comming from an {@link ItemsApi | `API`}
|
3486
|
-
* ```tsx
|
3487
|
-
* <Tags api={tagsApi} />
|
3488
|
-
* ```
|
3489
|
-
* or from on {@link Tag| `Array`}
|
3490
|
-
* ```tsx
|
3491
|
-
* <Tags data={tags} />
|
3492
|
-
* ```
|
3493
|
-
* Can be child of {@link AppShell | `AppShell`}
|
3494
|
-
* ```tsx
|
3495
|
-
* <AppShell>
|
3496
|
-
* ...
|
3497
|
-
* <Tags api={tagsApi} />
|
3498
|
-
* </AppShell>
|
3499
|
-
* ```
|
3500
|
-
* Or child of {@link UtopiaMap | `UtopiaMap`}
|
3501
|
-
* ```tsx
|
3502
|
-
* <UtopiaMap>
|
3503
|
-
* ...
|
3504
|
-
* <Tags api={tagsApi} />
|
3505
|
-
* </UtopiaMap>
|
3506
|
-
* ```
|
3507
|
-
* @category Map
|
3508
|
-
*/
|
3509
|
-
function Tags({ data, api }) {
|
3510
|
-
const setTagData = useSetTagData();
|
3511
|
-
const setTagApi = useSetTagApi();
|
3512
|
-
useEffect(() => {
|
3513
|
-
data && setTagData(data);
|
3514
|
-
api && setTagApi(api);
|
3515
|
-
// eslint-disable-next-line react-hooks/exhaustive-deps
|
3516
|
-
}, [api, data]);
|
3517
|
-
const location = useLocation();
|
3518
|
-
const addFilterTag = useAddFilterTag();
|
3519
|
-
const resetFilterTags = useResetFilterTags();
|
3520
|
-
const tags = useTags();
|
3521
|
-
const filterTags = useFilterTags();
|
3522
|
-
useEffect(() => {
|
3523
|
-
const params = new URLSearchParams(location.search);
|
3524
|
-
const urlTags = params.get('tags');
|
3525
|
-
const decodedTags = urlTags ? decodeURIComponent(urlTags) : '';
|
3526
|
-
const decodedTagsArray = decodedTags.split(';');
|
3527
|
-
if (decodedTagsArray.some((ut) => !filterTags.find((ft) => ut.toLocaleLowerCase() === ft.name.toLocaleLowerCase())) ||
|
3528
|
-
filterTags.some((ft) => !decodedTagsArray.find((ut) => ut.toLocaleLowerCase() === ft.name.toLocaleLowerCase()))) {
|
3529
|
-
resetFilterTags();
|
3530
|
-
decodedTagsArray.map((urlTag) => {
|
3531
|
-
const tag = tags.find((t) => t.name.toLocaleLowerCase() === urlTag.toLocaleLowerCase());
|
3532
|
-
tag && addFilterTag(tag);
|
3025
|
+
const tags = useTags();
|
3026
|
+
const filterTags = useFilterTags();
|
3027
|
+
useEffect(() => {
|
3028
|
+
const params = new URLSearchParams(location.search);
|
3029
|
+
const urlTags = params.get('tags');
|
3030
|
+
const decodedTags = urlTags ? decodeURIComponent(urlTags) : '';
|
3031
|
+
const decodedTagsArray = decodedTags.split(';');
|
3032
|
+
if (decodedTagsArray.some((ut) => !filterTags.find((ft) => ut.toLocaleLowerCase() === ft.name.toLocaleLowerCase())) ||
|
3033
|
+
filterTags.some((ft) => !decodedTagsArray.find((ut) => ut.toLocaleLowerCase() === ft.name.toLocaleLowerCase()))) {
|
3034
|
+
resetFilterTags();
|
3035
|
+
decodedTagsArray.map((urlTag) => {
|
3036
|
+
const tag = tags.find((t) => t.name.toLocaleLowerCase() === urlTag.toLocaleLowerCase());
|
3037
|
+
tag && addFilterTag(tag);
|
3533
3038
|
return null;
|
3534
3039
|
});
|
3535
3040
|
}
|
@@ -3569,110 +3074,13 @@ function Permissions({ data, api, adminRole, }) {
|
|
3569
3074
|
const setAdminRole = useSetAdminRole();
|
3570
3075
|
const { user } = useAuth();
|
3571
3076
|
useEffect(() => {
|
3572
|
-
adminRole && setAdminRole(adminRole);
|
3573
|
-
data && setPermissionData(data);
|
3574
|
-
api && setPermissionApi(api);
|
3575
|
-
// eslint-disable-next-line react-hooks/exhaustive-deps
|
3576
|
-
}, [api, data, adminRole, user]);
|
3577
|
-
return jsx(Fragment, {});
|
3578
|
-
}
|
3579
|
-
|
3580
|
-
/**
|
3581
|
-
* @category Map
|
3582
|
-
*/
|
3583
|
-
const ItemForm = ({ children, item, title, setPopupTitle, }) => {
|
3584
|
-
useEffect(() => {
|
3585
|
-
setPopupTitle && title && setPopupTitle(title);
|
3586
|
-
// eslint-disable-next-line react-hooks/exhaustive-deps
|
3587
|
-
}, [title]);
|
3588
|
-
return (jsx("div", { children: children
|
3589
|
-
? Children.toArray(children).map((child) => isValidElement(child)
|
3590
|
-
? cloneElement(child, { item, test: 'test' })
|
3591
|
-
: '')
|
3592
|
-
: '' }));
|
3593
|
-
};
|
3594
|
-
ItemForm.__TYPE = 'ItemForm';
|
3595
|
-
|
3596
|
-
/**
|
3597
|
-
* @category Map
|
3598
|
-
*/
|
3599
|
-
const ItemView = ({ children, item }) => {
|
3600
|
-
return (jsx("div", { children: children
|
3601
|
-
? Children.toArray(children).map((child) => isValidElement(child) ? cloneElement(child, { item }) : null)
|
3602
|
-
: null }));
|
3603
|
-
};
|
3604
|
-
ItemView.__TYPE = 'ItemView';
|
3605
|
-
|
3606
|
-
/**
|
3607
|
-
* @category Map
|
3608
|
-
*/
|
3609
|
-
const PopupTextAreaInput = ({ dataField, placeholder, style, item, }) => {
|
3610
|
-
return (jsx(TextAreaInput, { defaultValue: item?.text ? item.text : '', dataField: dataField, placeholder: placeholder, inputStyle: style }));
|
3611
|
-
};
|
3612
|
-
|
3613
|
-
/**
|
3614
|
-
* @category Map
|
3615
|
-
*/
|
3616
|
-
const PopupStartEndInput = ({ item, showLabels = true, updateStartValue, updateEndValue, }) => {
|
3617
|
-
return (jsxs("div", { className: 'tw:grid tw:grid-cols-2 tw:gap-2', children: [jsx(TextInput, { type: 'date', placeholder: 'start', dataField: 'start', inputStyle: 'tw:text-sm tw:px-2', labelTitle: showLabels ? 'start' : '', defaultValue: item && item.start ? item.start.substring(0, 10) : '', autocomplete: 'one-time-code', updateFormValue: updateStartValue }), jsx(TextInput, { type: 'date', placeholder: 'end', dataField: 'end', inputStyle: 'tw:text-sm tw:px-2', labelTitle: showLabels ? 'end' : '', defaultValue: item && item.end ? item.end.substring(0, 10) : '', autocomplete: 'one-time-code', updateFormValue: updateEndValue })] }));
|
3618
|
-
};
|
3619
|
-
|
3620
|
-
/**
|
3621
|
-
* @category Map
|
3622
|
-
*/
|
3623
|
-
const PopupTextInput = ({ dataField, placeholder, style, item, }) => {
|
3624
|
-
return (jsx(TextInput, { defaultValue: item?.name ? item.name : '', dataField: dataField, placeholder: placeholder, inputStyle: style, type: 'text', containerStyle: 'tw:mt-4' }));
|
3625
|
-
};
|
3626
|
-
|
3627
|
-
/**
|
3628
|
-
* @category Map
|
3629
|
-
*/
|
3630
|
-
const PopupCheckboxInput = ({ dataField, label, item, }) => {
|
3631
|
-
return (jsxs("label", { htmlFor: item?.id, className: 'tw:label tw:justify-normal tw:pt-1 tw:pb-1', children: [jsx("input", { id: item?.id, type: 'checkbox', name: dataField, className: 'tw:checkbox tw:checkbox-xs tw:checkbox-success', checked: item?.public_edit }), jsx("span", { className: 'tw:text-sm tw:label-text tw:mx-2 tw:cursor-pointer', children: label })] }));
|
3632
|
-
};
|
3633
|
-
|
3634
|
-
function CalendarDaysIcon({
|
3635
|
-
title,
|
3636
|
-
titleId,
|
3637
|
-
...props
|
3638
|
-
}, svgRef) {
|
3639
|
-
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3640
|
-
xmlns: "http://www.w3.org/2000/svg",
|
3641
|
-
viewBox: "0 0 24 24",
|
3642
|
-
fill: "currentColor",
|
3643
|
-
"aria-hidden": "true",
|
3644
|
-
"data-slot": "icon",
|
3645
|
-
ref: svgRef,
|
3646
|
-
"aria-labelledby": titleId
|
3647
|
-
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3648
|
-
id: titleId
|
3649
|
-
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3650
|
-
d: "M12.75 12.75a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM7.5 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM8.25 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM9.75 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM10.5 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM12 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM12.75 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM14.25 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM15 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM16.5 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM15 12.75a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM16.5 13.5a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5Z"
|
3651
|
-
}), /*#__PURE__*/React.createElement("path", {
|
3652
|
-
fillRule: "evenodd",
|
3653
|
-
d: "M6.75 2.25A.75.75 0 0 1 7.5 3v1.5h9V3A.75.75 0 0 1 18 3v1.5h.75a3 3 0 0 1 3 3v11.25a3 3 0 0 1-3 3H5.25a3 3 0 0 1-3-3V7.5a3 3 0 0 1 3-3H6V3a.75.75 0 0 1 .75-.75Zm13.5 9a1.5 1.5 0 0 0-1.5-1.5H5.25a1.5 1.5 0 0 0-1.5 1.5v7.5a1.5 1.5 0 0 0 1.5 1.5h13.5a1.5 1.5 0 0 0 1.5-1.5v-7.5Z",
|
3654
|
-
clipRule: "evenodd"
|
3655
|
-
}));
|
3656
|
-
}
|
3657
|
-
const ForwardRef$6 = /*#__PURE__*/ React.forwardRef(CalendarDaysIcon);
|
3658
|
-
|
3659
|
-
/**
|
3660
|
-
* @category Map
|
3661
|
-
*/
|
3662
|
-
const StartEndView = ({ item }) => {
|
3663
|
-
return (jsxs("div", { className: 'tw:flex tw:flex-row tw:mb-4 tw:mt-1', children: [jsxs("div", { className: 'tw:basis-2/5 tw:flex tw:flex-row', children: [jsx(ForwardRef$6, { className: 'tw:h-4 tw:w-4 tw:mr-2' }), jsx("time", { className: 'tw:align-middle', dateTime: item && item.start ? item.start.substring(0, 10) : '', children: item && item.start ? new Date(item.start).toLocaleDateString() : '' })] }), jsx("div", { className: 'tw:basis-1/5 tw:place-content-center', children: jsx("span", { children: "-" }) }), jsxs("div", { className: 'tw:basis-2/5 tw:flex tw:flex-row', children: [jsx(ForwardRef$6, { className: 'tw:h-4 tw:w-4 tw:mr-2' }), jsx("time", { className: 'tw:align-middle', dateTime: item && item.end ? item.end.substring(0, 10) : '', children: item && item.end ? new Date(item.end).toLocaleDateString() : '' })] })] }));
|
3664
|
-
};
|
3665
|
-
|
3666
|
-
/**
|
3667
|
-
* @category Map
|
3668
|
-
*/
|
3669
|
-
const PopupButton = ({ url, parameterField, text, item, }) => {
|
3670
|
-
const params = new URLSearchParams(window.location.search);
|
3671
|
-
const getItemTags = useGetItemTags();
|
3672
|
-
return (jsx(Link, { to: `${url}/${parameterField ? item?.id : ''}?${params}`, children: jsx("button", { style: {
|
3673
|
-
backgroundColor: `${item?.color ?? (item && (getItemTags(item) && getItemTags(item)[0] && getItemTags(item)[0].color ? getItemTags(item)[0].color : (item?.layer?.markerDefaultColor ?? '#000')))}`,
|
3674
|
-
}, className: 'tw:btn tw:text-white tw:btn-sm tw:float-right tw:mt-1', children: text }) }));
|
3675
|
-
};
|
3077
|
+
adminRole && setAdminRole(adminRole);
|
3078
|
+
data && setPermissionData(data);
|
3079
|
+
api && setPermissionApi(api);
|
3080
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
3081
|
+
}, [api, data, adminRole, user]);
|
3082
|
+
return jsx(Fragment, {});
|
3083
|
+
}
|
3676
3084
|
|
3677
3085
|
const themes = [
|
3678
3086
|
'default',
|
@@ -3702,6 +3110,27 @@ const ThemeControl = () => {
|
|
3702
3110
|
return (jsxs("div", { className: 'tw:dropdown tw:mr-2', children: [jsxs("div", { tabIndex: 0, role: 'button', className: 'tw:btn tw:m-1', children: ["Theme", jsx("svg", { width: '12px', height: '12px', className: 'tw:inline-block tw:h-2 tw:w-2 tw:fill-current tw:opacity-60', xmlns: 'http://www.w3.org/2000/svg', viewBox: '0 0 2048 2048', children: jsx("path", { d: 'M1799 349l242 241-1017 1017L7 590l242-241 775 775 775-775z' }) })] }), jsx("ul", { tabIndex: 0, className: 'tw:dropdown-content tw:bg-base-200 tw:rounded-box tw:z-1 tw:w-36 tw:p-2 tw:shadow-2xl', children: themes.map((t) => (jsx("li", { children: jsx("input", { className: `tw:btn ${theme === t ? 'tw:bg-base-300' : ''} tw:btn-sm tw:btn-block tw:btn-ghost tw:justify-start`, type: 'radio', name: 'theme', value: t, checked: theme === t, onChange: () => setTheme(t), "aria-label": t.toLowerCase() }) }, t))) })] }));
|
3703
3111
|
};
|
3704
3112
|
|
3113
|
+
function EllipsisVerticalIcon({
|
3114
|
+
title,
|
3115
|
+
titleId,
|
3116
|
+
...props
|
3117
|
+
}, svgRef) {
|
3118
|
+
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3119
|
+
xmlns: "http://www.w3.org/2000/svg",
|
3120
|
+
viewBox: "0 0 16 16",
|
3121
|
+
fill: "currentColor",
|
3122
|
+
"aria-hidden": "true",
|
3123
|
+
"data-slot": "icon",
|
3124
|
+
ref: svgRef,
|
3125
|
+
"aria-labelledby": titleId
|
3126
|
+
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3127
|
+
id: titleId
|
3128
|
+
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3129
|
+
d: "M8 2a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM8 6.5a1.5 1.5 0 1 1 0 3 1.5 1.5 0 0 1 0-3ZM9.5 12.5a1.5 1.5 0 1 0-3 0 1.5 1.5 0 0 0 3 0Z"
|
3130
|
+
}));
|
3131
|
+
}
|
3132
|
+
const ForwardRef$9 = /*#__PURE__*/ React.forwardRef(EllipsisVerticalIcon);
|
3133
|
+
|
3705
3134
|
const UserControl = () => {
|
3706
3135
|
const { isAuthenticated, user, logout } = useAuth();
|
3707
3136
|
const appState = useAppState();
|
@@ -4019,6 +3448,43 @@ function SetNewPasswordPage() {
|
|
4019
3448
|
onClick: () => onReset(), children: loading ? jsx("span", { className: 'tw:loading tw:loading-spinner' }) : 'Set' }) })] }));
|
4020
3449
|
}
|
4021
3450
|
|
3451
|
+
/**
|
3452
|
+
* @category Input
|
3453
|
+
*/
|
3454
|
+
function TextAreaInput({ labelTitle, dataField, labelStyle, containerStyle, inputStyle, defaultValue, placeholder, required = true, updateFormValue, }) {
|
3455
|
+
const ref = useRef(null);
|
3456
|
+
const [inputValue, setInputValue] = useState(defaultValue);
|
3457
|
+
useEffect(() => {
|
3458
|
+
setInputValue(defaultValue);
|
3459
|
+
}, [defaultValue]);
|
3460
|
+
const handleChange = (e) => {
|
3461
|
+
const newValue = e.target.value;
|
3462
|
+
setInputValue(newValue);
|
3463
|
+
if (updateFormValue) {
|
3464
|
+
updateFormValue(newValue);
|
3465
|
+
}
|
3466
|
+
};
|
3467
|
+
return (jsxs("div", { className: `tw:form-control tw:w-full ${containerStyle ?? ''}`, children: [labelTitle ? (jsx("label", { className: 'tw:label', children: jsx("span", { className: `tw:label-text tw:text-base-content ${labelStyle ?? ''}`, children: labelTitle }) })) : null, jsx("textarea", { required: required, ref: ref, value: inputValue, name: dataField, className: `tw:textarea tw:textarea-bordered tw:w-full tw:leading-5 ${inputStyle ?? ''}`, placeholder: placeholder ?? '', onChange: handleChange })] }));
|
3468
|
+
}
|
3469
|
+
|
3470
|
+
/**
|
3471
|
+
* @category Input
|
3472
|
+
*/
|
3473
|
+
function TextInput({ labelTitle, labelStyle, type, dataField, containerStyle, inputStyle, defaultValue, placeholder, autocomplete, pattern, required = true, updateFormValue, }) {
|
3474
|
+
const [inputValue, setInputValue] = useState(defaultValue ?? '');
|
3475
|
+
useEffect(() => {
|
3476
|
+
setInputValue(defaultValue ?? '');
|
3477
|
+
}, [defaultValue]);
|
3478
|
+
const handleChange = (e) => {
|
3479
|
+
const newValue = e.target.value;
|
3480
|
+
setInputValue(newValue);
|
3481
|
+
if (updateFormValue) {
|
3482
|
+
updateFormValue(newValue);
|
3483
|
+
}
|
3484
|
+
};
|
3485
|
+
return (jsxs("div", { className: `tw:form-control ${containerStyle ?? ''}`, children: [labelTitle ? (jsx("label", { className: 'tw:label', children: jsx("span", { className: `tw:label-text tw:text-base-content ${labelStyle ?? ''}`, children: labelTitle }) })) : null, jsx("input", { required: required, pattern: pattern, type: type ?? 'text', name: dataField, value: inputValue, placeholder: placeholder ?? '', autoComplete: autocomplete, onChange: handleChange, className: `tw:input tw:input-bordered tw:w-full ${inputStyle ?? ''}` })] }));
|
3486
|
+
}
|
3487
|
+
|
4022
3488
|
function Subtitle({ styleClass, children }) {
|
4023
3489
|
return jsx("div", { className: `tw:text-xl tw:font-semibold ${styleClass}`, children: children });
|
4024
3490
|
}
|
@@ -4053,6 +3519,77 @@ const SelectUser = () => {
|
|
4053
3519
|
}) }) }) }), jsx("div", { className: 'tw:w-full tw:grid tw:mt-4', children: jsx(Link, { className: 'tw:place-self-center ', to: '/attestation-form' + '?to=' + selectedUsers.map((u) => u, ','), children: jsx("button", { className: 'tw:btn tw:px-8', children: "Next" }) }) })] }));
|
4054
3520
|
};
|
4055
3521
|
|
3522
|
+
/**
|
3523
|
+
* @category Map
|
3524
|
+
*/
|
3525
|
+
const PopupTextAreaInput$1 = ({ dataField, placeholder, style, item, }) => {
|
3526
|
+
return (jsx(TextAreaInput, { defaultValue: item?.text ? item.text : '', dataField: dataField, placeholder: placeholder, inputStyle: style }));
|
3527
|
+
};
|
3528
|
+
|
3529
|
+
/**
|
3530
|
+
* @category Map
|
3531
|
+
*/
|
3532
|
+
const PopupStartEndInput$1 = ({ item, showLabels = true, updateStartValue, updateEndValue, }) => {
|
3533
|
+
return (jsxs("div", { className: 'tw:grid tw:grid-cols-2 tw:gap-2', children: [jsx(TextInput, { type: 'date', placeholder: 'start', dataField: 'start', inputStyle: 'tw:text-sm tw:px-2', labelTitle: showLabels ? 'start' : '', defaultValue: item && item.start ? item.start.substring(0, 10) : '', autocomplete: 'one-time-code', updateFormValue: updateStartValue }), jsx(TextInput, { type: 'date', placeholder: 'end', dataField: 'end', inputStyle: 'tw:text-sm tw:px-2', labelTitle: showLabels ? 'end' : '', defaultValue: item && item.end ? item.end.substring(0, 10) : '', autocomplete: 'one-time-code', updateFormValue: updateEndValue })] }));
|
3534
|
+
};
|
3535
|
+
|
3536
|
+
/**
|
3537
|
+
* @category Map
|
3538
|
+
*/
|
3539
|
+
const PopupTextInput$1 = ({ dataField, placeholder, style, item, }) => {
|
3540
|
+
return (jsx(TextInput, { defaultValue: item?.name ? item.name : '', dataField: dataField, placeholder: placeholder, inputStyle: style, type: 'text', containerStyle: 'tw:mt-4' }));
|
3541
|
+
};
|
3542
|
+
|
3543
|
+
/**
|
3544
|
+
* @category Map
|
3545
|
+
*/
|
3546
|
+
const PopupCheckboxInput$1 = ({ dataField, label, item, }) => {
|
3547
|
+
return (jsxs("label", { htmlFor: item?.id, className: 'tw:label tw:justify-normal tw:pt-1 tw:pb-1', children: [jsx("input", { id: item?.id, type: 'checkbox', name: dataField, className: 'tw:checkbox tw:checkbox-xs tw:checkbox-success', checked: item?.public_edit }), jsx("span", { className: 'tw:text-sm tw:label-text tw:mx-2 tw:cursor-pointer', children: label })] }));
|
3548
|
+
};
|
3549
|
+
|
3550
|
+
function CalendarDaysIcon({
|
3551
|
+
title,
|
3552
|
+
titleId,
|
3553
|
+
...props
|
3554
|
+
}, svgRef) {
|
3555
|
+
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3556
|
+
xmlns: "http://www.w3.org/2000/svg",
|
3557
|
+
viewBox: "0 0 24 24",
|
3558
|
+
fill: "currentColor",
|
3559
|
+
"aria-hidden": "true",
|
3560
|
+
"data-slot": "icon",
|
3561
|
+
ref: svgRef,
|
3562
|
+
"aria-labelledby": titleId
|
3563
|
+
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3564
|
+
id: titleId
|
3565
|
+
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3566
|
+
d: "M12.75 12.75a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM7.5 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM8.25 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM9.75 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM10.5 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM12 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM12.75 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM14.25 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM15 17.25a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM16.5 15.75a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5ZM15 12.75a.75.75 0 1 1-1.5 0 .75.75 0 0 1 1.5 0ZM16.5 13.5a.75.75 0 1 0 0-1.5.75.75 0 0 0 0 1.5Z"
|
3567
|
+
}), /*#__PURE__*/React.createElement("path", {
|
3568
|
+
fillRule: "evenodd",
|
3569
|
+
d: "M6.75 2.25A.75.75 0 0 1 7.5 3v1.5h9V3A.75.75 0 0 1 18 3v1.5h.75a3 3 0 0 1 3 3v11.25a3 3 0 0 1-3 3H5.25a3 3 0 0 1-3-3V7.5a3 3 0 0 1 3-3H6V3a.75.75 0 0 1 .75-.75Zm13.5 9a1.5 1.5 0 0 0-1.5-1.5H5.25a1.5 1.5 0 0 0-1.5 1.5v7.5a1.5 1.5 0 0 0 1.5 1.5h13.5a1.5 1.5 0 0 0 1.5-1.5v-7.5Z",
|
3570
|
+
clipRule: "evenodd"
|
3571
|
+
}));
|
3572
|
+
}
|
3573
|
+
const ForwardRef$8 = /*#__PURE__*/ React.forwardRef(CalendarDaysIcon);
|
3574
|
+
|
3575
|
+
/**
|
3576
|
+
* @category Map
|
3577
|
+
*/
|
3578
|
+
const StartEndView$1 = ({ item }) => {
|
3579
|
+
return (jsxs("div", { className: 'tw:flex tw:flex-row tw:mb-4 tw:mt-1', children: [jsxs("div", { className: 'tw:basis-2/5 tw:flex tw:flex-row', children: [jsx(ForwardRef$8, { className: 'tw:h-4 tw:w-4 tw:mr-2' }), jsx("time", { className: 'tw:align-middle', dateTime: item && item.start ? item.start.substring(0, 10) : '', children: item && item.start ? new Date(item.start).toLocaleDateString() : '' })] }), jsx("div", { className: 'tw:basis-1/5 tw:place-content-center', children: jsx("span", { children: "-" }) }), jsxs("div", { className: 'tw:basis-2/5 tw:flex tw:flex-row', children: [jsx(ForwardRef$8, { className: 'tw:h-4 tw:w-4 tw:mr-2' }), jsx("time", { className: 'tw:align-middle', dateTime: item && item.end ? item.end.substring(0, 10) : '', children: item && item.end ? new Date(item.end).toLocaleDateString() : '' })] })] }));
|
3580
|
+
};
|
3581
|
+
|
3582
|
+
/**
|
3583
|
+
* @category Map
|
3584
|
+
*/
|
3585
|
+
const PopupButton$1 = ({ url, parameterField, text, item, }) => {
|
3586
|
+
const params = new URLSearchParams(window.location.search);
|
3587
|
+
const getItemTags = useGetItemTags();
|
3588
|
+
return (jsx(Link, { to: `${url}/${parameterField ? item?.id : ''}?${params}`, children: jsx("button", { style: {
|
3589
|
+
backgroundColor: `${item?.color ?? (item && (getItemTags(item) && getItemTags(item)[0] && getItemTags(item)[0].color ? getItemTags(item)[0].color : (item?.layer?.markerDefaultColor ?? '#000')))}`,
|
3590
|
+
}, className: 'tw:btn tw:text-white tw:btn-sm tw:float-right tw:mt-1', children: text }) }));
|
3591
|
+
};
|
3592
|
+
|
4056
3593
|
function PlusIcon({
|
4057
3594
|
title,
|
4058
3595
|
titleId,
|
@@ -4076,15 +3613,203 @@ function PlusIcon({
|
|
4076
3613
|
d: "M12 4.5v15m7.5-7.5h-15"
|
4077
3614
|
}));
|
4078
3615
|
}
|
4079
|
-
const ForwardRef$
|
3616
|
+
const ForwardRef$7 = /*#__PURE__*/ React.forwardRef(PlusIcon);
|
4080
3617
|
|
4081
3618
|
function PlusButton({ layer, triggerAction, color, collection = 'items', }) {
|
4082
3619
|
const hasUserPermission = useHasUserPermission();
|
4083
|
-
return (jsx(Fragment, { children: hasUserPermission(collection, 'create', undefined, layer) && (jsx("div", { className: 'tw:dropdown tw:dropdown-top tw:dropdown-end tw:dropdown-hover tw:z-3000 tw:absolute tw:right-4 tw:bottom-4', children: jsx("button", { tabIndex: 0, className: 'tw:z-500 tw:btn tw:btn-circle tw:shadow', onClick: () => {
|
4084
|
-
triggerAction();
|
4085
|
-
}, style: { backgroundColor: color, color: '#fff' }, children: jsx(ForwardRef$
|
3620
|
+
return (jsx(Fragment, { children: hasUserPermission(collection, 'create', undefined, layer) && (jsx("div", { className: 'tw:dropdown tw:dropdown-top tw:dropdown-end tw:dropdown-hover tw:z-3000 tw:absolute tw:right-4 tw:bottom-4', children: jsx("button", { tabIndex: 0, className: 'tw:z-500 tw:btn tw:btn-circle tw:shadow', onClick: () => {
|
3621
|
+
triggerAction();
|
3622
|
+
}, style: { backgroundColor: color, color: '#fff' }, children: jsx(ForwardRef$7, { className: 'tw:w-5 tw:h-5 tw:stroke-[2.5]' }) }) })) }));
|
3623
|
+
}
|
3624
|
+
|
3625
|
+
const goldenRatioConjugate = 0.618033988749895;
|
3626
|
+
const randomColor = () => {
|
3627
|
+
return hsvToHex((Math.random() + goldenRatioConjugate) % 1, 0.8, 0.7);
|
3628
|
+
};
|
3629
|
+
function hsvToHex(h, s, v) {
|
3630
|
+
let r, g, b;
|
3631
|
+
const i = Math.floor(h * 6);
|
3632
|
+
const f = h * 6 - i;
|
3633
|
+
const p = v * (1 - s);
|
3634
|
+
const q = v * (1 - f * s);
|
3635
|
+
const t = v * (1 - (1 - f) * s);
|
3636
|
+
switch (i % 6) {
|
3637
|
+
case 0:
|
3638
|
+
r = v;
|
3639
|
+
g = t;
|
3640
|
+
b = p;
|
3641
|
+
break;
|
3642
|
+
case 1:
|
3643
|
+
r = q;
|
3644
|
+
g = v;
|
3645
|
+
b = p;
|
3646
|
+
break;
|
3647
|
+
case 2:
|
3648
|
+
r = p;
|
3649
|
+
g = v;
|
3650
|
+
b = t;
|
3651
|
+
break;
|
3652
|
+
case 3:
|
3653
|
+
r = p;
|
3654
|
+
g = q;
|
3655
|
+
b = v;
|
3656
|
+
break;
|
3657
|
+
case 4:
|
3658
|
+
r = t;
|
3659
|
+
g = p;
|
3660
|
+
b = v;
|
3661
|
+
break;
|
3662
|
+
case 5:
|
3663
|
+
r = v;
|
3664
|
+
g = p;
|
3665
|
+
b = q;
|
3666
|
+
break;
|
3667
|
+
}
|
3668
|
+
return rgbToHex(Math.round(r * 255), Math.round(g * 255), Math.round(b * 255));
|
3669
|
+
}
|
3670
|
+
const rgbToHex = (r, g, b) => '#' +
|
3671
|
+
[r, g, b]
|
3672
|
+
.map((x) => {
|
3673
|
+
const hex = x.toString(16);
|
3674
|
+
return hex.length === 1 ? `0${hex}` : hex;
|
3675
|
+
})
|
3676
|
+
.join('');
|
3677
|
+
|
3678
|
+
function PencilIcon({
|
3679
|
+
title,
|
3680
|
+
titleId,
|
3681
|
+
...props
|
3682
|
+
}, svgRef) {
|
3683
|
+
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3684
|
+
xmlns: "http://www.w3.org/2000/svg",
|
3685
|
+
viewBox: "0 0 24 24",
|
3686
|
+
fill: "currentColor",
|
3687
|
+
"aria-hidden": "true",
|
3688
|
+
"data-slot": "icon",
|
3689
|
+
ref: svgRef,
|
3690
|
+
"aria-labelledby": titleId
|
3691
|
+
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3692
|
+
id: titleId
|
3693
|
+
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3694
|
+
d: "M21.731 2.269a2.625 2.625 0 0 0-3.712 0l-1.157 1.157 3.712 3.712 1.157-1.157a2.625 2.625 0 0 0 0-3.712ZM19.513 8.199l-3.712-3.712-12.15 12.15a5.25 5.25 0 0 0-1.32 2.214l-.8 2.685a.75.75 0 0 0 .933.933l2.685-.8a5.25 5.25 0 0 0 2.214-1.32L19.513 8.2Z"
|
3695
|
+
}));
|
3696
|
+
}
|
3697
|
+
const ForwardRef$6 = /*#__PURE__*/ React.forwardRef(PencilIcon);
|
3698
|
+
|
3699
|
+
function TrashIcon({
|
3700
|
+
title,
|
3701
|
+
titleId,
|
3702
|
+
...props
|
3703
|
+
}, svgRef) {
|
3704
|
+
return /*#__PURE__*/React.createElement("svg", Object.assign({
|
3705
|
+
xmlns: "http://www.w3.org/2000/svg",
|
3706
|
+
viewBox: "0 0 24 24",
|
3707
|
+
fill: "currentColor",
|
3708
|
+
"aria-hidden": "true",
|
3709
|
+
"data-slot": "icon",
|
3710
|
+
ref: svgRef,
|
3711
|
+
"aria-labelledby": titleId
|
3712
|
+
}, props), title ? /*#__PURE__*/React.createElement("title", {
|
3713
|
+
id: titleId
|
3714
|
+
}, title) : null, /*#__PURE__*/React.createElement("path", {
|
3715
|
+
fillRule: "evenodd",
|
3716
|
+
d: "M16.5 4.478v.227a48.816 48.816 0 0 1 3.878.512.75.75 0 1 1-.256 1.478l-.209-.035-1.005 13.07a3 3 0 0 1-2.991 2.77H8.084a3 3 0 0 1-2.991-2.77L4.087 6.66l-.209.035a.75.75 0 0 1-.256-1.478A48.567 48.567 0 0 1 7.5 4.705v-.227c0-1.564 1.213-2.9 2.816-2.951a52.662 52.662 0 0 1 3.369 0c1.603.051 2.815 1.387 2.815 2.951Zm-6.136-1.452a51.196 51.196 0 0 1 3.273 0C14.39 3.05 15 3.684 15 4.478v.113a49.488 49.488 0 0 0-6 0v-.113c0-.794.609-1.428 1.364-1.452Zm-.355 5.945a.75.75 0 1 0-1.5.058l.347 9a.75.75 0 1 0 1.499-.058l-.346-9Zm5.48.058a.75.75 0 1 0-1.498-.058l-.347 9a.75.75 0 0 0 1.5.058l.345-9Z",
|
3717
|
+
clipRule: "evenodd"
|
3718
|
+
}));
|
3719
|
+
}
|
3720
|
+
const ForwardRef$5 = /*#__PURE__*/ React.forwardRef(TrashIcon);
|
3721
|
+
|
3722
|
+
var TargetDotSVG = '';
|
3723
|
+
|
3724
|
+
const isClickInsideRectangle = (e, element) => {
|
3725
|
+
const r = element.getBoundingClientRect();
|
3726
|
+
return e.clientX > r.left && e.clientX < r.right && e.clientY > r.top && e.clientY < r.bottom;
|
3727
|
+
};
|
3728
|
+
const DialogModal = ({ title, isOpened, onClose, children, showCloseButton = true, closeOnClickOutside = true, className, }) => {
|
3729
|
+
const ref = useRef(null);
|
3730
|
+
useEffect(() => {
|
3731
|
+
if (isOpened) {
|
3732
|
+
ref.current?.showModal();
|
3733
|
+
ref.current?.classList.remove('tw:hidden');
|
3734
|
+
document.body.classList.add('modal-open'); // prevent bg scroll
|
3735
|
+
}
|
3736
|
+
else {
|
3737
|
+
ref.current?.close();
|
3738
|
+
ref.current?.classList.add('tw:hidden');
|
3739
|
+
document.body.classList.remove('modal-open');
|
3740
|
+
}
|
3741
|
+
}, [isOpened]);
|
3742
|
+
if (isOpened) {
|
3743
|
+
return (jsx("dialog", { className: `${className ?? ''} card tw:shadow-xl tw:absolute tw:right-0 tw:top-0 tw:bottom-0 tw:left-0 tw:m-auto tw:transition-opacity tw:duration-300 tw:p-4 tw:max-w-xl tw:bg-base-100`, ref: ref, onCancel: onClose, onClick: (e) => ref.current && !isClickInsideRectangle(e, ref.current) && closeOnClickOutside && onClose(), children: jsxs("div", { className: 'card-body tw:p-2', children: [jsx("h2", { className: 'tw:text-2xl tw:font-semibold tw:mb-2 tw:text-center', children: title }), children, showCloseButton && (jsx("button", { className: 'btn btn-sm btn-circle btn-ghost tw:absolute tw:right-2 tw:top-2', onClick: onClose, children: "\u2715" }))] }) }));
|
3744
|
+
}
|
3745
|
+
else
|
3746
|
+
return jsx(Fragment, {});
|
3747
|
+
};
|
3748
|
+
|
3749
|
+
function HeaderView({ item, api, editCallback, deleteCallback, setPositionCallback, loading, hideMenu = false, big = false, truncateSubname = true, hideSubname = false, showAddress = false, }) {
|
3750
|
+
const [modalOpen, setModalOpen] = useState(false);
|
3751
|
+
const hasUserPermission = useHasUserPermission();
|
3752
|
+
const navigate = useNavigate();
|
3753
|
+
const appState = useAppState();
|
3754
|
+
const [imageLoaded, setImageLoaded] = useState(false);
|
3755
|
+
const avatar = item.image && appState.assetsApi.url + item.image + '?width=160&heigth=160';
|
3756
|
+
const title = item.name;
|
3757
|
+
const subtitle = item.subname;
|
3758
|
+
const [address] = useState('');
|
3759
|
+
const params = new URLSearchParams(window.location.search);
|
3760
|
+
const openDeleteModal = async (event) => {
|
3761
|
+
setModalOpen(true);
|
3762
|
+
event.stopPropagation();
|
3763
|
+
};
|
3764
|
+
return (jsxs(Fragment, { children: [jsxs("div", { className: 'tw:flex tw:flex-row', children: [jsx("div", { className: 'tw:grow tw:max-w-[calc(100%-60px)] }', children: jsxs("div", { className: 'flex items-center', children: [avatar && (jsx("div", { className: 'tw:avatar', children: jsxs("div", { className: `${big ? 'tw:w-20' : 'tw:w-10'} tw:inline tw:items-center tw:justify-center overflow-hidden`, children: [jsx("img", { className: 'tw:w-full tw:h-full tw:object-cover tw:rounded-full', src: avatar, alt: item.name + ' logo', onLoad: () => setImageLoaded(true), onError: () => setImageLoaded(false), style: { display: imageLoaded ? 'block' : 'none' } }), !imageLoaded && (jsx("div", { className: 'tw:w-full tw:h-full tw:bg-gray-200 tw:rounded-full' }))] }) })), jsxs("div", { className: `${avatar ? 'tw:ml-2' : ''} tw:overflow-hidden`, children: [jsx("div", { className: `${big ? 'tw:xl:text-3xl tw:text-2xl' : 'tw:text-xl'} tw:font-semibold tw:truncate`, title: title, children: title }), showAddress && address && !hideSubname && (jsx("div", { className: `tw:text-xs tw:text-gray-500 ${truncateSubname && 'tw:truncate'}`, children: address })), subtitle && !hideSubname && (jsx("div", { className: `tw:text-xs tw:opacity-50 ${truncateSubname && 'tw:truncate'}`, children: subtitle }))] })] }) }), jsx("div", { onClick: (e) => e.stopPropagation(), className: `${big ? 'tw:mt-5' : 'tw:mt-1'}`, children: (api?.deleteItem || item.layer?.api?.updateItem) &&
|
3765
|
+
(hasUserPermission(api?.collectionName, 'delete', item) ||
|
3766
|
+
hasUserPermission(api?.collectionName, 'update', item)) &&
|
3767
|
+
!hideMenu && (jsxs("div", { className: 'tw:dropdown tw:dropdown-bottom', children: [jsx("label", { tabIndex: 0, className: 'tw:bg-base-100 tw:btn tw:m-1 tw:leading-3 tw:border-none tw:min-h-0 tw:h-6', children: jsx(ForwardRef$9, { className: 'tw:h-5 tw:w-5' }) }), jsxs("ul", { tabIndex: 0, className: 'tw:dropdown-content tw:menu tw:p-2 tw:shadow tw:bg-base-100 tw:rounded-box tw:z-1000', children: [api?.updateItem &&
|
3768
|
+
hasUserPermission(api.collectionName, 'update', item) &&
|
3769
|
+
editCallback && (jsx("li", { children: jsx("a", { className: 'tw:text-base-content! tw:cursor-pointer', onClick: (e) => item.layer?.customEditLink
|
3770
|
+
? navigate(`${item.layer.customEditLink}${item.layer.customEditParameter ? `/${item.id}${params && '?' + params}` : ''} `)
|
3771
|
+
: editCallback(e), children: jsx(ForwardRef$6, { className: 'tw:h-5 tw:w-5' }) }) })), api?.updateItem &&
|
3772
|
+
hasUserPermission(api.collectionName, 'update', item) &&
|
3773
|
+
setPositionCallback && (jsx("li", { children: jsx("a", { className: 'tw:text-base-content! tw:cursor-pointer', onClick: setPositionCallback, children: jsx(SVG, { src: TargetDotSVG, className: 'tw:w-5 tw:h-5' }) }) })), api?.deleteItem &&
|
3774
|
+
hasUserPermission(api.collectionName, 'delete', item) &&
|
3775
|
+
deleteCallback && (jsx("li", { children: jsx("a", { className: 'tw:cursor-pointer tw:text-error!', onClick: openDeleteModal, children: loading ? (jsx("span", { className: 'tw:loading tw:loading-spinner tw:loading-sm' })) : (jsx(ForwardRef$5, { className: 'tw:h-5 tw:w-5' })) }) }))] })] })) })] }), jsx(DialogModal, { isOpened: modalOpen, title: 'Are you sure?', showCloseButton: false, onClose: () => setModalOpen(false), children: jsxs("div", { onClick: (e) => e.stopPropagation(), children: [jsxs("span", { children: ["Do you want to delete ", jsx("b", { children: item.name }), "?"] }), jsx("div", { className: 'tw:grid', children: jsxs("div", { className: 'tw:flex tw:justify-between', children: [jsx("label", { className: 'tw:btn tw:mt-4 tw:btn-error', onClick: (e) => {
|
3776
|
+
deleteCallback(e);
|
3777
|
+
setModalOpen(false);
|
3778
|
+
}, children: "Yes" }), jsx("label", { className: 'tw:btn tw:mt-4', onClick: () => setModalOpen(false), children: "No" })] }) })] }) })] }));
|
4086
3779
|
}
|
4087
3780
|
|
3781
|
+
// in miliseconds
|
3782
|
+
const units = [
|
3783
|
+
{ label: 'year', seconds: 31536000 },
|
3784
|
+
{ label: 'month', seconds: 2592000 },
|
3785
|
+
{ label: 'week', seconds: 604800 },
|
3786
|
+
{ label: 'day', seconds: 86400 },
|
3787
|
+
{ label: 'hour', seconds: 3600 },
|
3788
|
+
{ label: 'minute', seconds: 60 },
|
3789
|
+
{ label: 'second', seconds: 1 },
|
3790
|
+
];
|
3791
|
+
const timeAgo = (date) => {
|
3792
|
+
const time = Math.floor((new Date().valueOf() - new Date(date).valueOf()) / 1000);
|
3793
|
+
const { interval, unit } = calculateTimeDifference(time);
|
3794
|
+
const suffix = interval === 1 ? '' : 's';
|
3795
|
+
return `${interval} ${unit}${suffix} ago`;
|
3796
|
+
};
|
3797
|
+
const calculateTimeDifference = (time) => {
|
3798
|
+
for (const { label, seconds } of units) {
|
3799
|
+
const interval = Math.floor(time / seconds);
|
3800
|
+
if (interval >= 1) {
|
3801
|
+
return {
|
3802
|
+
interval,
|
3803
|
+
unit: label,
|
3804
|
+
};
|
3805
|
+
}
|
3806
|
+
}
|
3807
|
+
return {
|
3808
|
+
interval: 0,
|
3809
|
+
unit: '',
|
3810
|
+
};
|
3811
|
+
};
|
3812
|
+
|
4088
3813
|
const DateUserInfo = ({ item }) => {
|
4089
3814
|
const [infoExpanded, setInfoExpanded] = useState(false);
|
4090
3815
|
return (jsxs("div", { className: 'tw:flex tw:-mb-1 tw:flex-row tw:mr-2 tw:-mt-2', onClick: (e) => e.stopPropagation(), children: [infoExpanded ? (jsx("p", { className: 'tw:italic tw:min-h-[21px] tw:my-0! tw:text-gray-500', onClick: () => setInfoExpanded(false), children: `${item.date_updated && item.date_updated !== item.date_created ? 'updated' : 'posted'} ${item.user_created?.first_name ? `by ${item.user_created.first_name}` : ''} ${item.date_updated ? timeAgo(item.date_updated) : timeAgo(item.date_created)}` })) : (jsx("p", { className: 'tw:my-0! tw:min-h-[21px] tw:font-bold tw:cursor-pointer tw:text-gray-500', onClick: () => setInfoExpanded(true), children: "\u24D8" })), jsx("div", { className: 'tw:grow ' })] }));
|
@@ -4100,7 +3825,7 @@ const ItemCard = ({ i, loading, url, deleteCallback, }) => {
|
|
4100
3825
|
navigate('/' + i.id + `${params.size > 0 ? `?${params.toString()}` : ''}`);
|
4101
3826
|
else
|
4102
3827
|
navigate(url + i.id + `${params.size > 0 ? `?${params.toString()}` : ''}`);
|
4103
|
-
}, children: [jsx(HeaderView, { loading: loading, item: i, api: i.layer?.api, editCallback: () => navigate('/edit-item/' + i.id), deleteCallback: () => deleteCallback(i) }), jsxs("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: [i.layer?.itemType.show_start_end && jsx(StartEndView, { item: i }), i.layer?.itemType.show_text && jsx(TextView, { truncate: true, text: i.text, itemId: i.id })] }), jsx(DateUserInfo, { item: i })] }));
|
3828
|
+
}, children: [jsx(HeaderView, { loading: loading, item: i, api: i.layer?.api, editCallback: () => navigate('/edit-item/' + i.id), deleteCallback: () => deleteCallback(i) }), jsxs("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: [i.layer?.itemType.show_start_end && jsx(StartEndView$1, { item: i }), i.layer?.itemType.show_text && jsx(TextView$1, { truncate: true, text: i.text, itemId: i.id })] }), jsx(DateUserInfo, { item: i })] }));
|
4104
3829
|
};
|
4105
3830
|
|
4106
3831
|
/**
|
@@ -4202,7 +3927,7 @@ const OverlayItemsIndexPage = ({ url, layerName, parameterField, plusButton = tr
|
|
4202
3927
|
: 0;
|
4203
3928
|
return dateB - dateA; // Subtracts milliseconds which are numbers
|
4204
3929
|
})
|
4205
|
-
.map((i, k) => (jsx("div", { className: 'tw:break-inside-avoid tw:mb-6', children: jsx(ItemCard, { i: i, loading: loading, url: url, deleteCallback: () => deleteItem(i) }) }, k))), addItemPopupType === 'place' && (jsx("form", { ref: tabRef, autoComplete: 'off', onSubmit: (e) => submitNewItem(e), children: jsxs("div", { className: 'tw:cursor-pointer tw:break-inside-avoid card tw:border-[1px] tw:border-base-300 card-body tw:shadow-xl tw:bg-base-100 tw:text-base-content tw:p-6 tw:mb-10', children: [jsx("label", { className: 'btn btn-sm tw:rounded-2xl btn-circle btn-ghost tw:hover:bg-transparent tw:absolute tw:right-0 tw:top-0 tw:text-gray-600', onClick: () => setAddItemPopupType(''), children: jsx("p", { className: 'tw:text-center', children: "\u2715" }) }), jsx(TextInput, { type: 'text', placeholder: 'Name', dataField: 'name', defaultValue: '', inputStyle: '' }), layer?.itemType.show_start_end_input && jsx(PopupStartEndInput, {}), jsx(TextAreaInput, { placeholder: 'Text', dataField: 'text', defaultValue: '', inputStyle: 'tw:h-40 tw:mt-5' }), jsx("div", { className: 'tw:flex tw:justify-center', children: jsx("button", { className: loading
|
3930
|
+
.map((i, k) => (jsx("div", { className: 'tw:break-inside-avoid tw:mb-6', children: jsx(ItemCard, { i: i, loading: loading, url: url, deleteCallback: () => deleteItem(i) }) }, k))), addItemPopupType === 'place' && (jsx("form", { ref: tabRef, autoComplete: 'off', onSubmit: (e) => submitNewItem(e), children: jsxs("div", { className: 'tw:cursor-pointer tw:break-inside-avoid card tw:border-[1px] tw:border-base-300 card-body tw:shadow-xl tw:bg-base-100 tw:text-base-content tw:p-6 tw:mb-10', children: [jsx("label", { className: 'btn btn-sm tw:rounded-2xl btn-circle btn-ghost tw:hover:bg-transparent tw:absolute tw:right-0 tw:top-0 tw:text-gray-600', onClick: () => setAddItemPopupType(''), children: jsx("p", { className: 'tw:text-center', children: "\u2715" }) }), jsx(TextInput, { type: 'text', placeholder: 'Name', dataField: 'name', defaultValue: '', inputStyle: '' }), layer?.itemType.show_start_end_input && jsx(PopupStartEndInput$1, {}), jsx(TextAreaInput, { placeholder: 'Text', dataField: 'text', defaultValue: '', inputStyle: 'tw:h-40 tw:mt-5' }), jsx("div", { className: 'tw:flex tw:justify-center', children: jsx("button", { className: loading
|
4206
3931
|
? 'btn btn-disabled tw:mt-5 tw:place-self-center'
|
4207
3932
|
: 'btn tw:mt-5 tw:place-self-center', type: 'submit', children: loading ? jsx("span", { className: 'loading loading-spinner' }) : 'Save' }) })] }) }))] }) })] }) }), plusButton && (jsx(PlusButton, { layer: layer, triggerAction: () => {
|
4208
3933
|
setAddItemPopupType('place');
|
@@ -4891,7 +4616,7 @@ const GroupSubHeaderView = ({ item, shareBaseUrl, platforms, }) => (jsxs("div",
|
|
4891
4616
|
: window.location.protocol + '//' + window.location.host + '/item/' + item.id, title: item.name, platforms: platforms }) })] }));
|
4892
4617
|
|
4893
4618
|
const ProfileStartEndView = ({ item }) => {
|
4894
|
-
return (jsx("div", { className: 'tw:mt-2 tw:px-6 tw:max-w-xs', children: jsx(StartEndView, { item: item }) }));
|
4619
|
+
return (jsx("div", { className: 'tw:mt-2 tw:px-6 tw:max-w-xs', children: jsx(StartEndView$1, { item: item }) }));
|
4895
4620
|
};
|
4896
4621
|
|
4897
4622
|
const get = (value, path, defaultValue) => {
|
@@ -4915,7 +4640,7 @@ const get = (value, path, defaultValue) => {
|
|
4915
4640
|
const ProfileTextView = ({ item, dataField = 'text', heading, hideWhenEmpty, }) => {
|
4916
4641
|
const text = get(item, dataField);
|
4917
4642
|
const parsedText = typeof text !== 'string' ? '' : text;
|
4918
|
-
return (jsxs("div", { className: 'tw:my-10 tw:mt-2 tw:px-6', children: [!(text === '' && hideWhenEmpty) && (jsx("h2", { className: 'tw:text-lg tw:font-semibold', children: heading })), jsx("div", { className: 'tw:mt-2 tw:text-sm', children: jsx(TextView, { itemId: item.id, rawText: parsedText }) })] }));
|
4643
|
+
return (jsxs("div", { className: 'tw:my-10 tw:mt-2 tw:px-6', children: [!(text === '' && hideWhenEmpty) && (jsx("h2", { className: 'tw:text-lg tw:font-semibold', children: heading })), jsx("div", { className: 'tw:mt-2 tw:text-sm', children: jsx(TextView$1, { itemId: item.id, rawText: parsedText }) })] }));
|
4919
4644
|
};
|
4920
4645
|
|
4921
4646
|
const componentMap$1 = {
|
@@ -4937,11 +4662,11 @@ const FlexView = ({ item }) => {
|
|
4937
4662
|
};
|
4938
4663
|
|
4939
4664
|
const OnepagerView = ({ item }) => {
|
4940
|
-
return (jsxs("div", { className: 'tw:h-full tw:overflow-y-auto fade', children: [jsx(GroupSubHeaderView, { item: item, shareBaseUrl: `https://www.wuerdekompass.org/aktivitaeten/gruppensuche/#/gruppe/${item.slug}` }), item.user_created?.first_name && jsx(ContactInfoView, { heading: 'Du hast Fragen?', item: item }), jsx("div", { className: 'tw:my-10 tw:mt-2 tw:px-6 tw:text-sm ', children: jsx(TextView, { itemId: item.id, rawText: item.text ?? 'Keine Beschreibung vorhanden' }) }), item.next_appointment && (jsxs("div", { className: 'tw:my-10 tw:px-6', children: [jsx("h2", { className: 'tw:text-lg tw:font-semibold', children: "N\u00E4chste Termine" }), jsx("div", { className: 'tw:mt-2 tw:text-sm', children: jsx(TextView, { itemId: item.id, rawText: item.next_appointment }) })] })), ";"] }));
|
4665
|
+
return (jsxs("div", { className: 'tw:h-full tw:overflow-y-auto fade', children: [jsx(GroupSubHeaderView, { item: item, shareBaseUrl: `https://www.wuerdekompass.org/aktivitaeten/gruppensuche/#/gruppe/${item.slug}` }), item.user_created?.first_name && jsx(ContactInfoView, { heading: 'Du hast Fragen?', item: item }), jsx("div", { className: 'tw:my-10 tw:mt-2 tw:px-6 tw:text-sm ', children: jsx(TextView$1, { itemId: item.id, rawText: item.text ?? 'Keine Beschreibung vorhanden' }) }), item.next_appointment && (jsxs("div", { className: 'tw:my-10 tw:px-6', children: [jsx("h2", { className: 'tw:text-lg tw:font-semibold', children: "N\u00E4chste Termine" }), jsx("div", { className: 'tw:mt-2 tw:text-sm', children: jsx(TextView$1, { itemId: item.id, rawText: item.next_appointment }) })] })), ";"] }));
|
4941
4666
|
};
|
4942
4667
|
|
4943
4668
|
const SimpleView = ({ item }) => {
|
4944
|
-
return (jsx("div", { className: 'tw:mt-8 tw:h-full tw:overflow-y-auto fade tw:px-6', children: jsx(TextView, { text: item.text, itemId: item.id }) }));
|
4669
|
+
return (jsx("div", { className: 'tw:mt-8 tw:h-full tw:overflow-y-auto fade tw:px-6', children: jsx(TextView$1, { text: item.text, itemId: item.id }) }));
|
4945
4670
|
};
|
4946
4671
|
|
4947
4672
|
function LinkIcon({
|
@@ -4993,7 +4718,7 @@ function ActionButton({ item, triggerAddButton, triggerItemSelected, existingRel
|
|
4993
4718
|
}, style: {
|
4994
4719
|
backgroundColor,
|
4995
4720
|
color: '#fff',
|
4996
|
-
}, children: jsx(ForwardRef$
|
4721
|
+
}, children: jsx(ForwardRef$7, { className: 'tw:w-5 tw:h-5 tw:stroke-[2.5]' }) }))] }), jsxs(DialogModal, { title: 'Select', isOpened: modalOpen, onClose: () => setModalOpen(false), className: 'tw:w-xl tw:sm:w-2xl tw:min-h-80 tw:bg-base-200', children: [jsx(TextInput, { defaultValue: '', placeholder: '\uD83D\uDD0D Search', containerStyle: 'lg:col-span-2 tw:m-4 ', updateFormValue: (val) => {
|
4997
4722
|
setSearch(val);
|
4998
4723
|
} }), jsx("div", { className: 'tw:grid tw:grid-cols-1 tw:sm:grid-cols-2', children: filterdItems
|
4999
4724
|
.filter((item) => {
|
@@ -5073,7 +4798,7 @@ const TabsView = ({ attestations, item, offers, needs, relations, updatePermissi
|
|
5073
4798
|
setActiveTab(urlTab ? Number(urlTab) : 1);
|
5074
4799
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
5075
4800
|
}, [location.search]);
|
5076
|
-
return (jsxs("div", { role: 'tablist', className: 'tw:tabs tw:tabs-lift tw:mt-2 tw:mb-2 tw:px-6', children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab tw:font-bold tw:ps-2! tw:pe-2! ', "aria-label": `${item.layer?.itemType.icon_as_labels && activeTab !== 1 ? '📝' : '📝\u00A0Info'}`, checked: activeTab === 1 && true, onChange: () => updateActiveTab(1) }), jsxs("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:rounded-box tw:!h-[calc(100dvh-280px)] tw:overflow-y-auto fade tw:pt-2 tw:pb-4 tw:mb-4 tw:overflow-x-hidden', children: [item.layer?.itemType.show_start_end && (jsx("div", { className: 'tw:max-w-xs', children: jsx(StartEndView, { item: item }) })), jsx(TextView, { text: item.text, itemId: item.id }), jsx("div", { className: 'tw:h-4' }), jsx(TextView, { text: item.contact, itemId: item.id })] }), item.layer?.itemType.questlog && (jsxs(Fragment, { children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab tw:font-bold tw:ps-2! tw:pe-2!', "aria-label": `${item.layer.itemType.icon_as_labels && activeTab !== 2 ? '❤️' : '❤️\u00A0Trust'}`, checked: activeTab === 2 && true, onChange: () => updateActiveTab(2) }), jsx("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:rounded-box tw:!h-[calc(100dvh-280px)] tw:overflow-y-auto fade tw:pt-2 tw:pb-4 tw:mb-4 tw:overflow-x-hidden', children: jsx("table", { className: 'sm:tw:table-sm md:tw:table-md tw:w-full', children: jsx("tbody", { children: attestations
|
4801
|
+
return (jsxs("div", { role: 'tablist', className: 'tw:tabs tw:tabs-lift tw:mt-2 tw:mb-2 tw:px-6', children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab tw:font-bold tw:ps-2! tw:pe-2! ', "aria-label": `${item.layer?.itemType.icon_as_labels && activeTab !== 1 ? '📝' : '📝\u00A0Info'}`, checked: activeTab === 1 && true, onChange: () => updateActiveTab(1) }), jsxs("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:rounded-box tw:!h-[calc(100dvh-280px)] tw:overflow-y-auto fade tw:pt-2 tw:pb-4 tw:mb-4 tw:overflow-x-hidden', children: [item.layer?.itemType.show_start_end && (jsx("div", { className: 'tw:max-w-xs', children: jsx(StartEndView$1, { item: item }) })), jsx(TextView$1, { text: item.text, itemId: item.id }), jsx("div", { className: 'tw:h-4' }), jsx(TextView$1, { text: item.contact, itemId: item.id })] }), item.layer?.itemType.questlog && (jsxs(Fragment, { children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab tw:font-bold tw:ps-2! tw:pe-2!', "aria-label": `${item.layer.itemType.icon_as_labels && activeTab !== 2 ? '❤️' : '❤️\u00A0Trust'}`, checked: activeTab === 2 && true, onChange: () => updateActiveTab(2) }), jsx("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:rounded-box tw:!h-[calc(100dvh-280px)] tw:overflow-y-auto fade tw:pt-2 tw:pb-4 tw:mb-4 tw:overflow-x-hidden', children: jsx("table", { className: 'sm:tw:table-sm md:tw:table-md tw:w-full', children: jsx("tbody", { children: attestations
|
5077
4802
|
.filter((a) => a.to.some((t) => t.directus_users_id === item.user_created?.id))
|
5078
4803
|
.sort((a, b) => new Date(b.date_created).getTime() - new Date(a.date_created).getTime())
|
5079
4804
|
.map((a, i) => (jsxs("tr", { children: [jsx("td", { children: jsx("div", { className: `tw:cursor-pointer tw:text-3xl tw:mask ${a.shape === 'squircle' ? 'tw:mask-squircle' : a.shape === 'circle' ? 'tw:mask-circle' : 'tw:mask-hexagon-2'} tw:p-2 tw:my-2 tw:mr-2 tw:shadow-xl`,
|
@@ -5083,7 +4808,7 @@ const TabsView = ({ attestations, item, offers, needs, relations, updatePermissi
|
|
5083
4808
|
a.user_created.first_name, ' '] }), jsx("div", { className: 'tw:text-xs opacity-50 tw:text-zinc-500', children: timeAgo(a.date_created) })] })] }) })) : (jsxs("div", { children: [jsxs("div", { className: 'font-bold', children: [a.user_created.first_name, " "] }), jsx("div", { className: 'tw:text-xs opacity-50 tw:text-zinc-500', children: timeAgo(a.date_created) })] })) })] }, i))) }) }) })] })), item.layer?.itemType.offers_and_needs && (jsxs(Fragment, { children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: `tw:tab tw:font-bold tw:ps-2! tw:pe-2! ${!(item.layer.itemType.icon_as_labels && activeTab !== 3) && 'tw:min-w-[10.4em]'} `, "aria-label": `${item.layer.itemType.icon_as_labels && activeTab !== 3 ? '♻️' : '♻️\u00A0Offers & Needs'}`, checked: activeTab === 3 && true, onChange: () => updateActiveTab(3) }), jsx("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:rounded-box tw:h-[calc(100dvh-268px)] tw:overflow-y-auto fade tw:pt-4 tw:pb-1', children: jsx("div", { className: 'tw:h-full', children: jsxs("div", { className: 'tw:grid tw:grid-cols-1', children: [offers.length > 0 ? (jsxs("div", { className: 'tw:col-span-1', children: [jsx("h3", { className: 'tw:-mb-2', children: "Offers" }), jsx("div", { className: 'tw:flex tw:flex-wrap tw:mb-4', children: offers.map((o) => (jsx(TagView, { tag: o, onClick: () => {
|
5084
4809
|
addFilterTag(o);
|
5085
4810
|
} }, o.id))) })] })) : (''), needs.length > 0 ? (jsxs("div", { className: 'tw:col-span-1', children: [jsx("h3", { className: 'tw:-mb-2 tw:col-span-1', children: "Needs" }), jsx("div", { className: 'tw:flex tw:flex-wrap tw:mb-4', children: needs.map((n) => (jsx(TagView, { tag: n, onClick: () => addFilterTag(n) }, n.id))) })] })) : ('')] }) }) })] })), item.layer?.itemType.relations && (jsxs(Fragment, { children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab tw:font-bold tw:ps-2! tw:pe-2! ', "aria-label": `${item.layer.itemType.icon_as_labels && activeTab !== 7 ? '🔗' : '🔗\u00A0Links'}`, checked: activeTab === 7 && true, onChange: () => updateActiveTab(7) }), jsx("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:rounded-box tw:!h-[calc(100dvh-280px)] tw:overflow-y-auto tw:pt-4 tw:pb-1 tw:-mr-4 tw:-mb-4 tw:overflow-x-hidden', children: jsx("div", { className: 'tw:h-full', children: jsxs("div", { className: 'tw:grid tw:grid-cols-1 tw:sm:grid-cols-2 tw:md:grid-cols-1 tw:lg:grid-cols-1 tw:xl:grid-cols-1 tw:2xl:grid-cols-2 tw:pb-4', children: [relations &&
|
5086
|
-
relations.map((i) => (jsxs("div", { className: 'tw:cursor-pointer tw:card tw:bg-base-200 tw:border-[1px] tw:border-base-300 tw:card-body tw:shadow-xl tw:text-base-content tw:p-6 tw:mr-4 tw:mb-4', onClick: () => navigate('/item/' + i.id), children: [jsx(LinkedItemsHeaderView, { unlinkPermission: updatePermission, item: i, unlinkCallback: unlinkItem, loading: loading }), jsx("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: jsx(TextView, { truncate: true, text: i.text, itemId: item.id }) })] }, i.id))), updatePermission && (jsx(ActionButton, { collection: 'items', item: item, existingRelations: relations, triggerItemSelected: linkItem }))] }) }) })] }))] }));
|
4811
|
+
relations.map((i) => (jsxs("div", { className: 'tw:cursor-pointer tw:card tw:bg-base-200 tw:border-[1px] tw:border-base-300 tw:card-body tw:shadow-xl tw:text-base-content tw:p-6 tw:mr-4 tw:mb-4', onClick: () => navigate('/item/' + i.id), children: [jsx(LinkedItemsHeaderView, { unlinkPermission: updatePermission, item: i, unlinkCallback: unlinkItem, loading: loading }), jsx("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: jsx(TextView$1, { truncate: true, text: i.text, itemId: item.id }) })] }, i.id))), updatePermission && (jsx(ActionButton, { collection: 'items', item: item, existingRelations: relations, triggerItemSelected: linkItem }))] }) }) })] }))] }));
|
5087
4812
|
};
|
5088
4813
|
|
5089
4814
|
/**
|
@@ -5471,7 +5196,7 @@ const GroupSubheaderForm = ({ state, setState, groupStates, groupTypes, }) => {
|
|
5471
5196
|
};
|
5472
5197
|
|
5473
5198
|
const ProfileStartEndForm = ({ item, setState, }) => {
|
5474
|
-
return (jsx(PopupStartEndInput, { item: item, showLabels: false, updateEndValue: (e) => setState((prevState) => ({
|
5199
|
+
return (jsx(PopupStartEndInput$1, { item: item, showLabels: false, updateEndValue: (e) => setState((prevState) => ({
|
5475
5200
|
...prevState,
|
5476
5201
|
end: e,
|
5477
5202
|
})), updateStartValue: (s) => setState((prevState) => ({
|
@@ -5689,7 +5414,7 @@ const TabsForm = ({ item, state, setState, updatePermission, linkItem, unlinkIte
|
|
5689
5414
|
setActiveTab(urlTab ? Number(urlTab) : 1);
|
5690
5415
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
5691
5416
|
}, [location.search]);
|
5692
|
-
return (jsx("div", { className: 'tw:grow', children: jsxs("div", { role: 'tablist', className: 'tw:tabs tw:h-full tw:tabs-lift tw:mt-3', children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab ', "aria-label": 'Info', checked: activeTab === 1 && true, onChange: () => updateActiveTab(1) }), jsx("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:border-(--fallback-bc,oklch(var(--bc)/0.2)) tw:rounded-box tw:!h-[calc(100%-48px)] tw:min-h-56 tw:border-none', children: jsxs("div", { className: `tw:flex tw:flex-col tw:h-full ${item.layer.itemType.show_start_end_input && 'tw:pt-4'}`, children: [item.layer.itemType.show_start_end_input && (jsx(PopupStartEndInput, { item: item, showLabels: false, updateEndValue: (e) => setState((prevState) => ({
|
5417
|
+
return (jsx("div", { className: 'tw:grow', children: jsxs("div", { role: 'tablist', className: 'tw:tabs tw:h-full tw:tabs-lift tw:mt-3', children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab ', "aria-label": 'Info', checked: activeTab === 1 && true, onChange: () => updateActiveTab(1) }), jsx("div", { role: 'tabpanel', className: 'tw:tab-content tw:bg-base-100 tw:border-(--fallback-bc,oklch(var(--bc)/0.2)) tw:rounded-box tw:!h-[calc(100%-48px)] tw:min-h-56 tw:border-none', children: jsxs("div", { className: `tw:flex tw:flex-col tw:h-full ${item.layer.itemType.show_start_end_input && 'tw:pt-4'}`, children: [item.layer.itemType.show_start_end_input && (jsx(PopupStartEndInput$1, { item: item, showLabels: false, updateEndValue: (e) => setState((prevState) => ({
|
5693
5418
|
...prevState,
|
5694
5419
|
end: e,
|
5695
5420
|
})), updateStartValue: (s) => setState((prevState) => ({
|
@@ -5708,7 +5433,7 @@ const TabsForm = ({ item, state, setState, updatePermission, linkItem, unlinkIte
|
|
5708
5433
|
...prevState,
|
5709
5434
|
needs: v,
|
5710
5435
|
})), placeholder: 'enter your needs', containerStyle: 'tw:bg-transparent tw:w-full tw:h-full tw:mt-3 tw:text-xs tw:h-[calc(100%-1rem)] tw:min-h-[5em] tw:pb-2 tw:overflow-auto' }) })] }) })] })), item.layer?.itemType.relations && (jsxs(Fragment, { children: [jsx("input", { type: 'radio', name: 'my_tabs_2', role: 'tab', className: 'tw:tab ', "aria-label": 'Links', checked: activeTab === 7 && true, onChange: () => updateActiveTab(7) }), jsx("div", { role: 'tabpanel', className: 'tw:tab-content tw:rounded-box tw:!h-[calc(100%-48px)] tw:overflow-y-auto tw:pt-4 tw:overflow-x-hidden fade', children: jsx("div", { className: 'tw:h-full', children: jsxs("div", { className: 'tw:grid tw:grid-cols-1 tw:sm:grid-cols-2 tw:md:grid-cols-1 tw:lg:grid-cols-1 tw:xl:grid-cols-1 tw:2xl:grid-cols-2 tw:mb-4', children: [state.relations &&
|
5711
|
-
state.relations.map((i) => (jsxs("div", { className: 'tw:cursor-pointer tw:card tw:bg-base-200 tw:border-[1px] tw:border-base-300 tw:card-body tw:shadow-xl tw:text-base-content tw:mx-4 tw:p-6 tw:mb-4', onClick: () => navigate('/item/' + i.id), children: [jsx(LinkedItemsHeaderView, { unlinkPermission: updatePermission, item: i, unlinkCallback: (id) => unlinkItem(id, item, updateItem), loading: loading }), jsx("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: jsx(TextView, { truncate: true, itemId: item.id }) })] }, i.id))), updatePermission && (jsx(ActionButton, { customStyle: 'tw:bottom-24!', collection: 'items', item: item, existingRelations: state.relations, triggerItemSelected: (id) => linkItem(id, item, updateItem) }))] }) }) })] }))] }) }));
|
5436
|
+
state.relations.map((i) => (jsxs("div", { className: 'tw:cursor-pointer tw:card tw:bg-base-200 tw:border-[1px] tw:border-base-300 tw:card-body tw:shadow-xl tw:text-base-content tw:mx-4 tw:p-6 tw:mb-4', onClick: () => navigate('/item/' + i.id), children: [jsx(LinkedItemsHeaderView, { unlinkPermission: updatePermission, item: i, unlinkCallback: (id) => unlinkItem(id, item, updateItem), loading: loading }), jsx("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: jsx(TextView$1, { truncate: true, itemId: item.id }) })] }, i.id))), updatePermission && (jsx(ActionButton, { customStyle: 'tw:bottom-24!', collection: 'items', item: item, existingRelations: state.relations, triggerItemSelected: (id) => linkItem(id, item, updateItem) }))] }) }) })] }))] }) }));
|
5712
5437
|
};
|
5713
5438
|
|
5714
5439
|
/**
|
@@ -5875,5 +5600,308 @@ function Quests() {
|
|
5875
5600
|
return (jsx(Fragment, { children: questsOpen ? (jsx("div", { className: 'tw:card tw:w-48 tw:bg-base-100 tw:shadow-xl tw:absolute tw:bottom-4 tw:left-4 tw:z-2000', children: jsxs("div", { className: 'tw:card-body tw:p-4 tw:pt-0', children: [jsx("div", { className: 'tw:card-actions tw:justify-end', children: jsx("label", { className: 'tw:btn tw:btn-sm tw:btn-circle tw:btn-ghost tw:absolute tw:right-1 tw:top-1', onClick: () => setQuestsOpen(false), children: "\u2715" }) }), jsxs("h2", { className: 'tw:card-title tw:m-auto ', children: ["Level 1", jsx(QuestionMarkIcon, {})] }), jsxs("ul", { className: 'tw:flex-row', children: [jsx("li", { children: jsxs("label", { className: 'tw:label tw:justify-normal tw:pt-1 tw:pb-0', children: [jsx("input", { type: 'checkbox', readOnly: true, className: 'tw:checkbox tw:checkbox-xs tw:checkbox-success', checked: isAuthenticated || false }), jsx("span", { className: 'tw:text-sm tw:label-text tw:mx-2', children: "Sign Up" })] }) }), jsx("li", { children: jsxs("label", { className: 'tw:label tw:justify-normal tw:pt-1 tw:pb-0', children: [jsx("input", { type: 'checkbox', readOnly: true, className: 'tw:checkbox tw:checkbox-xs tw:checkbox-success', checked: !!profile?.text }), jsx("span", { className: 'tw:text-sm tw:label-text tw:mx-2', children: "Fill Profile" })] }) }), jsx("li", { children: jsxs("label", { className: 'tw:label tw:justify-normal tw:pt-1 tw:pb-0', children: [jsx("input", { type: 'checkbox', readOnly: true, className: 'tw:checkbox tw:checkbox-xs tw:checkbox-success', checked: !!profile?.image }), jsx("span", { className: 'tw:text-sm tw:label-text tw:mx-2', children: "Upload Avatar" })] }) })] }), ' '] }) })) : ('') }));
|
5876
5601
|
}
|
5877
5602
|
|
5878
|
-
|
5603
|
+
const ItemContext = createContext(undefined);
|
5604
|
+
|
5605
|
+
function templateify(Component) {
|
5606
|
+
const TemplateComponent = (props) => {
|
5607
|
+
const item = useContext(ItemContext);
|
5608
|
+
return jsx(Component, { ...props, item: item });
|
5609
|
+
};
|
5610
|
+
return TemplateComponent;
|
5611
|
+
}
|
5612
|
+
|
5613
|
+
function ItemFormPopup(props) {
|
5614
|
+
const layerContext = useContext(LayerContext);
|
5615
|
+
const { menuText, name: activeLayerName } = layerContext;
|
5616
|
+
const { popupForm, setPopupForm } = usePopupForm();
|
5617
|
+
const [spinner, setSpinner] = useState(false);
|
5618
|
+
const formRef = useRef(null);
|
5619
|
+
const map = useMap();
|
5620
|
+
const addItem = useAddItem();
|
5621
|
+
const updateItem = useUpdateItem();
|
5622
|
+
const items = useItems();
|
5623
|
+
const tags = useTags();
|
5624
|
+
const addTag = useAddTag();
|
5625
|
+
const resetFilterTags = useResetFilterTags();
|
5626
|
+
const { user } = useAuth();
|
5627
|
+
const handleSubmit = async (evt) => {
|
5628
|
+
if (!popupForm) {
|
5629
|
+
throw new Error('Popup form is not defined');
|
5630
|
+
}
|
5631
|
+
const formItem = {};
|
5632
|
+
Array.from(evt.target).forEach((input) => {
|
5633
|
+
if (input.name) {
|
5634
|
+
formItem[input.name] = input.value;
|
5635
|
+
}
|
5636
|
+
});
|
5637
|
+
formItem.position = {
|
5638
|
+
type: 'Point',
|
5639
|
+
coordinates: [popupForm.position.lng, popupForm.position.lat],
|
5640
|
+
};
|
5641
|
+
evt.preventDefault();
|
5642
|
+
const name = formItem.name ? formItem.name : user?.first_name;
|
5643
|
+
if (!name) {
|
5644
|
+
toast.error('Name is must be defined');
|
5645
|
+
return;
|
5646
|
+
}
|
5647
|
+
setSpinner(true);
|
5648
|
+
formItem.text &&
|
5649
|
+
formItem.text
|
5650
|
+
.toLocaleLowerCase()
|
5651
|
+
.match(hashTagRegex)
|
5652
|
+
?.map((tag) => {
|
5653
|
+
if (!tags.find((t) => t.name.toLocaleLowerCase() === tag.slice(1).toLocaleLowerCase())) {
|
5654
|
+
addTag({ id: crypto.randomUUID(), name: tag.slice(1), color: randomColor() });
|
5655
|
+
}
|
5656
|
+
return null;
|
5657
|
+
});
|
5658
|
+
if (popupForm.item) {
|
5659
|
+
let success = false;
|
5660
|
+
try {
|
5661
|
+
await popupForm.layer.api?.updateItem({ ...formItem, id: popupForm.item.id });
|
5662
|
+
success = true;
|
5663
|
+
// eslint-disable-next-line no-catch-all/no-catch-all
|
5664
|
+
}
|
5665
|
+
catch (error) {
|
5666
|
+
toast.error(error.toString());
|
5667
|
+
}
|
5668
|
+
if (success) {
|
5669
|
+
updateItem({ ...popupForm.item, ...formItem });
|
5670
|
+
toast.success('Item updated');
|
5671
|
+
}
|
5672
|
+
setSpinner(false);
|
5673
|
+
map.closePopup();
|
5674
|
+
}
|
5675
|
+
else {
|
5676
|
+
const item = items.find((i) => i.user_created?.id === user?.id && i.layer?.id === popupForm.layer.id);
|
5677
|
+
const uuid = crypto.randomUUID();
|
5678
|
+
let success = false;
|
5679
|
+
try {
|
5680
|
+
popupForm.layer.userProfileLayer &&
|
5681
|
+
item &&
|
5682
|
+
(await popupForm.layer.api?.updateItem({ ...formItem, id: item.id }));
|
5683
|
+
(!popupForm.layer.userProfileLayer || !item) &&
|
5684
|
+
(await popupForm.layer.api?.createItem({
|
5685
|
+
...formItem,
|
5686
|
+
name,
|
5687
|
+
id: uuid,
|
5688
|
+
}));
|
5689
|
+
success = true;
|
5690
|
+
// eslint-disable-next-line no-catch-all/no-catch-all
|
5691
|
+
}
|
5692
|
+
catch (error) {
|
5693
|
+
toast.error(error.toString());
|
5694
|
+
}
|
5695
|
+
if (success) {
|
5696
|
+
if (popupForm.layer.userProfileLayer && item)
|
5697
|
+
updateItem({ ...item, ...formItem });
|
5698
|
+
if (!popupForm.layer.userProfileLayer || !item) {
|
5699
|
+
addItem({
|
5700
|
+
...formItem,
|
5701
|
+
name: (formItem.name ? formItem.name : user?.first_name) ?? '',
|
5702
|
+
user_created: user ?? undefined,
|
5703
|
+
id: uuid,
|
5704
|
+
layer: popupForm.layer,
|
5705
|
+
public_edit: !user,
|
5706
|
+
});
|
5707
|
+
}
|
5708
|
+
toast.success('New item created');
|
5709
|
+
resetFilterTags();
|
5710
|
+
}
|
5711
|
+
setSpinner(false);
|
5712
|
+
map.closePopup();
|
5713
|
+
}
|
5714
|
+
setPopupForm(null);
|
5715
|
+
};
|
5716
|
+
const resetPopup = () => {
|
5717
|
+
if (formRef.current) {
|
5718
|
+
formRef.current.reset();
|
5719
|
+
}
|
5720
|
+
};
|
5721
|
+
useEffect(() => {
|
5722
|
+
resetPopup();
|
5723
|
+
}, [popupForm?.position]);
|
5724
|
+
return (popupForm &&
|
5725
|
+
popupForm.layer.name === activeLayerName && (jsx(Popup, { minWidth: 275, maxWidth: 275, autoPanPadding: [20, 80], eventHandlers: {
|
5726
|
+
remove: () => {
|
5727
|
+
setTimeout(function () {
|
5728
|
+
resetPopup();
|
5729
|
+
}, 100);
|
5730
|
+
},
|
5731
|
+
}, position: popupForm.position, children: jsxs("form", { ref: formRef, onReset: resetPopup, autoComplete: 'off', onSubmit: (e) => handleSubmit(e), children: [popupForm.item ? (jsx("div", { className: 'tw:h-3' })) : (jsx("div", { className: 'tw:flex tw:justify-center', children: jsx("b", { className: 'tw:text-xl tw:text-center tw:font-bold', children: menuText }) })), props.children ? (jsx(ItemContext.Provider, { value: popupForm.item, children: props.children })) : (jsxs(Fragment, { children: [jsx(TextInput, { type: 'text', placeholder: 'Name', dataField: 'name', defaultValue: popupForm.item ? popupForm.item.name : '', inputStyle: '' }), jsx(TextAreaInput, { placeholder: 'Text', dataField: 'text', defaultValue: popupForm.item?.text ?? '', inputStyle: 'tw:h-40 tw:mt-5' }, popupForm.position.toString())] })), jsx("div", { className: 'tw:flex tw:justify-center', children: jsx("button", { className: spinner
|
5732
|
+
? 'tw:btn tw:btn-disabled tw:mt-5 tw:place-self-center'
|
5733
|
+
: 'tw:btn tw:mt-5 tw:place-self-center', type: 'submit', children: spinner ? jsx("span", { className: 'tw:loading tw:loading-spinner' }) : 'Save' }) })] }) })));
|
5734
|
+
}
|
5735
|
+
|
5736
|
+
/**
|
5737
|
+
* @category Item
|
5738
|
+
*/
|
5739
|
+
const PopupForm = ({ children }) => {
|
5740
|
+
return jsx(ItemFormPopup, { children: children });
|
5741
|
+
};
|
5742
|
+
|
5743
|
+
// eslint-disable-next-line react/display-name
|
5744
|
+
const ItemViewPopup = forwardRef((props, ref) => {
|
5745
|
+
const map = useMap();
|
5746
|
+
const [loading, setLoading] = useState(false);
|
5747
|
+
const removeItem = useRemoveItem();
|
5748
|
+
const updadateItem = useUpdateItem();
|
5749
|
+
const navigate = useNavigate();
|
5750
|
+
const setSelectPosition = useSetSelectPosition();
|
5751
|
+
const { setPopupForm } = usePopupForm();
|
5752
|
+
const [infoExpanded, setInfoExpanded] = useState(false);
|
5753
|
+
const handleEdit = (event) => {
|
5754
|
+
event.stopPropagation();
|
5755
|
+
map.closePopup();
|
5756
|
+
if (!props.item.layer) {
|
5757
|
+
throw new Error('Layer is not defined');
|
5758
|
+
}
|
5759
|
+
setPopupForm({
|
5760
|
+
position: new LatLng(props.item.position?.coordinates[1], props.item.position?.coordinates[0]),
|
5761
|
+
layer: props.item.layer,
|
5762
|
+
item: props.item,
|
5763
|
+
});
|
5764
|
+
};
|
5765
|
+
const handleDelete = async (event) => {
|
5766
|
+
event.stopPropagation();
|
5767
|
+
setLoading(true);
|
5768
|
+
let success = false;
|
5769
|
+
try {
|
5770
|
+
!props.item.layer?.userProfileLayer &&
|
5771
|
+
(await props.item.layer?.api?.deleteItem(props.item.id));
|
5772
|
+
props.item.layer?.userProfileLayer &&
|
5773
|
+
(await props.item.layer.api?.updateItem({ id: props.item.id, position: null }));
|
5774
|
+
success = true;
|
5775
|
+
// eslint-disable-next-line no-catch-all/no-catch-all
|
5776
|
+
}
|
5777
|
+
catch (error) {
|
5778
|
+
// eslint-disable-next-line @typescript-eslint/no-unsafe-argument
|
5779
|
+
toast.error(error.toString());
|
5780
|
+
}
|
5781
|
+
if (success) {
|
5782
|
+
!props.item.layer?.userProfileLayer && removeItem(props.item);
|
5783
|
+
props.item.layer?.userProfileLayer && updadateItem({ ...props.item, position: undefined });
|
5784
|
+
toast.success('Item deleted');
|
5785
|
+
}
|
5786
|
+
setLoading(false);
|
5787
|
+
map.closePopup();
|
5788
|
+
const params = new URLSearchParams(window.location.search);
|
5789
|
+
window.history.pushState({}, '', '/' + `${params ? `?${params}` : ''}`);
|
5790
|
+
navigate('/');
|
5791
|
+
};
|
5792
|
+
return (jsx(Popup, { ref: ref, maxHeight: 377, minWidth: 275, maxWidth: 275, autoPanPadding: [20, 80], children: jsxs("div", { className: 'tw:bg-base-100 tw:text-base-content', children: [jsx(HeaderView, { api: props.item.layer?.api, item: props.item, editCallback: handleEdit, deleteCallback: handleDelete, setPositionCallback: () => {
|
5793
|
+
map.closePopup();
|
5794
|
+
setSelectPosition(props.item);
|
5795
|
+
navigate('/');
|
5796
|
+
}, loading: loading }), jsx("div", { className: 'tw:overflow-y-auto tw:overflow-x-hidden tw:max-h-64 fade', children: props.children ?? jsx(TextView$1, { text: props.item.text, itemId: props.item.id }) }), jsxs("div", { className: 'tw:flex tw:-mb-1 tw:flex-row tw:mr-2 tw:mt-1', children: [infoExpanded ? (jsx("p", { className: 'tw:italic tw:min-h-[21px] tw:my-0! tw:opacity-50', children: `${props.item.date_updated && props.item.date_updated !== props.item.date_created ? 'updated' : 'posted'} ${props.item && props.item.user_created && props.item.user_created.first_name ? `by ${props.item.user_created.first_name}` : ''} ${props.item.date_updated ? timeAgo(props.item.date_updated) : timeAgo(props.item.date_created)}` })) : (jsx("p", { className: 'tw:my-0! tw:min-h-[21px] tw:font-bold tw:cursor-pointer tw:text-gray-500', onClick: () => setInfoExpanded(true), children: "\u24D8" })), jsx("div", { className: 'tw:grow' })] })] }) }));
|
5797
|
+
});
|
5798
|
+
|
5799
|
+
/**
|
5800
|
+
* @category Item
|
5801
|
+
*/
|
5802
|
+
const PopupView = ({ children }) => {
|
5803
|
+
const layerContext = useContext(LayerContext);
|
5804
|
+
const { name, markerDefaultColor, markerDefaultColor2, markerShape, markerIcon } = layerContext;
|
5805
|
+
const filterTags = useFilterTags();
|
5806
|
+
const appState = useAppState();
|
5807
|
+
const items = useItems();
|
5808
|
+
const getItemTags = useGetItemTags();
|
5809
|
+
const addMarker = useAddMarker();
|
5810
|
+
const addPopup = useAddPopup();
|
5811
|
+
const leafletRefs = useLeafletRefs();
|
5812
|
+
const allTagsLoaded = useAllTagsLoaded();
|
5813
|
+
const allItemsLoaded = useAllItemsLoaded();
|
5814
|
+
const setMarkerClicked = useSetMarkerClicked();
|
5815
|
+
const selectPosition = useSelectPosition();
|
5816
|
+
const tags = useTags();
|
5817
|
+
const [newTagsToAdd, setNewTagsToAdd] = useState([]);
|
5818
|
+
const [tagsReady, setTagsReady] = useState(false);
|
5819
|
+
const isLayerVisible = useIsLayerVisible();
|
5820
|
+
const isGroupTypeVisible = useIsGroupTypeVisible();
|
5821
|
+
const visibleGroupTypes = useVisibleGroupType();
|
5822
|
+
const visibleItems = useMemo(() => items
|
5823
|
+
.filter((item) => item.layer?.name === name)
|
5824
|
+
.filter((item) => filterTags.length === 0
|
5825
|
+
? item
|
5826
|
+
: filterTags.some((tag) => getItemTags(item).some((filterTag) => filterTag.name.toLocaleLowerCase() === tag.name.toLocaleLowerCase())))
|
5827
|
+
.filter((item) => item.layer && isLayerVisible(item.layer))
|
5828
|
+
.filter((item) => (item.group_type && isGroupTypeVisible(item.group_type)) ||
|
5829
|
+
visibleGroupTypes.length === 0), [
|
5830
|
+
filterTags,
|
5831
|
+
getItemTags,
|
5832
|
+
isGroupTypeVisible,
|
5833
|
+
isLayerVisible,
|
5834
|
+
items,
|
5835
|
+
name,
|
5836
|
+
visibleGroupTypes.length,
|
5837
|
+
]);
|
5838
|
+
return visibleItems.map((item) => {
|
5839
|
+
if (!(item.position?.coordinates[0] && item.position.coordinates[1]))
|
5840
|
+
return null;
|
5841
|
+
if (item.tags) {
|
5842
|
+
item.text += '\n\n';
|
5843
|
+
item.tags.map((tag) => {
|
5844
|
+
if (!item.text?.includes(`#${encodeTag(tag)}`)) {
|
5845
|
+
item.text += `#${encodeTag(tag)}`;
|
5846
|
+
}
|
5847
|
+
return item.text;
|
5848
|
+
});
|
5849
|
+
}
|
5850
|
+
if (allTagsLoaded && allItemsLoaded) {
|
5851
|
+
item.text?.match(hashTagRegex)?.map((tag) => {
|
5852
|
+
if (!tags.find((t) => t.name.toLocaleLowerCase() === tag.slice(1).toLocaleLowerCase()) &&
|
5853
|
+
!newTagsToAdd.find((t) => t.name.toLocaleLowerCase() === tag.slice(1).toLocaleLowerCase())) {
|
5854
|
+
const newTag = {
|
5855
|
+
id: crypto.randomUUID(),
|
5856
|
+
name: tag.slice(1),
|
5857
|
+
color: randomColor(),
|
5858
|
+
};
|
5859
|
+
setNewTagsToAdd((current) => [...current, newTag]);
|
5860
|
+
}
|
5861
|
+
return null;
|
5862
|
+
});
|
5863
|
+
!tagsReady && setTagsReady(true);
|
5864
|
+
}
|
5865
|
+
const itemTags = getItemTags(item);
|
5866
|
+
const latitude = item.position.coordinates[1];
|
5867
|
+
const longitude = item.position.coordinates[0];
|
5868
|
+
let color1 = markerDefaultColor;
|
5869
|
+
let color2 = markerDefaultColor2;
|
5870
|
+
if (item.color) {
|
5871
|
+
color1 = item.color;
|
5872
|
+
}
|
5873
|
+
else if (itemTags[0]) {
|
5874
|
+
color1 = itemTags[0].color;
|
5875
|
+
}
|
5876
|
+
if (itemTags[0] && item.color) {
|
5877
|
+
color2 = itemTags[0].color;
|
5878
|
+
}
|
5879
|
+
else if (itemTags[1]) {
|
5880
|
+
color2 = itemTags[1].color;
|
5881
|
+
}
|
5882
|
+
return (jsx(ItemContext.Provider, { value: item, children: jsxs(Marker, { ref: (r) => {
|
5883
|
+
if (!(item.id in leafletRefs && leafletRefs[item.id].marker === r)) {
|
5884
|
+
r && addMarker(item, r);
|
5885
|
+
}
|
5886
|
+
}, eventHandlers: {
|
5887
|
+
click: () => {
|
5888
|
+
selectPosition && setMarkerClicked(item);
|
5889
|
+
},
|
5890
|
+
}, icon: MarkerIconFactory(markerShape, color1, color2, item.markerIcon ?? markerIcon, appState.assetsApi.url), position: [latitude, longitude], children: [jsx(ItemViewPopup, { ref: (r) => {
|
5891
|
+
if (!(item.id in leafletRefs && leafletRefs[item.id].popup === r)) {
|
5892
|
+
r && addPopup(item, r);
|
5893
|
+
}
|
5894
|
+
}, item: item, children: children }), jsx(Tooltip, { offset: [0, -38], direction: 'top', children: item.name })] }) }, item.id));
|
5895
|
+
});
|
5896
|
+
};
|
5897
|
+
|
5898
|
+
const TextView = templateify(TextView$1);
|
5899
|
+
const StartEndView = templateify(StartEndView$1);
|
5900
|
+
const PopupTextInput = templateify(PopupTextInput$1);
|
5901
|
+
const PopupButton = templateify(PopupButton$1);
|
5902
|
+
const PopupCheckboxInput = templateify(PopupCheckboxInput$1);
|
5903
|
+
const PopupTextAreaInput = templateify(PopupTextAreaInput$1);
|
5904
|
+
const PopupStartEndInput = templateify(PopupStartEndInput$1);
|
5905
|
+
|
5906
|
+
export { AppShell, AttestationForm, AuthProvider, CardPage, Content, Layer, LoginPage, MapOverlayPage, MarketView, Modal, OverlayItemsIndexPage, Permissions, PopupButton, PopupCheckboxInput, PopupForm, PopupStartEndInput, PopupTextAreaInput, PopupTextInput, PopupView, ProfileForm, ProfileView, Quests, RequestPasswordPage, SelectUser, SetNewPasswordPage, SideBar, SignupPage, StartEndView, Tags, TextAreaInput, TextInput, TextView, TitleCard, UserSettings, UtopiaMap };
|
5879
5907
|
//# sourceMappingURL=index.esm.js.map
|