2022-11-02 14:44:06 +01:00
import { UIEventSource } from "../../Logic/UIEventSource"
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"
2023-03-28 05:13:48 +02:00
import { SpecialVisualization , SpecialVisualizationState } from "../SpecialVisualization"
2022-10-28 04:33:05 +02:00
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 (
2023-03-28 05:13:48 +02:00
state : SpecialVisualizationState ,
tagSource : UIEventSource < Record < string , string > > ,
args : string [ ]
2022-10-28 04:33:05 +02:00
) : BaseUIElement {
const t = Translations . t . image . nearbyPictures
const mode : "open" | "expandable" | "collapsable" = < any > args [ 0 ]
2023-03-28 05:13:48 +02:00
const feature = state . indexedFeatures . featuresById . data . get ( tagSource . data . id )
2022-10-28 04:33:05 +02:00
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 , {
2023-03-28 05:13:48 +02:00
theme : state?.layout.id ,
2022-10-28 04:33:05 +02:00
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 )
2023-01-30 01:19:37 +01:00
const maxSearchRadius = 100
const stepSize = 10
const defaultValue = Math . floor ( maxSearchRadius / ( 2 * stepSize ) ) * stepSize
const fromOsmPreferences = state ? . osmConnection
? . GetPreference ( "nearby-images-radius" , "" + defaultValue )
. sync (
2022-10-28 04:33:05 +02:00
( s ) = > Number ( s ) ,
[ ] ,
( i ) = > "" + i
2023-01-30 01:19:37 +01:00
)
const radiusValue = new UIEventSource ( fromOsmPreferences . data )
radiusValue . addCallbackAndRunD ( ( v ) = > fromOsmPreferences . setData ( v ) )
2022-10-28 04:33:05 +02:00
2023-01-30 01:19:37 +01:00
const radius = new Slider ( stepSize , maxSearchRadius , {
2022-10-28 04:33:05 +02:00
value : radiusValue ,
2023-01-30 01:19:37 +01:00
step : 10 ,
2022-10-28 04:33:05 +02:00
} )
const alreadyInTheImage = AllImageProviders . LoadImagesFor ( tagSource )
const options : NearbyImageOptions & { value } = {
lon ,
lat ,
2023-01-30 01:19:37 +01:00
searchRadius : maxSearchRadius ,
2022-10-28 04:33:05 +02:00
shownRadius : radius.GetValue ( ) ,
value : selectedImage ,
blacklist : alreadyInTheImage ,
towardscenter : towardsCenter.GetValue ( ) ,
2023-01-30 01:19:37 +01:00
maxDaysOld : 365 * 3 ,
2022-10-28 04:33:05 +02:00
}
const slideshow = canBeEdited
2023-03-28 05:13:48 +02:00
? new SelectOneNearbyImage ( options , state . indexedFeatures )
: new NearbyImages ( options , state . indexedFeatures )
2022-10-28 04:33:05 +02:00
const controls = new Combine ( [
towardsCenter ,
new Combine ( [
new VariableUiElement (
2022-11-02 14:44:06 +01:00
radius . GetValue ( ) . map ( ( radius ) = > t . withinRadius . Subs ( { radius } ) )
2022-10-28 04:33:05 +02:00
) ,
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
)
}
}