var Persist = {Servers: {}, Sources: {}, Identities: {},}; var Present = CopyObj(Persist); var Tasker = {}; var ApiCache = { __Store__(Data, Key, Where) { ApiCache[Where][Key] = Data; ApiCache[Where][Key].__Time__ = Date.now(); }, __UrlStore__(Data) { ApiCache.__Store__(Data, Data.Url, 'Urls'); }, Urls: {}, }; Assets._ = function _(Name) { if (Name in Assets) { if (Assets[Name].startsWith('data:')) { return Assets[Name]; } else { return `./Assets/${Assets[Name]}`; }; }; }; Strings._ = function _(Name) { // TODO: Handle arbitrary nestation if (Name in Strings) { if (Strings[Name]['en']) { // TODO{ Select this language from user config return Strings[Name]['en']; } else { return Strings[Name].en; }; }; }; function DoAsync(First, Then, Data) { var Job = RndId(); Tasker[Job] = { Remains: 0, Return(Data) { this.Remains -= 1; this.Result = Data; }, }; // Call all First functs ForceList(First).forEach(function(Fun){ var Task = RndId(); var Proc = [Job, Task]; Tasker[Job][Task] = {}; Tasker[Job].Remains += 1; //Fun(Proc, Data); Data ? Fun(Data, Proc) : Fun(Proc); }); // Continuosly check when First functs completed Tasker[Job].Interval = setInterval(function(Job, Then){ if (Tasker[Job].Remains <= 0) { clearInterval(Tasker[Job].Interval); // Call all Then functs ForceList(Then).forEach(function(Fun){ Fun(Tasker[Job].Result); }); delete Tasker[Job]; }; }, 50, Job, Then); return Job; }; function HtmlAssign(Id, Data) { Array.from(document.getElementsByClassName(Id)).forEach(function(El){ HtmlAssignPropper(El, Data); }); Array.from(document.getElementById(Id).querySelectorAll(`*:not([class~="${Id}"])`)).forEach(function(El){ if (El.dataset.assign) { HtmlAssignPropper(El, Data); }; }); }; function HtmlAssignPropper(El, Data) { El.dataset.assign.trim().split(' ').forEach(function(Att){ var Toks = Att.split(':'); var Prop = Data; Toks[1].split('.').forEach(function(Key){ Prop = Prop[Key]; }); if (Prop !== undefined) { El[Toks[0]] = Prop; }; }); }; function DisplayProfile(Profile) { Profile = UrlObj(Profile); var Window = MkWindow({className: "Profile"}); Window.innerHTML += `
`; // TODO: Handle fetching notes of non-standard profiles like servers timelines DoAsync(FetchNotes, FillTimeline, Profile); }; function DisplayMastodonTimeline(Data) { var Window = MkWindow(); Window.innerHTML += ``; DoAsync(FetchNotes, FillTimeline, Profile); }; function FetchNotes(Profile, Proc) { var Soft = Profile.__Software__; NetApiCall({Target: UrlBase(Profile.Url), Method: ApiEndpoints.FetchNotes['Mastodon'](Profile), CallFine: function(Res){ var Notes = ApiTransform(Res.responseJson, Soft, 'Note'); LogDebug(Notes, 'l'); Tasker[Res.Proc[0]].Return(Notes); }}, Proc); }; function FetchMastodon(Proc) { //if (UseFakeApi) { // ResFetchMastodon({responseJson: [FakeApi.Mastodon.Status], Proc: Proc}); //} else { NetApiCall({Target: "Mastodon", Method: "GET timelines/public", CallFine: ResFetchMastodon}, Proc); //}; }; function ResFetchMastodon(Res) { var Notes = ApiTransform(Res.responseJson, 'Mastodon', 'Note'); LogDebug(Notes, 'l'); Tasker[Res.Proc[0]].Return(Notes); }; function FillTimeline(Notes) { Notes.forEach(function(Note){ ApiCache.__UrlStore__(Note.Profile); Root.lastChild.innerHTML += `Posting in: [Channel]
`; }; function PostNote(Text) { Obj = ExtrimObj(ApiSchema.Note); Obj.Content = Text; // Find out current profile and destination channel to do a proper net request }; function ManageSettings() { MkWindow().innerHTML = `
Language: ${MkSelectMenu([{innerHTML: "en"}, {innerHTML: "it"},]).outerHTML}
Theme:
`; }; FillHome(); CoverView.remove();