MapComplete/src/Logic/ImageProviders/Mapillary.ts

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

180 lines
6.4 KiB
TypeScript
Raw Normal View History

import ImageProvider, { ProvidedImage } from "./ImageProvider"
import BaseUIElement from "../../UI/BaseUIElement"
2021-07-03 22:24:12 +02:00
import { Utils } from "../../Utils"
import { LicenseInfo } from "./LicenseInfo"
import Constants from "../../Models/Constants"
import SvelteUIElement from "../../UI/Base/SvelteUIElement"
import MapillaryIcon from "./MapillaryIcon.svelte"
export class Mapillary extends ImageProvider {
public static readonly singleton = new Mapillary()
2021-10-01 02:57:41 +02:00
private static readonly valuePrefix = "https://a.mapillary.com"
2021-11-07 16:34:51 +01:00
public static readonly valuePrefixes = [
Mapillary.valuePrefix,
"http://mapillary.com",
"https://mapillary.com",
"http://www.mapillary.com",
"https://www.mapillary.com",
]
defaultKeyPrefixes = ["mapillary", "image"]
/**
* Indicates that this is the same URL
* Ignores 'stp' parameter
*
* const a = "https://scontent-bru2-1.xx.fbcdn.net/m1/v/t6/An8xm5SGLt20ETziNqzhhBd8b8S5GHLiIu8N6BbyqHFohFAQoaJJPG8i5yQiSwjYmEqXSfVeoCmpiyBJICEkQK98JOB21kkJoBS8VdhYa-Ty93lBnznQyesJBtKcb32foGut2Hgt10hEMWJbE3dDgA?stp=s1024x768&ccb=10-5&oh=00_AT-ZGTXHzihoaQYBILmEiAEKR64z_IWiTlcAYq_D7Ka0-Q&oe=6278C456&_nc_sid=122ab1"
* const b = "https://scontent-bru2-1.xx.fbcdn.net/m1/v/t6/An8xm5SGLt20ETziNqzhhBd8b8S5GHLiIu8N6BbyqHFohFAQoaJJPG8i5yQiSwjYmEqXSfVeoCmpiyBJICEkQK98JOB21kkJoBS8VdhYa-Ty93lBnznQyesJBtKcb32foGut2Hgt10hEMWJbE3dDgA?stp=s256x192&ccb=10-5&oh=00_AT9BZ1Rpc9zbY_uNu92A_4gj1joiy1b6VtgtLIu_7wh9Bg&oe=6278C456&_nc_sid=122ab1"
* Mapillary.sameUrl(a, b) => true
*/
static sameUrl(a: string, b: string): boolean {
if (a === b) {
return true
}
try {
const aUrl = new URL(a)
const bUrl = new URL(b)
if (aUrl.host !== bUrl.host || aUrl.pathname !== bUrl.pathname) {
return false
}
let allSame = true
aUrl.searchParams.forEach((value, key) => {
if (key === "stp") {
// This is the key indicating the image size on mapillary; we ignore it
return
}
if (value !== bUrl.searchParams.get(key)) {
allSame = false
return
}
})
return allSame
} catch (e) {
console.debug("Could not compare ", a, "and", b, "due to", e)
}
return false
}
static createLink(
location: {
lon: number
lat: number
} = undefined,
zoom: number = 17,
pKey?: string
) {
2023-12-02 03:12:34 +01:00
const params = {
focus: pKey === undefined ? "map" : "photo",
lat: location?.lat,
lng: location?.lon,
2023-12-02 03:12:34 +01:00
z: location === undefined ? undefined : Math.max((zoom ?? 2) - 1, 1),
pKey,
}
const baselink = `https://www.mapillary.com/app/?`
const paramsStr = Utils.NoNull(
Object.keys(params).map((k) =>
params[k] === undefined ? undefined : k + "=" + params[k]
)
)
2023-12-02 03:12:34 +01:00
return baselink + paramsStr.join("&")
}
/**
* Returns the correct key for API v4.0
*/
private static ExtractKeyFromURL(value: string): number {
let key: string
2021-11-07 16:34:51 +01:00
if (value.startsWith("http")) {
try {
const url = new URL(value.toLowerCase())
if (url.searchParams.has("pkey")) {
const pkey = Number(url.searchParams.get("pkey"))
if (!isNaN(pkey)) {
return pkey
}
}
} catch (e) {
console.log("Could not parse value for mapillary:", value)
}
}
if (value.startsWith(Mapillary.valuePrefix)) {
key = value.substring(0, value.lastIndexOf("?")).substring(value.lastIndexOf("/") + 1)
} else if (value.match("[0-9]*")) {
key = value
}
const keyAsNumber = Number(key)
if (!isNaN(keyAsNumber)) {
return keyAsNumber
}
2021-11-07 16:34:51 +01:00
return undefined
}
2023-12-02 03:12:34 +01:00
apiUrls(): string[] {
return ["https://mapillary.com", "https://www.mapillary.com", "https://graph.mapillary.com"]
}
SourceIcon(
id: string,
location?: {
lon: number
lat: number
}
): BaseUIElement {
let url: string = undefined
if (id) {
url = Mapillary.createLink(location, 16, "" + id)
2023-12-02 03:12:34 +01:00
}
return new SvelteUIElement(MapillaryIcon, { url })
}
async ExtractUrls(key: string, value: string): Promise<Promise<ProvidedImage>[]> {
2021-10-18 22:17:41 +02:00
return [this.PrepareUrlAsync(key, value)]
}
2024-07-19 11:57:53 +02:00
public async DownloadAttribution(providedImage: {id: string}): Promise<LicenseInfo> {
2024-04-01 02:00:48 +02:00
const mapillaryId = providedImage.id
const metadataUrl =
"https://graph.mapillary.com/" +
mapillaryId +
"?fields=thumb_1024_url,thumb_original_url,captured_at,creator&access_token=" +
Constants.mapillary_client_token_v4
const response = await Utils.downloadJsonCached(metadataUrl, 60 * 60)
const license = new LicenseInfo()
2024-04-01 02:00:48 +02:00
license.artist = response["creator"]["username"]
license.license = "CC BY-SA 4.0"
// license.license = "Creative Commons Attribution-ShareAlike 4.0 International License";
license.attributionRequired = true
2024-04-01 02:00:48 +02:00
license.date = new Date(response["captured_at"])
return license
}
2021-11-07 16:34:51 +01:00
private async PrepareUrlAsync(key: string, value: string): Promise<ProvidedImage> {
const mapillaryId = Mapillary.ExtractKeyFromURL(value)
if (mapillaryId === undefined) {
2021-11-07 16:34:51 +01:00
return undefined
}
const metadataUrl =
"https://graph.mapillary.com/" +
mapillaryId +
2024-04-01 02:00:48 +02:00
"?fields=thumb_1024_url,thumb_original_url,captured_at,creator&access_token=" +
Constants.mapillary_client_token_v4
2022-06-13 00:51:53 +02:00
const response = await Utils.downloadJsonCached(metadataUrl, 60 * 60)
const url = <string>response["thumb_1024_url"]
const url_hd = <string>response["thumb_original_url"]
2024-04-01 02:00:48 +02:00
const date = new Date()
date.setTime(response["captured_at"])
2024-04-13 02:40:21 +02:00
return <ProvidedImage>{
2023-12-02 03:12:34 +01:00
id: "" + mapillaryId,
url,
url_hd,
provider: this,
2024-04-01 02:00:48 +02:00
date,
key,
2021-11-07 16:34:51 +01:00
}
}
}