forked from MapComplete/MapComplete
		
	Refactoring: split 'menuDrawer' into two parts to prevent the index to load to much modules
This commit is contained in:
		
							parent
							
								
									b398f0d8bb
								
							
						
					
					
						commit
						2afca781b9
					
				
					 3 changed files with 352 additions and 273 deletions
				
			
		|  | @ -24,6 +24,7 @@ | ||||||
|   import { MenuState } from "../Models/MenuState" |   import { MenuState } from "../Models/MenuState" | ||||||
|   import { MenuIcon } from "@rgossiaux/svelte-heroicons/solid" |   import { MenuIcon } from "@rgossiaux/svelte-heroicons/solid" | ||||||
|   import AccordionSingle from "./Flowbite/AccordionSingle.svelte" |   import AccordionSingle from "./Flowbite/AccordionSingle.svelte" | ||||||
|  |   import MenuDrawerIndex from "./BigComponents/MenuDrawerIndex.svelte" | ||||||
| 
 | 
 | ||||||
|   AndroidPolyfill.init().then(() => console.log("Android polyfill setup completed")) |   AndroidPolyfill.init().then(() => console.log("Android polyfill setup completed")) | ||||||
|   const featureSwitches = new OsmConnectionFeatureSwitches() |   const featureSwitches = new OsmConnectionFeatureSwitches() | ||||||
|  | @ -156,7 +157,7 @@ | ||||||
|   <div class="h-full overflow-hidden"> |   <div class="h-full overflow-hidden"> | ||||||
|     <DrawerLeft shown={guistate.pageStates.menu}> |     <DrawerLeft shown={guistate.pageStates.menu}> | ||||||
|       <div class="h-screen overflow-y-auto"> |       <div class="h-screen overflow-y-auto"> | ||||||
|         <MenuDrawer onlyLink={true} state={menuDrawerState} /> |         <MenuDrawerIndex onlyLink={true} state={menuDrawerState} /> | ||||||
|       </div> |       </div> | ||||||
|     </DrawerLeft> |     </DrawerLeft> | ||||||
|   </div> |   </div> | ||||||
|  |  | ||||||
|  | @ -1,55 +1,28 @@ | ||||||
| <script lang="ts"> | <script lang="ts"> | ||||||
|   import Translations from "../i18n/Translations" |   import Translations from "../i18n/Translations" | ||||||
|   import { CogIcon, EyeIcon, HeartIcon, TranslateIcon } from "@rgossiaux/svelte-heroicons/solid" |  | ||||||
|   import Page from "../Base/Page.svelte" |   import Page from "../Base/Page.svelte" | ||||||
|   import PrivacyPolicy from "./PrivacyPolicy.svelte" |  | ||||||
|   import Tr from "../Base/Tr.svelte" |   import Tr from "../Base/Tr.svelte" | ||||||
|   import If from "../Base/If.svelte" |   import If from "../Base/If.svelte" | ||||||
|   import CommunityIndexView from "./CommunityIndexView.svelte" |   import CommunityIndexView from "./CommunityIndexView.svelte" | ||||||
|   import Community from "../../assets/svg/Community.svelte" |   import Community from "../../assets/svg/Community.svelte" | ||||||
|   import LoginToggle from "../Base/LoginToggle.svelte" |  | ||||||
|   import { CloseButton } from "flowbite-svelte" |  | ||||||
|   import HotkeyTable from "./HotkeyTable.svelte" |  | ||||||
|   import { Utils } from "../../Utils" |   import { Utils } from "../../Utils" | ||||||
|   import Constants from "../../Models/Constants" |  | ||||||
|   import Mastodon from "../../assets/svg/Mastodon.svelte" |  | ||||||
|   import Liberapay from "../../assets/svg/Liberapay.svelte" |  | ||||||
|   import DocumentMagnifyingGlass from "@babeard/svelte-heroicons/outline/DocumentMagnifyingGlass" |   import DocumentMagnifyingGlass from "@babeard/svelte-heroicons/outline/DocumentMagnifyingGlass" | ||||||
|   import OpenIdEditor from "./OpenIdEditor.svelte" |   import OpenIdEditor from "./OpenIdEditor.svelte" | ||||||
|   import OpenJosm from "../Base/OpenJosm.svelte" |   import OpenJosm from "../Base/OpenJosm.svelte" | ||||||
|   import MapillaryLink from "./MapillaryLink.svelte" |   import MapillaryLink from "./MapillaryLink.svelte" | ||||||
|   import Bug from "../../assets/svg/Bug.svelte" |  | ||||||
|   import CopyrightPanel from "./CopyrightPanel.svelte" |  | ||||||
|   import CopyrightAllIcons from "./CopyrightAllIcons.svelte" |  | ||||||
|   import LanguagePicker from "../InputElement/LanguagePicker.svelte" |  | ||||||
|   import LoginButton from "../Base/LoginButton.svelte" |  | ||||||
|   import SelectedElementView from "./SelectedElementView.svelte" |  | ||||||
|   import LayerConfig from "../../Models/ThemeConfig/LayerConfig" |  | ||||||
|   import type { LayerConfigJson } from "../../Models/ThemeConfig/Json/LayerConfigJson" |  | ||||||
|   import usersettings from "../../../public/assets/generated/layers/usersettings.json" |  | ||||||
|   import UserRelatedState from "../../Logic/State/UserRelatedState" |   import UserRelatedState from "../../Logic/State/UserRelatedState" | ||||||
|   import ArrowDownTray from "@babeard/svelte-heroicons/mini/ArrowDownTray" |   import ArrowDownTray from "@babeard/svelte-heroicons/mini/ArrowDownTray" | ||||||
|   import DownloadPanel from "../DownloadFlow/DownloadPanel.svelte" |   import DownloadPanel from "../DownloadFlow/DownloadPanel.svelte" | ||||||
|   import Favourites from "../Favourites/Favourites.svelte" |  | ||||||
|   import ReviewsOverview from "../Reviews/ReviewsOverview.svelte" |  | ||||||
|   import Share from "@babeard/svelte-heroicons/solid/Share" |   import Share from "@babeard/svelte-heroicons/solid/Share" | ||||||
|   import ShareScreen from "./ShareScreen.svelte" |   import ShareScreen from "./ShareScreen.svelte" | ||||||
|   import FilterPage from "./FilterPage.svelte" |   import FilterPage from "./FilterPage.svelte" | ||||||
|   import RasterLayerOverview from "../Map/RasterLayerOverview.svelte" |   import RasterLayerOverview from "../Map/RasterLayerOverview.svelte" | ||||||
|   import ThemeIntroPanel from "./ThemeIntroPanel.svelte" |   import ThemeIntroPanel from "./ThemeIntroPanel.svelte" | ||||||
|   import Marker from "../Map/Marker.svelte" |   import Marker from "../Map/Marker.svelte" | ||||||
|   import LogoutButton from "../Base/LogoutButton.svelte" |   import { ShareIcon } from "@babeard/svelte-heroicons/mini" | ||||||
|   import { BoltIcon, ShareIcon } from "@babeard/svelte-heroicons/mini" |  | ||||||
|   import Copyright from "../../assets/svg/Copyright.svelte" |  | ||||||
|   import Pencil from "../../assets/svg/Pencil.svelte" |  | ||||||
|   import SidebarUnit from "../Base/SidebarUnit.svelte" |   import SidebarUnit from "../Base/SidebarUnit.svelte" | ||||||
|   import Squares2x2 from "@babeard/svelte-heroicons/mini/Squares2x2" |  | ||||||
|   import EnvelopeOpen from "@babeard/svelte-heroicons/mini/EnvelopeOpen" |  | ||||||
|   import PanoramaxLink from "./PanoramaxLink.svelte" |   import PanoramaxLink from "./PanoramaxLink.svelte" | ||||||
|   import { UIEventSource } from "../../Logic/UIEventSource" |   import { UIEventSource } from "../../Logic/UIEventSource" | ||||||
|   import MagnifyingGlassCircle from "@babeard/svelte-heroicons/mini/MagnifyingGlassCircle" |  | ||||||
|   import { AndroidPolyfill } from "../../Logic/Web/AndroidPolyfill" |  | ||||||
|   import Forgejo from "../../assets/svg/Forgejo.svelte" |  | ||||||
|   import ChartBar from "@babeard/svelte-heroicons/solid/ChartBar" |   import ChartBar from "@babeard/svelte-heroicons/solid/ChartBar" | ||||||
|   import QueueList from "@babeard/svelte-heroicons/solid/QueueList" |   import QueueList from "@babeard/svelte-heroicons/solid/QueueList" | ||||||
|   import { MenuState } from "../../Models/MenuState" |   import { MenuState } from "../../Models/MenuState" | ||||||
|  | @ -59,12 +32,9 @@ | ||||||
|   import type { MapProperties } from "../../Models/MapProperties" |   import type { MapProperties } from "../../Models/MapProperties" | ||||||
|   import FavouritesFeatureSource from "../../Logic/FeatureSource/Sources/FavouritesFeatureSource" |   import FavouritesFeatureSource from "../../Logic/FeatureSource/Sources/FavouritesFeatureSource" | ||||||
|   import Hotkeys from "../Base/Hotkeys" |   import Hotkeys from "../Base/Hotkeys" | ||||||
|   import { ArrowTrendingUp } from "@babeard/svelte-heroicons/solid/ArrowTrendingUp" |   import MenuDrawerIndex from "./MenuDrawerIndex.svelte" | ||||||
|   import ArrowTopRightOnSquare from "@babeard/svelte-heroicons/mini/ArrowTopRightOnSquare" |  | ||||||
|   import { PhotoIcon } from "@babeard/svelte-heroicons/outline" |  | ||||||
|   import ImageUploadQueue from "../../Logic/ImageProviders/ImageUploadQueue" |  | ||||||
|   import QueuedImagesView from "../Image/QueuedImagesView.svelte" |  | ||||||
| 
 | 
 | ||||||
|  |   export let onlyLink: boolean | ||||||
|   export let state: { |   export let state: { | ||||||
|     favourites: FavouritesFeatureSource |     favourites: FavouritesFeatureSource | ||||||
|     guistate: MenuState |     guistate: MenuState | ||||||
|  | @ -77,14 +47,10 @@ | ||||||
|   let hotkeys = Hotkeys._docs |   let hotkeys = Hotkeys._docs | ||||||
|   let userdetails = state.osmConnection.userDetails |   let userdetails = state.osmConnection.userDetails | ||||||
| 
 | 
 | ||||||
|   let usersettingslayer = new LayerConfig(<LayerConfigJson>usersettings, "usersettings", true) |  | ||||||
| 
 |  | ||||||
|   let theme = state.theme |   let theme = state.theme | ||||||
|   let featureSwitches = state.featureSwitches |   let featureSwitches = state.featureSwitches | ||||||
|   let showHome = featureSwitches?.featureSwitchBackToThemeOverview |  | ||||||
|   let pg = state.guistate.pageStates |   let pg = state.guistate.pageStates | ||||||
|   let location = state.mapProperties?.location |   let location = state.mapProperties?.location | ||||||
|   export let onlyLink: boolean |  | ||||||
|   const t = Translations.t.general.menu |   const t = Translations.t.general.menu | ||||||
|   let shown = new UIEventSource(state.guistate.pageStates.menu.data || !onlyLink) |   let shown = new UIEventSource(state.guistate.pageStates.menu.data || !onlyLink) | ||||||
|   state.guistate.pageStates.menu.addCallback((isShown) => { |   state.guistate.pageStates.menu.addCallback((isShown) => { | ||||||
|  | @ -99,119 +65,13 @@ | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
|   let isAndroid = AndroidPolyfill.inAndroid |  | ||||||
|   let nrOfFailedImages = ImageUploadQueue.singleton.imagesInQueue |  | ||||||
|   let failedImagesOpen = pg.failedImages |  | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <div | <MenuDrawerIndex {state} {onlyLink} > | ||||||
|   class="low-interaction flex h-screen flex-col gap-y-2 overflow-y-auto p-2 sm:gap-y-3 sm:p-3" |  | ||||||
|   class:hidden={!$shown} |  | ||||||
| > |  | ||||||
|   <div class="flex justify-between"> |  | ||||||
|     <h2> |  | ||||||
|       <Tr t={t.title} /> |  | ||||||
|     </h2> |  | ||||||
|     <CloseButton |  | ||||||
|       on:click={() => { |  | ||||||
|         pg.menu.set(false) |  | ||||||
|       }} |  | ||||||
|     /> |  | ||||||
|   </div> |  | ||||||
|   {#if $showHome} |  | ||||||
|     <a class="button primary flex" href={Utils.HomepageLink()}> |  | ||||||
|       <Squares2x2 class="h-10 w-10" /> |  | ||||||
|       {#if Utils.isIframe} |  | ||||||
|         <Tr t={Translations.t.general.seeIndex} /> |  | ||||||
|       {:else} |  | ||||||
|         <Tr t={Translations.t.general.backToIndex} /> |  | ||||||
|       {/if} |  | ||||||
|     </a> |  | ||||||
|   {/if} |  | ||||||
| 
 |  | ||||||
|   <!-- User related: avatar, settings, favourits, logout --> |  | ||||||
|   <SidebarUnit> |  | ||||||
|     <LoginToggle {state}> |  | ||||||
|       <LoginButton osmConnection={state.osmConnection} slot="not-logged-in" /> |  | ||||||
|       <div class="flex items-center gap-x-4"> |  | ||||||
|         {#if $userdetails.img} |  | ||||||
|           <img alt="avatar" src={$userdetails.img} class="h-14 w-14 rounded-full" /> |  | ||||||
|         {/if} |  | ||||||
|         <b>{$userdetails.name}</b> |  | ||||||
|       </div> |  | ||||||
|     </LoginToggle> |  | ||||||
| 
 |  | ||||||
|     <Page {onlyLink} shown={pg.usersettings} bodyPadding="p-0 pb-4"> |  | ||||||
|       <svelte:fragment slot="header"> |  | ||||||
|         <CogIcon /> |  | ||||||
|         <Tr t={UserRelatedState.usersettingsConfig.title.GetRenderValue({})} /> |  | ||||||
|       </svelte:fragment> |  | ||||||
| 
 |  | ||||||
|       <!-- All shown components are set by 'usersettings.json', which happily uses some special visualisations created specifically for it --> |  | ||||||
|       <SelectedElementView |  | ||||||
|         highlightedRendering={state.guistate.highlightedUserSetting} |  | ||||||
|         layer={usersettingslayer} |  | ||||||
|         selectedElement={{ |  | ||||||
|           type: "Feature", |  | ||||||
|           properties: { id: "settings" }, |  | ||||||
|           geometry: { type: "Point", coordinates: [0, 0] }, |  | ||||||
|         }} |  | ||||||
|         {state} |  | ||||||
|         tags={state.userRelatedState.preferencesAsTags} |  | ||||||
|       /> |  | ||||||
|     </Page> |  | ||||||
| 
 |  | ||||||
|     {#if $nrOfFailedImages.length > 0 || $failedImagesOpen} |  | ||||||
|       <Page {onlyLink} shown={pg.failedImages} bodyPadding="p-0 pb-4"> |  | ||||||
|         <svelte:fragment slot="header"> |  | ||||||
|           <PhotoIcon /> |  | ||||||
|           <Tr t={Translations.t.imageQueue.menu.Subs({ count: $nrOfFailedImages.length })} /> |  | ||||||
|         </svelte:fragment> |  | ||||||
|         <QueuedImagesView {state} /> |  | ||||||
|       </Page> |  | ||||||
|     {/if} |  | ||||||
| 
 |  | ||||||
|     <LoginToggle {state} silentFail> |  | ||||||
|       {#if state.favourites} |  | ||||||
|         <Page {onlyLink} shown={pg.favourites}> |  | ||||||
|           <svelte:fragment slot="header"> |  | ||||||
|             <HeartIcon /> |  | ||||||
|             <Tr t={Translations.t.favouritePoi.tab} /> |  | ||||||
|           </svelte:fragment> |  | ||||||
| 
 |  | ||||||
|           <h3> |  | ||||||
|             <Tr t={Translations.t.favouritePoi.title} /> |  | ||||||
|           </h3> |  | ||||||
|           <div> |  | ||||||
|             <Favourites {state} /> |  | ||||||
|             <h3> |  | ||||||
|               <Tr t={Translations.t.reviews.your_reviews} /> |  | ||||||
|             </h3> |  | ||||||
|             <ReviewsOverview {state} /> |  | ||||||
|           </div> |  | ||||||
|         </Page> |  | ||||||
|       {/if} |  | ||||||
|       {#if $hotkeys.length > 0} |  | ||||||
|         <div class="hidden-on-mobile w-full"> |  | ||||||
|           <Page {onlyLink} shown={pg.hotkeys}> |  | ||||||
|             <svelte:fragment slot="header"> |  | ||||||
|               <BoltIcon /> |  | ||||||
|               <Tr t={Translations.t.hotkeyDocumentation.title} /> |  | ||||||
|             </svelte:fragment> |  | ||||||
|             <HotkeyTable /> |  | ||||||
|           </Page> |  | ||||||
|         </div> |  | ||||||
|       {/if} |  | ||||||
| 
 |  | ||||||
|       <div class="self-end"> |  | ||||||
|         <LogoutButton osmConnection={state.osmConnection} /> |  | ||||||
|       </div> |  | ||||||
|     </LoginToggle> |  | ||||||
|     <LanguagePicker /> |  | ||||||
|   </SidebarUnit> |  | ||||||
| 
 | 
 | ||||||
|   <!-- Theme related: documentation links, download, ... --> |   <!-- Theme related: documentation links, download, ... --> | ||||||
|   {#if state.theme} |   <svelte:fragment slot="theme-tools"> | ||||||
|  |     {#if state.theme} | ||||||
|     <SidebarUnit> |     <SidebarUnit> | ||||||
|       <h3> |       <h3> | ||||||
|         <Tr t={t.aboutCurrentThemeTitle} /> |         <Tr t={t.aboutCurrentThemeTitle} /> | ||||||
|  | @ -282,137 +142,39 @@ | ||||||
|       {/if} |       {/if} | ||||||
|     </SidebarUnit> |     </SidebarUnit> | ||||||
|   {/if} |   {/if} | ||||||
|  |   </svelte:fragment> | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
|   <!-- Other links and tools for the given location: open iD/JOSM; community index, ... --> |   <!-- Other links and tools for the given location: open iD/JOSM; community index, ... --> | ||||||
|   {#if state.mapProperties?.location} |   <svelte:fragment  slot="location-tools"> | ||||||
|     <SidebarUnit> |     {#if state.mapProperties?.location} | ||||||
|       <h3> |       <SidebarUnit> | ||||||
|         <Tr t={t.moreUtilsTitle} /> |         <h3> | ||||||
|       </h3> |           <Tr t={t.moreUtilsTitle} /> | ||||||
|  |         </h3> | ||||||
| 
 | 
 | ||||||
|       <Page {onlyLink} shown={pg.community_index}> |         <Page {onlyLink} shown={pg.community_index}> | ||||||
|         <svelte:fragment slot="header"> |           <svelte:fragment slot="header"> | ||||||
|           <Community /> |             <Community /> | ||||||
|           <Tr t={Translations.t.communityIndex.title} /> |             <Tr t={Translations.t.communityIndex.title} /> | ||||||
|         </svelte:fragment> |           </svelte:fragment> | ||||||
|         <CommunityIndexView location={state.mapProperties.location} /> |           <CommunityIndexView location={state.mapProperties.location} /> | ||||||
|       </Page> |         </Page> | ||||||
| 
 | 
 | ||||||
|       <If condition={featureSwitches?.featureSwitchEnableLogin}> |         <If condition={featureSwitches?.featureSwitchEnableLogin}> | ||||||
|         <OpenIdEditor mapProperties={state.mapProperties} /> |           <OpenIdEditor mapProperties={state.mapProperties} /> | ||||||
|         <OpenJosm {state} /> |           <OpenJosm {state} /> | ||||||
|         <PanoramaxLink large={false} mapProperties={state.mapProperties} /> |           <PanoramaxLink large={false} mapProperties={state.mapProperties} /> | ||||||
|         <MapillaryLink large={false} mapProperties={state.mapProperties} /> |           <MapillaryLink large={false} mapProperties={state.mapProperties} /> | ||||||
|       </If> |         </If> | ||||||
| 
 | 
 | ||||||
|       <a class="sidebar-button flex" href="geo:{$location.lat},{$location.lon}"> |         <a class="sidebar-button flex" href="geo:{$location.lat},{$location.lon}"> | ||||||
|         <ShareIcon /> |           <ShareIcon /> | ||||||
|         <Tr t={t.openHereDifferentApp} /> |           <Tr t={t.openHereDifferentApp} /> | ||||||
|       </a> |         </a> | ||||||
|     </SidebarUnit> |       </SidebarUnit> | ||||||
|   {/if} |  | ||||||
| 
 |  | ||||||
|   <!-- About MC: various outward links, legal info, ... --> |  | ||||||
|   <SidebarUnit> |  | ||||||
|     <h3> |  | ||||||
|       <Tr t={Translations.t.general.menu.aboutMapComplete} /> |  | ||||||
|     </h3> |  | ||||||
| 
 |  | ||||||
|     <a class="flex" href={window.location.protocol + "//" + window.location.host + "/studio.html"}> |  | ||||||
|       <Pencil class="mr-2 h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.general.morescreen.createYourOwnTheme} /> |  | ||||||
|     </a> |  | ||||||
| 
 |  | ||||||
|     <a class="flex" href="mailto:info@mapcomplete.org"> |  | ||||||
|       <EnvelopeOpen class="h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.general.attribution.emailCreators} /> |  | ||||||
|     </a> |  | ||||||
| 
 |  | ||||||
|     <a class="flex" href="https://en.osm.town/@MapComplete" target="_blank"> |  | ||||||
|       <Mastodon class="h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.general.attribution.followOnMastodon} /> |  | ||||||
|     </a> |  | ||||||
| 
 |  | ||||||
|     <a class="flex" href="https://liberapay.com/pietervdvn/" target="_blank"> |  | ||||||
|       <Liberapay class="h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.general.attribution.donate} /> |  | ||||||
|     </a> |  | ||||||
| 
 |  | ||||||
|     <a |  | ||||||
|       class="flex" |  | ||||||
|       href="https://source.mapcomplete.org/MapComplete/MapComplete/issues" |  | ||||||
|       target="_blank" |  | ||||||
|     > |  | ||||||
|       <Bug class="h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.general.attribution.openIssueTracker} /> |  | ||||||
|     </a> |  | ||||||
| 
 |  | ||||||
|     <a class="flex" href="https://source.mapcomplete.org/MapComplete/MapComplete/" target="_blank"> |  | ||||||
|       <Forgejo class="h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.general.attribution.gotoSourceCode} /> |  | ||||||
|     </a> |  | ||||||
|     <a class="flex" href={`${Constants.weblate}projects/mapcomplete/`} target="_blank"> |  | ||||||
|       <TranslateIcon class="h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.translations.activateButton} /> |  | ||||||
|     </a> |  | ||||||
| 
 |  | ||||||
|     <a |  | ||||||
|       class="flex" |  | ||||||
|       href={window.location.protocol + "//" + window.location.host + "/inspector.html"} |  | ||||||
|     > |  | ||||||
|       <MagnifyingGlassCircle class="mr-2 h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.inspector.menu} /> |  | ||||||
|     </a> |  | ||||||
| 
 |  | ||||||
|     {#if !state.theme} |  | ||||||
|       <a class="flex" href={`./statistics.html`} target="_blank"> |  | ||||||
|         <ChartBar class="h-6 w-6" /> |  | ||||||
|         <Tr t={Translations.t.general.attribution.openStatistics.Subs({ theme: "MapComplete" })} /> |  | ||||||
|       </a> |  | ||||||
|     {/if} |     {/if} | ||||||
|  |   </svelte:fragment> | ||||||
| 
 | 
 | ||||||
|     <a class="flex" href={Utils.OsmChaLinkFor(7)} target="_blank"> |  | ||||||
|       <ArrowTrendingUp class="mr-2 h-6 w-6" /> |  | ||||||
|       <Tr t={Translations.t.general.attribution.openOsmchaLastWeek} /> |  | ||||||
|     </a> |  | ||||||
|   </SidebarUnit> |  | ||||||
|   <SidebarUnit> |  | ||||||
|     <h3> |  | ||||||
|       <Tr t={Translations.t.general.menu.legal} /> |  | ||||||
|     </h3> |  | ||||||
|     <Page {onlyLink} shown={pg.copyright}> |  | ||||||
|       <svelte:fragment slot="header"> |  | ||||||
|         <Copyright /> |  | ||||||
|         <Tr t={Translations.t.general.attribution.attributionTitle} /> |  | ||||||
|       </svelte:fragment> |  | ||||||
|       <CopyrightPanel {state} /> |  | ||||||
|     </Page> |  | ||||||
|     {#if state?.theme} |  | ||||||
|       <Page {onlyLink} shown={pg.copyright_icons}> |  | ||||||
|         <svelte:fragment slot="header"> |  | ||||||
|           <Copyright /> |  | ||||||
|           <Tr t={Translations.t.general.attribution.iconAttribution.title} /> |  | ||||||
|         </svelte:fragment> |  | ||||||
|         <CopyrightAllIcons {state} /> |  | ||||||
|       </Page> |  | ||||||
|     {/if} |  | ||||||
| 
 | 
 | ||||||
|     <Page {onlyLink} shown={pg.privacy}> | </MenuDrawerIndex> | ||||||
|       <svelte:fragment slot="header"> |  | ||||||
|         <EyeIcon /> |  | ||||||
|         <Tr t={Translations.t.privacy.title} /> |  | ||||||
|       </svelte:fragment> |  | ||||||
|       <PrivacyPolicy {state} /> |  | ||||||
|       <a href="./privacy.html" class="button float-right w-fit" target="_blank"> |  | ||||||
|         <ArrowTopRightOnSquare class="h-8 w-8" /> |  | ||||||
|       </a> |  | ||||||
|     </Page> |  | ||||||
|   </SidebarUnit> |  | ||||||
| 
 |  | ||||||
|   <div class="subtle self-end"> |  | ||||||
|     {Constants.vNumber} |  | ||||||
|     {#if $isAndroid} |  | ||||||
|       Android |  | ||||||
|     {/if} |  | ||||||
|   </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
							
								
								
									
										316
									
								
								src/UI/BigComponents/MenuDrawerIndex.svelte
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										316
									
								
								src/UI/BigComponents/MenuDrawerIndex.svelte
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,316 @@ | ||||||
|  | <script lang="ts"> | ||||||
|  |   /** | ||||||
|  |    * All the generic (non-theme/map view specific) components of the drawer, e.g.: | ||||||
|  |    * | ||||||
|  |    * - user settings | ||||||
|  |    * - mapcomplete links | ||||||
|  |    * - ... | ||||||
|  |    * | ||||||
|  |    * But NOT: open ID here, info about the curren theme, ... | ||||||
|  |    * Slots are provided for that | ||||||
|  |    */ | ||||||
|  |   import Translations from "../i18n/Translations" | ||||||
|  |   import { CogIcon, EyeIcon, HeartIcon, TranslateIcon } from "@rgossiaux/svelte-heroicons/solid" | ||||||
|  |   import Page from "../Base/Page.svelte" | ||||||
|  |   import PrivacyPolicy from "./PrivacyPolicy.svelte" | ||||||
|  |   import Tr from "../Base/Tr.svelte" | ||||||
|  |   import LoginToggle from "../Base/LoginToggle.svelte" | ||||||
|  |   import { CloseButton } from "flowbite-svelte" | ||||||
|  |   import HotkeyTable from "./HotkeyTable.svelte" | ||||||
|  |   import { Utils } from "../../Utils" | ||||||
|  |   import Constants from "../../Models/Constants" | ||||||
|  |   import Mastodon from "../../assets/svg/Mastodon.svelte" | ||||||
|  |   import Liberapay from "../../assets/svg/Liberapay.svelte" | ||||||
|  |   import Bug from "../../assets/svg/Bug.svelte" | ||||||
|  |   import CopyrightPanel from "./CopyrightPanel.svelte" | ||||||
|  |   import CopyrightAllIcons from "./CopyrightAllIcons.svelte" | ||||||
|  |   import LanguagePicker from "../InputElement/LanguagePicker.svelte" | ||||||
|  |   import LoginButton from "../Base/LoginButton.svelte" | ||||||
|  |   import SelectedElementView from "./SelectedElementView.svelte" | ||||||
|  |   import LayerConfig from "../../Models/ThemeConfig/LayerConfig" | ||||||
|  |   import type { LayerConfigJson } from "../../Models/ThemeConfig/Json/LayerConfigJson" | ||||||
|  |   import usersettings from "../../../public/assets/generated/layers/usersettings.json" | ||||||
|  |   import UserRelatedState from "../../Logic/State/UserRelatedState" | ||||||
|  |   import ArrowDownTray from "@babeard/svelte-heroicons/mini/ArrowDownTray" | ||||||
|  |   import Favourites from "../Favourites/Favourites.svelte" | ||||||
|  |   import ReviewsOverview from "../Reviews/ReviewsOverview.svelte" | ||||||
|  |   import Share from "@babeard/svelte-heroicons/solid/Share" | ||||||
|  |   import LogoutButton from "../Base/LogoutButton.svelte" | ||||||
|  |   import { BoltIcon, ShareIcon } from "@babeard/svelte-heroicons/mini" | ||||||
|  |   import Copyright from "../../assets/svg/Copyright.svelte" | ||||||
|  |   import Pencil from "../../assets/svg/Pencil.svelte" | ||||||
|  |   import SidebarUnit from "../Base/SidebarUnit.svelte" | ||||||
|  |   import Squares2x2 from "@babeard/svelte-heroicons/mini/Squares2x2" | ||||||
|  |   import EnvelopeOpen from "@babeard/svelte-heroicons/mini/EnvelopeOpen" | ||||||
|  |   import { UIEventSource } from "../../Logic/UIEventSource" | ||||||
|  |   import MagnifyingGlassCircle from "@babeard/svelte-heroicons/mini/MagnifyingGlassCircle" | ||||||
|  |   import { AndroidPolyfill } from "../../Logic/Web/AndroidPolyfill" | ||||||
|  |   import Forgejo from "../../assets/svg/Forgejo.svelte" | ||||||
|  |   import ChartBar from "@babeard/svelte-heroicons/solid/ChartBar" | ||||||
|  |   import { MenuState } from "../../Models/MenuState" | ||||||
|  |   import { OsmConnection } from "../../Logic/Osm/OsmConnection" | ||||||
|  |   import FeatureSwitchState from "../../Logic/State/FeatureSwitchState" | ||||||
|  |   import ThemeConfig from "../../Models/ThemeConfig/ThemeConfig" | ||||||
|  |   import type { MapProperties } from "../../Models/MapProperties" | ||||||
|  |   import FavouritesFeatureSource from "../../Logic/FeatureSource/Sources/FavouritesFeatureSource" | ||||||
|  |   import Hotkeys from "../Base/Hotkeys" | ||||||
|  |   import { ArrowTrendingUp } from "@babeard/svelte-heroicons/solid/ArrowTrendingUp" | ||||||
|  |   import ArrowTopRightOnSquare from "@babeard/svelte-heroicons/mini/ArrowTopRightOnSquare" | ||||||
|  |   import { PhotoIcon } from "@babeard/svelte-heroicons/outline" | ||||||
|  |   import ImageUploadQueue from "../../Logic/ImageProviders/ImageUploadQueue" | ||||||
|  |   import QueuedImagesView from "../Image/QueuedImagesView.svelte" | ||||||
|  | 
 | ||||||
|  |   export let state: { | ||||||
|  |     favourites: FavouritesFeatureSource | ||||||
|  |     guistate: MenuState | ||||||
|  |     osmConnection: OsmConnection | ||||||
|  |     theme?: ThemeConfig | ||||||
|  |     featureSwitches: Partial<FeatureSwitchState> | ||||||
|  |     mapProperties?: MapProperties | ||||||
|  |     userRelatedState?: UserRelatedState | ||||||
|  |   } | ||||||
|  |   let hotkeys = Hotkeys._docs | ||||||
|  |   let userdetails = state.osmConnection.userDetails | ||||||
|  | 
 | ||||||
|  |   let usersettingslayer = new LayerConfig(<LayerConfigJson>usersettings, "usersettings", true) | ||||||
|  | 
 | ||||||
|  |   let theme = state.theme | ||||||
|  |   let featureSwitches = state.featureSwitches | ||||||
|  |   let showHome = featureSwitches?.featureSwitchBackToThemeOverview | ||||||
|  |   let pg = state.guistate.pageStates | ||||||
|  |   let location = state.mapProperties?.location | ||||||
|  |   export let onlyLink: boolean | ||||||
|  |   const t = Translations.t.general.menu | ||||||
|  |   let shown = new UIEventSource(state.guistate.pageStates.menu.data || !onlyLink) | ||||||
|  |   state.guistate.pageStates.menu.addCallback((isShown) => { | ||||||
|  |     if (!onlyLink) { | ||||||
|  |       return true | ||||||
|  |     } | ||||||
|  |     if (isShown) { | ||||||
|  |       shown.setData(true) | ||||||
|  |     } else { | ||||||
|  |       Utils.waitFor(250).then(() => { | ||||||
|  |         shown.setData(state.guistate.pageStates.menu.data) | ||||||
|  |       }) | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  |   let isAndroid = AndroidPolyfill.inAndroid | ||||||
|  |   let nrOfFailedImages = ImageUploadQueue.singleton.imagesInQueue | ||||||
|  |   let failedImagesOpen = pg.failedImages | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <div | ||||||
|  |   class="low-interaction flex h-screen flex-col gap-y-2 overflow-y-auto p-2 sm:gap-y-3 sm:p-3" | ||||||
|  |   class:hidden={!$shown} | ||||||
|  | > | ||||||
|  |   <div class="flex justify-between"> | ||||||
|  |     <h2> | ||||||
|  |       <Tr t={t.title} /> | ||||||
|  |     </h2> | ||||||
|  |     <CloseButton | ||||||
|  |       on:click={() => { | ||||||
|  |         pg.menu.set(false) | ||||||
|  |       }} | ||||||
|  |     /> | ||||||
|  |   </div> | ||||||
|  |   {#if $showHome} | ||||||
|  |     <a class="button primary flex" href={Utils.HomepageLink()}> | ||||||
|  |       <Squares2x2 class="h-10 w-10" /> | ||||||
|  |       {#if Utils.isIframe} | ||||||
|  |         <Tr t={Translations.t.general.seeIndex} /> | ||||||
|  |       {:else} | ||||||
|  |         <Tr t={Translations.t.general.backToIndex} /> | ||||||
|  |       {/if} | ||||||
|  |     </a> | ||||||
|  |   {/if} | ||||||
|  | 
 | ||||||
|  |   <!-- User related: avatar, settings, favourits, logout --> | ||||||
|  |   <SidebarUnit> | ||||||
|  |     <LoginToggle {state}> | ||||||
|  |       <LoginButton osmConnection={state.osmConnection} slot="not-logged-in" /> | ||||||
|  |       <div class="flex items-center gap-x-4"> | ||||||
|  |         {#if $userdetails.img} | ||||||
|  |           <img alt="avatar" src={$userdetails.img} class="h-14 w-14 rounded-full" /> | ||||||
|  |         {/if} | ||||||
|  |         <b>{$userdetails.name}</b> | ||||||
|  |       </div> | ||||||
|  |     </LoginToggle> | ||||||
|  | 
 | ||||||
|  |     <Page {onlyLink} shown={pg.usersettings} bodyPadding="p-0 pb-4"> | ||||||
|  |       <svelte:fragment slot="header"> | ||||||
|  |         <CogIcon /> | ||||||
|  |         <Tr t={UserRelatedState.usersettingsConfig.title.GetRenderValue({})} /> | ||||||
|  |       </svelte:fragment> | ||||||
|  | 
 | ||||||
|  |       <!-- All shown components are set by 'usersettings.json', which happily uses some special visualisations created specifically for it --> | ||||||
|  |       <SelectedElementView | ||||||
|  |         highlightedRendering={state.guistate.highlightedUserSetting} | ||||||
|  |         layer={usersettingslayer} | ||||||
|  |         selectedElement={{ | ||||||
|  |           type: "Feature", | ||||||
|  |           properties: { id: "settings" }, | ||||||
|  |           geometry: { type: "Point", coordinates: [0, 0] }, | ||||||
|  |         }} | ||||||
|  |         {state} | ||||||
|  |         tags={state.userRelatedState.preferencesAsTags} | ||||||
|  |       /> | ||||||
|  |     </Page> | ||||||
|  | 
 | ||||||
|  |     {#if $nrOfFailedImages.length > 0 || $failedImagesOpen} | ||||||
|  |       <Page {onlyLink} shown={pg.failedImages} bodyPadding="p-0 pb-4"> | ||||||
|  |         <svelte:fragment slot="header"> | ||||||
|  |           <PhotoIcon /> | ||||||
|  |           <Tr t={Translations.t.imageQueue.menu.Subs({ count: $nrOfFailedImages.length })} /> | ||||||
|  |         </svelte:fragment> | ||||||
|  |         <QueuedImagesView {state} /> | ||||||
|  |       </Page> | ||||||
|  |     {/if} | ||||||
|  | 
 | ||||||
|  |     <LoginToggle {state} silentFail> | ||||||
|  |       {#if state.favourites} | ||||||
|  |         <Page {onlyLink} shown={pg.favourites}> | ||||||
|  |           <svelte:fragment slot="header"> | ||||||
|  |             <HeartIcon /> | ||||||
|  |             <Tr t={Translations.t.favouritePoi.tab} /> | ||||||
|  |           </svelte:fragment> | ||||||
|  | 
 | ||||||
|  |           <h3> | ||||||
|  |             <Tr t={Translations.t.favouritePoi.title} /> | ||||||
|  |           </h3> | ||||||
|  |           <div> | ||||||
|  |             <Favourites {state} /> | ||||||
|  |             <h3> | ||||||
|  |               <Tr t={Translations.t.reviews.your_reviews} /> | ||||||
|  |             </h3> | ||||||
|  |             <ReviewsOverview {state} /> | ||||||
|  |           </div> | ||||||
|  |         </Page> | ||||||
|  |       {/if} | ||||||
|  |       {#if $hotkeys.length > 0} | ||||||
|  |         <div class="hidden-on-mobile w-full"> | ||||||
|  |           <Page {onlyLink} shown={pg.hotkeys}> | ||||||
|  |             <svelte:fragment slot="header"> | ||||||
|  |               <BoltIcon /> | ||||||
|  |               <Tr t={Translations.t.hotkeyDocumentation.title} /> | ||||||
|  |             </svelte:fragment> | ||||||
|  |             <HotkeyTable /> | ||||||
|  |           </Page> | ||||||
|  |         </div> | ||||||
|  |       {/if} | ||||||
|  | 
 | ||||||
|  |       <div class="self-end"> | ||||||
|  |         <LogoutButton osmConnection={state.osmConnection} /> | ||||||
|  |       </div> | ||||||
|  |     </LoginToggle> | ||||||
|  |     <LanguagePicker /> | ||||||
|  |   </SidebarUnit> | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |   <slot name="theme-tools"/> | ||||||
|  |   <slot name="location-tools"/> | ||||||
|  | 
 | ||||||
|  |   <!-- About MC: various outward links, legal info, ... --> | ||||||
|  |   <SidebarUnit> | ||||||
|  |     <h3> | ||||||
|  |       <Tr t={Translations.t.general.menu.aboutMapComplete} /> | ||||||
|  |     </h3> | ||||||
|  | 
 | ||||||
|  |     <a class="flex" href={window.location.protocol + "//" + window.location.host + "/studio.html"}> | ||||||
|  |       <Pencil class="mr-2 h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.general.morescreen.createYourOwnTheme} /> | ||||||
|  |     </a> | ||||||
|  | 
 | ||||||
|  |     <a class="flex" href="mailto:info@mapcomplete.org"> | ||||||
|  |       <EnvelopeOpen class="h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.general.attribution.emailCreators} /> | ||||||
|  |     </a> | ||||||
|  | 
 | ||||||
|  |     <a class="flex" href="https://en.osm.town/@MapComplete" target="_blank"> | ||||||
|  |       <Mastodon class="h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.general.attribution.followOnMastodon} /> | ||||||
|  |     </a> | ||||||
|  | 
 | ||||||
|  |     <a class="flex" href="https://liberapay.com/pietervdvn/" target="_blank"> | ||||||
|  |       <Liberapay class="h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.general.attribution.donate} /> | ||||||
|  |     </a> | ||||||
|  | 
 | ||||||
|  |     <a | ||||||
|  |       class="flex" | ||||||
|  |       href="https://source.mapcomplete.org/MapComplete/MapComplete/issues" | ||||||
|  |       target="_blank" | ||||||
|  |     > | ||||||
|  |       <Bug class="h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.general.attribution.openIssueTracker} /> | ||||||
|  |     </a> | ||||||
|  | 
 | ||||||
|  |     <a class="flex" href="https://source.mapcomplete.org/MapComplete/MapComplete/" target="_blank"> | ||||||
|  |       <Forgejo class="h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.general.attribution.gotoSourceCode} /> | ||||||
|  |     </a> | ||||||
|  |     <a class="flex" href={`${Constants.weblate}projects/mapcomplete/`} target="_blank"> | ||||||
|  |       <TranslateIcon class="h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.translations.activateButton} /> | ||||||
|  |     </a> | ||||||
|  | 
 | ||||||
|  |     <a | ||||||
|  |       class="flex" | ||||||
|  |       href={window.location.protocol + "//" + window.location.host + "/inspector.html"} | ||||||
|  |     > | ||||||
|  |       <MagnifyingGlassCircle class="mr-2 h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.inspector.menu} /> | ||||||
|  |     </a> | ||||||
|  | 
 | ||||||
|  |     {#if !state.theme} | ||||||
|  |       <a class="flex" href={`./statistics.html`} target="_blank"> | ||||||
|  |         <ChartBar class="h-6 w-6" /> | ||||||
|  |         <Tr t={Translations.t.general.attribution.openStatistics.Subs({ theme: "MapComplete" })} /> | ||||||
|  |       </a> | ||||||
|  |     {/if} | ||||||
|  | 
 | ||||||
|  |     <a class="flex" href={Utils.OsmChaLinkFor(7)} target="_blank"> | ||||||
|  |       <ArrowTrendingUp class="mr-2 h-6 w-6" /> | ||||||
|  |       <Tr t={Translations.t.general.attribution.openOsmchaLastWeek} /> | ||||||
|  |     </a> | ||||||
|  |   </SidebarUnit> | ||||||
|  | 
 | ||||||
|  |   <SidebarUnit> | ||||||
|  |     <h3> | ||||||
|  |       <Tr t={Translations.t.general.menu.legal} /> | ||||||
|  |     </h3> | ||||||
|  |     <Page {onlyLink} shown={pg.copyright}> | ||||||
|  |       <svelte:fragment slot="header"> | ||||||
|  |         <Copyright /> | ||||||
|  |         <Tr t={Translations.t.general.attribution.attributionTitle} /> | ||||||
|  |       </svelte:fragment> | ||||||
|  |       <CopyrightPanel {state} /> | ||||||
|  |     </Page> | ||||||
|  |     {#if state?.theme} | ||||||
|  |       <Page {onlyLink} shown={pg.copyright_icons}> | ||||||
|  |         <svelte:fragment slot="header"> | ||||||
|  |           <Copyright /> | ||||||
|  |           <Tr t={Translations.t.general.attribution.iconAttribution.title} /> | ||||||
|  |         </svelte:fragment> | ||||||
|  |         <CopyrightAllIcons {state} /> | ||||||
|  |       </Page> | ||||||
|  |     {/if} | ||||||
|  | 
 | ||||||
|  |     <Page {onlyLink} shown={pg.privacy}> | ||||||
|  |       <svelte:fragment slot="header"> | ||||||
|  |         <EyeIcon /> | ||||||
|  |         <Tr t={Translations.t.privacy.title} /> | ||||||
|  |       </svelte:fragment> | ||||||
|  |       <PrivacyPolicy {state} /> | ||||||
|  |       <a href="./privacy.html" class="button float-right w-fit" target="_blank"> | ||||||
|  |         <ArrowTopRightOnSquare class="h-8 w-8" /> | ||||||
|  |       </a> | ||||||
|  |     </Page> | ||||||
|  |   </SidebarUnit> | ||||||
|  | 
 | ||||||
|  |   <div class="subtle self-end"> | ||||||
|  |     {Constants.vNumber} | ||||||
|  |     {#if $isAndroid} | ||||||
|  |       Android | ||||||
|  |     {/if} | ||||||
|  |   </div> | ||||||
|  | </div> | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue