import {VariableUiElement} from "../Base/VariableUIElement"; import {UIEventSource} from "../../Logic/UIEventSource"; import Wikidata, {WikidataResponse} from "../../Logic/Web/Wikidata"; import {Translation} from "../i18n/Translation"; import {FixedUiElement} from "../Base/FixedUiElement"; import Loading from "../Base/Loading"; import {Transform} from "stream"; import Translations from "../i18n/Translations"; import Combine from "../Base/Combine"; import Img from "../Base/Img"; import {WikimediaImageProvider} from "../../Logic/ImageProviders/WikimediaImageProvider"; import Link from "../Base/Link"; import Svg from "../../Svg"; import BaseUIElement from "../BaseUIElement"; import {Utils} from "../../Utils"; export default class WikidataPreviewBox extends VariableUiElement { constructor(wikidataId: UIEventSource) { let inited = false; const wikidata = wikidataId .stabilized(250) .bind(id => { if (id === undefined || id === "" || id === "Q") { return null; } inited = true; return Wikidata.LoadWikidataEntry(id) }) super(wikidata.map(maybeWikidata => { if (maybeWikidata === null || !inited) { return undefined; } if (maybeWikidata === undefined) { return new Loading(Translations.t.general.loading) } if (maybeWikidata["error"] !== undefined) { return new FixedUiElement(maybeWikidata["error"]).SetClass("alert") } const wikidata = maybeWikidata["success"] return WikidataPreviewBox.WikidataResponsePreview(wikidata) })) } public static WikidataResponsePreview(wikidata: WikidataResponse): BaseUIElement { let link = new Link( new Combine([ wikidata.id, Svg.wikidata_ui().SetStyle("width: 2.5rem").SetClass("block") ]).SetClass("flex"), Wikidata.IdToArticle(wikidata.id), true)?.SetClass("must-link") let info = new Combine([ new Combine( [Translation.fromMap(wikidata.labels)?.SetClass("font-bold"), link]).SetClass("flex justify-between"), Translation.fromMap(wikidata.descriptions), WikidataPreviewBox.QuickFacts(wikidata) ]).SetClass("flex flex-col link-underline") let imageUrl = undefined if (wikidata.claims.get("P18")?.size > 0) { imageUrl = Array.from(wikidata.claims.get("P18"))[0] } if (imageUrl) { imageUrl = WikimediaImageProvider.singleton.PrepUrl(imageUrl).url info = new Combine([new Img(imageUrl).SetStyle("max-width: 5rem; width: unset; height: 4rem").SetClass("rounded-xl mr-2"), info.SetClass("w-full")]).SetClass("flex") } info.SetClass("p-2 w-full") return info } private static isHuman = [ {p: 31/*is a*/, q: 5 /* human */}, ] // @ts-ignore // @ts-ignore private static extraProperties: { requires?: { p: number, q?: number }[], property: string, display: Translation | Map BaseUIElement) /*If translation: Subs({value: * }) */> }[] = [ { requires: WikidataPreviewBox.isHuman, property: "P21", display: new Map([ ['Q6581097', () => Svg.gender_male_ui().SetStyle("width: 1rem; height: auto")], ['Q6581072', () => Svg.gender_female_ui().SetStyle("width: 1rem; height: auto")], ['Q1097630',() => Svg.gender_inter_ui().SetStyle("width: 1rem; height: auto")], ['Q1052281',() => Svg.gender_trans_ui().SetStyle("width: 1rem; height: auto")/*'transwomen'*/], ['Q2449503',() => Svg.gender_trans_ui().SetStyle("width: 1rem; height: auto")/*'transmen'*/], ['Q48270',() => Svg.gender_queer_ui().SetStyle("width: 1rem; height: auto")] ]) }, { property: "P569", requires: WikidataPreviewBox.isHuman, display: new Translation({ "*":"Born: {value}" }) }, { property: "P570", requires: WikidataPreviewBox.isHuman, display: new Translation({ "*":"Died: {value}" }) } ] public static QuickFacts(wikidata: WikidataResponse): BaseUIElement { const els : BaseUIElement[] = [] for (const extraProperty of WikidataPreviewBox.extraProperties) { let hasAllRequirements =true for (const requirement of extraProperty.requires) { if(!wikidata.claims.has("P"+requirement.p)){ hasAllRequirements = false; break } if(!wikidata.claims.get("P"+requirement.p).has("Q"+requirement.q)){ hasAllRequirements = false; break } } if(!hasAllRequirements){ continue } const key = extraProperty.property const display = extraProperty.display const value: string[] = Array.from(wikidata.claims.get(key)) if(value === undefined){ continue } if(display instanceof Translation){ els.push(display.Subs({value: value.join(", ")}).SetClass("m-2")) continue } const constructors = Utils.NoNull(value.map(property => display.get(property))) const elems = constructors.map(v => { if(typeof v === "string"){ return new FixedUiElement(v) }else{ return v(); } }) els.push(new Combine(elems).SetClass("flex m-2")) } if(els.length === 0){ return undefined; } return new Combine(els).SetClass("flex") } }