forked from MapComplete/MapComplete
UX: make margins in note comment element smaller, add image preview option
This commit is contained in:
parent
15856d7047
commit
be1e1673d9
3 changed files with 56 additions and 41 deletions
|
@ -2,7 +2,7 @@ import { Mapillary } from "./Mapillary"
|
||||||
import { WikimediaImageProvider } from "./WikimediaImageProvider"
|
import { WikimediaImageProvider } from "./WikimediaImageProvider"
|
||||||
import { Imgur } from "./Imgur"
|
import { Imgur } from "./Imgur"
|
||||||
import GenericImageProvider from "./GenericImageProvider"
|
import GenericImageProvider from "./GenericImageProvider"
|
||||||
import { Store, UIEventSource } from "../UIEventSource"
|
import { ImmutableStore, Store, UIEventSource } from "../UIEventSource"
|
||||||
import ImageProvider, { ProvidedImage } from "./ImageProvider"
|
import ImageProvider, { ProvidedImage } from "./ImageProvider"
|
||||||
import { WikidataImageProvider } from "./WikidataImageProvider"
|
import { WikidataImageProvider } from "./WikidataImageProvider"
|
||||||
import Panoramax from "./Panoramax"
|
import Panoramax from "./Panoramax"
|
||||||
|
@ -34,17 +34,17 @@ export default class AllImageProviders {
|
||||||
AllImageProviders.genericImageProvider,
|
AllImageProviders.genericImageProvider,
|
||||||
]
|
]
|
||||||
public static apiUrls: string[] = [].concat(
|
public static apiUrls: string[] = [].concat(
|
||||||
...AllImageProviders.ImageAttributionSource.map((src) => src.apiUrls())
|
...AllImageProviders.ImageAttributionSource.map((src) => src.apiUrls()),
|
||||||
)
|
)
|
||||||
public static defaultKeys = [].concat(
|
public static defaultKeys = [].concat(
|
||||||
AllImageProviders.ImageAttributionSource.map((provider) => provider.defaultKeyPrefixes)
|
AllImageProviders.ImageAttributionSource.map((provider) => provider.defaultKeyPrefixes),
|
||||||
)
|
)
|
||||||
private static providersByName = {
|
private static providersByName = {
|
||||||
imgur: Imgur.singleton,
|
imgur: Imgur.singleton,
|
||||||
mapillary: Mapillary.singleton,
|
mapillary: Mapillary.singleton,
|
||||||
wikidata: WikidataImageProvider.singleton,
|
wikidata: WikidataImageProvider.singleton,
|
||||||
wikimedia: WikimediaImageProvider.singleton,
|
wikimedia: WikimediaImageProvider.singleton,
|
||||||
panoramax: Panoramax.singleton
|
panoramax: Panoramax.singleton,
|
||||||
}
|
}
|
||||||
|
|
||||||
public static byName(name: string) {
|
public static byName(name: string) {
|
||||||
|
@ -71,7 +71,7 @@ export default class AllImageProviders {
|
||||||
*/
|
*/
|
||||||
public static LoadImagesFor(
|
public static LoadImagesFor(
|
||||||
tags: Store<Record<string, string>>,
|
tags: Store<Record<string, string>>,
|
||||||
tagKey?: string[]
|
tagKey?: string[],
|
||||||
): Store<ProvidedImage[]> {
|
): Store<ProvidedImage[]> {
|
||||||
if (tags?.data?.id === undefined) {
|
if (tags?.data?.id === undefined) {
|
||||||
return undefined
|
return undefined
|
||||||
|
@ -96,4 +96,19 @@ export default class AllImageProviders {
|
||||||
}
|
}
|
||||||
return source
|
return source
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a list of URLs, tries to detect the images. Used in e.g. the comments
|
||||||
|
* @param url
|
||||||
|
*/
|
||||||
|
public static loadImagesFrom(urls: string[]): Store<ProvidedImage[]> {
|
||||||
|
const tags = {
|
||||||
|
id:"na"
|
||||||
|
}
|
||||||
|
for (let i = 0; i < urls.length; i++) {
|
||||||
|
const url = urls[i]
|
||||||
|
tags["image:" + i] = url
|
||||||
|
}
|
||||||
|
return this.LoadImagesFor(new ImmutableStore(tags))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -99,7 +99,6 @@ export default class LayerState {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
const filter = fl.layerDef.filters.find(f => f.id === filtername)
|
const filter = fl.layerDef.filters.find(f => f.id === filtername)
|
||||||
console.log("Updating active filters for flayer", fl.layerDef.id,"with filterconfig",filter)
|
|
||||||
if(typeof appliedFilter.data === "number"){
|
if(typeof appliedFilter.data === "number"){
|
||||||
if(filter.options[appliedFilter.data].osmTags === undefined){
|
if(filter.options[appliedFilter.data].osmTags === undefined){
|
||||||
// This is probably the first, generic option which doesn't _actually_ filter
|
// This is probably the first, generic option which doesn't _actually_ filter
|
||||||
|
|
|
@ -4,12 +4,11 @@
|
||||||
import Note from "../../../assets/svg/Note.svelte"
|
import Note from "../../../assets/svg/Note.svelte"
|
||||||
import Resolved from "../../../assets/svg/Resolved.svelte"
|
import Resolved from "../../../assets/svg/Resolved.svelte"
|
||||||
import Speech_bubble from "../../../assets/svg/Speech_bubble.svelte"
|
import Speech_bubble from "../../../assets/svg/Speech_bubble.svelte"
|
||||||
import { ImmutableStore, Stores } from "../../../Logic/UIEventSource"
|
import { Stores } from "../../../Logic/UIEventSource"
|
||||||
import { Utils } from "../../../Utils"
|
import { Utils } from "../../../Utils"
|
||||||
import Img from "../../Base/Img"
|
|
||||||
import { SlideShow } from "../../Image/SlideShow"
|
|
||||||
import ToSvelte from "../../Base/ToSvelte.svelte"
|
|
||||||
import Tr from "../../Base/Tr.svelte"
|
import Tr from "../../Base/Tr.svelte"
|
||||||
|
import AllImageProviders from "../../../Logic/ImageProviders/AllImageProviders"
|
||||||
|
import AttributedImage from "../../Image/AttributedImage.svelte"
|
||||||
|
|
||||||
export let comment: {
|
export let comment: {
|
||||||
date: string
|
date: string
|
||||||
|
@ -46,58 +45,60 @@
|
||||||
.filter((link) => !link.startsWith("https://wiki.openstreetmap.org/wiki/File:"))
|
.filter((link) => !link.startsWith("https://wiki.openstreetmap.org/wiki/File:"))
|
||||||
|
|
||||||
|
|
||||||
let imgStore = new ImmutableStore(
|
const attributedImages = AllImageProviders.loadImagesFrom(images)
|
||||||
images.map((i) =>
|
/**
|
||||||
new Img(i).SetClass("w-full block cursor-pointer")
|
* Class of the little icons indicating 'opened', 'comment' and 'resolved'
|
||||||
.onClick(() =>
|
*/
|
||||||
state?.previewedImage?.setData(
|
export let iconClass = "shrink-0 w-6 mr-3 my-2 "
|
||||||
<any>{
|
|
||||||
url_hd: i,
|
|
||||||
url: i,
|
|
||||||
}),
|
|
||||||
)))
|
|
||||||
|
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="flex flex-col py-2 my-2 border-gray-500 border-b" class:border-interactive={comment.highlighted}>
|
<div class="flex flex-col my-2 border-gray-500 border-b" class:border-interactive={comment.highlighted}>
|
||||||
|
|
||||||
<div class="flex">
|
<div class="flex items-center">
|
||||||
|
|
||||||
<!-- Action icon, e.g. 'created', 'commented', 'closed' -->
|
<!-- Action icon, e.g. 'created', 'commented', 'closed' -->
|
||||||
{#if comment.action === "opened" || comment.action === "reopened"}
|
|
||||||
<Note class="shrink-0 mr-4 w-6" />
|
{#if $userinfo?.user?.img?.href}
|
||||||
|
<img alt="avatar" aria-hidden="true" src={$userinfo?.user?.img?.href} class="rounded-full w-10 h-10 mr-3" />
|
||||||
|
{:else if comment.action === "opened" || comment.action === "reopened"}
|
||||||
|
<Note class={iconClass} />
|
||||||
{:else if comment.action === "closed"}
|
{:else if comment.action === "closed"}
|
||||||
<Resolved class="shrink-0 mr-4 w-6" />
|
<Resolved class={iconClass} />
|
||||||
{:else}
|
{:else}
|
||||||
<Speech_bubble class="shrink-0 mr-4 w-6" />
|
<Speech_bubble class={iconClass} />
|
||||||
{/if}
|
{/if}
|
||||||
<div class="flex flex-col gap-y-2">
|
<div class="flex flex-col gap-y-2">
|
||||||
{@html comment.html}
|
{@html comment.html}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#if images.length > 0}
|
{#if $attributedImages?.length > 0}
|
||||||
<ToSvelte
|
<div class="flex justify-center w-full space-x-4 overflow-x-auto" style="scroll-snap-type: x proximity">
|
||||||
construct={() => new SlideShow(imgStore) .SetClass("mb-1").SetStyle("min-width: 50px; background: grey;")} />
|
{#each $attributedImages as image (image.id)}
|
||||||
|
<AttributedImage
|
||||||
|
{state}
|
||||||
|
{image}
|
||||||
|
imgClass="max-h-64 w-auto sm:h-32 md:h-64"
|
||||||
|
previewedImage={state.previewedImage}
|
||||||
|
attributionFormat="minimal"
|
||||||
|
>
|
||||||
|
</AttributedImage>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
|
|
||||||
<div class="flex justify-end items-center subtle pt-4 pb-2">
|
<div class="flex justify-end items-center subtle py-2">
|
||||||
<!-- commenter info -->
|
<!-- commenter info -->
|
||||||
|
|
||||||
{#if $userinfo?.user?.img?.href}
|
|
||||||
<img alt="avatar" aria-hidden="true" src={$userinfo?.user?.img?.href} class="rounded-full w-8 h-8 mr-4" />
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<span class="mr-2">
|
<span class="mr-2">
|
||||||
{#if comment.user === undefined}
|
{#if comment.user === undefined}
|
||||||
<Tr t={t.anonymous} />
|
<Tr t={t.anonymous} />
|
||||||
{:else}
|
{:else}
|
||||||
<a href={comment.user_url} target="_blank">{comment.user}</a>
|
<a href={comment.user_url} target="_blank">{comment.user}</a>
|
||||||
{/if}
|
{/if}
|
||||||
{comment.date}
|
{comment.date}
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Reference in a new issue