/**
 * @name BDGitHubDownloader
 * @author Ahlawat
 * @authorId 1025214794766221384
 * @version 2.3.0
 * @invite SgKSKyh9gY
 * @description Download BetterDiscord plugins and themes by right clicking on a message containing a GitHub link.
 * @website https://tharki-god.github.io/
 * @source https://github.com/Tharki-God/BetterDiscordPlugins
 * @updateUrl https://tharki-god.github.io/BetterDiscordPlugins/BDGitHubDownloader.plugin.js
 */
/*@cc_on
@if (@_jscript)
var shell = WScript.CreateObject("WScript.Shell");
var fs = new ActiveXObject("Scripting.FileSystemObject");
var pathPlugins = shell.ExpandEnvironmentStrings("%APPDATA%\\BetterDiscord\\plugins");
var pathSelf = WScript.ScriptFullName;
shell.Popup("It looks like you've mistakenly tried to run me directly. \n(Don't do that!)", 0, "I'm a plugin for BetterDiscord", 0x30);
if (fs.GetParentFolderName(pathSelf) === fs.GetAbsolutePathName(pathPlugins)) {
shell.Popup("I'm in the correct folder already.", 0, "I'm already installed", 0x40);
} else if (!fs.FolderExists(pathPlugins)) {
shell.Popup("I can't find the BetterDiscord plugins folder.\nAre you sure it's even installed?", 0, "Can't install myself", 0x10);
} else if (shell.Popup("Should I move myself to BetterDiscord's plugins folder for you?", 0, "Do you need some help?", 0x34) === 6) {
fs.MoveFile(pathSelf, fs.BuildPath(pathPlugins, fs.GetFileName(pathSelf)));
shell.Exec("explorer " + pathPlugins);
shell.Popup("I'm installed!", 0, "Successfully installed", 0x40);
}
WScript.Quit();
@else@*/
module.exports = ((_) => {
  const config = {
    info: {
      name: "BDGitHubDownloader",
      authors: [
        {
          name: "Ahlawat",
          discord_id: "1025214794766221384",
          github_username: "Tharki-God",
        }
      ],
      version: "2.3.0",
      description:
        "Download BetterDiscord plugins and themes by right clicking on a message containing a GitHub link.",
      github: "https://github.com/Tharki-God/BetterDiscordPlugins",
      github_raw:
        "https://tharki-god.github.io/BetterDiscordPlugins/BDGitHubDownloader.plugin.js",
    },
    changelog: [
      {
        title: "v0.0.1",
        items: ["Idea in mind"],
      },
      {
        title: "v0.0.5",
        items: ["Base Model"],
      },
      {
        title: "Initial Release v1.0.0",
        items: [
          "This is the initial release of the plugin :)",
          "Couldn't Have been possible without my sis, Thank you Kirai",
          "No Incest but you are great",
          "btw guys get those illegal plugins easily (>'-'<)",
        ],
      },
      {
        title: "v2.0.0",
        items: ["Better Code", "Theme Support"],
      },
      {
        title: "v2.2.2",
        items: ["Corrected text."],
      },
    ],
    main: "BDGitHubDownloader.plugin.js",
  };
  const RequiredLibs = [{
    window: "ZeresPluginLibrary",
    filename: "0PluginLibrary.plugin.js",
    external: "https://rauenzi.github.io/BDPluginLibrary/release/0PluginLibrary.plugin.js",
    downloadUrl: "https://rauenzi.github.io/BDPluginLibrary/release/0PluginLibrary.plugin.js"
  },
  {
    window: "BunnyLib",
    filename: "1BunnyLib.plugin.js",
    external: "https://github.com/Tharki-God/BetterDiscordPlugins",
    downloadUrl: "https://tharki-god.github.io/BetterDiscordPlugins/1BunnyLib.plugin.js"
  },
  ];
  class handleMissingLibrarys {
    load() {
      for (const Lib of RequiredLibs.filter(lib => !window.hasOwnProperty(lib.window)))
        BdApi.showConfirmationModal(
          "Library Missing",
          `The library plugin (${Lib.window}) needed for ${config.info.name} is missing. Please click Download Now to install it.`,
          {
            confirmText: "Download Now",
            cancelText: "Cancel",
            onConfirm: () => this.downloadLib(Lib),
          }
        );
    }
    async downloadLib(Lib) {
      const fs = require("fs");
      const path = require("path");
      const { Plugins } = BdApi;
      const LibFetch = await fetch(
        Lib.downloadUrl
      );
      if (!LibFetch.ok) return this.errorDownloadLib(Lib);
      const LibContent = await LibFetch.text();
      try {
        await fs.writeFile(
          path.join(Plugins.folder, Lib.filename),
          LibContent,
          (err) => {
            if (err) return this.errorDownloadLib(Lib);
          }
        );
      } catch (err) {
        return this.errorDownloadLib(Lib);
      }
    }
    errorDownloadZLib(Lib) {
      const { shell } = require("electron");
      BdApi.showConfirmationModal(
        "Error Downloading",
        [
          `${Lib.window} download failed. Manually install plugin library from the link below.`,
        ],
        {
          confirmText: "Download",
          cancelText: "Cancel",
          onConfirm: () => {
            shell.openExternal(
              Lib.external
            );
          },
        }
      );
    }
    start() { }
    stop() { }
  }
  return RequiredLibs.some(m => !window.hasOwnProperty(m.window))
    ? handleMissingLibrarys
    : (([Plugin, ZLibrary]) => {
      const {
        Toasts,
        Utilities,
        Logger,
        PluginUpdater,
        Settings: { SettingPanel, SettingGroup, Switch },
      } = ZLibrary;
      const { ContextMenu } = BdApi;
      const {
        LibraryUtils,
        LibraryIcons,
        LibraryRequires: { fs }
      } = BunnyLib.build(config)
      const isGithubUrl = new RegExp(
        "(?:git|https?|git@)(?:\\:\\/\\/)?github.com[/|:][A-Za-z0-9-]+?"
      );
      const isGithubRawUrl = new RegExp(
        "(?:git|https?|git@)(?:\\:\\/\\/)?raw.githubusercontent.com[/|:][A-Za-z0-9-]+?"
      );
      const nameRegex =
        /@name\s+([^\t^\r^\n]+)|\/\/\**META.*["']name["']\s*:\s*["'](.+?)["']/i;
      const fileNameRegex = /([^\\\/]+)$/i;
      const defaultSettings = {
        showToast: true,
        autoEnablePlugin: true,
        showPluginDownload: true,
        autoEnableTheme: true,
        showThemeDownload: true,
      };
      return class BDGitHubDownloader extends Plugin {
        constructor() {
          super();
          this.settings = Utilities.loadData(
            config.info.name,
            "settings",
            defaultSettings
          );
          this.pluginDownloadPatch = this.pluginDownloadPatch.bind(this);
          this.themeDownloadPatch = this.themeDownloadPatch.bind(this);
        }

        checkForUpdates() {
          try {
            PluginUpdater.checkForUpdate(
              config.info.name,
              config.info.version,
              config.info.github_raw
            );
          } catch (err) {
            Logger.err("Plugin Updater could not be reached.", err);
          }
        }
        onStart() {
          this.checkForUpdates();
          this.addMenu();
        }
        addMenu() {
          if (this.settings["showPluginDownload"])
            ContextMenu.patch("message", this.pluginDownloadPatch);
          if (this.settings["showThemeDownload"])
            ContextMenu.patch("message", this.themeDownloadPatch);
        }
        pluginDownloadPatch(menu, { message }) {
          let links = LibraryUtils.getLinks(message.content);
          links = links?.filter((link) => link.endsWith(".plugin.js"));
          if (links?.length)
            for (var link of links) {
              if (isGithubUrl.test(link))
                link = `https://raw.githubusercontent.com/${link.split("github.com/")[1]
                  }`.replace("/blob/", "/");
              const [fileName] = fileNameRegex.exec(link)[0].split(".plugin.js");
              menu.props.children.splice(
                3,
                0,
                ContextMenu.buildItem(
                  {
                    name: `Download ${fileName}`,
                    separate: true,
                    id: `download-${fileName
                      .toLowerCase()
                      .replaceAll(" ", "-")}`,
                    label: `Download ${fileName}`,
                    icon: () => LibraryIcons.Download("20", "20"),
                    action: async () => {
                      if (isGithubRawUrl.test(link))
                        return this.download(
                          link,
                          `${fileName}.plugin.js`,
                          "Plugin"
                        );
                      else if (this.settings["showToast"])
                        Toasts.show(`That link type is not supported`, {
                          icon: "https://tharki-god.github.io/files-random-host/ic_fluent_error_circle_24_regular.png",
                          timeout: 5000,
                          type: "error",
                        });
                    },
                  },
                  true
                )
              );
            }
        }
        themeDownloadPatch(menu, { message }) {
          let links = LibraryUtils.getLinks(message.content);
          links = links?.filter((link, index) => link.endsWith(".theme.css"));
          if (links?.length)
            for (var link of links) {
              if (isGithubUrl.test(link))
                link = `https://raw.githubusercontent.com/${link.split("github.com/")[1]
                  }`.replace("/blob/", "/");
              const [fileName] = fileNameRegex.exec(link)[0].split(".theme.css");
              menu.props.children.splice(
                3,
                0,
                ContextMenu.buildItem(
                  {
                    name: `Download ${fileName}`,
                    separate: true,
                    id: `download-${fileName
                      .toLowerCase()
                      .replaceAll(" ", "-")}`,
                    label: `Download ${fileName}`,
                    icon: () => LibraryIcons.Download("20", "20"),
                    action: async () => {
                      if (isGithubRawUrl.test(link))
                        return this.download(
                          link,
                          `${fileName}.theme.css`,
                          "Theme"
                        );
                      else if (this.settings["showToast"])
                        Toasts.show(`That link type is not supported`, {
                          icon: "https://tharki-god.github.io/files-random-host/ic_fluent_error_circle_24_regular.png",
                          timeout: 5000,
                          type: "error",
                        });
                    },
                  },
                  true
                )
              );
            }
        }
        async download(url, fileName, type) {
          const response = await fetch(url);
          if (!response.ok && this.settings["showToast"])
            Toasts.show(`Downloading Error!`, {
              icon: "https://tharki-god.github.io/files-random-host/ic_fluent_error_circle_24_regular.png",
              timeout: 5000,
              type: "error",
            });
          const data = await response.text();
          let name = (nameRegex.exec(data) || []).filter((n) => n)[1];
          if (this.settings["showToast"])
            Toasts.show(`Downloading ${type}: ${name}`, {
              icon: "https://tharki-god.github.io/files-random-host/ic_fluent_arrow_download_24_filled.png",
              timeout: 5000,
              type: "error",
            });
          try {
            await fs
              .writeFileSync(
                require("path").join(
                  type === "Plugin"
                    ? BdApi.Plugins.folder
                    : BdApi.Themes.folder,
                  fileName
                ),
                data,
                (err) => {
                  if (err) {
                    if (this.settings["showToast"])
                      Toasts.show(` Error: ${err}.`, {
                        icon: "https://tharki-god.github.io/files-random-host/ic_fluent_error_circle_24_regular.png",
                        timeout: 5000,
                        type: "error",
                      });
                    Logger.err(err);
                  }
                }
              )
            if (this.settings["autoEnableTheme"] && type === "Theme") {
              setTimeout(() => {
                BdApi.Themes.enable(name);
              }, 2000);
            } else if (
              this.settings["autoEnablePlugin"] &&
              type === "Plugin"
            ) {
              setTimeout(() => {
                BdApi.Plugins.enable(name);
              }, 2000);
            }

          } catch (err) {
            if (this.settings["showToast"])
              Toasts.show(` Error: ${err}.`, {
                icon: "https://tharki-god.github.io/files-random-host/ic_fluent_error_circle_24_regular.png",
                timeout: 5000,
                type: "error",
              });
            Logger.warn("Something went wrong.", err);
          }
        }
        onStop() {
          ContextMenu.unpatch("message", this.pluginDownloadPatch);
          ContextMenu.unpatch("message", this.themeDownloadPatch);
        }
        getSettingsPanel() {
          return SettingPanel.build(
            this.saveSettings.bind(this),
            new Switch(
              "Pop-up/Toast",
              "Display error/success toast.",
              this.settings["showToast"],
              (e) => {
                this.settings["showToast"] = e;
              }
            ),
            new SettingGroup("Plugins", {
              collapsible: true,
              shown: false,
            }).append(
              new Switch(
                "Show option",
                "Provide the option to download a plugin from a link.",
                this.settings["showPluginDownload"],
                (e) => {
                  this.settings["showPluginDownload"] = e;
                }
              ),
              new Switch(
                "Auto enable",
                "Automatically enable each plugin after downloading.",
                this.settings["autoEnablePlugin"],
                (e) => {
                  this.settings["autoEnablePlugin"] = e;
                }
              )
            ),
            new SettingGroup("Themes", {
              collapsible: true,
              shown: false,
            }).append(
              new Switch(
                "Show option",
                "Provide the option to download a theme from a link.",
                this.settings["showThemeDownload"],
                (e) => {
                  this.settings["showThemeDownload"] = e;
                }
              ),
              new Switch(
                "Auto enable",
                "Automatically enable each theme after downloading.",
                this.settings["autoEnableTheme"],
                (e) => {
                  this.settings["autoEnableTheme"] = e;
                }
              )
            )
          );
        }
        saveSettings() {
          Utilities.saveData(config.info.name, "settings", this.settings);
        }
      };
    })(ZLibrary.buildPlugin(config));
})();
/*@end@*/