2025-04-09 23:30:39 +02:00
|
|
|
import ImageProvider, { PanoramaView, ProvidedImage } from "./ImageProvider"
|
2021-09-29 23:56:59 +02:00
|
|
|
import BaseUIElement from "../../UI/BaseUIElement"
|
|
|
|
import { Utils } from "../../Utils"
|
|
|
|
import { LicenseInfo } from "./LicenseInfo"
|
2021-10-06 02:30:23 +02:00
|
|
|
import Wikimedia from "../Web/Wikimedia"
|
2024-01-10 23:46:57 +01:00
|
|
|
import SvelteUIElement from "../../UI/Base/SvelteUIElement"
|
|
|
|
import Wikimedia_commons_white from "../../assets/svg/Wikimedia_commons_white.svelte"
|
2025-04-09 23:30:39 +02:00
|
|
|
import { Feature, Point } from "geojson"
|
2021-09-29 23:56:59 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This module provides endpoints for wikimedia and others
|
|
|
|
*/
|
|
|
|
export class WikimediaImageProvider extends ImageProvider {
|
|
|
|
public static readonly singleton = new WikimediaImageProvider()
|
2023-09-27 22:21:35 +02:00
|
|
|
public static readonly apiUrls = [
|
2021-10-06 17:48:07 +02:00
|
|
|
"https://commons.wikimedia.org/wiki/",
|
2024-04-13 02:40:21 +02:00
|
|
|
"https://upload.wikimedia.org",
|
2021-10-06 17:48:07 +02:00
|
|
|
]
|
2023-09-27 22:21:35 +02:00
|
|
|
public static readonly commonsPrefixes = [...WikimediaImageProvider.apiUrls, "File:"]
|
2021-11-07 16:34:51 +01:00
|
|
|
private readonly commons_key = "wikimedia_commons"
|
|
|
|
public readonly defaultKeyPrefixes = [this.commons_key, "image"]
|
2024-07-27 12:59:38 +02:00
|
|
|
public readonly name = "Wikimedia"
|
2021-09-29 23:56:59 +02:00
|
|
|
|
|
|
|
private constructor() {
|
|
|
|
super()
|
|
|
|
}
|
2021-09-30 00:26:21 +02:00
|
|
|
|
2025-04-28 00:53:23 +02:00
|
|
|
/**
|
|
|
|
* Replaces (multiple) spaces to underscores.
|
|
|
|
* Will remove a "File:"-prefix
|
|
|
|
*
|
|
|
|
* WikimediaImageProvider.makeCanonical("Some File.jpg") // => "Some_File.jpg"
|
|
|
|
*
|
|
|
|
* // Double spaces
|
|
|
|
* WikimediaImageProvider.makeCanonical("Some File.jpg") // => "Some_File.jpg"
|
|
|
|
* WikimediaImageProvider.makeCanonical("Some+File.jpg") // => "Some+File.jpg"
|
|
|
|
*
|
|
|
|
* // Remove File: prefix
|
|
|
|
*/
|
|
|
|
private static makeCanonical(filename: string): string {
|
|
|
|
if (filename.startsWith("File:")) {
|
|
|
|
filename = filename.substring(5)
|
|
|
|
}
|
|
|
|
return filename.trim().replace(/\s+/g, "_")
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* WikimediaImageProvider.extractFileName("https://commons.wikimedia.org/wiki/File:Somefile.jpg") // => "Somefile.jpg"
|
|
|
|
* WikimediaImageProvider.extractFileName("https://commons.wikimedia.org/wiki/File:S%C3%A8vres%20-%20square_madame_de_Pompadour_-_bo%C3%AEte_%C3%A0_livres.jpg?uselang=en") // => "Sèvres_-_square_madame_de_Pompadour_-_boîte_à_livres.jpg"
|
|
|
|
*/
|
|
|
|
private static extractFileName(url: string) {
|
2021-09-29 23:56:59 +02:00
|
|
|
if (!url.startsWith("http")) {
|
|
|
|
return url
|
|
|
|
}
|
2025-04-28 00:53:23 +02:00
|
|
|
const path = decodeURIComponent(new URL(url).pathname)
|
|
|
|
return WikimediaImageProvider.makeCanonical(path.substring(path.lastIndexOf("/") + 1))
|
2021-09-29 23:56:59 +02:00
|
|
|
}
|
|
|
|
|
2024-03-22 16:22:57 +01:00
|
|
|
private static PrepareUrl(value: string, useHd = false): string {
|
2021-11-07 16:34:51 +01:00
|
|
|
if (value.toLowerCase().startsWith("https://commons.wikimedia.org/wiki/")) {
|
|
|
|
return value
|
|
|
|
}
|
2024-03-22 16:22:57 +01:00
|
|
|
const baseUrl = `https://commons.wikimedia.org/wiki/Special:FilePath/${encodeURIComponent(
|
2024-10-19 14:44:55 +02:00
|
|
|
value
|
2024-03-22 16:22:57 +01:00
|
|
|
)}`
|
|
|
|
if (useHd) {
|
|
|
|
return baseUrl
|
|
|
|
}
|
|
|
|
return baseUrl + `?width=500&height=400`
|
2021-11-07 16:34:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private static startsWithCommonsPrefix(value: string): boolean {
|
|
|
|
return WikimediaImageProvider.commonsPrefixes.some((prefix) => value.startsWith(prefix))
|
|
|
|
}
|
|
|
|
|
|
|
|
private static removeCommonsPrefix(value: string): string {
|
|
|
|
if (value.startsWith("https://upload.wikimedia.org/")) {
|
|
|
|
value = value.substring(value.lastIndexOf("/") + 1)
|
|
|
|
value = decodeURIComponent(value)
|
|
|
|
if (!value.startsWith("File:")) {
|
|
|
|
value = "File:" + value
|
|
|
|
}
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const prefix of WikimediaImageProvider.commonsPrefixes) {
|
|
|
|
if (value.startsWith(prefix)) {
|
|
|
|
let part = value.substr(prefix.length)
|
|
|
|
if (prefix.startsWith("http")) {
|
|
|
|
part = decodeURIComponent(part)
|
|
|
|
}
|
|
|
|
return part
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
2023-09-27 22:21:35 +02:00
|
|
|
apiUrls(): string[] {
|
|
|
|
return WikimediaImageProvider.apiUrls
|
|
|
|
}
|
|
|
|
|
2023-12-02 03:12:34 +01:00
|
|
|
SourceIcon(): BaseUIElement {
|
2024-01-10 23:46:57 +01:00
|
|
|
return new SvelteUIElement(Wikimedia_commons_white).SetStyle("width:2em;height: 2em")
|
2021-09-29 23:56:59 +02:00
|
|
|
}
|
|
|
|
|
2024-07-11 16:59:10 +02:00
|
|
|
public PrepUrl(value: NonNullable<string>): ProvidedImage
|
|
|
|
public PrepUrl(value: undefined): undefined
|
|
|
|
|
|
|
|
public PrepUrl(value: string): ProvidedImage
|
2024-07-21 10:52:51 +02:00
|
|
|
public PrepUrl(value: string | undefined): ProvidedImage | undefined {
|
|
|
|
if (value === undefined) {
|
2024-07-11 16:59:10 +02:00
|
|
|
return undefined
|
|
|
|
}
|
2021-11-07 16:34:51 +01:00
|
|
|
value = WikimediaImageProvider.removeCommonsPrefix(value)
|
2021-09-29 23:56:59 +02:00
|
|
|
|
2021-11-07 16:34:51 +01:00
|
|
|
if (value.startsWith("File:")) {
|
|
|
|
return this.UrlForImage(value)
|
2021-09-29 23:56:59 +02:00
|
|
|
}
|
2021-11-07 16:34:51 +01:00
|
|
|
|
|
|
|
// We do a last effort and assume this is a file
|
|
|
|
return this.UrlForImage("File:" + value)
|
|
|
|
}
|
|
|
|
|
2025-04-28 00:53:23 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param key
|
|
|
|
* @param value
|
|
|
|
* @constructor
|
|
|
|
*
|
|
|
|
* const result = await WikimediaImageProvider.singleton.ExtractUrls("wikimedia_commons", "File:Sèvres_-_square_madame_de_Pompadour_-_boîte_à_livres.jpg")
|
|
|
|
* result[0].url_hd // => "https://commons.wikimedia.org/wiki/Special:FilePath/File%3AS%C3%A8vres_-_square_madame_de_Pompadour_-_bo%C3%AEte_%C3%A0_livres.jpg"
|
|
|
|
*/
|
2024-09-28 02:04:14 +02:00
|
|
|
public async ExtractUrls(key: string, value: string): undefined | Promise<ProvidedImage[]> {
|
2021-11-07 16:34:51 +01:00
|
|
|
const hasCommonsPrefix = WikimediaImageProvider.startsWithCommonsPrefix(value)
|
|
|
|
if (key !== undefined && key !== this.commons_key && !hasCommonsPrefix) {
|
2024-09-28 02:04:14 +02:00
|
|
|
return undefined
|
2021-11-07 16:34:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
value = WikimediaImageProvider.removeCommonsPrefix(value)
|
|
|
|
if (value.startsWith("Category:")) {
|
|
|
|
const urls = await Wikimedia.GetCategoryContents(value)
|
2024-10-19 14:44:55 +02:00
|
|
|
return urls
|
|
|
|
.filter((url) => url.startsWith("File:"))
|
2024-09-28 02:04:14 +02:00
|
|
|
.map((image) => this.UrlForImage(image))
|
2021-11-07 16:34:51 +01:00
|
|
|
}
|
|
|
|
if (value.startsWith("File:")) {
|
2025-06-07 02:52:06 +02:00
|
|
|
return [this.UrlForImage(value, key, value)]
|
2021-11-07 16:34:51 +01:00
|
|
|
}
|
|
|
|
if (value.startsWith("http")) {
|
2024-09-28 02:04:14 +02:00
|
|
|
// Probably an error
|
|
|
|
return undefined
|
2021-11-07 16:34:51 +01:00
|
|
|
}
|
|
|
|
// We do a last effort and assume this is a file
|
2025-06-07 02:52:06 +02:00
|
|
|
return [this.UrlForImage("File:" + value, key, value)]
|
2021-09-29 23:56:59 +02:00
|
|
|
}
|
|
|
|
|
2025-06-05 12:21:38 +02:00
|
|
|
public async DownloadAttribution(img: { id: string }): Promise<LicenseInfo> {
|
|
|
|
const filename = "File:" + WikimediaImageProvider.extractFileName(img.id)
|
|
|
|
console.log("Downloading attribution for", filename, img.id)
|
2021-09-29 23:56:59 +02:00
|
|
|
if (filename === "") {
|
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
|
|
|
|
const url =
|
|
|
|
"https://en.wikipedia.org/w/" +
|
|
|
|
"api.php?action=query&prop=imageinfo&iiprop=extmetadata&" +
|
|
|
|
"titles=" +
|
|
|
|
filename +
|
|
|
|
"&format=json&origin=*"
|
2024-08-09 16:55:08 +02:00
|
|
|
const data = await Utils.downloadJsonCached<{
|
|
|
|
query: { pages: { title: string; imageinfo: { extmetadata }[] }[] }
|
|
|
|
}>(url, 365 * 24 * 60 * 60)
|
2021-09-29 23:56:59 +02:00
|
|
|
const licenseInfo = new LicenseInfo()
|
2025-01-25 02:06:29 +01:00
|
|
|
const pages = data.query.pages
|
|
|
|
/*jup, a literal "-1" in an object, not a list!*/
|
|
|
|
let pageInfo = pages["-1"]
|
|
|
|
if (Array.isArray(pages)) {
|
|
|
|
pageInfo = pages.at(-1)
|
|
|
|
}
|
2021-11-07 16:34:51 +01:00
|
|
|
if (pageInfo === undefined) {
|
2025-04-28 00:53:23 +02:00
|
|
|
console.warn("No attribution found for wikimedia image:", filename)
|
2021-10-07 22:06:47 +02:00
|
|
|
return undefined
|
|
|
|
}
|
2022-09-08 21:40:48 +02:00
|
|
|
|
2021-10-07 22:06:47 +02:00
|
|
|
const license = (pageInfo.imageinfo ?? [])[0]?.extmetadata
|
2021-09-29 23:56:59 +02:00
|
|
|
if (license === undefined) {
|
2021-11-07 16:34:51 +01:00
|
|
|
console.warn(
|
2025-05-03 23:48:35 +02:00
|
|
|
"The file",
|
|
|
|
filename,
|
|
|
|
"has no usable metedata or license attached... Please fix the license info file yourself!"
|
2021-11-07 16:34:51 +01:00
|
|
|
)
|
2021-09-29 23:56:59 +02:00
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
|
2025-04-28 00:53:23 +02:00
|
|
|
let title = WikimediaImageProvider.makeCanonical(pageInfo.title)
|
2021-11-07 16:34:51 +01:00
|
|
|
if (title.endsWith(".jpg") || title.endsWith(".png")) {
|
2021-10-07 22:06:47 +02:00
|
|
|
title = title.substring(0, title.length - 4)
|
|
|
|
}
|
2021-11-07 16:34:51 +01:00
|
|
|
|
2021-10-07 22:06:47 +02:00
|
|
|
licenseInfo.title = title
|
2021-09-29 23:56:59 +02:00
|
|
|
licenseInfo.artist = license.Artist?.value
|
|
|
|
licenseInfo.license = license.License?.value
|
|
|
|
licenseInfo.copyrighted = license.Copyrighted?.value
|
|
|
|
licenseInfo.attributionRequired = license.AttributionRequired?.value
|
|
|
|
licenseInfo.usageTerms = license.UsageTerms?.value
|
|
|
|
licenseInfo.licenseShortName = license.LicenseShortName?.value
|
|
|
|
licenseInfo.credit = license.Credit?.value
|
|
|
|
licenseInfo.description = license.ImageDescription?.value
|
2022-08-30 20:29:49 +02:00
|
|
|
licenseInfo.informationLocation = new URL("https://en.wikipedia.org/wiki/" + pageInfo.title)
|
2021-09-29 23:56:59 +02:00
|
|
|
return licenseInfo
|
|
|
|
}
|
|
|
|
|
2025-06-07 02:52:06 +02:00
|
|
|
private UrlForImage(image: string, key?: string, value?: string): ProvidedImage {
|
2025-04-28 00:53:23 +02:00
|
|
|
image = "File:" + WikimediaImageProvider.makeCanonical(image)
|
2025-06-07 02:52:06 +02:00
|
|
|
const providedImage: ProvidedImage = {
|
2023-12-19 22:08:00 +01:00
|
|
|
url: WikimediaImageProvider.PrepareUrl(image),
|
2024-03-22 16:22:57 +01:00
|
|
|
url_hd: WikimediaImageProvider.PrepareUrl(image, true),
|
2023-12-19 22:08:00 +01:00
|
|
|
key: undefined,
|
|
|
|
provider: this,
|
2024-04-13 02:40:21 +02:00
|
|
|
id: image,
|
2025-04-15 18:18:44 +02:00
|
|
|
isSpherical: false,
|
2023-12-19 22:08:00 +01:00
|
|
|
}
|
2025-06-18 21:40:01 +02:00
|
|
|
if (key && value) {
|
|
|
|
providedImage.originalAttribute = { key, value }
|
2025-06-07 02:52:06 +02:00
|
|
|
}
|
|
|
|
return providedImage
|
2021-09-29 23:56:59 +02:00
|
|
|
}
|
2025-04-09 23:30:39 +02:00
|
|
|
|
2025-04-23 21:35:43 +02:00
|
|
|
getPanoramaInfo(): Promise<Feature<Point, PanoramaView>> | undefined {
|
|
|
|
return undefined
|
|
|
|
}
|
|
|
|
|
|
|
|
visitUrl(): string | undefined {
|
2025-04-09 23:30:39 +02:00
|
|
|
return undefined
|
|
|
|
}
|
2021-09-29 23:56:59 +02:00
|
|
|
}
|