| 
									
										
										
										
											2021-04-07 01:32:39 +02:00
										 |  |  | import Combine from "../Base/Combine" | 
					
						
							|  |  |  | import Translations from "../i18n/Translations" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  | import { Store } from "../../Logic/UIEventSource" | 
					
						
							|  |  |  | import { FixedUiElement } from "../Base/FixedUiElement" | 
					
						
							| 
									
										
										
										
											2023-02-06 22:43:34 +01:00
										 |  |  | import licenses from "../../assets/generated/license_info.json" | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  | import SmallLicense from "../../Models/smallLicense" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  | import { Utils } from "../../Utils" | 
					
						
							| 
									
										
										
										
											2021-04-23 17:22:01 +02:00
										 |  |  | import Link from "../Base/Link" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  | import { VariableUiElement } from "../Base/VariableUIElement" | 
					
						
							| 
									
										
										
										
											2023-02-06 22:43:34 +01:00
										 |  |  | import contributors from "../../assets/contributors.json" | 
					
						
							|  |  |  | import translators from "../../assets/translators.json" | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  | import BaseUIElement from "../BaseUIElement" | 
					
						
							| 
									
										
										
										
											2021-08-07 23:11:34 +02:00
										 |  |  | import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig" | 
					
						
							| 
									
										
										
										
											2021-10-30 01:55:32 +02:00
										 |  |  | import Title from "../Base/Title" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  | import { BBox } from "../../Logic/BBox" | 
					
						
							|  |  |  | import { OsmConnection } from "../../Logic/Osm/OsmConnection" | 
					
						
							| 
									
										
										
										
											2021-10-30 01:55:32 +02:00
										 |  |  | import Constants from "../../Models/Constants" | 
					
						
							| 
									
										
										
										
											2021-11-21 02:44:35 +01:00
										 |  |  | import ContributorCount from "../../Logic/ContributorCount" | 
					
						
							| 
									
										
										
										
											2022-02-09 22:37:21 +01:00
										 |  |  | import Img from "../Base/Img" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  | import { TypedTranslation } from "../i18n/Translation" | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  | import GeoIndexedStore from "../../Logic/FeatureSource/Actors/GeoIndexedStore" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  | import { RasterLayerPolygon } from "../../Models/RasterLayers" | 
					
						
							| 
									
										
										
										
											2021-05-19 23:29:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 01:32:39 +02:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2024-04-22 14:43:05 +02:00
										 |  |  |  * The attribution panel in the theme menu. Shows the licenses of the artwork and of the map data | 
					
						
							| 
									
										
										
										
											2021-04-07 01:32:39 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-10-30 01:55:32 +02:00
										 |  |  | export default class CopyrightPanel extends Combine { | 
					
						
							|  |  |  |     private static LicenseObject = CopyrightPanel.GenerateLicenses() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constructor(state: { | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |         layout: LayoutConfig | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |         mapProperties: { | 
					
						
							|  |  |  |             readonly bounds: Store<BBox> | 
					
						
							|  |  |  |             readonly rasterLayer: Store<RasterLayerPolygon> | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |         osmConnection: OsmConnection | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |         dataIsLoading: Store<boolean> | 
					
						
							|  |  |  |         perLayer: ReadonlyMap<string, GeoIndexedStore> | 
					
						
							| 
									
										
										
										
											2021-11-21 02:44:35 +01:00
										 |  |  |     }) { | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |         const t = Translations.t.general.attribution | 
					
						
							| 
									
										
										
										
											2023-03-28 05:13:48 +02:00
										 |  |  |         const layoutToUse = state.layout | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-22 14:43:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const iconAttributions: BaseUIElement[] = (layoutToUse.getUsedImages()).map( | 
					
						
							| 
									
										
										
										
											2023-02-03 03:56:31 +01:00
										 |  |  |             CopyrightPanel.IconAttribution | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         let maintainer: BaseUIElement = undefined | 
					
						
							| 
									
										
										
										
											2022-08-22 14:46:55 +02:00
										 |  |  |         if (layoutToUse.credits !== undefined && layoutToUse.credits !== "") { | 
					
						
							|  |  |  |             maintainer = t.themeBy.Subs({ author: layoutToUse.credits }) | 
					
						
							| 
									
										
										
										
											2021-10-30 01:55:32 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-26 21:40:38 +01:00
										 |  |  |         const contributions = new ContributorCount(state).Contributors | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |         const dataContributors = new VariableUiElement( | 
					
						
							|  |  |  |             contributions.map((contributions) => { | 
					
						
							| 
									
										
										
										
											2021-11-07 16:34:51 +01:00
										 |  |  |                 if (contributions === undefined) { | 
					
						
							| 
									
										
										
										
											2021-09-21 02:10:42 +02:00
										 |  |  |                     return "" | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2021-06-27 02:41:30 +02:00
										 |  |  |                 const sorted = Array.from(contributions, ([name, value]) => ({ | 
					
						
							|  |  |  |                     name, | 
					
						
							|  |  |  |                     value, | 
					
						
							|  |  |  |                 })).filter((x) => x.name !== undefined && x.name !== "undefined") | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |                 if (sorted.length === 0) { | 
					
						
							|  |  |  |                     return "" | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 sorted.sort((a, b) => b.value - a.value) | 
					
						
							|  |  |  |                 let hiddenCount = 0 | 
					
						
							|  |  |  |                 if (sorted.length > 10) { | 
					
						
							|  |  |  |                     hiddenCount = sorted.length - 10 | 
					
						
							|  |  |  |                     sorted.splice(10, sorted.length - 10) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 const links = sorted.map( | 
					
						
							|  |  |  |                     (kv) => | 
					
						
							|  |  |  |                         `<a href="https://openstreetmap.org/user/${kv.name}" target="_blank">${kv.name}</a>` | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |                 const contribs = links.join(", ") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-24 14:21:41 +02:00
										 |  |  |                 if (hiddenCount <= 0) { | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |                     return t.mapContributionsBy.Subs({ | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |                         contributors: contribs, | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |                     }) | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |                     return t.mapContributionsByAndHidden.Subs({ | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |                         contributors: contribs, | 
					
						
							|  |  |  |                         hiddenCount: hiddenCount, | 
					
						
							| 
									
										
										
										
											2021-06-10 01:36:20 +02:00
										 |  |  |                     }) | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |             }) | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |         super( | 
					
						
							|  |  |  |             [ | 
					
						
							|  |  |  |                 new Title(t.attributionTitle), | 
					
						
							|  |  |  |                 t.attributionContent, | 
					
						
							| 
									
										
										
										
											2023-05-18 15:44:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |                 new VariableUiElement( | 
					
						
							|  |  |  |                     state.mapProperties.rasterLayer.mapD((layer) => { | 
					
						
							|  |  |  |                         const props = layer.properties | 
					
						
							|  |  |  |                         const attrUrl = props.attribution?.url | 
					
						
							|  |  |  |                         const attrText = props.attribution?.text | 
					
						
							| 
									
										
										
										
											2023-05-18 15:44:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |                         let bgAttr: BaseUIElement | string = undefined | 
					
						
							|  |  |  |                         if (attrText && attrUrl) { | 
					
						
							|  |  |  |                             bgAttr = | 
					
						
							| 
									
										
										
										
											2023-09-21 15:29:34 +02:00
										 |  |  |                                 "<a href='" + | 
					
						
							|  |  |  |                                 attrUrl + | 
					
						
							|  |  |  |                                 "' target='_blank' rel='noopener'>" + | 
					
						
							|  |  |  |                                 attrText + | 
					
						
							|  |  |  |                                 "</a>" | 
					
						
							| 
									
										
										
										
											2023-06-14 20:39:36 +02:00
										 |  |  |                         } else if (attrUrl) { | 
					
						
							|  |  |  |                             bgAttr = attrUrl | 
					
						
							|  |  |  |                         } else { | 
					
						
							|  |  |  |                             bgAttr = attrText | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         if (bgAttr) { | 
					
						
							|  |  |  |                             return Translations.t.general.attribution.attributionBackgroundLayerWithCopyright.Subs( | 
					
						
							|  |  |  |                                 { | 
					
						
							|  |  |  |                                     name: props.name, | 
					
						
							|  |  |  |                                     copyright: bgAttr, | 
					
						
							|  |  |  |                                 } | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         return Translations.t.general.attribution.attributionBackgroundLayer.Subs( | 
					
						
							|  |  |  |                             props | 
					
						
							|  |  |  |                         ) | 
					
						
							|  |  |  |                     }) | 
					
						
							|  |  |  |                 ), | 
					
						
							| 
									
										
										
										
											2023-05-18 15:44:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |                 maintainer, | 
					
						
							|  |  |  |                 dataContributors, | 
					
						
							|  |  |  |                 CopyrightPanel.CodeContributors(contributors, t.codeContributionsBy), | 
					
						
							|  |  |  |                 CopyrightPanel.CodeContributors(translators, t.translatedBy), | 
					
						
							|  |  |  |                 new FixedUiElement("MapComplete " + Constants.vNumber).SetClass("font-bold"), | 
					
						
							| 
									
										
										
										
											2021-10-30 01:55:32 +02:00
										 |  |  |                 new Title(t.iconAttribution.title, 3), | 
					
						
							| 
									
										
										
										
											2021-11-21 02:44:35 +01:00
										 |  |  |                 ...iconAttributions, | 
					
						
							| 
									
										
										
										
											2021-10-30 01:55:32 +02:00
										 |  |  |             ].map((e) => e?.SetClass("mt-4")) | 
					
						
							| 
									
										
										
										
											2022-09-08 21:40:48 +02:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-07-03 21:18:19 +02:00
										 |  |  |         this.SetClass("flex flex-col link-underline overflow-hidden") | 
					
						
							| 
									
										
										
										
											2022-02-01 04:14:54 +01:00
										 |  |  |         this.SetStyle("max-width:100%; width: 40rem; margin-left: 0.75rem; margin-right: 0.5rem") | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-13 01:19:28 +02:00
										 |  |  |     private static CodeContributors( | 
					
						
							|  |  |  |         contributors, | 
					
						
							|  |  |  |         translation: TypedTranslation<{ contributors; hiddenCount }> | 
					
						
							|  |  |  |     ): BaseUIElement { | 
					
						
							| 
									
										
										
										
											2021-05-19 19:39:55 +02:00
										 |  |  |         const total = contributors.contributors.length | 
					
						
							| 
									
										
										
										
											2021-06-27 02:41:30 +02:00
										 |  |  |         let filtered = [...contributors.contributors] | 
					
						
							| 
									
										
										
										
											2021-05-19 23:29:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-27 02:41:30 +02:00
										 |  |  |         filtered.splice(10, total - 10) | 
					
						
							| 
									
										
										
										
											2021-05-19 23:29:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-27 02:41:30 +02:00
										 |  |  |         let contribsStr = filtered.map((c) => c.contributor).join(", ") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (contribsStr === "") { | 
					
						
							| 
									
										
										
										
											2021-05-27 18:56:02 +02:00
										 |  |  |             // Hmm, something went wrong loading the contributors list. Lets show nothing
 | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-19 23:29:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-14 19:53:23 +01:00
										 |  |  |         return translation.Subs({ | 
					
						
							| 
									
										
										
										
											2021-05-19 23:29:23 +02:00
										 |  |  |             contributors: contribsStr, | 
					
						
							|  |  |  |             hiddenCount: total - 10, | 
					
						
							|  |  |  |         }) | 
					
						
							| 
									
										
										
										
											2021-05-19 19:39:55 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-19 23:29:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-12 02:58:32 +02:00
										 |  |  |     private static IconAttribution(iconPath: string): BaseUIElement { | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |         if (iconPath.startsWith("http")) { | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |             try { | 
					
						
							| 
									
										
										
										
											2022-02-22 13:43:36 +01:00
										 |  |  |                 iconPath = "." + new URL(iconPath).pathname | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |             } catch (e) { | 
					
						
							| 
									
										
										
										
											2022-02-22 13:43:36 +01:00
										 |  |  |                 console.warn(e) | 
					
						
							| 
									
										
										
										
											2022-02-14 15:41:14 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-30 01:55:32 +02:00
										 |  |  |         const license: SmallLicense = CopyrightPanel.LicenseObject[iconPath] | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |         if (license == undefined) { | 
					
						
							|  |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |         if (license.license.indexOf("trivial") >= 0) { | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |             return undefined | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-10 23:48:00 +02:00
										 |  |  |         const sources = Utils.NoNull(Utils.NoEmpty(license.sources)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |         return new Combine([ | 
					
						
							| 
									
										
										
										
											2022-02-09 22:37:21 +01:00
										 |  |  |             new Img(iconPath).SetClass("w-12 min-h-12 mr-2 mb-2"), | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |             new Combine([ | 
					
						
							|  |  |  |                 new FixedUiElement(license.authors.join("; ")).SetClass("font-bold"), | 
					
						
							| 
									
										
										
										
											2022-04-01 12:51:55 +02:00
										 |  |  |                 license.license, | 
					
						
							|  |  |  |                 new Combine([ | 
					
						
							|  |  |  |                     ...sources.map((lnk) => { | 
					
						
							|  |  |  |                         let sourceLinkContent = lnk | 
					
						
							|  |  |  |                         try { | 
					
						
							|  |  |  |                             sourceLinkContent = new URL(lnk).hostname | 
					
						
							|  |  |  |                         } catch { | 
					
						
							|  |  |  |                             console.error("Not a valid URL:", lnk) | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         return new Link(sourceLinkContent, lnk, true).SetClass("mr-2 mb-2") | 
					
						
							|  |  |  |                     }), | 
					
						
							|  |  |  |                 ]).SetClass("flex flex-wrap"), | 
					
						
							| 
									
										
										
										
											2021-07-03 21:18:19 +02:00
										 |  |  |             ]) | 
					
						
							|  |  |  |                 .SetClass("flex flex-col") | 
					
						
							|  |  |  |                 .SetStyle("width: calc(100% - 50px - 0.5em); min-width: 12rem;"), | 
					
						
							|  |  |  |         ]).SetClass("flex flex-wrap border-b border-gray-300 m-2 border-box") | 
					
						
							| 
									
										
										
										
											2021-04-09 02:57:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private static GenerateLicenses() { | 
					
						
							|  |  |  |         const allLicenses = {} | 
					
						
							|  |  |  |         for (const key in licenses) { | 
					
						
							|  |  |  |             const license: SmallLicense = licenses[key] | 
					
						
							|  |  |  |             allLicenses[license.path] = license | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return allLicenses | 
					
						
							| 
									
										
										
										
											2021-04-07 01:32:39 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } |