148 lines
6 KiB
TypeScript
148 lines
6 KiB
TypeScript
|
import FeaturePipelineState from "../../Logic/State/FeaturePipelineState";
|
||
|
import {UIEventSource} from "../../Logic/UIEventSource";
|
||
|
import {DefaultGuiState} from "../DefaultGuiState";
|
||
|
import BaseUIElement from "../BaseUIElement";
|
||
|
import Translations from "../i18n/Translations";
|
||
|
import {GeoOperations} from "../../Logic/GeoOperations";
|
||
|
import NearbyImages, {NearbyImageOptions, P4CPicture, SelectOneNearbyImage} from "./NearbyImages";
|
||
|
import {SubstitutedTranslation} from "../SubstitutedTranslation";
|
||
|
import {Tag} from "../../Logic/Tags/Tag";
|
||
|
import ChangeTagAction from "../../Logic/Osm/Actions/ChangeTagAction";
|
||
|
import {And} from "../../Logic/Tags/And";
|
||
|
import {SaveButton} from "./SaveButton";
|
||
|
import Lazy from "../Base/Lazy";
|
||
|
import {CheckBox} from "../Input/Checkboxes";
|
||
|
import Slider from "../Input/Slider";
|
||
|
import AllImageProviders from "../../Logic/ImageProviders/AllImageProviders";
|
||
|
import Combine from "../Base/Combine";
|
||
|
import {VariableUiElement} from "../Base/VariableUIElement";
|
||
|
import Toggle from "../Input/Toggle";
|
||
|
import Title from "../Base/Title";
|
||
|
import {MapillaryLinkVis} from "./MapillaryLinkVis";
|
||
|
import {SpecialVisualization} from "../SpecialVisualization";
|
||
|
|
||
|
export class NearbyImageVis implements SpecialVisualization {
|
||
|
args: { name: string; defaultValue?: string; doc: string; required?: boolean }[] = [
|
||
|
{
|
||
|
name: "mode",
|
||
|
defaultValue: "expandable",
|
||
|
doc: "Indicates how this component is initialized. Options are: \n\n- `open`: always show and load the pictures\n- `collapsable`: show the pictures, but a user can collapse them\n- `expandable`: shown by default; but a user can collapse them.",
|
||
|
},
|
||
|
{
|
||
|
name: "mapillary",
|
||
|
defaultValue: "true",
|
||
|
doc: "If 'true', includes a link to mapillary on this location.",
|
||
|
},
|
||
|
]
|
||
|
docs =
|
||
|
"A component showing nearby images loaded from various online services such as Mapillary. In edit mode and when used on a feature, the user can select an image to add to the feature"
|
||
|
funcName = "nearby_images"
|
||
|
|
||
|
constr(
|
||
|
state: FeaturePipelineState,
|
||
|
tagSource: UIEventSource<any>,
|
||
|
args: string[],
|
||
|
guistate: DefaultGuiState
|
||
|
): BaseUIElement {
|
||
|
const t = Translations.t.image.nearbyPictures
|
||
|
const mode: "open" | "expandable" | "collapsable" = <any>args[0]
|
||
|
const feature = state.allElements.ContainingFeatures.get(tagSource.data.id)
|
||
|
const [lon, lat] = GeoOperations.centerpointCoordinates(feature)
|
||
|
const id: string = tagSource.data["id"]
|
||
|
const canBeEdited: boolean = !!id?.match("(node|way|relation)/-?[0-9]+")
|
||
|
const selectedImage = new UIEventSource<P4CPicture>(undefined)
|
||
|
|
||
|
let saveButton: BaseUIElement = undefined
|
||
|
if (canBeEdited) {
|
||
|
const confirmText: BaseUIElement = new SubstitutedTranslation(
|
||
|
t.confirm,
|
||
|
tagSource,
|
||
|
state
|
||
|
)
|
||
|
|
||
|
const onSave = async () => {
|
||
|
console.log("Selected a picture...", selectedImage.data)
|
||
|
const osmTags = selectedImage.data.osmTags
|
||
|
const tags: Tag[] = []
|
||
|
for (const key in osmTags) {
|
||
|
tags.push(new Tag(key, osmTags[key]))
|
||
|
}
|
||
|
await state?.changes?.applyAction(
|
||
|
new ChangeTagAction(id, new And(tags), tagSource.data, {
|
||
|
theme: state?.layoutToUse.id,
|
||
|
changeType: "link-image",
|
||
|
})
|
||
|
)
|
||
|
}
|
||
|
saveButton = new SaveButton(
|
||
|
selectedImage,
|
||
|
state.osmConnection,
|
||
|
confirmText,
|
||
|
t.noImageSelected
|
||
|
)
|
||
|
.onClick(onSave)
|
||
|
.SetClass("flex justify-end")
|
||
|
}
|
||
|
|
||
|
const nearby = new Lazy(() => {
|
||
|
const towardsCenter = new CheckBox(t.onlyTowards, false)
|
||
|
|
||
|
const radiusValue =
|
||
|
state?.osmConnection?.GetPreference("nearby-images-radius", "300").sync(
|
||
|
(s) => Number(s),
|
||
|
[],
|
||
|
(i) => "" + i
|
||
|
) ?? new UIEventSource(300)
|
||
|
|
||
|
const radius = new Slider(25, 500, {
|
||
|
value: radiusValue,
|
||
|
step: 25,
|
||
|
})
|
||
|
const alreadyInTheImage = AllImageProviders.LoadImagesFor(tagSource)
|
||
|
const options: NearbyImageOptions & { value } = {
|
||
|
lon,
|
||
|
lat,
|
||
|
searchRadius: 500,
|
||
|
shownRadius: radius.GetValue(),
|
||
|
value: selectedImage,
|
||
|
blacklist: alreadyInTheImage,
|
||
|
towardscenter: towardsCenter.GetValue(),
|
||
|
maxDaysOld: 365 * 5,
|
||
|
}
|
||
|
const slideshow = canBeEdited
|
||
|
? new SelectOneNearbyImage(options, state)
|
||
|
: new NearbyImages(options, state)
|
||
|
const controls = new Combine([
|
||
|
towardsCenter,
|
||
|
new Combine([
|
||
|
new VariableUiElement(
|
||
|
radius.GetValue().map((radius) => t.withinRadius.Subs({radius}))
|
||
|
),
|
||
|
radius,
|
||
|
]).SetClass("flex justify-between"),
|
||
|
]).SetClass("flex flex-col")
|
||
|
return new Combine([
|
||
|
slideshow,
|
||
|
controls,
|
||
|
saveButton,
|
||
|
new MapillaryLinkVis().constr(state, tagSource, []).SetClass("mt-6"),
|
||
|
])
|
||
|
})
|
||
|
|
||
|
let withEdit: BaseUIElement = nearby
|
||
|
if (canBeEdited) {
|
||
|
withEdit = new Combine([t.hasMatchingPicture, nearby]).SetClass("flex flex-col")
|
||
|
}
|
||
|
|
||
|
if (mode === "open") {
|
||
|
return withEdit
|
||
|
}
|
||
|
const toggleState = new UIEventSource<boolean>(mode === "collapsable")
|
||
|
return new Toggle(
|
||
|
new Combine([new Title(t.title), withEdit]),
|
||
|
new Title(t.browseNearby).onClick(() => toggleState.setData(true)),
|
||
|
toggleState
|
||
|
)
|
||
|
}
|
||
|
}
|