| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  | import Combine from "../Base/Combine" | 
					
						
							|  |  |  | import { Store, Stores, UIEventSource } from "../../Logic/UIEventSource" | 
					
						
							|  |  |  | import { SlideShow } from "../Image/SlideShow" | 
					
						
							|  |  |  | import { ClickableToggle } from "../Input/Toggle" | 
					
						
							|  |  |  | import Loading from "../Base/Loading" | 
					
						
							|  |  |  | import { AttributedImage } from "../Image/AttributedImage" | 
					
						
							|  |  |  | import AllImageProviders from "../../Logic/ImageProviders/AllImageProviders" | 
					
						
							|  |  |  | import Svg from "../../Svg" | 
					
						
							|  |  |  | import BaseUIElement from "../BaseUIElement" | 
					
						
							|  |  |  | import { InputElement } from "../Input/InputElement" | 
					
						
							|  |  |  | import { VariableUiElement } from "../Base/VariableUIElement" | 
					
						
							|  |  |  | import Translations from "../i18n/Translations" | 
					
						
							|  |  |  | import { Mapillary } from "../../Logic/ImageProviders/Mapillary" | 
					
						
							|  |  |  | import { SubtleButton } from "../Base/SubtleButton" | 
					
						
							|  |  |  | import { GeoOperations } from "../../Logic/GeoOperations" | 
					
						
							|  |  |  | import { ElementStorage } from "../../Logic/ElementStorage" | 
					
						
							|  |  |  | import Lazy from "../Base/Lazy" | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | export interface P4CPicture { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     pictureUrl: string | 
					
						
							|  |  |  |     date?: number | 
					
						
							|  |  |  |     coordinates: { lat: number; lng: number } | 
					
						
							|  |  |  |     provider: "Mapillary" | string | 
					
						
							|  |  |  |     author? | 
					
						
							|  |  |  |     license? | 
					
						
							|  |  |  |     detailsUrl?: string | 
					
						
							|  |  |  |     direction? | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |     osmTags?: object /*To copy straight into OSM!*/ | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     thumbUrl: string | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     details: { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         isSpherical: boolean | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-13 01:19:53 +02:00
										 |  |  | export interface NearbyImageOptions { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     lon: number | 
					
						
							|  |  |  |     lat: number | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |     // Radius of the upstream search
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     searchRadius?: 500 | number | 
					
						
							|  |  |  |     maxDaysOld?: 1095 | number | 
					
						
							|  |  |  |     blacklist: Store<{ url: string }[]> | 
					
						
							|  |  |  |     shownImagesCount?: UIEventSource<number> | 
					
						
							|  |  |  |     towardscenter?: UIEventSource<boolean> | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |     allowSpherical?: UIEventSource<boolean> | 
					
						
							|  |  |  |     // Radius of what is shown. Useless to select a value > searchRadius; defaults to searchRadius
 | 
					
						
							|  |  |  |     shownRadius?: UIEventSource<number> | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | class ImagesInLoadedDataFetcher { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     private allElements: ElementStorage | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |     constructor(state: { allElements: ElementStorage }) { | 
					
						
							|  |  |  |         this.allElements = state.allElements | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     public fetchAround(loc: { lon: number; lat: number; searchRadius?: number }): P4CPicture[] { | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |         const foundImages: P4CPicture[] = [] | 
					
						
							|  |  |  |         this.allElements.ContainingFeatures.forEach((feature) => { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |             const props = feature.properties | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |             const images = [] | 
					
						
							|  |  |  |             if (props.image) { | 
					
						
							|  |  |  |                 images.push(props.image) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             for (let i = 0; i < 10; i++) { | 
					
						
							|  |  |  |                 if (props["image:" + i]) { | 
					
						
							|  |  |  |                     images.push(props["image:" + i]) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (images.length == 0) { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                 return | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |             const centerpoint = GeoOperations.centerpointCoordinates(feature) | 
					
						
							|  |  |  |             const d = GeoOperations.distanceBetween(centerpoint, [loc.lon, loc.lat]) | 
					
						
							|  |  |  |             if (loc.searchRadius !== undefined && d > loc.searchRadius) { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                 return | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |             for (const image of images) { | 
					
						
							|  |  |  |                 foundImages.push({ | 
					
						
							|  |  |  |                     pictureUrl: image, | 
					
						
							|  |  |  |                     thumbUrl: image, | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                     coordinates: { lng: centerpoint[0], lat: centerpoint[1] }, | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |                     provider: "OpenStreetMap", | 
					
						
							|  |  |  |                     details: { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                         isSpherical: false, | 
					
						
							|  |  |  |                     }, | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |                 }) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |         const cleaned: P4CPicture[] = [] | 
					
						
							|  |  |  |         const seen = new Set<string>() | 
					
						
							|  |  |  |         for (const foundImage of foundImages) { | 
					
						
							|  |  |  |             if (seen.has(foundImage.pictureUrl)) { | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             seen.add(foundImage.pictureUrl) | 
					
						
							|  |  |  |             cleaned.push(foundImage) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return cleaned | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default class NearbyImages extends Lazy { | 
					
						
							|  |  |  |     constructor(options: NearbyImageOptions, state?: { allElements: ElementStorage }) { | 
					
						
							|  |  |  |         super(() => { | 
					
						
							|  |  |  |             const t = Translations.t.image.nearbyPictures | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |             const shownImages = options.shownImagesCount ?? new UIEventSource(25) | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             const loadedPictures = NearbyImages.buildPictureFetcher(options, state) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |             const loadMoreButton = new Combine([ | 
					
						
							|  |  |  |                 new SubtleButton(Svg.add_svg(), t.loadMore).onClick(() => { | 
					
						
							|  |  |  |                     shownImages.setData(shownImages.data + 25) | 
					
						
							|  |  |  |                 }), | 
					
						
							|  |  |  |             ]).SetClass("flex flex-col justify-center") | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |             const imageElements = loadedPictures.map( | 
					
						
							|  |  |  |                 (imgs) => { | 
					
						
							|  |  |  |                     if (imgs === undefined) { | 
					
						
							|  |  |  |                         return [] | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     const elements = (imgs.images ?? []) | 
					
						
							|  |  |  |                         .slice(0, shownImages.data) | 
					
						
							|  |  |  |                         .map((i) => this.prepareElement(i)) | 
					
						
							|  |  |  |                     if (imgs.images !== undefined && elements.length < imgs.images.length) { | 
					
						
							|  |  |  |                         // We effectively sliced some items, so we can increase the count
 | 
					
						
							|  |  |  |                         elements.push(loadMoreButton) | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     return elements | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 [shownImages] | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return new VariableUiElement( | 
					
						
							|  |  |  |                 loadedPictures.map((loaded) => { | 
					
						
							|  |  |  |                     if (loaded?.images === undefined) { | 
					
						
							|  |  |  |                         return NearbyImages.NoImagesView(new Loading(t.loading)).SetClass( | 
					
						
							|  |  |  |                             "animate-pulse" | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     const images = loaded.images | 
					
						
							|  |  |  |                     const beforeFilter = loaded?.beforeFilter | 
					
						
							|  |  |  |                     if (beforeFilter === 0) { | 
					
						
							|  |  |  |                         return NearbyImages.NoImagesView(t.nothingFound.SetClass("alert block")) | 
					
						
							|  |  |  |                     } else if (images.length === 0) { | 
					
						
							|  |  |  |                         const removeFiltersButton = new SubtleButton( | 
					
						
							|  |  |  |                             Svg.filter_disable_svg(), | 
					
						
							|  |  |  |                             t.removeFilters | 
					
						
							|  |  |  |                         ).onClick(() => { | 
					
						
							|  |  |  |                             options.shownRadius.setData(options.searchRadius) | 
					
						
							|  |  |  |                             options.allowSpherical.setData(true) | 
					
						
							|  |  |  |                             options.towardscenter.setData(false) | 
					
						
							|  |  |  |                         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         return NearbyImages.NoImagesView( | 
					
						
							|  |  |  |                             t.allFiltered.SetClass("font-bold"), | 
					
						
							|  |  |  |                             removeFiltersButton | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                     return new SlideShow(imageElements) | 
					
						
							|  |  |  |                 }) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |         }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     private static NoImagesView(...elems: BaseUIElement[]) { | 
					
						
							|  |  |  |         return new Combine(elems) | 
					
						
							|  |  |  |             .SetClass("flex flex-col justify-center items-center bg-gray-200 mb-2 rounded-lg") | 
					
						
							|  |  |  |             .SetStyle( | 
					
						
							|  |  |  |                 "height: calc( var(--image-carousel-height) - 0.5rem ) ; max-height: calc( var(--image-carousel-height) - 0.5rem );" | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     private static buildPictureFetcher( | 
					
						
							|  |  |  |         options: NearbyImageOptions, | 
					
						
							|  |  |  |         state?: { allElements: ElementStorage } | 
					
						
							|  |  |  |     ) { | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |         const P4C = require("../../vendor/P4C.min") | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         const picManager = new P4C.PicturesManager({}) | 
					
						
							|  |  |  |         const searchRadius = options.searchRadius ?? 500 | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         const nearbyImages = | 
					
						
							|  |  |  |             state !== undefined ? new ImagesInLoadedDataFetcher(state).fetchAround(options) : [] | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-05 02:24:14 +02:00
										 |  |  |         return Stores.FromPromise<P4CPicture[]>( | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |             picManager.startPicsRetrievalAround( | 
					
						
							|  |  |  |                 new P4C.LatLng(options.lat, options.lon), | 
					
						
							|  |  |  |                 options.searchRadius ?? 500, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     mindate: | 
					
						
							|  |  |  |                         new Date().getTime() - | 
					
						
							|  |  |  |                         (options.maxDaysOld ?? 3 * 365) * 24 * 60 * 60 * 1000, | 
					
						
							|  |  |  |                     towardscenter: false, | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         ).map( | 
					
						
							|  |  |  |             (images) => { | 
					
						
							|  |  |  |                 if (images === undefined) { | 
					
						
							|  |  |  |                     return undefined | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 images = (images ?? []).concat(nearbyImages) | 
					
						
							|  |  |  |                 const blacklisted = options.blacklist?.data | 
					
						
							|  |  |  |                 images = images?.filter( | 
					
						
							|  |  |  |                     (i) => | 
					
						
							|  |  |  |                         !blacklisted?.some((notAllowed) => | 
					
						
							|  |  |  |                             Mapillary.sameUrl(i.pictureUrl, notAllowed.url) | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 const beforeFilterCount = images.length | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (!options?.allowSpherical?.data) { | 
					
						
							|  |  |  |                     images = images?.filter((i) => i.details.isSpherical !== true) | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                 const shownRadius = options?.shownRadius?.data ?? searchRadius | 
					
						
							|  |  |  |                 if (shownRadius !== searchRadius) { | 
					
						
							|  |  |  |                     images = images.filter((i) => { | 
					
						
							|  |  |  |                         const d = GeoOperations.distanceBetween( | 
					
						
							|  |  |  |                             [i.coordinates.lng, i.coordinates.lat], | 
					
						
							|  |  |  |                             [options.lon, options.lat] | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                         return d <= shownRadius | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 if (options.towardscenter?.data) { | 
					
						
							|  |  |  |                     images = images.filter((i) => { | 
					
						
							|  |  |  |                         if (i.direction === undefined || isNaN(i.direction)) { | 
					
						
							|  |  |  |                             return false | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         const bearing = GeoOperations.bearing( | 
					
						
							|  |  |  |                             [i.coordinates.lng, i.coordinates.lat], | 
					
						
							|  |  |  |                             [options.lon, options.lat] | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                         const diff = Math.abs((i.direction - bearing) % 360) | 
					
						
							|  |  |  |                         return diff < 40 | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                 images?.sort((a, b) => { | 
					
						
							|  |  |  |                     const distanceA = GeoOperations.distanceBetween( | 
					
						
							|  |  |  |                         [a.coordinates.lng, a.coordinates.lat], | 
					
						
							|  |  |  |                         [options.lon, options.lat] | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     const distanceB = GeoOperations.distanceBetween( | 
					
						
							|  |  |  |                         [b.coordinates.lng, b.coordinates.lat], | 
					
						
							|  |  |  |                         [options.lon, options.lat] | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     return distanceA - distanceB | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |                 }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                 return { images, beforeFilter: beforeFilterCount } | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             [options.blacklist, options.allowSpherical, options.towardscenter, options.shownRadius] | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected prepareElement(info: P4CPicture): BaseUIElement { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         const provider = AllImageProviders.byName(info.provider) | 
					
						
							|  |  |  |         return new AttributedImage({ url: info.pictureUrl, provider }) | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |     private static asAttributedImage(info: P4CPicture): AttributedImage { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         const provider = AllImageProviders.byName(info.provider) | 
					
						
							|  |  |  |         return new AttributedImage({ url: info.thumbUrl, provider, date: new Date(info.date) }) | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-05 02:24:14 +02:00
										 |  |  |     protected asToggle(info: P4CPicture): ClickableToggle { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         const imgNonSelected = NearbyImages.asAttributedImage(info) | 
					
						
							|  |  |  |         const imageSelected = NearbyImages.asAttributedImage(info) | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const nonSelected = new Combine([imgNonSelected]).SetClass("relative block") | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         const hoveringCheckmark = new Combine([ | 
					
						
							|  |  |  |             Svg.confirm_svg().SetClass("block w-24 h-24 -ml-12 -mt-12"), | 
					
						
							|  |  |  |         ]).SetClass("absolute left-1/2 top-1/2 w-0") | 
					
						
							|  |  |  |         const selected = new Combine([imageSelected, hoveringCheckmark]).SetClass("relative block") | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         return new ClickableToggle(selected, nonSelected).SetClass("").ToggleOnClick() | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export class SelectOneNearbyImage extends NearbyImages implements InputElement<P4CPicture> { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     private readonly value: UIEventSource<P4CPicture> | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |     constructor( | 
					
						
							|  |  |  |         options: NearbyImageOptions & { value?: UIEventSource<P4CPicture> }, | 
					
						
							|  |  |  |         state?: { allElements: ElementStorage } | 
					
						
							|  |  |  |     ) { | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |         super(options, state) | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         this.value = options.value ?? new UIEventSource<P4CPicture>(undefined) | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     GetValue(): UIEventSource<P4CPicture> { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         return this.value | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IsValid(t: P4CPicture): boolean { | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         return false | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected prepareElement(info: P4CPicture): BaseUIElement { | 
					
						
							|  |  |  |         const toggle = super.asToggle(info) | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         toggle.isEnabled.addCallback((enabled) => { | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |             if (enabled) { | 
					
						
							|  |  |  |                 this.value.setData(info) | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |             } else if (this.value.data === info) { | 
					
						
							|  |  |  |                 this.value.setData(undefined) | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         this.value.addCallback((inf) => { | 
					
						
							| 
									
										
										
										
											2022-06-03 01:33:41 +02:00
										 |  |  |             if (inf !== info) { | 
					
						
							| 
									
										
										
										
											2022-05-06 12:41:24 +02:00
										 |  |  |                 toggle.isEnabled.setData(false) | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return toggle | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |