forked from MapComplete/MapComplete
		
	Merge develop
This commit is contained in:
		
						commit
						d0ebd0e233
					
				
					 187 changed files with 1201 additions and 22352 deletions
				
			
		
							
								
								
									
										
											BIN
										
									
								
								Docs/Security/MapComplete Pentest Report 2023 1.0.pdf
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								Docs/Security/MapComplete Pentest Report 2023 1.0.pdf
									
										
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							|  | @ -34,7 +34,7 @@ OpenStreetMap data can be reused freely, including for commercial purposes. Howe | ||||||
| 1. Give attribution | 1. Give attribution | ||||||
| 2. Keep the data open - changes to data based on OpenStreetMap must be published under the same license. | 2. Keep the data open - changes to data based on OpenStreetMap must be published under the same license. | ||||||
| 
 | 
 | ||||||
| You can read the [full copyright notice here](https://osm.org/copyright) | See the [full copyright notice](https://osm.org/copyright) for details | ||||||
| 
 | 
 | ||||||
| This also means that we are *not* allowed to copy data from other maps. Do not enter data based on Google Maps! | This also means that we are *not* allowed to copy data from other maps. Do not enter data based on Google Maps! | ||||||
| 
 | 
 | ||||||
|  | @ -83,7 +83,7 @@ These features are based on OpenStreetMap. If some data is not known, the user w | ||||||
| 
 | 
 | ||||||
| <img src="../../public/assets/docs/UIExample.png" class="w-1/2"/> | <img src="../../public/assets/docs/UIExample.png" class="w-1/2"/> | ||||||
| 
 | 
 | ||||||
| Data can also be loaded and visualised from external sources. No changes can be made to the data in that case. | Data can also be loaded and visualised from external sources. No changes can be made to this externally loaded data in that case. | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -29,7 +29,7 @@ Note that this also works withing predifined options | ||||||
| 
 | 
 | ||||||
| # Special values | # Special values | ||||||
| 
 | 
 | ||||||
| Special components can be summoned by calling them. For example, the relevant wikipedia will be displayed by entering the text `{wikipedia()}`. A table with opening hours is displayed with `{opening_hours()}`. For a full reference, [see the documentation](../SpecialRenderings.md). | Special components can be summoned by calling them. For example, the relevant wikipedia will be displayed by entering the text `{wikipedia()}`. A table with opening hours is displayed with `{opening_hours()}`. For a full reference, [see the documentation](https://github.com/pietervdvn/MapComplete/blob/master/Docs/SpecialRenderings.md). | ||||||
| 
 | 
 | ||||||
| # Requesting data with predefined options | # Requesting data with predefined options | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -189,9 +189,6 @@ | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       "id": "rating", |       "id": "rating", | ||||||
|       "labels": [ |  | ||||||
|         "defaults" |  | ||||||
|       ], |  | ||||||
|       "icon": { |       "icon": { | ||||||
|         "class": "w-20 mx-1 flex items-center" |         "class": "w-20 mx-1 flex items-center" | ||||||
|       }, |       }, | ||||||
|  |  | ||||||
|  | @ -66,11 +66,11 @@ | ||||||
|       "marker": [ |       "marker": [ | ||||||
|         { |         { | ||||||
|           "icon": { |           "icon": { | ||||||
|             "render": "note", |             "render": "./assets/svg/note.svg", | ||||||
|             "mappings": [ |             "mappings": [ | ||||||
|               { |               { | ||||||
|                 "if": "closed_at~*", |                 "if": "closed_at~*", | ||||||
|                 "then": "resolved" |                 "then": "./assets/svg/resolved.svg" | ||||||
|               } |               } | ||||||
|             ] |             ] | ||||||
|           } |           } | ||||||
|  | @ -80,11 +80,11 @@ | ||||||
|       "iconBadges": [ |       "iconBadges": [ | ||||||
|         { |         { | ||||||
|           "if": "_total_comments>1", |           "if": "_total_comments>1", | ||||||
|           "then": "circle:white;speech_bubble" |           "then": "circle:white;./assets/svg/speech_bubble.svg" | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|           "if": "_is_import_note~*", |           "if": "_is_import_note~*", | ||||||
|           "then": "addSmall" |           "then": "./assets/svg/addSmall.svg" | ||||||
|         } |         } | ||||||
|       ], |       ], | ||||||
|       "anchor": "bottom" |       "anchor": "bottom" | ||||||
|  |  | ||||||
|  | @ -160,7 +160,7 @@ | ||||||
|               "craft=key_cutter" |               "craft=key_cutter" | ||||||
|             ] |             ] | ||||||
|           }, |           }, | ||||||
|           "then": "./assets/layers/id_presets/fas-key.svg" |           "then": "circle:white;./assets/layers/id_presets/fas-key.svg" | ||||||
|         } |         } | ||||||
|       ], |       ], | ||||||
|       "label": { |       "label": { | ||||||
|  |  | ||||||
|  | @ -372,7 +372,7 @@ | ||||||
|       "mappings": [ |       "mappings": [ | ||||||
|         { |         { | ||||||
|           "if": "mapcomplete-show_crosshair=yes", |           "if": "mapcomplete-show_crosshair=yes", | ||||||
|           "then": "Show a crosshair in the center of the map (when zoomed in above level 17)" |           "then": "Show a crosshair in the center of the map when zoomed in above level 17" | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|           "if": "mapcomplete-show_crosshair=no", |           "if": "mapcomplete-show_crosshair=no", | ||||||
|  | @ -382,6 +382,10 @@ | ||||||
|           "if": "mapcomplete-show_crosshair=", |           "if": "mapcomplete-show_crosshair=", | ||||||
|           "then": "Do not show a crosshair in the center of the map", |           "then": "Do not show a crosshair in the center of the map", | ||||||
|           "hideInAnswer": true |           "hideInAnswer": true | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |           "if": "mapcomplete-show_crosshair=always", | ||||||
|  |           "then": "Always show a crosshair in the center of the map" | ||||||
|         } |         } | ||||||
|       ] |       ] | ||||||
|     }, |     }, | ||||||
|  |  | ||||||
|  | @ -1,4 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="UTF-8" standalone="yes"?> |  | ||||||
| <svg xmlns="http://www.w3.org/2000/svg" width="374px" height="259px" viewBox="0 0 374 259" version="1.1"> |  | ||||||
|   <g id="surface1"/> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 189 B | 
|  | @ -1,4 +0,0 @@ | ||||||
| <?xml version="1.0" encoding="UTF-8" standalone="yes"?> |  | ||||||
| <svg xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1"> |  | ||||||
|   <g id="surface1"/> |  | ||||||
| </svg> |  | ||||||
| Before Width: | Height: | Size: 189 B | 
|  | @ -84,7 +84,7 @@ | ||||||
|         "eu": "Akanpatzeko tokiak", |         "eu": "Akanpatzeko tokiak", | ||||||
|         "pl": "Miejsca kamperowe" |         "pl": "Miejsca kamperowe" | ||||||
|       }, |       }, | ||||||
|       "minzoom": 10, |       "minzoom": 7, | ||||||
|       "source": { |       "source": { | ||||||
|         "osmTags": { |         "osmTags": { | ||||||
|           "and": [ |           "and": [ | ||||||
|  | @ -967,7 +967,7 @@ | ||||||
|         "es": "Estaciones sanitarias", |         "es": "Estaciones sanitarias", | ||||||
|         "pl": "Stacje zrzutów sanitarnych" |         "pl": "Stacje zrzutów sanitarnych" | ||||||
|       }, |       }, | ||||||
|       "minzoom": 10, |       "minzoom": 7, | ||||||
|       "source": { |       "source": { | ||||||
|         "osmTags": { |         "osmTags": { | ||||||
|           "and": [ |           "and": [ | ||||||
|  |  | ||||||
|  | @ -40,8 +40,20 @@ | ||||||
|     "Originally created during Open Summer of Code by Pieter Fiers, Thibault Declercq, Pierre Barban, Joost Schouppe and Pieter Vander Vennet" |     "Originally created during Open Summer of Code by Pieter Fiers, Thibault Declercq, Pierre Barban, Joost Schouppe and Pieter Vander Vennet" | ||||||
|   ], |   ], | ||||||
|   "layers": [ |   "layers": [ | ||||||
|  |     { | ||||||
|  |       "builtin": [ | ||||||
|         "bike_cafe", |         "bike_cafe", | ||||||
|         "bike_shop", |         "bike_shop", | ||||||
|  |         "bike_repair_station", | ||||||
|  |         "bicycle_tube_vending_machine", | ||||||
|  |         "drinking_water", | ||||||
|  |         "bike_themed_object", | ||||||
|  |         "bike_cleaning" | ||||||
|  |       ], | ||||||
|  |       "override": { | ||||||
|  |         "minzoom": 11 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     { |     { | ||||||
|       "builtin": [ |       "builtin": [ | ||||||
|         "bicycle_rental" |         "bicycle_rental" | ||||||
|  | @ -113,11 +125,6 @@ | ||||||
|         "minzoom": 13 |         "minzoom": 13 | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "bike_repair_station", |  | ||||||
|     "bicycle_tube_vending_machine", |  | ||||||
|     "drinking_water", |  | ||||||
|     "bike_themed_object", |  | ||||||
|     "bike_cleaning", |  | ||||||
|     "bike_parking", |     "bike_parking", | ||||||
|     { |     { | ||||||
|       "builtin": [ |       "builtin": [ | ||||||
|  |  | ||||||
|  | @ -785,7 +785,7 @@ | ||||||
|     } |     } | ||||||
|   ], |   ], | ||||||
|   "overrideAll": { |   "overrideAll": { | ||||||
|     "minzoom": 18 |     "minzoom": 17 | ||||||
|   }, |   }, | ||||||
|   "widenFactor": 2, |   "widenFactor": 2, | ||||||
|   "overpassMaxZoom": 15, |   "overpassMaxZoom": 15, | ||||||
|  |  | ||||||
|  | @ -768,7 +768,7 @@ | ||||||
|             ], |             ], | ||||||
|             "marker": [ |             "marker": [ | ||||||
|               { |               { | ||||||
|                 "icon": "statistics:black" |                 "icon": "./assets/svg/statistics.svg" | ||||||
|               } |               } | ||||||
|             ], |             ], | ||||||
|             "iconSize": "30,30", |             "iconSize": "30,30", | ||||||
|  |  | ||||||
|  | @ -294,7 +294,7 @@ | ||||||
|             ], |             ], | ||||||
|             "marker": [ |             "marker": [ | ||||||
|               { |               { | ||||||
|                 "icon": "statistics:black" |                 "icon": "./assets/svg/statistics.svg" | ||||||
|               } |               } | ||||||
|             ], |             ], | ||||||
|             "iconSize": "30,30", |             "iconSize": "30,30", | ||||||
|  |  | ||||||
|  | @ -467,7 +467,7 @@ | ||||||
|             ], |             ], | ||||||
|             "marker": [ |             "marker": [ | ||||||
|               { |               { | ||||||
|                 "icon": "statistics" |                 "icon": "./assets/svg/statistics.svg" | ||||||
|               } |               } | ||||||
|             ] |             ] | ||||||
|           } |           } | ||||||
|  |  | ||||||
|  | @ -166,31 +166,31 @@ | ||||||
|                 { |                 { | ||||||
|                   "if": "sidewalk:left|right=yes", |                   "if": "sidewalk:left|right=yes", | ||||||
|                   "then": { |                   "then": { | ||||||
|                     "en": "Yes, there is a sidewalk on this side of the road", |                     "en": "There is a sidewalk on this side of the road", | ||||||
|                     "de": "Ja, es gibt einen Bürgersteig auf dieser Straßenseite", |                     "de": "Es gibt einen Bürgersteig auf dieser Straßenseite", | ||||||
|                     "da": "Ja, der er et fortov på denne side af vejen", |                     "da": "Der er et fortov på denne side af vejen", | ||||||
|                     "nl": "Ja, er is een stoep aan deze kant van de weg", |                     "nl": "Er is een stoep aan deze kant van de weg", | ||||||
|                     "fr": "Oui, il y a un trottoir de ce côté de la route", |                     "fr": "Il y a un trottoir de ce côté de la route", | ||||||
|                     "ca": "Sí, hi ha una vorera a aquest costat del carrer", |                     "ca": "Hi ha una vorera a aquest costat del carrer", | ||||||
|                     "es": "Sí, hay una acera en este lado de la calle", |                     "es": "Hay una acera en este lado de la calle", | ||||||
|                     "cs": "Ano, na této straně silnice je chodník", |                     "cs": "Na této straně silnice je chodník", | ||||||
|                     "it": "Sì, c'è un marciapiede su questo lato della strada", |                     "it": "C'è un marciapiede su questo lato della strada", | ||||||
|                     "pl": "Tak, jest chodnik z boku drogi" |                     "pl": "Jest chodnik z boku drogi" | ||||||
|                   } |                   } | ||||||
|                 }, |                 }, | ||||||
|                 { |                 { | ||||||
|                   "if": "sidewalk:left|right=no", |                   "if": "sidewalk:left|right=no", | ||||||
|                   "then": { |                   "then": { | ||||||
|                     "en": "No, there is no sidewalk to walk on", |                     "en": "There is no sidewalk to walk on", | ||||||
|                     "de": "Nein, es gibt keinen Bürgersteig für Fußgänger", |                     "de": "Es gibt keinen Bürgersteig für Fußgänger", | ||||||
|                     "da": "Nej, der er ikke noget fortov at gå på", |                     "da": "Der er ikke noget fortov at gå på", | ||||||
|                     "nl": "Nee, er is geen stoep om op te lopen", |                     "nl": "Er is geen stoep om op te lopen", | ||||||
|                     "fr": "Non, il n'y a pas de trottoir où marcher", |                     "fr": "Il n'y a pas de trottoir où marcher", | ||||||
|                     "ca": "No, no hi ha vorera per la que caminar", |                     "ca": "No hi ha vorera per la que caminar", | ||||||
|                     "es": "No, no hay acera por la que caminar", |                     "es": "No hay acera por la que caminar", | ||||||
|                     "cs": "Ne, není tu žádný chodník", |                     "cs": "Není tu žádný chodník", | ||||||
|                     "it": "No, non c'è un marciapiede su cui camminare", |                     "it": "Non c'è un marciapiede su cui camminare", | ||||||
|                     "pl": "Nie, nie ma chodnika, którym można chodzić" |                     "pl": "Nie ma chodnika, którym można chodzić" | ||||||
|                   } |                   } | ||||||
|                 }, |                 }, | ||||||
|                 { |                 { | ||||||
|  |  | ||||||
|  | @ -1,6 +1,6 @@ | ||||||
| { | { | ||||||
|   "name": "mapcomplete", |   "name": "mapcomplete", | ||||||
|   "version": "0.34.9", |   "version": "0.35.1", | ||||||
|   "repository": "https://github.com/pietervdvn/MapComplete", |   "repository": "https://github.com/pietervdvn/MapComplete", | ||||||
|   "description": "A small website to edit OSM easily", |   "description": "A small website to edit OSM easily", | ||||||
|   "bugs": "https://github.com/pietervdvn/MapComplete/issues", |   "bugs": "https://github.com/pietervdvn/MapComplete/issues", | ||||||
|  |  | ||||||
|  | @ -725,14 +725,6 @@ video { | ||||||
|   left: 0px; |   left: 0px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .top-2 { |  | ||||||
|   top: 0.5rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .right-3 { |  | ||||||
|   right: 0.75rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .bottom-0 { | .bottom-0 { | ||||||
|   bottom: 0px; |   bottom: 0px; | ||||||
| } | } | ||||||
|  | @ -777,18 +769,22 @@ video { | ||||||
|   margin: 2rem; |   margin: 2rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .m-5 { | .m-4 { | ||||||
|   margin: 1.25rem; |   margin: 1rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .m-3 { | ||||||
|  |   margin: 0.75rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .m-0 { | ||||||
|  |   margin: 0px; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .m-2 { | .m-2 { | ||||||
|   margin: 0.5rem; |   margin: 0.5rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .m-4 { |  | ||||||
|   margin: 1rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .m-1 { | .m-1 { | ||||||
|   margin: 0.25rem; |   margin: 0.25rem; | ||||||
| } | } | ||||||
|  | @ -797,14 +793,6 @@ video { | ||||||
|   margin: 0.125rem; |   margin: 0.125rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .m-0 { |  | ||||||
|   margin: 0px; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .m-3 { |  | ||||||
|   margin: 0.75rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .m-6 { | .m-6 { | ||||||
|   margin: 1.5rem; |   margin: 1.5rem; | ||||||
| } | } | ||||||
|  | @ -823,16 +811,16 @@ video { | ||||||
|   margin-bottom: 0.25rem; |   margin-bottom: 0.25rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mx-1 { |  | ||||||
|   margin-left: 0.25rem; |  | ||||||
|   margin-right: 0.25rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .my-4 { | .my-4 { | ||||||
|   margin-top: 1rem; |   margin-top: 1rem; | ||||||
|   margin-bottom: 1rem; |   margin-bottom: 1rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .mx-1 { | ||||||
|  |   margin-left: 0.25rem; | ||||||
|  |   margin-right: 0.25rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .my-2 { | .my-2 { | ||||||
|   margin-top: 0.5rem; |   margin-top: 0.5rem; | ||||||
|   margin-bottom: 0.5rem; |   margin-bottom: 0.5rem; | ||||||
|  | @ -853,14 +841,22 @@ video { | ||||||
|   margin-right: 3rem; |   margin-right: 3rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mr-2 { | .mt-4 { | ||||||
|   margin-right: 0.5rem; |   margin-top: 1rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mr-4 { | .mr-4 { | ||||||
|   margin-right: 1rem; |   margin-right: 1rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .mr-2 { | ||||||
|  |   margin-right: 0.5rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .mb-16 { | ||||||
|  |   margin-bottom: 4rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .mr-6 { | .mr-6 { | ||||||
|   margin-right: 1.5rem; |   margin-right: 1.5rem; | ||||||
| } | } | ||||||
|  | @ -897,10 +893,6 @@ video { | ||||||
|   margin-top: 0.5rem; |   margin-top: 0.5rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mt-4 { |  | ||||||
|   margin-top: 1rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .mb-2 { | .mb-2 { | ||||||
|   margin-bottom: 0.5rem; |   margin-bottom: 0.5rem; | ||||||
| } | } | ||||||
|  | @ -917,10 +909,6 @@ video { | ||||||
|   margin-left: 1rem; |   margin-left: 1rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .mt-3 { |  | ||||||
|   margin-top: 0.75rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .mb-10 { | .mb-10 { | ||||||
|   margin-bottom: 2.5rem; |   margin-bottom: 2.5rem; | ||||||
| } | } | ||||||
|  | @ -1058,24 +1046,24 @@ video { | ||||||
|   height: 2rem; |   height: 2rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .h-16 { | .h-12 { | ||||||
|   height: 4rem; |   height: 3rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .h-6 { | .h-6 { | ||||||
|   height: 1.5rem; |   height: 1.5rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .h-12 { |  | ||||||
|   height: 3rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .h-fit { | .h-fit { | ||||||
|   height: -webkit-fit-content; |   height: -webkit-fit-content; | ||||||
|   height: -moz-fit-content; |   height: -moz-fit-content; | ||||||
|   height: fit-content; |   height: fit-content; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .h-16 { | ||||||
|  |   height: 4rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .h-4 { | .h-4 { | ||||||
|   height: 1rem; |   height: 1rem; | ||||||
| } | } | ||||||
|  | @ -1152,16 +1140,16 @@ video { | ||||||
|   width: 2rem; |   width: 2rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .w-16 { | .w-12 { | ||||||
|   width: 4rem; |   width: 3rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .w-6 { | .w-6 { | ||||||
|   width: 1.5rem; |   width: 1.5rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .w-12 { | .w-16 { | ||||||
|   width: 3rem; |   width: 4rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .w-screen { | .w-screen { | ||||||
|  | @ -1809,6 +1797,11 @@ video { | ||||||
|   line-height: 1.75rem; |   line-height: 1.75rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .text-base { | ||||||
|  |   font-size: 1rem; | ||||||
|  |   line-height: 1.5rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .text-lg { | .text-lg { | ||||||
|   font-size: 1.125rem; |   font-size: 1.125rem; | ||||||
|   line-height: 1.75rem; |   line-height: 1.75rem; | ||||||
|  | @ -1829,20 +1822,11 @@ video { | ||||||
|   line-height: 1.25rem; |   line-height: 1.25rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .text-base { |  | ||||||
|   font-size: 1rem; |  | ||||||
|   line-height: 1.5rem; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .text-4xl { | .text-4xl { | ||||||
|   font-size: 2.25rem; |   font-size: 2.25rem; | ||||||
|   line-height: 2.5rem; |   line-height: 2.5rem; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .font-bold { |  | ||||||
|   font-weight: 700; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .font-extrabold { | .font-extrabold { | ||||||
|   font-weight: 800; |   font-weight: 800; | ||||||
| } | } | ||||||
|  | @ -1851,6 +1835,10 @@ video { | ||||||
|   font-weight: 600; |   font-weight: 600; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | .font-bold { | ||||||
|  |   font-weight: 700; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| .font-normal { | .font-normal { | ||||||
|   font-weight: 400; |   font-weight: 400; | ||||||
| } | } | ||||||
|  | @ -1927,21 +1915,6 @@ video { | ||||||
|   color: rgb(255 255 255 / var(--tw-text-opacity)); |   color: rgb(255 255 255 / var(--tw-text-opacity)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .text-gray-900 { |  | ||||||
|   --tw-text-opacity: 1; |  | ||||||
|   color: rgb(17 24 39 / var(--tw-text-opacity)); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .text-gray-800 { |  | ||||||
|   --tw-text-opacity: 1; |  | ||||||
|   color: rgb(31 41 55 / var(--tw-text-opacity)); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .text-gray-500 { |  | ||||||
|   --tw-text-opacity: 1; |  | ||||||
|   color: rgb(107 114 128 / var(--tw-text-opacity)); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .underline { | .underline { | ||||||
|   text-decoration-line: underline; |   text-decoration-line: underline; | ||||||
| } | } | ||||||
|  | @ -2753,11 +2726,6 @@ a.link-underline { | ||||||
|     margin-right: 0.25rem; |     margin-right: 0.25rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .sm\:mx-auto { |  | ||||||
|     margin-left: auto; |  | ||||||
|     margin-right: auto; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .sm\:mt-2 { |   .sm\:mt-2 { | ||||||
|     margin-top: 0.5rem; |     margin-top: 0.5rem; | ||||||
|   } |   } | ||||||
|  | @ -2766,10 +2734,6 @@ a.link-underline { | ||||||
|     margin-right: 0.25rem; |     margin-right: 0.25rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .sm\:mt-5 { |  | ||||||
|     margin-top: 1.25rem; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .sm\:mr-4 { |   .sm\:mr-4 { | ||||||
|     margin-right: 1rem; |     margin-right: 1rem; | ||||||
|   } |   } | ||||||
|  | @ -2790,10 +2754,6 @@ a.link-underline { | ||||||
|     width: 6rem; |     width: 6rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .sm\:max-w-xl { |  | ||||||
|     max-width: 36rem; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .sm\:flex-nowrap { |   .sm\:flex-nowrap { | ||||||
|     flex-wrap: nowrap; |     flex-wrap: nowrap; | ||||||
|   } |   } | ||||||
|  | @ -2822,15 +2782,6 @@ a.link-underline { | ||||||
|     padding-top: 0.25rem; |     padding-top: 0.25rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .sm\:text-center { |  | ||||||
|     text-align: center; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .sm\:text-5xl { |  | ||||||
|     font-size: 3rem; |  | ||||||
|     line-height: 1; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .sm\:text-lg { |   .sm\:text-lg { | ||||||
|     font-size: 1.125rem; |     font-size: 1.125rem; | ||||||
|     line-height: 1.75rem; |     line-height: 1.75rem; | ||||||
|  | @ -2842,25 +2793,17 @@ a.link-underline { | ||||||
|     margin: 0.25rem; |     margin: 0.25rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .md\:m-2 { |  | ||||||
|     margin: 0.5rem; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .md\:mx-2 { |   .md\:mx-2 { | ||||||
|     margin-left: 0.5rem; |     margin-left: 0.5rem; | ||||||
|     margin-right: 0.5rem; |     margin-right: 0.5rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .md\:mr-2 { |  | ||||||
|     margin-right: 0.5rem; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .md\:mt-5 { |   .md\:mt-5 { | ||||||
|     margin-top: 1.25rem; |     margin-top: 1.25rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .md\:mt-4 { |   .md\:mr-2 { | ||||||
|     margin-top: 1rem; |     margin-right: 0.5rem; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .md\:grid { |   .md\:grid { | ||||||
|  | @ -2932,14 +2875,6 @@ a.link-underline { | ||||||
|     margin-right: 0px; |     margin-right: 0px; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .lg\:ml-40 { |  | ||||||
|     margin-left: 10rem; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .lg\:w-3\/4 { |  | ||||||
|     width: 75%; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .lg\:w-5\/12 { |   .lg\:w-5\/12 { | ||||||
|     width: 41.666667%; |     width: 41.666667%; | ||||||
|   } |   } | ||||||
|  | @ -2951,17 +2886,9 @@ a.link-underline { | ||||||
|   .lg\:grid-cols-3 { |   .lg\:grid-cols-3 { | ||||||
|     grid-template-columns: repeat(3, minmax(0, 1fr)); |     grid-template-columns: repeat(3, minmax(0, 1fr)); | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .lg\:text-left { |  | ||||||
|     text-align: left; |  | ||||||
|   } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @media (min-width: 1280px) { | @media (min-width: 1280px) { | ||||||
|   .xl\:inline { |  | ||||||
|     display: inline; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   .xl\:w-4\/12 { |   .xl\:w-4\/12 { | ||||||
|     width: 33.333333%; |     width: 33.333333%; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -1,11 +1,71 @@ | ||||||
| import * as fs from "fs" | import * as fs from "fs" | ||||||
|  | 
 | ||||||
| function genImages(dryrun = false) { | function genImages(dryrun = false) { | ||||||
|     console.log("Generating images") |     console.log("Generating images") | ||||||
|  |     // These images are not referenced via 'Svg.ts' anymore and can be ignored
 | ||||||
|  |     const blacklist: string[] = [ | ||||||
|  |         "add", | ||||||
|  |         "addSmall", | ||||||
|  |         "brick_wall_square", | ||||||
|  |         "clock", | ||||||
|  |         "community", | ||||||
|  |         "copyright", | ||||||
|  |         "cross", | ||||||
|  |         "cross_bottom_right", | ||||||
|  |         "crosshair_locked", | ||||||
|  |         "delete_not_allowed", | ||||||
|  |         "direction_gradient", | ||||||
|  |         "direction_stroke", | ||||||
|  |         "duplicate", | ||||||
|  |         "elevator", | ||||||
|  |         "elevator_wheelchair", | ||||||
|  |         "liberapay", | ||||||
|  |         "length_crosshair", | ||||||
|  |         "speech_bubble_black_outline", | ||||||
|  |         "square", | ||||||
|  |         "star_half", | ||||||
|  |         "star_outline", | ||||||
|  |         "star", | ||||||
|  |         "osm_logo_us", | ||||||
|  | 
 | ||||||
|  |         "SocialImageForeground", | ||||||
|  |         "wikipedia", | ||||||
|  |         "Upload", | ||||||
|  |         "pin", | ||||||
|  |         "mapillary_black", | ||||||
|  |         "plantnet_logo", | ||||||
|  |         "mastodon", | ||||||
|  |         "move-arrows", | ||||||
|  |         "mapcomplete_logo", | ||||||
|  |         "logo", | ||||||
|  |         "logout", | ||||||
|  |         "hand", | ||||||
|  |         "help", | ||||||
|  |         "home", | ||||||
|  |         "reload", | ||||||
|  |         "min", | ||||||
|  |         "plus", | ||||||
|  |         "not_found", | ||||||
|  |         "osm_logo_us", | ||||||
|  |         "party", | ||||||
|  |         "filter", | ||||||
|  |         "filter_disable", | ||||||
|  |         "floppy", | ||||||
|  |         "eye", | ||||||
|  |         "gear", | ||||||
|  |         "gender_bi", | ||||||
|  |         "compass", | ||||||
|  |         "blocked", | ||||||
|  |         "brick_wall", | ||||||
|  |         "brick_wall_raw", | ||||||
|  |         "brick_wall_round", | ||||||
|  |         "bug", | ||||||
|  |         "back", | ||||||
|  |     ].map((s) => s.toLowerCase()) | ||||||
|     const dir = fs.readdirSync("./assets/svg") |     const dir = fs.readdirSync("./assets/svg") | ||||||
| 
 | 
 | ||||||
|     let module = |     let module = | ||||||
|         'import Img from "./UI/Base/Img";\nimport {FixedUiElement} from "./UI/Base/FixedUiElement";\n\n/* @deprecated */\nexport default class Svg {\n\n\n' |         'import Img from "./UI/Base/Img";\nimport {FixedUiElement} from "./UI/Base/FixedUiElement";\n\n/* @deprecated */\nexport default class Svg {\n\n\n' | ||||||
|     const allNames: string[] = [] |  | ||||||
|     for (const path of dir) { |     for (const path of dir) { | ||||||
|         if (path.endsWith("license_info.json")) { |         if (path.endsWith("license_info.json")) { | ||||||
|             continue |             continue | ||||||
|  | @ -21,6 +81,7 @@ function genImages(dryrun = false) { | ||||||
|         let svg: string = fs |         let svg: string = fs | ||||||
|             .readFileSync("./assets/svg/" + path, "utf-8") |             .readFileSync("./assets/svg/" + path, "utf-8") | ||||||
|             .replace(/<\?xml.*?>/, "") |             .replace(/<\?xml.*?>/, "") | ||||||
|  |             .replace(/<!DOCTYPE [^>]*>/, "") | ||||||
|             .replace(/fill: ?none;/g, "fill: none !important;") // This is such a brittle hack...
 |             .replace(/fill: ?none;/g, "fill: none !important;") // This is such a brittle hack...
 | ||||||
|             .replace(/\n/g, " ") |             .replace(/\n/g, " ") | ||||||
|             .replace(/\r/g, "") |             .replace(/\r/g, "") | ||||||
|  | @ -37,18 +98,6 @@ function genImages(dryrun = false) { | ||||||
|         } |         } | ||||||
|         const name = path.substring(0, path.length - 4).replace(/[ -]/g, "_") |         const name = path.substring(0, path.length - 4).replace(/[ -]/g, "_") | ||||||
| 
 | 
 | ||||||
|         if (dryrun) { |  | ||||||
|             svg = "<omitting svg - dryrun>" |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         let rawName = name |  | ||||||
| 
 |  | ||||||
|         module += `    public static ${name} = "${svg}"\n` |  | ||||||
|         module += `    public static ${name}_svg() { return new Img(Svg.${rawName}, true);}\n` |  | ||||||
|         if (!dryrun) { |  | ||||||
|             allNames.push(`"${path}": Svg.${name}`) |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         const nameUC = name.toUpperCase().at(0) + name.substring(1) |         const nameUC = name.toUpperCase().at(0) + name.substring(1) | ||||||
|         const svelteCode = |         const svelteCode = | ||||||
|             '<script>\nexport let color = "#000000"\n</script>\n' + |             '<script>\nexport let color = "#000000"\n</script>\n' + | ||||||
|  | @ -60,8 +109,23 @@ function genImages(dryrun = false) { | ||||||
|                 .replace(/\\"/g, '"') |                 .replace(/\\"/g, '"') | ||||||
|                 .replace(/(rgb\(0%,0%,0%\)|#000000|#000)/g, "{color}") |                 .replace(/(rgb\(0%,0%,0%\)|#000000|#000)/g, "{color}") | ||||||
|         fs.writeFileSync("./src/assets/svg/" + nameUC + ".svelte", svelteCode, "utf8") |         fs.writeFileSync("./src/assets/svg/" + nameUC + ".svelte", svelteCode, "utf8") | ||||||
|  | 
 | ||||||
|  |         if (blacklist.some((item) => path.toLowerCase().endsWith(item + ".svg"))) { | ||||||
|  |             continue | ||||||
|  |         } | ||||||
|  |         if (dryrun) { | ||||||
|  |             svg = "<omitting svg - dryrun>" | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         let rawName = name | ||||||
|  | 
 | ||||||
|  |         module += `    public static ${name} = "${svg}"\n` | ||||||
|  |         if (!dryrun) { | ||||||
|  |             module += `    public static ${name}_svg() { return new Img(Svg.${rawName}, true);}\n` | ||||||
|  |         } else { | ||||||
|  |             module += `    public static ${name}_svg() { return new Img("", true);}\n` | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|     module += `public static All = {${allNames.join(",")}};` |  | ||||||
|     module += "}\n" |     module += "}\n" | ||||||
|     fs.writeFileSync("src/Svg.ts", module) |     fs.writeFileSync("src/Svg.ts", module) | ||||||
|     console.log("Done") |     console.log("Done") | ||||||
|  |  | ||||||
|  | @ -15,6 +15,7 @@ import { ImmutableStore } from "../src/Logic/UIEventSource" | ||||||
| import * as crypto from "crypto" | import * as crypto from "crypto" | ||||||
| import * as eli from "../src/assets/editor-layer-index.json" | import * as eli from "../src/assets/editor-layer-index.json" | ||||||
| import * as eli_global from "../src/assets/global-raster-layers.json" | import * as eli_global from "../src/assets/global-raster-layers.json" | ||||||
|  | import ValidationUtils from "../src/Models/ThemeConfig/Conversion/ValidationUtils" | ||||||
| 
 | 
 | ||||||
| const sharp = require("sharp") | const sharp = require("sharp") | ||||||
| const template = readFileSync("theme.html", "utf8") | const template = readFileSync("theme.html", "utf8") | ||||||
|  | @ -264,6 +265,7 @@ async function eliUrls(): Promise<string[]> { | ||||||
| 
 | 
 | ||||||
| async function generateCsp( | async function generateCsp( | ||||||
|     layout: LayoutConfig, |     layout: LayoutConfig, | ||||||
|  |     layoutJson: LayoutConfigJson, | ||||||
|     options: { |     options: { | ||||||
|         scriptSrcs: string[] |         scriptSrcs: string[] | ||||||
|     } |     } | ||||||
|  | @ -273,11 +275,28 @@ async function generateCsp( | ||||||
|         ...Constants.defaultOverpassUrls, |         ...Constants.defaultOverpassUrls, | ||||||
|         Constants.countryCoderEndpoint, |         Constants.countryCoderEndpoint, | ||||||
|         Constants.nominatimEndpoint, |         Constants.nominatimEndpoint, | ||||||
|  |         "https://www.openstreetmap.org", | ||||||
|         "https://api.openstreetmap.org", |         "https://api.openstreetmap.org", | ||||||
|         "https://pietervdvn.goatcounter.com", |         "https://pietervdvn.goatcounter.com", | ||||||
|     ] |     ].concat(...(await eliUrls())) | ||||||
|         .concat(...SpecialVisualizations.specialVisualizations.map((sv) => sv.needsUrls)) | 
 | ||||||
|         .concat(...(await eliUrls())) |     SpecialVisualizations.specialVisualizations.forEach(sv => { | ||||||
|  |         if(typeof sv.needsUrls === "function"){ | ||||||
|  |             return | ||||||
|  |         } | ||||||
|  |         apiUrls.push(...sv.needsUrls) | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     const usedSpecialVisualisations = ValidationUtils.getSpecialVisualisationsWithArgs(layoutJson) | ||||||
|  |     for (const usedSpecialVisualisation of usedSpecialVisualisations) { | ||||||
|  |         if (typeof usedSpecialVisualisation === "string") { | ||||||
|  |             continue | ||||||
|  |         } | ||||||
|  |         const neededUrls = usedSpecialVisualisation.func.needsUrls | ||||||
|  |         if (typeof neededUrls === "function") { | ||||||
|  |             apiUrls.push(...neededUrls(usedSpecialVisualisation.args)) | ||||||
|  |         } | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     const geojsonSources: string[] = layout.layers.map((l) => l.source?.geojsonSource) |     const geojsonSources: string[] = layout.layers.map((l) => l.source?.geojsonSource) | ||||||
|     const hosts = new Set<string>() |     const hosts = new Set<string>() | ||||||
|  | @ -299,6 +318,10 @@ async function generateCsp( | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     if (hosts.has("*")) { | ||||||
|  |         throw "* is not allowed as connect-src" | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     const connectSrc = Array.from(hosts).sort() |     const connectSrc = Array.from(hosts).sort() | ||||||
| 
 | 
 | ||||||
|     const newSrcs = connectSrc.filter((newItem) => !previousSrc.has(newItem)) |     const newSrcs = connectSrc.filter((newItem) => !previousSrc.has(newItem)) | ||||||
|  | @ -344,7 +367,13 @@ const removeOtherLanguagesHash = crypto | ||||||
|     .update(removeOtherLanguages) |     .update(removeOtherLanguages) | ||||||
|     .digest("base64") |     .digest("base64") | ||||||
| 
 | 
 | ||||||
| async function createLandingPage(layout: LayoutConfig, manifest, whiteIcons, alreadyWritten) { | async function createLandingPage( | ||||||
|  |     layout: LayoutConfig, | ||||||
|  |     layoutJson: LayoutConfigJson, | ||||||
|  |     manifest, | ||||||
|  |     whiteIcons, | ||||||
|  |     alreadyWritten | ||||||
|  | ) { | ||||||
|     Locale.language.setData(layout.language[0]) |     Locale.language.setData(layout.language[0]) | ||||||
|     const targetLanguage = layout.language[0] |     const targetLanguage = layout.language[0] | ||||||
|     const ogTitle = Translations.T(layout.title).textFor(targetLanguage).replace(/"/g, '\\"') |     const ogTitle = Translations.T(layout.title).textFor(targetLanguage).replace(/"/g, '\\"') | ||||||
|  | @ -428,7 +457,7 @@ async function createLandingPage(layout: LayoutConfig, manifest, whiteIcons, alr | ||||||
|         .replace(/<!-- THEME-SPECIFIC -->.*<!-- THEME-SPECIFIC-END-->/s, themeSpecific) |         .replace(/<!-- THEME-SPECIFIC -->.*<!-- THEME-SPECIFIC-END-->/s, themeSpecific) | ||||||
|         .replace( |         .replace( | ||||||
|             /<!-- CSP -->/, |             /<!-- CSP -->/, | ||||||
|             await generateCsp(layout, { |             await generateCsp(layout, layoutJson, { | ||||||
|                 scriptSrcs: [`'sha256-${removeOtherLanguagesHash}'`], |                 scriptSrcs: [`'sha256-${removeOtherLanguagesHash}'`], | ||||||
|             }) |             }) | ||||||
|         ) |         ) | ||||||
|  | @ -518,7 +547,13 @@ async function main(): Promise<void> { | ||||||
|         writeFile("public/" + manifestLocation, manif, err) |         writeFile("public/" + manifestLocation, manif, err) | ||||||
| 
 | 
 | ||||||
|         // Create a landing page for the given theme
 |         // Create a landing page for the given theme
 | ||||||
|         const landing = await createLandingPage(layout, manifest, whiteIcons, alreadyWritten) |         const landing = await createLandingPage( | ||||||
|  |             layout, | ||||||
|  |             layoutConfigJson, | ||||||
|  |             manifest, | ||||||
|  |             whiteIcons, | ||||||
|  |             alreadyWritten | ||||||
|  |         ) | ||||||
| 
 | 
 | ||||||
|         writeFile(enc(layout.id) + ".html", landing, err) |         writeFile(enc(layout.id) + ".html", landing, err) | ||||||
|         await createIndexFor(layout) |         await createIndexFor(layout) | ||||||
|  |  | ||||||
|  | @ -137,7 +137,6 @@ export default class GeoLocationHandler { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         console.trace("Moving the map to the GPS-location") |  | ||||||
|         mapLocation.setData({ |         mapLocation.setData({ | ||||||
|             lon: newLocation.longitude, |             lon: newLocation.longitude, | ||||||
|             lat: newLocation.latitude, |             lat: newLocation.latitude, | ||||||
|  | @ -152,7 +151,6 @@ export default class GeoLocationHandler { | ||||||
|     private CopyGeolocationIntoMapstate() { |     private CopyGeolocationIntoMapstate() { | ||||||
|         const features: UIEventSource<Feature[]> = new UIEventSource<Feature[]>([]) |         const features: UIEventSource<Feature[]> = new UIEventSource<Feature[]>([]) | ||||||
|         this.currentUserLocation = new StaticFeatureSource(features) |         this.currentUserLocation = new StaticFeatureSource(features) | ||||||
|         const keysToCopy = ["speed", "accuracy", "altitude", "altitudeAccuracy", "heading"] |  | ||||||
|         let i = 0 |         let i = 0 | ||||||
|         this.geolocationState.currentGPSLocation.addCallbackAndRun((location) => { |         this.geolocationState.currentGPSLocation.addCallbackAndRun((location) => { | ||||||
|             if (location === undefined) { |             if (location === undefined) { | ||||||
|  | @ -163,19 +161,15 @@ export default class GeoLocationHandler { | ||||||
|                 id: "gps-" + i, |                 id: "gps-" + i, | ||||||
|                 "user:location": "yes", |                 "user:location": "yes", | ||||||
|                 date: new Date().toISOString(), |                 date: new Date().toISOString(), | ||||||
|  |                 // GeolocationObject behaves really weird when indexing, so copying it one by one is the most stable
 | ||||||
|  |                 accuracy: location.accuracy, | ||||||
|  |                 speed: location.speed, | ||||||
|  |                 altitude: location.altitude, | ||||||
|  |                 altitudeAccuracy: location.altitudeAccuracy, | ||||||
|  |                 heading: location.heading, | ||||||
|             } |             } | ||||||
|             i++ |             i++ | ||||||
| 
 | 
 | ||||||
|             for (const k in keysToCopy) { |  | ||||||
|                 // For some weird reason, the 'Object.keys' method doesn't work for the 'location: GeolocationCoordinates'-object and will thus not copy all the properties when using {...location}
 |  | ||||||
|                 // As such, they are copied here
 |  | ||||||
|                 if (location[k]) { |  | ||||||
|                     properties[k] = location[k] |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             console.debug("Current location object:", location) |  | ||||||
|             properties["_all"] = JSON.stringify(location) |  | ||||||
| 
 |  | ||||||
|             const feature = <Feature>{ |             const feature = <Feature>{ | ||||||
|                 type: "Feature", |                 type: "Feature", | ||||||
|                 properties, |                 properties, | ||||||
|  | @ -184,7 +178,6 @@ export default class GeoLocationHandler { | ||||||
|                     coordinates: [location.longitude, location.latitude], |                     coordinates: [location.longitude, location.latitude], | ||||||
|                 }, |                 }, | ||||||
|             } |             } | ||||||
| 
 |  | ||||||
|             features.setData([feature]) |             features.setData([feature]) | ||||||
|         }) |         }) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -21,6 +21,7 @@ export default class UserDetails { | ||||||
|     public account_created: string |     public account_created: string | ||||||
|     public tracesCount: number = 0 |     public tracesCount: number = 0 | ||||||
|     public description: string |     public description: string | ||||||
|  |     public languages: string[] | ||||||
| 
 | 
 | ||||||
|     constructor(backend: string) { |     constructor(backend: string) { | ||||||
|         this.backend = backend |         this.backend = backend | ||||||
|  | @ -89,6 +90,7 @@ export class OsmConnection { | ||||||
|             ud.unreadMessages = 0 |             ud.unreadMessages = 0 | ||||||
|             ud.name = "Fake user" |             ud.name = "Fake user" | ||||||
|             ud.totalMessages = 42 |             ud.totalMessages = 42 | ||||||
|  |             ud.languages = ["en"] | ||||||
|         } |         } | ||||||
|         const self = this |         const self = this | ||||||
|         this.UpdateCapabilities() |         this.UpdateCapabilities() | ||||||
|  | @ -193,7 +195,7 @@ export class OsmConnection { | ||||||
|                 method: "GET", |                 method: "GET", | ||||||
|                 path: "/api/0.6/user/details", |                 path: "/api/0.6/user/details", | ||||||
|             }, |             }, | ||||||
|             function (err, details) { |             function (err, details: XMLDocument) { | ||||||
|                 if (err != null) { |                 if (err != null) { | ||||||
|                     console.log(err) |                     console.log(err) | ||||||
|                     self.loadingStatus.setData("error") |                     self.loadingStatus.setData("error") | ||||||
|  | @ -222,11 +224,14 @@ export class OsmConnection { | ||||||
|                 data.name = userInfo.getAttribute("display_name") |                 data.name = userInfo.getAttribute("display_name") | ||||||
|                 data.account_created = userInfo.getAttribute("account_created") |                 data.account_created = userInfo.getAttribute("account_created") | ||||||
|                 data.uid = Number(userInfo.getAttribute("id")) |                 data.uid = Number(userInfo.getAttribute("id")) | ||||||
|  |                 data.languages = Array.from( | ||||||
|  |                     userInfo.getElementsByTagName("languages")[0].getElementsByTagName("lang") | ||||||
|  |                 ).map((l) => l.textContent) | ||||||
|                 data.csCount = Number.parseInt( |                 data.csCount = Number.parseInt( | ||||||
|                     userInfo.getElementsByTagName("changesets")[0].getAttribute("count") ?? 0 |                     userInfo.getElementsByTagName("changesets")[0].getAttribute("count") ?? "0" | ||||||
|                 ) |                 ) | ||||||
|                 data.tracesCount = Number.parseInt( |                 data.tracesCount = Number.parseInt( | ||||||
|                     userInfo.getElementsByTagName("traces")[0].getAttribute("count") ?? 0 |                     userInfo.getElementsByTagName("traces")[0].getAttribute("count") ?? "0" | ||||||
|                 ) |                 ) | ||||||
| 
 | 
 | ||||||
|                 data.img = undefined |                 data.img = undefined | ||||||
|  |  | ||||||
|  | @ -42,6 +42,10 @@ export default class UserRelatedState { | ||||||
|     public readonly showCrosshair: UIEventSource<"yes" | undefined> |     public readonly showCrosshair: UIEventSource<"yes" | undefined> | ||||||
|     public readonly fixateNorth: UIEventSource<undefined | "yes"> |     public readonly fixateNorth: UIEventSource<undefined | "yes"> | ||||||
|     public readonly homeLocation: FeatureSource |     public readonly homeLocation: FeatureSource | ||||||
|  |     /** | ||||||
|  |      * The language as saved into the preferences of the user, if logged in. | ||||||
|  |      * Note that this is _different_ from the languages a user can set via the osm.org interface here: https://www.openstreetmap.org/preferences
 | ||||||
|  |      */ | ||||||
|     public readonly language: UIEventSource<string> |     public readonly language: UIEventSource<string> | ||||||
|     public readonly preferredBackgroundLayer: UIEventSource< |     public readonly preferredBackgroundLayer: UIEventSource< | ||||||
|         string | "photo" | "map" | "osmbasedmap" | undefined |         string | "photo" | "map" | "osmbasedmap" | undefined | ||||||
|  | @ -134,7 +138,7 @@ export default class UserRelatedState { | ||||||
|             return |             return | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         this.language.addCallbackAndRunD((language) => Locale.language.setData(language)) |         this.language.syncWith(Locale.language) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private static initUserRelatedState(): LayerConfig { |     private static initUserRelatedState(): LayerConfig { | ||||||
|  |  | ||||||
|  | @ -123,6 +123,14 @@ export default class Constants { | ||||||
|         "teardrop", |         "teardrop", | ||||||
|         "triangle", |         "triangle", | ||||||
|         "crosshair", |         "crosshair", | ||||||
|  |         "brick_wall_square", | ||||||
|  |         "brick_wall_round", | ||||||
|  |         "gps_arrow", | ||||||
|  |         "checkmark", | ||||||
|  |         "help", | ||||||
|  |         "clock", | ||||||
|  |         "invalid", | ||||||
|  |         "close", | ||||||
|     ] as const |     ] as const | ||||||
|     public static readonly defaultPinIcons: string[] = <any>Constants._defaultPinIcons |     public static readonly defaultPinIcons: string[] = <any>Constants._defaultPinIcons | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -9,7 +9,9 @@ import { RasterLayerProperties } from "./RasterLayerProperties" | ||||||
| export class AvailableRasterLayers { | export class AvailableRasterLayers { | ||||||
|     public static EditorLayerIndex: (Feature<Polygon, EditorLayerIndexProperties> & |     public static EditorLayerIndex: (Feature<Polygon, EditorLayerIndexProperties> & | ||||||
|         RasterLayerPolygon)[] = <any>editorlayerindex.features |         RasterLayerPolygon)[] = <any>editorlayerindex.features | ||||||
|     public static globalLayers: RasterLayerPolygon[] = globallayers.layers.map( |     public static globalLayers: RasterLayerPolygon[] = globallayers.layers | ||||||
|  |         .filter((properties) => properties.id !== "osm.carto" /*Added separately*/) | ||||||
|  |         .map( | ||||||
|             (properties) => |             (properties) => | ||||||
|                 <RasterLayerPolygon>{ |                 <RasterLayerPolygon>{ | ||||||
|                     type: "Feature", |                     type: "Feature", | ||||||
|  | @ -74,6 +76,7 @@ export class AvailableRasterLayers { | ||||||
|                     } |                     } | ||||||
|                     return GeoOperations.inside(lonlat, eliPolygon) |                     return GeoOperations.inside(lonlat, eliPolygon) | ||||||
|                 }) |                 }) | ||||||
|  |                 matching.unshift(AvailableRasterLayers.osmCarto) | ||||||
|                 matching.push(AvailableRasterLayers.maptilerDefaultLayer) |                 matching.push(AvailableRasterLayers.maptilerDefaultLayer) | ||||||
|                 matching.push(...AvailableRasterLayers.globalLayers) |                 matching.push(...AvailableRasterLayers.globalLayers) | ||||||
|                 return matching |                 return matching | ||||||
|  |  | ||||||
|  | @ -26,7 +26,7 @@ import predifined_filters from "../../../../assets/layers/filters/filters.json" | ||||||
| import { TagConfigJson } from "../Json/TagConfigJson" | import { TagConfigJson } from "../Json/TagConfigJson" | ||||||
| import PointRenderingConfigJson, { IconConfigJson } from "../Json/PointRenderingConfigJson" | import PointRenderingConfigJson, { IconConfigJson } from "../Json/PointRenderingConfigJson" | ||||||
| import ValidationUtils from "./ValidationUtils" | import ValidationUtils from "./ValidationUtils" | ||||||
| import { RenderingSpecification } from "../../../UI/SpecialVisualization" | import { RenderingSpecification, SpecialVisualization } from "../../../UI/SpecialVisualization" | ||||||
| import { QuestionableTagRenderingConfigJson } from "../Json/QuestionableTagRenderingConfigJson" | import { QuestionableTagRenderingConfigJson } from "../Json/QuestionableTagRenderingConfigJson" | ||||||
| import { ConfigMeta } from "../../../UI/Studio/configMeta" | import { ConfigMeta } from "../../../UI/Studio/configMeta" | ||||||
| import LineRenderingConfigJson from "../Json/LineRenderingConfigJson" | import LineRenderingConfigJson from "../Json/LineRenderingConfigJson" | ||||||
|  | @ -1193,6 +1193,34 @@ class ExpandMarkerRenderings extends DesugaringStep<IconConfigJson> { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export class AddRatingBadge extends DesugaringStep<LayerConfigJson> { | ||||||
|  |     constructor() { | ||||||
|  |         super( | ||||||
|  |             "Adds the 'rating'-element if a reviews-element is used in the tagRenderings", | ||||||
|  |             ["titleIcons"], | ||||||
|  |             "AddRatingBadge" | ||||||
|  |         ) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     convert(json: LayerConfigJson, context: ConversionContext): LayerConfigJson { | ||||||
|  |         if (!json.tagRenderings) { | ||||||
|  |             return json | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const specialVis: Exclude<RenderingSpecification, string>[] = < | ||||||
|  |             Exclude<RenderingSpecification, string>[] | ||||||
|  |         >ValidationUtils.getAllSpecialVisualisations(<any>json.tagRenderings).filter( | ||||||
|  |             (rs) => typeof rs !== "string" | ||||||
|  |         ) | ||||||
|  |         const funcs = new Set<string>(specialVis.map((rs) => rs.func.funcName)) | ||||||
|  | 
 | ||||||
|  |         if (funcs.has("list_reviews")) { | ||||||
|  |             ;(<(string | TagRenderingConfigJson)[]>json.titleIcons).push("icons.rating") | ||||||
|  |         } | ||||||
|  |         return json | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export class PrepareLayer extends Fuse<LayerConfigJson> { | export class PrepareLayer extends Fuse<LayerConfigJson> { | ||||||
|     constructor(state: DesugaringContext) { |     constructor(state: DesugaringContext) { | ||||||
|         super( |         super( | ||||||
|  | @ -1218,6 +1246,7 @@ export class PrepareLayer extends Fuse<LayerConfigJson> { | ||||||
|                 (layer) => new Each(new PreparePointRendering(state, layer)) |                 (layer) => new Each(new PreparePointRendering(state, layer)) | ||||||
|             ), |             ), | ||||||
|             new SetDefault("titleIcons", ["icons.defaults"]), |             new SetDefault("titleIcons", ["icons.defaults"]), | ||||||
|  |             new AddRatingBadge(), | ||||||
|             new On( |             new On( | ||||||
|                 "titleIcons", |                 "titleIcons", | ||||||
|                 (layer) => |                 (layer) => | ||||||
|  |  | ||||||
|  | @ -92,7 +92,7 @@ export class DoesImageExist extends DesugaringStep<string> { | ||||||
|             return image |             return image | ||||||
|         } |         } | ||||||
|         if (image.match(/[a-z]*/)) { |         if (image.match(/[a-z]*/)) { | ||||||
|             if (Svg.All[image + ".svg"] !== undefined) { |             if (Constants.defaultPinIcons.indexOf(image) >= 0) { | ||||||
|                 // This is a builtin img, e.g. 'checkmark' or 'crosshair'
 |                 // This is a builtin img, e.g. 'checkmark' or 'crosshair'
 | ||||||
|                 return image |                 return image | ||||||
|             } |             } | ||||||
|  | @ -110,7 +110,7 @@ export class DoesImageExist extends DesugaringStep<string> { | ||||||
|                 ) |                 ) | ||||||
|             } else if (!this.doesPathExist(image)) { |             } else if (!this.doesPathExist(image)) { | ||||||
|                 context.err( |                 context.err( | ||||||
|                     `Image with path ${image} does not exist; it is used in ${context}.\n     Check for typo's and missing directories in the path.` |                     `Image with path ${image} does not exist.\n     Check for typo's and missing directories in the path.` | ||||||
|                 ) |                 ) | ||||||
|             } else { |             } else { | ||||||
|                 context.err( |                 context.err( | ||||||
|  | @ -813,6 +813,12 @@ class MiscTagRenderingChecks extends DesugaringStep<TagRenderingConfigJson> { | ||||||
|             ) |             ) | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         if (Object.keys(json).length === 1 && typeof json["render"] === "string") { | ||||||
|  |             context.warn( | ||||||
|  |                 `use the content directly instead of {render: ${JSON.stringify(json["render"])}}` | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         { |         { | ||||||
|             for (const key of ["question", "questionHint", "render"]) { |             for (const key of ["question", "questionHint", "render"]) { | ||||||
|                 CheckTranslation.allowUndefined.convert(json[key], context.enter(key)) |                 CheckTranslation.allowUndefined.convert(json[key], context.enter(key)) | ||||||
|  |  | ||||||
|  | @ -9,6 +9,12 @@ export default class ValidationUtils { | ||||||
|         renderingConfigs: (TagRenderingConfigJson | QuestionableTagRenderingConfigJson)[] |         renderingConfigs: (TagRenderingConfigJson | QuestionableTagRenderingConfigJson)[] | ||||||
|     ): RenderingSpecification[] { |     ): RenderingSpecification[] { | ||||||
|         const visualisations: RenderingSpecification[] = [] |         const visualisations: RenderingSpecification[] = [] | ||||||
|  |         if (!Array.isArray(renderingConfigs)) { | ||||||
|  |             throw ( | ||||||
|  |                 "Could not inspect renderingConfigs, not an array: " + | ||||||
|  |                 JSON.stringify(renderingConfigs) | ||||||
|  |             ) | ||||||
|  |         } | ||||||
|         for (const renderConfig of renderingConfigs) { |         for (const renderConfig of renderingConfigs) { | ||||||
|             visualisations.push(...ValidationUtils.getSpecialVisualisationsWithArgs(renderConfig)) |             visualisations.push(...ValidationUtils.getSpecialVisualisationsWithArgs(renderConfig)) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -3,12 +3,10 @@ import TagRenderingConfig from "./TagRenderingConfig" | ||||||
| import { TagsFilter } from "../../Logic/Tags/TagsFilter" | import { TagsFilter } from "../../Logic/Tags/TagsFilter" | ||||||
| import { TagUtils } from "../../Logic/Tags/TagUtils" | import { TagUtils } from "../../Logic/Tags/TagUtils" | ||||||
| import { Utils } from "../../Utils" | import { Utils } from "../../Utils" | ||||||
| import Svg from "../../Svg" |  | ||||||
| import WithContextLoader from "./WithContextLoader" | import WithContextLoader from "./WithContextLoader" | ||||||
| import { ImmutableStore, Store } from "../../Logic/UIEventSource" | import { ImmutableStore, Store } from "../../Logic/UIEventSource" | ||||||
| import BaseUIElement from "../../UI/BaseUIElement" | import BaseUIElement from "../../UI/BaseUIElement" | ||||||
| import { FixedUiElement } from "../../UI/Base/FixedUiElement" | import { FixedUiElement } from "../../UI/Base/FixedUiElement" | ||||||
| import Img from "../../UI/Base/Img" |  | ||||||
| import Combine from "../../UI/Base/Combine" | import Combine from "../../UI/Base/Combine" | ||||||
| import { VariableUiElement } from "../../UI/Base/VariableUIElement" | import { VariableUiElement } from "../../UI/Base/VariableUIElement" | ||||||
| import { TagRenderingConfigJson } from "./Json/TagRenderingConfigJson" | import { TagRenderingConfigJson } from "./Json/TagRenderingConfigJson" | ||||||
|  | @ -133,71 +131,23 @@ export default class PointRenderingConfig extends WithContextLoader { | ||||||
|             context + ".rotationAlignment" |             context + ".rotationAlignment" | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
| 
 |     private static FromHtmlMulti(multiSpec: string, tags: Store<Record<string, string>>) { | ||||||
|     /** |         const icons: IconConfig[] = [] | ||||||
|      * Given a single HTML spec (either a single image path OR "image_path_to_known_svg:fill-colour", returns a fixedUIElement containing that |         for (const subspec of multiSpec.split(";")) { | ||||||
|      * The element will fill 100% and be positioned absolutely with top:0 and left: 0 |             const [icon, color] = subspec.split(":") | ||||||
|      */ |             icons.push(new IconConfig({ icon, color })) | ||||||
|     private static FromHtmlSpec(htmlSpec: string, style: string, isBadge = false): BaseUIElement { |  | ||||||
|         if (htmlSpec === undefined) { |  | ||||||
|             return undefined |  | ||||||
|         } |         } | ||||||
|         const match = htmlSpec.match(/([a-zA-Z0-9_]*):([^;]*)/) |         return new SvelteUIElement(DynamicMarker, { marker: icons, tags }).SetClass( | ||||||
|         if (match !== null && Svg.All[match[1] + ".svg"] !== undefined) { |             "w-full h-full block absolute top-0 left-0" | ||||||
|             const svg = Svg.All[match[1] + ".svg"] as string |  | ||||||
|             const targetColor = match[2] |  | ||||||
|             const img = new Img( |  | ||||||
|                 svg.replace(/(rgb\(0%,0%,0%\)|#000000|#000)/g, targetColor), |  | ||||||
|                 true |  | ||||||
|             ).SetStyle(style) |  | ||||||
|             if (isBadge) { |  | ||||||
|                 img.SetClass("badge") |  | ||||||
|             } |  | ||||||
|             return img |  | ||||||
|         } else if (Svg.All[htmlSpec + ".svg"] !== undefined) { |  | ||||||
|             const svg = Svg.All[htmlSpec + ".svg"] as string |  | ||||||
|             const img = new Img(svg, true).SetStyle(style) |  | ||||||
|             if (isBadge) { |  | ||||||
|                 img.SetClass("badge") |  | ||||||
|             } |  | ||||||
|             return img |  | ||||||
|         } else { |  | ||||||
|             return new FixedUiElement(`<img src="${htmlSpec}" style="${style}" />`) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static FromHtmlMulti( |  | ||||||
|         multiSpec: string, |  | ||||||
|         rotation: string, |  | ||||||
|         isBadge: boolean, |  | ||||||
|         defaultElement: BaseUIElement = undefined, |  | ||||||
|         options?: { |  | ||||||
|             noFullWidth?: boolean |  | ||||||
|         } |  | ||||||
|     ) { |  | ||||||
|         if (multiSpec === undefined) { |  | ||||||
|             return defaultElement |  | ||||||
|         } |  | ||||||
|         const style = `width:100%;height:100%;transform: rotate( ${rotation} );display:block;position: absolute; top: 0; left: 0` |  | ||||||
| 
 |  | ||||||
|         const htmlDefs = multiSpec.trim()?.split(";") ?? [] |  | ||||||
|         const elements = Utils.NoEmpty(htmlDefs).map((def) => |  | ||||||
|             PointRenderingConfig.FromHtmlSpec(def, style, isBadge) |  | ||||||
|         ) |         ) | ||||||
|         if (elements.length === 0) { |  | ||||||
|             return defaultElement |  | ||||||
|         } else { |  | ||||||
|             const combine = new Combine(elements).SetClass("relative block") |  | ||||||
|             if (options?.noFullWidth) { |  | ||||||
|                 return combine |  | ||||||
|             } |  | ||||||
|             combine.SetClass("w-full h-full") |  | ||||||
|             return combine |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public GetBaseIcon(tags?: Record<string, string>): BaseUIElement { |     public GetBaseIcon(tags?: Record<string, string>): BaseUIElement { | ||||||
|         return new SvelteUIElement(DynamicMarker, { config: this, tags: new ImmutableStore(tags) }) |         return new SvelteUIElement(DynamicMarker, { | ||||||
|  |             marker: this.marker, | ||||||
|  |             rotation: this.rotation, | ||||||
|  |             tags: new ImmutableStore(tags), | ||||||
|  |         }) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public RenderIcon( |     public RenderIcon( | ||||||
|  | @ -249,9 +199,11 @@ export default class PointRenderingConfig extends WithContextLoader { | ||||||
|             anchorH = -iconH / 2 |             anchorH = -iconH / 2 | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         const icon = new SvelteUIElement(DynamicMarker, { config: this, tags }).SetClass( |         const icon = new SvelteUIElement(DynamicMarker, { | ||||||
|             "w-full h-full" |             marker: this.marker, | ||||||
|         ) |             rotation: this.rotation, | ||||||
|  |             tags, | ||||||
|  |         }).SetClass("w-full h-full") | ||||||
|         let badges = undefined |         let badges = undefined | ||||||
|         if (options?.includeBadges ?? true) { |         if (options?.includeBadges ?? true) { | ||||||
|             badges = this.GetBadges(tags, options?.metatags) |             badges = this.GetBadges(tags, options?.metatags) | ||||||
|  | @ -306,9 +258,9 @@ export default class PointRenderingConfig extends WithContextLoader { | ||||||
|         } |         } | ||||||
|         return new VariableUiElement( |         return new VariableUiElement( | ||||||
|             tags.map( |             tags.map( | ||||||
|                 (tags) => { |                 (tagsData) => { | ||||||
|                     const badgeElements = this.iconBadges.map((badge) => { |                     const badgeElements = this.iconBadges.map((badge) => { | ||||||
|                         if (!badge.if.matchesProperties(tags)) { |                         if (!badge.if.matchesProperties(tagsData)) { | ||||||
|                             // Doesn't match...
 |                             // Doesn't match...
 | ||||||
|                             return undefined |                             return undefined | ||||||
|                         } |                         } | ||||||
|  | @ -323,19 +275,23 @@ export default class PointRenderingConfig extends WithContextLoader { | ||||||
|                         } |                         } | ||||||
| 
 | 
 | ||||||
|                         const htmlDefs = Utils.SubstituteKeys( |                         const htmlDefs = Utils.SubstituteKeys( | ||||||
|                             badge.then.GetRenderValue(tags)?.txt, |                             badge.then.GetRenderValue(tagsData)?.txt, | ||||||
|                             tags |                             tagsData | ||||||
|                         ) |                         ) | ||||||
|                         if (htmlDefs.startsWith("<") && htmlDefs.endsWith(">")) { |                         if (htmlDefs.startsWith("<") && htmlDefs.endsWith(">")) { | ||||||
|                             // This is probably an HTML-element
 |                             // This is probably an HTML-element
 | ||||||
|                             return new FixedUiElement(Utils.SubstituteKeys(htmlDefs, tags)) |                             return new FixedUiElement(Utils.SubstituteKeys(htmlDefs, tagsData)) | ||||||
|                                 .SetStyle("width: 1.5rem") |                                 .SetStyle("width: 1.5rem") | ||||||
|                                 .SetClass("block") |                                 .SetClass("block") | ||||||
|                         } |                         } | ||||||
|  | 
 | ||||||
|  |                         if (!htmlDefs) { | ||||||
|  |                             return undefined | ||||||
|  |                         } | ||||||
|  | 
 | ||||||
|                         const badgeElement = PointRenderingConfig.FromHtmlMulti( |                         const badgeElement = PointRenderingConfig.FromHtmlMulti( | ||||||
|                             htmlDefs, |                             htmlDefs, | ||||||
|                             "0", |                             tags | ||||||
|                             true |  | ||||||
|                         )?.SetClass("block relative") |                         )?.SetClass("block relative") | ||||||
|                         if (badgeElement === undefined) { |                         if (badgeElement === undefined) { | ||||||
|                             return undefined |                             return undefined | ||||||
|  |  | ||||||
|  | @ -19,6 +19,7 @@ import { Paragraph } from "../../UI/Base/Paragraph" | ||||||
| import Svg from "../../Svg" | import Svg from "../../Svg" | ||||||
| import Validators, { ValidatorType } from "../../UI/InputElement/Validators" | import Validators, { ValidatorType } from "../../UI/InputElement/Validators" | ||||||
| import { TagRenderingConfigJson } from "./Json/TagRenderingConfigJson" | import { TagRenderingConfigJson } from "./Json/TagRenderingConfigJson" | ||||||
|  | import Constants from "../Constants" | ||||||
| 
 | 
 | ||||||
| export interface Icon {} | export interface Icon {} | ||||||
| 
 | 
 | ||||||
|  | @ -362,7 +363,7 @@ export default class TagRenderingConfig { | ||||||
|                 if (stripped.endsWith(".svg")) { |                 if (stripped.endsWith(".svg")) { | ||||||
|                     stripped = stripped.substring(0, stripped.length - 4) |                     stripped = stripped.substring(0, stripped.length - 4) | ||||||
|                 } |                 } | ||||||
|                 if (Svg.All[stripped + ".svg"] !== undefined) { |                 if (Constants.defaultPinIcons.indexOf(stripped) >= 0) { | ||||||
|                     icon = "./assets/svg/" + mapping.icon |                     icon = "./assets/svg/" + mapping.icon | ||||||
|                     if (!icon.endsWith(".svg")) { |                     if (!icon.endsWith(".svg")) { | ||||||
|                         icon += ".svg" |                         icon += ".svg" | ||||||
|  |  | ||||||
							
								
								
									
										74
									
								
								src/UI/AllThemesGui.svelte
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								src/UI/AllThemesGui.svelte
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,74 @@ | ||||||
|  | <script lang="ts"> | ||||||
|  |   import { OsmConnectionFeatureSwitches } from "../Logic/State/FeatureSwitchState"; | ||||||
|  |   import { OsmConnection } from "../Logic/Osm/OsmConnection"; | ||||||
|  |   import { QueryParameters } from "../Logic/Web/QueryParameters"; | ||||||
|  |   import UserRelatedState from "../Logic/State/UserRelatedState"; | ||||||
|  |   import LanguagePicker from "./InputElement/LanguagePicker.svelte"; | ||||||
|  |   import Translations from "./i18n/Translations"; | ||||||
|  |   import Logo from "../assets/svg/Logo.svelte"; | ||||||
|  |   import Tr from "./Base/Tr.svelte"; | ||||||
|  |   import ToSvelte from "./Base/ToSvelte.svelte"; | ||||||
|  |   import MoreScreen from "./BigComponents/MoreScreen"; | ||||||
|  |   import LoginToggle from "./Base/LoginToggle.svelte"; | ||||||
|  |   import Pencil from "../assets/svg/Pencil.svelte"; | ||||||
|  |   import Login from "../assets/svg/Login.svelte"; | ||||||
|  |   import Constants from "../Models/Constants"; | ||||||
|  | 
 | ||||||
|  |   const featureSwitches = new OsmConnectionFeatureSwitches(); | ||||||
|  |   const osmConnection = new OsmConnection({ | ||||||
|  |     fakeUser: featureSwitches.featureSwitchFakeUser.data, | ||||||
|  |     oauth_token: QueryParameters.GetQueryParameter( | ||||||
|  |       "oauth_token", | ||||||
|  |       undefined, | ||||||
|  |       "Used to complete the login" | ||||||
|  |     ) | ||||||
|  |   }); | ||||||
|  |   const state = new UserRelatedState(osmConnection); | ||||||
|  |   const t = Translations.t.index; | ||||||
|  |   let userLanguages = osmConnection.userDetails.map(ud => ud.languages); | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <div class="flex flex-col m-4"> | ||||||
|  |   <div class="self-end"> | ||||||
|  |     <LanguagePicker assignTo={state.language} availableLanguages={t.title.SupportedLanguages()} | ||||||
|  |                     preferredLanguages={userLanguages} /> | ||||||
|  |   </div> | ||||||
|  | 
 | ||||||
|  |   <div class="flex mt-4"> | ||||||
|  | 
 | ||||||
|  |     <div class="flex-none m-3"> | ||||||
|  |       <Logo alt="MapComplete Logo" class="w-12 h-12 sm:h-24 sm:w-24" /> | ||||||
|  |     </div> | ||||||
|  | 
 | ||||||
|  |     <div class="flex flex-col"> | ||||||
|  |       <h1 class="tracking-tight font-extrabold md:text-6xl m-0"> | ||||||
|  |         <Tr t={t.title} /> | ||||||
|  |       </h1> | ||||||
|  |       <Tr cls="my-4 mr-4 text-base font-semibold sm:text-lg md:mt-5 md:text-xl lg:mx-0" | ||||||
|  |           t={Translations.t.index.intro} /> | ||||||
|  | 
 | ||||||
|  |     </div> | ||||||
|  | 
 | ||||||
|  |   </div> | ||||||
|  | 
 | ||||||
|  |   <ToSvelte construct={new MoreScreen(state, true)} /> | ||||||
|  | 
 | ||||||
|  |   <LoginToggle state={state}> | ||||||
|  |     <div slot="not-logged-in"> | ||||||
|  |       <button class="w-full" on:click={() => osmConnection.AttemptLogin()}> | ||||||
|  |         <Login class="w-6 h-6 mr-2 "/> | ||||||
|  |         <Tr t={Translations.t.index.logIn} /> | ||||||
|  |       </button> | ||||||
|  |     </div> | ||||||
|  |     <a class="w-full h-fit button" href={window.location.protocol + "//" + window.location.host + "/studio.html"}> | ||||||
|  |       <Pencil class="w-6 h-6 mr-2" /> | ||||||
|  |       <Tr t={ Translations.t.general.morescreen.createYourOwnTheme} /> | ||||||
|  |     </a> | ||||||
|  |   </LoginToggle> | ||||||
|  |    | ||||||
|  |   <Tr cls="link-underline" t={Translations.t.general.aboutMapComplete.intro}/> | ||||||
|  |   <div class="subtle self-end mb-16"> | ||||||
|  |     v{Constants.vNumber} | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
|  | 
 | ||||||
|  | @ -1,69 +0,0 @@ | ||||||
| import UserRelatedState from "../Logic/State/UserRelatedState" |  | ||||||
| import { FixedUiElement } from "./Base/FixedUiElement" |  | ||||||
| import Combine from "./Base/Combine" |  | ||||||
| import MoreScreen from "./BigComponents/MoreScreen" |  | ||||||
| import Translations from "./i18n/Translations" |  | ||||||
| import Constants from "../Models/Constants" |  | ||||||
| import LanguagePicker from "./LanguagePicker" |  | ||||||
| import IndexText from "./BigComponents/IndexText" |  | ||||||
| import { LoginToggle } from "./Popup/LoginButton" |  | ||||||
| import { ImmutableStore } from "../Logic/UIEventSource" |  | ||||||
| import { OsmConnection } from "../Logic/Osm/OsmConnection" |  | ||||||
| import { QueryParameters } from "../Logic/Web/QueryParameters" |  | ||||||
| import { OsmConnectionFeatureSwitches } from "../Logic/State/FeatureSwitchState" |  | ||||||
| import { SubtleButton } from "./Base/SubtleButton" |  | ||||||
| import Svg from "../Svg" |  | ||||||
| import Link from "./Base/Link" |  | ||||||
| 
 |  | ||||||
| export default class AllThemesGui { |  | ||||||
|     setup() { |  | ||||||
|         try { |  | ||||||
|             const featureSwitches = new OsmConnectionFeatureSwitches() |  | ||||||
|             const osmConnection = new OsmConnection({ |  | ||||||
|                 fakeUser: featureSwitches.featureSwitchFakeUser.data, |  | ||||||
|                 oauth_token: QueryParameters.GetQueryParameter( |  | ||||||
|                     "oauth_token", |  | ||||||
|                     undefined, |  | ||||||
|                     "Used to complete the login" |  | ||||||
|                 ), |  | ||||||
|             }) |  | ||||||
|             const state = new UserRelatedState(osmConnection) |  | ||||||
|             const intro = new Combine([ |  | ||||||
|                 new LanguagePicker( |  | ||||||
|                     Translations.t.index.title.SupportedLanguages(), |  | ||||||
|                     state.language |  | ||||||
|                 ).SetClass("flex absolute top-2 right-3"), |  | ||||||
|                 new IndexText(), |  | ||||||
|             ]) |  | ||||||
|             new Combine([ |  | ||||||
|                 intro, |  | ||||||
|                 new MoreScreen(state, true), |  | ||||||
|                 new LoginToggle( |  | ||||||
|                     new Link( |  | ||||||
|                         new Combine([ |  | ||||||
|                             Svg.pencil_svg().SetClass("w-6 h-6 mr-2"), |  | ||||||
|                             Translations.t.general.morescreen.createYourOwnTheme, |  | ||||||
|                         ]).SetClass("flex p-2"), |  | ||||||
|                         window.location.protocol + "//" + window.location.host + "/studio.html" |  | ||||||
|                     ).SetClass("w-full h-fit button"), |  | ||||||
|                     Translations.t.index.logIn, |  | ||||||
|                     { |  | ||||||
|                         osmConnection, |  | ||||||
|                         featureSwitchUserbadge: new ImmutableStore(true), |  | ||||||
|                     } |  | ||||||
|                 ).SetClass("flex justify-center w-full"), |  | ||||||
|                 Translations.t.general.aboutMapComplete.intro.SetClass("link-underline"), |  | ||||||
|                 new FixedUiElement("v" + Constants.vNumber).SetClass("block"), |  | ||||||
|             ]) |  | ||||||
|                 .SetClass("block m-5 lg:w-3/4 lg:ml-40") |  | ||||||
|                 .AttachTo("main") |  | ||||||
|         } catch (e) { |  | ||||||
|             console.error(">>>> CRITICAL", e) |  | ||||||
|             new FixedUiElement( |  | ||||||
|                 "Seems like no layers are compiled - check the output of `npm run generate:layeroverview`. Is this visible online? Contact pietervdvn immediately!" |  | ||||||
|             ) |  | ||||||
|                 .SetClass("alert") |  | ||||||
|                 .AttachTo("main") |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -5,6 +5,7 @@ | ||||||
|    */ |    */ | ||||||
|   import { Store } from "../../Logic/UIEventSource" |   import { Store } from "../../Logic/UIEventSource" | ||||||
|   import { onDestroy } from "svelte" |   import { onDestroy } from "svelte" | ||||||
|  |   import Hand from "../../assets/svg/Hand.svelte"; | ||||||
| 
 | 
 | ||||||
|   let mainElem: HTMLElement |   let mainElem: HTMLElement | ||||||
|   export let hideSignal: Store<any> |   export let hideSignal: Store<any> | ||||||
|  | @ -34,7 +35,7 @@ | ||||||
| 
 | 
 | ||||||
| <div bind:this={mainElem} class="pointer-events-none absolute bottom-0 right-0 h-full w-full"> | <div bind:this={mainElem} class="pointer-events-none absolute bottom-0 right-0 h-full w-full"> | ||||||
|   <div id="hand-container"> |   <div id="hand-container"> | ||||||
|     <img src="./assets/svg/hand.svg" /> |     <Hand /> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,14 +1,31 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import { UIEventSource } from "../../Logic/UIEventSource.js" |   import { UIEventSource } from "../../Logic/UIEventSource.js"; | ||||||
| 
 | 
 | ||||||
|   /** |   export let value: UIEventSource<any> | ||||||
|    * For some stupid reason, it is very hard to bind inputs |   let i: any = value.data | ||||||
|    */ |   let htmlElement : HTMLSelectElement | ||||||
|   export let value: UIEventSource<number> |   function selectAppropriateValue(){ | ||||||
|   let i: number = value.data |     if(!htmlElement){ | ||||||
|   $: value.setData(i) |       return; | ||||||
|  |     } | ||||||
|  |     const v = value.data | ||||||
|  |     for (let option of htmlElement.getElementsByTagName("option")) { | ||||||
|  |       if(option.value === v){ | ||||||
|  |         option.selected = true | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |    | ||||||
|  |   value.addCallbackD(() => selectAppropriateValue()) | ||||||
|  |   $: { | ||||||
|  |     if(htmlElement){ | ||||||
|  |       selectAppropriateValue() | ||||||
|  |     } | ||||||
|  |   } | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <select bind:value={i}> | <select bind:this={htmlElement} on:change={(e) => {value.setData(e.srcElement.value)}}> | ||||||
|   <slot /> |   <slot /> | ||||||
| </select> | </select> | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ | ||||||
|   import Tr from "./Tr.svelte" |   import Tr from "./Tr.svelte" | ||||||
|   import { OsmConnection } from "../../Logic/Osm/OsmConnection" |   import { OsmConnection } from "../../Logic/Osm/OsmConnection" | ||||||
|   import { ImmutableStore, UIEventSource } from "../../Logic/UIEventSource" |   import { ImmutableStore, UIEventSource } from "../../Logic/UIEventSource" | ||||||
|  |   import Invalid from "../../assets/svg/Invalid.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let state: { |   export let state: { | ||||||
|     osmConnection: OsmConnection |     osmConnection: OsmConnection | ||||||
|  | @ -35,7 +36,7 @@ | ||||||
|     </slot> |     </slot> | ||||||
|   {:else if $loadingStatus === "error"} |   {:else if $loadingStatus === "error"} | ||||||
|     <div class="alert max-w-64 flex items-center"> |     <div class="alert max-w-64 flex items-center"> | ||||||
|       <img src="./assets/svg/invalid.svg" class="m-2 h-8 w-8 shrink-0" /> |       <Invalid class="m-2 h-8 w-8 shrink-0" /> | ||||||
|       <Tr t={offlineModes[$apiState]} /> |       <Tr t={offlineModes[$apiState]} /> | ||||||
|     </div> |     </div> | ||||||
|   {:else if $loadingStatus === "logged-in"} |   {:else if $loadingStatus === "logged-in"} | ||||||
|  |  | ||||||
							
								
								
									
										15
									
								
								src/UI/Base/LogoutButton.svelte
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/UI/Base/LogoutButton.svelte
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,15 @@ | ||||||
|  | <script lang="ts"> | ||||||
|  |   import { OsmConnection } from "../../Logic/Osm/OsmConnection"; | ||||||
|  |   import Logout from "../../assets/svg/Logout.svelte"; | ||||||
|  |   import Translations from "../i18n/Translations"; | ||||||
|  |   import Tr from "./Tr.svelte"; | ||||||
|  | 
 | ||||||
|  |   export let osmConnection: OsmConnection | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <button on:click={() => { | ||||||
|  |                         state.osmConnection.LogOut() | ||||||
|  |                     }}> | ||||||
|  |   <Logout class="w-6 h-6"/> | ||||||
|  |   <Tr t={Translations.t.general.logout}/> | ||||||
|  | </button> | ||||||
							
								
								
									
										48
									
								
								src/UI/Base/OpenJosm.svelte
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								src/UI/Base/OpenJosm.svelte
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,48 @@ | ||||||
|  | <script lang="ts"> | ||||||
|  |   import { UIEventSource } from "../../Logic/UIEventSource"; | ||||||
|  |   import Translations from "../i18n/Translations"; | ||||||
|  |   import Tr from "./Tr.svelte"; | ||||||
|  |   import Josm_logo from "../../assets/svg/Josm_logo.svelte"; | ||||||
|  |   import Constants from "../../Models/Constants"; | ||||||
|  |   import type { SpecialVisualizationState } from "../SpecialVisualization"; | ||||||
|  | 
 | ||||||
|  |   export let state : SpecialVisualizationState | ||||||
|  |   const t = Translations.t.general.attribution; | ||||||
|  |   const josmState = new UIEventSource<"OK" | string>(undefined); | ||||||
|  |   // Reset after 15s | ||||||
|  |   josmState.stabilized(15000).addCallbackD(() => josmState.setData(undefined)); | ||||||
|  | 
 | ||||||
|  |   const showButton = state.osmConnection.userDetails.map( | ||||||
|  |     (ud) => ud.loggedIn && ud.csCount >= Constants.userJourney.historyLinkVisible | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   function openJosm() { | ||||||
|  |     const bbox = state.mapProperties. bounds.data; | ||||||
|  |     if (bbox === undefined) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     const top = bbox.getNorth(); | ||||||
|  |     const bottom = bbox.getSouth(); | ||||||
|  |     const right = bbox.getEast(); | ||||||
|  |     const left = bbox.getWest(); | ||||||
|  |     const josmLink = `http://127.0.0.1:8111/load_and_zoom?left=${left}&right=${right}&top=${top}&bottom=${bottom}`; | ||||||
|  |     Utils.download(josmLink) | ||||||
|  |       .then((answer) => josmState.setData(answer.replace(/\n/g, "").trim())) | ||||||
|  |       .catch(() => josmState.setData("ERROR")); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | </script> | ||||||
|  | {#if $showButton} | ||||||
|  |   {#if $josmState === undefined} | ||||||
|  |     <!-- empty --> | ||||||
|  |   {:else if state === "OK"} | ||||||
|  |     <Tr cls="thanks" t={t.josmOpened} /> | ||||||
|  |   {:else} | ||||||
|  |     <Tr cls="alert" t={t.josmNotOpened} /> | ||||||
|  |   {/if} | ||||||
|  | 
 | ||||||
|  |   <button class="flex items-center" on:click={openJosm}> | ||||||
|  |     <Josm_logo class="h-12 w-12 p-2 pr-4" /> | ||||||
|  |     <Tr t={t.editJosm} /> | ||||||
|  |   </button> | ||||||
|  | {/if} | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import ToSvelte from "./ToSvelte.svelte" |   import ToSvelte from "./ToSvelte.svelte" | ||||||
|   import Svg from "../../Svg" |   import Svg from "../../Svg" | ||||||
|  |   import Share from "../../assets/svg/Share.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let generateShareData: () => { |   export let generateShareData: () => { | ||||||
|     text: string |     text: string | ||||||
|  | @ -25,6 +26,6 @@ | ||||||
| 
 | 
 | ||||||
| <button on:click={share} class="secondary m-0 h-8 w-8 p-0"> | <button on:click={share} class="secondary m-0 h-8 w-8 p-0"> | ||||||
|   <slot name="content"> |   <slot name="content"> | ||||||
|     <ToSvelte construct={Svg.share_svg().SetClass("w-7 h-7 p-1")} /> |     <Share class="w-7 h-7 p-1"/> | ||||||
|   </slot> |   </slot> | ||||||
| </button> | </button> | ||||||
|  |  | ||||||
|  | @ -23,10 +23,10 @@ export default class SvelteUIElement< | ||||||
|     private readonly _events: Events |     private readonly _events: Events | ||||||
|     private readonly _slots: Slots |     private readonly _slots: Slots | ||||||
| 
 | 
 | ||||||
|     constructor(svelteElement, props: Props, events?: Events, slots?: Slots) { |     constructor(svelteElement, props?: Props, events?: Events, slots?: Slots) { | ||||||
|         super() |         super() | ||||||
|         this._svelteComponent = svelteElement |         this._svelteComponent = svelteElement | ||||||
|         this._props = props |         this._props = props ?? <Props>{} | ||||||
|         this._events = events |         this._events = events | ||||||
|         this._slots = slots |         this._slots = slots | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -1,6 +1,7 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import Locale from "../i18n/Locale" |   import Locale from "../i18n/Locale" | ||||||
|   import LinkToWeblate from "./LinkToWeblate" |   import LinkToWeblate from "./LinkToWeblate" | ||||||
|  |   import Translate from "../../assets/svg/Translate.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * Shows a small icon which will open up weblate; a contributor can translate the item for 'context' there |    * Shows a small icon which will open up weblate; a contributor can translate the item for 'context' there | ||||||
|  | @ -19,7 +20,7 @@ | ||||||
|       target="_blank" |       target="_blank" | ||||||
|       class="weblate-link mx-1" |       class="weblate-link mx-1" | ||||||
|     > |     > | ||||||
|       <img src="./assets/svg/translate.svg" class="font-gray" /> |       <Translate class="font-gray" /> | ||||||
|     </a> |     </a> | ||||||
|   {:else if $linkToWeblate} |   {:else if $linkToWeblate} | ||||||
|     <a |     <a | ||||||
|  | @ -27,7 +28,7 @@ | ||||||
|       class="weblate-link hidden-on-mobile mx-1" |       class="weblate-link hidden-on-mobile mx-1" | ||||||
|       target="_blank" |       target="_blank" | ||||||
|     > |     > | ||||||
|       <img src="./assets/svg/translate.svg" class="font-gray inline-block" /> |       <Translate class="font-gray inline-block" /> | ||||||
|     </a> |     </a> | ||||||
|   {/if} |   {/if} | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
|  | @ -4,25 +4,7 @@ import { FixedUiElement } from "../Base/FixedUiElement" | ||||||
| 
 | 
 | ||||||
| export default class IndexText extends Combine { | export default class IndexText extends Combine { | ||||||
|     constructor() { |     constructor() { | ||||||
|         super([ |         super([]) | ||||||
|             new FixedUiElement( |  | ||||||
|                 `<img class="w-12 h-12 sm:h-24 sm:w-24" src="./assets/svg/logo.svg" alt="MapComplete Logo">` |  | ||||||
|             ).SetClass("flex-none m-3"), |  | ||||||
| 
 |  | ||||||
|             new Combine([ |  | ||||||
|                 Translations.t.index.title.SetClass( |  | ||||||
|                     "text-2xl tracking-tight font-extrabold text-gray-900 sm:text-5xl md:text-6xl block text-gray-800 xl:inline" |  | ||||||
|                 ), |  | ||||||
| 
 |  | ||||||
|                 Translations.t.index.intro.SetClass( |  | ||||||
|                     "mt-3 text-base font-semibold text-gray-500 sm:mt-5 sm:text-lg sm:max-w-xl sm:mx-auto md:mt-5 md:text-xl lg:mx-0" |  | ||||||
|                 ), |  | ||||||
| 
 |  | ||||||
|                 Translations.t.index.pickTheme.SetClass( |  | ||||||
|                     "mt-3 text-base sm:mt-5 sm:text-lg sm:max-w-xl sm:mx-auto md:mt-5 md:text-xl lg:mx-0" |  | ||||||
|                 ), |  | ||||||
|             ]).SetClass("flex flex-col sm:text-center lg:text-left m-1 mt-2 md:m-2 md:mt-4"), |  | ||||||
|         ]) |  | ||||||
| 
 | 
 | ||||||
|         this.SetClass("flex flex-row") |         this.SetClass("flex flex-row") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -4,6 +4,7 @@ | ||||||
|   import { Store } from "../../Logic/UIEventSource" |   import { Store } from "../../Logic/UIEventSource" | ||||||
|   import Tr from "../Base/Tr.svelte" |   import Tr from "../Base/Tr.svelte" | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import ToSvelte from "../Base/ToSvelte.svelte" | ||||||
|  |   import Mapillary_black from "../../assets/svg/Mapillary_black.svelte"; | ||||||
| 
 | 
 | ||||||
|   /* |   /* | ||||||
|     A subtleButton which opens mapillary in a new tab at the current location |     A subtleButton which opens mapillary in a new tab at the current location | ||||||
|  | @ -21,7 +22,7 @@ | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <a class="button flex items-center" href={mapillaryLink} target="_blank"> | <a class="button flex items-center" href={mapillaryLink} target="_blank"> | ||||||
|   <ToSvelte construct={() => Svg.mapillary_black_svg().SetClass("w-12 h-12 m-2 mr-4 shrink-0")} /> |   <Mapillary_black class="w-12 h-12 m-2 mr-4 shrink-0"/> | ||||||
|   <div class="flex flex-col"> |   <div class="flex flex-col"> | ||||||
|     <Tr t={Translations.t.general.attribution.openMapillary} /> |     <Tr t={Translations.t.general.attribution.openMapillary} /> | ||||||
|     <Tr cls="subtle" t={Translations.t.general.attribution.mapillaryHelp} /> |     <Tr cls="subtle" t={Translations.t.general.attribution.mapillaryHelp} /> | ||||||
|  |  | ||||||
|  | @ -16,6 +16,7 @@ | ||||||
|   import LayerConfig from "../../Models/ThemeConfig/LayerConfig" |   import LayerConfig from "../../Models/ThemeConfig/LayerConfig" | ||||||
|   import { Utils } from "../../Utils" |   import { Utils } from "../../Utils" | ||||||
|   import { createEventDispatcher } from "svelte" |   import { createEventDispatcher } from "svelte" | ||||||
|  |   import Move_arrows from "../../assets/svg/Move_arrows.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * An advanced location input, which has support to: |    * An advanced location input, which has support to: | ||||||
|  | @ -125,6 +126,6 @@ | ||||||
|   maxDistanceInMeters="50" |   maxDistanceInMeters="50" | ||||||
| > | > | ||||||
|   <slot name="image" slot="image"> |   <slot name="image" slot="image"> | ||||||
|     <img class="h-full max-h-24" src="./assets/svg/move-arrows.svg" /> |     <Move_arrows  class="h-full max-h-24" /> | ||||||
|   </slot> |   </slot> | ||||||
| </LocationInput> | </LocationInput> | ||||||
|  |  | ||||||
|  | @ -1,60 +0,0 @@ | ||||||
| import Combine from "../Base/Combine" |  | ||||||
| import { OsmConnection } from "../../Logic/Osm/OsmConnection" |  | ||||||
| import { Store, UIEventSource } from "../../Logic/UIEventSource" |  | ||||||
| import { BBox } from "../../Logic/BBox" |  | ||||||
| import Translations from "../i18n/Translations" |  | ||||||
| import { VariableUiElement } from "../Base/VariableUIElement" |  | ||||||
| import Toggle from "../Input/Toggle" |  | ||||||
| import { SubtleButton } from "../Base/SubtleButton" |  | ||||||
| import Svg from "../../Svg" |  | ||||||
| import { Utils } from "../../Utils" |  | ||||||
| import Constants from "../../Models/Constants" |  | ||||||
| 
 |  | ||||||
| export class OpenJosm extends Combine { |  | ||||||
|     public static readonly needsUrls = ["http://127.0.0.1:8111/load_and_zoom"] |  | ||||||
|     constructor(osmConnection: OsmConnection, bounds: Store<BBox>, iconStyle?: string) { |  | ||||||
|         const t = Translations.t.general.attribution |  | ||||||
| 
 |  | ||||||
|         const josmState = new UIEventSource<string>(undefined) |  | ||||||
|         // Reset after 15s
 |  | ||||||
|         josmState.stabilized(15000).addCallbackD(() => josmState.setData(undefined)) |  | ||||||
| 
 |  | ||||||
|         const stateIndication = new VariableUiElement( |  | ||||||
|             josmState.map((state) => { |  | ||||||
|                 if (state === undefined) { |  | ||||||
|                     return undefined |  | ||||||
|                 } |  | ||||||
|                 state = state.toUpperCase() |  | ||||||
|                 if (state === "OK") { |  | ||||||
|                     return t.josmOpened.SetClass("thanks") |  | ||||||
|                 } |  | ||||||
|                 return t.josmNotOpened.SetClass("alert") |  | ||||||
|             }) |  | ||||||
|         ) |  | ||||||
| 
 |  | ||||||
|         const toggle = new Toggle( |  | ||||||
|             new SubtleButton(Svg.josm_logo_svg().SetStyle(iconStyle), t.editJosm) |  | ||||||
|                 .onClick(() => { |  | ||||||
|                     const bbox = bounds.data |  | ||||||
|                     if (bbox === undefined) { |  | ||||||
|                         return |  | ||||||
|                     } |  | ||||||
|                     const top = bbox.getNorth() |  | ||||||
|                     const bottom = bbox.getSouth() |  | ||||||
|                     const right = bbox.getEast() |  | ||||||
|                     const left = bbox.getWest() |  | ||||||
|                     const josmLink = `http://127.0.0.1:8111/load_and_zoom?left=${left}&right=${right}&top=${top}&bottom=${bottom}` |  | ||||||
|                     Utils.download(josmLink) |  | ||||||
|                         .then((answer) => josmState.setData(answer.replace(/\n/g, "").trim())) |  | ||||||
|                         .catch(() => josmState.setData("ERROR")) |  | ||||||
|                 }) |  | ||||||
|                 .SetClass("w-full"), |  | ||||||
|             undefined, |  | ||||||
|             osmConnection.userDetails.map( |  | ||||||
|                 (ud) => ud.loggedIn && ud.csCount >= Constants.userJourney.historyLinkVisible |  | ||||||
|             ) |  | ||||||
|         ) |  | ||||||
| 
 |  | ||||||
|         super([stateIndication, toggle]) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,20 +1,21 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import Translations from "../i18n/Translations" |   import Translations from "../i18n/Translations"; | ||||||
|   import Svg from "../../Svg" |   import Tr from "../Base/Tr.svelte"; | ||||||
|   import Tr from "../Base/Tr.svelte" |   import NextButton from "../Base/NextButton.svelte"; | ||||||
|   import NextButton from "../Base/NextButton.svelte" |   import Geosearch from "./Geosearch.svelte"; | ||||||
|   import Geosearch from "./Geosearch.svelte" |   import ThemeViewState from "../../Models/ThemeViewState"; | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import { UIEventSource } from "../../Logic/UIEventSource"; | ||||||
|   import ThemeViewState from "../../Models/ThemeViewState" |   import { SearchIcon } from "@rgossiaux/svelte-heroicons/solid"; | ||||||
|   import { UIEventSource } from "../../Logic/UIEventSource" |   import { twJoin } from "tailwind-merge"; | ||||||
|   import { SearchIcon } from "@rgossiaux/svelte-heroicons/solid" |   import { Utils } from "../../Utils"; | ||||||
|   import { twJoin } from "tailwind-merge" |   import type { GeolocationPermissionState } from "../../Logic/State/GeoLocationState"; | ||||||
|   import { Utils } from "../../Utils" |   import { GeoLocationState } from "../../Logic/State/GeoLocationState"; | ||||||
|   import type { GeolocationPermissionState } from "../../Logic/State/GeoLocationState" |   import If from "../Base/If.svelte"; | ||||||
|   import { GeoLocationState } from "../../Logic/State/GeoLocationState" |   import { ExclamationTriangleIcon } from "@babeard/svelte-heroicons/mini"; | ||||||
|   import If from "../Base/If.svelte" |   import type { Readable } from "svelte/store"; | ||||||
|   import { ExclamationTriangleIcon } from "@babeard/svelte-heroicons/mini" |   import Add from "../../assets/svg/Add.svelte"; | ||||||
|   import type { Readable } from "svelte/store" |   import Location_refused from "../../assets/svg/Location_refused.svelte"; | ||||||
|  |   import Crosshair from "../../assets/svg/Crosshair.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * The theme introduction panel |    * The theme introduction panel | ||||||
|  | @ -70,7 +71,7 @@ | ||||||
|       <If condition={state.featureSwitches.featureSwitchGeolocation}> |       <If condition={state.featureSwitches.featureSwitchGeolocation}> | ||||||
|         {#if $currentGPSLocation !== undefined || $geopermission === "prompt"} |         {#if $currentGPSLocation !== undefined || $geopermission === "prompt"} | ||||||
|           <button class="flex w-full items-center gap-x-2" on:click={jumpToCurrentLocation}> |           <button class="flex w-full items-center gap-x-2" on:click={jumpToCurrentLocation}> | ||||||
|             <ToSvelte construct={Svg.crosshair_svg().SetClass("w-8 h-8")} /> |             <Crosshair  class="w-8 h-8"/> | ||||||
|             <Tr t={Translations.t.general.openTheMapAtGeolocation} /> |             <Tr t={Translations.t.general.openTheMapAtGeolocation} /> | ||||||
|           </button> |           </button> | ||||||
|           <!-- No geolocation granted - we don't show the button --> |           <!-- No geolocation granted - we don't show the button --> | ||||||
|  | @ -80,25 +81,17 @@ | ||||||
|             on:click={jumpToCurrentLocation} |             on:click={jumpToCurrentLocation} | ||||||
|           > |           > | ||||||
|             <!-- Even though disabled, when clicking we request the location again in case the contributor dismissed the location popup --> |             <!-- Even though disabled, when clicking we request the location again in case the contributor dismissed the location popup --> | ||||||
|             <ToSvelte |             <Crosshair class="w-8 h-8" style="animation: 3s linear 0s infinite normal none running spin;" /> | ||||||
|               construct={Svg.crosshair_svg() |  | ||||||
|                 .SetClass("w-8 h-8") |  | ||||||
|                 .SetStyle("animation: 3s linear 0s infinite normal none running spin;")} |  | ||||||
|             /> |  | ||||||
|             <Tr t={Translations.t.general.waitingForGeopermission} /> |             <Tr t={Translations.t.general.waitingForGeopermission} /> | ||||||
|           </button> |           </button> | ||||||
|         {:else if $geopermission === "denied"} |         {:else if $geopermission === "denied"} | ||||||
|           <button class="disabled flex w-full items-center gap-x-2"> |           <button class="disabled flex w-full items-center gap-x-2"> | ||||||
|             <ToSvelte construct={Svg.location_refused_svg().SetClass("w-8 h-8")} /> |             <Location_refused class="w-8 h-8"/> | ||||||
|             <Tr t={Translations.t.general.geopermissionDenied} /> |             <Tr t={Translations.t.general.geopermissionDenied} /> | ||||||
|           </button> |           </button> | ||||||
|         {:else} |         {:else} | ||||||
|           <button class="disabled flex w-full items-center gap-x-2"> |           <button class="disabled flex w-full items-center gap-x-2"> | ||||||
|             <ToSvelte |             <Crosshair class="w-8 h-8" style="animation: 3s linear 0s infinite normal none running spin;" /> | ||||||
|               construct={Svg.crosshair_svg() |  | ||||||
|                 .SetClass("w-8 h-8") |  | ||||||
|                 .SetStyle("animation: 3s linear 0s infinite normal none running spin;")} |  | ||||||
|             /> |  | ||||||
|             <Tr t={Translations.t.general.waitingForLocation} /> |             <Tr t={Translations.t.general.waitingForLocation} /> | ||||||
|           </button> |           </button> | ||||||
|         {/if} |         {/if} | ||||||
|  | @ -156,7 +149,7 @@ | ||||||
|   <div class="links-as-button links-w-full m-2 flex flex-col gap-y-1"> |   <div class="links-as-button links-w-full m-2 flex flex-col gap-y-1"> | ||||||
|     <!-- bottom buttons, a bit hidden away: switch layout --> |     <!-- bottom buttons, a bit hidden away: switch layout --> | ||||||
|     <a class="flex" href={Utils.HomepageLink()}> |     <a class="flex" href={Utils.HomepageLink()}> | ||||||
|       <img class="h-6 w-6" src="./assets/svg/add.svg" /> |       <Add  class="h-6 w-6"/> | ||||||
|       <Tr t={Translations.t.general.backToIndex} /> |       <Tr t={Translations.t.general.backToIndex} /> | ||||||
|     </a> |     </a> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|  | @ -12,6 +12,8 @@ import { OsmConnection } from "../../Logic/Osm/OsmConnection" | ||||||
| import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig" | import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig" | ||||||
| import { Translation } from "../i18n/Translation" | import { Translation } from "../i18n/Translation" | ||||||
| import { LoginToggle } from "../Popup/LoginButton" | import { LoginToggle } from "../Popup/LoginButton" | ||||||
|  | import SvelteUIElement from "../Base/SvelteUIElement" | ||||||
|  | import Upload from "../../assets/svg/Upload.svelte" | ||||||
| 
 | 
 | ||||||
| export default class UploadTraceToOsmUI extends LoginToggle { | export default class UploadTraceToOsmUI extends LoginToggle { | ||||||
|     constructor( |     constructor( | ||||||
|  | @ -83,7 +85,7 @@ export default class UploadTraceToOsmUI extends LoginToggle { | ||||||
|                         clicked.setData(false) |                         clicked.setData(false) | ||||||
|                     }) |                     }) | ||||||
|                     .SetClass(""), |                     .SetClass(""), | ||||||
|                 new SubtleButton(Svg.upload_svg(), t.confirm).OnClickWithLoading( |                 new SubtleButton(new SvelteUIElement(Upload, {}), t.confirm).OnClickWithLoading( | ||||||
|                     t.uploading, |                     t.uploading, | ||||||
|                     async () => { |                     async () => { | ||||||
|                         const titleStr = UploadTraceToOsmUI.createDefault( |                         const titleStr = UploadTraceToOsmUI.createDefault( | ||||||
|  | @ -119,7 +121,7 @@ export default class UploadTraceToOsmUI extends LoginToggle { | ||||||
|                     ]).SetClass("flex p-2 rounded-xl border-2 subtle-border items-center"), |                     ]).SetClass("flex p-2 rounded-xl border-2 subtle-border items-center"), | ||||||
|                     new Toggle( |                     new Toggle( | ||||||
|                         confirmPanel, |                         confirmPanel, | ||||||
|                         new SubtleButton(Svg.upload_svg(), t.title).onClick(() => |                         new SubtleButton(new SvelteUIElement(Upload), t.title).onClick(() => | ||||||
|                             clicked.setData(true) |                             clicked.setData(true) | ||||||
|                         ), |                         ), | ||||||
|                         clicked |                         clicked | ||||||
|  |  | ||||||
|  | @ -3,16 +3,15 @@ | ||||||
|    * Shows an 'upload'-button which will start the upload for this feature |    * Shows an 'upload'-button which will start the upload for this feature | ||||||
|    */ |    */ | ||||||
| 
 | 
 | ||||||
|   import type { SpecialVisualizationState } from "../SpecialVisualization" |   import type { SpecialVisualizationState } from "../SpecialVisualization"; | ||||||
|   import { ImmutableStore, Store } from "../../Logic/UIEventSource" |   import { ImmutableStore, Store } from "../../Logic/UIEventSource"; | ||||||
|   import type { OsmTags } from "../../Models/OsmFeature" |   import type { OsmTags } from "../../Models/OsmFeature"; | ||||||
|   import LoginToggle from "../Base/LoginToggle.svelte" |   import LoginToggle from "../Base/LoginToggle.svelte"; | ||||||
|   import Translations from "../i18n/Translations" |   import Translations from "../i18n/Translations"; | ||||||
|   import Tr from "../Base/Tr.svelte" |   import Tr from "../Base/Tr.svelte"; | ||||||
|   import UploadingImageCounter from "./UploadingImageCounter.svelte" |   import UploadingImageCounter from "./UploadingImageCounter.svelte"; | ||||||
|   import FileSelector from "../Base/FileSelector.svelte" |   import FileSelector from "../Base/FileSelector.svelte"; | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import Camera_plus from "../../assets/svg/Camera_plus.svelte"; | ||||||
|   import Svg from "../../Svg" |  | ||||||
| 
 | 
 | ||||||
|   export let state: SpecialVisualizationState |   export let state: SpecialVisualizationState | ||||||
| 
 | 
 | ||||||
|  | @ -58,7 +57,7 @@ | ||||||
|         {#if image !== undefined} |         {#if image !== undefined} | ||||||
|           <img src={image} /> |           <img src={image} /> | ||||||
|         {:else} |         {:else} | ||||||
|           <ToSvelte construct={Svg.camera_plus_svg().SetClass("block w-12 h-12 p-1 text-4xl ")} /> |           <Camera_plus class="block w-12 h-12 p-1 text-4xl"/> | ||||||
|         {/if} |         {/if} | ||||||
|         {#if labelText} |         {#if labelText} | ||||||
|           {labelText} |           {labelText} | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ | ||||||
|   import MaplibreMap from "../../Map/MaplibreMap.svelte" |   import MaplibreMap from "../../Map/MaplibreMap.svelte" | ||||||
|   import ToSvelte from "../../Base/ToSvelte.svelte" |   import ToSvelte from "../../Base/ToSvelte.svelte" | ||||||
|   import Svg from "../../../Svg.js" |   import Svg from "../../../Svg.js" | ||||||
|  |   import Direction_stroke from "../../../assets/svg/Direction_stroke.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * A visualisation to pick a direction on a map background. |    * A visualisation to pick a direction on a map background. | ||||||
|  | @ -67,6 +68,6 @@ | ||||||
|   </div> |   </div> | ||||||
| 
 | 
 | ||||||
|   <div bind:this={directionElem} class="absolute top-0 left-0 h-full w-full"> |   <div bind:this={directionElem} class="absolute top-0 left-0 h-full w-full"> | ||||||
|     <ToSvelte construct={Svg.direction_stroke_svg} /> |     <Direction_stroke/> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|  | @ -12,6 +12,7 @@ | ||||||
|   import * as turf from "@turf/turf" |   import * as turf from "@turf/turf" | ||||||
|   import LayerConfig from "../../../Models/ThemeConfig/LayerConfig" |   import LayerConfig from "../../../Models/ThemeConfig/LayerConfig" | ||||||
|   import { createEventDispatcher, onDestroy } from "svelte" |   import { createEventDispatcher, onDestroy } from "svelte" | ||||||
|  |   import Move_arrows from "../../../assets/svg/Move_arrows.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * A visualisation to pick a location on a map background |    * A visualisation to pick a location on a map background | ||||||
|  | @ -90,7 +91,7 @@ | ||||||
|     class="pointer-events-none absolute top-0 left-0 flex h-full w-full items-center p-8 opacity-50" |     class="pointer-events-none absolute top-0 left-0 flex h-full w-full items-center p-8 opacity-50" | ||||||
|   > |   > | ||||||
|     <slot name="image"> |     <slot name="image"> | ||||||
|       <img class="h-full max-h-24" src="./assets/svg/move-arrows.svg" /> |       <Move_arrows class="h-full max-h-24"/> | ||||||
|     </slot> |     </slot> | ||||||
|   </div> |   </div> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										66
									
								
								src/UI/InputElement/LanguagePicker.svelte
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								src/UI/InputElement/LanguagePicker.svelte
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,66 @@ | ||||||
|  | <script lang="ts"> | ||||||
|  |   // Languages in the language itself | ||||||
|  |   import native from "../../assets/language_native.json"; | ||||||
|  |   // Translated languages | ||||||
|  |   import language_translations from "../../assets/language_translations.json"; | ||||||
|  | 
 | ||||||
|  |   import { UIEventSource } from "../../Logic/UIEventSource"; | ||||||
|  |   import Locale from "../i18n/Locale"; | ||||||
|  |   import { LanguageIcon } from "@babeard/svelte-heroicons/solid"; | ||||||
|  |   import Dropdown from "../Base/Dropdown.svelte"; | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Languages one can choose from | ||||||
|  |    * Defaults to _all_ languages known by MapComplete | ||||||
|  |    */ | ||||||
|  |   export let availableLanguages: string[] = Object.keys(native); | ||||||
|  |   /** | ||||||
|  |    * EventStore to assign to, defaults to 'Locale.langauge' | ||||||
|  |    */ | ||||||
|  |   export let assignTo: UIEventSource<string> = Locale.language; | ||||||
|  |   export let preferredLanguages: UIEventSource<string[]> = undefined; | ||||||
|  |   let preferredFiltered: string[] = undefined; | ||||||
|  |   preferredLanguages?.addCallbackAndRunD(preferredLanguages => { | ||||||
|  |     let lng = navigator.language; | ||||||
|  |     if (lng === "en-US") { | ||||||
|  |       lng = "en"; | ||||||
|  |     } | ||||||
|  |     if (preferredLanguages?.indexOf(lng) < 0) { | ||||||
|  |       preferredLanguages?.push(lng); | ||||||
|  |     } | ||||||
|  |     preferredFiltered = preferredLanguages?.filter(l => availableLanguages.indexOf(l) >= 0); | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   let current = Locale.language; | ||||||
|  | 
 | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | {#if availableLanguages?.length > 1} | ||||||
|  |   <form class="flex items-center"> | ||||||
|  | 
 | ||||||
|  |     <LanguageIcon class="h-4 w-4 mr-1" /> | ||||||
|  |     <Dropdown value={assignTo}> | ||||||
|  |       {#if preferredFiltered} | ||||||
|  |         {#each preferredFiltered as language} | ||||||
|  |           <option value={language} class="font-bold"> | ||||||
|  |             {native[language] ?? ""} | ||||||
|  |             {#if language !== $current} | ||||||
|  |               ({language_translations[language]?.[$current] ?? language}) | ||||||
|  |             {/if} | ||||||
|  |           </option> | ||||||
|  |         {/each} | ||||||
|  |         <option disabled></option> | ||||||
|  |       {/if} | ||||||
|  | 
 | ||||||
|  |       {#each availableLanguages as language} | ||||||
|  |         <option value={language} class="font-bold"> | ||||||
|  |           {native[language] ?? ""} | ||||||
|  |           {#if language !== $current} | ||||||
|  |             ({(language_translations[language]?.[$current] + " - " + language) ?? language}) | ||||||
|  |           {/if} | ||||||
|  |         </option> | ||||||
|  |       {/each} | ||||||
|  |     </Dropdown> | ||||||
|  |   </form> | ||||||
|  | 
 | ||||||
|  | {/if} | ||||||
|  | @ -1,67 +0,0 @@ | ||||||
| import { DropDown } from "./Input/DropDown" |  | ||||||
| import Locale from "./i18n/Locale" |  | ||||||
| import BaseUIElement from "./BaseUIElement" |  | ||||||
| import native from "../assets/language_native.json" |  | ||||||
| import language_translations from "../assets/language_translations.json" |  | ||||||
| import { Translation } from "./i18n/Translation" |  | ||||||
| import Lazy from "./Base/Lazy" |  | ||||||
| import Toggle from "./Input/Toggle" |  | ||||||
| import LanguageUtils from "../Utils/LanguageUtils" |  | ||||||
| import { UIEventSource } from "../Logic/UIEventSource" |  | ||||||
| import { QueryParameters } from "../Logic/Web/QueryParameters" |  | ||||||
| 
 |  | ||||||
| export default class LanguagePicker extends Toggle { |  | ||||||
|     constructor(languages: string[], assignTo: UIEventSource<string>) { |  | ||||||
|         console.log("Constructing a language picker for languages", languages) |  | ||||||
|         if ( |  | ||||||
|             languages === undefined || |  | ||||||
|             languages.length <= 1 || |  | ||||||
|             QueryParameters.wasInitialized("language") |  | ||||||
|         ) { |  | ||||||
|             super(undefined, undefined, undefined) |  | ||||||
|         } else { |  | ||||||
|             const normalPicker = LanguagePicker.dropdownFor(languages, assignTo ?? Locale.language) |  | ||||||
|             const fullPicker = new Lazy(() => |  | ||||||
|                 LanguagePicker.dropdownFor(allLanguages, assignTo ?? Locale.language) |  | ||||||
|             ) |  | ||||||
|             super(fullPicker, normalPicker, Locale.showLinkToWeblate) |  | ||||||
|             const allLanguages: string[] = LanguageUtils.usedLanguagesSorted |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static dropdownFor( |  | ||||||
|         languages: string[], |  | ||||||
|         assignTo: UIEventSource<string> |  | ||||||
|     ): BaseUIElement { |  | ||||||
|         return new DropDown( |  | ||||||
|             undefined, |  | ||||||
|             languages |  | ||||||
|                 .filter((lang) => lang !== "_context") |  | ||||||
|                 .map((lang) => { |  | ||||||
|                     return { value: lang, shown: LanguagePicker.hybrid(lang) } |  | ||||||
|                 }), |  | ||||||
|             assignTo |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     private static hybrid(lang: string): Translation { |  | ||||||
|         const nativeText = native[lang] ?? lang |  | ||||||
|         const translation = {} |  | ||||||
|         const trans = language_translations[lang] |  | ||||||
|         if (trans === undefined) { |  | ||||||
|             return new Translation({ "*": nativeText }) |  | ||||||
|         } |  | ||||||
|         for (const key in trans) { |  | ||||||
|             if (key.startsWith("_")) { |  | ||||||
|                 continue |  | ||||||
|             } |  | ||||||
|             const translationInKey = language_translations[lang][key] |  | ||||||
|             if (nativeText.toLowerCase() === translationInKey.toLowerCase()) { |  | ||||||
|                 translation[key] = nativeText |  | ||||||
|             } else { |  | ||||||
|                 translation[key] = nativeText + " (" + translationInKey + ")" |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return new Translation(translation) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  | @ -1,20 +1,22 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import PointRenderingConfig, { IconConfig } from "../../Models/ThemeConfig/PointRenderingConfig" |   import { IconConfig } from "../../Models/ThemeConfig/PointRenderingConfig"; | ||||||
|   import { Store } from "../../Logic/UIEventSource" |   import { ImmutableStore, Store } from "../../Logic/UIEventSource"; | ||||||
|   import DynamicIcon from "./DynamicIcon.svelte" |   import DynamicIcon from "./DynamicIcon.svelte"; | ||||||
|  |   import TagRenderingConfig from "../../Models/ThemeConfig/TagRenderingConfig"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * Renders a 'marker', which consists of multiple 'icons' |    * Renders a 'marker', which consists of multiple 'icons' | ||||||
|    */ |    */ | ||||||
|   export let config: PointRenderingConfig |   export let marker: IconConfig[] = config?.marker; | ||||||
|   let icons: IconConfig[] = config.marker |   export let tags: Store<Record<string, string>>; | ||||||
|   export let tags: Store<Record<string, string>> |   export let rotation: TagRenderingConfig | ||||||
|   let rotation = tags.map((tags) => config.rotation.GetRenderValue(tags).Subs(tags).txt) |   let _rotation = rotation ? tags.map(tags => rotation.GetRenderValue(tags).Subs(tags).txt) : new ImmutableStore(0); | ||||||
|  |    | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| {#if config !== undefined} | {#if marker && marker} | ||||||
|   <div class="relative h-full w-full" style={`transform: rotate(${$rotation})`}> |   <div class="relative h-full w-full" style={`transform: rotate(${$_rotation})`}> | ||||||
|     {#each icons as icon} |     {#each marker as icon} | ||||||
|       <DynamicIcon {icon} {tags} /> |       <DynamicIcon {icon} {tags} /> | ||||||
|     {/each} |     {/each} | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|  | @ -19,6 +19,9 @@ | ||||||
|   import Teardrop from "../../assets/svg/Teardrop.svelte" |   import Teardrop from "../../assets/svg/Teardrop.svelte" | ||||||
|   import Teardrop_with_hole_green from "../../assets/svg/Teardrop_with_hole_green.svelte" |   import Teardrop_with_hole_green from "../../assets/svg/Teardrop_with_hole_green.svelte" | ||||||
|   import Triangle from "../../assets/svg/Triangle.svelte" |   import Triangle from "../../assets/svg/Triangle.svelte" | ||||||
|  |   import Brick_wall_square from "../../assets/svg/Brick_wall_square.svelte"; | ||||||
|  |   import Brick_wall_round from "../../assets/svg/Brick_wall_round.svelte"; | ||||||
|  |   import Gps_arrow from "../../assets/svg/Gps_arrow.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * Renders a single icon. |    * Renders a single icon. | ||||||
|  | @ -72,6 +75,20 @@ | ||||||
|       <Teardrop_with_hole_green {color} /> |       <Teardrop_with_hole_green {color} /> | ||||||
|     {:else if icon === "triangle"} |     {:else if icon === "triangle"} | ||||||
|       <Triangle {color} /> |       <Triangle {color} /> | ||||||
|  |     {:else if icon === "brick_wall_square"} | ||||||
|  |       <Brick_wall_square {color} /> | ||||||
|  |     {:else if icon === "brick_wall_round"} | ||||||
|  |       <Brick_wall_round {color} /> | ||||||
|  |     {:else if icon === "gps_arrow"} | ||||||
|  |       <Gps_arrow {color} /> | ||||||
|  |     {:else if icon === "checkmark"} | ||||||
|  |       <Checkmark {color} /> | ||||||
|  |     {:else if icon === "help"} | ||||||
|  |       <Help {color} /> | ||||||
|  |     {:else if icon === "close"} | ||||||
|  |       <Close {color} /> | ||||||
|  |     {:else if icon === "invalid"} | ||||||
|  |       <Invalid {color} /> | ||||||
|     {:else} |     {:else} | ||||||
|       <img class="h-full w-full" src={icon} /> |       <img class="h-full w-full" src={icon} /> | ||||||
|     {/if} |     {/if} | ||||||
|  |  | ||||||
|  | @ -12,6 +12,7 @@ | ||||||
|   import { createEventDispatcher } from "svelte" |   import { createEventDispatcher } from "svelte" | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import ToSvelte from "../Base/ToSvelte.svelte" | ||||||
|   import Svg from "../../Svg" |   import Svg from "../../Svg" | ||||||
|  |   import Plantnet_logo from "../../assets/svg/Plantnet_logo.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * The main entry point for the plantnet wizard |    * The main entry point for the plantnet wizard | ||||||
|  | @ -142,9 +143,7 @@ | ||||||
|     </BackButton> |     </BackButton> | ||||||
|   {/if} |   {/if} | ||||||
|   <div class="low-interaction flex self-end rounded-xl p-2"> |   <div class="low-interaction flex self-end rounded-xl p-2"> | ||||||
|     <ToSvelte |     <Plantnet_logo class="w-8 h-8 p-1 mr-1 bg-white rounded-full"/> | ||||||
|       construct={Svg.plantnet_logo_svg().SetClass("w-8 h-8 p-1 mr-1 bg-white rounded-full")} |  | ||||||
|     /> |  | ||||||
|     <Tr t={t.poweredByPlantnet} /> |     <Tr t={t.poweredByPlantnet} /> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|  | @ -3,109 +3,110 @@ | ||||||
|    * This component ties together all the steps that are needed to create a new point. |    * This component ties together all the steps that are needed to create a new point. | ||||||
|    * There are many subcomponents which help with that |    * There are many subcomponents which help with that | ||||||
|    */ |    */ | ||||||
|   import type { SpecialVisualizationState } from "../../SpecialVisualization" |   import type { SpecialVisualizationState } from "../../SpecialVisualization"; | ||||||
|   import PresetList from "./PresetList.svelte" |   import PresetList from "./PresetList.svelte"; | ||||||
|   import type PresetConfig from "../../../Models/ThemeConfig/PresetConfig" |   import type PresetConfig from "../../../Models/ThemeConfig/PresetConfig"; | ||||||
|   import LayerConfig from "../../../Models/ThemeConfig/LayerConfig" |   import LayerConfig from "../../../Models/ThemeConfig/LayerConfig"; | ||||||
|   import Tr from "../../Base/Tr.svelte" |   import Tr from "../../Base/Tr.svelte"; | ||||||
|   import SubtleButton from "../../Base/SubtleButton.svelte" |   import SubtleButton from "../../Base/SubtleButton.svelte"; | ||||||
|   import FromHtml from "../../Base/FromHtml.svelte" |   import Translations from "../../i18n/Translations.js"; | ||||||
|   import Translations from "../../i18n/Translations.js" |   import TagHint from "../TagHint.svelte"; | ||||||
|   import TagHint from "../TagHint.svelte" |   import { And } from "../../../Logic/Tags/And.js"; | ||||||
|   import { And } from "../../../Logic/Tags/And.js" |   import LoginToggle from "../../Base/LoginToggle.svelte"; | ||||||
|   import LoginToggle from "../../Base/LoginToggle.svelte" |   import Constants from "../../../Models/Constants.js"; | ||||||
|   import Constants from "../../../Models/Constants.js" |   import FilteredLayer from "../../../Models/FilteredLayer"; | ||||||
|   import FilteredLayer from "../../../Models/FilteredLayer" |   import { Store, UIEventSource } from "../../../Logic/UIEventSource"; | ||||||
|   import { Store, UIEventSource } from "../../../Logic/UIEventSource" |   import { EyeIcon, EyeOffIcon } from "@rgossiaux/svelte-heroicons/solid"; | ||||||
|   import { EyeIcon, EyeOffIcon } from "@rgossiaux/svelte-heroicons/solid" |   import LoginButton from "../../Base/LoginButton.svelte"; | ||||||
|   import LoginButton from "../../Base/LoginButton.svelte" |   import NewPointLocationInput from "../../BigComponents/NewPointLocationInput.svelte"; | ||||||
|   import NewPointLocationInput from "../../BigComponents/NewPointLocationInput.svelte" |   import CreateNewNodeAction from "../../../Logic/Osm/Actions/CreateNewNodeAction"; | ||||||
|   import CreateNewNodeAction from "../../../Logic/Osm/Actions/CreateNewNodeAction" |   import { OsmWay } from "../../../Logic/Osm/OsmObject"; | ||||||
|   import { OsmWay } from "../../../Logic/Osm/OsmObject" |   import { Tag } from "../../../Logic/Tags/Tag"; | ||||||
|   import { Tag } from "../../../Logic/Tags/Tag" |   import type { WayId } from "../../../Models/OsmFeature"; | ||||||
|   import type { WayId } from "../../../Models/OsmFeature" |   import Loading from "../../Base/Loading.svelte"; | ||||||
|   import Loading from "../../Base/Loading.svelte" |   import type { GlobalFilter } from "../../../Models/GlobalFilter"; | ||||||
|   import type { GlobalFilter } from "../../../Models/GlobalFilter" |   import { onDestroy } from "svelte"; | ||||||
|   import { onDestroy } from "svelte" |   import NextButton from "../../Base/NextButton.svelte"; | ||||||
|   import NextButton from "../../Base/NextButton.svelte" |   import BackButton from "../../Base/BackButton.svelte"; | ||||||
|   import BackButton from "../../Base/BackButton.svelte" |   import ToSvelte from "../../Base/ToSvelte.svelte"; | ||||||
|   import ToSvelte from "../../Base/ToSvelte.svelte" |   import Svg from "../../../Svg"; | ||||||
|   import Svg from "../../../Svg" |   import OpenBackgroundSelectorButton from "../../BigComponents/OpenBackgroundSelectorButton.svelte"; | ||||||
|   import OpenBackgroundSelectorButton from "../../BigComponents/OpenBackgroundSelectorButton.svelte" |   import { twJoin } from "tailwind-merge"; | ||||||
|   import { twJoin } from "tailwind-merge" |   import Confirm from "../../../assets/svg/Confirm.svelte"; | ||||||
|  |   import Close from "../../../assets/svg/Close.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let coordinate: { lon: number; lat: number } |   export let coordinate: { lon: number; lat: number }; | ||||||
|   export let state: SpecialVisualizationState |   export let state: SpecialVisualizationState; | ||||||
| 
 | 
 | ||||||
|   let selectedPreset: { |   let selectedPreset: { | ||||||
|     preset: PresetConfig |     preset: PresetConfig | ||||||
|     layer: LayerConfig |     layer: LayerConfig | ||||||
|     icon: string |     icon: string | ||||||
|     tags: Record<string, string> |     tags: Record<string, string> | ||||||
|   } = undefined |   } = undefined; | ||||||
|   let checkedOfGlobalFilters: number = 0 |   let checkedOfGlobalFilters: number = 0; | ||||||
|   let confirmedCategory = false |   let confirmedCategory = false; | ||||||
|   $: if (selectedPreset === undefined) { |   $: if (selectedPreset === undefined) { | ||||||
|     confirmedCategory = false |     confirmedCategory = false; | ||||||
|     creating = false |     creating = false; | ||||||
|     checkedOfGlobalFilters = 0 |     checkedOfGlobalFilters = 0; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   let flayer: FilteredLayer = undefined |   let flayer: FilteredLayer = undefined; | ||||||
|   let layerIsDisplayed: UIEventSource<boolean> | undefined = undefined |   let layerIsDisplayed: UIEventSource<boolean> | undefined = undefined; | ||||||
|   let layerHasFilters: Store<boolean> | undefined = undefined |   let layerHasFilters: Store<boolean> | undefined = undefined; | ||||||
|   let globalFilter: UIEventSource<GlobalFilter[]> = state.layerState.globalFilters |   let globalFilter: UIEventSource<GlobalFilter[]> = state.layerState.globalFilters; | ||||||
|   let _globalFilter: GlobalFilter[] = [] |   let _globalFilter: GlobalFilter[] = []; | ||||||
|   onDestroy( |   onDestroy( | ||||||
|     globalFilter.addCallbackAndRun((globalFilter) => { |     globalFilter.addCallbackAndRun((globalFilter) => { | ||||||
|       console.log("Global filters are", globalFilter) |       console.log("Global filters are", globalFilter); | ||||||
|       _globalFilter = globalFilter ?? [] |       _globalFilter = globalFilter ?? []; | ||||||
|     }) |     }) | ||||||
|   ) |   ); | ||||||
|   $: { |   $: { | ||||||
|     flayer = state.layerState.filteredLayers.get(selectedPreset?.layer?.id) |     flayer = state.layerState.filteredLayers.get(selectedPreset?.layer?.id); | ||||||
|     layerIsDisplayed = flayer?.isDisplayed |     layerIsDisplayed = flayer?.isDisplayed; | ||||||
|     layerHasFilters = flayer?.hasFilter |     layerHasFilters = flayer?.hasFilter; | ||||||
|   } |   } | ||||||
|   const t = Translations.t.general.add |   const t = Translations.t.general.add; | ||||||
| 
 | 
 | ||||||
|   const zoom = state.mapProperties.zoom |   const zoom = state.mapProperties.zoom; | ||||||
| 
 | 
 | ||||||
|   const isLoading = state.dataIsLoading |   const isLoading = state.dataIsLoading; | ||||||
|   let preciseCoordinate: UIEventSource<{ lon: number; lat: number }> = new UIEventSource(undefined) |   let preciseCoordinate: UIEventSource<{ lon: number; lat: number }> = new UIEventSource(undefined); | ||||||
|   let snappedToObject: UIEventSource<string> = new UIEventSource<string>(undefined) |   let snappedToObject: UIEventSource<string> = new UIEventSource<string>(undefined); | ||||||
| 
 | 
 | ||||||
|   // Small helper variable: if the map is tapped, we should let the 'Next'-button grab some attention as users have to click _that_ to continue, not the map |   // Small helper variable: if the map is tapped, we should let the 'Next'-button grab some attention as users have to click _that_ to continue, not the map | ||||||
|   let preciseInputIsTapped = false |   let preciseInputIsTapped = false; | ||||||
| 
 | 
 | ||||||
|   let creating = false |   let creating = false; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * Call when the user should restart the flow by clicking on the map, e.g. because they disabled filters. |    * Call when the user should restart the flow by clicking on the map, e.g. because they disabled filters. | ||||||
|    * Will delete the lastclick-location |    * Will delete the lastclick-location | ||||||
|    */ |    */ | ||||||
|   function abort() { |   function abort() { | ||||||
|     state.selectedElement.setData(undefined) |     state.selectedElement.setData(undefined); | ||||||
|     // When aborted, we force the contributors to place the pin _again_ |     // When aborted, we force the contributors to place the pin _again_ | ||||||
|     // This is because there might be a nearby object that was disabled; this forces them to re-evaluate the map |     // This is because there might be a nearby object that was disabled; this forces them to re-evaluate the map | ||||||
|     state.lastClickObject.features.setData([]) |     state.lastClickObject.features.setData([]); | ||||||
|     preciseInputIsTapped = false |     preciseInputIsTapped = false; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   async function confirm() { |   async function confirm() { | ||||||
|     creating = true |     creating = true; | ||||||
|     const location: { lon: number; lat: number } = preciseCoordinate.data |     const location: { lon: number; lat: number } = preciseCoordinate.data; | ||||||
|     const snapTo: WayId | undefined = <WayId>snappedToObject.data |     const snapTo: WayId | undefined = <WayId>snappedToObject.data; | ||||||
|     const tags: Tag[] = selectedPreset.preset.tags.concat( |     const tags: Tag[] = selectedPreset.preset.tags.concat( | ||||||
|       ..._globalFilter.map((f) => f?.onNewPoint?.tags ?? []) |       ..._globalFilter.map((f) => f?.onNewPoint?.tags ?? []) | ||||||
|     ) |     ); | ||||||
|     console.log("Creating new point at", location, "snapped to", snapTo, "with tags", tags) |     console.log("Creating new point at", location, "snapped to", snapTo, "with tags", tags); | ||||||
| 
 | 
 | ||||||
|     let snapToWay: undefined | OsmWay = undefined |     let snapToWay: undefined | OsmWay = undefined; | ||||||
|     if (snapTo !== undefined && snapTo !== null) { |     if (snapTo !== undefined && snapTo !== null) { | ||||||
|       const downloaded = await state.osmObjectDownloader.DownloadObjectAsync(snapTo, 0) |       const downloaded = await state.osmObjectDownloader.DownloadObjectAsync(snapTo, 0); | ||||||
|       if (downloaded !== "deleted") { |       if (downloaded !== "deleted") { | ||||||
|         snapToWay = downloaded |         snapToWay = downloaded; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -113,44 +114,44 @@ | ||||||
|       theme: state.layout?.id ?? "unkown", |       theme: state.layout?.id ?? "unkown", | ||||||
|       changeType: "create", |       changeType: "create", | ||||||
|       snapOnto: snapToWay, |       snapOnto: snapToWay, | ||||||
|       reusePointWithinMeters: 1, |       reusePointWithinMeters: 1 | ||||||
|     }) |     }); | ||||||
|     await state.changes.applyAction(newElementAction) |     await state.changes.applyAction(newElementAction); | ||||||
|     state.newFeatures.features.ping() |     state.newFeatures.features.ping(); | ||||||
|     // The 'changes' should have created a new point, which added this into the 'featureProperties' |     // The 'changes' should have created a new point, which added this into the 'featureProperties' | ||||||
|     const newId = newElementAction.newElementId |     const newId = newElementAction.newElementId; | ||||||
|     console.log("Applied pending changes, fetching store for", newId) |     console.log("Applied pending changes, fetching store for", newId); | ||||||
|     const tagsStore = state.featureProperties.getStore(newId) |     const tagsStore = state.featureProperties.getStore(newId); | ||||||
|     if (!tagsStore) { |     if (!tagsStore) { | ||||||
|       console.error("Bug: no tagsStore found for", newId) |       console.error("Bug: no tagsStore found for", newId); | ||||||
|     } |     } | ||||||
|     { |     { | ||||||
|       // Set some metainfo |       // Set some metainfo | ||||||
|       const properties = tagsStore.data |       const properties = tagsStore.data; | ||||||
|       if (snapTo) { |       if (snapTo) { | ||||||
|         // metatags (starting with underscore) are not uploaded, so we can safely mark this |         // metatags (starting with underscore) are not uploaded, so we can safely mark this | ||||||
|         delete properties["_referencing_ways"] |         delete properties["_referencing_ways"]; | ||||||
|         properties["_referencing_ways"] = `["${snapTo}"]` |         properties["_referencing_ways"] = `["${snapTo}"]`; | ||||||
|       } |       } | ||||||
|       properties["_backend"] = state.osmConnection.Backend() |       properties["_backend"] = state.osmConnection.Backend(); | ||||||
|       properties["_last_edit:timestamp"] = new Date().toISOString() |       properties["_last_edit:timestamp"] = new Date().toISOString(); | ||||||
|       const userdetails = state.osmConnection.userDetails.data |       const userdetails = state.osmConnection.userDetails.data; | ||||||
|       properties["_last_edit:contributor"] = userdetails.name |       properties["_last_edit:contributor"] = userdetails.name; | ||||||
|       properties["_last_edit:uid"] = "" + userdetails.uid |       properties["_last_edit:uid"] = "" + userdetails.uid; | ||||||
|       tagsStore.ping() |       tagsStore.ping(); | ||||||
|     } |     } | ||||||
|     const feature = state.indexedFeatures.featuresById.data.get(newId) |     const feature = state.indexedFeatures.featuresById.data.get(newId); | ||||||
|     console.log("Selecting feature", feature, "and opening their popup") |     console.log("Selecting feature", feature, "and opening their popup"); | ||||||
|     abort() |     abort(); | ||||||
|     state.selectedLayer.setData(selectedPreset.layer) |     state.selectedLayer.setData(selectedPreset.layer); | ||||||
|     state.selectedElement.setData(feature) |     state.selectedElement.setData(feature); | ||||||
|     tagsStore.ping() |     tagsStore.ping(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   function confirmSync() { |   function confirmSync() { | ||||||
|     confirm() |     confirm() | ||||||
|       .then((_) => console.debug("New point successfully handled")) |       .then((_) => console.debug("New point successfully handled")) | ||||||
|       .catch((e) => console.error("Handling the new point went wrong due to", e)) |       .catch((e) => console.error("Handling the new point went wrong due to", e)); | ||||||
|   } |   } | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
|  | @ -285,7 +286,7 @@ | ||||||
|         <NextButton on:click={() => (confirmedCategory = true)} clss="primary w-full"> |         <NextButton on:click={() => (confirmedCategory = true)} clss="primary w-full"> | ||||||
|           <div slot="image" class="relative"> |           <div slot="image" class="relative"> | ||||||
|             <ToSvelte construct={selectedPreset.icon} /> |             <ToSvelte construct={selectedPreset.icon} /> | ||||||
|             <img class="absolute bottom-0 right-0 h-4 w-4" src="./assets/svg/confirm.svg" /> |             <Confirm class="absolute bottom-0 right-0 h-4 w-4" /> | ||||||
|           </div> |           </div> | ||||||
|           <div class="w-full"> |           <div class="w-full"> | ||||||
|             <Tr t={selectedPreset.text} /> |             <Tr t={selectedPreset.text} /> | ||||||
|  | @ -299,11 +300,7 @@ | ||||||
|           checkedOfGlobalFilters = checkedOfGlobalFilters + 1 |           checkedOfGlobalFilters = checkedOfGlobalFilters + 1 | ||||||
|         }} |         }} | ||||||
|       > |       > | ||||||
|         <img |         <Confirm slot="image" class="h-12 w-12" /> | ||||||
|           slot="image" |  | ||||||
|           src={_globalFilter[checkedOfGlobalFilters].onNewPoint?.icon ?? "./assets/svg/confirm.svg"} |  | ||||||
|           class="h-12 w-12" |  | ||||||
|         /> |  | ||||||
|         <Tr |         <Tr | ||||||
|           slot="message" |           slot="message" | ||||||
|           t={_globalFilter[checkedOfGlobalFilters].onNewPoint?.confirmAddNew.Subs({ |           t={_globalFilter[checkedOfGlobalFilters].onNewPoint?.confirmAddNew.Subs({ | ||||||
|  | @ -317,7 +314,7 @@ | ||||||
|           abort() |           abort() | ||||||
|         }} |         }} | ||||||
|       > |       > | ||||||
|         <img slot="image" src="./assets/svg/close.svg" class="h-8 w-8" /> |         <Close slot="image" class="h-8 w-8" /> | ||||||
|         <Tr slot="message" t={Translations.t.general.cancel} /> |         <Tr slot="message" t={Translations.t.general.cancel} /> | ||||||
|       </SubtleButton> |       </SubtleButton> | ||||||
|     {:else if !creating} |     {:else if !creating} | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ | ||||||
|   import Lazy from "../Base/Lazy" |   import Lazy from "../Base/Lazy" | ||||||
|   import BaseUIElement from "../BaseUIElement" |   import BaseUIElement from "../BaseUIElement" | ||||||
|   import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig" |   import LayoutConfig from "../../Models/ThemeConfig/LayoutConfig" | ||||||
|  |   import { VariableUiElement } from "../Base/VariableUIElement"; | ||||||
| 
 | 
 | ||||||
|   //Svelte props |   //Svelte props | ||||||
|   export let tags: UIEventSource<any> |   export let tags: UIEventSource<any> | ||||||
|  | @ -54,13 +55,7 @@ | ||||||
|     return parts |     return parts | ||||||
|   }) |   }) | ||||||
| 
 | 
 | ||||||
|   let _allTags = [] |   const tagsTable = new VariableUiElement(allTags.mapD(_allTags => new Table(["Key", "Value"], _allTags).SetClass("zebra-table break-all"))) | ||||||
|   onDestroy( |  | ||||||
|     allTags.addCallbackAndRunD((allTags) => { |  | ||||||
|       _allTags = allTags |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   const tagsTable = new Table(["Key", "Value"], _allTags).SetClass("zebra-table break-all") |  | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <section> | <section> | ||||||
|  |  | ||||||
|  | @ -15,6 +15,8 @@ | ||||||
|   import NewPointLocationInput from "../BigComponents/NewPointLocationInput.svelte" |   import NewPointLocationInput from "../BigComponents/NewPointLocationInput.svelte" | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import ToSvelte from "../Base/ToSvelte.svelte" | ||||||
|   import Svg from "../../Svg" |   import Svg from "../../Svg" | ||||||
|  |   import Layers from "../../assets/svg/Layers.svelte"; | ||||||
|  |   import AddSmall from "../../assets/svg/AddSmall.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let coordinate: UIEventSource<{ lon: number; lat: number }> |   export let coordinate: UIEventSource<{ lon: number; lat: number }> | ||||||
|   export let state: SpecialVisualizationState |   export let state: SpecialVisualizationState | ||||||
|  | @ -97,7 +99,7 @@ | ||||||
|           <Tr t={Translations.t.notes.noteLayerHasFilters} /> |           <Tr t={Translations.t.notes.noteLayerHasFilters} /> | ||||||
|         </div> |         </div> | ||||||
|         <SubtleButton on:click={() => notelayer.disableAllFilters()}> |         <SubtleButton on:click={() => notelayer.disableAllFilters()}> | ||||||
|           <img slot="image" src="./assets/svg/filter.svg" class="mr-4 h-8 w-8" /> |           <Layers class="mr-4 h-8 w-8"/> | ||||||
|           <Tr slot="message" t={Translations.t.notes.disableAllNoteFilters} /> |           <Tr slot="message" t={Translations.t.notes.disableAllNoteFilters} /> | ||||||
|         </SubtleButton> |         </SubtleButton> | ||||||
|       </div> |       </div> | ||||||
|  | @ -126,7 +128,7 @@ | ||||||
| 
 | 
 | ||||||
|           {#if $comment?.length >= 3} |           {#if $comment?.length >= 3} | ||||||
|             <SubtleButton on:click={uploadNote}> |             <SubtleButton on:click={uploadNote}> | ||||||
|               <img slot="image" src="./assets/svg/addSmall.svg" class="mr-4 h-8 w-8" /> |               <AddSmall slot="image" class="mr-4 h-8 w-8" /> | ||||||
|               <Tr slot="message" t={Translations.t.notes.createNote} /> |               <Tr slot="message" t={Translations.t.notes.createNote} /> | ||||||
|             </SubtleButton> |             </SubtleButton> | ||||||
|           {:else} |           {:else} | ||||||
|  | @ -143,7 +145,7 @@ | ||||||
|         <Tr t={Translations.t.notes.noteLayerNotEnabled} /> |         <Tr t={Translations.t.notes.noteLayerNotEnabled} /> | ||||||
|       </div> |       </div> | ||||||
|       <SubtleButton on:click={enableNoteLayer}> |       <SubtleButton on:click={enableNoteLayer}> | ||||||
|         <img slot="image" src="./assets/svg/layers.svg" class="mr-4 h-8 w-8" /> |         <Layers slot="image" class="mr-4 h-8 w-8" /> | ||||||
|         <Tr slot="message" t={Translations.t.notes.noteLayerDoEnable} /> |         <Tr slot="message" t={Translations.t.notes.noteLayerDoEnable} /> | ||||||
|       </SubtleButton> |       </SubtleButton> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|  | @ -11,6 +11,7 @@ | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import ToSvelte from "../Base/ToSvelte.svelte" | ||||||
|   import { XCircleIcon } from "@babeard/svelte-heroicons/solid" |   import { XCircleIcon } from "@babeard/svelte-heroicons/solid" | ||||||
|   import exp from "constants" |   import exp from "constants" | ||||||
|  |   import Camera_plus from "../../assets/svg/Camera_plus.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let tags: Store<OsmTags> |   export let tags: Store<OsmTags> | ||||||
|   export let state: SpecialVisualizationState |   export let state: SpecialVisualizationState | ||||||
|  | @ -42,7 +43,7 @@ | ||||||
|       expanded = true |       expanded = true | ||||||
|     }} |     }} | ||||||
|   > |   > | ||||||
|     <ToSvelte construct={Svg.camera_plus_svg().SetClass("block w-8 h-8 p-1 mr-2 ")} /> |     <Camera_plus class="block w-8 h-8 p-1 mr-2"/> | ||||||
|     <Tr t={t.seeNearby} /> |     <Tr t={t.seeNearby} /> | ||||||
|   </button> |   </button> | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
|  | @ -23,6 +23,8 @@ | ||||||
|   import UserRelatedState from "../../../Logic/State/UserRelatedState" |   import UserRelatedState from "../../../Logic/State/UserRelatedState" | ||||||
|   import { twJoin } from "tailwind-merge" |   import { twJoin } from "tailwind-merge" | ||||||
|   import { TagUtils } from "../../../Logic/Tags/TagUtils" |   import { TagUtils } from "../../../Logic/Tags/TagUtils" | ||||||
|  |   import Search from "../../../assets/svg/Search.svelte"; | ||||||
|  |   import Login from "../../../assets/svg/Login.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let config: TagRenderingConfig |   export let config: TagRenderingConfig | ||||||
|   export let tags: UIEventSource<Record<string, string>> |   export let tags: UIEventSource<Record<string, string>> | ||||||
|  | @ -211,7 +213,7 @@ | ||||||
| 
 | 
 | ||||||
|     {#if config.mappings?.length >= 8} |     {#if config.mappings?.length >= 8} | ||||||
|       <div class="sticky flex w-full"> |       <div class="sticky flex w-full"> | ||||||
|         <img src="./assets/svg/search.svg" class="h-6 w-6" /> |         <Search class="h-6 w-6"/> | ||||||
|         <input type="text" bind:value={$searchTerm} class="w-full" /> |         <input type="text" bind:value={$searchTerm} class="w-full" /> | ||||||
|       </div> |       </div> | ||||||
|     {/if} |     {/if} | ||||||
|  | @ -318,7 +320,7 @@ | ||||||
|     <LoginToggle {state}> |     <LoginToggle {state}> | ||||||
|       <Loading slot="loading" /> |       <Loading slot="loading" /> | ||||||
|       <SubtleButton slot="not-logged-in" on:click={() => state?.osmConnection?.AttemptLogin()}> |       <SubtleButton slot="not-logged-in" on:click={() => state?.osmConnection?.AttemptLogin()}> | ||||||
|         <img slot="image" src="./assets/svg/login.svg" class="h-8 w-8" /> |         <Login slot="image" class="h-8 w-8" /> | ||||||
|         <Tr t={Translations.t.general.loginToStart} slot="message" /> |         <Tr t={Translations.t.general.loginToStart} slot="message" /> | ||||||
|       </SubtleButton> |       </SubtleButton> | ||||||
|       {#if $feedback !== undefined} |       {#if $feedback !== undefined} | ||||||
|  |  | ||||||
|  | @ -12,6 +12,7 @@ | ||||||
|   import LayerConfig from "../../Models/ThemeConfig/LayerConfig" |   import LayerConfig from "../../Models/ThemeConfig/LayerConfig" | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import ToSvelte from "../Base/ToSvelte.svelte" | ||||||
|   import Svg from "../../Svg" |   import Svg from "../../Svg" | ||||||
|  |   import Mangrove_logo from "../../assets/svg/Mangrove_logo.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * An element showing all reviews |    * An element showing all reviews | ||||||
|  | @ -40,7 +41,7 @@ | ||||||
|     <Tr t={Translations.t.reviews.no_reviews_yet} /> |     <Tr t={Translations.t.reviews.no_reviews_yet} /> | ||||||
|   {/if} |   {/if} | ||||||
|   <div class="flex justify-end"> |   <div class="flex justify-end"> | ||||||
|     <ToSvelte construct={Svg.mangrove_logo_svg().SetClass("w-12 h-12 shrink-0 p-1 ")} /> |     <Mangrove_logo class="w-12 h-12 shrink-0 p-1"/> | ||||||
|     <Tr cls="text-sm subtle" t={Translations.t.reviews.attribution} /> |     <Tr cls="text-sm subtle" t={Translations.t.reviews.attribution} /> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|  | @ -2,6 +2,9 @@ | ||||||
|   import ToSvelte from "../Base/ToSvelte.svelte" |   import ToSvelte from "../Base/ToSvelte.svelte" | ||||||
|   import Svg from "../../Svg" |   import Svg from "../../Svg" | ||||||
|   import { createEventDispatcher } from "svelte" |   import { createEventDispatcher } from "svelte" | ||||||
|  |   import Star from "../../assets/svg/Star.svelte"; | ||||||
|  |   import Star_half from "../../assets/svg/Star_half.svelte"; | ||||||
|  |   import Star_outline from "../../assets/svg/Star_outline.svelte"; | ||||||
| 
 | 
 | ||||||
|   export let score: number |   export let score: number | ||||||
|   export let cutoff: number |   export let cutoff: number | ||||||
|  | @ -23,10 +26,10 @@ | ||||||
|   on:mousemove={(e) => dispatch("hover", { score: getScore(e) })} |   on:mousemove={(e) => dispatch("hover", { score: getScore(e) })} | ||||||
| > | > | ||||||
|   {#if score >= cutoff} |   {#if score >= cutoff} | ||||||
|     <ToSvelte construct={Svg.star_svg().SetClass(starSize)} /> |     <Star class={starSize}/> | ||||||
|   {:else if score + 10 >= cutoff} |   {:else if score + 10 >= cutoff} | ||||||
|     <ToSvelte construct={Svg.star_half_svg().SetClass(starSize)} /> |     <Star_half class={starSize}/> | ||||||
|   {:else} |   {:else} | ||||||
|     <ToSvelte construct={Svg.star_outline_svg().SetClass(starSize)} /> |     <Star_outline class={starSize}/> | ||||||
|   {/if} |   {/if} | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|  | @ -88,7 +88,7 @@ export interface SpecialVisualization { | ||||||
|     readonly funcName: string |     readonly funcName: string | ||||||
|     readonly docs: string | BaseUIElement |     readonly docs: string | BaseUIElement | ||||||
|     readonly example?: string |     readonly example?: string | ||||||
|     readonly needsUrls: string[] |     readonly needsUrls: string[] | ((args: string[]) => string) | ||||||
| 
 | 
 | ||||||
|     /** |     /** | ||||||
|      * Indicates that this special visualisation will make requests to the 'alLNodesDatabase' and that it thus should be included |      * Indicates that this special visualisation will make requests to the 'alLNodesDatabase' and that it thus should be included | ||||||
|  |  | ||||||
|  | @ -45,7 +45,6 @@ import { GeoOperations } from "../Logic/GeoOperations" | ||||||
| import CreateNewNote from "./Popup/CreateNewNote.svelte" | import CreateNewNote from "./Popup/CreateNewNote.svelte" | ||||||
| import AddNewPoint from "./Popup/AddNewPoint/AddNewPoint.svelte" | import AddNewPoint from "./Popup/AddNewPoint/AddNewPoint.svelte" | ||||||
| import UserProfile from "./BigComponents/UserProfile.svelte" | import UserProfile from "./BigComponents/UserProfile.svelte" | ||||||
| import LanguagePicker from "./LanguagePicker" |  | ||||||
| import Link from "./Base/Link" | import Link from "./Base/Link" | ||||||
| import LayerConfig from "../Models/ThemeConfig/LayerConfig" | import LayerConfig from "../Models/ThemeConfig/LayerConfig" | ||||||
| import TagRenderingConfig from "../Models/ThemeConfig/TagRenderingConfig" | import TagRenderingConfig from "../Models/ThemeConfig/TagRenderingConfig" | ||||||
|  | @ -59,7 +58,6 @@ import { PointImportButtonViz } from "./Popup/ImportButtons/PointImportButtonViz | ||||||
| import WayImportButtonViz from "./Popup/ImportButtons/WayImportButtonViz" | import WayImportButtonViz from "./Popup/ImportButtons/WayImportButtonViz" | ||||||
| import ConflateImportButtonViz from "./Popup/ImportButtons/ConflateImportButtonViz" | import ConflateImportButtonViz from "./Popup/ImportButtons/ConflateImportButtonViz" | ||||||
| import DeleteWizard from "./Popup/DeleteFlow/DeleteWizard.svelte" | import DeleteWizard from "./Popup/DeleteFlow/DeleteWizard.svelte" | ||||||
| import { OpenJosm } from "./BigComponents/OpenJosm" |  | ||||||
| import OpenIdEditor from "./BigComponents/OpenIdEditor.svelte" | import OpenIdEditor from "./BigComponents/OpenIdEditor.svelte" | ||||||
| import FediverseValidator from "./InputElement/Validators/FediverseValidator" | import FediverseValidator from "./InputElement/Validators/FediverseValidator" | ||||||
| import SendEmail from "./Popup/SendEmail.svelte" | import SendEmail from "./Popup/SendEmail.svelte" | ||||||
|  | @ -78,6 +76,9 @@ import Questionbox from "./Popup/TagRendering/Questionbox.svelte" | ||||||
| import { TagUtils } from "../Logic/Tags/TagUtils" | import { TagUtils } from "../Logic/Tags/TagUtils" | ||||||
| import Giggity from "./BigComponents/Giggity.svelte" | import Giggity from "./BigComponents/Giggity.svelte" | ||||||
| import ThemeViewState from "../Models/ThemeViewState" | import ThemeViewState from "../Models/ThemeViewState" | ||||||
|  | import LanguagePicker from "./InputElement/LanguagePicker.svelte" | ||||||
|  | import LogoutButton from "./Base/LogoutButton.svelte" | ||||||
|  | import OpenJosm from "./Base/OpenJosm.svelte" | ||||||
| 
 | 
 | ||||||
| class NearbyImageVis implements SpecialVisualization { | class NearbyImageVis implements SpecialVisualization { | ||||||
|     // Class must be in SpecialVisualisations due to weird cyclical import that breaks the tests
 |     // Class must be in SpecialVisualisations due to weird cyclical import that breaks the tests
 | ||||||
|  | @ -453,10 +454,13 @@ export default class SpecialVisualizations { | ||||||
|                 needsUrls: [], |                 needsUrls: [], | ||||||
|                 docs: "A component to set the language of the user interface", |                 docs: "A component to set the language of the user interface", | ||||||
|                 constr(state: SpecialVisualizationState): BaseUIElement { |                 constr(state: SpecialVisualizationState): BaseUIElement { | ||||||
|                     return new LanguagePicker( |                     return new SvelteUIElement(LanguagePicker, { | ||||||
|                         state.layout.language, |                         assignTo: state.userRelatedState.language, | ||||||
|                         state.userRelatedState.language |                         availableLanguages: state.layout.language, | ||||||
|                     ) |                         preferredLanguages: state.osmConnection.userDetails.map( | ||||||
|  |                             (ud) => ud.languages | ||||||
|  |                         ), | ||||||
|  |                     }) | ||||||
|                 }, |                 }, | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|  | @ -465,11 +469,7 @@ export default class SpecialVisualizations { | ||||||
|                 needsUrls: [Constants.osmAuthConfig.url], |                 needsUrls: [Constants.osmAuthConfig.url], | ||||||
|                 docs: "Shows a button where the user can log out", |                 docs: "Shows a button where the user can log out", | ||||||
|                 constr(state: SpecialVisualizationState): BaseUIElement { |                 constr(state: SpecialVisualizationState): BaseUIElement { | ||||||
|                     return new SubtleButton(Svg.logout_svg(), Translations.t.general.logout, { |                     return new SvelteUIElement(LogoutButton, { osmConnection: state.osmConnection }) | ||||||
|                         imgSize: "w-6 h-6", |  | ||||||
|                     }).onClick(() => { |  | ||||||
|                         state.osmConnection.LogOut() |  | ||||||
|                     }) |  | ||||||
|                 }, |                 }, | ||||||
|             }, |             }, | ||||||
|             new HistogramViz(), |             new HistogramViz(), | ||||||
|  | @ -903,10 +903,10 @@ export default class SpecialVisualizations { | ||||||
|                 funcName: "open_in_josm", |                 funcName: "open_in_josm", | ||||||
|                 docs: "Opens the current view in the JOSM-editor", |                 docs: "Opens the current view in the JOSM-editor", | ||||||
|                 args: [], |                 args: [], | ||||||
|                 needsUrls: OpenJosm.needsUrls, |                 needsUrls: ["http://127.0.0.1:8111/load_and_zoom"], | ||||||
| 
 | 
 | ||||||
|                 constr: (state) => { |                 constr: (state) => { | ||||||
|                     return new OpenJosm(state.osmConnection, state.mapProperties.bounds) |                     return new SvelteUIElement(OpenJosm, { state }) | ||||||
|                 }, |                 }, | ||||||
|             }, |             }, | ||||||
|             { |             { | ||||||
|  | @ -1099,12 +1099,6 @@ export default class SpecialVisualizations { | ||||||
|                     if (maproulette_id_key === "" || maproulette_id_key === undefined) { |                     if (maproulette_id_key === "" || maproulette_id_key === undefined) { | ||||||
|                         maproulette_id_key = "mr_taskId" |                         maproulette_id_key = "mr_taskId" | ||||||
|                     } |                     } | ||||||
|                     if (Svg.All[image] !== undefined || Svg.All[image + ".svg"] !== undefined) { |  | ||||||
|                         if (image.endsWith(".svg")) { |  | ||||||
|                             image = image.substring(0, image.length - 4) |  | ||||||
|                         } |  | ||||||
|                         image = Svg[image + "_svg"]() |  | ||||||
|                     } |  | ||||||
|                     const failed = new UIEventSource(false) |                     const failed = new UIEventSource(false) | ||||||
| 
 | 
 | ||||||
|                     const closeButton = new SubtleButton(image, message).OnClickWithLoading( |                     const closeButton = new SubtleButton(image, message).OnClickWithLoading( | ||||||
|  | @ -1460,7 +1454,7 @@ export default class SpecialVisualizations { | ||||||
|                     }, |                     }, | ||||||
|                 ], |                 ], | ||||||
|                 docs: "Shows events that are happening based on a Giggity URL", |                 docs: "Shows events that are happening based on a Giggity URL", | ||||||
|                 needsUrls: ["*"], |                 needsUrls: (args) => args[0], | ||||||
|                 constr( |                 constr( | ||||||
|                     state: SpecialVisualizationState, |                     state: SpecialVisualizationState, | ||||||
|                     tagSource: UIEventSource<Record<string, string>>, |                     tagSource: UIEventSource<Record<string, string>>, | ||||||
|  |  | ||||||
|  | @ -15,7 +15,6 @@ | ||||||
|     import If from "./Base/If.svelte" |     import If from "./Base/If.svelte" | ||||||
|     import BackButton from "./Base/BackButton.svelte" |     import BackButton from "./Base/BackButton.svelte" | ||||||
|     import ChooseLayerToEdit from "./Studio/ChooseLayerToEdit.svelte" |     import ChooseLayerToEdit from "./Studio/ChooseLayerToEdit.svelte" | ||||||
|   import { LocalStorageSource } from "../Logic/Web/LocalStorageSource" |  | ||||||
|     import FloatOver from "./Base/FloatOver.svelte" |     import FloatOver from "./Base/FloatOver.svelte" | ||||||
|     import Walkthrough from "./Walkthrough/Walkthrough.svelte" |     import Walkthrough from "./Walkthrough/Walkthrough.svelte" | ||||||
|     import * as intro from "../assets/studio_introduction.json" |     import * as intro from "../assets/studio_introduction.json" | ||||||
|  | @ -29,6 +28,7 @@ | ||||||
|     import { Utils } from "../Utils" |     import { Utils } from "../Utils" | ||||||
|     import Translations from "./i18n/Translations" |     import Translations from "./i18n/Translations" | ||||||
|     import Tr from "./Base/Tr.svelte" |     import Tr from "./Base/Tr.svelte" | ||||||
|  |     import Add from "../assets/svg/Add.svelte" | ||||||
| 
 | 
 | ||||||
|     export let studioUrl = |     export let studioUrl = | ||||||
|     window.location.hostname === "127.0.0.2" |     window.location.hostname === "127.0.0.2" | ||||||
|  | @ -196,8 +196,8 @@ | ||||||
|             <QuestionMarkCircleIcon class="h-6 w-6" /> |             <QuestionMarkCircleIcon class="h-6 w-6" /> | ||||||
|             Show the introduction again |             Show the introduction again | ||||||
|           </button> |           </button> | ||||||
|           <a class="button flex" href={Utils.HomepageLink()}> |           <a class="flex button" href={Utils.HomepageLink()}> | ||||||
|             <img class="h-6 w-6" src="./assets/svg/add.svg" /> |             <Add class="h-6 w-6" /> | ||||||
|             <Tr t={Translations.t.general.backToIndex} /> |             <Tr t={Translations.t.general.backToIndex} /> | ||||||
|           </a> |           </a> | ||||||
|         </div> |         </div> | ||||||
|  |  | ||||||
|  | @ -2,6 +2,7 @@ | ||||||
|     import Svg from "../Svg" |     import Svg from "../Svg" | ||||||
|     import Loading from "./Base/Loading.svelte" |     import Loading from "./Base/Loading.svelte" | ||||||
|     import ToSvelte from "./Base/ToSvelte.svelte" |     import ToSvelte from "./Base/ToSvelte.svelte" | ||||||
|  |     import Community from "../assets/svg/Community.svelte" | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <div> | <div> | ||||||
|  | @ -40,16 +41,16 @@ | ||||||
| 
 | 
 | ||||||
|     <div class="flex"> |     <div class="flex"> | ||||||
|       <button class="primary"> |       <button class="primary"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Main action |         Main action | ||||||
|       </button> |       </button> | ||||||
|       <button class="primary disabled"> |       <button class="primary disabled"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Main action (disabled) |         Main action (disabled) | ||||||
|       </button> |       </button> | ||||||
| 
 | 
 | ||||||
|       <button class="small"> |       <button class="small"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Small button |         Small button | ||||||
|       </button> |       </button> | ||||||
| 
 | 
 | ||||||
|  | @ -58,11 +59,11 @@ | ||||||
|     </div> |     </div> | ||||||
|     <div class="flex"> |     <div class="flex"> | ||||||
|       <button> |       <button> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Secondary action |         Secondary action | ||||||
|       </button> |       </button> | ||||||
|       <button class="disabled"> |       <button class="disabled"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Secondary action (disabled) |         Secondary action (disabled) | ||||||
|       </button> |       </button> | ||||||
|     </div> |     </div> | ||||||
|  | @ -81,7 +82,7 @@ | ||||||
|       </label> |       </label> | ||||||
|       <label for="javascript"> |       <label for="javascript"> | ||||||
|         <input id="javascript" name="fav_language" type="radio" value="JavaScript" /> |         <input id="javascript" name="fav_language" type="radio" value="JavaScript" /> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-8 h-8")} /> |         <Community class="w-8 h-8"/> | ||||||
|         JavaScript |         JavaScript | ||||||
|       </label> |       </label> | ||||||
|     </div> |     </div> | ||||||
|  | @ -105,26 +106,26 @@ | ||||||
| 
 | 
 | ||||||
|     <div class="flex"> |     <div class="flex"> | ||||||
|       <button class="primary"> |       <button class="primary"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Main action |         Main action | ||||||
|       </button> |       </button> | ||||||
|       <button class="primary disabled"> |       <button class="primary disabled"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Main action (disabled) |         Main action (disabled) | ||||||
|       </button> |       </button> | ||||||
|       <button class="small"> |       <button class="small"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Small button |         Small button | ||||||
|       </button> |       </button> | ||||||
|     </div> |     </div> | ||||||
| 
 | 
 | ||||||
|     <div class="flex"> |     <div class="flex"> | ||||||
|       <button> |       <button> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Secondary action |         Secondary action | ||||||
|       </button> |       </button> | ||||||
|       <button class="disabled"> |       <button class="disabled"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6" /> | ||||||
|         Secondary action (disabled) |         Secondary action (disabled) | ||||||
|       </button> |       </button> | ||||||
|     </div> |     </div> | ||||||
|  |  | ||||||
|  | @ -1,16 +1,6 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   // Testing grounds |   // Testing grounds | ||||||
|   import { UIEventSource } from "../Logic/UIEventSource" |  | ||||||
|   import TabbedGroup from "./Base/TabbedGroup.svelte" |  | ||||||
| 
 |  | ||||||
|   let tab = new UIEventSource(1) |  | ||||||
|   console.log("Tab control", tab) |  | ||||||
| </script> | </script> | ||||||
| 
 | <div class="w-full"> | ||||||
| <TabbedGroup {tab}> |   No tests | ||||||
|   <div slot="title0">Title 0</div> | </div> | ||||||
|   <div slot="content0">Content 0 loaded</div> |  | ||||||
| 
 |  | ||||||
|   <div slot="title1">Title 1</div> |  | ||||||
|   <div slot="content1">Content 1</div> |  | ||||||
| </TabbedGroup> |  | ||||||
|  |  | ||||||
|  | @ -36,25 +36,34 @@ | ||||||
|     import LevelSelector from "./BigComponents/LevelSelector.svelte" |     import LevelSelector from "./BigComponents/LevelSelector.svelte" | ||||||
|     import ExtraLinkButton from "./BigComponents/ExtraLinkButton" |     import ExtraLinkButton from "./BigComponents/ExtraLinkButton" | ||||||
|     import SelectedElementTitle from "./BigComponents/SelectedElementTitle.svelte" |     import SelectedElementTitle from "./BigComponents/SelectedElementTitle.svelte" | ||||||
|   import Svg from "../Svg" |  | ||||||
|     import ThemeIntroPanel from "./BigComponents/ThemeIntroPanel.svelte" |     import ThemeIntroPanel from "./BigComponents/ThemeIntroPanel.svelte" | ||||||
|     import type { RasterLayerPolygon } from "../Models/RasterLayers" |     import type { RasterLayerPolygon } from "../Models/RasterLayers" | ||||||
|     import { AvailableRasterLayers } from "../Models/RasterLayers" |     import { AvailableRasterLayers } from "../Models/RasterLayers" | ||||||
|     import RasterLayerOverview from "./Map/RasterLayerOverview.svelte" |     import RasterLayerOverview from "./Map/RasterLayerOverview.svelte" | ||||||
|     import IfHidden from "./Base/IfHidden.svelte" |     import IfHidden from "./Base/IfHidden.svelte" | ||||||
|     import { onDestroy } from "svelte" |     import { onDestroy } from "svelte" | ||||||
|   import { OpenJosm } from "./BigComponents/OpenJosm" |  | ||||||
|     import MapillaryLink from "./BigComponents/MapillaryLink.svelte" |     import MapillaryLink from "./BigComponents/MapillaryLink.svelte" | ||||||
|     import OpenIdEditor from "./BigComponents/OpenIdEditor.svelte" |     import OpenIdEditor from "./BigComponents/OpenIdEditor.svelte" | ||||||
|     import OpenBackgroundSelectorButton from "./BigComponents/OpenBackgroundSelectorButton.svelte" |     import OpenBackgroundSelectorButton from "./BigComponents/OpenBackgroundSelectorButton.svelte" | ||||||
|     import StateIndicator from "./BigComponents/StateIndicator.svelte" |     import StateIndicator from "./BigComponents/StateIndicator.svelte" | ||||||
|   import LanguagePicker from "./LanguagePicker" |  | ||||||
|   import Locale from "./i18n/Locale" |  | ||||||
|     import ShareScreen from "./BigComponents/ShareScreen.svelte" |     import ShareScreen from "./BigComponents/ShareScreen.svelte" | ||||||
|     import UploadingImageCounter from "./Image/UploadingImageCounter.svelte" |     import UploadingImageCounter from "./Image/UploadingImageCounter.svelte" | ||||||
|     import PendingChangesIndicator from "./BigComponents/PendingChangesIndicator.svelte" |     import PendingChangesIndicator from "./BigComponents/PendingChangesIndicator.svelte" | ||||||
|     import Cross from "../assets/svg/Cross.svelte" |     import Cross from "../assets/svg/Cross.svelte" | ||||||
|     import Summary from "./BigComponents/Summary.svelte" |     import Summary from "./BigComponents/Summary.svelte" | ||||||
|  |     import Mastodon from "../assets/svg/Mastodon.svelte" | ||||||
|  |     import Bug from "../assets/svg/Bug.svelte" | ||||||
|  |     import Liberapay from "../assets/svg/Liberapay.svelte" | ||||||
|  |     import Min from "../assets/svg/Min.svelte" | ||||||
|  |     import Plus from "../assets/svg/Plus.svelte" | ||||||
|  |     import Filter from "../assets/svg/Filter.svelte" | ||||||
|  |     import Add from "../assets/svg/Add.svelte" | ||||||
|  |     import Statistics from "../assets/svg/Statistics.svelte" | ||||||
|  |     import Community from "../assets/svg/Community.svelte" | ||||||
|  |     import Download from "../assets/svg/Download.svelte" | ||||||
|  |     import Share from "../assets/svg/Share.svelte" | ||||||
|  |     import LanguagePicker from "./InputElement/LanguagePicker.svelte" | ||||||
|  |     import OpenJosm from "./Base/OpenJosm.svelte" | ||||||
| 
 | 
 | ||||||
|     export let state: ThemeViewState |     export let state: ThemeViewState | ||||||
|   let layout = state.layout |   let layout = state.layout | ||||||
|  | @ -205,7 +214,7 @@ | ||||||
|         <!-- bottom left elements --> |         <!-- bottom left elements --> | ||||||
|         <If condition={state.featureSwitches.featureSwitchFilter}> |         <If condition={state.featureSwitches.featureSwitchFilter}> | ||||||
|           <MapControlButton on:click={() => state.guistate.openFilterView()}> |           <MapControlButton on:click={() => state.guistate.openFilterView()}> | ||||||
|             <ToSvelte construct={Svg.filter_svg().SetClass("h-6 w-6")} /> |             <Filter class="h-6 w-6"/> | ||||||
|           </MapControlButton> |           </MapControlButton> | ||||||
|         </If> |         </If> | ||||||
|         <If condition={state.featureSwitches.featureSwitchBackgroundSelection}> |         <If condition={state.featureSwitches.featureSwitchBackgroundSelection}> | ||||||
|  | @ -245,10 +254,10 @@ | ||||||
|         </div> |         </div> | ||||||
|       </If> |       </If> | ||||||
|       <MapControlButton on:click={() => mapproperties.zoom.update((z) => z + 1)}> |       <MapControlButton on:click={() => mapproperties.zoom.update((z) => z + 1)}> | ||||||
|         <ToSvelte construct={Svg.plus_svg().SetClass("w-8 h-8")} /> |         <Plus class="w-8 h-8" /> | ||||||
|       </MapControlButton> |       </MapControlButton> | ||||||
|       <MapControlButton on:click={() => mapproperties.zoom.update((z) => z - 1)}> |       <MapControlButton on:click={() => mapproperties.zoom.update((z) => z - 1)}> | ||||||
|         <ToSvelte construct={Svg.min_svg().SetClass("w-8 h-8")} /> |         <Min class="w-8 h-8"/> | ||||||
|       </MapControlButton> |       </MapControlButton> | ||||||
|       <If condition={featureSwitches.featureSwitchGeolocation}> |       <If condition={featureSwitches.featureSwitchGeolocation}> | ||||||
|         <MapControlButton> |         <MapControlButton> | ||||||
|  | @ -264,7 +273,7 @@ | ||||||
| </div> | </div> | ||||||
| 
 | 
 | ||||||
| <LoginToggle ignoreLoading={true} {state}> | <LoginToggle ignoreLoading={true} {state}> | ||||||
|   {#if $showCrosshair === "yes" && ($currentZoom >= 17 || $arrowKeysWereUsed !== undefined)} |   {#if ($showCrosshair === "yes" && $currentZoom >= 17) || $showCrosshair === "always" || $arrowKeysWereUsed !== undefined } | ||||||
|     <div |     <div | ||||||
|       class="pointer-events-none absolute top-0 left-0 flex h-full w-full items-center justify-center" |       class="pointer-events-none absolute top-0 left-0 flex h-full w-full items-center justify-center" | ||||||
|     > |     > | ||||||
|  | @ -341,7 +350,7 @@ | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <div class="flex" slot="title1"> |       <div class="flex" slot="title1"> | ||||||
|         <ToSvelte construct={Svg.filter_svg().SetClass("w-4 h-4")} /> |         <Filter class="w-4 h-4"/> | ||||||
|         <Tr t={Translations.t.general.menu.filter} /> |         <Tr t={Translations.t.general.menu.filter} /> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|  | @ -365,7 +374,7 @@ | ||||||
| 
 | 
 | ||||||
|       <div class="flex" slot="title2"> |       <div class="flex" slot="title2"> | ||||||
|         <If condition={state.featureSwitches.featureSwitchEnableExport}> |         <If condition={state.featureSwitches.featureSwitchEnableExport}> | ||||||
|           <ToSvelte construct={Svg.download_svg().SetClass("w-4 h-4")} /> |           <Download class="w-4 h-4"/> | ||||||
|           <Tr t={Translations.t.general.download.title} /> |           <Tr t={Translations.t.general.download.title} /> | ||||||
|         </If> |         </If> | ||||||
|       </div> |       </div> | ||||||
|  | @ -380,7 +389,7 @@ | ||||||
|       <ToSvelte construct={() => new CopyrightPanel(state)} slot="content3" /> |       <ToSvelte construct={() => new CopyrightPanel(state)} slot="content3" /> | ||||||
| 
 | 
 | ||||||
|       <div class="flex" slot="title4"> |       <div class="flex" slot="title4"> | ||||||
|         <ToSvelte construct={Svg.share_svg().SetClass("w-4 h-4")} /> |         <Share class="w-4 h-4"/> | ||||||
|         <Tr t={Translations.t.general.sharescreen.title} /> |         <Tr t={Translations.t.general.sharescreen.title} /> | ||||||
|       </div> |       </div> | ||||||
|       <div class="m-2" slot="content4"> |       <div class="m-2" slot="content4"> | ||||||
|  | @ -432,27 +441,27 @@ | ||||||
|         <Tr t={Translations.t.general.aboutMapComplete.intro} /> |         <Tr t={Translations.t.general.aboutMapComplete.intro} /> | ||||||
| 
 | 
 | ||||||
|         <a class="flex" href={Utils.HomepageLink()}> |         <a class="flex" href={Utils.HomepageLink()}> | ||||||
|           <img class="h-6 w-6" src="./assets/svg/add.svg" /> |           <Add class="h-6 w-6"/> | ||||||
|           <Tr t={Translations.t.general.backToIndex} /> |           <Tr t={Translations.t.general.backToIndex} /> | ||||||
|         </a> |         </a> | ||||||
| 
 | 
 | ||||||
|         <a class="flex" href="https://github.com/pietervdvn/MapComplete/issues" target="_blank"> |         <a class="flex" href="https://github.com/pietervdvn/MapComplete/issues" target="_blank"> | ||||||
|           <img class="h-6 w-6" src="./assets/svg/bug.svg" /> |           <Bug class="h-6 w-6"/> | ||||||
|           <Tr t={Translations.t.general.attribution.openIssueTracker} /> |           <Tr t={Translations.t.general.attribution.openIssueTracker} /> | ||||||
|         </a> |         </a> | ||||||
| 
 | 
 | ||||||
|         <a class="flex" href="https://en.osm.town/@MapComplete" target="_blank"> |         <a class="flex" href="https://en.osm.town/@MapComplete" target="_blank"> | ||||||
|           <img class="h-6 w-6" src="./assets/svg/mastodon.svg" /> |           <Mastodon class="w-6 h-6" /> | ||||||
|           <Tr t={Translations.t.general.attribution.followOnMastodon} /> |           <Tr t={Translations.t.general.attribution.followOnMastodon} /> | ||||||
|         </a> |         </a> | ||||||
| 
 | 
 | ||||||
|         <a class="flex" href="https://liberapay.com/pietervdvn/" target="_blank"> |         <a class="flex" href="https://liberapay.com/pietervdvn/" target="_blank"> | ||||||
|           <img class="h-6 w-6" src="./assets/svg/liberapay.svg" /> |           <Liberapay class="h-6 w-6" /> | ||||||
|           <Tr t={Translations.t.general.attribution.donate} /> |           <Tr t={Translations.t.general.attribution.donate} /> | ||||||
|         </a> |         </a> | ||||||
| 
 | 
 | ||||||
|         <a class="flex" href={Utils.OsmChaLinkFor(7)} target="_blank"> |         <a class="flex" href={Utils.OsmChaLinkFor(7)} target="_blank"> | ||||||
|           <img class="h-6 w-6" src="./assets/svg/statistics.svg" /> |           <Statistics class="h-6 w-6" /> | ||||||
|           <Tr t={Translations.t.general.attribution.openOsmcha.Subs({ theme: "MapComplete" })} /> |           <Tr t={Translations.t.general.attribution.openOsmcha.Subs({ theme: "MapComplete" })} /> | ||||||
|         </a> |         </a> | ||||||
|         {Constants.vNumber} |         {Constants.vNumber} | ||||||
|  | @ -467,7 +476,7 @@ | ||||||
|         <!-- All shown components are set by 'usersettings.json', which happily uses some special visualisations created specifically for it --> |         <!-- All shown components are set by 'usersettings.json', which happily uses some special visualisations created specifically for it --> | ||||||
|         <LoginToggle {state}> |         <LoginToggle {state}> | ||||||
|           <div class="flex flex-col" slot="not-logged-in"> |           <div class="flex flex-col" slot="not-logged-in"> | ||||||
|             <ToSvelte construct={() => new LanguagePicker(layout.language, Locale.language)} /> |             <LanguagePicker availableLanguages={layout.language} /> | ||||||
|             <Tr cls="alert" t={Translations.t.userinfo.notLoggedIn} /> |             <Tr cls="alert" t={Translations.t.userinfo.notLoggedIn} /> | ||||||
|             <LoginButton clss="primary" osmConnection={state.osmConnection} /> |             <LoginButton clss="primary" osmConnection={state.osmConnection} /> | ||||||
|           </div> |           </div> | ||||||
|  | @ -486,7 +495,7 @@ | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <div class="flex" slot="title2"> |       <div class="flex" slot="title2"> | ||||||
|         <ToSvelte construct={Svg.community_svg().SetClass("w-6 h-6")} /> |         <Community class="w-6 h-6"/> | ||||||
|         <Tr t={Translations.t.communityIndex.title} /> |         <Tr t={Translations.t.communityIndex.title} /> | ||||||
|       </div> |       </div> | ||||||
|       <div class="m-2" slot="content2"> |       <div class="m-2" slot="content2"> | ||||||
|  | @ -504,10 +513,7 @@ | ||||||
|       <div class="m-2 flex flex-col" slot="content4"> |       <div class="m-2 flex flex-col" slot="content4"> | ||||||
|         <If condition={featureSwitches.featureSwitchEnableLogin}> |         <If condition={featureSwitches.featureSwitchEnableLogin}> | ||||||
|           <OpenIdEditor mapProperties={state.mapProperties} /> |           <OpenIdEditor mapProperties={state.mapProperties} /> | ||||||
|           <ToSvelte |           <OpenJosm {state}/> | ||||||
|             construct={() => |  | ||||||
|               new OpenJosm(state.osmConnection, state.mapProperties.bounds).SetClass("w-full")} |  | ||||||
|           /> |  | ||||||
|           <MapillaryLink mapProperties={state.mapProperties} /> |           <MapillaryLink mapProperties={state.mapProperties} /> | ||||||
|         </If> |         </If> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -28,6 +28,8 @@ | ||||||
|   on:next={() => step(1)} |   on:next={() => step(1)} | ||||||
|   isFirst={currentPage === 0} |   isFirst={currentPage === 0} | ||||||
|   islast={currentPage + 1 === pages.length} |   islast={currentPage + 1 === pages.length} | ||||||
|  |   totalPages={pages.length} | ||||||
|  |   pageNumber={currentPage} | ||||||
| > | > | ||||||
|   <FromHtml src={nmd(pages[currentPage])} /> |   <FromHtml src={nmd(pages[currentPage])} /> | ||||||
| </WalkthroughStep> | </WalkthroughStep> | ||||||
|  |  | ||||||
|  | @ -1,18 +1,27 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import BackButton from "../Base/BackButton.svelte" |   import BackButton from "../Base/BackButton.svelte"; | ||||||
|   import NextButton from "../Base/NextButton.svelte" |   import NextButton from "../Base/NextButton.svelte"; | ||||||
|   import { createEventDispatcher } from "svelte" |   import { createEventDispatcher } from "svelte"; | ||||||
| 
 | 
 | ||||||
|   const dispatch = createEventDispatcher<{ back; next }>() |   const dispatch = createEventDispatcher<{ back; next }>(); | ||||||
|   export let islast = false |   export let islast = false; | ||||||
|   export let isFirst = false |   export let isFirst = false; | ||||||
|  |   export let pageNumber: number = undefined; | ||||||
|  |   export let totalPages: number = undefined; | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <div class="flex h-full w-full flex-col justify-between"> | <div class="flex h-full w-full flex-col justify-between link-underline"> | ||||||
|   <div class="overflow-y-auto"> |   <div class="overflow-y-auto"> | ||||||
|     <slot /> |     <slot /> | ||||||
|   </div> |   </div> | ||||||
| 
 | 
 | ||||||
|  |   <div class="flex flex-col"> | ||||||
|  | 
 | ||||||
|  |     {#if pageNumber !== undefined && totalPages !== undefined} | ||||||
|  |       <div class="flex justify-end"> | ||||||
|  |       <div class="subtle">{pageNumber+1}/{totalPages}</div> | ||||||
|  |       </div> | ||||||
|  |     {/if} | ||||||
|     <div class="flex w-full"> |     <div class="flex w-full"> | ||||||
|       {#if !isFirst} |       {#if !isFirst} | ||||||
|         <BackButton clss="w-full" on:click={() => dispatch("back")}>Back</BackButton> |         <BackButton clss="w-full" on:click={() => dispatch("back")}>Back</BackButton> | ||||||
|  | @ -26,5 +35,8 @@ | ||||||
|           Next |           Next | ||||||
|         {/if} |         {/if} | ||||||
|       </NextButton> |       </NextButton> | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     </div> | ||||||
|   </div> |   </div> | ||||||
| </div> | </div> | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ | ||||||
|   import WikidataPreviewBox from "./WikidataPreviewBox" |   import WikidataPreviewBox from "./WikidataPreviewBox" | ||||||
|   import Tr from "../Base/Tr.svelte" |   import Tr from "../Base/Tr.svelte" | ||||||
|   import Translations from "../i18n/Translations" |   import Translations from "../i18n/Translations" | ||||||
|  |   import Wikipedia from "../../assets/svg/Wikipedia.svelte"; | ||||||
| 
 | 
 | ||||||
|   /** |   /** | ||||||
|    * Shows a wikipedia-article + wikidata preview for the given item |    * Shows a wikipedia-article + wikidata preview for the given item | ||||||
|  | @ -18,7 +19,7 @@ | ||||||
| 
 | 
 | ||||||
| {#if $wikipediaDetails.articleUrl} | {#if $wikipediaDetails.articleUrl} | ||||||
|   <a class="flex" href={$wikipediaDetails.articleUrl} rel="noreferrer" target="_blank"> |   <a class="flex" href={$wikipediaDetails.articleUrl} rel="noreferrer" target="_blank"> | ||||||
|     <img class="h-6 w-6" src="./assets/svg/wikipedia.svg" /> |     <Wikipedia class="h-6 w-6"/> | ||||||
|     <Tr t={Translations.t.general.wikipedia.fromWikipedia} /> |     <Tr t={Translations.t.general.wikipedia.fromWikipedia} /> | ||||||
|   </a> |   </a> | ||||||
| {/if} | {/if} | ||||||
|  |  | ||||||
|  | @ -56,6 +56,9 @@ export default class Locale { | ||||||
|             if (typeof navigator !== "undefined") { |             if (typeof navigator !== "undefined") { | ||||||
|                 browserLanguage = navigator.languages?.[0] ?? navigator.language ?? "en" |                 browserLanguage = navigator.languages?.[0] ?? navigator.language ?? "en" | ||||||
|                 console.log("Browser language is", browserLanguage) |                 console.log("Browser language is", browserLanguage) | ||||||
|  |                 if (browserLanguage === "en-US") { | ||||||
|  |                     browserLanguage = "en" | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             source = LocalStorageSource.Get("language", browserLanguage) |             source = LocalStorageSource.Get("language", browserLanguage) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -999,7 +999,7 @@ class SvgToPdfPage { | ||||||
| 
 | 
 | ||||||
| export interface PdfTemplateInfo { | export interface PdfTemplateInfo { | ||||||
|     pages: string[] |     pages: string[] | ||||||
|     description: string | Translation |     description?: string | Translation | ||||||
|     format: "a3" | "a4" | "a2" |     format: "a3" | "a4" | "a2" | ||||||
|     orientation: "portrait" | "landscape" |     orientation: "portrait" | "landscape" | ||||||
|     isPublic: boolean |     isPublic: boolean | ||||||
|  | @ -1043,22 +1043,18 @@ export class SvgToPdf { | ||||||
|             format: "a4", |             format: "a4", | ||||||
|             orientation: "landscape", |             orientation: "landscape", | ||||||
|             pages: ["./assets/templates/CurrentMapWithHeaderA4.svg"], |             pages: ["./assets/templates/CurrentMapWithHeaderA4.svg"], | ||||||
|             description: Translations.t.general.download.pdf.current_view_a4, |  | ||||||
| 
 |  | ||||||
|             isPublic: true, |             isPublic: true, | ||||||
|         }, |         }, | ||||||
|         current_view_a3_landscape: { |         current_view_a3_landscape: { | ||||||
|             format: "a3", |             format: "a3", | ||||||
|             orientation: "landscape", |             orientation: "landscape", | ||||||
|             pages: ["./assets/templates/CurrentMapWithHeader_A3_Landscape.svg"], |             pages: ["./assets/templates/CurrentMapWithHeader_A3_Landscape.svg"], | ||||||
|             description: Translations.t.general.download.pdf.current_view_a3, |  | ||||||
|             isPublic: true, |             isPublic: true, | ||||||
|         }, |         }, | ||||||
|         current_view_a3_portrait: { |         current_view_a3_portrait: { | ||||||
|             format: "a3", |             format: "a3", | ||||||
|             orientation: "portrait", |             orientation: "portrait", | ||||||
|             pages: ["./assets/templates/CurrentMapWithHeader_A3_Portrait.svg"], |             pages: ["./assets/templates/CurrentMapWithHeader_A3_Portrait.svg"], | ||||||
|             description: Translations.t.general.download.pdf.current_view_a3, |  | ||||||
|             isPublic: true, |             isPublic: true, | ||||||
|         }, |         }, | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| import AllThemesGui from "./UI/AllThemesGui" |  | ||||||
| import { QueryParameters } from "./Logic/Web/QueryParameters" | import { QueryParameters } from "./Logic/Web/QueryParameters" | ||||||
|  | import SvelteUIElement from "./UI/Base/SvelteUIElement" | ||||||
|  | import AllThemesGui from "./UI/AllThemesGui.svelte" | ||||||
| 
 | 
 | ||||||
| const layout = QueryParameters.GetQueryParameter("layout", undefined).data ?? "" | const layout = QueryParameters.GetQueryParameter("layout", undefined).data ?? "" | ||||||
| const customLayout = QueryParameters.GetQueryParameter("userlayout", undefined).data ?? "" | const customLayout = QueryParameters.GetQueryParameter("userlayout", undefined).data ?? "" | ||||||
|  | @ -27,4 +28,4 @@ if (layout !== "") { | ||||||
|     ) |     ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| new AllThemesGui().setup() | new SvelteUIElement(AllThemesGui, {}).AttachTo("main") | ||||||
|  |  | ||||||
|  | @ -1,38 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="303px" height="374px" viewBox="0 0 303 374" version="1.1">   <g id="surface1">     <path style=" stroke:none;fill-rule:nonzero;fill:rgb(43.921569%,77.254903%,28.627452%);fill-opacity:1;" d="M 163.890625 344.988281 C 158.277344 356.675781 141.632812 356.675781 136.019531 344.988281 L 41.4375 148.097656 C 36.507812 137.839844 43.988281 125.953125 55.375 125.953125 L 244.535156 125.953125 C 255.917969 125.953125 263.398438 137.839844 258.46875 148.097656 Z M 163.890625 344.988281 "/>     <path style=" stroke:none;fill-rule:nonzero;fill:rgb(43.921569%,77.254903%,28.627452%);fill-opacity:1;" d="M 303 151.453125 C 303 235.101562 235.171875 302.910156 151.5 302.910156 C 67.828125 302.910156 0 235.101562 0 151.453125 C 0 67.808594 67.828125 0 151.5 0 C 235.171875 0 303 67.808594 303 151.453125 Z M 303 151.453125 "/>     <path style="fill: none !important;stroke-width:7.514114;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" d="M 22.101015 291.358766 L 5.785842 275.043593 " transform="matrix(4.837809,-4.836357,4.837809,4.836357,-1287.063942,-1153.71106)"/>     <path style="fill: none !important;stroke-width:7.514114;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" d="M 22.125279 274.965231 L 5.810459 291.28005 " transform="matrix(4.837809,-4.836357,4.837809,4.836357,-1287.063942,-1153.71106)"/>   </g> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="303px" |  | ||||||
|   height="374px" |  | ||||||
|   viewBox="0 0 303 374" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:rgb(43.921569%,77.254903%,28.627452%);fill-opacity:1;" |  | ||||||
|       d="M 163.890625 344.988281 C 158.277344 356.675781 141.632812 356.675781 136.019531 344.988281 L 41.4375 148.097656 C 36.507812 137.839844 43.988281 125.953125 55.375 125.953125 L 244.535156 125.953125 C 255.917969 125.953125 263.398438 137.839844 258.46875 148.097656 Z M 163.890625 344.988281 " |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:rgb(43.921569%,77.254903%,28.627452%);fill-opacity:1;" |  | ||||||
|       d="M 303 151.453125 C 303 235.101562 235.171875 302.910156 151.5 302.910156 C 67.828125 302.910156 0 235.101562 0 151.453125 C 0 67.808594 67.828125 0 151.5 0 C 235.171875 0 303 67.808594 303 151.453125 Z M 303 151.453125 " |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:7.514114;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 22.101015 291.358766 L 5.785842 275.043593 " |  | ||||||
|       transform="matrix(4.837809,-4.836357,4.837809,4.836357,-1287.063942,-1153.71106)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:7.514114;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 22.125279 274.965231 L 5.810459 291.28005 " |  | ||||||
|       transform="matrix(4.837809,-4.836357,4.837809,4.836357,-1287.063942,-1153.71106)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,29 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style="fill-rule:nonzero;fill:rgb(21.568628%,83.92157%,28.627452%);fill-opacity:1;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke:rgb(37.64706%,37.64706%,37.64706%);stroke-opacity:1;stroke-miterlimit:4;" d="M 98 49.021438 C 98 76.083729 76.062292 98.021438 49 98.021438 C 21.937708 98.021438 0 76.083729 0 49.021438 C 0 21.959146 21.937708 0.0214375 49 0.0214375 C 76.062292 0.0214375 98 21.959146 98 49.021438 Z M 98 49.021438 " transform="matrix(3.826531,0,0,3.826531,0,0)"/>     <path style=" stroke:none;fill-rule:nonzero;fill:rgb(100%,100%,100%);fill-opacity:1;" d="M 185.398438 53.390625 C 176.808594 53.523438 168.636719 57.125 162.738281 63.378906 C 157.074219 69.394531 153.96875 77.378906 154.085938 85.640625 L 154.085938 152.132812 L 88.527344 152.132812 C 79.929688 152.011719 71.644531 155.371094 65.566406 161.453125 C 56.40625 170.613281 53.710938 184.414062 58.761719 196.347656 C 63.808594 208.28125 75.585938 215.960938 88.539062 215.765625 L 154.085938 215.765625 L 154.085938 281.003906 C 153.898438 292.488281 159.917969 303.183594 169.835938 308.984375 C 179.753906 314.785156 192.027344 314.785156 201.945312 308.992188 C 211.863281 303.199219 217.890625 292.503906 217.707031 281.019531 L 217.707031 215.765625 L 283.894531 215.765625 C 295.296875 215.820312 305.855469 209.769531 311.574219 199.90625 C 317.292969 190.042969 317.296875 177.871094 311.582031 168.007812 C 305.871094 158.140625 295.3125 152.085938 283.910156 152.132812 L 217.703125 152.132812 L 217.703125 85.640625 C 217.820312 77.042969 214.453125 68.765625 208.367188 62.6875 C 202.28125 56.613281 193.996094 53.261719 185.398438 53.390625 Z M 185.398438 53.390625 "/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill-rule:nonzero;fill:rgb(21.568628%,83.92157%,28.627452%);fill-opacity:1;stroke-width:1;stroke-linecap:butt;stroke-linejoin:miter;stroke:rgb(37.64706%,37.64706%,37.64706%);stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 98 49.021438 C 98 76.083729 76.062292 98.021438 49 98.021438 C 21.937708 98.021438 0 76.083729 0 49.021438 C 0 21.959146 21.937708 0.0214375 49 0.0214375 C 76.062292 0.0214375 98 21.959146 98 49.021438 Z M 98 49.021438 " |  | ||||||
|       transform="matrix(3.826531,0,0,3.826531,0,0)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:rgb(100%,100%,100%);fill-opacity:1;" |  | ||||||
|       d="M 185.398438 53.390625 C 176.808594 53.523438 168.636719 57.125 162.738281 63.378906 C 157.074219 69.394531 153.96875 77.378906 154.085938 85.640625 L 154.085938 152.132812 L 88.527344 152.132812 C 79.929688 152.011719 71.644531 155.371094 65.566406 161.453125 C 56.40625 170.613281 53.710938 184.414062 58.761719 196.347656 C 63.808594 208.28125 75.585938 215.960938 88.539062 215.765625 L 154.085938 215.765625 L 154.085938 281.003906 C 153.898438 292.488281 159.917969 303.183594 169.835938 308.984375 C 179.753906 314.785156 192.027344 314.785156 201.945312 308.992188 C 211.863281 303.199219 217.890625 292.503906 217.707031 281.019531 L 217.707031 215.765625 L 283.894531 215.765625 C 295.296875 215.820312 305.855469 209.769531 311.574219 199.90625 C 317.292969 190.042969 317.296875 177.871094 311.582031 168.007812 C 305.871094 158.140625 295.3125 152.085938 283.910156 152.132812 L 217.703125 152.132812 L 217.703125 85.640625 C 217.820312 77.042969 214.453125 68.765625 208.367188 62.6875 C 202.28125 56.613281 193.996094 53.261719 185.398438 53.390625 Z M 185.398438 53.390625 " |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,24 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="303px" height="374px" viewBox="0 0 303 374" version="1.1">   <g id="surface1">     <path style=" stroke:none;fill-rule:nonzero;fill:rgb(43.921569%,77.254903%,28.627452%);fill-opacity:1;" d="M 151.5 0 C 67.828125 0 0 67.808594 0 151.453125 C 0.0742188 203.539062 26.914062 251.933594 71.0625 279.589844 C 83.773438 287.726562 98.976562 305.320312 111.8125 321.363281 C 125.679688 338.699219 133.386719 374.292969 150.203125 374.5625 C 167.015625 374.292969 174.730469 338.699219 188.597656 321.363281 C 199.703125 307.480469 211.496094 292.414062 224.496094 283.984375 C 272.859375 257.398438 302.933594 206.628906 303 151.453125 C 303 67.808594 235.171875 0 151.5 0 Z M 151.5 0 "/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="303px" |  | ||||||
|   height="374px" |  | ||||||
|   viewBox="0 0 303 374" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:rgb(43.921569%,77.254903%,28.627452%);fill-opacity:1;" |  | ||||||
|       d="M 151.5 0 C 67.828125 0 0 67.808594 0 151.453125 C 0.0742188 203.539062 26.914062 251.933594 71.0625 279.589844 C 83.773438 287.726562 98.976562 305.320312 111.8125 321.363281 C 125.679688 338.699219 133.386719 374.292969 150.203125 374.5625 C 167.015625 374.292969 174.730469 338.699219 188.597656 321.363281 C 199.703125 307.480469 211.496094 292.414062 224.496094 283.984375 C 272.859375 257.398438 302.933594 206.628906 303 151.453125 C 303 67.808594 235.171875 0 151.5 0 Z M 151.5 0 " |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,30 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style="fill: none !important;stroke-width:5.291667;stroke-linecap:round;stroke-linejoin:round;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 14.287499 23.618748 L 4.143209 13.229167 L 14.287499 2.871556 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,0)"/>     <path style="fill: none !important;stroke-width:3.96875;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 5.532271 13.229167 C 12.346946 13.21704 23.467714 13.190031 23.8125 13.200504 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,0)"/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:5.291667;stroke-linecap:round;stroke-linejoin:round;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 14.287499 23.618748 L 4.143209 13.229167 L 14.287499 2.871556 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,0)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:3.96875;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 5.532271 13.229167 C 12.346946 13.21704 23.467714 13.190031 23.8125 13.200504 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,0)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							|  | @ -1,24 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="268px" height="373px" viewBox="0 0 268 373" version="1.1">   <g id="surface1">     <path style=" stroke:none;fill-rule:nonzero;fill:{color};fill-opacity:1;" d="M 88.972656 57.019531 C 69.667969 77.699219 74.996094 110.734375 74.996094 110.734375 C 74.996094 110.734375 94.699219 134.046875 133.414062 134.046875 C 172.128906 134.046875 191.84375 110.734375 191.84375 110.734375 C 191.84375 110.734375 197.09375 78.144531 178.199219 57.4375 C 190.019531 50.871094 196.980469 41.378906 194.105469 34.921875 C 190.839844 27.648438 176.441406 27 161.882812 33.476562 C 156.472656 35.878906 151.863281 38.929688 148.28125 42.175781 C 143.742188 41.355469 138.925781 40.796875 133.414062 40.796875 C 128.28125 40.796875 123.714844 41.285156 119.414062 41.980469 C 115.886719 38.792969 111.359375 35.835938 106.074219 33.476562 C 91.515625 27.023438 77.09375 27.671875 73.851562 34.921875 C 71.023438 41.242188 77.609375 50.472656 88.972656 57.019531 Z M 235.433594 207.535156 C 232.417969 206.898438 229.542969 206.535156 226.761719 206.261719 C 226.761719 205.488281 226.898438 204.757812 226.898438 203.960938 C 226.898438 191.726562 225.164062 180.046875 222.355469 168.949219 C 228.359375 169.460938 236.003906 168.082031 243.738281 164.65625 C 258.296875 158.179688 267.472656 147.035156 264.230469 139.808594 C 260.992188 132.535156 246.566406 131.886719 232.007812 138.363281 C 225.207031 141.390625 219.664062 145.453125 215.945312 149.621094 C 213.113281 142.972656 209.917969 136.585938 206.152344 130.78125 C 195.132812 139.945312 174.273438 153.625 145.15625 156.664062 L 145.15625 273.875 C 145.15625 273.875 145.0625 285.53125 133.460938 285.53125 C 121.867188 285.53125 121.777344 273.875 121.777344 273.875 L 121.777344 156.710938 C 92.636719 153.671875 71.765625 139.964844 60.765625 130.824219 C 57.160156 136.402344 54.058594 142.480469 51.296875 148.800781 C 47.542969 144.929688 42.304688 141.207031 35.972656 138.382812 C 21.414062 131.929688 6.992188 132.578125 3.75 139.828125 C 0.5 147.082031 9.683594 158.203125 24.242188 164.679688 C 31.488281 167.902344 38.675781 169.347656 44.496094 169.085938 C 41.699219 180.160156 39.941406 191.792969 39.941406 203.984375 C 39.941406 204.78125 40.082031 205.535156 40.101562 206.351562 C 37.648438 206.625 35.105469 206.992188 32.515625 207.535156 C 13.539062 211.453125 -0.996094 221.242188 0.078125 229.324219 C 1.148438 237.453125 17.421875 240.796875 36.441406 236.859375 C 38.917969 236.359375 41.347656 235.722656 43.664062 235.015625 C 47.007812 249.882812 52.359375 263.675781 59.761719 275.5625 C 55.371094 277.769531 50.804688 281.023438 46.605469 285.257812 C 34.96875 296.871094 30.246094 310.984375 36.0625 316.789062 C 41.882812 322.597656 56.019531 317.882812 67.660156 306.296875 C 71.070312 302.878906 73.808594 299.285156 75.929688 295.730469 C 91.824219 311.257812 111.734375 320.546875 133.460938 320.546875 C 155.492188 320.546875 175.734375 310.984375 191.707031 295.070312 C 193.855469 298.828125 196.730469 302.652344 200.335938 306.25 C 211.949219 317.859375 226.121094 322.574219 231.941406 316.769531 C 237.757812 310.960938 233.035156 296.871094 221.398438 285.257812 C 216.902344 280.75 212.019531 277.335938 207.363281 275.082031 C 214.710938 263.246094 220.003906 249.449219 223.316406 234.628906 C 225.9375 235.445312 228.675781 236.222656 231.550781 236.835938 C 250.5625 240.773438 266.832031 237.40625 267.925781 229.300781 C 268.953125 221.242188 254.417969 211.453125 235.433594 207.535156 Z M 235.433594 207.535156 "/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="268px" |  | ||||||
|   height="373px" |  | ||||||
|   viewBox="0 0 268 373" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:{color};fill-opacity:1;" |  | ||||||
|       d="M 88.972656 57.019531 C 69.667969 77.699219 74.996094 110.734375 74.996094 110.734375 C 74.996094 110.734375 94.699219 134.046875 133.414062 134.046875 C 172.128906 134.046875 191.84375 110.734375 191.84375 110.734375 C 191.84375 110.734375 197.09375 78.144531 178.199219 57.4375 C 190.019531 50.871094 196.980469 41.378906 194.105469 34.921875 C 190.839844 27.648438 176.441406 27 161.882812 33.476562 C 156.472656 35.878906 151.863281 38.929688 148.28125 42.175781 C 143.742188 41.355469 138.925781 40.796875 133.414062 40.796875 C 128.28125 40.796875 123.714844 41.285156 119.414062 41.980469 C 115.886719 38.792969 111.359375 35.835938 106.074219 33.476562 C 91.515625 27.023438 77.09375 27.671875 73.851562 34.921875 C 71.023438 41.242188 77.609375 50.472656 88.972656 57.019531 Z M 235.433594 207.535156 C 232.417969 206.898438 229.542969 206.535156 226.761719 206.261719 C 226.761719 205.488281 226.898438 204.757812 226.898438 203.960938 C 226.898438 191.726562 225.164062 180.046875 222.355469 168.949219 C 228.359375 169.460938 236.003906 168.082031 243.738281 164.65625 C 258.296875 158.179688 267.472656 147.035156 264.230469 139.808594 C 260.992188 132.535156 246.566406 131.886719 232.007812 138.363281 C 225.207031 141.390625 219.664062 145.453125 215.945312 149.621094 C 213.113281 142.972656 209.917969 136.585938 206.152344 130.78125 C 195.132812 139.945312 174.273438 153.625 145.15625 156.664062 L 145.15625 273.875 C 145.15625 273.875 145.0625 285.53125 133.460938 285.53125 C 121.867188 285.53125 121.777344 273.875 121.777344 273.875 L 121.777344 156.710938 C 92.636719 153.671875 71.765625 139.964844 60.765625 130.824219 C 57.160156 136.402344 54.058594 142.480469 51.296875 148.800781 C 47.542969 144.929688 42.304688 141.207031 35.972656 138.382812 C 21.414062 131.929688 6.992188 132.578125 3.75 139.828125 C 0.5 147.082031 9.683594 158.203125 24.242188 164.679688 C 31.488281 167.902344 38.675781 169.347656 44.496094 169.085938 C 41.699219 180.160156 39.941406 191.792969 39.941406 203.984375 C 39.941406 204.78125 40.082031 205.535156 40.101562 206.351562 C 37.648438 206.625 35.105469 206.992188 32.515625 207.535156 C 13.539062 211.453125 -0.996094 221.242188 0.078125 229.324219 C 1.148438 237.453125 17.421875 240.796875 36.441406 236.859375 C 38.917969 236.359375 41.347656 235.722656 43.664062 235.015625 C 47.007812 249.882812 52.359375 263.675781 59.761719 275.5625 C 55.371094 277.769531 50.804688 281.023438 46.605469 285.257812 C 34.96875 296.871094 30.246094 310.984375 36.0625 316.789062 C 41.882812 322.597656 56.019531 317.882812 67.660156 306.296875 C 71.070312 302.878906 73.808594 299.285156 75.929688 295.730469 C 91.824219 311.257812 111.734375 320.546875 133.460938 320.546875 C 155.492188 320.546875 175.734375 310.984375 191.707031 295.070312 C 193.855469 298.828125 196.730469 302.652344 200.335938 306.25 C 211.949219 317.859375 226.121094 322.574219 231.941406 316.769531 C 237.757812 310.960938 233.035156 296.871094 221.398438 285.257812 C 216.902344 280.75 212.019531 277.335938 207.363281 275.082031 C 214.710938 263.246094 220.003906 249.449219 223.316406 234.628906 C 225.9375 235.445312 228.675781 236.222656 231.550781 236.835938 C 250.5625 240.773438 266.832031 237.40625 267.925781 229.300781 C 268.953125 221.242188 254.417969 211.453125 235.433594 207.535156 Z M 235.433594 207.535156 " |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,38 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style=" stroke:none;fill-rule:nonzero;fill:{color};fill-opacity:1;" d="M 169.460938 135.585938 C 184.714844 135.585938 197.761719 141.003906 208.59375 151.835938 C 219.425781 162.667969 224.84375 175.714844 224.84375 190.972656 C 224.84375 206.226562 219.425781 219.273438 208.59375 230.105469 C 197.761719 240.9375 184.714844 246.355469 169.460938 246.355469 C 154.203125 246.355469 141.160156 240.9375 130.324219 230.105469 C 119.492188 219.273438 114.074219 206.226562 114.074219 190.972656 C 114.074219 175.714844 119.492188 162.667969 130.324219 151.835938 C 141.160156 141.003906 154.203125 135.585938 169.460938 135.585938 M 322.921875 74.816406 C 336.511719 74.816406 348.113281 79.625 357.730469 89.242188 C 367.34375 98.855469 372.152344 110.457031 372.152344 124.046875 L 372.152344 296.355469 C 372.152344 309.945312 367.34375 321.546875 357.730469 331.164062 C 348.113281 340.777344 336.511719 345.585938 322.921875 345.585938 L 52.152344 345.585938 C 38.5625 345.585938 26.960938 340.777344 17.34375 331.164062 C 7.730469 321.546875 2.921875 309.945312 2.921875 296.355469 L 2.921875 124.046875 C 2.921875 110.457031 7.730469 98.855469 17.34375 89.242188 C 26.960938 79.625 38.5625 74.816406 52.152344 74.816406 L 77.152344 74.816406 L 86.960938 48.664062 C 89.394531 42.382812 93.851562 36.964844 100.324219 32.414062 C 106.800781 27.863281 113.433594 25.585938 120.230469 25.585938 L 218.691406 25.585938 C 225.484375 25.585938 232.121094 27.863281 238.59375 32.414062 C 245.070312 36.964844 249.523438 42.382812 251.960938 48.664062 L 261.765625 74.816406 L 322.921875 74.816406 M 169.460938 277.125 C 193.175781 277.125 213.464844 268.695312 230.324219 251.835938 C 247.183594 234.976562 255.613281 214.6875 255.613281 190.972656 C 255.613281 167.253906 247.183594 146.964844 230.324219 130.105469 C 213.464844 113.246094 193.175781 104.816406 169.460938 104.816406 C 145.742188 104.816406 125.453125 113.246094 108.59375 130.105469 C 91.734375 146.964844 83.304688 167.253906 83.304688 190.972656 C 83.304688 214.6875 91.734375 234.976562 108.59375 251.835938 C 125.453125 268.695312 145.742188 277.125 169.460938 277.125 "/>     <path style=" stroke:none;fill-rule:nonzero;fill:rgb(49.411765%,73.725492%,43.529412%);fill-opacity:1;" d="M 374.332031 308.261719 C 374.332031 343.875 345.464844 372.746094 309.851562 372.746094 C 274.238281 372.746094 245.371094 343.875 245.371094 308.261719 C 245.371094 272.652344 274.238281 243.78125 309.851562 243.78125 C 345.464844 243.78125 374.332031 272.652344 374.332031 308.261719 Z M 374.332031 308.261719 "/>     <path style="fill: none !important;stroke-width:95.518036;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" d="M 1154.783124 1044.338782 L 1543.909686 1044.338782 " transform="matrix(0.192308,0,0,0.192308,49.973197,107.396088)"/>     <path style="fill: none !important;stroke-width:95.518036;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" d="M 1349.925311 849.359094 L 1349.925311 1238.505969 " transform="matrix(0.192308,0,0,0.192308,49.973197,107.396088)"/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:{color};fill-opacity:1;" |  | ||||||
|       d="M 169.460938 135.585938 C 184.714844 135.585938 197.761719 141.003906 208.59375 151.835938 C 219.425781 162.667969 224.84375 175.714844 224.84375 190.972656 C 224.84375 206.226562 219.425781 219.273438 208.59375 230.105469 C 197.761719 240.9375 184.714844 246.355469 169.460938 246.355469 C 154.203125 246.355469 141.160156 240.9375 130.324219 230.105469 C 119.492188 219.273438 114.074219 206.226562 114.074219 190.972656 C 114.074219 175.714844 119.492188 162.667969 130.324219 151.835938 C 141.160156 141.003906 154.203125 135.585938 169.460938 135.585938 M 322.921875 74.816406 C 336.511719 74.816406 348.113281 79.625 357.730469 89.242188 C 367.34375 98.855469 372.152344 110.457031 372.152344 124.046875 L 372.152344 296.355469 C 372.152344 309.945312 367.34375 321.546875 357.730469 331.164062 C 348.113281 340.777344 336.511719 345.585938 322.921875 345.585938 L 52.152344 345.585938 C 38.5625 345.585938 26.960938 340.777344 17.34375 331.164062 C 7.730469 321.546875 2.921875 309.945312 2.921875 296.355469 L 2.921875 124.046875 C 2.921875 110.457031 7.730469 98.855469 17.34375 89.242188 C 26.960938 79.625 38.5625 74.816406 52.152344 74.816406 L 77.152344 74.816406 L 86.960938 48.664062 C 89.394531 42.382812 93.851562 36.964844 100.324219 32.414062 C 106.800781 27.863281 113.433594 25.585938 120.230469 25.585938 L 218.691406 25.585938 C 225.484375 25.585938 232.121094 27.863281 238.59375 32.414062 C 245.070312 36.964844 249.523438 42.382812 251.960938 48.664062 L 261.765625 74.816406 L 322.921875 74.816406 M 169.460938 277.125 C 193.175781 277.125 213.464844 268.695312 230.324219 251.835938 C 247.183594 234.976562 255.613281 214.6875 255.613281 190.972656 C 255.613281 167.253906 247.183594 146.964844 230.324219 130.105469 C 213.464844 113.246094 193.175781 104.816406 169.460938 104.816406 C 145.742188 104.816406 125.453125 113.246094 108.59375 130.105469 C 91.734375 146.964844 83.304688 167.253906 83.304688 190.972656 C 83.304688 214.6875 91.734375 234.976562 108.59375 251.835938 C 125.453125 268.695312 145.742188 277.125 169.460938 277.125 " |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:rgb(49.411765%,73.725492%,43.529412%);fill-opacity:1;" |  | ||||||
|       d="M 374.332031 308.261719 C 374.332031 343.875 345.464844 372.746094 309.851562 372.746094 C 274.238281 372.746094 245.371094 343.875 245.371094 308.261719 C 245.371094 272.652344 274.238281 243.78125 309.851562 243.78125 C 345.464844 243.78125 374.332031 272.652344 374.332031 308.261719 Z M 374.332031 308.261719 " |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:95.518036;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 1154.783124 1044.338782 L 1543.909686 1044.338782 " |  | ||||||
|       transform="matrix(0.192308,0,0,0.192308,49.973197,107.396088)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:95.518036;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 1349.925311 849.359094 L 1349.925311 1238.505969 " |  | ||||||
|       transform="matrix(0.192308,0,0,0.192308,49.973197,107.396088)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,25 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="374px" height="259px" viewBox="0 0 374 259" version="1.1">   <g id="surface1">     <path style="fill: none !important;stroke-width:4;stroke-linecap:butt;stroke-linejoin:round;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 2.999896 7.285624 L 10.826182 14.99991 L 23.000104 3.00009 " transform="matrix(14.384615,0,0,14.388889,0,0)"/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="374px" |  | ||||||
|   height="259px" |  | ||||||
|   viewBox="0 0 374 259" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:4;stroke-linecap:butt;stroke-linejoin:round;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 2.999896 7.285624 L 10.826182 14.99991 L 23.000104 3.00009 " |  | ||||||
|       transform="matrix(14.384615,0,0,14.388889,0,0)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,25 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">     <g id="surface1">         <path                 style="fill:{color};"                 class="selectable"               d="M 375 187.5 C 375 291.054688 291.054688 375 187.5 375 C 83.945312 375 0 291.054688 0 187.5 C 0 83.945312 83.945312 0 187.5 0 C 291.054688 0 375 83.945312 375 187.5 Z M 375 187.5 "/>     </g> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill:{color};" |  | ||||||
|       class="selectable" |  | ||||||
|       d="M 375 187.5 C 375 291.054688 291.054688 375 187.5 375 C 83.945312 375 0 291.054688 0 187.5 C 0 83.945312 83.945312 0 187.5 0 C 291.054688 0 375 83.945312 375 187.5 Z M 375 187.5 " |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,30 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style="fill: none !important;stroke-width:9;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 46 55.932292 L 66.551042 55.932292 " transform="matrix(3.75,0,0,3.75,0,0)"/>     <path style="fill: none !important;stroke-width:9;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 46 55.534375 L 46 27.729167 " transform="matrix(3.75,0,0,3.75,0,0)"/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:9;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 46 55.932292 L 66.551042 55.932292 " |  | ||||||
|       transform="matrix(3.75,0,0,3.75,0,0)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:9;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 46 55.534375 L 46 27.729167 " |  | ||||||
|       transform="matrix(3.75,0,0,3.75,0,0)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,30 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style="fill: none !important;stroke-width:6.012842;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 23.165924 293.603402 L 3.541282 273.978759 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)"/>     <path style="fill: none !important;stroke-width:6.012921;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 23.195414 273.884226 L 3.571047 293.508593 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)"/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:6.012842;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 23.165924 293.603402 L 3.541282 273.978759 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:6.012921;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 23.195414 273.884226 L 3.571047 293.508593 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,35 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <!-- Created with Inkscape (http://www.inkscape.org/) -->  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown    width="1124.975mm"    height="1111.9373mm"    viewBox="0 0 1124.975 1111.9373"    version="1.1"    id="svg5"    xmlns="http://www.w3.org/2000/svg"    xmlns:svg="http://www.w3.org/2000/svg">   <defs      id="defs2" />   <g      id="layer1"      transform="translate(429.38456,373.85159)"      style="display:inline">     <path        id="path858"        d="m 465.50195,-1412.9824 c -48.13018,0 -95.14167,6.6605 -140.92968,19.3242 -517.78906,36.4545 -1004.59215,263.4542 -1364.78127,639.11132 -125.578,130.97093 -232.0356,276.80492 -318.709,432.875 l -3.0996,0.006 0.012,5.62891 c -169.1451,306.7883664 -260.6208,653.52706 -260.8652,1009.62695 v 0.0645 c 0,555.45592 220.6943,1088.26072 613.4609,1481.02732 378.0703,378.0702 885.96964,596.0875 1418.85161,611.92 18.51351,1.9864 37.20113,3.0136 56.06054,3.0136 17.25834,0 34.36078,-0.9283 51.33008,-2.5937 224.45873,-4.8504 444.64455,-45.7063 652.83597,-119.1836 -60.8675,-43.0776 -118.1478,-91.0122 -171.27933,-143.334 -121.72268,40.4323 -230.24225,90.5328 -428.68164,58.6602 l -3.38086,-1775.19732 233.78516,-0.44532 c 42.42213,-67.09791 90.54812,-130.41487 143.84375,-189.24804 l -377.98828,0.7207 -1.43946,-755.89648 634.26176,-1.38672 c 32.7782,154.928763 56.3858,319.47347 69.9629,489.41797 58.6612,-33.17515 119.6711,-62.01386 182.541,-86.28516 -13.1747,-138.65014 -32.7954,-273.7956044 -58.7852,-403.55664 l 791.918,-1.73242 c 54.2995,111.507047 97.5037,228.170813 129.2012,348.26953 72.3897,18.82425 143.1313,43.49582 211.5273,73.77148 -80.6791,-402.00209 -278.3137,-774.77273 -572.5098,-1068.96875 -363.7812,-363.78173 -847.7515,-579.57553 -1358.52731,-609.49993 -41.8967,-10.5343 -84.78727,-16.1094 -128.61524,-16.1094 z m 96.98438,204.4375 c 159.67283,49.2174 326.54513,218.8615 462.03907,516.93552 49.6393,109.20202 92.1295,232.60044 128.4531,364.44727 l -588.81053,1.28711 z m -188.97461,0.9179 1.67969,882.16411 -597.98243,1.30664 c 36.4911,-132.99204 79.25432,-257.42426 129.269536,-367.45313 133.780665,-294.30513 298.164144,-463.66612 455.992184,-515.27932 3.68603,-0.2115 7.36574,-0.4597 11.04102,-0.7383 z m 591.27148,61.1954 c 320.2851,85.8427 615.6781,254.12924 854.2344,492.68551 99.0716,99.07158 185.6177,208.20254 259.6738,324.5625 l -729.584,1.59375 c -41.2536,-159.09493 -91.7465,-308.46744 -152.5469,-442.22266 -67.1207,-147.65948 -144.9895,-275.0596 -231.7773,-376.6191 z m -1003.50195,5.9433 c -84.79247,100.5848 -161.00089,225.84047 -226.83789,370.6758 -61.29809,134.85015 -112.17438,285.51379 -153.59961,446.08594 l -720.14645,1.57422 c 67.8012,-107.37947 146.33969,-208.61822 235.49997,-301.60743 238.54174,-248.78579 538.10225,-425.81263 865.08398,-516.72853 z m 414.26953,1004.00002 1.43945,755.8418 -727.68554,1.38671 c 4.20418,-264.24481 32.77031,-520.983169 82.2207,-755.82031 z m -837.92773,1.83203 c -47.33699,237.75431 -73.46184,493.4318 -77.33789,755.75781 l -892.42186,1.70117 c 10.0141,-264.11069 74.8439,-521.07351 188.0937,-755.75 z m 839.72656,942.98437 1.3457,707.00198 -646.08984,1.414 c -46.71986,-220.2353 -75.01919,-459.8106 -82.07617,-707.03121 z m -915.85156,1.74415 c 6.57514,245.41033 32.61831,484.16193 77.38867,707.09573 l -784.53902,1.7148 c -105.8005,-220.1735 -168.8074,-460.0874 -184.1446,-707.11327 z m 917.55859,894.23433 1.68359,884.2715 C 217.38802,2544.4462 45.439072,2373.9426 -93.521484,2068.2422 -142.93255,1959.5424 -185.25941,1836.7732 -221.48047,1705.6211 Z m -796.66211,1.7422 c 41.17164,158.3984 91.48351,307.1603 152.04883,440.3985 69.28935,152.4302 150.07483,283.1451 240.259765,386.2558 C -344.1327,2446.5457 -638.16035,2278.6776 -875.7832,2041.0547 -977.36504,1939.4729 -1065.801,1827.3298 -1141.0586,1707.6328 Z"        transform="scale(0.26458333)" />     <path        id="path928"        d="" />     <path        id="path890"        d="" />     <path        id="path3123"        d="m 500.42418,99.52416 c -162.9348,1.72512 -108.06353,223.75011 -69.65465,274.06343 l 1.09709,35.29034 c -176.038,13.83721 -191.80907,75.90897 -198.43798,158.82719 -2.02223,25.29521 3.89762,50.2628 9.81667,80.72646 97.56502,-0.71585 177.50757,-0.62612 275.07185,-0.49796 84.27329,-45.47436 147.33813,-122.20106 175.63579,-213.68411 -28.16643,-12.43076 -66.9608,-21.14889 -120.68876,-25.37209 l 1.09762,-35.29036 C 613.0731,322.8775 663.35897,97.79884 500.42418,99.52416 Z" />   </g> </svg>  | ||||||
| <!-- Created with Inkscape (http://www.inkscape.org/) --> |  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   width="1124.975mm" |  | ||||||
|   height="1111.9373mm" |  | ||||||
|   viewBox="0 0 1124.975 1111.9373" |  | ||||||
|   version="1.1" |  | ||||||
|   id="svg5" |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   xmlns:svg="http://www.w3.org/2000/svg" |  | ||||||
| > |  | ||||||
|   <defs id="defs2" /> |  | ||||||
|   <g id="layer1" transform="translate(429.38456,373.85159)" style="display:inline"> |  | ||||||
|     <path |  | ||||||
|       id="path858" |  | ||||||
|       d="m 465.50195,-1412.9824 c -48.13018,0 -95.14167,6.6605 -140.92968,19.3242 -517.78906,36.4545 -1004.59215,263.4542 -1364.78127,639.11132 -125.578,130.97093 -232.0356,276.80492 -318.709,432.875 l -3.0996,0.006 0.012,5.62891 c -169.1451,306.7883664 -260.6208,653.52706 -260.8652,1009.62695 v 0.0645 c 0,555.45592 220.6943,1088.26072 613.4609,1481.02732 378.0703,378.0702 885.96964,596.0875 1418.85161,611.92 18.51351,1.9864 37.20113,3.0136 56.06054,3.0136 17.25834,0 34.36078,-0.9283 51.33008,-2.5937 224.45873,-4.8504 444.64455,-45.7063 652.83597,-119.1836 -60.8675,-43.0776 -118.1478,-91.0122 -171.27933,-143.334 -121.72268,40.4323 -230.24225,90.5328 -428.68164,58.6602 l -3.38086,-1775.19732 233.78516,-0.44532 c 42.42213,-67.09791 90.54812,-130.41487 143.84375,-189.24804 l -377.98828,0.7207 -1.43946,-755.89648 634.26176,-1.38672 c 32.7782,154.928763 56.3858,319.47347 69.9629,489.41797 58.6612,-33.17515 119.6711,-62.01386 182.541,-86.28516 -13.1747,-138.65014 -32.7954,-273.7956044 -58.7852,-403.55664 l 791.918,-1.73242 c 54.2995,111.507047 97.5037,228.170813 129.2012,348.26953 72.3897,18.82425 143.1313,43.49582 211.5273,73.77148 -80.6791,-402.00209 -278.3137,-774.77273 -572.5098,-1068.96875 -363.7812,-363.78173 -847.7515,-579.57553 -1358.52731,-609.49993 -41.8967,-10.5343 -84.78727,-16.1094 -128.61524,-16.1094 z m 96.98438,204.4375 c 159.67283,49.2174 326.54513,218.8615 462.03907,516.93552 49.6393,109.20202 92.1295,232.60044 128.4531,364.44727 l -588.81053,1.28711 z m -188.97461,0.9179 1.67969,882.16411 -597.98243,1.30664 c 36.4911,-132.99204 79.25432,-257.42426 129.269536,-367.45313 133.780665,-294.30513 298.164144,-463.66612 455.992184,-515.27932 3.68603,-0.2115 7.36574,-0.4597 11.04102,-0.7383 z m 591.27148,61.1954 c 320.2851,85.8427 615.6781,254.12924 854.2344,492.68551 99.0716,99.07158 185.6177,208.20254 259.6738,324.5625 l -729.584,1.59375 c -41.2536,-159.09493 -91.7465,-308.46744 -152.5469,-442.22266 -67.1207,-147.65948 -144.9895,-275.0596 -231.7773,-376.6191 z m -1003.50195,5.9433 c -84.79247,100.5848 -161.00089,225.84047 -226.83789,370.6758 -61.29809,134.85015 -112.17438,285.51379 -153.59961,446.08594 l -720.14645,1.57422 c 67.8012,-107.37947 146.33969,-208.61822 235.49997,-301.60743 238.54174,-248.78579 538.10225,-425.81263 865.08398,-516.72853 z m 414.26953,1004.00002 1.43945,755.8418 -727.68554,1.38671 c 4.20418,-264.24481 32.77031,-520.983169 82.2207,-755.82031 z m -837.92773,1.83203 c -47.33699,237.75431 -73.46184,493.4318 -77.33789,755.75781 l -892.42186,1.70117 c 10.0141,-264.11069 74.8439,-521.07351 188.0937,-755.75 z m 839.72656,942.98437 1.3457,707.00198 -646.08984,1.414 c -46.71986,-220.2353 -75.01919,-459.8106 -82.07617,-707.03121 z m -915.85156,1.74415 c 6.57514,245.41033 32.61831,484.16193 77.38867,707.09573 l -784.53902,1.7148 c -105.8005,-220.1735 -168.8074,-460.0874 -184.1446,-707.11327 z m 917.55859,894.23433 1.68359,884.2715 C 217.38802,2544.4462 45.439072,2373.9426 -93.521484,2068.2422 -142.93255,1959.5424 -185.25941,1836.7732 -221.48047,1705.6211 Z m -796.66211,1.7422 c 41.17164,158.3984 91.48351,307.1603 152.04883,440.3985 69.28935,152.4302 150.07483,283.1451 240.259765,386.2558 C -344.1327,2446.5457 -638.16035,2278.6776 -875.7832,2041.0547 -977.36504,1939.4729 -1065.801,1827.3298 -1141.0586,1707.6328 Z" |  | ||||||
|       transform="scale(0.26458333)" |  | ||||||
|     /> |  | ||||||
|     <path id="path928" d="" /> |  | ||||||
|     <path id="path890" d="" /> |  | ||||||
|     <path |  | ||||||
|       id="path3123" |  | ||||||
|       d="m 500.42418,99.52416 c -162.9348,1.72512 -108.06353,223.75011 -69.65465,274.06343 l 1.09709,35.29034 c -176.038,13.83721 -191.80907,75.90897 -198.43798,158.82719 -2.02223,25.29521 3.89762,50.2628 9.81667,80.72646 97.56502,-0.71585 177.50757,-0.62612 275.07185,-0.49796 84.27329,-45.47436 147.33813,-122.20106 175.63579,-213.68411 -28.16643,-12.43076 -66.9608,-21.14889 -120.68876,-25.37209 l 1.09762,-35.29036 C 613.0731,322.8775 663.35897,97.79884 500.42418,99.52416 Z" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,26 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 | <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" viewBox="0 0 36 36"><circle fill="#F4900C" cx="18" cy="18" r="18"/><circle fill="#FFD983" cx="18" cy="18" r="14.5"/><circle fill="#F5F8FA" cx="18" cy="18" r="13"/><path fill="#CCD6DD" d="M18 8l1.531 6.304 5.54-3.375-3.375 5.54L28 18l-6.304 1.531 3.375 5.54-5.54-3.375L18 28l-1.531-6.304-5.54 3.375 3.375-5.54L8 18l6.304-1.531-3.375-5.54 5.54 3.375z"/><path fill="#292F33" d="M17.343 20.748l8.777 5.381-5.379-8.778z"/><path fill="#DD2E44" d="M18.657 15.267L9.879 9.886l5.38 8.779z"/><circle fill="#8899A6" cx="18" cy="18.008" r="3.055"/><circle fill="#F5F8FA" cx="18" cy="18.008" r="1.648"/></svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   viewBox="0 0 36 36" |  | ||||||
| > |  | ||||||
|   <circle fill="#F4900C" cx="18" cy="18" r="18" /> |  | ||||||
|   <circle fill="#FFD983" cx="18" cy="18" r="14.5" /> |  | ||||||
|   <circle fill="#F5F8FA" cx="18" cy="18" r="13" /> |  | ||||||
|   <path |  | ||||||
|     fill="#CCD6DD" |  | ||||||
|     d="M18 8l1.531 6.304 5.54-3.375-3.375 5.54L28 18l-6.304 1.531 3.375 5.54-5.54-3.375L18 28l-1.531-6.304-5.54 3.375 3.375-5.54L8 18l6.304-1.531-3.375-5.54 5.54 3.375z" |  | ||||||
|   /> |  | ||||||
|   <path fill="#292F33" d="M17.343 20.748l8.777 5.381-5.379-8.778z" /> |  | ||||||
|   <path fill="#DD2E44" d="M18.657 15.267L9.879 9.886l5.38 8.779z" /> |  | ||||||
|   <circle fill="#8899A6" cx="18" cy="18.008" r="3.055" /> |  | ||||||
|   <circle fill="#F5F8FA" cx="18" cy="18.008" r="1.648" /> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,49 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown    width="157.33984"    height="157.33984"    viewBox="0 0 157.33984 157.33984"    version="1.1"    id="svg9"    sodipodi:docname="confirm.svg"    inkscape:version="1.1.2 (1:1.1+202202050950+0a00cf5339)"    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"    xmlns="http://www.w3.org/2000/svg"    xmlns:svg="http://www.w3.org/2000/svg">   <defs      id="defs13" />   <sodipodi:namedview      id="namedview11"      pagecolor="#505050"      bordercolor="#eeeeee"      borderopacity="1"      inkscape:pageshadow="0"      inkscape:pageopacity="0"      inkscape:pagecheckerboard="0"      showgrid="false"      inkscape:zoom="2.312"      inkscape:cx="-18.598616"      inkscape:cy="57.093426"      inkscape:current-layer="svg9" />   <path      style="fill:#35d447;fill-opacity:1;fill-rule:nonzero;stroke:none"      d="m 157.33984,78.66796 c 0,43.44922 -35.21875,78.67188 -78.66796,78.67188 C 35.22266,157.33984 0,122.11718 0,78.66796 0,35.22265 35.22266,0 78.67188,0 c 43.44921,0 78.66796,35.22265 78.66796,78.66796 z m 0,0"      id="path4" />   <path      style="fill: none !important;stroke:#ffffff;stroke-width:19.7495;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1"      d="m 37.69921,75.49609 35.55078,39.5 47.39844,-63.19922"      id="path6" /> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   width="157.33984" |  | ||||||
|   height="157.33984" |  | ||||||
|   viewBox="0 0 157.33984 157.33984" |  | ||||||
|   version="1.1" |  | ||||||
|   id="svg9" |  | ||||||
|   sodipodi:docname="confirm.svg" |  | ||||||
|   inkscape:version="1.1.2 (1:1.1+202202050950+0a00cf5339)" |  | ||||||
|   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" |  | ||||||
|   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   xmlns:svg="http://www.w3.org/2000/svg" |  | ||||||
| > |  | ||||||
|   <defs id="defs13" /> |  | ||||||
|   <sodipodi:namedview |  | ||||||
|     id="namedview11" |  | ||||||
|     pagecolor="#505050" |  | ||||||
|     bordercolor="#eeeeee" |  | ||||||
|     borderopacity="1" |  | ||||||
|     inkscape:pageshadow="0" |  | ||||||
|     inkscape:pageopacity="0" |  | ||||||
|     inkscape:pagecheckerboard="0" |  | ||||||
|     showgrid="false" |  | ||||||
|     inkscape:zoom="2.312" |  | ||||||
|     inkscape:cx="-18.598616" |  | ||||||
|     inkscape:cy="57.093426" |  | ||||||
|     inkscape:current-layer="svg9" |  | ||||||
|   /> |  | ||||||
|   <path |  | ||||||
|     style="fill:#35d447;fill-opacity:1;fill-rule:nonzero;stroke:none" |  | ||||||
|     d="m 157.33984,78.66796 c 0,43.44922 -35.21875,78.67188 -78.66796,78.67188 C 35.22266,157.33984 0,122.11718 0,78.66796 0,35.22265 35.22266,0 78.67188,0 c 43.44921,0 78.66796,35.22265 78.66796,78.66796 z m 0,0" |  | ||||||
|     id="path4" |  | ||||||
|   /> |  | ||||||
|   <path |  | ||||||
|     style="fill: none !important;stroke:#ffffff;stroke-width:19.7495;stroke-linecap:round;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1" |  | ||||||
|     d="m 37.69921,75.49609 35.55078,39.5 47.39844,-63.19922" |  | ||||||
|     id="path6" |  | ||||||
|   /> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,51 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown    width="375px"    height="375px"    viewBox="0 0 375 375"    version="1.1"    id="svg5"    sodipodi:docname="copyright.svg"    inkscape:version="1.1.2 (0a00cf5339, 2022-02-04)"    xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"    xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"    xmlns="http://www.w3.org/2000/svg"    xmlns:svg="http://www.w3.org/2000/svg">   <defs      id="defs9" />   <sodipodi:namedview      id="namedview7"      pagecolor="#505050"      bordercolor="#eeeeee"      borderopacity="1"      inkscape:pageshadow="0"      inkscape:pageopacity="0"      inkscape:pagecheckerboard="0"      showgrid="false"      inkscape:zoom="2.4012481"      inkscape:cx="194.06574"      inkscape:cy="30.192632"      inkscape:window-width="1920"      inkscape:window-height="995"      inkscape:window-x="0"      inkscape:window-y="0"      inkscape:window-maximized="1"      inkscape:current-layer="svg5" />   <g      id="surface1"      transform="matrix(1.2644961,0,0,1.2644961,-49.593016,-49.593016)">     <path        style="fill:{color};fill-opacity:1;fill-rule:nonzero;stroke:none"        d="m 159.26562,170.73437 c 0.73438,-4.85156 2.35157,-9.11718 4.41016,-12.79296 2.0625,-3.67579 5,-6.76172 8.67969,-9.11719 3.52734,-2.20313 7.9375,-3.23438 13.3789,-3.37891 3.38282,0.14453 6.47266,0.73438 9.26563,1.91016 2.94141,1.32422 5.58594,3.08594 7.64453,5.29297 2.05859,2.20703 3.67969,4.85156 5,7.79297 1.32422,2.9414 1.91406,6.17578 2.05859,9.41406 h 26.32422 c -0.29297,-6.91406 -1.61718,-13.23438 -4.11718,-18.97266 -2.5,-5.73437 -5.88282,-10.73437 -10.29297,-14.85156 -4.41407,-4.11719 -9.70703,-7.35156 -15.88282,-9.70312 -6.17578,-2.35547 -12.9414,-3.38282 -20.4414,-3.38282 -9.5586,0 -17.9375,1.61719 -24.9961,5 -7.05859,3.38281 -12.9414,7.79297 -17.64843,13.52735 -4.70313,5.73437 -8.23438,12.35156 -10.44141,20 -2.20312,7.64453 -3.52734,15.58593 -3.52734,24.11718 v 3.96875 c 0,8.52735 1.17578,16.46875 3.38281,24.11719 2.20312,7.64453 5.73437,14.26172 10.44141,19.85156 4.70312,5.58594 10.58593,10.14453 17.64453,13.38282 7.05859,3.23437 15.4414,5 25,5 6.91015,0 13.3789,-1.17969 19.41015,-3.38282 6.02735,-2.20703 11.32422,-5.29687 15.87891,-9.26562 4.55859,-3.96875 8.23828,-8.52735 10.88281,-13.82422 2.64844,-5.29297 4.26563,-10.87891 4.41406,-16.91016 h -26.32421 c -0.14454,3.08985 -0.88282,5.88282 -2.20704,8.53125 -1.32031,2.64453 -3.08593,4.85157 -5.29296,6.76172 -2.20313,1.91406 -4.70313,3.38281 -7.64454,4.41406 -2.79687,1.02735 -5.73437,1.32032 -8.82421,1.46875 -5.29297,-0.14843 -9.70704,-1.17578 -13.08594,-3.38281 -3.67969,-2.35156 -6.61719,-5.4414 -8.67969,-9.11719 -2.05859,-3.67578 -3.67578,-8.08593 -4.41016,-12.9375 -0.73437,-4.85546 -1.17578,-9.85546 -1.17578,-14.70703 v -3.96875 c 0,-5.14843 0.44141,-10 1.17578,-14.85547 z M 187.5,40.449219 c -81.17187,0 -147.050781,65.878911 -147.050781,147.050781 0,81.17187 65.878911,147.05078 147.050781,147.05078 81.17187,0 147.05078,-65.87891 147.05078,-147.05078 0,-81.17187 -65.87891,-147.050781 -147.05078,-147.050781 z m 0,264.691401 c -64.84766,0 -117.640625,-52.79296 -117.640625,-117.64062 0,-64.84766 52.792965,-117.640625 117.640625,-117.640625 64.84766,0 117.64062,52.792965 117.64062,117.640625 0,64.84766 -52.79296,117.64062 -117.64062,117.64062 z m 0,0"        id="path2" />   </g> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
|   id="svg5" |  | ||||||
|   sodipodi:docname="copyright.svg" |  | ||||||
|   inkscape:version="1.1.2 (0a00cf5339, 2022-02-04)" |  | ||||||
|   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" |  | ||||||
|   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   xmlns:svg="http://www.w3.org/2000/svg" |  | ||||||
| > |  | ||||||
|   <defs id="defs9" /> |  | ||||||
|   <sodipodi:namedview |  | ||||||
|     id="namedview7" |  | ||||||
|     pagecolor="#505050" |  | ||||||
|     bordercolor="#eeeeee" |  | ||||||
|     borderopacity="1" |  | ||||||
|     inkscape:pageshadow="0" |  | ||||||
|     inkscape:pageopacity="0" |  | ||||||
|     inkscape:pagecheckerboard="0" |  | ||||||
|     showgrid="false" |  | ||||||
|     inkscape:zoom="2.4012481" |  | ||||||
|     inkscape:cx="194.06574" |  | ||||||
|     inkscape:cy="30.192632" |  | ||||||
|     inkscape:window-width="1920" |  | ||||||
|     inkscape:window-height="995" |  | ||||||
|     inkscape:window-x="0" |  | ||||||
|     inkscape:window-y="0" |  | ||||||
|     inkscape:window-maximized="1" |  | ||||||
|     inkscape:current-layer="svg5" |  | ||||||
|   /> |  | ||||||
|   <g id="surface1" transform="matrix(1.2644961,0,0,1.2644961,-49.593016,-49.593016)"> |  | ||||||
|     <path |  | ||||||
|       style="fill:{color};fill-opacity:1;fill-rule:nonzero;stroke:none" |  | ||||||
|       d="m 159.26562,170.73437 c 0.73438,-4.85156 2.35157,-9.11718 4.41016,-12.79296 2.0625,-3.67579 5,-6.76172 8.67969,-9.11719 3.52734,-2.20313 7.9375,-3.23438 13.3789,-3.37891 3.38282,0.14453 6.47266,0.73438 9.26563,1.91016 2.94141,1.32422 5.58594,3.08594 7.64453,5.29297 2.05859,2.20703 3.67969,4.85156 5,7.79297 1.32422,2.9414 1.91406,6.17578 2.05859,9.41406 h 26.32422 c -0.29297,-6.91406 -1.61718,-13.23438 -4.11718,-18.97266 -2.5,-5.73437 -5.88282,-10.73437 -10.29297,-14.85156 -4.41407,-4.11719 -9.70703,-7.35156 -15.88282,-9.70312 -6.17578,-2.35547 -12.9414,-3.38282 -20.4414,-3.38282 -9.5586,0 -17.9375,1.61719 -24.9961,5 -7.05859,3.38281 -12.9414,7.79297 -17.64843,13.52735 -4.70313,5.73437 -8.23438,12.35156 -10.44141,20 -2.20312,7.64453 -3.52734,15.58593 -3.52734,24.11718 v 3.96875 c 0,8.52735 1.17578,16.46875 3.38281,24.11719 2.20312,7.64453 5.73437,14.26172 10.44141,19.85156 4.70312,5.58594 10.58593,10.14453 17.64453,13.38282 7.05859,3.23437 15.4414,5 25,5 6.91015,0 13.3789,-1.17969 19.41015,-3.38282 6.02735,-2.20703 11.32422,-5.29687 15.87891,-9.26562 4.55859,-3.96875 8.23828,-8.52735 10.88281,-13.82422 2.64844,-5.29297 4.26563,-10.87891 4.41406,-16.91016 h -26.32421 c -0.14454,3.08985 -0.88282,5.88282 -2.20704,8.53125 -1.32031,2.64453 -3.08593,4.85157 -5.29296,6.76172 -2.20313,1.91406 -4.70313,3.38281 -7.64454,4.41406 -2.79687,1.02735 -5.73437,1.32032 -8.82421,1.46875 -5.29297,-0.14843 -9.70704,-1.17578 -13.08594,-3.38281 -3.67969,-2.35156 -6.61719,-5.4414 -8.67969,-9.11719 -2.05859,-3.67578 -3.67578,-8.08593 -4.41016,-12.9375 -0.73437,-4.85546 -1.17578,-9.85546 -1.17578,-14.70703 v -3.96875 c 0,-5.14843 0.44141,-10 1.17578,-14.85547 z M 187.5,40.449219 c -81.17187,0 -147.050781,65.878911 -147.050781,147.050781 0,81.17187 65.878911,147.05078 147.050781,147.05078 81.17187,0 147.05078,-65.87891 147.05078,-147.05078 0,-81.17187 -65.87891,-147.050781 -147.05078,-147.050781 z m 0,264.691401 c -64.84766,0 -117.640625,-52.79296 -117.640625,-117.64062 0,-64.84766 52.792965,-117.640625 117.640625,-117.640625 64.84766,0 117.64062,52.792965 117.64062,117.640625 0,64.84766 -52.79296,117.64062 -117.64062,117.64062 z m 0,0" |  | ||||||
|       id="path2" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,42 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown         xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"         xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"         width="9"         height="9"         viewBox="0 0 9 9"         version="1.1"         sodipodi:docname="cross.svg"         inkscape:version="1.1.2 (0a00cf5339, 2022-02-04)"         xmlns="http://www.w3.org/2000/svg" >       <path             style="fill: none !important;stroke:{color};stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403"             d="M 1,4.5 H 8"             id="path941"/>     <path             style="fill: none !important;stroke:{color};stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403"             d="M 4.5,1 V 4"             id="path941"/>     <path             style="fill: none !important;stroke:{color};stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403"             d="M 4.5,5 V 8"             id="path941"/>      <path             style="fill: none !important;stroke:#ffffff;stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403"             d="M 3.5,5 v 3.5 h 2 v -3 h 3 v -2 h -3 v -3 h -2 v 3 h -3 v 2 h3"             id="path941"/> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" |  | ||||||
|   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" |  | ||||||
|   width="9" |  | ||||||
|   height="9" |  | ||||||
|   viewBox="0 0 9 9" |  | ||||||
|   version="1.1" |  | ||||||
|   sodipodi:docname="cross.svg" |  | ||||||
|   inkscape:version="1.1.2 (0a00cf5339, 2022-02-04)" |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
| > |  | ||||||
|   <path |  | ||||||
|     style="fill: none !important;stroke:{color};stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403" |  | ||||||
|     d="M 1,4.5 H 8" |  | ||||||
|     id="path941" |  | ||||||
|   /> |  | ||||||
|   <path |  | ||||||
|     style="fill: none !important;stroke:{color};stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403" |  | ||||||
|     d="M 4.5,1 V 4" |  | ||||||
|     id="path941" |  | ||||||
|   /> |  | ||||||
|   <path |  | ||||||
|     style="fill: none !important;stroke:{color};stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403" |  | ||||||
|     d="M 4.5,5 V 8" |  | ||||||
|     id="path941" |  | ||||||
|   /> |  | ||||||
|   <path |  | ||||||
|     style="fill: none !important;stroke:#ffffff;stroke-width:1;stroke-linecap:butt;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:0.50314403" |  | ||||||
|     d="M 3.5,5 v 3.5 h 2 v -3 h 3 v -2 h -3 v -3 h -2 v 3 h -3 v 2 h3" |  | ||||||
|     id="path941" |  | ||||||
|   /> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,30 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style="fill: none !important;stroke-width:3.439583;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 18.97295 289.3838 L 7.74695 278.1578 " transform="matrix(11.65716,0,0,11.65716,130.176942,-3023.744742)"/>     <path style="fill: none !important;stroke-width:3.439583;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 18.989705 278.103849 L 7.763705 289.329515 " transform="matrix(11.65716,0,0,11.65716,130.176942,-3023.744742)"/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:3.439583;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 18.97295 289.3838 L 7.74695 278.1578 " |  | ||||||
|       transform="matrix(11.65716,0,0,11.65716,130.176942,-3023.744742)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:3.439583;stroke-linecap:round;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 18.989705 278.103849 L 7.763705 289.329515 " |  | ||||||
|       transform="matrix(11.65716,0,0,11.65716,130.176942,-3023.744742)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,25 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown    width="375px"    height="375px"    viewBox="0 0 375 375"    version="1.1"    id="svg15"    xmlns="http://www.w3.org/2000/svg"    xmlns:svg="http://www.w3.org/2000/svg">   <path      id="path2"      style="color:{color};fill:{color};fill-opacity:0.988235;stroke-linecap:round;stroke-linejoin:round;-inkscape-stroke:none"      d="M 187.5 0.17578125 A 15 15 0 0 0 172.5 15.175781 L 172.5 43.708984 C 104.37541 50.343145 49.843699 104.61884 42.837891 172.63672 L 14.765625 172.63672 A 14.86235 14.86235 0 0 0 -0.09765625 187.5 A 14.86235 14.86235 0 0 0 14.765625 202.36133 L 42.837891 202.36133 C 49.842792 270.37794 104.37427 324.65648 172.5 331.29102 L 172.5 360.10156 A 15 15 0 0 0 187.5 375.10156 A 15 15 0 0 0 202.5 360.10156 L 202.5 331.08594 C 269.70108 323.65092 323.26953 269.82518 330.26953 202.5 L 360.08203 202.5 A 15 15 0 0 0 375.08203 187.5 A 15 15 0 0 0 360.08203 172.5 L 330.26953 172.5 C 323.26946 105.17266 269.70061 51.348735 202.5 43.914062 L 202.5 15.175781 A 15 15 0 0 0 187.5 0.17578125 z M 186.5625 80.511719 C 245.87515 80.511719 293.55078 128.18741 293.55078 187.5 C 293.55078 246.80822 245.87443 294.48828 186.5625 294.48828 C 127.25056 294.48828 79.574219 246.80822 79.574219 187.5 C 79.574219 128.1874 127.24984 80.511719 186.5625 80.511719 z M 187.5 139.21094 C 160.83203 139.21094 139.21094 160.83203 139.21094 187.5 C 139.21094 214.16797 160.83203 235.78906 187.5 235.78906 C 214.16797 235.78906 235.78906 214.16797 235.78906 187.5 C 235.78906 160.83203 214.16797 139.21094 187.5 139.21094 z " /> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
|   id="svg15" |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   xmlns:svg="http://www.w3.org/2000/svg" |  | ||||||
| > |  | ||||||
|   <path |  | ||||||
|     id="path2" |  | ||||||
|     style="color:{color};fill:{color};fill-opacity:0.988235;stroke-linecap:round;stroke-linejoin:round;-inkscape-stroke:none" |  | ||||||
|     d="M 187.5 0.17578125 A 15 15 0 0 0 172.5 15.175781 L 172.5 43.708984 C 104.37541 50.343145 49.843699 104.61884 42.837891 172.63672 L 14.765625 172.63672 A 14.86235 14.86235 0 0 0 -0.09765625 187.5 A 14.86235 14.86235 0 0 0 14.765625 202.36133 L 42.837891 202.36133 C 49.842792 270.37794 104.37427 324.65648 172.5 331.29102 L 172.5 360.10156 A 15 15 0 0 0 187.5 375.10156 A 15 15 0 0 0 202.5 360.10156 L 202.5 331.08594 C 269.70108 323.65092 323.26953 269.82518 330.26953 202.5 L 360.08203 202.5 A 15 15 0 0 0 375.08203 187.5 A 15 15 0 0 0 360.08203 172.5 L 330.26953 172.5 C 323.26946 105.17266 269.70061 51.348735 202.5 43.914062 L 202.5 15.175781 A 15 15 0 0 0 187.5 0.17578125 z M 186.5625 80.511719 C 245.87515 80.511719 293.55078 128.18741 293.55078 187.5 C 293.55078 246.80822 245.87443 294.48828 186.5625 294.48828 C 127.25056 294.48828 79.574219 246.80822 79.574219 187.5 C 79.574219 128.1874 127.24984 80.511719 186.5625 80.511719 z M 187.5 139.21094 C 160.83203 139.21094 139.21094 160.83203 139.21094 187.5 C 139.21094 214.16797 160.83203 235.78906 187.5 235.78906 C 214.16797 235.78906 235.78906 214.16797 235.78906 187.5 C 235.78906 160.83203 214.16797 139.21094 187.5 139.21094 z " |  | ||||||
|   /> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,49 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style="fill: none !important;stroke-width:2.645833;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" d="M 22.034555 283.770823 C 22.034555 288.670301 18.062773 292.642358 13.16302 292.642358 C 8.263267 292.642358 4.291486 288.670301 4.291486 283.770823 C 4.291486 278.871071 8.263267 274.899289 13.16302 274.899289 C 18.062773 274.899289 22.034555 278.871071 22.034555 283.770823 Z M 22.034555 283.770823 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)"/>     <path style="fill: none !important;stroke-width:2.097239;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" d="M 3.28414 283.770823 L 1.041796 283.770823 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)"/>     <path style="fill: none !important;stroke-width:2.116667;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" d="M 25.405787 283.770823 L 23.286365 283.770823 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)"/>     <path style="fill: none !important;stroke-width:2.116667;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" d="M 13.229166 295.948823 L 13.229166 293.831329 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)"/>     <path style="fill: none !important;stroke-width:2.116667;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" d="M 13.229166 275.057488 L 13.229166 271.612392 " transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)"/>     <path style=" stroke:none;fill-rule:nonzero;fill:rgb(33.333334%,33.333334%,92.54902%);fill-opacity:0.988235;" d="M 238.824219 161.203125 L 229.523438 161.203125 L 229.523438 134.96875 C 229.523438 134.96875 229.523438 91.246094 186.066406 91.246094 C 142.609375 91.246094 142.609375 134.96875 142.609375 134.96875 L 142.609375 161.203125 L 133.308594 161.203125 C 124.042969 161.203125 116.535156 168.757812 116.535156 178.082031 L 116.535156 249.347656 C 116.582031 258.636719 124.078125 266.136719 133.308594 266.136719 L 238.824219 266.136719 C 248.085938 266.136719 255.597656 258.582031 255.597656 249.261719 L 255.597656 178.082031 C 255.597656 168.757812 248.085938 161.203125 238.824219 161.203125 Z M 186.066406 231.160156 C 176.464844 231.160156 168.683594 223.328125 168.683594 213.671875 C 168.683594 204.011719 176.464844 196.179688 186.066406 196.179688 C 195.664062 196.179688 203.449219 204.011719 203.449219 213.671875 C 203.449219 223.328125 195.664062 231.160156 186.066406 231.160156 Z M 212.140625 161.203125 L 159.992188 161.203125 L 159.992188 139.339844 C 159.992188 126.226562 159.992188 108.734375 186.066406 108.734375 C 212.140625 108.734375 212.140625 126.226562 212.140625 139.339844 Z M 212.140625 161.203125 "/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:2.645833;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" |  | ||||||
|       d="M 22.034555 283.770823 C 22.034555 288.670301 18.062773 292.642358 13.16302 292.642358 C 8.263267 292.642358 4.291486 288.670301 4.291486 283.770823 C 4.291486 278.871071 8.263267 274.899289 13.16302 274.899289 C 18.062773 274.899289 22.034555 278.871071 22.034555 283.770823 Z M 22.034555 283.770823 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:2.097239;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" |  | ||||||
|       d="M 3.28414 283.770823 L 1.041796 283.770823 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:2.116667;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" |  | ||||||
|       d="M 25.405787 283.770823 L 23.286365 283.770823 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:2.116667;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" |  | ||||||
|       d="M 13.229166 295.948823 L 13.229166 293.831329 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:2.116667;stroke-linecap:round;stroke-linejoin:miter;stroke:rgb(33.333334%,33.333334%,92.54902%);stroke-opacity:0.988235;stroke-miterlimit:4;" |  | ||||||
|       d="M 13.229166 275.057488 L 13.229166 271.612392 " |  | ||||||
|       transform="matrix(14.173228,0,0,14.173228,0.0000135166,-3834.448583)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:rgb(33.333334%,33.333334%,92.54902%);fill-opacity:0.988235;" |  | ||||||
|       d="M 238.824219 161.203125 L 229.523438 161.203125 L 229.523438 134.96875 C 229.523438 134.96875 229.523438 91.246094 186.066406 91.246094 C 142.609375 91.246094 142.609375 134.96875 142.609375 134.96875 L 142.609375 161.203125 L 133.308594 161.203125 C 124.042969 161.203125 116.535156 168.757812 116.535156 178.082031 L 116.535156 249.347656 C 116.582031 258.636719 124.078125 266.136719 133.308594 266.136719 L 238.824219 266.136719 C 248.085938 266.136719 255.597656 258.582031 255.597656 249.261719 L 255.597656 178.082031 C 255.597656 168.757812 248.085938 161.203125 238.824219 161.203125 Z M 186.066406 231.160156 C 176.464844 231.160156 168.683594 223.328125 168.683594 213.671875 C 168.683594 204.011719 176.464844 196.179688 186.066406 196.179688 C 195.664062 196.179688 203.449219 204.011719 203.449219 213.671875 C 203.449219 223.328125 195.664062 231.160156 186.066406 231.160156 Z M 212.140625 161.203125 L 159.992188 161.203125 L 159.992188 139.339844 C 159.992188 126.226562 159.992188 108.734375 186.066406 108.734375 C 212.140625 108.734375 212.140625 126.226562 212.140625 139.339844 Z M 212.140625 161.203125 " |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,24 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style=" stroke:none;fill-rule:nonzero;fill:rgb(100%,0%,0%);fill-opacity:1;" d="M 148.457031 148.796875 L 148.457031 269.332031 C 148.457031 271.285156 147.828125 272.890625 146.574219 274.144531 C 145.316406 275.402344 143.714844 276.027344 141.761719 276.027344 L 128.367188 276.027344 C 126.414062 276.027344 124.808594 275.402344 123.554688 274.144531 C 122.296875 272.890625 121.671875 271.285156 121.671875 269.332031 L 121.671875 148.796875 C 121.671875 146.84375 122.296875 145.238281 123.554688 143.984375 C 124.808594 142.726562 126.414062 142.101562 128.367188 142.101562 L 141.761719 142.101562 C 143.714844 142.101562 145.316406 142.726562 146.574219 143.984375 C 147.828125 145.238281 148.457031 146.84375 148.457031 148.796875 Z M 202.027344 148.796875 L 202.027344 269.332031 C 202.027344 271.285156 201.398438 272.890625 200.144531 274.144531 C 198.890625 275.402344 197.285156 276.027344 195.332031 276.027344 L 181.9375 276.027344 C 179.984375 276.027344 178.382812 275.402344 177.125 274.144531 C 175.871094 272.890625 175.242188 271.285156 175.242188 269.332031 L 175.242188 148.796875 C 175.242188 146.84375 175.871094 145.238281 177.125 143.984375 C 178.382812 142.726562 179.984375 142.101562 181.9375 142.101562 L 195.332031 142.101562 C 197.285156 142.101562 198.890625 142.726562 200.144531 143.984375 C 201.398438 145.238281 202.027344 146.84375 202.027344 148.796875 Z M 255.597656 148.796875 L 255.597656 269.332031 C 255.597656 271.285156 254.972656 272.890625 253.714844 274.144531 C 252.460938 275.402344 250.855469 276.027344 248.902344 276.027344 L 235.511719 276.027344 C 233.558594 276.027344 231.953125 275.402344 230.695312 274.144531 C 229.441406 272.890625 228.8125 271.285156 228.8125 269.332031 L 228.8125 148.796875 C 228.8125 146.84375 229.441406 145.238281 230.695312 143.984375 C 231.953125 142.726562 233.558594 142.101562 235.511719 142.101562 L 248.902344 142.101562 C 250.855469 142.101562 252.460938 142.726562 253.714844 143.984375 C 254.972656 145.238281 255.597656 146.84375 255.597656 148.796875 Z M 282.386719 300.304688 L 282.386719 101.921875 L 94.886719 101.921875 L 94.886719 300.304688 C 94.886719 303.371094 95.375 306.199219 96.351562 308.777344 C 97.328125 311.359375 98.335938 313.242188 99.382812 314.429688 C 100.429688 315.613281 101.164062 316.207031 101.582031 316.207031 L 275.6875 316.207031 C 276.105469 316.207031 276.839844 315.613281 277.886719 314.429688 C 278.933594 313.242188 279.945312 311.359375 280.921875 308.777344 C 281.898438 306.199219 282.386719 303.371094 282.386719 300.304688 Z M 141.761719 75.136719 L 235.511719 75.136719 L 225.464844 50.652344 C 224.488281 49.398438 223.304688 48.628906 221.90625 48.351562 L 155.570312 48.351562 C 154.175781 48.628906 152.992188 49.398438 152.015625 50.652344 Z M 335.957031 81.832031 L 335.957031 95.226562 C 335.957031 97.179688 335.328125 98.78125 334.074219 100.039062 C 332.816406 101.292969 331.214844 101.921875 329.261719 101.921875 L 309.171875 101.921875 L 309.171875 300.304688 C 309.171875 311.882812 305.890625 321.894531 299.335938 330.332031 C 292.777344 338.773438 284.894531 342.992188 275.6875 342.992188 L 101.582031 342.992188 C 92.375 342.992188 84.492188 338.914062 77.933594 330.75 C 71.378906 322.589844 68.097656 312.71875 68.097656 301.140625 L 68.097656 101.921875 L 48.011719 101.921875 C 46.058594 101.921875 44.453125 101.292969 43.195312 100.039062 C 41.941406 98.78125 41.3125 97.179688 41.3125 95.226562 L 41.3125 81.832031 C 41.3125 79.878906 41.941406 78.273438 43.195312 77.019531 C 44.453125 75.765625 46.058594 75.136719 48.011719 75.136719 L 112.671875 75.136719 L 127.320312 40.1875 C 129.414062 35.027344 133.179688 30.632812 138.621094 27.003906 C 144.0625 23.378906 149.574219 21.566406 155.152344 21.566406 L 222.117188 21.566406 C 227.699219 21.566406 233.207031 23.378906 238.648438 27.003906 C 244.089844 30.632812 247.855469 35.027344 249.949219 40.1875 L 264.597656 75.136719 L 329.261719 75.136719 C 331.214844 75.136719 332.816406 75.765625 334.074219 77.019531 C 335.328125 78.273438 335.957031 79.878906 335.957031 81.832031 Z M 335.957031 81.832031 "/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style=" stroke:none;fill-rule:nonzero;fill:rgb(100%,0%,0%);fill-opacity:1;" |  | ||||||
|       d="M 148.457031 148.796875 L 148.457031 269.332031 C 148.457031 271.285156 147.828125 272.890625 146.574219 274.144531 C 145.316406 275.402344 143.714844 276.027344 141.761719 276.027344 L 128.367188 276.027344 C 126.414062 276.027344 124.808594 275.402344 123.554688 274.144531 C 122.296875 272.890625 121.671875 271.285156 121.671875 269.332031 L 121.671875 148.796875 C 121.671875 146.84375 122.296875 145.238281 123.554688 143.984375 C 124.808594 142.726562 126.414062 142.101562 128.367188 142.101562 L 141.761719 142.101562 C 143.714844 142.101562 145.316406 142.726562 146.574219 143.984375 C 147.828125 145.238281 148.457031 146.84375 148.457031 148.796875 Z M 202.027344 148.796875 L 202.027344 269.332031 C 202.027344 271.285156 201.398438 272.890625 200.144531 274.144531 C 198.890625 275.402344 197.285156 276.027344 195.332031 276.027344 L 181.9375 276.027344 C 179.984375 276.027344 178.382812 275.402344 177.125 274.144531 C 175.871094 272.890625 175.242188 271.285156 175.242188 269.332031 L 175.242188 148.796875 C 175.242188 146.84375 175.871094 145.238281 177.125 143.984375 C 178.382812 142.726562 179.984375 142.101562 181.9375 142.101562 L 195.332031 142.101562 C 197.285156 142.101562 198.890625 142.726562 200.144531 143.984375 C 201.398438 145.238281 202.027344 146.84375 202.027344 148.796875 Z M 255.597656 148.796875 L 255.597656 269.332031 C 255.597656 271.285156 254.972656 272.890625 253.714844 274.144531 C 252.460938 275.402344 250.855469 276.027344 248.902344 276.027344 L 235.511719 276.027344 C 233.558594 276.027344 231.953125 275.402344 230.695312 274.144531 C 229.441406 272.890625 228.8125 271.285156 228.8125 269.332031 L 228.8125 148.796875 C 228.8125 146.84375 229.441406 145.238281 230.695312 143.984375 C 231.953125 142.726562 233.558594 142.101562 235.511719 142.101562 L 248.902344 142.101562 C 250.855469 142.101562 252.460938 142.726562 253.714844 143.984375 C 254.972656 145.238281 255.597656 146.84375 255.597656 148.796875 Z M 282.386719 300.304688 L 282.386719 101.921875 L 94.886719 101.921875 L 94.886719 300.304688 C 94.886719 303.371094 95.375 306.199219 96.351562 308.777344 C 97.328125 311.359375 98.335938 313.242188 99.382812 314.429688 C 100.429688 315.613281 101.164062 316.207031 101.582031 316.207031 L 275.6875 316.207031 C 276.105469 316.207031 276.839844 315.613281 277.886719 314.429688 C 278.933594 313.242188 279.945312 311.359375 280.921875 308.777344 C 281.898438 306.199219 282.386719 303.371094 282.386719 300.304688 Z M 141.761719 75.136719 L 235.511719 75.136719 L 225.464844 50.652344 C 224.488281 49.398438 223.304688 48.628906 221.90625 48.351562 L 155.570312 48.351562 C 154.175781 48.628906 152.992188 49.398438 152.015625 50.652344 Z M 335.957031 81.832031 L 335.957031 95.226562 C 335.957031 97.179688 335.328125 98.78125 334.074219 100.039062 C 332.816406 101.292969 331.214844 101.921875 329.261719 101.921875 L 309.171875 101.921875 L 309.171875 300.304688 C 309.171875 311.882812 305.890625 321.894531 299.335938 330.332031 C 292.777344 338.773438 284.894531 342.992188 275.6875 342.992188 L 101.582031 342.992188 C 92.375 342.992188 84.492188 338.914062 77.933594 330.75 C 71.378906 322.589844 68.097656 312.71875 68.097656 301.140625 L 68.097656 101.921875 L 48.011719 101.921875 C 46.058594 101.921875 44.453125 101.292969 43.195312 100.039062 C 41.941406 98.78125 41.3125 97.179688 41.3125 95.226562 L 41.3125 81.832031 C 41.3125 79.878906 41.941406 78.273438 43.195312 77.019531 C 44.453125 75.765625 46.058594 75.136719 48.011719 75.136719 L 112.671875 75.136719 L 127.320312 40.1875 C 129.414062 35.027344 133.179688 30.632812 138.621094 27.003906 C 144.0625 23.378906 149.574219 21.566406 155.152344 21.566406 L 222.117188 21.566406 C 227.699219 21.566406 233.207031 23.378906 238.648438 27.003906 C 244.089844 30.632812 247.855469 35.027344 249.949219 40.1875 L 264.597656 75.136719 L 329.261719 75.136719 C 331.214844 75.136719 332.816406 75.765625 334.074219 77.019531 C 335.328125 78.273438 335.957031 79.878906 335.957031 81.832031 Z M 335.957031 81.832031 " |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							|  | @ -1,104 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown         xmlns:dc="http://purl.org/dc/elements/1.1/"         xmlns:cc="http://creativecommons.org/ns#"         xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"         xmlns:xlink="http://www.w3.org/1999/xlink"         xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"         xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"         xmlns="http://www.w3.org/2000/svg"         version="1.0"         width="860.50732pt"         height="860.50732pt"         viewBox="0 0 860.50732 860.50732"         preserveAspectRatio="xMidYMid meet"         id="svg14"         sodipodi:docname="direction_gradient.svg"         inkscape:version="0.92.5 (2060ec1f9f, 2020-04-08)">     <defs             id="defs18">         <linearGradient                 inkscape:collect="always"                 id="linearGradient832">             <stop                     style="stop-color:{color};stop-opacity:1;"                     offset="0"                     id="stop828"/>             <stop                     style="stop-color:{color};stop-opacity:0;"                     offset="1"                     id="stop830"/>         </linearGradient>         <radialGradient                 inkscape:collect="always"                 xlink:href="#linearGradient832"                 id="radialGradient838"                 cx="430.25363"                 cy="519.61188"                 fx="430.25363"                 fy="519.61188"                 r="305.54589"                 gradientTransform="matrix(0.95288409,-0.94890664,0.94542304,0.94938587,-470.98122,345.21193)"                 gradientUnits="userSpaceOnUse"/>     </defs>     <sodipodi:namedview             pagecolor="#ffffff"             bordercolor="#666666"             borderopacity="1"             objecttolerance="10"             gridtolerance="10"             guidetolerance="10"             inkscape:pageopacity="0"             inkscape:pageshadow="2"             inkscape:window-width="1920"             inkscape:window-height="999"             id="namedview16"             showgrid="false"             showguides="true"             inkscape:guide-bbox="true"             inkscape:zoom="0.70710678"             inkscape:cx="279.00239"             inkscape:cy="856.75313"             inkscape:window-x="0"             inkscape:window-y="0"             inkscape:window-maximized="1"             inkscape:current-layer="svg14">         <sodipodi:guide                 position="430.25363,862.49682"                 orientation="1,0"                 id="guide832"                 inkscape:locked="false"/>         <sodipodi:guide                 position="-42.427977,430.25368"                 orientation="0,1"                 id="guide834"                 inkscape:locked="false"/>         <sodipodi:guide                 position="398.27788,720.18823"                 orientation="0,1"                 id="guide840"                 inkscape:locked="false"/>     </sodipodi:namedview>     <metadata             id="metadata2">         Created by potrace 1.15, written by Peter Selinger 2001-2017         <rdf:RDF>             <cc:Work                     rdf:about="">                 <dc:format>image/svg+xml</dc:format>                 <dc:type                         rdf:resource="http://purl.org/dc/dcmitype/StillImage"/>                 <dc:title/>             </cc:Work>         </rdf:RDF>     </metadata>     <path             style="fill:url(#radialGradient838);fill-opacity:1;stroke:none;stroke-width:2.83464575;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"             d="M 735.79979,124.70799 C 654.79116,43.598883 544.88842,-2.0206645 430.25389,-2.121103 315.61937,-2.0206592 205.71663,43.598888 124.70801,124.70799 l 305.54588,305.54589 z"             id="path836"             inkscape:connector-curvature="0"             sodipodi:nodetypes="ccccc"/> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns:dc="http://purl.org/dc/elements/1.1/" |  | ||||||
|   xmlns:cc="http://creativecommons.org/ns#" |  | ||||||
|   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" |  | ||||||
|   xmlns:xlink="http://www.w3.org/1999/xlink" |  | ||||||
|   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" |  | ||||||
|   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   version="1.0" |  | ||||||
|   width="860.50732pt" |  | ||||||
|   height="860.50732pt" |  | ||||||
|   viewBox="0 0 860.50732 860.50732" |  | ||||||
|   preserveAspectRatio="xMidYMid meet" |  | ||||||
|   id="svg14" |  | ||||||
|   sodipodi:docname="direction_gradient.svg" |  | ||||||
|   inkscape:version="0.92.5 (2060ec1f9f, 2020-04-08)" |  | ||||||
| > |  | ||||||
|   <defs id="defs18"> |  | ||||||
|     <linearGradient inkscape:collect="always" id="linearGradient832"> |  | ||||||
|       <stop style="stop-color:{color};stop-opacity:1;" offset="0" id="stop828" /> |  | ||||||
|       <stop style="stop-color:{color};stop-opacity:0;" offset="1" id="stop830" /> |  | ||||||
|     </linearGradient> |  | ||||||
|     <radialGradient |  | ||||||
|       inkscape:collect="always" |  | ||||||
|       xlink:href="#linearGradient832" |  | ||||||
|       id="radialGradient838" |  | ||||||
|       cx="430.25363" |  | ||||||
|       cy="519.61188" |  | ||||||
|       fx="430.25363" |  | ||||||
|       fy="519.61188" |  | ||||||
|       r="305.54589" |  | ||||||
|       gradientTransform="matrix(0.95288409,-0.94890664,0.94542304,0.94938587,-470.98122,345.21193)" |  | ||||||
|       gradientUnits="userSpaceOnUse" |  | ||||||
|     /> |  | ||||||
|   </defs> |  | ||||||
|   <sodipodi:namedview |  | ||||||
|     pagecolor="#ffffff" |  | ||||||
|     bordercolor="#666666" |  | ||||||
|     borderopacity="1" |  | ||||||
|     objecttolerance="10" |  | ||||||
|     gridtolerance="10" |  | ||||||
|     guidetolerance="10" |  | ||||||
|     inkscape:pageopacity="0" |  | ||||||
|     inkscape:pageshadow="2" |  | ||||||
|     inkscape:window-width="1920" |  | ||||||
|     inkscape:window-height="999" |  | ||||||
|     id="namedview16" |  | ||||||
|     showgrid="false" |  | ||||||
|     showguides="true" |  | ||||||
|     inkscape:guide-bbox="true" |  | ||||||
|     inkscape:zoom="0.70710678" |  | ||||||
|     inkscape:cx="279.00239" |  | ||||||
|     inkscape:cy="856.75313" |  | ||||||
|     inkscape:window-x="0" |  | ||||||
|     inkscape:window-y="0" |  | ||||||
|     inkscape:window-maximized="1" |  | ||||||
|     inkscape:current-layer="svg14" |  | ||||||
|   > |  | ||||||
|     <sodipodi:guide |  | ||||||
|       position="430.25363,862.49682" |  | ||||||
|       orientation="1,0" |  | ||||||
|       id="guide832" |  | ||||||
|       inkscape:locked="false" |  | ||||||
|     /> |  | ||||||
|     <sodipodi:guide |  | ||||||
|       position="-42.427977,430.25368" |  | ||||||
|       orientation="0,1" |  | ||||||
|       id="guide834" |  | ||||||
|       inkscape:locked="false" |  | ||||||
|     /> |  | ||||||
|     <sodipodi:guide |  | ||||||
|       position="398.27788,720.18823" |  | ||||||
|       orientation="0,1" |  | ||||||
|       id="guide840" |  | ||||||
|       inkscape:locked="false" |  | ||||||
|     /> |  | ||||||
|   </sodipodi:namedview> |  | ||||||
|   <metadata id="metadata2"> |  | ||||||
|     Created by potrace 1.15, written by Peter Selinger 2001-2017 <rdf:RDF> |  | ||||||
|       <cc:Work rdf:about=""> |  | ||||||
|         <dc:format>image/svg+xml</dc:format> |  | ||||||
|         <dc:type rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> |  | ||||||
|         <dc:title /> |  | ||||||
|       </cc:Work> |  | ||||||
|     </rdf:RDF> |  | ||||||
|   </metadata> |  | ||||||
|   <path |  | ||||||
|     style="fill:url(#radialGradient838);fill-opacity:1;stroke:none;stroke-width:2.83464575;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" |  | ||||||
|     d="M 735.79979,124.70799 C 654.79116,43.598883 544.88842,-2.0206645 430.25389,-2.121103 315.61937,-2.0206592 205.71663,43.598888 124.70801,124.70799 l 305.54588,305.54589 z" |  | ||||||
|     id="path836" |  | ||||||
|     inkscape:connector-curvature="0" |  | ||||||
|     sodipodi:nodetypes="ccccc" |  | ||||||
|   /> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,40 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path style="fill: none !important;stroke-width:6;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(97.254902%,100%,96.078432%);stroke-opacity:1;stroke-miterlimit:4;" d="M -177.48351 -16.993714 C -177.484166 101.48875 -226.288922 214.739751 -312.411923 296.10684 C -398.528411 377.467771 -514.363074 419.770216 -632.651731 413.060164 " transform="matrix(-0.316636,-0.299423,0.299423,-0.316636,0,0)"/>     <path style="fill: none !important;stroke-width:6;stroke-linecap:butt;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" d="M 743.478328 134.561833 L 430.253662 430.253662 L 117.002105 134.508051 " transform="matrix(0.435789,0,0,0.435789,0,0)"/>     <path style="fill: none !important;stroke-width:3;stroke-linecap:round;stroke-linejoin:round;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M -177.48351 -16.993714 C -177.484166 101.48875 -226.288922 214.739751 -312.411923 296.10684 C -398.528411 377.467771 -514.363074 419.770216 -632.651731 413.060164 " transform="matrix(-0.316636,-0.299423,0.299423,-0.316636,0,0)"/>     <path style="fill: none !important;stroke-width:3;stroke-linecap:butt;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" d="M 743.478328 134.561833 L 430.253662 430.253662 L 117.002105 134.508051 " transform="matrix(0.435789,0,0,0.435789,0,0)"/>   </g> </svg> | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:6;stroke-linecap:round;stroke-linejoin:round;stroke:rgb(97.254902%,100%,96.078432%);stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M -177.48351 -16.993714 C -177.484166 101.48875 -226.288922 214.739751 -312.411923 296.10684 C -398.528411 377.467771 -514.363074 419.770216 -632.651731 413.060164 " |  | ||||||
|       transform="matrix(-0.316636,-0.299423,0.299423,-0.316636,0,0)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:6;stroke-linecap:butt;stroke-linejoin:miter;stroke:rgb(100%,100%,100%);stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 743.478328 134.561833 L 430.253662 430.253662 L 117.002105 134.508051 " |  | ||||||
|       transform="matrix(0.435789,0,0,0.435789,0,0)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:3;stroke-linecap:round;stroke-linejoin:round;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M -177.48351 -16.993714 C -177.484166 101.48875 -226.288922 214.739751 -312.411923 296.10684 C -398.528411 377.467771 -514.363074 419.770216 -632.651731 413.060164 " |  | ||||||
|       transform="matrix(-0.316636,-0.299423,0.299423,-0.316636,0,0)" |  | ||||||
|     /> |  | ||||||
|     <path |  | ||||||
|       style="fill: none !important;stroke-width:3;stroke-linecap:butt;stroke-linejoin:miter;stroke:{color};stroke-opacity:1;stroke-miterlimit:4;" |  | ||||||
|       d="M 743.478328 134.561833 L 430.253662 430.253662 L 117.002105 134.508051 " |  | ||||||
|       transform="matrix(0.435789,0,0,0.435789,0,0)" |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
|  | @ -1,23 +1,4 @@ | ||||||
| <script> | <script> | ||||||
|   export let color = "#000000" | export let color = "#000000" | ||||||
| </script> | </script> | ||||||
| 
 |  <svg {...$$restProps} on:click on:mouseover on:mouseenter on:mouseleave on:keydown xmlns="http://www.w3.org/2000/svg" width="375px" height="375px" viewBox="0 0 375 375" version="1.1">   <g id="surface1">     <path d="M 280.75 123.625 L 218.375 123.625 L 218.375 27.8125 C 218.375 19.0625 211.375 11.875 202.75 11.875 L 171.5625 11.875 C 163 11.875 155.9375 19.0625 155.9375 27.8125 L 155.9375 123.625 L 93.5625 123.625 L 187.125 251.375 Z M 351 171.5625 C 338.0625 171.5625 327.625 182.3125 327.625 195.5 L 327.625 315.25 L 46.8125 315.25 L 46.8125 195.5 C 46.8125 182.25 36.3125 171.5625 23.4375 171.5625 C 10.5 171.5625 0 182.25 0 195.5 L 0 339.1875 C 0 352.375 10.5 363.125 23.375 363.125 L 350.9375 363.125 C 363.875 363.125 374.3125 352.375 374.3125 339.1875 L 374.3125 195.5 C 374.375 182.25 363.875 171.5625 351 171.5625 Z M 351 171.5625 "/>   </g> </svg>  | ||||||
| <svg |  | ||||||
|   {...$$restProps} |  | ||||||
|   on:click |  | ||||||
|   on:mouseover |  | ||||||
|   on:mouseenter |  | ||||||
|   on:mouseleave |  | ||||||
|   on:keydown |  | ||||||
|   xmlns="http://www.w3.org/2000/svg" |  | ||||||
|   width="375px" |  | ||||||
|   height="375px" |  | ||||||
|   viewBox="0 0 375 375" |  | ||||||
|   version="1.1" |  | ||||||
| > |  | ||||||
|   <g id="surface1"> |  | ||||||
|     <path |  | ||||||
|       d="M 280.75 123.625 L 218.375 123.625 L 218.375 27.8125 C 218.375 19.0625 211.375 11.875 202.75 11.875 L 171.5625 11.875 C 163 11.875 155.9375 19.0625 155.9375 27.8125 L 155.9375 123.625 L 93.5625 123.625 L 187.125 251.375 Z M 351 171.5625 C 338.0625 171.5625 327.625 182.3125 327.625 195.5 L 327.625 315.25 L 46.8125 315.25 L 46.8125 195.5 C 46.8125 182.25 36.3125 171.5625 23.4375 171.5625 C 10.5 171.5625 0 182.25 0 195.5 L 0 339.1875 C 0 352.375 10.5 363.125 23.375 363.125 L 350.9375 363.125 C 363.875 363.125 374.3125 352.375 374.3125 339.1875 L 374.3125 195.5 C 374.375 182.25 363.875 171.5625 351 171.5625 Z M 351 171.5625 " |  | ||||||
|     /> |  | ||||||
|   </g> |  | ||||||
| </svg> |  | ||||||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue