forked from MapComplete/MapComplete
		
	
		
			
				
	
	
		
			116 lines
		
	
	
		
			No EOL
		
	
	
		
			4.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			116 lines
		
	
	
		
			No EOL
		
	
	
		
			4.2 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import {UIEventSource} from "../Logic/UIEventSource";
 | |
| import {VariableUiElement} from "./Base/VariableUIElement";
 | |
| import Wikipedia from "../Logic/Web/Wikipedia";
 | |
| import Loading from "./Base/Loading";
 | |
| import {FixedUiElement} from "./Base/FixedUiElement";
 | |
| import Combine from "./Base/Combine";
 | |
| import BaseUIElement from "./BaseUIElement";
 | |
| import Title from "./Base/Title";
 | |
| import Translations from "./i18n/Translations";
 | |
| import Svg from "../Svg";
 | |
| import Wikidata from "../Logic/Web/Wikidata";
 | |
| import Locale from "./i18n/Locale";
 | |
| 
 | |
| export default class WikipediaBox extends Combine {
 | |
| 
 | |
|     private static async ExtractWikiPages(wikidata): Promise<Map<string, string>> {
 | |
|         return (await Wikidata.LoadWikidataEntry(wikidata)).wikisites
 | |
|     }
 | |
|     
 | |
|     
 | |
|     private static _cache = new Map()
 | |
| 
 | |
|     constructor(wikidataId: string | UIEventSource<string>) {
 | |
|         const wp = Translations.t.general.wikipedia;
 | |
|         if(typeof wikidataId === "string"){
 | |
|             wikidataId = new UIEventSource(wikidataId)
 | |
|         }
 | |
|         
 | |
|         const knownPages = new UIEventSource<{success:Map<string, string>}|{error:any}>(undefined)
 | |
|             
 | |
|         wikidataId.addCallbackAndRunD(wikidataId => {
 | |
|             WikipediaBox.ExtractWikiPages(wikidataId).then(pages => {
 | |
|                 knownPages.setData({success:pages})
 | |
|             }).catch(err=> {
 | |
|                 knownPages.setData({error: err})
 | |
|             })
 | |
|         })
 | |
|         
 | |
|         const cachedPages = new Map<string, BaseUIElement>()
 | |
|         
 | |
|         const contents = new VariableUiElement(
 | |
|             knownPages.map(pages => {
 | |
| 
 | |
|                 if (pages === undefined) {
 | |
|                     return new Loading(wp.loading.Clone())
 | |
|                 }
 | |
|                 if (pages["error"] !== undefined) {
 | |
|                     return wp.failed.Clone().SetClass("alert p-4")
 | |
|                 }
 | |
|                 const dict: Map<string, string> = pages["success"]
 | |
| 
 | |
|                 const preferredLanguage = [Locale.language.data, "en", Array.from(dict.keys())[0]]
 | |
|                 let language
 | |
|                 let pagetitle;
 | |
|                 let i = 0
 | |
|                 do {
 | |
|                     language = preferredLanguage[i]
 | |
|                     pagetitle = dict.get(language)
 | |
|                     i++;
 | |
|                     if(i >= preferredLanguage.length){
 | |
|                         return wp.noWikipediaPage.Clone()
 | |
|                     }
 | |
|                 } while (pagetitle === undefined)
 | |
|                 
 | |
|                 if(cachedPages.has(language)){
 | |
|                     return cachedPages.get(language)
 | |
|                 }
 | |
| 
 | |
|                 const page = WikipediaBox.createContents(pagetitle, language);
 | |
|                 cachedPages.set(language, page)
 | |
|                 return page
 | |
|             }, [Locale.language])
 | |
|         ).SetClass("overflow-auto normal-background rounded-lg")
 | |
| 
 | |
| 
 | |
|         super([
 | |
|             new Combine([Svg.wikipedia_ui().SetStyle("width: 1.5rem").SetClass("mr-3"),
 | |
|                 new Title(Translations.t.general.wikipedia.wikipediaboxTitle.Clone(), 2)]).SetClass("flex"),
 | |
|             contents])
 | |
| 
 | |
|         this
 | |
|             .SetClass("block rounded-xl subtle-background m-1 p-2 flex flex-col")
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the actual content in a scrollable way
 | |
|      * @param pagename
 | |
|      * @param language
 | |
|      * @private
 | |
|      */
 | |
|     private static createContents(pagename: string, language: string): BaseUIElement {
 | |
|         const htmlContent = Wikipedia.GetArticle({
 | |
|             pageName: pagename,
 | |
|             language: language
 | |
|         })
 | |
|         const wp = Translations.t.general.wikipedia
 | |
|         const contents: UIEventSource<string | BaseUIElement> = htmlContent.map(htmlContent => {
 | |
|             if (htmlContent === undefined) {
 | |
|                 // Still loading
 | |
|                 return new Loading(wp.loading.Clone())
 | |
|             }
 | |
|             if (htmlContent["success"] !== undefined) {
 | |
|                 return new FixedUiElement(htmlContent["success"]).SetClass("wikipedia-article")
 | |
|             }
 | |
|             if (htmlContent["error"]) {
 | |
|                 return wp.failed.Clone().SetClass("alert p-4")
 | |
|             }
 | |
| 
 | |
|             return undefined
 | |
|         })
 | |
| 
 | |
|         return new Combine([new VariableUiElement(contents).SetClass("block pl-6 pt-2")])
 | |
|             .SetClass("block")
 | |
|     }
 | |
| 
 | |
| } |