MapComplete/UI/Image/ImageUploadFlow.ts

203 lines
7.6 KiB
TypeScript
Raw Normal View History

2023-05-11 17:29:25 +02:00
import {Store, UIEventSource} from "../../Logic/UIEventSource"
2022-09-08 21:40:48 +02:00
import Combine from "../Base/Combine"
import Translations from "../i18n/Translations"
import Svg from "../../Svg"
2023-05-11 17:29:25 +02:00
import {Tag} from "../../Logic/Tags/Tag"
2022-09-08 21:40:48 +02:00
import BaseUIElement from "../BaseUIElement"
import Toggle from "../Input/Toggle"
import FileSelectorButton from "../Input/FileSelectorButton"
import ImgurUploader from "../../Logic/ImageProviders/ImgurUploader"
import ChangeTagAction from "../../Logic/Osm/Actions/ChangeTagAction"
import LayerConfig from "../../Models/ThemeConfig/LayerConfig"
2023-05-11 17:29:25 +02:00
import {FixedUiElement} from "../Base/FixedUiElement"
import {VariableUiElement} from "../Base/VariableUIElement"
2022-09-08 21:40:48 +02:00
import Loading from "../Base/Loading"
2023-05-11 17:29:25 +02:00
import {LoginToggle} from "../Popup/LoginButton"
import Constants from "../../Models/Constants"
2023-05-11 17:29:25 +02:00
import {SpecialVisualizationState} from "../SpecialVisualization"
2021-06-14 19:21:33 +02:00
export class ImageUploadFlow extends Toggle {
2021-10-20 00:34:26 +02:00
private static readonly uploadCountsPerId = new Map<string, UIEventSource<number>>()
2021-11-07 16:34:51 +01:00
2022-09-08 21:40:48 +02:00
constructor(
tagsSource: Store<any>,
2023-03-28 05:13:48 +02:00
state: SpecialVisualizationState,
2022-09-08 21:40:48 +02:00
imagePrefix: string = "image",
text: string = undefined
) {
2021-10-20 00:34:26 +02:00
const perId = ImageUploadFlow.uploadCountsPerId
const id = tagsSource.data.id
2021-11-07 16:34:51 +01:00
if (!perId.has(id)) {
2021-10-20 00:34:26 +02:00
perId.set(id, new UIEventSource<number>(0))
}
const uploadedCount = perId.get(id)
2022-09-08 21:40:48 +02:00
const uploader = new ImgurUploader(async (url) => {
2021-06-11 22:51:45 +02:00
// A file was uploaded - we add it to the tags of the object
2021-06-11 22:51:45 +02:00
const tags = tagsSource.data
let key = imagePrefix
if (tags[imagePrefix] !== undefined) {
2022-09-08 21:40:48 +02:00
let freeIndex = 0
2021-06-11 22:51:45 +02:00
while (tags[imagePrefix + ":" + freeIndex] !== undefined) {
2022-09-08 21:40:48 +02:00
freeIndex++
2021-06-11 22:51:45 +02:00
}
2022-09-08 21:40:48 +02:00
key = imagePrefix + ":" + freeIndex
2021-06-11 22:51:45 +02:00
}
2022-09-08 21:40:48 +02:00
await state.changes.applyAction(
new ChangeTagAction(tags.id, new Tag(key, url), tagsSource.data, {
changeType: "add-image",
2023-03-28 05:13:48 +02:00
theme: state.layout.id,
2022-09-08 21:40:48 +02:00
})
)
console.log("Adding image:" + key, url)
uploadedCount.data++
uploadedCount.ping()
2021-06-11 22:51:45 +02:00
})
2021-11-07 16:34:51 +01:00
2022-09-08 21:40:48 +02:00
const t = Translations.t.image
2021-11-07 16:34:51 +01:00
let labelContent: BaseUIElement
if (text === undefined) {
2022-09-08 21:40:48 +02:00
labelContent = Translations.t.image.addPicture
.Clone()
.SetClass("block align-middle mt-1 ml-3 text-4xl ")
2021-11-07 16:34:51 +01:00
} else {
2022-09-08 21:40:48 +02:00
labelContent = new FixedUiElement(text).SetClass(
"block align-middle mt-1 ml-3 text-2xl "
)
2021-11-07 16:34:51 +01:00
}
const label = new Combine([
Svg.camera_plus_svg().SetClass("block w-12 h-12 p-1 text-4xl "),
2022-09-08 21:40:48 +02:00
labelContent,
2023-05-11 17:29:25 +02:00
]).SetClass("w-full flex justify-center items-center")
const licenseStore = state?.osmConnection?.GetPreference(
Constants.OsmPreferenceKeyPicturesLicense,
"CC0"
)
2023-05-11 17:29:25 +02:00
const fileSelector = new FileSelectorButton(label, {
acceptType: "image/*",
allowMultiple: true,
labelClasses: "rounded-full border-2 border-black font-bold"
})
/* fileSelector.SetClass(
"p-2 border-4 border-detail rounded-full font-bold h-full align-middle w-full flex justify-center"
)
.SetStyle(" border-color: var(--foreground-color);")*/
2022-09-08 21:40:48 +02:00
fileSelector.GetValue().addCallback((filelist) => {
if (filelist === undefined || filelist.length === 0) {
2022-09-08 21:40:48 +02:00
return
2021-06-11 22:51:45 +02:00
}
for (var i = 0; i < filelist.length; i++) {
2021-11-07 16:34:51 +01:00
const sizeInBytes = filelist[i].size
2022-09-08 21:40:48 +02:00
console.log(filelist[i].name + " has a size of " + sizeInBytes + " Bytes")
2021-11-07 16:34:51 +01:00
if (sizeInBytes > uploader.maxFileSizeInMegabytes * 1000000) {
2022-09-08 21:40:48 +02:00
alert(
Translations.t.image.toBig.Subs({
actual_size: Math.floor(sizeInBytes / 1000000) + "MB",
max_size: uploader.maxFileSizeInMegabytes + "MB",
}).txt
)
return
}
}
2021-11-07 16:34:51 +01:00
const license = licenseStore?.data ?? "CC0"
2020-10-14 12:15:09 +02:00
2022-09-08 21:40:48 +02:00
const tags = tagsSource.data
2020-10-14 12:15:09 +02:00
2023-03-28 05:13:48 +02:00
const layout = state?.layout
2021-06-11 22:51:45 +02:00
let matchingLayer: LayerConfig = undefined
2021-06-14 19:21:33 +02:00
for (const layer of layout?.layers ?? []) {
2021-06-11 22:51:45 +02:00
if (layer.source.osmTags.matchesProperties(tags)) {
2022-09-08 21:40:48 +02:00
matchingLayer = layer
break
2021-06-11 22:51:45 +02:00
}
2020-10-14 12:15:09 +02:00
}
2022-09-08 21:40:48 +02:00
const title =
matchingLayer?.title?.GetRenderValue(tags)?.Subs(tags)?.ConstructElement()
?.textContent ??
tags.name ??
"https//osm.org/" + tags.id
2021-06-11 22:51:45 +02:00
const description = [
2021-12-21 18:35:31 +01:00
"author:" + state.osmConnection.userDetails.data.name,
2021-06-11 22:51:45 +02:00
"license:" + license,
"osmid:" + tags.id,
2022-09-08 21:40:48 +02:00
].join("\n")
2020-07-27 13:04:38 +02:00
2021-06-11 22:51:45 +02:00
uploader.uploadMany(title, description, filelist)
})
const uploadFlow: BaseUIElement = new Combine([
2022-09-08 21:40:48 +02:00
new VariableUiElement(
uploader.queue
.map((q) => q.length)
.map((l) => {
if (l == 0) {
return undefined
}
if (l == 1) {
return new Loading(t.uploadingPicture).SetClass("alert")
} else {
return new Loading(
2023-05-11 17:29:25 +02:00
t.uploadingMultiple.Subs({count: "" + l})
2022-09-08 21:40:48 +02:00
).SetClass("alert")
}
})
),
new VariableUiElement(
uploader.failed
.map((q) => q.length)
.map((l) => {
if (l == 0) {
return undefined
}
console.log(l)
return t.uploadFailed.SetClass("block alert")
})
),
new VariableUiElement(
uploadedCount.map((l) => {
if (l == 0) {
return undefined
}
if (l == 1) {
return t.uploadDone.Clone().SetClass("thanks block")
}
2023-05-11 17:29:25 +02:00
return t.uploadMultipleDone.Subs({count: l}).SetClass("thanks block")
2022-09-08 21:40:48 +02:00
})
),
2021-11-07 16:34:51 +01:00
2021-06-11 22:51:45 +02:00
fileSelector,
new Combine([
Translations.t.image.respectPrivacy,
new VariableUiElement(
licenseStore.map((license) =>
2023-05-11 17:29:25 +02:00
Translations.t.image.currentLicense.Subs({license})
)
)
.onClick(() => {
console.log("Opening the license settings... ")
state.guistate.openUsersettings("picture-license")
})
.SetClass("underline"),
]).SetStyle("font-size:small;"),
2023-01-13 14:31:28 +01:00
]).SetClass("flex flex-col image-upload-flow mt-4 mb-8 text-center leading-none")
2021-06-11 22:51:45 +02:00
2021-06-14 19:21:33 +02:00
super(
2023-01-06 03:37:22 +01:00
new LoginToggle(
2021-06-11 22:51:45 +02:00
/*We can show the actual upload button!*/
uploadFlow,
2023-01-06 03:37:22 +01:00
/* User not logged in*/ t.pleaseLogin.Clone(),
state
2021-06-11 22:51:45 +02:00
),
undefined /* Nothing as the user badge is disabled*/,
state?.featureSwitchUserbadge
2021-06-11 22:51:45 +02:00
)
}
2022-09-08 21:40:48 +02:00
}