You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
25 lines
254 KiB
25 lines
254 KiB
var je=Object.create;var K=Object.defineProperty;var qe=Object.getOwnPropertyDescriptor;var Je=Object.getOwnPropertyNames;var We=Object.getPrototypeOf,Qe=Object.prototype.hasOwnProperty;var Ke=(i,t)=>()=>(t||i((t={exports:{}}).exports,t),t.exports),Ge=(i,t)=>{for(var e in t)K(i,e,{get:t[e],enumerable:!0})},pe=(i,t,e,n)=>{if(t&&typeof t=="object"||typeof t=="function")for(let s of Je(t))!Qe.call(i,s)&&s!==e&&K(i,s,{get:()=>t[s],enumerable:!(n=qe(t,s))||n.enumerable});return i};var Ze=(i,t,e)=>(e=i!=null?je(We(i)):{},pe(t||!i||!i.__esModule?K(e,"default",{value:i,enumerable:!0}):e,i)),Xe=i=>pe(K({},"__esModule",{value:!0}),i);var a=(i,t,e)=>new Promise((n,s)=>{var o=g=>{try{l(e.next(g))}catch(u){s(u)}},r=g=>{try{l(e.throw(g))}catch(u){s(u)}},l=g=>g.done?n(g.value):Promise.resolve(g.value).then(o,r);l((e=e.apply(i,t)).next())});var ze=Ke(c=>{"use strict";Object.defineProperty(c,"__esModule",{value:!0});var f=require("obsidian"),ae="YYYY-MM-DD",re="gggg-[W]ww",Ne="YYYY-MM",Ee="YYYY-[Q]Q",Le="YYYY";function z(i){var e,n;let t=window.app.plugins.getPlugin("periodic-notes");return t&&((n=(e=t.settings)==null?void 0:e[i])==null?void 0:n.enabled)}function _(){var i,t,e,n;try{let{internalPlugins:s,plugins:o}=window.app;if(z("daily")){let{format:u,folder:m,template:d}=((t=(i=o.getPlugin("periodic-notes"))==null?void 0:i.settings)==null?void 0:t.daily)||{};return{format:u||ae,folder:(m==null?void 0:m.trim())||"",template:(d==null?void 0:d.trim())||""}}let{folder:r,format:l,template:g}=((n=(e=s.getPluginById("daily-notes"))==null?void 0:e.instance)==null?void 0:n.options)||{};return{format:l||ae,folder:(r==null?void 0:r.trim())||"",template:(g==null?void 0:g.trim())||""}}catch(s){console.info("No custom daily note settings found!",s)}}function H(){var i,t,e,n,s,o,r;try{let l=window.app.plugins,g=(i=l.getPlugin("calendar"))==null?void 0:i.options,u=(e=(t=l.getPlugin("periodic-notes"))==null?void 0:t.settings)==null?void 0:e.weekly;if(z("weekly"))return{format:u.format||re,folder:((n=u.folder)==null?void 0:n.trim())||"",template:((s=u.template)==null?void 0:s.trim())||""};let m=g||{};return{format:m.weeklyNoteFormat||re,folder:((o=m.weeklyNoteFolder)==null?void 0:o.trim())||"",template:((r=m.weeklyNoteTemplate)==null?void 0:r.trim())||""}}catch(l){console.info("No custom weekly note settings found!",l)}}function V(){var t,e,n,s;let i=window.app.plugins;try{let o=z("monthly")&&((e=(t=i.getPlugin("periodic-notes"))==null?void 0:t.settings)==null?void 0:e.monthly)||{};return{format:o.format||Ne,folder:((n=o.folder)==null?void 0:n.trim())||"",template:((s=o.template)==null?void 0:s.trim())||""}}catch(o){console.info("No custom monthly note settings found!",o)}}function Y(){var t,e,n,s;let i=window.app.plugins;try{let o=z("quarterly")&&((e=(t=i.getPlugin("periodic-notes"))==null?void 0:t.settings)==null?void 0:e.quarterly)||{};return{format:o.format||Ee,folder:((n=o.folder)==null?void 0:n.trim())||"",template:((s=o.template)==null?void 0:s.trim())||""}}catch(o){console.info("No custom quarterly note settings found!",o)}}function j(){var t,e,n,s;let i=window.app.plugins;try{let o=z("yearly")&&((e=(t=i.getPlugin("periodic-notes"))==null?void 0:t.settings)==null?void 0:e.yearly)||{};return{format:o.format||Le,folder:((n=o.folder)==null?void 0:n.trim())||"",template:((s=o.template)==null?void 0:s.trim())||""}}catch(o){console.info("No custom yearly note settings found!",o)}}function Ie(...i){let t=[];for(let n=0,s=i.length;n<s;n++)t=t.concat(i[n].split("/"));let e=[];for(let n=0,s=t.length;n<s;n++){let o=t[n];!o||o==="."||e.push(o)}return t[0]===""&&e.unshift(""),e.join("/")}function st(i){let t=i.substring(i.lastIndexOf("/")+1);return t.lastIndexOf(".")!=-1&&(t=t.substring(0,t.lastIndexOf("."))),t}function ot(i){return a(this,null,function*(){let t=i.replace(/\\/g,"/").split("/");if(t.pop(),t.length){let e=Ie(...t);window.app.vault.getAbstractFileByPath(e)||(yield window.app.vault.createFolder(e))}})}function q(i,t){return a(this,null,function*(){t.endsWith(".md")||(t+=".md");let e=f.normalizePath(Ie(i,t));return yield ot(e),e})}function D(i){return a(this,null,function*(){let{metadataCache:t,vault:e}=window.app,n=f.normalizePath(i);if(n==="/")return Promise.resolve(["",null]);try{let s=t.getFirstLinkpathDest(n,""),o=yield e.cachedRead(s),r=window.app.foldManager.load(s);return[o,r]}catch(s){return console.error(`Failed to read the daily note template '${n}'`,s),new f.Notice("Failed to read the daily note template"),["",null]}})}function C(i,t="day"){let e=i.clone().startOf(t).format();return`${t}-${e}`}function ke(i){return i.replace(/\[[^\]]*\]/g,"")}function at(i,t){if(t==="week"){let e=ke(i);return/w{1,2}/i.test(e)&&(/M{1,4}/.test(e)||/D{1,4}/.test(e))}return!1}function B(i,t){return De(i.basename,t)}function rt(i,t){return De(st(i),t)}function De(i,t){let n={day:_,week:H,month:V,quarter:Y,year:j}[t]().format.split("/").pop(),s=window.moment(i,n,!0);if(!s.isValid())return null;if(at(n,t)&&t==="week"){let o=ke(n);if(/w{1,2}/i.test(o))return window.moment(i,n.replace(/M{1,4}/g,"").replace(/D{1,4}/g,""),!1)}return s}var le=class extends Error{};function Be(i){return a(this,null,function*(){let t=window.app,{vault:e}=t,n=window.moment,{template:s,format:o,folder:r}=_(),[l,g]=yield D(s),u=i.format(o),m=yield q(r,u);try{let d=yield e.create(m,l.replace(/{{\s*date\s*}}/gi,u).replace(/{{\s*time\s*}}/gi,n().format("HH:mm")).replace(/{{\s*title\s*}}/gi,u).replace(/{{\s*(date|time)\s*(([+-]\d+)([yqmwdhs]))?\s*(:.+?)?}}/gi,(h,T,w,b,P,y)=>{let se=n(),oe=i.clone().set({hour:se.get("hour"),minute:se.get("minute"),second:se.get("second")});return w&&oe.add(parseInt(b,10),P),y?oe.format(y.substring(1).trim()):oe.format(o)}).replace(/{{\s*yesterday\s*}}/gi,i.clone().subtract(1,"day").format(o)).replace(/{{\s*tomorrow\s*}}/gi,i.clone().add(1,"d").format(o)));return t.foldManager.save(d,g),d}catch(d){console.error(`Failed to create file: '${m}'`,d),new f.Notice("Unable to create new file.")}})}function lt(i,t){var e;return(e=t[C(i,"day")])!=null?e:null}function gt(){let{vault:i}=window.app,{folder:t}=_(),e=i.getAbstractFileByPath(f.normalizePath(t));if(!e)throw new le("Failed to find daily notes folder");let n={};return f.Vault.recurseChildren(e,s=>{if(s instanceof f.TFile){let o=B(s,"day");if(o){let r=C(o,"day");n[r]=s}}}),n}var ge=class extends Error{};function ut(){let{moment:i}=window,t=i.localeData()._week.dow,e=["sunday","monday","tuesday","wednesday","thursday","friday","saturday"];for(;t;)e.push(e.shift()),t--;return e}function ct(i){return ut().indexOf(i.toLowerCase())}function Me(i){return a(this,null,function*(){let{vault:t}=window.app,{template:e,format:n,folder:s}=H(),[o,r]=yield D(e),l=i.format(n),g=yield q(s,l);try{let u=yield t.create(g,o.replace(/{{\s*(date|time)\s*(([+-]\d+)([yqmwdhs]))?\s*(:.+?)?}}/gi,(m,d,h,T,w,b)=>{let P=window.moment(),y=i.clone().set({hour:P.get("hour"),minute:P.get("minute"),second:P.get("second")});return h&&y.add(parseInt(T,10),w),b?y.format(b.substring(1).trim()):y.format(n)}).replace(/{{\s*title\s*}}/gi,l).replace(/{{\s*time\s*}}/gi,window.moment().format("HH:mm")).replace(/{{\s*(sunday|monday|tuesday|wednesday|thursday|friday|saturday)\s*:(.*?)}}/gi,(m,d,h)=>{let T=ct(d);return i.weekday(T).format(h.trim())}));return window.app.foldManager.save(u,r),u}catch(u){console.error(`Failed to create file: '${g}'`,u),new f.Notice("Unable to create new file.")}})}function dt(i,t){var e;return(e=t[C(i,"week")])!=null?e:null}function pt(){let i={};if(!xe())return i;let{vault:t}=window.app,{folder:e}=H(),n=t.getAbstractFileByPath(f.normalizePath(e));if(!n)throw new ge("Failed to find weekly notes folder");return f.Vault.recurseChildren(n,s=>{if(s instanceof f.TFile){let o=B(s,"week");if(o){let r=C(o,"week");i[r]=s}}}),i}var ue=class extends Error{};function Re(i){return a(this,null,function*(){let{vault:t}=window.app,{template:e,format:n,folder:s}=V(),[o,r]=yield D(e),l=i.format(n),g=yield q(s,l);try{let u=yield t.create(g,o.replace(/{{\s*(date|time)\s*(([+-]\d+)([yqmwdhs]))?\s*(:.+?)?}}/gi,(m,d,h,T,w,b)=>{let P=window.moment(),y=i.clone().set({hour:P.get("hour"),minute:P.get("minute"),second:P.get("second")});return h&&y.add(parseInt(T,10),w),b?y.format(b.substring(1).trim()):y.format(n)}).replace(/{{\s*date\s*}}/gi,l).replace(/{{\s*time\s*}}/gi,window.moment().format("HH:mm")).replace(/{{\s*title\s*}}/gi,l));return window.app.foldManager.save(u,r),u}catch(u){console.error(`Failed to create file: '${g}'`,u),new f.Notice("Unable to create new file.")}})}function mt(i,t){var e;return(e=t[C(i,"month")])!=null?e:null}function ft(){let i={};if(!Oe())return i;let{vault:t}=window.app,{folder:e}=V(),n=t.getAbstractFileByPath(f.normalizePath(e));if(!n)throw new ue("Failed to find monthly notes folder");return f.Vault.recurseChildren(n,s=>{if(s instanceof f.TFile){let o=B(s,"month");if(o){let r=C(o,"month");i[r]=s}}}),i}var ce=class extends Error{};function ht(i){return a(this,null,function*(){let{vault:t}=window.app,{template:e,format:n,folder:s}=Y(),[o,r]=yield D(e),l=i.format(n),g=yield q(s,l);try{let u=yield t.create(g,o.replace(/{{\s*(date|time)\s*(([+-]\d+)([yqmwdhs]))?\s*(:.+?)?}}/gi,(m,d,h,T,w,b)=>{let P=window.moment(),y=i.clone().set({hour:P.get("hour"),minute:P.get("minute"),second:P.get("second")});return h&&y.add(parseInt(T,10),w),b?y.format(b.substring(1).trim()):y.format(n)}).replace(/{{\s*date\s*}}/gi,l).replace(/{{\s*time\s*}}/gi,window.moment().format("HH:mm")).replace(/{{\s*title\s*}}/gi,l));return window.app.foldManager.save(u,r),u}catch(u){console.error(`Failed to create file: '${g}'`,u),new f.Notice("Unable to create new file.")}})}function bt(i,t){var e;return(e=t[C(i,"quarter")])!=null?e:null}function wt(){let i={};if(!$e())return i;let{vault:t}=window.app,{folder:e}=Y(),n=t.getAbstractFileByPath(f.normalizePath(e));if(!n)throw new ce("Failed to find quarterly notes folder");return f.Vault.recurseChildren(n,s=>{if(s instanceof f.TFile){let o=B(s,"quarter");if(o){let r=C(o,"quarter");i[r]=s}}}),i}var de=class extends Error{};function yt(i){return a(this,null,function*(){let{vault:t}=window.app,{template:e,format:n,folder:s}=j(),[o,r]=yield D(e),l=i.format(n),g=yield q(s,l);try{let u=yield t.create(g,o.replace(/{{\s*(date|time)\s*(([+-]\d+)([yqmwdhs]))?\s*(:.+?)?}}/gi,(m,d,h,T,w,b)=>{let P=window.moment(),y=i.clone().set({hour:P.get("hour"),minute:P.get("minute"),second:P.get("second")});return h&&y.add(parseInt(T,10),w),b?y.format(b.substring(1).trim()):y.format(n)}).replace(/{{\s*date\s*}}/gi,l).replace(/{{\s*time\s*}}/gi,window.moment().format("HH:mm")).replace(/{{\s*title\s*}}/gi,l));return window.app.foldManager.save(u,r),u}catch(u){console.error(`Failed to create file: '${g}'`,u),new f.Notice("Unable to create new file.")}})}function Tt(i,t){var e;return(e=t[C(i,"year")])!=null?e:null}function Pt(){let i={};if(!Ue())return i;let{vault:t}=window.app,{folder:e}=j(),n=t.getAbstractFileByPath(f.normalizePath(e));if(!n)throw new de("Failed to find yearly notes folder");return f.Vault.recurseChildren(n,s=>{if(s instanceof f.TFile){let o=B(s,"year");if(o){let r=C(o,"year");i[r]=s}}}),i}function vt(){var n,s;let{app:i}=window,t=i.internalPlugins.plugins["daily-notes"];if(t&&t.enabled)return!0;let e=i.plugins.getPlugin("periodic-notes");return e&&((s=(n=e.settings)==null?void 0:n.daily)==null?void 0:s.enabled)}function xe(){var e,n;let{app:i}=window;if(i.plugins.getPlugin("calendar"))return!0;let t=i.plugins.getPlugin("periodic-notes");return t&&((n=(e=t.settings)==null?void 0:e.weekly)==null?void 0:n.enabled)}function Oe(){var e,n;let{app:i}=window,t=i.plugins.getPlugin("periodic-notes");return t&&((n=(e=t.settings)==null?void 0:e.monthly)==null?void 0:n.enabled)}function $e(){var e,n;let{app:i}=window,t=i.plugins.getPlugin("periodic-notes");return t&&((n=(e=t.settings)==null?void 0:e.quarterly)==null?void 0:n.enabled)}function Ue(){var e,n;let{app:i}=window,t=i.plugins.getPlugin("periodic-notes");return t&&((n=(e=t.settings)==null?void 0:e.yearly)==null?void 0:n.enabled)}function Ct(i){let t={day:_,week:H,month:V,quarter:Y,year:j}[i];return t()}function St(i,t){return{day:Be,month:Re,week:Me}[i](t)}c.DEFAULT_DAILY_NOTE_FORMAT=ae;c.DEFAULT_MONTHLY_NOTE_FORMAT=Ne;c.DEFAULT_QUARTERLY_NOTE_FORMAT=Ee;c.DEFAULT_WEEKLY_NOTE_FORMAT=re;c.DEFAULT_YEARLY_NOTE_FORMAT=Le;c.appHasDailyNotesPluginLoaded=vt;c.appHasMonthlyNotesPluginLoaded=Oe;c.appHasQuarterlyNotesPluginLoaded=$e;c.appHasWeeklyNotesPluginLoaded=xe;c.appHasYearlyNotesPluginLoaded=Ue;c.createDailyNote=Be;c.createMonthlyNote=Re;c.createPeriodicNote=St;c.createQuarterlyNote=ht;c.createWeeklyNote=Me;c.createYearlyNote=yt;c.getAllDailyNotes=gt;c.getAllMonthlyNotes=ft;c.getAllQuarterlyNotes=wt;c.getAllWeeklyNotes=pt;c.getAllYearlyNotes=Pt;c.getDailyNote=lt;c.getDailyNoteSettings=_;c.getDateFromFile=B;c.getDateFromPath=rt;c.getDateUID=C;c.getMonthlyNote=mt;c.getMonthlyNoteSettings=V;c.getPeriodicNoteSettings=Ct;c.getQuarterlyNote=bt;c.getQuarterlyNoteSettings=Y;c.getTemplateInfo=D;c.getWeeklyNote=dt;c.getWeeklyNoteSettings=H;c.getYearlyNote=Tt;c.getYearlyNoteSettings=j});var Ft={};Ge(Ft,{default:()=>ie});module.exports=Xe(Ft);var Ye=require("obsidian");var v=require("obsidian");var N=require("obsidian");var F=require("obsidian");var et="https://raw.githubusercontent.com/",G=(i,t,e,n=!0)=>a(void 0,null,function*(){let s=`https://github.com/${i}/releases/download/${t}/${e}`;try{let o=yield(0,F.request)({url:s});return o==="Not Found"||o==='{"error":"Not Found"}'?null:o}catch(o){return n&&console.log("error in grabReleaseFileFromRepository",s,o),null}}),me=(i,t=!0,e=!0)=>a(void 0,null,function*(){let n=et+i+(t===!0?"/HEAD/manifest.json":"/HEAD/manifest-beta.json");try{let s=yield(0,F.request)({url:n});return s==="404: Not Found"?null:yield JSON.parse(s)}catch(s){return s!="Error: Request failed, status 404"&&e&&console.log(`error in grabManifestJsonFromRepository for ${n}`,s),null}}),fe=(i=!0)=>a(void 0,null,function*(){let t="https://raw.githubusercontent.com/obsidianmd/obsidian-releases/HEAD/community-plugins.json";try{let e=yield(0,F.request)({url:t});return e==="404: Not Found"?null:yield JSON.parse(e)}catch(e){return i&&console.log("error in grabCommmunityPluginList",e),null}}),he=(i=!0)=>a(void 0,null,function*(){let t="https://raw.githubusercontent.com/obsidianmd/obsidian-releases/HEAD/community-css-themes.json";try{let e=yield(0,F.request)({url:t});return e==="404: Not Found"?null:yield JSON.parse(e)}catch(e){return i&&console.log("error in grabCommmunityThemesList",e),null}}),L=(i,t=!1,e)=>a(void 0,null,function*(){let n=`https://raw.githubusercontent.com/${i}/HEAD/theme${t?"-beta":""}.css`;try{let s=yield(0,F.request)({url:n});return s==="404: Not Found"?null:s}catch(s){return e&&console.log("error in grabCommmunityThemeCssFile",s),null}}),be=(i,t=!0)=>a(void 0,null,function*(){let e=`https://raw.githubusercontent.com/${i}/HEAD/manifest.json`;try{let n=yield(0,F.request)({url:e});return n==="404: Not Found"?null:n}catch(n){return t&&console.log("error in grabCommmunityThemeManifestFile",n),null}}),tt=i=>{let t=0;for(let e=0;e<i.length;e++)t+=i.charCodeAt(e);return t},M=i=>tt(i).toString(),R=(i,t,e)=>a(void 0,null,function*(){let n=yield L(i,t,e);return n?M(n):"0"}),nt=(i,t,e=!0)=>a(void 0,null,function*(){let n=`https://api.github.com/repos/${i}/commits?path=${t}&page=1&per_page=1`;try{let s=yield(0,F.request)({url:n});return s==="404: Not Found"?null:JSON.parse(s)}catch(s){return e&&console.log("error in grabLastCommitInfoForAFile",s),null}}),we=(i,t)=>a(void 0,null,function*(){let e=yield nt(i,t);return e[0].commit.committer.date?e[0].commit.committer.date:""});var ye={pluginList:[],pluginSubListFrozenVersion:[],themesList:[],updateAtStartup:!1,updateThemesAtStartup:!1,ribbonIconEnabled:!0,loggingEnabled:!1,loggingPath:"BRAT-log",loggingVerboseEnabled:!1,debuggingMode:!0,notificationsEnabled:!0};function Te(i,t,e=""){return a(this,null,function*(){let n=!1;i.settings.pluginList.contains(t)||(i.settings.pluginList.unshift(t),n=!0),e!==""&&i.settings.pluginSubListFrozenVersion.filter(s=>s.repo===t).length===0&&(i.settings.pluginSubListFrozenVersion.unshift({repo:t,version:e}),n=!0),n&&i.saveSettings()})}function Pe(i,t){return a(this,null,function*(){return i.settings.pluginList.contains(t)})}function ve(i,t,e){return a(this,null,function*(){let n={repo:t,lastUpdate:M(e)};i.settings.themesList.unshift(n),i.saveSettings()})}function Ce(i,t){return a(this,null,function*(){return!!i.settings.themesList.find(n=>n.repo===t)})}function Se(i,t,e){i.settings.themesList.forEach(n=>{n.repo===t&&(n.lastUpdate=e,i.saveSettings())})}var Z=require("obsidian");function p(i,t,e=10,n){if(i.settings.notificationsEnabled===!1)return;let s=n?Z.Platform.isDesktop?"(click=dismiss, right-click=Info)":"(click=dismiss)":"",o=new Z.Notice(`BRAT
|
|
${t}
|
|
${s}`,e*1e3);n&&(o.noticeEl.oncontextmenu=()=>a(this,null,function*(){n()}))}function X(){return a(this,null,function*(){try{let i=yield fetch("https://obsidian.md/?"+Math.random());return i.status>=200&&i.status<300}catch(i){return!1}})}var x=(i,t,e)=>a(void 0,null,function*(){let n=yield L(t,!0,i.settings.debuggingMode);if(n||(n=yield L(t,!1,i.settings.debuggingMode)),!n)return p(i,"There is no theme.css or theme-beta.css file in the root path of this repository, so there is no theme to install."),!1;let s=yield be(t,i.settings.debuggingMode);if(!s)return p(i,"There is no manifest.json file in the root path of this repository, so theme cannot be installed."),!1;let o=yield JSON.parse(s),r=(0,N.normalizePath)(it(i)+o.name),l=i.app.vault.adapter;(yield l.exists(r))===!1&&(yield l.mkdir(r)),yield l.write((0,N.normalizePath)(r+"/theme.css"),n),yield l.write((0,N.normalizePath)(r+"/manifest.json"),s),Se(i,t,M(n));let g="";return e?(yield ve(i,t,n),g=`${o.name} theme installed from ${t}. `,setTimeout(()=>{i.app.customCss.setTheme(o.name)},500)):g=`${o.name} theme updated from ${t}.`,i.log(g+`[Theme Info](https://github.com/${t})`,!1),p(i,`${g}`,20,()=>a(void 0,null,function*(){window.open(`https://github.com/${t}`)})),!0}),I=(i,t)=>a(void 0,null,function*(){if((yield X())===!1){console.log("BRAT: No internet detected.");return}let e,n="Checking for beta theme updates STARTED";i.log(n,!0),t&&i.settings.notificationsEnabled&&(e=new N.Notice(`BRAT
|
|
${n}`,3e4));for(let o of i.settings.themesList){let r=yield R(o.repo,!0,i.settings.debuggingMode);r==="0"&&(r=yield R(o.repo,!1,i.settings.debuggingMode)),r!==o.lastUpdate&&(yield x(i,o.repo,!1))}let s="Checking for beta theme updates COMPLETED";i.log(s,!0),t&&(i.settings.notificationsEnabled&&e.hide(),p(i,s))}),ee=(i,t)=>a(void 0,null,function*(){i.settings.themesList=i.settings.themesList.filter(n=>n.repo!=t),i.saveSettings();let e=`Removed ${t} from BRAT themes list and will no longer be updated. However, the theme files still exist in the vault. To remove them, go into Settings > Appearance and remove the theme.`;i.log(e,!0),p(i,`${e}`)}),it=i=>(0,N.normalizePath)(i.app.vault.configDir+"/themes")+"/";var te=require("obsidian");var k=(i,t=!0)=>{let e=t?40:30,n=i.createEl("div");n.style.float="right",t===!1?(n.style.padding="10px",n.style.paddingLeft="15px",n.style.paddingRight="15px"):(n.style.padding="15px",n.style.paddingLeft="15px",n.style.paddingRight="15px",n.style.marginLeft="15px");let s=n.createSpan("coffee");s.addClass("ex-coffee-span");let r=s.createEl("a",{href:"https://bit.ly/o42-kofi"}).createEl("img",{attr:{src:"https://cdn.ko-fi.com/cdn/kofi3.png?v=3"}});r.height=e;let l=n.createSpan("coffee");l.addClass("ex-twitter-span"),l.style.paddingLeft="10px";let u=l.createEl("a",{href:"https://bit.ly/o42-twitter"}).createEl("img",{attr:{src:"https://cdn.cdnlogo.com/logos/t/96/twitter-icon.svg"}});u.height=e;let m=n.createSpan("coffee");m.addClass("ex-twitter-span"),m.style.paddingLeft="10px",m.style.right="0";let h=m.createEl("a",{href:"https://bit.ly/o42-medium"}).createEl("img",{attr:{src:"https://miro.medium.com/v2/resize:fill:176:176/1*sHhtYhaCe2Uc3IU0IgKwIQ.png"}});return h.height=e,n};var E=class extends te.Modal{constructor(e,n=!1){super(e.app);this.plugin=e,this.address="",this.openSettingsTabAfterwards=n}submitForm(){return a(this,null,function*(){if(this.address==="")return;let e=this.address.replace("https://github.com/","");if(yield Ce(this.plugin,e)){p(this.plugin,"This plugin is already in the list for beta testing",10);return}(yield x(this.plugin,e,!0))&&this.close()})}onOpen(){this.contentEl.createEl("h4",{text:"Github repository for beta theme:"}),this.contentEl.createEl("form",{},e=>{e.addClass("brat-modal"),new te.Setting(e).addText(o=>{o.setPlaceholder("Repository (example: https://github.com/GitubUserName/repository-name"),o.onChange(r=>{this.address=r.trim()}),o.inputEl.addEventListener("keydown",r=>a(this,null,function*(){r.key==="Enter"&&this.address!==" "&&(r.preventDefault(),yield this.submitForm())})),o.inputEl.style.width="100%",window.setTimeout(()=>{let r=document.querySelector(".setting-item-info");r&&r.remove(),o.inputEl.focus()},10)}),e.createDiv("modal-button-container",o=>{o.createEl("button",{attr:{type:"button"},text:"Never mind"}).addEventListener("click",()=>this.close()),o.createEl("button",{attr:{type:"submit"},cls:"mod-cta",text:"Add Theme"})});let n=e.createDiv();n.style.borderTop="1px solid #ccc",n.style.marginTop="30px";let s=n.createSpan();s.innerHTML="BRAT by <a href='https://bit.ly/o42-twitter'>TFTHacker</a>",s.style.fontStyle="italic",n.appendChild(s),k(n,!1),window.setTimeout(()=>{e.querySelectorAll(".brat-modal .setting-item-info").forEach(r=>{r.remove()})},50),e.addEventListener("submit",o=>a(this,null,function*(){o.preventDefault(),this.address!==""&&(yield this.submitForm())}))})}onClose(){return a(this,null,function*(){this.openSettingsTabAfterwards&&(yield this.plugin.app.setting.open(),yield this.plugin.app.setting.openTabById("obsidian42-brat"))})}};var ne=class extends v.PluginSettingTab{constructor(e,n){super(e,n);this.plugin=n}display(){let{containerEl:e}=this;e.empty(),k(e,!0),e.createEl("h1",{text:this.plugin.appName}),e.createEl("h2",{text:"by TfTHacker"}),new v.Setting(e).setName("Auto-update plugins at startup").setDesc("If enabled all beta plugins will be checked for updates each time Obsidian starts. Note: this does not update frozen version plugins.").addToggle(s=>{s.setValue(this.plugin.settings.updateAtStartup),s.onChange(o=>a(this,null,function*(){this.plugin.settings.updateAtStartup=o,yield this.plugin.saveSettings()}))}),new v.Setting(e).setName("Auto-update themes at startup").setDesc("If enabled all beta themes will be checked for updates each time Obsidian starts.").addToggle(s=>{s.setValue(this.plugin.settings.updateThemesAtStartup),s.onChange(o=>a(this,null,function*(){this.plugin.settings.updateThemesAtStartup=o,yield this.plugin.saveSettings()}))}),new v.Setting(e).setName("Ribbon Button").setDesc("Toggle ribbon button off and on.").addToggle(s=>{s.setValue(this.plugin.settings.ribbonIconEnabled),s.onChange(o=>a(this,null,function*(){this.plugin.settings.ribbonIconEnabled=o,this.plugin.settings.ribbonIconEnabled===!1?this.plugin.ribbonIcon.remove():this.plugin.showRibbonButton(),yield this.plugin.saveSettings()}))}),e.createEl("hr"),e.createEl("h2",{text:"Beta Plugin List"}),e.createEl("div",{text:'The following is a list of beta plugins added via the command palette "Add a beta plugin for testing" or "Add a beta plugin with frozen version for testing". A frozen version is a specific release of a plugin based on its releease tag. '}),e.createEl("p"),e.createEl("div",{text:"Click the x button next to a plugin to remove it from the list."}),e.createEl("p"),e.createEl("span").createEl("b",{text:"Note: "}),e.createSpan({text:"This does not delete the plugin, this should be done from the Community Plugins tab in Settings."}),new v.Setting(e).addButton(s=>{s.setButtonText("Add Beta plugin"),s.onClick(()=>a(this,null,function*(){this.plugin.app.setting.close(),yield this.plugin.betaPlugins.displayAddNewPluginModal(!0,!1)}))});let n=new Set(this.plugin.settings.pluginSubListFrozenVersion.map(s=>s.repo));for(let s of this.plugin.settings.pluginList)n.has(s)||new v.Setting(e).setName(s).addButton(o=>{o.setIcon("cross"),o.setTooltip("Delete this beta plugin"),o.onClick(()=>a(this,null,function*(){o.buttonEl.textContent===""?o.setButtonText("Click once more to confirm removal"):(o.buttonEl.parentElement.parentElement.remove(),yield this.plugin.betaPlugins.deletePlugin(s))}))});new v.Setting(e).addButton(s=>{s.setButtonText("Add Beta plugin with frozen version"),s.onClick(()=>a(this,null,function*(){this.plugin.app.setting.close(),yield this.plugin.betaPlugins.displayAddNewPluginModal(!0,!0)}))});for(let s of this.plugin.settings.pluginSubListFrozenVersion)new v.Setting(e).setName(`${s.repo} (version ${s.version})`).addButton(o=>{o.setIcon("cross"),o.setTooltip("Delete this beta plugin"),o.onClick(()=>a(this,null,function*(){o.buttonEl.textContent===""?o.setButtonText("Click once more to confirm removal"):(o.buttonEl.parentElement.parentElement.remove(),yield this.plugin.betaPlugins.deletePlugin(s.repo))}))});e.createEl("hr"),e.createEl("h2",{text:"Beta Themes List"}),new v.Setting(e).addButton(s=>{s.setButtonText("Add Beta Theme"),s.onClick(()=>a(this,null,function*(){this.plugin.app.setting.close(),new E(this.plugin).open()}))});for(let s of this.plugin.settings.themesList)new v.Setting(e).setName(s.repo).addButton(o=>{o.setIcon("cross"),o.setTooltip("Delete this beta theme"),o.onClick(()=>a(this,null,function*(){o.buttonEl.textContent===""?o.setButtonText("Click once more to confirm removal"):(o.buttonEl.parentElement.parentElement.remove(),yield ee(this.plugin,s.repo))}))});e.createEl("hr"),e.createEl("h2",{text:"Monitoring"}),new v.Setting(e).setName("Enable Notifications").setDesc("BRAT will provide popup notifications for its various activities. Turn this off means no notifications from BRAT.").addToggle(s=>{s.setValue(this.plugin.settings.notificationsEnabled),s.onChange(o=>a(this,null,function*(){this.plugin.settings.notificationsEnabled=o,yield this.plugin.saveSettings()}))}),new v.Setting(e).setName("Enable Logging").setDesc("Plugin updates will be logged to a file in the log file.").addToggle(s=>{s.setValue(this.plugin.settings.loggingEnabled),s.onChange(o=>a(this,null,function*(){this.plugin.settings.loggingEnabled=o,yield this.plugin.saveSettings()}))}),new v.Setting(this.containerEl).setName("BRAT Log File Location").setDesc("Logs will be saved to this file. Don't add .md to the file name.").addSearch(s=>{s.setPlaceholder("Example: BRAT-log").setValue(this.plugin.settings.loggingPath).onChange(o=>a(this,null,function*(){this.plugin.settings.loggingPath=o,yield this.plugin.saveSettings()}))}),new v.Setting(e).setName("Enable Verbose Logging").setDesc("Get a lot more information in the log.").addToggle(s=>{s.setValue(this.plugin.settings.loggingVerboseEnabled),s.onChange(o=>a(this,null,function*(){this.plugin.settings.loggingVerboseEnabled=o,yield this.plugin.saveSettings()}))}),new v.Setting(e).setName("Debugging Mode").setDesc("Atomic Bomb level console logging. Can be used for troubleshoting and development.").addToggle(s=>{s.setValue(this.plugin.settings.debuggingMode),s.onChange(o=>a(this,null,function*(){this.plugin.settings.debuggingMode=o,yield this.plugin.saveSettings()}))})}};var O=require("obsidian");var $=class extends O.Modal{constructor(e,n,s=!1,o=!1){super(e.app);this.plugin=e,this.betaPlugins=n,this.address="",this.openSettingsTabAfterwards=s,this.useFrozenVersion=o,this.version=""}submitForm(){return a(this,null,function*(){if(this.address==="")return;let e=this.address.replace("https://github.com/","");if(yield Pe(this.plugin,e)){p(this.plugin,"This plugin is already in the list for beta testing",10);return}(yield this.betaPlugins.addPlugin(e,!1,!1,!1,this.version))&&this.close()})}onOpen(){this.contentEl.createEl("h4",{text:"Github repository for beta plugin:"}),this.contentEl.createEl("form",{},e=>{e.addClass("brat-modal"),new O.Setting(e).addText(o=>{o.setPlaceholder("Repository (example: https://github.com/GitubUserName/repository-name)"),o.onChange(r=>{this.address=r.trim()}),o.inputEl.addEventListener("keydown",r=>a(this,null,function*(){r.key==="Enter"&&this.address!==" "&&(this.useFrozenVersion&&this.version!==""||!this.useFrozenVersion)&&(r.preventDefault(),yield this.submitForm())})),o.inputEl.style.width="100%"}),this.useFrozenVersion&&new O.Setting(e).addText(o=>{o.setPlaceholder("Specify the release version tag (example: 1.0.0)"),o.onChange(r=>{this.version=r.trim()}),o.inputEl.style.width="100%"}),e.createDiv("modal-button-container",o=>{o.createEl("button",{attr:{type:"button"},text:"Never mind"}).addEventListener("click",()=>this.close()),o.createEl("button",{attr:{type:"submit"},cls:"mod-cta",text:"Add Plugin"})});let n=e.createDiv();n.style.borderTop="1px solid #ccc",n.style.marginTop="30px";let s=n.createSpan();s.innerHTML="BRAT by <a href='https://bit.ly/o42-twitter'>TFTHacker</a>",s.style.fontStyle="italic",n.appendChild(s),k(n,!1),window.setTimeout(()=>{e.querySelectorAll(".brat-modal .setting-item-info").forEach(r=>{r.remove()})},50),e.addEventListener("submit",o=>a(this,null,function*(){o.preventDefault(),this.address!==""&&(this.useFrozenVersion&&this.version!==""||!this.useFrozenVersion)&&(yield this.submitForm())}))})}onClose(){return a(this,null,function*(){this.openSettingsTabAfterwards&&(yield this.plugin.app.setting.open(),yield this.plugin.app.setting.openTabById("obsidian42-brat"))})}};var A=require("obsidian");var U=class{constructor(t){this.plugin=t}displayAddNewPluginModal(t=!1,e=!1){return a(this,null,function*(){new $(this.plugin,this,t,e).open()})}validateRepository(t,e=!1,n=!1){return a(this,null,function*(){let o=yield me(t,!e,this.plugin.settings.debuggingMode);return o?"id"in o?"version"in o?o:(n&&p(this.plugin,`${t}
|
|
The version attribute for the release is missing from the manifest file`,15),null):(n&&p(this.plugin,`${t}
|
|
The plugin id attribute for the release is missing from the manifest file`,15),null):(n&&p(this.plugin,`${t}
|
|
This does not seem to be an obsidian plugin, as there is no manifest.json file.`,15),null)})}getAllReleaseFiles(t,e,n,s=""){return a(this,null,function*(){let o=s===""?e.version:s,r=n||s!=="";return{mainJs:yield G(t,o,"main.js",this.plugin.settings.debuggingMode),manifest:r?yield G(t,o,"manifest.json",this.plugin.settings.debuggingMode):"",styles:yield G(t,o,"styles.css",this.plugin.settings.debuggingMode)}})}writeReleaseFilesToPluginFolder(t,e){return a(this,null,function*(){let n=(0,A.normalizePath)(this.plugin.app.vault.configDir+"/plugins/"+t)+"/",s=this.plugin.app.vault.adapter;((yield s.exists(n))===!1||!(yield s.exists(n+"manifest.json")))&&(yield s.mkdir(n)),yield s.write(n+"main.js",e.mainJs),yield s.write(n+"manifest.json",e.manifest),e.styles&&(yield s.write(n+"styles.css",e.styles))})}addPlugin(t,e=!1,n=!1,s=!1,o=""){return a(this,null,function*(){var m;let l=yield this.validateRepository(t,!0,!1),g=!!l;if(g===!1&&(l=yield this.validateRepository(t,!1,!0)),l===null){let d=`${t}
|
|
A manifest.json or manifest-beta.json file does not exist in the root directory of the repository. This plugin cannot be installed.`;return this.plugin.log(d,!0),p(this.plugin,`${d}`,10),!1}if(!l.hasOwnProperty("version")){let d=`${t}
|
|
The manifest${g?"-beta":""}.json file in the root directory of the repository does not have a version number in the file. This plugin cannot be installed.`;return this.plugin.log(d,!0),p(this.plugin,`${d}`,10),!1}if(l.hasOwnProperty("minAppVersion")&&!(0,A.requireApiVersion)(l.minAppVersion)){let d=`Plugin: ${t}
|
|
|
|
The manifest${g?"-beta":""}.json for this plugin indicates that the Obsidian version of the app needs to be ${l.minAppVersion}, but this installation of Obsidian is ${A.apiVersion}.
|
|
|
|
You will need to update your Obsidian to use this plugin or contact the plugin developer for more information.`;return this.plugin.log(d,!0),p(this.plugin,`${d}`,30),!1}let u=()=>a(this,null,function*(){let d=yield this.getAllReleaseFiles(t,l,g,o);if((g||d.manifest==="")&&(d.manifest=JSON.stringify(l)),d.mainJs===null){let h=`${t}
|
|
The release is not complete and cannot be download. main.js is missing from the Release`;return this.plugin.log(h,!0),p(this.plugin,`${h}`,10),null}return d});if(e===!1){let d=yield u();if(d===null)return!1;yield this.writeReleaseFilesToPluginFolder(l.id,d),yield Te(this.plugin,t,o),yield this.plugin.app.plugins.loadManifests();let h=o===""?"":` (version: ${o})`,T=`${t}${h}
|
|
The plugin has been registered with BRAT. You may still need to enable it the Community Plugin List.`;this.plugin.log(T,!0),p(this.plugin,T,10)}else{let d=this.plugin.app.vault.configDir+"/plugins/"+l.id+"/",h="";try{h=yield this.plugin.app.vault.adapter.read(d+"manifest.json")}catch(w){if(w.errno===-4058||w.errno===-2)return yield this.addPlugin(t,!1,g,!1,o),!0;console.log("BRAT - Local Manifest Load",l.id,JSON.stringify(w,null,2))}if(o!==""||this.plugin.settings.pluginSubListFrozenVersion.map(w=>w.repo).includes(t))return p(this.plugin,`The version of ${t} is frozen, not updating.`,3),!1;let T=yield JSON.parse(h);if(T.version!==l.version){let w=yield u();if(w===null)return!1;if(n){let b=`There is an update available for ${l.id} from version ${T.version} to ${l.version}. `;this.plugin.log(b+`[Release Info](https://github.com/${t}/releases/tag/${l.version})`,!1),p(this.plugin,b,30,()=>a(this,null,function*(){window.open(`https://github.com/${t}/releases/tag/${l.version}`)}))}else{yield this.writeReleaseFilesToPluginFolder(l.id,w),yield this.plugin.app.plugins.loadManifests(),(m=this.plugin.app.plugins.plugins[l.id])!=null&&m.manifest&&(yield this.reloadPlugin(l.id));let b=`${l.id}
|
|
Plugin has been updated from version ${T.version} to ${l.version}. `;this.plugin.log(b+`[Release Info](https://github.com/${t}/releases/tag/${l.version})`,!1),p(this.plugin,b,30,()=>a(this,null,function*(){window.open(`https://github.com/${t}/releases/tag/${l.version}`)}))}}else s&&p(this.plugin,`No update available for ${t}`,3)}return!0})}reloadPlugin(t){return a(this,null,function*(){let e=this.plugin.app.plugins;try{yield e.disablePlugin(t),yield e.enablePlugin(t)}catch(n){this.plugin.settings.debuggingMode&&console.log("reload plugin",n)}})}updatePlugin(t,e=!1,n=!1){return a(this,null,function*(){let s=yield this.addPlugin(t,!0,e,n);return s===!1&&e===!1&&p(this.plugin,`${t}
|
|
Update of plugin failed.`),s})}checkForUpdatesAndInstallUpdates(t=!1,e=!1){return a(this,null,function*(){if((yield X())===!1){console.log("BRAT: No internet detected.");return}let n,s="Checking for plugin updates STARTED";this.plugin.log(s,!0),t&&this.plugin.settings.notificationsEnabled&&(n=new A.Notice(`BRAT
|
|
${s}`,3e4));let o=new Set(this.plugin.settings.pluginSubListFrozenVersion.map(l=>l.repo));for(let l of this.plugin.settings.pluginList)o.has(l)||(yield this.updatePlugin(l,e));let r="Checking for plugin updates COMPLETED";this.plugin.log(r,!0),t&&(n.hide(),p(this.plugin,r,10))})}deletePlugin(t){return a(this,null,function*(){let e=`Removed ${t} from BRAT plugin list`;this.plugin.log(e,!0),this.plugin.settings.pluginList=this.plugin.settings.pluginList.filter(n=>n!=t),this.plugin.settings.pluginSubListFrozenVersion=this.plugin.settings.pluginSubListFrozenVersion.filter(n=>n.repo!=t),this.plugin.saveSettings()})}getEnabledDisabledPlugins(t){let e=this.plugin.app.plugins,n=Object.values(e.manifests),s=Object.values(e.plugins).map(o=>o.manifest);return t?n.filter(o=>s.find(r=>o.id===r.id)):n.filter(o=>!s.find(r=>o.id===r.id))}};var Fe=require("obsidian");function Ae(){(0,Fe.addIcon)("BratIcon",'<path fill="currentColor" stroke="currentColor" d="M 41.667969 41.667969 C 41.667969 39.367188 39.800781 37.5 37.5 37.5 C 35.199219 37.5 33.332031 39.367188 33.332031 41.667969 C 33.332031 43.96875 35.199219 45.832031 37.5 45.832031 C 39.800781 45.832031 41.667969 43.96875 41.667969 41.667969 Z M 60.417969 58.582031 C 59.460938 58.023438 58.320312 57.867188 57.25 58.148438 C 56.179688 58.429688 55.265625 59.125 54.707031 60.082031 C 53.746094 61.777344 51.949219 62.820312 50 62.820312 C 48.050781 62.820312 46.253906 61.777344 45.292969 60.082031 C 44.734375 59.125 43.820312 58.429688 42.75 58.148438 C 41.679688 57.867188 40.539062 58.023438 39.582031 58.582031 C 37.597656 59.726562 36.910156 62.257812 38.042969 64.25 C 40.5 68.53125 45.0625 71.171875 50 71.171875 C 54.9375 71.171875 59.5 68.53125 61.957031 64.25 C 63.089844 62.257812 62.402344 59.726562 60.417969 58.582031 Z M 62.5 37.5 C 60.199219 37.5 58.332031 39.367188 58.332031 41.667969 C 58.332031 43.96875 60.199219 45.832031 62.5 45.832031 C 64.800781 45.832031 66.667969 43.96875 66.667969 41.667969 C 66.667969 39.367188 64.800781 37.5 62.5 37.5 Z M 50 8.332031 C 26.988281 8.332031 8.332031 26.988281 8.332031 50 C 8.332031 73.011719 26.988281 91.667969 50 91.667969 C 73.011719 91.667969 91.667969 73.011719 91.667969 50 C 91.667969 26.988281 73.011719 8.332031 50 8.332031 Z M 50 83.332031 C 33.988281 83.402344 20.191406 72.078125 17.136719 56.363281 C 14.078125 40.644531 22.628906 24.976562 37.5 19.042969 C 37.457031 19.636719 37.457031 20.238281 37.5 20.832031 C 37.5 27.738281 43.097656 33.332031 50 33.332031 C 52.300781 33.332031 54.167969 31.46875 54.167969 29.167969 C 54.167969 26.867188 52.300781 25 50 25 C 47.699219 25 45.832031 23.132812 45.832031 20.832031 C 45.832031 18.53125 47.699219 16.667969 50 16.667969 C 68.410156 16.667969 83.332031 31.589844 83.332031 50 C 83.332031 68.410156 68.410156 83.332031 50 83.332031 Z M 50 83.332031 " />')}var J=require("obsidian"),_e=Ze(ze());function He(i,t,e=!1){if(i.settings.debuggingMode&&console.log("BRAT: "+t),i.settings.loggingEnabled){if(i.settings.loggingVerboseEnabled===!1&&e===!0)return;{let n=i.settings.loggingPath+".md",s="[["+(0,J.moment)().format((0,_e.getDailyNoteSettings)().format).toString()+"]] "+(0,J.moment)().format("HH:mm"),o=J.Platform.isDesktop?window.require("os").hostname():"MOBILE",r=s+" "+o+" "+t.replace(`
|
|
`," ")+`
|
|
|
|
`;setTimeout(()=>a(this,null,function*(){if((yield i.app.vault.adapter.exists(n))===!0){let l=yield i.app.vault.adapter.read(n);r=r+l;let g=i.app.vault.getAbstractFileByPath(n);yield i.app.vault.modify(g,r)}else yield i.app.vault.create(n,r)}),10)}}}var Ve=require("obsidian");var S=class extends Ve.FuzzySuggestModal{constructor(e){super(e.app);this.scope.register(["Shift"],"Enter",n=>this.enterTrigger(n)),this.scope.register(["Ctrl"],"Enter",n=>this.enterTrigger(n))}setSuggesterData(e){this.data=e}display(e){return a(this,null,function*(){this.callbackFunction=e,this.open()})}getItems(){return this.data}getItemText(e){return e.display}onChooseItem(){}renderSuggestion(e,n){n.createEl("div",{text:e.item.display})}enterTrigger(e){let n=document.querySelector(".suggestion-item.is-selected div").textContent,s=this.data.find(o=>o.display===n);s&&(this.invokeCallback(s,e),this.close())}onChooseSuggestion(e,n){this.invokeCallback(e.item,n)}invokeCallback(e,n){this.callbackFunction(e,n)}};var W=class{constructor(t){this.bratCommands=[{id:"BRAT-AddBetaPlugin",icon:"BratIcon",name:"Plugins: Add a beta plugin for testing",showInRibbon:!0,callback:()=>a(this,null,function*(){yield this.plugin.betaPlugins.displayAddNewPluginModal(!1,!1)})},{id:"BRAT-AddBetaPluginWithFrozenVersion",icon:"BratIcon",name:"Plugins: Add a beta plugin with frozen version based on a release tag",showInRibbon:!0,callback:()=>a(this,null,function*(){yield this.plugin.betaPlugins.displayAddNewPluginModal(!1,!0)})},{id:"BRAT-checkForUpdatesAndUpdate",icon:"BratIcon",name:"Plugins: Check for updates to all beta plugins and UPDATE",showInRibbon:!0,callback:()=>a(this,null,function*(){yield this.plugin.betaPlugins.checkForUpdatesAndInstallUpdates(!0,!1)})},{id:"BRAT-checkForUpdatesAndDontUpdate",icon:"BratIcon",name:"Plugins: Only check for updates to beta plugins, but don't Update",showInRibbon:!0,callback:()=>a(this,null,function*(){yield this.plugin.betaPlugins.checkForUpdatesAndInstallUpdates(!0,!0)})},{id:"BRAT-updateOnePlugin",icon:"BratIcon",name:"Plugins: Choose a single plugin version to update",showInRibbon:!0,callback:()=>a(this,null,function*(){let t=new Set(this.plugin.settings.pluginSubListFrozenVersion.map(s=>s.repo)),e=Object.values(this.plugin.settings.pluginList).filter(s=>!t.has(s)).map(s=>({display:s,info:s})),n=new S(this.plugin);n.setSuggesterData(e),yield n.display(s=>a(this,null,function*(){let o=`Checking for updates for ${s.info}`;this.plugin.log(o,!0),p(this.plugin,`
|
|
${o}`,3),yield this.plugin.betaPlugins.updatePlugin(s.info,!1,!0)}))})},{id:"BRAT-restartPlugin",icon:"BratIcon",name:"Plugins: Restart a plugin that is already installed",showInRibbon:!0,callback:()=>a(this,null,function*(){let t=Object.values(this.plugin.app.plugins.manifests).map(n=>({display:n.id,info:n.id})),e=new S(this.plugin);e.setSuggesterData(t),yield e.display(n=>a(this,null,function*(){p(this.plugin,`${n.info}
|
|
Plugin reloading .....`,5),yield this.plugin.betaPlugins.reloadPlugin(n.info)}))})},{id:"BRAT-disablePlugin",icon:"BratIcon",name:"Plugins: Disable a plugin - toggle it off",showInRibbon:!0,callback:()=>a(this,null,function*(){let t=this.plugin.betaPlugins.getEnabledDisabledPlugins(!0).map(n=>({display:`${n.name} (${n.id})`,info:n.id})),e=new S(this.plugin);e.setSuggesterData(t),yield e.display(n=>a(this,null,function*(){this.plugin.log(`${n.display} plugin disabled`,!1),this.plugin.settings.debuggingMode&&console.log(n.info),yield this.plugin.app.plugins.disablePluginAndSave(n.info)}))})},{id:"BRAT-enablePlugin",icon:"BratIcon",name:"Plugins: Enable a plugin - toggle it on",showInRibbon:!0,callback:()=>a(this,null,function*(){let t=this.plugin.betaPlugins.getEnabledDisabledPlugins(!1).map(n=>({display:`${n.name} (${n.id})`,info:n.id})),e=new S(this.plugin);e.setSuggesterData(t),yield e.display(n=>a(this,null,function*(){this.plugin.log(`${n.display} plugin enabled`,!1),yield this.plugin.app.plugins.enablePluginAndSave(n.info)}))})},{id:"BRAT-openGitHubZRepository",icon:"BratIcon",name:"Plugins: Open the GitHub repository for a plugin",showInRibbon:!0,callback:()=>a(this,null,function*(){let t=yield fe(this.plugin.settings.debuggingMode),e=Object.values(t).map(o=>({display:`Plugin: ${o.name} (${o.repo})`,info:o.repo})),n=Object.values(this.plugin.settings.pluginList).map(o=>({display:"BRAT: "+o,info:o}));e.forEach(o=>n.push(o));let s=new S(this.plugin);s.setSuggesterData(n),yield s.display(o=>a(this,null,function*(){o.info&&window.open(`https://github.com/${o.info}`)}))})},{id:"BRAT-openGitHubRepoTheme",icon:"BratIcon",name:"Themes: Open the GitHub repository for a theme (appearance)",showInRibbon:!0,callback:()=>a(this,null,function*(){let t=yield he(this.plugin.settings.debuggingMode),e=Object.values(t).map(s=>({display:`Theme: ${s.name} (${s.repo})`,info:s.repo})),n=new S(this.plugin);n.setSuggesterData(e),yield n.display(s=>a(this,null,function*(){s.info&&window.open(`https://github.com/${s.info}`)}))})},{id:"BRAT-opentPluginSettings",icon:"BratIcon",name:"Plugins: Open Plugin Settings Tab",showInRibbon:!0,callback:()=>a(this,null,function*(){let t=this.plugin.app.setting,e=Object.values(t.pluginTabs).map(o=>({display:"Plugin: "+o.name,info:o.id})),n=new S(this.plugin),s=Object.values(t.settingTabs).map(o=>({display:"Core: "+o.name,info:o.id}));e.forEach(o=>s.push(o)),n.setSuggesterData(s),yield n.display(o=>a(this,null,function*(){t.open(),t.openTabById(o.info)}))})},{id:"BRAT-GrabBetaTheme",icon:"BratIcon",name:"Themes: Grab a beta theme for testing from a Github repository",showInRibbon:!0,callback:()=>a(this,null,function*(){new E(this.plugin).open()})},{id:"BRAT-updateBetaThemes",icon:"BratIcon",name:"Themes: Update beta themes",showInRibbon:!0,callback:()=>a(this,null,function*(){return yield I(this.plugin,!0)})},{id:"BRAT-allCommands",icon:"BratIcon",name:"All Commands list",showInRibbon:!1,callback:()=>a(this,null,function*(){return this.ribbonDisplayCommands()})}];this.plugin=t,this.bratCommands.forEach(e=>a(this,null,function*(){this.plugin.addCommand({id:e.id,name:e.name,icon:e.icon,callback:()=>a(this,null,function*(){yield e.callback()})})}))}ribbonDisplayCommands(){return a(this,null,function*(){let t=[];this.bratCommands.forEach(r=>{r.showInRibbon&&t.push({display:r.name,info:r.callback})});let e=new S(this.plugin),n=this.plugin.app.setting,s=Object.values(n.settingTabs).map(r=>({display:"Core: "+r.name,info:()=>a(this,null,function*(){n.open(),n.openTabById(r.id)})})),o=Object.values(n.pluginTabs).map(r=>({display:"Plugin: "+r.name,info:()=>a(this,null,function*(){n.open(),n.openTabById(r.id)})}));t.push({display:"---- Core Plugin Settings ----",info:()=>a(this,null,function*(){yield this.ribbonDisplayCommands()})}),s.forEach(r=>t.push(r)),t.push({display:"---- Plugin Settings ----",info:()=>a(this,null,function*(){yield this.ribbonDisplayCommands()})}),o.forEach(r=>t.push(r)),e.setSuggesterData(t),yield e.display(r=>a(this,null,function*(){return yield r.info()}))})}};var Q=class{constructor(t){this.console=(t,...e)=>{console.log("BRAT: "+t,e)};this.themes={themeseCheckAndUpates:t=>a(this,null,function*(){yield I(this.plugin,t)}),themeInstallTheme:t=>a(this,null,function*(){let e=t.replace("https://github.com/","");yield x(this.plugin,e,!0)}),themesDelete:t=>a(this,null,function*(){let e=t.replace("https://github.com/","");yield ee(this.plugin,e)}),grabCommmunityThemeCssFile:(t,e=!1)=>a(this,null,function*(){return yield L(t,e,this.plugin.settings.debuggingMode)}),grabChecksumOfThemeCssFile:(t,e=!1)=>a(this,null,function*(){return yield R(t,e,this.plugin.settings.debuggingMode)}),grabLastCommitDateForAFile:(t,e)=>a(this,null,function*(){return yield we(t,e)})};this.plugin=t}};var ie=class extends Ye.Plugin{constructor(){super(...arguments);this.appName="Obsidian42 - Beta Reviewer's Auto-update Tool (BRAT)";this.appID="obsidian42-brat"}onload(){return a(this,null,function*(){console.log("loading Obsidian42 - BRAT"),yield this.loadSettings(),this.addSettingTab(new ne(this.app,this)),this.betaPlugins=new U(this),this.commands=new W(this),Ae(),this.settings.ribbonIconEnabled&&this.showRibbonButton(),this.app.workspace.onLayoutReady(()=>{this.settings.updateAtStartup&&setTimeout(()=>a(this,null,function*(){yield this.betaPlugins.checkForUpdatesAndInstallUpdates(!1)}),6e4),this.settings.updateThemesAtStartup&&setTimeout(()=>a(this,null,function*(){yield I(this,!1)}),12e4),setTimeout(()=>a(this,null,function*(){this.bratAPI=new Q(this),globalThis.bratAPI=this.bratAPI}),500)})})}showRibbonButton(){this.ribbonIcon=this.addRibbonIcon("BratIcon","BRAT",()=>a(this,null,function*(){return this.commands.ribbonDisplayCommands()}))}log(e,n=!1){He(this,e,n)}onunload(){console.log("unloading "+this.appName)}loadSettings(){return a(this,null,function*(){this.settings=Object.assign({},ye,yield this.loadData())})}saveSettings(){return a(this,null,function*(){yield this.saveData(this.settings)})}};
|
|
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../node_modules/obsidian-daily-notes-interface/dist/main.js", "../src/main.ts", "../src/ui/SettingsTab.ts", "../src/features/themes.ts", "../src/features/githubUtils.ts", "../src/ui/settings.ts", "../src/utils/notifications.ts", "../src/utils/internetconnection.ts", "../src/ui/AddNewTheme.ts", "../src/ui/Promotional.ts", "../src/ui/AddNewPluginModal.ts", "../src/features/BetaPlugins.ts", "../src/ui/icons.ts", "../src/utils/logging.ts", "../src/ui/GenericFuzzySuggester.ts", "../src/ui/PluginCommands.ts", "../src/utils/BratAPI.ts"],
  "sourcesContent": ["'use strict';\n\nObject.defineProperty(exports, '__esModule', { value: true });\n\nvar obsidian = require('obsidian');\n\nconst DEFAULT_DAILY_NOTE_FORMAT = \"YYYY-MM-DD\";\nconst DEFAULT_WEEKLY_NOTE_FORMAT = \"gggg-[W]ww\";\nconst DEFAULT_MONTHLY_NOTE_FORMAT = \"YYYY-MM\";\nconst DEFAULT_QUARTERLY_NOTE_FORMAT = \"YYYY-[Q]Q\";\nconst DEFAULT_YEARLY_NOTE_FORMAT = \"YYYY\";\n\nfunction shouldUsePeriodicNotesSettings(periodicity) {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const periodicNotes = window.app.plugins.getPlugin(\"periodic-notes\");\n    return periodicNotes && periodicNotes.settings?.[periodicity]?.enabled;\n}\n/**\n * Read the user settings for the `daily-notes` plugin\n * to keep behavior of creating a new note in-sync.\n */\nfunction getDailyNoteSettings() {\n    try {\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        const { internalPlugins, plugins } = window.app;\n        if (shouldUsePeriodicNotesSettings(\"daily\")) {\n            const { format, folder, template } = plugins.getPlugin(\"periodic-notes\")?.settings?.daily || {};\n            return {\n                format: format || DEFAULT_DAILY_NOTE_FORMAT,\n                folder: folder?.trim() || \"\",\n                template: template?.trim() || \"\",\n            };\n        }\n        const { folder, format, template } = internalPlugins.getPluginById(\"daily-notes\")?.instance?.options || {};\n        return {\n            format: format || DEFAULT_DAILY_NOTE_FORMAT,\n            folder: folder?.trim() || \"\",\n            template: template?.trim() || \"\",\n        };\n    }\n    catch (err) {\n        console.info(\"No custom daily note settings found!\", err);\n    }\n}\n/**\n * Read the user settings for the `weekly-notes` plugin\n * to keep behavior of creating a new note in-sync.\n */\nfunction getWeeklyNoteSettings() {\n    try {\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        const pluginManager = window.app.plugins;\n        const calendarSettings = pluginManager.getPlugin(\"calendar\")?.options;\n        const periodicNotesSettings = pluginManager.getPlugin(\"periodic-notes\")?.settings?.weekly;\n        if (shouldUsePeriodicNotesSettings(\"weekly\")) {\n            return {\n                format: periodicNotesSettings.format || DEFAULT_WEEKLY_NOTE_FORMAT,\n                folder: periodicNotesSettings.folder?.trim() || \"\",\n                template: periodicNotesSettings.template?.trim() || \"\",\n            };\n        }\n        const settings = calendarSettings || {};\n        return {\n            format: settings.weeklyNoteFormat || DEFAULT_WEEKLY_NOTE_FORMAT,\n            folder: settings.weeklyNoteFolder?.trim() || \"\",\n            template: settings.weeklyNoteTemplate?.trim() || \"\",\n        };\n    }\n    catch (err) {\n        console.info(\"No custom weekly note settings found!\", err);\n    }\n}\n/**\n * Read the user settings for the `periodic-notes` plugin\n * to keep behavior of creating a new note in-sync.\n */\nfunction getMonthlyNoteSettings() {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const pluginManager = window.app.plugins;\n    try {\n        const settings = (shouldUsePeriodicNotesSettings(\"monthly\") &&\n            pluginManager.getPlugin(\"periodic-notes\")?.settings?.monthly) ||\n            {};\n        return {\n            format: settings.format || DEFAULT_MONTHLY_NOTE_FORMAT,\n            folder: settings.folder?.trim() || \"\",\n            template: settings.template?.trim() || \"\",\n        };\n    }\n    catch (err) {\n        console.info(\"No custom monthly note settings found!\", err);\n    }\n}\n/**\n * Read the user settings for the `periodic-notes` plugin\n * to keep behavior of creating a new note in-sync.\n */\nfunction getQuarterlyNoteSettings() {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const pluginManager = window.app.plugins;\n    try {\n        const settings = (shouldUsePeriodicNotesSettings(\"quarterly\") &&\n            pluginManager.getPlugin(\"periodic-notes\")?.settings?.quarterly) ||\n            {};\n        return {\n            format: settings.format || DEFAULT_QUARTERLY_NOTE_FORMAT,\n            folder: settings.folder?.trim() || \"\",\n            template: settings.template?.trim() || \"\",\n        };\n    }\n    catch (err) {\n        console.info(\"No custom quarterly note settings found!\", err);\n    }\n}\n/**\n * Read the user settings for the `periodic-notes` plugin\n * to keep behavior of creating a new note in-sync.\n */\nfunction getYearlyNoteSettings() {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const pluginManager = window.app.plugins;\n    try {\n        const settings = (shouldUsePeriodicNotesSettings(\"yearly\") &&\n            pluginManager.getPlugin(\"periodic-notes\")?.settings?.yearly) ||\n            {};\n        return {\n            format: settings.format || DEFAULT_YEARLY_NOTE_FORMAT,\n            folder: settings.folder?.trim() || \"\",\n            template: settings.template?.trim() || \"\",\n        };\n    }\n    catch (err) {\n        console.info(\"No custom yearly note settings found!\", err);\n    }\n}\n\n// Credit: @creationix/path.js\nfunction join(...partSegments) {\n    // Split the inputs into a list of path commands.\n    let parts = [];\n    for (let i = 0, l = partSegments.length; i < l; i++) {\n        parts = parts.concat(partSegments[i].split(\"/\"));\n    }\n    // Interpret the path commands to get the new resolved path.\n    const newParts = [];\n    for (let i = 0, l = parts.length; i < l; i++) {\n        const part = parts[i];\n        // Remove leading and trailing slashes\n        // Also remove \".\" segments\n        if (!part || part === \".\")\n            continue;\n        // Push new path segments.\n        else\n            newParts.push(part);\n    }\n    // Preserve the initial slash if there was one.\n    if (parts[0] === \"\")\n        newParts.unshift(\"\");\n    // Turn back into a single string path.\n    return newParts.join(\"/\");\n}\nfunction basename(fullPath) {\n    let base = fullPath.substring(fullPath.lastIndexOf(\"/\") + 1);\n    if (base.lastIndexOf(\".\") != -1)\n        base = base.substring(0, base.lastIndexOf(\".\"));\n    return base;\n}\nasync function ensureFolderExists(path) {\n    const dirs = path.replace(/\\\\/g, \"/\").split(\"/\");\n    dirs.pop(); // remove basename\n    if (dirs.length) {\n        const dir = join(...dirs);\n        if (!window.app.vault.getAbstractFileByPath(dir)) {\n            await window.app.vault.createFolder(dir);\n        }\n    }\n}\nasync function getNotePath(directory, filename) {\n    if (!filename.endsWith(\".md\")) {\n        filename += \".md\";\n    }\n    const path = obsidian.normalizePath(join(directory, filename));\n    await ensureFolderExists(path);\n    return path;\n}\nasync function getTemplateInfo(template) {\n    const { metadataCache, vault } = window.app;\n    const templatePath = obsidian.normalizePath(template);\n    if (templatePath === \"/\") {\n        return Promise.resolve([\"\", null]);\n    }\n    try {\n        const templateFile = metadataCache.getFirstLinkpathDest(templatePath, \"\");\n        const contents = await vault.cachedRead(templateFile);\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        const IFoldInfo = window.app.foldManager.load(templateFile);\n        return [contents, IFoldInfo];\n    }\n    catch (err) {\n        console.error(`Failed to read the daily note template '${templatePath}'`, err);\n        new obsidian.Notice(\"Failed to read the daily note template\");\n        return [\"\", null];\n    }\n}\n\n/**\n * dateUID is a way of weekly identifying daily/weekly/monthly notes.\n * They are prefixed with the granularity to avoid ambiguity.\n */\nfunction getDateUID(date, granularity = \"day\") {\n    const ts = date.clone().startOf(granularity).format();\n    return `${granularity}-${ts}`;\n}\nfunction removeEscapedCharacters(format) {\n    return format.replace(/\\[[^\\]]*\\]/g, \"\"); // remove everything within brackets\n}\n/**\n * XXX: When parsing dates that contain both week numbers and months,\n * Moment choses to ignore the week numbers. For the week dateUID, we\n * want the opposite behavior. Strip the MMM from the format to patch.\n */\nfunction isFormatAmbiguous(format, granularity) {\n    if (granularity === \"week\") {\n        const cleanFormat = removeEscapedCharacters(format);\n        return (/w{1,2}/i.test(cleanFormat) &&\n            (/M{1,4}/.test(cleanFormat) || /D{1,4}/.test(cleanFormat)));\n    }\n    return false;\n}\nfunction getDateFromFile(file, granularity) {\n    return getDateFromFilename(file.basename, granularity);\n}\nfunction getDateFromPath(path, granularity) {\n    return getDateFromFilename(basename(path), granularity);\n}\nfunction getDateFromFilename(filename, granularity) {\n    const getSettings = {\n        day: getDailyNoteSettings,\n        week: getWeeklyNoteSettings,\n        month: getMonthlyNoteSettings,\n        quarter: getQuarterlyNoteSettings,\n        year: getYearlyNoteSettings,\n    };\n    const format = getSettings[granularity]().format.split(\"/\").pop();\n    const noteDate = window.moment(filename, format, true);\n    if (!noteDate.isValid()) {\n        return null;\n    }\n    if (isFormatAmbiguous(format, granularity)) {\n        if (granularity === \"week\") {\n            const cleanFormat = removeEscapedCharacters(format);\n            if (/w{1,2}/i.test(cleanFormat)) {\n                return window.moment(filename, \n                // If format contains week, remove day & month formatting\n                format.replace(/M{1,4}/g, \"\").replace(/D{1,4}/g, \"\"), false);\n            }\n        }\n    }\n    return noteDate;\n}\n\nclass DailyNotesFolderMissingError extends Error {\n}\n/**\n * This function mimics the behavior of the daily-notes plugin\n * so it will replace {{date}}, {{title}}, and {{time}} with the\n * formatted timestamp.\n *\n * Note: it has an added bonus that it's not 'today' specific.\n */\nasync function createDailyNote(date) {\n    const app = window.app;\n    const { vault } = app;\n    const moment = window.moment;\n    const { template, format, folder } = getDailyNoteSettings();\n    const [templateContents, IFoldInfo] = await getTemplateInfo(template);\n    const filename = date.format(format);\n    const normalizedPath = await getNotePath(folder, filename);\n    try {\n        const createdFile = await vault.create(normalizedPath, templateContents\n            .replace(/{{\\s*date\\s*}}/gi, filename)\n            .replace(/{{\\s*time\\s*}}/gi, moment().format(\"HH:mm\"))\n            .replace(/{{\\s*title\\s*}}/gi, filename)\n            .replace(/{{\\s*(date|time)\\s*(([+-]\\d+)([yqmwdhs]))?\\s*(:.+?)?}}/gi, (_, _timeOrDate, calc, timeDelta, unit, momentFormat) => {\n            const now = moment();\n            const currentDate = date.clone().set({\n                hour: now.get(\"hour\"),\n                minute: now.get(\"minute\"),\n                second: now.get(\"second\"),\n            });\n            if (calc) {\n                currentDate.add(parseInt(timeDelta, 10), unit);\n            }\n            if (momentFormat) {\n                return currentDate.format(momentFormat.substring(1).trim());\n            }\n            return currentDate.format(format);\n        })\n            .replace(/{{\\s*yesterday\\s*}}/gi, date.clone().subtract(1, \"day\").format(format))\n            .replace(/{{\\s*tomorrow\\s*}}/gi, date.clone().add(1, \"d\").format(format)));\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        app.foldManager.save(createdFile, IFoldInfo);\n        return createdFile;\n    }\n    catch (err) {\n        console.error(`Failed to create file: '${normalizedPath}'`, err);\n        new obsidian.Notice(\"Unable to create new file.\");\n    }\n}\nfunction getDailyNote(date, dailyNotes) {\n    return dailyNotes[getDateUID(date, \"day\")] ?? null;\n}\nfunction getAllDailyNotes() {\n    /**\n     * Find all daily notes in the daily note folder\n     */\n    const { vault } = window.app;\n    const { folder } = getDailyNoteSettings();\n    const dailyNotesFolder = vault.getAbstractFileByPath(obsidian.normalizePath(folder));\n    if (!dailyNotesFolder) {\n        throw new DailyNotesFolderMissingError(\"Failed to find daily notes folder\");\n    }\n    const dailyNotes = {};\n    obsidian.Vault.recurseChildren(dailyNotesFolder, (note) => {\n        if (note instanceof obsidian.TFile) {\n            const date = getDateFromFile(note, \"day\");\n            if (date) {\n                const dateString = getDateUID(date, \"day\");\n                dailyNotes[dateString] = note;\n            }\n        }\n    });\n    return dailyNotes;\n}\n\nclass WeeklyNotesFolderMissingError extends Error {\n}\nfunction getDaysOfWeek() {\n    const { moment } = window;\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    let weekStart = moment.localeData()._week.dow;\n    const daysOfWeek = [\n        \"sunday\",\n        \"monday\",\n        \"tuesday\",\n        \"wednesday\",\n        \"thursday\",\n        \"friday\",\n        \"saturday\",\n    ];\n    while (weekStart) {\n        daysOfWeek.push(daysOfWeek.shift());\n        weekStart--;\n    }\n    return daysOfWeek;\n}\nfunction getDayOfWeekNumericalValue(dayOfWeekName) {\n    return getDaysOfWeek().indexOf(dayOfWeekName.toLowerCase());\n}\nasync function createWeeklyNote(date) {\n    const { vault } = window.app;\n    const { template, format, folder } = getWeeklyNoteSettings();\n    const [templateContents, IFoldInfo] = await getTemplateInfo(template);\n    const filename = date.format(format);\n    const normalizedPath = await getNotePath(folder, filename);\n    try {\n        const createdFile = await vault.create(normalizedPath, templateContents\n            .replace(/{{\\s*(date|time)\\s*(([+-]\\d+)([yqmwdhs]))?\\s*(:.+?)?}}/gi, (_, _timeOrDate, calc, timeDelta, unit, momentFormat) => {\n            const now = window.moment();\n            const currentDate = date.clone().set({\n                hour: now.get(\"hour\"),\n                minute: now.get(\"minute\"),\n                second: now.get(\"second\"),\n            });\n            if (calc) {\n                currentDate.add(parseInt(timeDelta, 10), unit);\n            }\n            if (momentFormat) {\n                return currentDate.format(momentFormat.substring(1).trim());\n            }\n            return currentDate.format(format);\n        })\n            .replace(/{{\\s*title\\s*}}/gi, filename)\n            .replace(/{{\\s*time\\s*}}/gi, window.moment().format(\"HH:mm\"))\n            .replace(/{{\\s*(sunday|monday|tuesday|wednesday|thursday|friday|saturday)\\s*:(.*?)}}/gi, (_, dayOfWeek, momentFormat) => {\n            const day = getDayOfWeekNumericalValue(dayOfWeek);\n            return date.weekday(day).format(momentFormat.trim());\n        }));\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        window.app.foldManager.save(createdFile, IFoldInfo);\n        return createdFile;\n    }\n    catch (err) {\n        console.error(`Failed to create file: '${normalizedPath}'`, err);\n        new obsidian.Notice(\"Unable to create new file.\");\n    }\n}\nfunction getWeeklyNote(date, weeklyNotes) {\n    return weeklyNotes[getDateUID(date, \"week\")] ?? null;\n}\nfunction getAllWeeklyNotes() {\n    const weeklyNotes = {};\n    if (!appHasWeeklyNotesPluginLoaded()) {\n        return weeklyNotes;\n    }\n    const { vault } = window.app;\n    const { folder } = getWeeklyNoteSettings();\n    const weeklyNotesFolder = vault.getAbstractFileByPath(obsidian.normalizePath(folder));\n    if (!weeklyNotesFolder) {\n        throw new WeeklyNotesFolderMissingError(\"Failed to find weekly notes folder\");\n    }\n    obsidian.Vault.recurseChildren(weeklyNotesFolder, (note) => {\n        if (note instanceof obsidian.TFile) {\n            const date = getDateFromFile(note, \"week\");\n            if (date) {\n                const dateString = getDateUID(date, \"week\");\n                weeklyNotes[dateString] = note;\n            }\n        }\n    });\n    return weeklyNotes;\n}\n\nclass MonthlyNotesFolderMissingError extends Error {\n}\n/**\n * This function mimics the behavior of the daily-notes plugin\n * so it will replace {{date}}, {{title}}, and {{time}} with the\n * formatted timestamp.\n *\n * Note: it has an added bonus that it's not 'today' specific.\n */\nasync function createMonthlyNote(date) {\n    const { vault } = window.app;\n    const { template, format, folder } = getMonthlyNoteSettings();\n    const [templateContents, IFoldInfo] = await getTemplateInfo(template);\n    const filename = date.format(format);\n    const normalizedPath = await getNotePath(folder, filename);\n    try {\n        const createdFile = await vault.create(normalizedPath, templateContents\n            .replace(/{{\\s*(date|time)\\s*(([+-]\\d+)([yqmwdhs]))?\\s*(:.+?)?}}/gi, (_, _timeOrDate, calc, timeDelta, unit, momentFormat) => {\n            const now = window.moment();\n            const currentDate = date.clone().set({\n                hour: now.get(\"hour\"),\n                minute: now.get(\"minute\"),\n                second: now.get(\"second\"),\n            });\n            if (calc) {\n                currentDate.add(parseInt(timeDelta, 10), unit);\n            }\n            if (momentFormat) {\n                return currentDate.format(momentFormat.substring(1).trim());\n            }\n            return currentDate.format(format);\n        })\n            .replace(/{{\\s*date\\s*}}/gi, filename)\n            .replace(/{{\\s*time\\s*}}/gi, window.moment().format(\"HH:mm\"))\n            .replace(/{{\\s*title\\s*}}/gi, filename));\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        window.app.foldManager.save(createdFile, IFoldInfo);\n        return createdFile;\n    }\n    catch (err) {\n        console.error(`Failed to create file: '${normalizedPath}'`, err);\n        new obsidian.Notice(\"Unable to create new file.\");\n    }\n}\nfunction getMonthlyNote(date, monthlyNotes) {\n    return monthlyNotes[getDateUID(date, \"month\")] ?? null;\n}\nfunction getAllMonthlyNotes() {\n    const monthlyNotes = {};\n    if (!appHasMonthlyNotesPluginLoaded()) {\n        return monthlyNotes;\n    }\n    const { vault } = window.app;\n    const { folder } = getMonthlyNoteSettings();\n    const monthlyNotesFolder = vault.getAbstractFileByPath(obsidian.normalizePath(folder));\n    if (!monthlyNotesFolder) {\n        throw new MonthlyNotesFolderMissingError(\"Failed to find monthly notes folder\");\n    }\n    obsidian.Vault.recurseChildren(monthlyNotesFolder, (note) => {\n        if (note instanceof obsidian.TFile) {\n            const date = getDateFromFile(note, \"month\");\n            if (date) {\n                const dateString = getDateUID(date, \"month\");\n                monthlyNotes[dateString] = note;\n            }\n        }\n    });\n    return monthlyNotes;\n}\n\nclass QuarterlyNotesFolderMissingError extends Error {\n}\n/**\n * This function mimics the behavior of the daily-notes plugin\n * so it will replace {{date}}, {{title}}, and {{time}} with the\n * formatted timestamp.\n *\n * Note: it has an added bonus that it's not 'today' specific.\n */\nasync function createQuarterlyNote(date) {\n    const { vault } = window.app;\n    const { template, format, folder } = getQuarterlyNoteSettings();\n    const [templateContents, IFoldInfo] = await getTemplateInfo(template);\n    const filename = date.format(format);\n    const normalizedPath = await getNotePath(folder, filename);\n    try {\n        const createdFile = await vault.create(normalizedPath, templateContents\n            .replace(/{{\\s*(date|time)\\s*(([+-]\\d+)([yqmwdhs]))?\\s*(:.+?)?}}/gi, (_, _timeOrDate, calc, timeDelta, unit, momentFormat) => {\n            const now = window.moment();\n            const currentDate = date.clone().set({\n                hour: now.get(\"hour\"),\n                minute: now.get(\"minute\"),\n                second: now.get(\"second\"),\n            });\n            if (calc) {\n                currentDate.add(parseInt(timeDelta, 10), unit);\n            }\n            if (momentFormat) {\n                return currentDate.format(momentFormat.substring(1).trim());\n            }\n            return currentDate.format(format);\n        })\n            .replace(/{{\\s*date\\s*}}/gi, filename)\n            .replace(/{{\\s*time\\s*}}/gi, window.moment().format(\"HH:mm\"))\n            .replace(/{{\\s*title\\s*}}/gi, filename));\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        window.app.foldManager.save(createdFile, IFoldInfo);\n        return createdFile;\n    }\n    catch (err) {\n        console.error(`Failed to create file: '${normalizedPath}'`, err);\n        new obsidian.Notice(\"Unable to create new file.\");\n    }\n}\nfunction getQuarterlyNote(date, quarterly) {\n    return quarterly[getDateUID(date, \"quarter\")] ?? null;\n}\nfunction getAllQuarterlyNotes() {\n    const quarterly = {};\n    if (!appHasQuarterlyNotesPluginLoaded()) {\n        return quarterly;\n    }\n    const { vault } = window.app;\n    const { folder } = getQuarterlyNoteSettings();\n    const quarterlyFolder = vault.getAbstractFileByPath(obsidian.normalizePath(folder));\n    if (!quarterlyFolder) {\n        throw new QuarterlyNotesFolderMissingError(\"Failed to find quarterly notes folder\");\n    }\n    obsidian.Vault.recurseChildren(quarterlyFolder, (note) => {\n        if (note instanceof obsidian.TFile) {\n            const date = getDateFromFile(note, \"quarter\");\n            if (date) {\n                const dateString = getDateUID(date, \"quarter\");\n                quarterly[dateString] = note;\n            }\n        }\n    });\n    return quarterly;\n}\n\nclass YearlyNotesFolderMissingError extends Error {\n}\n/**\n * This function mimics the behavior of the daily-notes plugin\n * so it will replace {{date}}, {{title}}, and {{time}} with the\n * formatted timestamp.\n *\n * Note: it has an added bonus that it's not 'today' specific.\n */\nasync function createYearlyNote(date) {\n    const { vault } = window.app;\n    const { template, format, folder } = getYearlyNoteSettings();\n    const [templateContents, IFoldInfo] = await getTemplateInfo(template);\n    const filename = date.format(format);\n    const normalizedPath = await getNotePath(folder, filename);\n    try {\n        const createdFile = await vault.create(normalizedPath, templateContents\n            .replace(/{{\\s*(date|time)\\s*(([+-]\\d+)([yqmwdhs]))?\\s*(:.+?)?}}/gi, (_, _timeOrDate, calc, timeDelta, unit, momentFormat) => {\n            const now = window.moment();\n            const currentDate = date.clone().set({\n                hour: now.get(\"hour\"),\n                minute: now.get(\"minute\"),\n                second: now.get(\"second\"),\n            });\n            if (calc) {\n                currentDate.add(parseInt(timeDelta, 10), unit);\n            }\n            if (momentFormat) {\n                return currentDate.format(momentFormat.substring(1).trim());\n            }\n            return currentDate.format(format);\n        })\n            .replace(/{{\\s*date\\s*}}/gi, filename)\n            .replace(/{{\\s*time\\s*}}/gi, window.moment().format(\"HH:mm\"))\n            .replace(/{{\\s*title\\s*}}/gi, filename));\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\n        window.app.foldManager.save(createdFile, IFoldInfo);\n        return createdFile;\n    }\n    catch (err) {\n        console.error(`Failed to create file: '${normalizedPath}'`, err);\n        new obsidian.Notice(\"Unable to create new file.\");\n    }\n}\nfunction getYearlyNote(date, yearlyNotes) {\n    return yearlyNotes[getDateUID(date, \"year\")] ?? null;\n}\nfunction getAllYearlyNotes() {\n    const yearlyNotes = {};\n    if (!appHasYearlyNotesPluginLoaded()) {\n        return yearlyNotes;\n    }\n    const { vault } = window.app;\n    const { folder } = getYearlyNoteSettings();\n    const yearlyNotesFolder = vault.getAbstractFileByPath(obsidian.normalizePath(folder));\n    if (!yearlyNotesFolder) {\n        throw new YearlyNotesFolderMissingError(\"Failed to find yearly notes folder\");\n    }\n    obsidian.Vault.recurseChildren(yearlyNotesFolder, (note) => {\n        if (note instanceof obsidian.TFile) {\n            const date = getDateFromFile(note, \"year\");\n            if (date) {\n                const dateString = getDateUID(date, \"year\");\n                yearlyNotes[dateString] = note;\n            }\n        }\n    });\n    return yearlyNotes;\n}\n\nfunction appHasDailyNotesPluginLoaded() {\n    const { app } = window;\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const dailyNotesPlugin = app.internalPlugins.plugins[\"daily-notes\"];\n    if (dailyNotesPlugin && dailyNotesPlugin.enabled) {\n        return true;\n    }\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const periodicNotes = app.plugins.getPlugin(\"periodic-notes\");\n    return periodicNotes && periodicNotes.settings?.daily?.enabled;\n}\n/**\n * XXX: \"Weekly Notes\" live in either the Calendar plugin or the periodic-notes plugin.\n * Check both until the weekly notes feature is removed from the Calendar plugin.\n */\nfunction appHasWeeklyNotesPluginLoaded() {\n    const { app } = window;\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    if (app.plugins.getPlugin(\"calendar\")) {\n        return true;\n    }\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const periodicNotes = app.plugins.getPlugin(\"periodic-notes\");\n    return periodicNotes && periodicNotes.settings?.weekly?.enabled;\n}\nfunction appHasMonthlyNotesPluginLoaded() {\n    const { app } = window;\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const periodicNotes = app.plugins.getPlugin(\"periodic-notes\");\n    return periodicNotes && periodicNotes.settings?.monthly?.enabled;\n}\nfunction appHasQuarterlyNotesPluginLoaded() {\n    const { app } = window;\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const periodicNotes = app.plugins.getPlugin(\"periodic-notes\");\n    return periodicNotes && periodicNotes.settings?.quarterly?.enabled;\n}\nfunction appHasYearlyNotesPluginLoaded() {\n    const { app } = window;\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    const periodicNotes = app.plugins.getPlugin(\"periodic-notes\");\n    return periodicNotes && periodicNotes.settings?.yearly?.enabled;\n}\nfunction getPeriodicNoteSettings(granularity) {\n    const getSettings = {\n        day: getDailyNoteSettings,\n        week: getWeeklyNoteSettings,\n        month: getMonthlyNoteSettings,\n        quarter: getQuarterlyNoteSettings,\n        year: getYearlyNoteSettings,\n    }[granularity];\n    return getSettings();\n}\nfunction createPeriodicNote(granularity, date) {\n    const createFn = {\n        day: createDailyNote,\n        month: createMonthlyNote,\n        week: createWeeklyNote,\n    };\n    return createFn[granularity](date);\n}\n\nexports.DEFAULT_DAILY_NOTE_FORMAT = DEFAULT_DAILY_NOTE_FORMAT;\nexports.DEFAULT_MONTHLY_NOTE_FORMAT = DEFAULT_MONTHLY_NOTE_FORMAT;\nexports.DEFAULT_QUARTERLY_NOTE_FORMAT = DEFAULT_QUARTERLY_NOTE_FORMAT;\nexports.DEFAULT_WEEKLY_NOTE_FORMAT = DEFAULT_WEEKLY_NOTE_FORMAT;\nexports.DEFAULT_YEARLY_NOTE_FORMAT = DEFAULT_YEARLY_NOTE_FORMAT;\nexports.appHasDailyNotesPluginLoaded = appHasDailyNotesPluginLoaded;\nexports.appHasMonthlyNotesPluginLoaded = appHasMonthlyNotesPluginLoaded;\nexports.appHasQuarterlyNotesPluginLoaded = appHasQuarterlyNotesPluginLoaded;\nexports.appHasWeeklyNotesPluginLoaded = appHasWeeklyNotesPluginLoaded;\nexports.appHasYearlyNotesPluginLoaded = appHasYearlyNotesPluginLoaded;\nexports.createDailyNote = createDailyNote;\nexports.createMonthlyNote = createMonthlyNote;\nexports.createPeriodicNote = createPeriodicNote;\nexports.createQuarterlyNote = createQuarterlyNote;\nexports.createWeeklyNote = createWeeklyNote;\nexports.createYearlyNote = createYearlyNote;\nexports.getAllDailyNotes = getAllDailyNotes;\nexports.getAllMonthlyNotes = getAllMonthlyNotes;\nexports.getAllQuarterlyNotes = getAllQuarterlyNotes;\nexports.getAllWeeklyNotes = getAllWeeklyNotes;\nexports.getAllYearlyNotes = getAllYearlyNotes;\nexports.getDailyNote = getDailyNote;\nexports.getDailyNoteSettings = getDailyNoteSettings;\nexports.getDateFromFile = getDateFromFile;\nexports.getDateFromPath = getDateFromPath;\nexports.getDateUID = getDateUID;\nexports.getMonthlyNote = getMonthlyNote;\nexports.getMonthlyNoteSettings = getMonthlyNoteSettings;\nexports.getPeriodicNoteSettings = getPeriodicNoteSettings;\nexports.getQuarterlyNote = getQuarterlyNote;\nexports.getQuarterlyNoteSettings = getQuarterlyNoteSettings;\nexports.getTemplateInfo = getTemplateInfo;\nexports.getWeeklyNote = getWeeklyNote;\nexports.getWeeklyNoteSettings = getWeeklyNoteSettings;\nexports.getYearlyNote = getYearlyNote;\nexports.getYearlyNoteSettings = getYearlyNoteSettings;\n", "import { Plugin } from \"obsidian\";\nimport { BratSettingsTab } from \"./ui/SettingsTab\";\nimport { Settings, DEFAULT_SETTINGS } from \"./ui/settings\";\nimport BetaPlugins from \"./features/BetaPlugins\";\nimport { addIcons } from \"./ui/icons\";\nimport { logger } from \"./utils/logging\";\nimport PluginCommands from \"./ui/PluginCommands\";\nimport { themesCheckAndUpdates } from \"./features/themes\";\nimport BratAPI from \"./utils/BratAPI\";\n\nexport default class ThePlugin extends Plugin {\n\tappName = \"Obsidian42 - Beta Reviewer's Auto-update Tool (BRAT)\";\n\tappID = \"obsidian42-brat\";\n\tsettings: Settings;\n\tbetaPlugins: BetaPlugins;\n\tribbonIcon: HTMLElement;\n\tcommands: PluginCommands;\n\tbratAPI: BratAPI\n\n\tasync onload(): Promise<void> {\n\t\tconsole.log(\"loading Obsidian42 - BRAT\");\t\t\n\n\t\tawait this.loadSettings();\n\t\tthis.addSettingTab(new BratSettingsTab(this.app, this));\n\n\t\tthis.betaPlugins = new BetaPlugins(this);\n\t\tthis.commands = new PluginCommands(this);\n\n\t\taddIcons();\n\t\tif (this.settings.ribbonIconEnabled) this.showRibbonButton();\n\n\t\tthis.app.workspace.onLayoutReady((): void => { // let obsidian load and calm down before check\n\t\t\tif (this.settings.updateAtStartup) { \n\t\t\t\tsetTimeout(async () => {\n\t\t\t\t\tawait this.betaPlugins.checkForUpdatesAndInstallUpdates(false)\n\t\t\t\t}, 60000);\n\t\t\t}\n\t\t\tif (this.settings.updateThemesAtStartup) { \n\t\t\t\tsetTimeout(async () => {\n\t\t\t\t\tawait themesCheckAndUpdates(this, false);\n\t\t\t\t}, 120000);\n\t\t\t}\n\t\t\tsetTimeout(async () => {\n\t\t\t\tthis.bratAPI = new BratAPI(this);\n\t\t\t\tglobalThis.bratAPI = this.bratAPI;\n\t\t\t}, 500);\n\t\t});\n\t}\n\n\tshowRibbonButton(): void { this.ribbonIcon = this.addRibbonIcon(\"BratIcon\", \"BRAT\", async () => this.commands.ribbonDisplayCommands()) }\n\n\tlog(textToLog: string, verbose = false): void { logger(this, textToLog, verbose) }\n\t\n\tonunload(): void { console.log(\"unloading \" + this.appName) }\n\n\tasync loadSettings(): Promise<void> { this.settings = Object.assign({}, DEFAULT_SETTINGS, await this.loadData()) }\n\n\tasync saveSettings(): Promise<void> { await this.saveData(this.settings) }\n}", "import { App, PluginSettingTab, Setting, ToggleComponent, ButtonComponent } from 'obsidian';\nimport { themeDelete } from '../features/themes';\nimport ThePlugin from '../main';\nimport AddNewTheme from './AddNewTheme';\nimport { promotionalLinks } from './Promotional';\n\nexport class BratSettingsTab extends PluginSettingTab {\n\tplugin: ThePlugin;\n\n\tconstructor(app: App, plugin: ThePlugin) {\n\t\tsuper(app, plugin);\n\t\tthis.plugin = plugin;\n\t}\n\n\tdisplay(): void {\n\t\tconst { containerEl } = this;\n\t\tcontainerEl.empty();\n\n\t\tpromotionalLinks(containerEl, true)\n\n\t\tcontainerEl.createEl('h1', { text: this.plugin.appName }) \n\t\t// .style.marginTop = \"50px\";\n\t\tcontainerEl.createEl('h2', { text: \"by TfTHacker\" }) \n\n\t\tnew Setting(containerEl)\n\t\t\t.setName('Auto-update plugins at startup')\n\t\t\t.setDesc('If enabled all beta plugins will be checked for updates each time Obsidian starts. Note: this does not update frozen version plugins.')\n\t\t\t.addToggle((cb: ToggleComponent) => {\n\t\t\t\tcb.setValue(this.plugin.settings.updateAtStartup);\n\t\t\t\tcb.onChange(async (value: boolean) => {\n\t\t\t\t\tthis.plugin.settings.updateAtStartup = value;\n\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t});\n\t\t\t})\n\n\t\tnew Setting(containerEl)\n\t\t\t.setName('Auto-update themes at startup')\n\t\t\t.setDesc('If enabled all beta themes will be checked for updates each time Obsidian starts.')\n\t\t\t.addToggle((cb: ToggleComponent) => {\n\t\t\t\tcb.setValue(this.plugin.settings.updateThemesAtStartup);\n\t\t\t\tcb.onChange(async (value: boolean) => {\n\t\t\t\t\tthis.plugin.settings.updateThemesAtStartup = value;\n\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t});\n\t\t\t})\n\n\n\t\tnew Setting(containerEl)\n\t\t\t.setName('Ribbon Button')\n\t\t\t.setDesc('Toggle ribbon button off and on.')\n\t\t\t.addToggle((cb: ToggleComponent) => {\n\t\t\t\tcb.setValue(this.plugin.settings.ribbonIconEnabled);\n\t\t\t\tcb.onChange(async (value: boolean) => {\n\t\t\t\t\tthis.plugin.settings.ribbonIconEnabled = value;\n\t\t\t\t\tif (this.plugin.settings.ribbonIconEnabled === false)\n\t\t\t\t\t\tthis.plugin.ribbonIcon.remove();\n\t\t\t\t\telse\n\t\t\t\t\t\tthis.plugin.showRibbonButton();\n\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t});\n\t\t\t})\t\t\t\n\n\t\tcontainerEl.createEl(\"hr\");\n\t\tcontainerEl.createEl(\"h2\", { text: \"Beta Plugin List\" });\n\t\tcontainerEl.createEl(\"div\", { text: `The following is a list of beta plugins added via the command palette \"Add a beta plugin for testing\" or \"Add a beta plugin with frozen version for testing\". A frozen version is a specific release of a plugin based on its releease tag. ` });\n\t\tcontainerEl.createEl(\"p\");\n\t\tcontainerEl.createEl(\"div\", { text: `Click the x button next to a plugin to remove it from the list.` });\n\t\tcontainerEl.createEl(\"p\");\n\t\tcontainerEl.createEl(\"span\")\n\t\t\t.createEl(\"b\", { text: \"Note: \" })\n\t\tcontainerEl.createSpan({ text: \"This does not delete the plugin, this should be done from the  Community Plugins tab in Settings.\" });\n\n\t\tnew Setting(containerEl)\n\t\t\t.addButton((cb: ButtonComponent)=>{\n\t\t\t\tcb.setButtonText(\"Add Beta plugin\")\n\t\t\t\tcb.onClick(async ()=>{\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tthis.plugin.app.setting.close();\n\t\t\t\t\tawait this.plugin.betaPlugins.displayAddNewPluginModal(true, false);\n\t\t\t\t})\n\t\t\t});\n\n\t\tconst pluginSubListFrozenVersionNames\n\t\t\t= new Set(this.plugin.settings.pluginSubListFrozenVersion.map(x => x.repo));\n\t\tfor (const bp of this.plugin.settings.pluginList) {\n\t\t\tif (pluginSubListFrozenVersionNames.has(bp)) {\n\t\t\t\tcontinue;\n\t\t\t}\n\t\t\tnew Setting(containerEl)\n\t\t\t\t.setName(bp)\n\t\t\t\t.addButton((btn: ButtonComponent) => {\n\t\t\t\t\tbtn.setIcon(\"cross\");\n\t\t\t\t\tbtn.setTooltip(\"Delete this beta plugin\");\n\t\t\t\t\tbtn.onClick(async () => {\n\t\t\t\t\t\t// await this.plugin.betaPlugins.deletePlugin(bp);\n\t\t\t\t\t\tif (btn.buttonEl.textContent === \"\")\n\t\t\t\t\t\t\tbtn.setButtonText(\"Click once more to confirm removal\");\n\t\t\t\t\t\telse {\n\t\t\t\t\t\t\tbtn.buttonEl.parentElement.parentElement.remove();\n\t\t\t\t\t\t\tawait this.plugin.betaPlugins.deletePlugin(bp)\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\t\t\t\t})\n\t\t}\n\n\t\tnew Setting(containerEl)\n\t\t\t.addButton((cb: ButtonComponent)=>{\n\t\t\t\tcb.setButtonText(\"Add Beta plugin with frozen version\")\n\t\t\t\tcb.onClick(async ()=>{\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tthis.plugin.app.setting.close();\n\t\t\t\t\tawait this.plugin.betaPlugins.displayAddNewPluginModal(true, true);\n\t\t\t\t})\n\t\t\t});\n\t\tfor (const bp of this.plugin.settings.pluginSubListFrozenVersion) {\n\t\t\tnew Setting(containerEl)\n\t\t\t\t.setName(`${bp.repo} (version ${bp.version})`)\n\t\t\t\t.addButton((btn: ButtonComponent) => {\n\t\t\t\t\tbtn.setIcon(\"cross\");\n\t\t\t\t\tbtn.setTooltip(\"Delete this beta plugin\");\n\t\t\t\t\tbtn.onClick(async () => {\n\t\t\t\t\t\t// await this.plugin.betaPlugins.deletePlugin(bp);\n\t\t\t\t\t\tif (btn.buttonEl.textContent === \"\")\n\t\t\t\t\t\t\tbtn.setButtonText(\"Click once more to confirm removal\");\n\t\t\t\t\t\telse {\n\t\t\t\t\t\t\tbtn.buttonEl.parentElement.parentElement.remove();\n\t\t\t\t\t\t\tawait this.plugin.betaPlugins.deletePlugin(bp.repo);\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\t\t\t\t})\n\t\t}\n\n\t\tcontainerEl.createEl(\"hr\");\n\t\tcontainerEl.createEl(\"h2\", { text: \"Beta Themes List\" });\n\n\t\tnew Setting(containerEl)\n\t\t\t.addButton((cb: ButtonComponent)=>{\n\t\t\t\tcb.setButtonText(\"Add Beta Theme\")\n\t\t\t\tcb.onClick(async ()=>{\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tthis.plugin.app.setting.close();\n\t\t\t\t\t(new AddNewTheme(this.plugin)).open();\n\t\t\t\t})\n\t\t\t});\t\t\n\n\n\t\tfor (const bp of this.plugin.settings.themesList) {\n\t\t\tnew Setting(containerEl)\n\t\t\t\t.setName(bp.repo)\n\t\t\t\t.addButton((btn: ButtonComponent) => {\n\t\t\t\t\tbtn.setIcon(\"cross\");\n\t\t\t\t\tbtn.setTooltip(\"Delete this beta theme\");\n\t\t\t\t\tbtn.onClick(async () => {\n\t\t\t\t\t\tif (btn.buttonEl.textContent === \"\")\n\t\t\t\t\t\t\tbtn.setButtonText(\"Click once more to confirm removal\");\n\t\t\t\t\t\telse {\n\t\t\t\t\t\t\tbtn.buttonEl.parentElement.parentElement.remove();\n\t\t\t\t\t\t\tawait themeDelete(this.plugin, bp.repo);\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\t\t\t\t})\n\t\t}\n\n\t\tcontainerEl.createEl(\"hr\");\n\t\tcontainerEl.createEl(\"h2\", { text: \"Monitoring\" });\n\n\t\tnew Setting(containerEl)\n\t\t\t.setName('Enable Notifications')\n\t\t\t.setDesc('BRAT will provide popup notifications for its various activities. Turn this off means  no notifications from BRAT.')\n\t\t\t.addToggle((cb: ToggleComponent) => {\n\t\t\t\tcb.setValue(this.plugin.settings.notificationsEnabled);\n\t\t\t\tcb.onChange(async (value: boolean) => {\n\t\t\t\t\tthis.plugin.settings.notificationsEnabled = value;\n\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t});\n\t\t\t})\n\n\t\tnew Setting(containerEl)\n\t\t\t.setName('Enable Logging')\n\t\t\t.setDesc('Plugin updates will be logged to a file in the log file.')\n\t\t\t.addToggle((cb: ToggleComponent) => {\n\t\t\t\tcb.setValue(this.plugin.settings.loggingEnabled);\n\t\t\t\tcb.onChange(async (value: boolean) => {\n\t\t\t\t\tthis.plugin.settings.loggingEnabled = value;\n\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t});\n\t\t\t})\n\n\t\tnew Setting(this.containerEl)\n            .setName(\"BRAT Log File Location\")\n            .setDesc(\"Logs will be saved to this file. Don't add .md to the file name.\")\n            .addSearch((cb) => {\n                cb.setPlaceholder(\"Example: BRAT-log\")\n                    .setValue(this.plugin.settings.loggingPath)\n                    .onChange(async (new_folder) => {\n                        this.plugin.settings.loggingPath = new_folder;\n\t\t\t\t\t\tawait this.plugin.saveSettings();\n                    });\n            });\t\t\n\n\t\tnew Setting(containerEl)\n\t\t\t.setName('Enable Verbose Logging')\n\t\t\t.setDesc('Get a lot  more information in  the log.')\n\t\t\t.addToggle((cb: ToggleComponent) => {\n\t\t\t\tcb.setValue(this.plugin.settings.loggingVerboseEnabled);\n\t\t\t\tcb.onChange(async (value: boolean) => {\n\t\t\t\t\tthis.plugin.settings.loggingVerboseEnabled = value;\n\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t});\n\t\t\t})\n\n\n\t\tnew Setting(containerEl)\n\t\t\t.setName('Debugging Mode')\n\t\t\t.setDesc('Atomic Bomb level console logging. Can be used for troubleshoting and development.')\n\t\t\t.addToggle((cb: ToggleComponent) => {\n\t\t\t\tcb.setValue(this.plugin.settings.debuggingMode);\n\t\t\t\tcb.onChange(async (value: boolean) => {\n\t\t\t\t\tthis.plugin.settings.debuggingMode = value;\n\t\t\t\t\tawait this.plugin.saveSettings();\n\t\t\t\t});\n\t\t\t})\t\t\t\n\t\n\t}\n}\n", "import { normalizePath, Notice } from \"obsidian\";\nimport ThePlugin from \"../main\";\nimport { addBetaThemeToList, updateBetaThemeLastUpdateChecksum } from \"../ui/settings\";\nimport { checksumForString, grabChecksumOfThemeCssFile, grabCommmunityThemeCssFile, grabCommmunityThemeManifestFile } from \"./githubUtils\";\nimport { ToastMessage } from \"../utils/notifications\";\nimport { isConnectedToInternet } from \"../utils/internetconnection\";\n\n\n/**\n * Installs or updates a theme\n *\n * @param   {ThePlugin}     plugin               ThePlugin\n * @param   {string}        cssGithubRepository  The repository with the theme\n * @param   {boolean}       newInstall           true = New theme install, false update the theme\n *\n * @return  {Promise<boolean>}                   true for succcess\n */\nexport const themeSave = async (plugin: ThePlugin, cssGithubRepository: string, newInstall: boolean): Promise<boolean> => {\n    let themeCSS = await grabCommmunityThemeCssFile(cssGithubRepository, true, plugin.settings.debuggingMode); //test for themes-beta.css\n    if(!themeCSS) themeCSS = await grabCommmunityThemeCssFile(cssGithubRepository, false, plugin.settings.debuggingMode); // grabe themes.css if no beta\n\n    if(!themeCSS) {\n        ToastMessage(plugin,\"There is no theme.css or theme-beta.css file in the root path of this repository, so there is no theme to install.\")\n        return false;\n    }\n\n    const themeManifest = await grabCommmunityThemeManifestFile(cssGithubRepository, plugin.settings.debuggingMode);\n    if(!themeManifest) {\n        ToastMessage(plugin,\"There is no manifest.json file in the root path of this repository, so theme cannot be installed.\")\n        return false;\n    }\n\n    const manifestInfo = await JSON.parse(themeManifest);\n\n    const themeTargetFolderPath = normalizePath(themesRootPath(plugin) + manifestInfo.name);\n\n    const adapter = plugin.app.vault.adapter;\n    if (await adapter.exists(themeTargetFolderPath) === false) await adapter.mkdir(themeTargetFolderPath);\n\n    await adapter.write( normalizePath(themeTargetFolderPath + \"/theme.css\"), themeCSS);\n    await adapter.write( normalizePath(themeTargetFolderPath + \"/manifest.json\"), themeManifest);\n\n    updateBetaThemeLastUpdateChecksum(plugin, cssGithubRepository, checksumForString(themeCSS))\n\n    let msg = ``;\n    \n    if(newInstall) {\n        await addBetaThemeToList(plugin, cssGithubRepository, themeCSS);\n        msg = `${manifestInfo.name} theme installed from ${cssGithubRepository}. `;\n        setTimeout(() => {\n            // @ts-ignore            \n            plugin.app.customCss.setTheme(manifestInfo.name);\n        }, 500);    \n    } else {\n        msg = `${manifestInfo.name} theme updated from ${cssGithubRepository}.`;\n    }\n\n    plugin.log(msg + `[Theme Info](https://github.com/${cssGithubRepository})`, false);\n    ToastMessage(plugin,`${msg}`,20, async ():Promise<void>=>{ window.open(`https://github.com/${cssGithubRepository}`)});\n    return true;\n}\n\n/**\n * Checks  if there  are theme updates based on the commit date of the obsidian.css file on github in comparison to what is stored in the BRAT theme list\n *\n * @param   {ThePlugin}      plugin    ThePlugin\n * @param   {boolean<void>}  showInfo  provide  notices during the update proces\n *\n * @return  {Promise<void>}            \n */\nexport const themesCheckAndUpdates = async (plugin: ThePlugin, showInfo: boolean): Promise<void> => {\n    if(await isConnectedToInternet()===false) { \n        console.log(\"BRAT: No internet detected.\") \n        return;\n    }\n    let newNotice: Notice;\n    const msg1 = `Checking for beta theme updates STARTED`;\n    plugin.log(msg1, true);\n    if (showInfo && plugin.settings.notificationsEnabled) newNotice = new Notice(`BRAT\\n${msg1}`, 30000);\n    for(const t of plugin.settings.themesList) {\n        // first test to see if theme-beta.css exists\n        let lastUpdateOnline = await grabChecksumOfThemeCssFile(t.repo, true, plugin.settings.debuggingMode);\n        // if theme-beta.css does NOT exist, try to get theme.css\n        if(lastUpdateOnline===\"0\") lastUpdateOnline = await grabChecksumOfThemeCssFile(t.repo, false, plugin.settings.debuggingMode);\n        if(lastUpdateOnline!==t.lastUpdate) \n            await themeSave(plugin,t.repo,false)\n    }\n    const msg2 = `Checking for beta theme updates COMPLETED`;\n    plugin.log(msg2, true);\n    if (showInfo) {\n        if(plugin.settings.notificationsEnabled) newNotice.hide();\n        ToastMessage(plugin, msg2);\n    }\n} \n\n/**\n * Deletes a theme from the BRAT list (Does not physically delete the theme)\n *\n * @param   {ThePlugin}  plugin               ThePlugin\n * @param   {string}     cssGithubRepository  Repository path\n *\n * @return  {void}\n */\nexport const themeDelete = async (plugin: ThePlugin, cssGithubRepository: string): Promise<void> => {\n    plugin.settings.themesList = plugin.settings.themesList.filter((t) => t.repo != cssGithubRepository);\n    plugin.saveSettings();\n    const msg = `Removed ${cssGithubRepository} from BRAT themes list and will no longer be updated. However, the theme files still exist in the vault. To remove them, go into Settings > Appearance and remove the theme.`;\n    plugin.log(msg, true);\n    ToastMessage(plugin, `${msg}`);\n}\n\n\n/**\n * Get the path to the themes folder fo rthis vault\n *\n * @param   {ThePlugin}  plugin  ThPlugin\n *\n * @return  {string}             path to themes folder\n */\nexport const themesRootPath = (plugin: ThePlugin): string => {\n    return normalizePath(plugin.app.vault.configDir + \"/themes\") + \"/\";\n}\n\n", "import { PluginManifest, request } from \"obsidian\";\n\nconst GITHUB_RAW_USERCONTENT_PATH = \"https://raw.githubusercontent.com/\";\n\n/**\n * pulls from github a release file by its version number\n *\n * @param   {string}           repository  path to GitHub repository in format USERNAME/repository\n * @param   {string}           version     version of release to retrive \n * @param   {string<string>}   fileName    name of file to retrieve from release\n *\n * @return  {Promise<string>}              contents of file as string from the repository's release\n */\nexport const grabReleaseFileFromRepository = async (repository: string, version: string, fileName: string, debugLogging = true): Promise<string|null> => {\n    const URL = `https://github.com/${repository}/releases/download/${version}/${fileName}`;\n    try {\n        const download = await request({ url: URL });\n        return ((download === \"Not Found\" || download === `{\"error\":\"Not Found\"}`) ? null : download);\n    } catch (error) {\n        if(debugLogging) console.log(\"error in grabReleaseFileFromRepository\", URL, error)\n        return null;\n    }\n}\n \n/**\n * grabs the manifest.json from the repository. rootManifest - if true grabs manifest.json if false grabs manifest-beta.json\n *\n * @param   {string}                     repositoryPath  path to GitHub repository in format USERNAME/repository\n * @param   {[type]}                     rootManifest    if true grabs manifest.json if false grabs manifest-beta.json\n *\n * @return  {Promise<PluginManifest>}                    returns manifest file for  a plugin\n */\nexport const grabManifestJsonFromRepository = async (repositoryPath: string, rootManifest = true, debugLogging = true): Promise<PluginManifest|null> => {\n    const manifestJsonPath = GITHUB_RAW_USERCONTENT_PATH + repositoryPath +\n        (rootManifest === true ? \"/HEAD/manifest.json\" : \"/HEAD/manifest-beta.json\");\n    try {\n        const response = await request({ url: manifestJsonPath });\n        return (response === \"404: Not Found\" ? null : await JSON.parse(response));\n    } catch (error) {\n        if(error!=\"Error: Request failed, status 404\" && debugLogging)  { //normal error, ignore\n            console.log(`error in grabManifestJsonFromRepository for ${manifestJsonPath}`, error);\n        }\n        return null;\n    }\n}\n\n\nexport const grabCommmunityPluginList = async (debugLogging = true): Promise<JSON|null> => {\n    const pluginListURL = `https://raw.githubusercontent.com/obsidianmd/obsidian-releases/HEAD/community-plugins.json`;\n    try {\n        const response = await request({ url: pluginListURL });\n        return (response === \"404: Not Found\" ? null : await JSON.parse(response));\n    } catch (error) {\n        if(debugLogging) console.log(\"error in grabCommmunityPluginList\", error)\n        return null;\n    }\n}\n\nexport const grabCommmunityThemesList = async (debugLogging = true): Promise<JSON|null> => {\n    const themesURL = `https://raw.githubusercontent.com/obsidianmd/obsidian-releases/HEAD/community-css-themes.json`;\n    try {\n        const response = await request({ url: themesURL });\n        return (response === \"404: Not Found\" ? null : await JSON.parse(response));\n    } catch (error) {\n        if(debugLogging) console.log(\"error in grabCommmunityThemesList\", error)\n        return null;\n    }\n}\n\n\nexport const grabCommmunityThemeCssFile = async (repositoryPath: string, betaVersion = false, debugLogging): Promise<string|null> => {\n    const themesURL = `https://raw.githubusercontent.com/${repositoryPath}/HEAD/theme${betaVersion ? \"-beta\" : \"\"}.css`;\n    try {\n        const response = await request({ url: themesURL });\n        return (response === \"404: Not Found\" ? null : response);\n    } catch (error) {\n        if(debugLogging) console.log(\"error in grabCommmunityThemeCssFile\", error)\n        return null;\n    }\n}\n\nexport const grabCommmunityThemeManifestFile = async (repositoryPath: string, debugLogging = true): Promise<string|null> => {\n    const themesURL = `https://raw.githubusercontent.com/${repositoryPath}/HEAD/manifest.json`;\n    try {\n        const response = await request({ url: themesURL });\n        return (response === \"404: Not Found\" ? null : response);\n    } catch (error) {\n        if(debugLogging) console.log(\"error in grabCommmunityThemeManifestFile\", error)\n        return null;\n    }\n}\n\nconst checksum = (str: string): number => {\n    let sum = 0;\n    for (let i = 0; i < str.length; i++) {\n        sum += str.charCodeAt(i);\n    }\n    return sum;\n}\n\nexport const checksumForString = (str: string): string => {\n    return checksum(str).toString();\n}\n\nexport const grabChecksumOfThemeCssFile = async (repositoryPath: string, betaVersion, debugLogging): Promise<string> =>{\n    const themeCSS = await grabCommmunityThemeCssFile(repositoryPath, betaVersion, debugLogging)\n    return themeCSS ? checksumForString(themeCSS) : \"0\";\n}\n\nexport const grabLastCommitInfoForAFile = async (repositoryPath: string, path: string, debugLogging = true): Promise<string|null> => {\n    const url = `https://api.github.com/repos/${repositoryPath}/commits?path=${path}&page=1&per_page=1`;\n    try {\n        const response = await request({ url: url });\n        return (response === \"404: Not Found\" ? null : JSON.parse(response));\n    } catch (error) {\n        if(debugLogging) console.log(\"error in grabLastCommitInfoForAFile\", error)\n        return null;\n    }\n}\n\nexport const grabLastCommitDateForAFile = async (repositoryPath: string, path: string): Promise<string> => {\n    const test = await grabLastCommitInfoForAFile(repositoryPath, path);\n    //@ts-ignore\n    if(test[0].commit.committer.date){\n        //@ts-ignore\n        return test[0].commit.committer.date\n    }\n    else\n        return \"\";\n}\n\n", "import { checksumForString } from \"../features/githubUtils\";\nimport ThePlugin from \"../main\";\n\nexport interface ThemeInforamtion {\n    repo: string;\n    lastUpdate: string; //checksum of theme file (either theme.css or theme-beta.css)\n}\n\nexport interface PluginFrozenVersion {\n    repo: string;\n    version: string;\n}\n\nexport interface Settings {\n    pluginList: string[];\n    pluginSubListFrozenVersion: PluginFrozenVersion[],\n    themesList: ThemeInforamtion[];\n    updateAtStartup: boolean;\n    updateThemesAtStartup:  boolean;\n    ribbonIconEnabled: boolean;\n    loggingEnabled: boolean;\n    loggingPath: string;\n    loggingVerboseEnabled: boolean;\n    debuggingMode: boolean;\n    notificationsEnabled: boolean;\n}\n\nexport const DEFAULT_SETTINGS: Settings = {\n    pluginList: [],\n    pluginSubListFrozenVersion: [],\n    themesList: [],\n    updateAtStartup: false,\n    updateThemesAtStartup: false,\n    ribbonIconEnabled: true,\n    loggingEnabled: false,\n    loggingPath: \"BRAT-log\",\n    loggingVerboseEnabled: false,\n    debuggingMode: true,\n    notificationsEnabled: true\n}\n\n/**\n * Adds a plugin for beta testing to the data.json file of this  plugin\n *\n * @param   {ThePlugin}      plugin         \n * @param   {string<void>}   repositoryPath  path to the GitHub repository\n * @param   {string}         specifyVersion  if the plugin needs to stay at the frozen version, we need to also record the version\n *\n * @return  {Promise<void>}                  \n */\nexport async function addBetaPluginToList(plugin: ThePlugin, repositoryPath: string, specifyVersion = \"\"): Promise<void> {\n    let save = false;\n    if (!plugin.settings.pluginList.contains(repositoryPath)) {\n        plugin.settings.pluginList.unshift(repositoryPath);\n        save = true;\n    }\n    if (\n        specifyVersion !== \"\" \n        && (plugin.settings.pluginSubListFrozenVersion.filter(x => x.repo === repositoryPath).length === 0)\n    ) {\n        plugin.settings.pluginSubListFrozenVersion.unshift({\n            repo: repositoryPath,\n            version: specifyVersion\n        });\n        save = true;\n    }\n    if (save) {\n        plugin.saveSettings();\n    }\n}\n\n/**\n * Tests if  a  plugin  is in data.json\n *\n * @param   {ThePlugin}         plugin          \n * @param   {string<boolean>}   repositoryPath  path to the GitHub repository\n *\n * @return  {Promise<boolean>}  true if exists      \n */\nexport async function existBetaPluginInList(plugin: ThePlugin, repositoryPath: string): Promise<boolean> {\n    return plugin.settings.pluginList.contains(repositoryPath);\n}\n\n\n/**\n * Adds a theme for beta testing to the data.json file of this  plugin\n *\n * @param   {ThePlugin}      plugin         \n * @param   {string<void>}   repositoryPath  path to the GitHub repository\n * @param   {string<void>}   themeCSS  raw text of the theme. It is checksummed and this is used for tracking if changes occurred to the theme\n *\n * @return  {Promise<void>}                  \n */\n export async function addBetaThemeToList(plugin: ThePlugin, repositoryPath: string, themeCSS: string): Promise<void> {\n     const newTheme: ThemeInforamtion = { \n         repo: repositoryPath, \n         lastUpdate: checksumForString(themeCSS)\n    }\n    plugin.settings.themesList.unshift(newTheme);\n    plugin.saveSettings();\n}\n\n/**\n * Tests if a  theme  is in data.json\n *\n * @param   {ThePlugin}         plugin          \n * @param   {string<boolean>}   repositoryPath  path to the GitHub repository\n *\n * @return  {Promise<boolean>}  true if exists      \n */\nexport async function existBetaThemeinInList(plugin: ThePlugin, repositoryPath: string): Promise<boolean> {\n    const testIfThemExists = plugin.settings.themesList.find(t=> t.repo === repositoryPath);\n    return testIfThemExists ? true : false;\n}\n\n\n/**\n * Update the lastUpate field for the theme\n *\n * @param   {ThePlugin}         plugin          \n * @param   {string<boolean>}   repositoryPath  path to the GitHub repository\n * @param   {string<checksum>}  checksum  checksum of file. In past we used the date of file update, but this proved to not be consisent with the GitHub cache.\n *\n */\n export function updateBetaThemeLastUpdateChecksum(plugin: ThePlugin, repositoryPath: string, checksum: string): void {\n    plugin.settings.themesList.forEach(t=>{\n        if(t.repo === repositoryPath) {\n            t.lastUpdate = checksum;\n            plugin.saveSettings();\n        }\n    });\n}\n\n", "import { Notice, Platform } from \"obsidian\";\nimport ThePlugin from \"../main\";\n\n/**\n * Displays a notice to the user\n *\n * @param   {ThePlugin}  plugin              Plugin object\n * @param   {string}     msg                 Text to display to the user\n * @param   {number}     timeoutInSeconds    Number of seconds to show the Toast message\n * @param   {null}       contextMenuCallback function to call if right mouse clicked\n * @return  {void}                         \n */\nexport function ToastMessage(plugin: ThePlugin, msg: string, timeoutInSeconds = 10, contextMenuCallback?:()=>void): void {\n    if(plugin.settings.notificationsEnabled===false) return;\n    const additionalInfo = contextMenuCallback  ? \n                            (Platform.isDesktop ? \"(click=dismiss, right-click=Info)\" : \"(click=dismiss)\") : \"\";\n    const newNotice: Notice = new Notice(`BRAT\\n${msg}\\n${additionalInfo}`, timeoutInSeconds*1000);\n    //@ts-ignore\n    if(contextMenuCallback) newNotice.noticeEl.oncontextmenu = async () => { contextMenuCallback() };    \n}", "\n/**\n * Tests if there is an internet connection\n * @returns true if connected, false if no internet\n */\nexport async function isConnectedToInternet(): Promise<boolean> {\n    try {\n        const online = await fetch(\"https://obsidian.md/?\" + Math.random());\n        return online.status >= 200 && online.status < 300;\n    } catch(err) {\n        return false;\n    }\n}", "import { Modal, Setting } from 'obsidian';\nimport { themeSave } from '../features/themes';\nimport ThePlugin from '../main';\nimport { ToastMessage } from '../utils/notifications';\nimport {  existBetaThemeinInList } from './settings';\nimport { promotionalLinks } from './Promotional';\n\n/**\n * Add a beta theme to the list of plugins being tracked and updated\n */\nexport default class AddNewTheme extends Modal {\n    plugin: ThePlugin;\n    address: string;\n    openSettingsTabAfterwards: boolean;\n\n    constructor(plugin: ThePlugin, openSettingsTabAfterwards = false) {\n        super(plugin.app);\n        this.plugin = plugin;\n        this.address = \"\";\n        this.openSettingsTabAfterwards = openSettingsTabAfterwards;\n    }\n\n    async submitForm(): Promise<void> {\n        if (this.address === \"\") return;\n        const scrubbedAddress = this.address.replace(\"https://github.com/\", \"\");\n        if (await existBetaThemeinInList(this.plugin, scrubbedAddress)) {\n            ToastMessage(this.plugin, `This plugin is already in the list for beta testing`, 10);\n            return;\n        }\n        \n        if(await themeSave(this.plugin, scrubbedAddress, true)) {\n            this.close();    \n        }\n    }\n\n    onOpen(): void {\n        this.contentEl.createEl('h4', { text: \"Github repository for beta theme:\" });\n        this.contentEl.createEl('form', {}, (formEl) => {\n            formEl.addClass(\"brat-modal\");\n            new Setting(formEl)\n                .addText((textEl) => {\n                    textEl.setPlaceholder('Repository (example: https://github.com/GitubUserName/repository-name');\n                    textEl.onChange((value) => {\n                        this.address = value.trim();\n                    });\n                    textEl.inputEl.addEventListener('keydown', async (e: KeyboardEvent) => {\n                        if (e.key === 'Enter' && this.address !== ' ') {\n                            e.preventDefault();\n                            await this.submitForm();\n                        }\n                    });\n                    textEl.inputEl.style.width = \"100%\";\n                    window.setTimeout(() => {\n                        const title = document.querySelector(\".setting-item-info\");\n                        if (title) title.remove();\n                        textEl.inputEl.focus()\n                    }, 10);\n                });\n\n            formEl.createDiv('modal-button-container', (buttonContainerEl) => {\n                buttonContainerEl\n                    .createEl('button', { attr: { type: 'button' }, text: 'Never mind' })\n                    .addEventListener('click', () => this.close());\n                buttonContainerEl.createEl('button', {\n                    attr: { type: 'submit' },\n                    cls: 'mod-cta',\n                    text: 'Add Theme',\n                });\n            });\n\n            const newDiv = formEl.createDiv();\n            newDiv.style.borderTop = \"1px solid #ccc\";\n            newDiv.style.marginTop = \"30px\";\n            const byTfThacker = newDiv.createSpan();\n            byTfThacker.innerHTML = \"BRAT by <a href='https://bit.ly/o42-twitter'>TFTHacker</a>\";\n            byTfThacker.style.fontStyle = \"italic\";\n            newDiv.appendChild(byTfThacker);\n            promotionalLinks(newDiv, false);\n\n            window.setTimeout(() => {\n                const title = formEl.querySelectorAll(\".brat-modal .setting-item-info\");\n                title.forEach((titleEl) => {\n                    titleEl.remove();\n                })\n            }, 50)\n\n            // invoked when button is clicked. \n            formEl.addEventListener('submit', async (e: Event) => {\n                e.preventDefault();\n                if (this.address !== '') await this.submitForm();\n            });\n        });\n    }\n\n    async onClose(): Promise<void> {\n        if (this.openSettingsTabAfterwards) {\n            await (this.plugin as any).app.setting.open();\n            await (this.plugin as any).app.setting.openTabById(\"obsidian42-brat\");\n        }\n\n    }\n}", "\n\nexport const promotionalLinks = (containerEl: HTMLElement, settingsTab = true) : HTMLElement => {\n\n    const linkHeight = settingsTab ? 40 : 30;\n\n    const linksDiv = containerEl.createEl(\"div\");\n    linksDiv.style.float = \"right\";\n    \n    if(settingsTab===false) {\n        linksDiv.style.padding = \"10px\";\n        linksDiv.style.paddingLeft = \"15px\";\n        linksDiv.style.paddingRight = \"15px\";\n    } else {\n        linksDiv.style.padding = \"15px\";\n        linksDiv.style.paddingLeft = \"15px\";\n        linksDiv.style.paddingRight = \"15px\";\n        linksDiv.style.marginLeft = \"15px\";    \n    }\n\n    const coffeeSpan = linksDiv.createSpan(\"coffee\");\n    coffeeSpan.addClass(\"ex-coffee-span\");\n    const coffeeLink = coffeeSpan.createEl(\"a\", { href: \"https://bit.ly/o42-kofi\" });\n    // const coffeeLogo = settingsTab ? \"https://cdn.ko-fi.com/cdn/kofi3.png?v=3\" : \"https://uploads-ssl.webflow.com/5c14e387dab576fe667689cf/61e1116779fc0a9bd5bdbcc7_Frame%206.png\"\n    const coffeeImg = coffeeLink.createEl(\"img\", { attr: { src: \"https://cdn.ko-fi.com/cdn/kofi3.png?v=3\" } });\n    coffeeImg.height = linkHeight;\n    \n    const twitterSpan = linksDiv.createSpan(\"coffee\");\n    twitterSpan.addClass(\"ex-twitter-span\");\n    twitterSpan.style.paddingLeft = \"10px\";\n    const twitterLink = twitterSpan.createEl(\"a\", { href: \"https://bit.ly/o42-twitter\" });\n    const twitterImg = twitterLink.createEl(\"img\", { attr: { src: \"https://cdn.cdnlogo.com/logos/t/96/twitter-icon.svg\" } });\n    twitterImg.height = linkHeight;\n\n    const mediumSpan = linksDiv.createSpan(\"coffee\");\n    mediumSpan.addClass(\"ex-twitter-span\");\n    mediumSpan.style.paddingLeft = \"10px\";\n    mediumSpan.style.right = \"0\";\n    const mediumLink = mediumSpan.createEl(\"a\", { href: \"https://bit.ly/o42-medium\" });\n    const mediumImg = mediumLink.createEl(\"img\", { attr: { src: \"https://miro.medium.com/v2/resize:fill:176:176/1*sHhtYhaCe2Uc3IU0IgKwIQ.png\" } });\n    mediumImg.height = linkHeight;\n\n    return linksDiv;\n}\n\n", "import { Modal, Setting } from 'obsidian';\nimport BetaPlugins from '../features/BetaPlugins';\nimport ThePlugin from '../main';\nimport { ToastMessage } from '../utils/notifications';\nimport { promotionalLinks } from './Promotional';\nimport { existBetaPluginInList } from './settings';\n\n/**\n * Add a beta plugin to the list of plugins being tracked and updated\n */\nexport default class AddNewPluginModal extends Modal {\n    plugin: ThePlugin;\n    betaPlugins: BetaPlugins;\n    address: string;\n    openSettingsTabAfterwards: boolean;\n    readonly useFrozenVersion: boolean;\n    version: string;\n\n    constructor(plugin: ThePlugin, betaPlugins: BetaPlugins, openSettingsTabAfterwards = false, useFrozenVersion = false) {\n        super(plugin.app);\n        this.plugin = plugin;\n        this.betaPlugins = betaPlugins;\n        this.address = \"\";\n        this.openSettingsTabAfterwards = openSettingsTabAfterwards;\n        this.useFrozenVersion = useFrozenVersion;\n        this.version = \"\";\n    }\n\n    async submitForm(): Promise<void> {\n        if (this.address === \"\") return;\n        const scrubbedAddress = this.address.replace(\"https://github.com/\",\"\");\n        if (await existBetaPluginInList(this.plugin, scrubbedAddress)) {\n            ToastMessage(this.plugin, `This plugin is already in the list for beta testing`, 10);\n            return;\n        }\n        const result = await this.betaPlugins.addPlugin(scrubbedAddress, false, false, false, this.version);\n        if (result) {\n            this.close();\n        }\n    }\n\n    onOpen(): void {\n        this.contentEl.createEl('h4', { text: \"Github repository for beta plugin:\" });\n        this.contentEl.createEl('form', {}, (formEl) => {\n            formEl.addClass(\"brat-modal\")\n            new Setting(formEl)\n                .addText((textEl) => {\n                    textEl.setPlaceholder('Repository (example: https://github.com/GitubUserName/repository-name)');\n                    textEl.onChange((value) => {\n                        this.address = value.trim();\n                    });\n                    textEl.inputEl.addEventListener('keydown', async (e: KeyboardEvent) => {\n                        if (e.key === 'Enter' && this.address !== ' ') {\n                            if (\n                                (this.useFrozenVersion && this.version !== \"\") \n                                || (!this.useFrozenVersion)\n                            ) {\n                                e.preventDefault();\n                                await this.submitForm();\n                            }\n                        }\n                    });\n                    textEl.inputEl.style.width = \"100%\";\n                });\n\n            if (this.useFrozenVersion) {\n                new Setting(formEl)\n                    .addText((textEl) => {\n                        textEl.setPlaceholder('Specify the release version tag (example: 1.0.0)');\n                        textEl.onChange((value) => {\n                            this.version = value.trim();\n                        });\n                        textEl.inputEl.style.width = \"100%\";\n                    });\n            }\n\n            formEl.createDiv('modal-button-container', (buttonContainerEl) => {\n                buttonContainerEl\n                    .createEl('button', { attr: { type: 'button' }, text: 'Never mind' })\n                    .addEventListener('click', () => this.close());\n                buttonContainerEl.createEl('button', {\n                    attr: { type: 'submit' },\n                    cls: 'mod-cta',\n                    text: 'Add Plugin',\n                });\n            });\n\n            const newDiv = formEl.createDiv();\n            newDiv.style.borderTop = \"1px solid #ccc\";\n            newDiv.style.marginTop = \"30px\";\n            const byTfThacker = newDiv.createSpan();\n            byTfThacker.innerHTML = \"BRAT by <a href='https://bit.ly/o42-twitter'>TFTHacker</a>\";\n            byTfThacker.style.fontStyle = \"italic\";\n            newDiv.appendChild(byTfThacker);\n            promotionalLinks(newDiv, false);\n\n            window.setTimeout(() => {\n                const title = formEl.querySelectorAll(\".brat-modal .setting-item-info\");\n                title.forEach((titleEl) => {\n                    titleEl.remove();\n                })\n            }, 50)\n\n\n            // invoked when button is clicked. \n            formEl.addEventListener('submit', async (e: Event) => {\n                e.preventDefault();\n                if (this.address !== '') {\n                    if (\n                        (this.useFrozenVersion && this.version !== \"\") \n                        || (!this.useFrozenVersion)\n                    ) {\n                        await this.submitForm();\n                    }\n                }\n            });\n        });\n    }\n    \n    async onClose(): Promise<void> {\n        if(this.openSettingsTabAfterwards) {\n            await (this.plugin as any).app.setting.open();\n            await (this.plugin as any).app.setting.openTabById(\"obsidian42-brat\");\n        }\n\n    }\n}", "import ThePlugin from \"../main\";\nimport AddNewPluginModal from \"../ui/AddNewPluginModal\";\nimport { grabManifestJsonFromRepository, grabReleaseFileFromRepository } from \"./githubUtils\";\nimport { normalizePath, PluginManifest, Notice, requireApiVersion, apiVersion } from \"obsidian\";\nimport { addBetaPluginToList } from \"../ui/settings\";\nimport { ToastMessage } from \"../utils/notifications\";\nimport { isConnectedToInternet } from \"../utils/internetconnection\";\n\n/**\n * all the files needed for a plugin based on the release files are hre\n */\ninterface ReleaseFiles {\n    mainJs:     string | null;\n    manifest:   string | null;\n    styles:     string | null;\n}\n\n/**\n * Primary handler for adding, updating, deleting beta plugins tracked by this plugin\n */\nexport default class BetaPlugins {\n    plugin: ThePlugin;\n\n    constructor(plugin: ThePlugin) {\n        this.plugin = plugin;\n    }\n\n    /**\n     * opens the AddNewPluginModal to get info for  a new beta plugin\n     * @param   {boolean}   openSettingsTabAfterwards will open settings screen afterwards. Used when this command is called from settings tab\n     * @param   {boolean}   useFrozenVersion          install the plugin using frozen version.\n     * @return  {<Promise><void>}\n     */\n    async displayAddNewPluginModal(openSettingsTabAfterwards = false, useFrozenVersion = false): Promise<void> {\n        const newPlugin = new AddNewPluginModal(this.plugin, this, openSettingsTabAfterwards, useFrozenVersion);\n        newPlugin.open();\n    }\n\n    /**\n     * Validates that a GitHub repository is plugin\n     *\n     * @param   {string}                     repositoryPath   GithubUser/RepositoryName (example: TfThacker/obsidian42-brat)\n     * @param   {[type]}                     getBetaManifest  test the beta version of the manifest, not at the root\n     * @param   {[type]}                     false            [false description]\n     * @param   {[type]}                     reportIssues      will display notices as it finds issues\n     *\n     * @return  {Promise<PluginManifest>}                     the manifest file if found, or null if its incomplete\n     */\n    async validateRepository(repositoryPath: string, getBetaManifest = false, reportIssues = false): Promise<PluginManifest|null> {\n        const noticeTimeout = 15;\n        const manifestJson = await grabManifestJsonFromRepository(repositoryPath, !getBetaManifest, this.plugin.settings.debuggingMode);\n        if (!manifestJson) { // this is a plugin with a manifest json, try to see if there is a beta version\n            if (reportIssues) ToastMessage(this.plugin, `${repositoryPath}\\nThis does not seem to be an obsidian plugin, as there is no manifest.json file.`, noticeTimeout);\n            return null;\n        }\n        // Test that the mainfest has some key elements, like ID and version\n        if (!(\"id\" in manifestJson)) { // this is a plugin with a manifest json, try to see if there is a beta version\n            if (reportIssues) ToastMessage(this.plugin,`${repositoryPath}\\nThe plugin id attribute for the release is missing from the manifest file`, noticeTimeout);\n            return null;\n        }\n        if (!(\"version\" in manifestJson)) { // this is a plugin with a manifest json, try to see if there is a beta version\n            if (reportIssues) ToastMessage(this.plugin,`${repositoryPath}\\nThe version attribute for the release is missing from the manifest file`, noticeTimeout);\n            return null;\n        }\n        return manifestJson;\n    }\n\n    /**\n     * Gets all the release files based on the version number in the manifest\n     *\n     * @param   {string}                        repositoryPath  path to the GitHub repository\n     * @param   {PluginManifest<ReleaseFiles>}  manifest        manifest file\n     * @param   {boolean}                       getManifest     grab the remote manifest file\n     * @param   {string}                        specifyVersion  grab the specified version if set\n     *\n     * @return  {Promise<ReleaseFiles>}                         all relase files as strings based on the ReleaseFiles interaface\n     */\n    async getAllReleaseFiles(repositoryPath: string, manifest: PluginManifest, getManifest: boolean, specifyVersion = \"\"): Promise<ReleaseFiles> {\n        const version = specifyVersion === \"\" ? manifest.version : specifyVersion;\n\n        // if we have version specified, we always want to get the remote manifest file.\n        const reallyGetManifestOrNot = getManifest || (specifyVersion !== \"\");\n\n        return {\n            mainJs: await grabReleaseFileFromRepository(repositoryPath, version, \"main.js\", this.plugin.settings.debuggingMode),\n            manifest: reallyGetManifestOrNot ? await grabReleaseFileFromRepository(repositoryPath, version, \"manifest.json\", this.plugin.settings.debuggingMode) : \"\",\n            styles: await grabReleaseFileFromRepository(repositoryPath, version, \"styles.css\", this.plugin.settings.debuggingMode)\n        }\n    }\n\n    /**\n     * Writes the plugin release files to the local obsidian .plugins folder\n     *\n     * @param   {string}              betaPluginID  the id of the plugin (not the repository path)\n     * @param   {ReleaseFiles<void>}  relFiles      release file as strings, based on the ReleaseFiles interface\n     *\n     * @return  {Promise<void>}                     \n     */\n    async writeReleaseFilesToPluginFolder(betaPluginID: string, relFiles: ReleaseFiles): Promise<void> {\n        const pluginTargetFolderPath = normalizePath(this.plugin.app.vault.configDir + \"/plugins/\" + betaPluginID) + \"/\";\n        const adapter = this.plugin.app.vault.adapter;\n        if (await adapter.exists(pluginTargetFolderPath) === false ||\n            !(await adapter.exists(pluginTargetFolderPath + \"manifest.json\"))) {\n            // if plugin folder doesnt exist or manifest.json doesn't exist, create it and save the plugin files\n            await adapter.mkdir(pluginTargetFolderPath);\n        }\n        await adapter.write(pluginTargetFolderPath + \"main.js\", relFiles.mainJs);\n        await adapter.write(pluginTargetFolderPath + \"manifest.json\", relFiles.manifest);\n        if (relFiles.styles) await adapter.write(pluginTargetFolderPath + \"styles.css\", relFiles.styles);\n    }\n\n    /**\n     * Primary function for adding a new beta plugin to Obsidian. \n     * Also this function is used for updating existing plugins.\n     *\n     * @param   {string}              repositoryPath     path to GitHub repository formated as USERNAME/repository\n     * @param   {boolean}             updatePluginFiles  true if this is just an update not an install\n     * @param   {boolean}             seeIfUpdatedOnly   if true, and updatePluginFiles true, will just check for updates, but not do the update. will report to user that there is a new plugin\n     * @param   {boolean}             reportIfNotUpdted  if true, report if an update has not succed\n     * @param   {string}              specifyVersion     if not empty, need to install a specified version instead of the value in manifest{-beta}.json\n     *\n     * @return  {Promise<boolean>}                       true if succeeds\n     */\n    async addPlugin(repositoryPath: string, updatePluginFiles = false, seeIfUpdatedOnly = false, reportIfNotUpdted = false, specifyVersion = \"\"): Promise<boolean> {\n        const noticeTimeout = 10;\n        let primaryManifest = await this.validateRepository(repositoryPath, true, false); // attempt to get manifest-beta.json\n        const usingBetaManifest: boolean = primaryManifest ? true : false;\n        if (usingBetaManifest === false)\n            primaryManifest = await this.validateRepository(repositoryPath, false, true); // attempt to get manifest.json\n\n        if (primaryManifest === null) {\n            const msg = `${repositoryPath}\\nA manifest.json or manifest-beta.json file does not exist in the root directory of the repository. This plugin cannot be installed.`;\n            this.plugin.log(msg, true);\n            ToastMessage(this.plugin, `${msg}`, noticeTimeout);\n            return false;\n        }\n\n        if (!primaryManifest.hasOwnProperty('version')) {\n            const msg = `${repositoryPath}\\nThe manifest${usingBetaManifest ? \"-beta\" : \"\"}.json file in the root directory of the repository does not have a version number in the file. This plugin cannot be installed.`;\n            this.plugin.log(msg, true);\n            ToastMessage(this.plugin, `${msg}`, noticeTimeout);\n            return false;\n        }\n\n        // Check manifest minAppVersion and current version of Obisidan, don't load plugin if not compatible\n        if(primaryManifest.hasOwnProperty('minAppVersion')) { \n            if( !requireApiVersion(primaryManifest.minAppVersion) ) {\n                const msg = `Plugin: ${repositoryPath}\\n\\n`+\n                            `The manifest${usingBetaManifest ? \"-beta\" : \"\"}.json for this plugin indicates that the Obsidian ` +\n                            `version of the app needs to be ${primaryManifest.minAppVersion}, ` +\n                            `but this installation of Obsidian is ${apiVersion}. \\n\\nYou will need to update your ` +\n                            `Obsidian to use this plugin or contact the plugin developer for more information.`;\n                this.plugin.log(msg, true);\n                ToastMessage(this.plugin, `${msg}`, 30);\n                return false;    \n            }\n        }\n\n        const getRelease = async () => { \n            \n            const rFiles = await this.getAllReleaseFiles(repositoryPath, primaryManifest as PluginManifest, usingBetaManifest, specifyVersion);\n            if (usingBetaManifest || rFiles.manifest === \"\")  //if beta, use that manifest, or if there is no manifest in release, use the primaryManifest\n                rFiles.manifest = JSON.stringify(primaryManifest);\n\n            if (rFiles.mainJs === null) {\n                const msg = `${repositoryPath}\\nThe release is not complete and cannot be download. main.js is missing from the Release`;\n                this.plugin.log(msg, true);\n                ToastMessage(this.plugin, `${msg}`, noticeTimeout);\n                return null;\n            }\n            return rFiles;\n        }\n\n        if (updatePluginFiles === false) {\n            const releaseFiles = await getRelease();\n            if (releaseFiles === null) return false;\n            await this.writeReleaseFilesToPluginFolder(primaryManifest.id, releaseFiles);\n            await addBetaPluginToList(this.plugin, repositoryPath, specifyVersion);\n            //@ts-ignore\n            await this.plugin.app.plugins.loadManifests();\n            const versionText = specifyVersion === \"\" ? \"\" : ` (version: ${specifyVersion})`;\n            const msg = `${repositoryPath}${versionText}\\nThe plugin has been registered with BRAT. You may still need to enable it the Community Plugin List.`;\n            this.plugin.log(msg, true);\n            ToastMessage(this.plugin, msg, noticeTimeout);\n        } else {\n            // test if the plugin needs to be updated\n            // if a specified version is provided, then we shall skip the update\n            const pluginTargetFolderPath = this.plugin.app.vault.configDir + \"/plugins/\" + primaryManifest.id + \"/\";\n            let localManifestContents = \"\";\n            try {\n                localManifestContents = await this.plugin.app.vault.adapter.read(pluginTargetFolderPath + \"manifest.json\")\n            } catch (e) {\n                if (e.errno === -4058 || e.errno === -2) { // file does not exist, try installing the plugin\n                    await this.addPlugin(repositoryPath, false, usingBetaManifest, false, specifyVersion);\n                    return true; // even though failed, return true since install will be attempted\n                }\n                else\n                    console.log(\"BRAT - Local Manifest Load\", primaryManifest.id, JSON.stringify(e, null, 2));\n            }\n\n            if (\n                specifyVersion !== \"\" \n                || this.plugin.settings.pluginSubListFrozenVersion.map(x=>x.repo).includes(repositoryPath)\n            ) {\n                // skip the frozen version plugin\n                ToastMessage(this.plugin, `The version of ${repositoryPath} is frozen, not updating.`, 3);\n                return false;\n            }\n\n            const localManifestJSON = await JSON.parse(localManifestContents);\n            if (localManifestJSON.version !== primaryManifest.version) { //manifest files are not the same, do an update\n                const releaseFiles = await getRelease();\n                if (releaseFiles === null) return false;\n\n                if (seeIfUpdatedOnly) { // dont update, just report it\n                    const msg = `There is an update available for ${primaryManifest.id} from version ${localManifestJSON.version} to ${primaryManifest.version}. `;\n                    this.plugin.log(msg + `[Release Info](https://github.com/${repositoryPath}/releases/tag/${primaryManifest.version})`, false);\n                    ToastMessage(this.plugin, msg, 30, async () => { window.open(`https://github.com/${repositoryPath}/releases/tag/${primaryManifest.version}`)});\n                } else {\n                    await this.writeReleaseFilesToPluginFolder(primaryManifest.id, releaseFiles);\n                    //@ts-ignore\n                    await this.plugin.app.plugins.loadManifests();\n                    //@ts-ignore\n                    if (this.plugin.app.plugins.plugins[primaryManifest.id]?.manifest) await this.reloadPlugin(primaryManifest.id); //reload if enabled\n                    const msg = `${primaryManifest.id}\\nPlugin has been updated from version ${localManifestJSON.version} to ${primaryManifest.version}. `;\n                    this.plugin.log(msg + `[Release Info](https://github.com/${repositoryPath}/releases/tag/${primaryManifest.version})`, false);\n                    ToastMessage(this.plugin, msg, 30, async () => { window.open(`https://github.com/${repositoryPath}/releases/tag/${primaryManifest.version}`) } );\n                }\n            } else\n                if (reportIfNotUpdted) ToastMessage(this.plugin, `No update available for ${repositoryPath}`, 3);\n        }\n        return true;\n    }\n\n    /**\n     * reloads a plugin (assuming it has been enabled by user)\n     * pjeby, Thanks Bro https://github.com/pjeby/hot-reload/blob/master/main.js\n     * \n     * @param   {string<void>}   pluginName  name of plugin\n     *\n     * @return  {Promise<void>}              \n     */\n    async reloadPlugin(pluginName: string): Promise<void> {\n        // @ts-ignore\n        const plugins = this.plugin.app.plugins;\n        try {\n            await plugins.disablePlugin(pluginName);\n            await plugins.enablePlugin(pluginName);\n        } catch (e) { \n            if(this.plugin.settings.debuggingMode)\n                console.log(\"reload plugin\", e) \n        }\n    }\n\n    /**\n     * updates a beta plugin\n     *\n     * @param   {string}   repositoryPath  repository path on GitHub\n     * @param   {boolean}  onlyCheckDontUpdate only looks for update\n     *\n     * @return  {Promise<void>}                  \n     */\n    async updatePlugin(repositoryPath: string, onlyCheckDontUpdate = false, reportIfNotUpdted = false): Promise<boolean> {\n        const result = await this.addPlugin(repositoryPath, true, onlyCheckDontUpdate, reportIfNotUpdted);\n        if (result === false && onlyCheckDontUpdate === false)\n        ToastMessage(this.plugin, `${repositoryPath}\\nUpdate of plugin failed.`)\n        return result;\n    }\n\n    /**\n     * walks through the list of plugins without frozen version and performs an update\n     *\n     * @param   {boolean}           showInfo  should this with a started/completed message - useful when ran from CP\n     * @return  {Promise<void>}              \n     */\n    async checkForUpdatesAndInstallUpdates(showInfo = false, onlyCheckDontUpdate = false): Promise<void> {\n        if(await isConnectedToInternet()===false) { \n            console.log(\"BRAT: No internet detected.\") \n            return;\n        }\n        let newNotice: Notice;\n        const msg1 = `Checking for plugin updates STARTED`;\n        this.plugin.log(msg1, true);\n        if (showInfo && this.plugin.settings.notificationsEnabled) newNotice = new Notice(`BRAT\\n${msg1}`, 30000);\n        const pluginSubListFrozenVersionNames = \n            new Set(this.plugin.settings.pluginSubListFrozenVersion.map(f => f.repo));\n        for (const bp of this.plugin.settings.pluginList) {\n            if (pluginSubListFrozenVersionNames.has(bp)) {\n                continue;\n            }\n            await this.updatePlugin(bp, onlyCheckDontUpdate);\n        }\n        const msg2 = `Checking for plugin updates COMPLETED`;\n        this.plugin.log(msg2, true);\n        if (showInfo) {\n            newNotice.hide();\n            ToastMessage(this.plugin, msg2, 10);\n        }\n    }\n\n    /**\n     * Removes the beta plugin from the list of beta plugins (does not delete them from disk)\n     *\n     * @param   {string<void>}   betaPluginID  repository path\n     *\n     * @return  {Promise<void>}                [return description]\n     */\n    async deletePlugin(repositoryPath: string): Promise<void> {\n        const msg = `Removed ${repositoryPath} from BRAT plugin list`;\n        this.plugin.log(msg, true);\n        this.plugin.settings.pluginList = this.plugin.settings.pluginList.filter((b) => b != repositoryPath);\n        this.plugin.settings.pluginSubListFrozenVersion = \n            this.plugin.settings.pluginSubListFrozenVersion.filter(\n                (b) => b.repo != repositoryPath\n            );\n        this.plugin.saveSettings();\n    }\n\n    /**\n     * Returns a list of plugins that are currently enabled or currently disabled\n     *\n     * @param   {boolean[]}        enabled  true for enabled plugins, false for disabled plutings\n     *\n     * @return  {PluginManifest[]}           manifests  of plugins\n     */\n    getEnabledDisabledPlugins(enabled: boolean): PluginManifest[] {\n        // @ts-ignore\n        const pl = this.plugin.app.plugins;\n        const manifests: PluginManifest[] = Object.values(pl.manifests);\n        // @ts-ignore\n        const enabledPlugins: PluginManifest[] = Object.values(pl.plugins).map(p => p.manifest);\n        return enabled ?\n            manifests.filter(manifest => enabledPlugins.find(pluginName => manifest.id === pluginName.id)) :\n            manifests.filter(manifest => !enabledPlugins.find(pluginName => manifest.id === pluginName.id));\n    }\n}", "import { addIcon } from 'obsidian';\n\nexport function addIcons(): void {\n    addIcon(\n        \"BratIcon\",\n        `<path fill=\"currentColor\" stroke=\"currentColor\"  d=\"M 41.667969 41.667969 C 41.667969 39.367188 39.800781 37.5 37.5 37.5 C 35.199219 37.5 33.332031 39.367188 33.332031 41.667969 C 33.332031 43.96875 35.199219 45.832031 37.5 45.832031 C 39.800781 45.832031 41.667969 43.96875 41.667969 41.667969 Z M 60.417969 58.582031 C 59.460938 58.023438 58.320312 57.867188 57.25 58.148438 C 56.179688 58.429688 55.265625 59.125 54.707031 60.082031 C 53.746094 61.777344 51.949219 62.820312 50 62.820312 C 48.050781 62.820312 46.253906 61.777344 45.292969 60.082031 C 44.734375 59.125 43.820312 58.429688 42.75 58.148438 C 41.679688 57.867188 40.539062 58.023438 39.582031 58.582031 C 37.597656 59.726562 36.910156 62.257812 38.042969 64.25 C 40.5 68.53125 45.0625 71.171875 50 71.171875 C 54.9375 71.171875 59.5 68.53125 61.957031 64.25 C 63.089844 62.257812 62.402344 59.726562 60.417969 58.582031 Z M 62.5 37.5 C 60.199219 37.5 58.332031 39.367188 58.332031 41.667969 C 58.332031 43.96875 60.199219 45.832031 62.5 45.832031 C 64.800781 45.832031 66.667969 43.96875 66.667969 41.667969 C 66.667969 39.367188 64.800781 37.5 62.5 37.5 Z M 50 8.332031 C 26.988281 8.332031 8.332031 26.988281 8.332031 50 C 8.332031 73.011719 26.988281 91.667969 50 91.667969 C 73.011719 91.667969 91.667969 73.011719 91.667969 50 C 91.667969 26.988281 73.011719 8.332031 50 8.332031 Z M 50 83.332031 C 33.988281 83.402344 20.191406 72.078125 17.136719 56.363281 C 14.078125 40.644531 22.628906 24.976562 37.5 19.042969 C 37.457031 19.636719 37.457031 20.238281 37.5 20.832031 C 37.5 27.738281 43.097656 33.332031 50 33.332031 C 52.300781 33.332031 54.167969 31.46875 54.167969 29.167969 C 54.167969 26.867188 52.300781 25 50 25 C 47.699219 25 45.832031 23.132812 45.832031 20.832031 C 45.832031 18.53125 47.699219 16.667969 50 16.667969 C 68.410156 16.667969 83.332031 31.589844 83.332031 50 C 83.332031 68.410156 68.410156 83.332031 50 83.332031 Z M 50 83.332031 \" />`\n    );\n}", "import { moment, TFile, Platform } from \"obsidian\";\nimport { getDailyNoteSettings } from \"obsidian-daily-notes-interface\";\nimport ThePlugin from \"../main\";\n\n/**\n * Logs events to a log file\n *\n * @param   {ThePlugin}  plugin            Plugin object\n * @param   {string}     textToLog         text to be saved to log file\n * @param   {[type]}     verboseLoggingOn  True if should only be logged if verbose logging is enabled\n *\n * @return  {void}                         \n */\nexport function logger(plugin: ThePlugin, textToLog: string, verboseLoggingOn = false): void {\n    if(plugin.settings.debuggingMode) console.log(\"BRAT: \" + textToLog);\n    if (plugin.settings.loggingEnabled) {\n        if (plugin.settings.loggingVerboseEnabled === false && verboseLoggingOn === true) {\n            return;\n        } else {\n            const fileName = plugin.settings.loggingPath + \".md\";\n            const dateOutput = \"[[\" + moment().format(getDailyNoteSettings().format).toString() + \"]] \" +\n                moment().format(\"HH:mm\");\n            const machineName = Platform.isDesktop ? window.require(\"os\").hostname() : \"MOBILE\";\n            let output = dateOutput + \" \" + machineName + \" \" + textToLog.replace(\"\\n\",\" \") + \"\\n\\n\";\n            setTimeout(async () => {\n                if (await plugin.app.vault.adapter.exists(fileName) === true) {\n                    const fileContents = await plugin.app.vault.adapter.read(fileName);\n                    output = output + fileContents;\n                    const file = plugin.app.vault.getAbstractFileByPath(fileName) as TFile;\n                    await plugin.app.vault.modify(file, output);\n                } else\n                    await plugin.app.vault.create(fileName, output);\n            }, 10);\n        }\n    }\n}", "import { FuzzySuggestModal, FuzzyMatch } from 'obsidian';\nimport ThePlugin from '../main';\n\n/**\n * Simple interface for what should be displayed and stored for suggester\n */\nexport interface SuggesterItem {\n    display: string,        // displayed to user\n    info: any               // supplmental info for the callback\n}\n\n/**\n * Generic suggester for quick reuse\n */\nexport class GenericFuzzySuggester extends FuzzySuggestModal<SuggesterItem>{\n    data: SuggesterItem[];\n    callbackFunction: any;\n\n    constructor(plugin: ThePlugin) {\n        super(plugin.app);\n        this.scope.register([\"Shift\"], \"Enter\", evt => this.enterTrigger(evt));\n        this.scope.register([\"Ctrl\"], \"Enter\", evt => this.enterTrigger(evt));\n    }\n\n    setSuggesterData(suggesterData: Array<SuggesterItem>): void { this.data = suggesterData }\n\n    async display(callBack: (item: SuggesterItem, evt: MouseEvent | KeyboardEvent) => void): Promise<any> {\n        this.callbackFunction = callBack;\n        this.open();\n    }\n\n    getItems(): SuggesterItem[] { return this.data }\n\n    getItemText(item: SuggesterItem): string { return item.display }\n\n    onChooseItem(): void { return } // required by TS, but not using\n\n    renderSuggestion(item: FuzzyMatch<SuggesterItem>, el: HTMLElement): void { el.createEl('div', { text: item.item.display }) }\n\n    enterTrigger(evt: KeyboardEvent): void {\n        const selectedText = document.querySelector(\".suggestion-item.is-selected div\").textContent;\n        const item = this.data.find(i => i.display === selectedText);\n        if (item) {\n            this.invokeCallback(item, evt);\n            this.close();\n        }\n    }\n\n    onChooseSuggestion(item: FuzzyMatch<SuggesterItem>, evt: MouseEvent | KeyboardEvent): void { this.invokeCallback(item.item, evt) }\n\n    invokeCallback(item: SuggesterItem, evt: MouseEvent | KeyboardEvent): void { this.callbackFunction(item, evt) }\n}\n", "import ThePlugin from \"../main\";\nimport { GenericFuzzySuggester, SuggesterItem } from \"./GenericFuzzySuggester\";\nimport { grabCommmunityPluginList, grabCommmunityThemesList } from \"../features/githubUtils\";\nimport { themesCheckAndUpdates } from \"../features/themes\";\nimport AddNewTheme from \"./AddNewTheme\";\nimport { ToastMessage } from \"../utils/notifications\";\n\nexport default class PluginCommands {\n    plugin: ThePlugin;\n    bratCommands = [\n        {\n            id: \"BRAT-AddBetaPlugin\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Add a beta plugin for testing\",\n            showInRibbon: true,\n            callback: async () => { await this.plugin.betaPlugins.displayAddNewPluginModal(false, false) }\n        },\n        {\n            id: \"BRAT-AddBetaPluginWithFrozenVersion\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Add a beta plugin with frozen version based on a release tag\",\n            showInRibbon: true,\n            callback: async () => { await this.plugin.betaPlugins.displayAddNewPluginModal(false, true) }\n        },\n        {\n            id: \"BRAT-checkForUpdatesAndUpdate\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Check for updates to all beta plugins and UPDATE\",\n            showInRibbon: true,\n            callback: async () => { await this.plugin.betaPlugins.checkForUpdatesAndInstallUpdates(true, false) }\n        },\n        {\n            id: \"BRAT-checkForUpdatesAndDontUpdate\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Only check for updates to beta plugins, but don't Update\",\n            showInRibbon: true,\n            callback: async () => { await this.plugin.betaPlugins.checkForUpdatesAndInstallUpdates(true, true) }\n        },\n        {\n            id: \"BRAT-updateOnePlugin\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Choose a single plugin version to update\",\n            showInRibbon: true,\n            callback: async () => {\n                const pluginSubListFrozenVersionNames = \n                    new Set(this.plugin.settings.pluginSubListFrozenVersion.map(f => f.repo));\n                const pluginList: SuggesterItem[] = \n                    Object\n                        .values(this.plugin.settings.pluginList)\n                        .filter((f) => !pluginSubListFrozenVersionNames.has(f))\n                        .map((m) => { return { display: m, info: m } });\n                const gfs = new GenericFuzzySuggester(this.plugin);\n                gfs.setSuggesterData(pluginList);\n                await gfs.display(async (results) => {\n                    const msg = `Checking for updates for ${results.info}`;\n                    this.plugin.log(msg,true);\n                    ToastMessage(this.plugin, `\\n${msg}`, 3);\n                    await this.plugin.betaPlugins.updatePlugin(results.info, false, true);\n                });\n            }\n        },\n        {\n            id: \"BRAT-restartPlugin\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Restart a plugin that is already installed\",\n            showInRibbon: true,\n            callback: async () => {\n                // @ts-ignore\n                const pluginList: SuggesterItem[] = Object.values(this.plugin.app.plugins.manifests).map((m) => { return { display: m.id, info: m.id } });\n                const gfs = new GenericFuzzySuggester(this.plugin);\n                gfs.setSuggesterData(pluginList);\n                await gfs.display(async (results) => {\n                    ToastMessage(this.plugin, `${results.info}\\nPlugin reloading .....`, 5);\n                    await this.plugin.betaPlugins.reloadPlugin(results.info);\n                });\n            }\n        },\n        {\n            id: \"BRAT-disablePlugin\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Disable a plugin - toggle it off\",\n            showInRibbon: true,\n            callback: async () => {\n                const pluginList = this.plugin.betaPlugins.getEnabledDisabledPlugins(true).map(manifest => { return { display: `${manifest.name} (${manifest.id})`, info: manifest.id } });\n                const gfs = new GenericFuzzySuggester(this.plugin);\n                gfs.setSuggesterData(pluginList);\n                await gfs.display(async (results) => {\n                    this.plugin.log(`${results.display} plugin disabled`, false);\n                    if(this.plugin.settings.debuggingMode) console.log(results.info)\n                    // @ts-ignore\n                    await this.plugin.app.plugins.disablePluginAndSave(results.info);\n                });\n            }\n        },\n        {\n            id: \"BRAT-enablePlugin\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Enable a plugin - toggle it on\",\n            showInRibbon: true,\n            callback: async () => {\n                const pluginList = this.plugin.betaPlugins.getEnabledDisabledPlugins(false).map(manifest => { return { display: `${manifest.name} (${manifest.id})`, info: manifest.id } });\n                const gfs = new GenericFuzzySuggester(this.plugin);\n                gfs.setSuggesterData(pluginList);\n                await gfs.display(async (results) => {\n                    this.plugin.log(`${results.display} plugin enabled`, false);\n                    // @ts-ignore\n                    await this.plugin.app.plugins.enablePluginAndSave(results.info);\n                });\n            }\n        },\n        {\n            id: \"BRAT-openGitHubZRepository\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Open the GitHub repository for a plugin\",\n            showInRibbon: true,\n            callback: async () => {\n                const communityPlugins = await grabCommmunityPluginList(this.plugin.settings.debuggingMode);\n                const communityPluginList: SuggesterItem[] = Object.values(communityPlugins).map((p) => { return { display: `Plugin: ${p.name}  (${p.repo})`, info: p.repo } });\n                const bratList: SuggesterItem[] = Object.values(this.plugin.settings.pluginList).map((p) => { return { display: \"BRAT: \" + p, info: p } });\n                communityPluginList.forEach(si => bratList.push(si));\n                const gfs = new GenericFuzzySuggester(this.plugin);\n                gfs.setSuggesterData(bratList);\n                await gfs.display(async (results) => {\n                    if (results.info) window.open(`https://github.com/${results.info}`)\n                });\n            }\n        },\n        {\n            id: \"BRAT-openGitHubRepoTheme\",\n            icon: \"BratIcon\",\n            name: \"Themes: Open the GitHub repository for a theme (appearance)\",\n            showInRibbon: true,\n            callback: async () => {\n                const communityTheme = await grabCommmunityThemesList(this.plugin.settings.debuggingMode);\n                const communityThemeList: SuggesterItem[] = Object.values(communityTheme).map((p) => { return { display: `Theme: ${p.name}  (${p.repo})`, info: p.repo } });\n                const gfs = new GenericFuzzySuggester(this.plugin);\n                gfs.setSuggesterData(communityThemeList);\n                await gfs.display(async (results) => {\n                    if (results.info) window.open(`https://github.com/${results.info}`)\n                });\n            }\n        },\n        {\n            id: \"BRAT-opentPluginSettings\",\n            icon: \"BratIcon\",\n            name: \"Plugins: Open Plugin Settings Tab\",\n            showInRibbon: true,\n            callback: async () => {\n                // @ts-ignore\n                const settings = this.plugin.app.setting;\n                // @ts-ignore\n                const listOfPluginSettingsTabs: SuggesterItem[] = Object.values(settings.pluginTabs).map((t) => { return { display: \"Plugin: \" + t.name, info: t.id } });\n                const gfs = new GenericFuzzySuggester(this.plugin);\n                // @ts-ignore\n                const listOfCoreSettingsTabs: SuggesterItem[] = Object.values(settings.settingTabs).map((t) => { return { display: \"Core: \" + t.name, info: t.id } });\n                listOfPluginSettingsTabs.forEach(si => listOfCoreSettingsTabs.push(si));\n                gfs.setSuggesterData(listOfCoreSettingsTabs);\n                await gfs.display(async (results) => {\n                    settings.open();\n                    settings.openTabById(results.info);\n                });\n            }\n        },\n        {\n            id: \"BRAT-GrabBetaTheme\",\n            icon: \"BratIcon\",\n            name: \"Themes: Grab a beta theme for testing from a Github repository\",\n            showInRibbon: true,\n            callback: async () => { (new AddNewTheme(this.plugin)).open() }\n        },\n        {\n            id: \"BRAT-updateBetaThemes\",\n            icon: \"BratIcon\",\n            name: \"Themes: Update beta themes\",\n            showInRibbon: true,\n            callback: async () => await themesCheckAndUpdates(this.plugin, true) \n        },        \n        {\n            id: \"BRAT-allCommands\",\n            icon: \"BratIcon\",\n            name: \"All Commands list\",\n            showInRibbon: false,\n            callback: async () => this.ribbonDisplayCommands()\n        },\n    ]\n\n    async ribbonDisplayCommands(): Promise<void> {\n        const bratCommandList: SuggesterItem[] = [];\n        this.bratCommands.forEach(cmd => { if (cmd.showInRibbon) bratCommandList.push({ display: cmd.name, info: cmd.callback }) });\n        const gfs = new GenericFuzzySuggester(this.plugin);\n        // @ts-ignore\n        const settings = this.plugin.app.setting;\n        // @ts-ignore\n        const listOfCoreSettingsTabs: SuggesterItem[] = Object.values(settings.settingTabs).map((t: any) => {\n            return {\n                display: \"Core: \" + t.name,\n                info: async () => {\n                    settings.open();\n                    settings.openTabById(t.id);\n                }\n            }\n        });\n        // @ts-ignore\n        const listOfPluginSettingsTabs: SuggesterItem[] = Object.values(settings.pluginTabs).map((t: any) => {\n            return {\n                display: \"Plugin: \" + t.name,\n                info: async () => {\n                    settings.open();\n                    settings.openTabById(t.id);\n                }\n            }\n        });\n\n        bratCommandList.push({ display: \"---- Core Plugin Settings ----\", info: async () => { await this.ribbonDisplayCommands() } })\n        listOfCoreSettingsTabs.forEach(si => bratCommandList.push(si));\n        bratCommandList.push({ display: \"---- Plugin Settings ----\", info: async () => { await this.ribbonDisplayCommands() } })\n        listOfPluginSettingsTabs.forEach(si => bratCommandList.push(si));\n\n        gfs.setSuggesterData(bratCommandList);\n        await gfs.display(async (results) => await results.info());\n    }\n\n    constructor(plugin: ThePlugin) {\n        this.plugin = plugin;\n\n        this.bratCommands.forEach(async (item) => {\n            this.plugin.addCommand({\n                id: item.id,\n                name: item.name,\n                icon: item.icon,\n                callback: async () => { await item.callback() }\n            })\n        });\n    }\n\n}\n\n", "import { grabChecksumOfThemeCssFile, grabCommmunityThemeCssFile, grabLastCommitDateForAFile } from \"../features/githubUtils\";\nimport { themeSave, themeDelete, themesCheckAndUpdates } from \"../features/themes\";\nimport ThePlugin from \"../main\";\nimport { Settings } from \"../ui/settings\";\n\n\n// This module is for API access for use in debuging console \n\nexport default class BratAPI {\n    plugin: ThePlugin;\n    settings: Settings; \n\n    constructor(plugin: ThePlugin) {\n        this.plugin = plugin    \n    }\n\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    console = (logDescription: string, ...outputs: any[]): void => {\n        console.log(\"BRAT: \" + logDescription, outputs)\n    }\n\n    themes = {\n\n        themeseCheckAndUpates: async (showInfo: boolean): Promise<void> => {\n            await themesCheckAndUpdates(this.plugin, showInfo);\n        },\n\n        themeInstallTheme: async (cssGithubRepository: string): Promise<void> => {\n            const scrubbedAddress = cssGithubRepository.replace(\"https://github.com/\", \"\");\n            await themeSave(this.plugin, scrubbedAddress, true);\n        },\n\n        themesDelete: async (cssGithubRepository: string): Promise<void> => {\n            const scrubbedAddress = cssGithubRepository.replace(\"https://github.com/\", \"\");\n            await themeDelete(this.plugin, scrubbedAddress)\n        },\n\n        grabCommmunityThemeCssFile: async (repositoryPath: string, betaVersion = false): Promise<string|null> =>  {\n            return await grabCommmunityThemeCssFile(repositoryPath, betaVersion, this.plugin.settings.debuggingMode);\n        },\n\n        grabChecksumOfThemeCssFile: async (repositoryPath: string, betaVersion = false): Promise<string> =>  {\n            return await grabChecksumOfThemeCssFile(repositoryPath, betaVersion, this.plugin.settings.debuggingMode);\n        },\n\n        grabLastCommitDateForAFile: async (repositoryPath: string, path: string): Promise<string> => {\n            // example await grabLastCommitDateForAFile(t.repo, \"theme-beta.css\");\n            return await grabLastCommitDateForAFile(repositoryPath, path)\n        },\n\n\n\n\n        \n    }\n\n}"],
  "mappings": "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",
  "names": ["require_main", "__commonJSMin", "exports", "obsidian", "DEFAULT_DAILY_NOTE_FORMAT", "DEFAULT_WEEKLY_NOTE_FORMAT", "DEFAULT_MONTHLY_NOTE_FORMAT", "DEFAULT_QUARTERLY_NOTE_FORMAT", "DEFAULT_YEARLY_NOTE_FORMAT", "shouldUsePeriodicNotesSettings", "periodicity", "_a", "_b", "periodicNotes", "getDailyNoteSettings", "_c", "_d", "internalPlugins", "plugins", "format", "folder", "template", "err", "getWeeklyNoteSettings", "_e", "_f", "_g", "pluginManager", "calendarSettings", "periodicNotesSettings", "settings", "getMonthlyNoteSettings", "getQuarterlyNoteSettings", "getYearlyNoteSettings", "join", "partSegments", "parts", "i", "l", "newParts", "part", "basename", "fullPath", "base", "ensureFolderExists", "path", "__async", "dirs", "dir", "getNotePath", "directory", "filename", "getTemplateInfo", "metadataCache", "vault", "templatePath", "templateFile", "contents", "IFoldInfo", "getDateUID", "date", "granularity", "ts", "removeEscapedCharacters", "isFormatAmbiguous", "cleanFormat", "getDateFromFile", "file", "getDateFromFilename", "getDateFromPath", "noteDate", "DailyNotesFolderMissingError", "createDailyNote", "app", "moment", "templateContents", "normalizedPath", "createdFile", "_", "_timeOrDate", "calc", "timeDelta", "unit", "momentFormat", "now", "currentDate", "getDailyNote", "dailyNotes", "getAllDailyNotes", "dailyNotesFolder", "note", "dateString", "WeeklyNotesFolderMissingError", "getDaysOfWeek", "weekStart", "daysOfWeek", "getDayOfWeekNumericalValue", "dayOfWeekName", "createWeeklyNote", "dayOfWeek", "day", "getWeeklyNote", "weeklyNotes", "getAllWeeklyNotes", "appHasWeeklyNotesPluginLoaded", "weeklyNotesFolder", "MonthlyNotesFolderMissingError", "createMonthlyNote", "getMonthlyNote", "monthlyNotes", "getAllMonthlyNotes", "appHasMonthlyNotesPluginLoaded", "monthlyNotesFolder", "QuarterlyNotesFolderMissingError", "createQuarterlyNote", "getQuarterlyNote", "quarterly", "getAllQuarterlyNotes", "appHasQuarterlyNotesPluginLoaded", "quarterlyFolder", "YearlyNotesFolderMissingError", "createYearlyNote", "getYearlyNote", "yearlyNotes", "getAllYearlyNotes", "appHasYearlyNotesPluginLoaded", "yearlyNotesFolder", "appHasDailyNotesPluginLoaded", "dailyNotesPlugin", "getPeriodicNoteSettings", "getSettings", "createPeriodicNote", "main_exports", "__export", "ThePlugin", "__toCommonJS", "import_obsidian", "import_obsidian", "import_obsidian", "import_obsidian", "GITHUB_RAW_USERCONTENT_PATH", "grabReleaseFileFromRepository", "repository", "version", "fileName", "debugLogging", "__async", "URL", "download", "error", "grabManifestJsonFromRepository", "repositoryPath", "rootManifest", "manifestJsonPath", "response", "grabCommmunityPluginList", "pluginListURL", "grabCommmunityThemesList", "themesURL", "grabCommmunityThemeCssFile", "betaVersion", "grabCommmunityThemeManifestFile", "checksum", "str", "sum", "i", "checksumForString", "grabChecksumOfThemeCssFile", "themeCSS", "grabLastCommitInfoForAFile", "path", "url", "grabLastCommitDateForAFile", "test", "DEFAULT_SETTINGS", "addBetaPluginToList", "plugin", "repositoryPath", "specifyVersion", "__async", "save", "x", "existBetaPluginInList", "addBetaThemeToList", "themeCSS", "newTheme", "checksumForString", "existBetaThemeinInList", "t", "updateBetaThemeLastUpdateChecksum", "checksum", "import_obsidian", "ToastMessage", "plugin", "msg", "timeoutInSeconds", "contextMenuCallback", "additionalInfo", "newNotice", "__async", "isConnectedToInternet", "__async", "online", "err", "themeSave", "plugin", "cssGithubRepository", "newInstall", "__async", "themeCSS", "grabCommmunityThemeCssFile", "ToastMessage", "themeManifest", "grabCommmunityThemeManifestFile", "manifestInfo", "themeTargetFolderPath", "themesRootPath", "adapter", "updateBetaThemeLastUpdateChecksum", "checksumForString", "msg", "addBetaThemeToList", "themesCheckAndUpdates", "showInfo", "isConnectedToInternet", "newNotice", "msg1", "t", "lastUpdateOnline", "grabChecksumOfThemeCssFile", "msg2", "themeDelete", "import_obsidian", "promotionalLinks", "containerEl", "settingsTab", "linkHeight", "linksDiv", "coffeeSpan", "coffeeImg", "twitterSpan", "twitterImg", "mediumSpan", "mediumImg", "AddNewTheme", "plugin", "openSettingsTabAfterwards", "__async", "scrubbedAddress", "existBetaThemeinInList", "ToastMessage", "themeSave", "formEl", "textEl", "value", "e", "title", "buttonContainerEl", "newDiv", "byTfThacker", "promotionalLinks", "titleEl", "BratSettingsTab", "app", "plugin", "containerEl", "promotionalLinks", "cb", "value", "__async", "pluginSubListFrozenVersionNames", "x", "bp", "btn", "AddNewTheme", "themeDelete", "new_folder", "import_obsidian", "AddNewPluginModal", "plugin", "betaPlugins", "openSettingsTabAfterwards", "useFrozenVersion", "__async", "scrubbedAddress", "existBetaPluginInList", "ToastMessage", "formEl", "textEl", "value", "e", "buttonContainerEl", "newDiv", "byTfThacker", "promotionalLinks", "titleEl", "import_obsidian", "BetaPlugins", "plugin", "openSettingsTabAfterwards", "useFrozenVersion", "__async", "AddNewPluginModal", "repositoryPath", "getBetaManifest", "reportIssues", "manifestJson", "grabManifestJsonFromRepository", "ToastMessage", "manifest", "getManifest", "specifyVersion", "version", "reallyGetManifestOrNot", "grabReleaseFileFromRepository", "betaPluginID", "relFiles", "pluginTargetFolderPath", "adapter", "updatePluginFiles", "seeIfUpdatedOnly", "reportIfNotUpdted", "_a", "primaryManifest", "usingBetaManifest", "msg", "getRelease", "rFiles", "releaseFiles", "addBetaPluginToList", "versionText", "localManifestContents", "e", "x", "localManifestJSON", "pluginName", "plugins", "onlyCheckDontUpdate", "result", "showInfo", "isConnectedToInternet", "newNotice", "msg1", "pluginSubListFrozenVersionNames", "f", "bp", "msg2", "b", "enabled", "pl", "manifests", "enabledPlugins", "p", "import_obsidian", "addIcons", "import_obsidian", "import_obsidian_daily_notes_interface", "logger", "plugin", "textToLog", "verboseLoggingOn", "fileName", "dateOutput", "machineName", "output", "__async", "fileContents", "file", "import_obsidian", "GenericFuzzySuggester", "plugin", "evt", "suggesterData", "callBack", "__async", "item", "el", "selectedText", "i", "PluginCommands", "plugin", "__async", "pluginSubListFrozenVersionNames", "f", "pluginList", "m", "gfs", "GenericFuzzySuggester", "results", "msg", "ToastMessage", "manifest", "communityPlugins", "grabCommmunityPluginList", "communityPluginList", "p", "bratList", "si", "communityTheme", "grabCommmunityThemesList", "communityThemeList", "settings", "listOfPluginSettingsTabs", "t", "listOfCoreSettingsTabs", "AddNewTheme", "themesCheckAndUpdates", "item", "bratCommandList", "cmd", "BratAPI", "plugin", "logDescription", "outputs", "showInfo", "__async", "themesCheckAndUpdates", "cssGithubRepository", "scrubbedAddress", "themeSave", "themeDelete", "repositoryPath", "betaVersion", "grabCommmunityThemeCssFile", "grabChecksumOfThemeCssFile", "path", "grabLastCommitDateForAFile", "ThePlugin", "__async", "BratSettingsTab", "BetaPlugins", "PluginCommands", "addIcons", "themesCheckAndUpdates", "BratAPI", "textToLog", "verbose", "logger", "DEFAULT_SETTINGS"]
}

|