diff --git a/data/emotefilter.json b/data/emotefilter.json deleted file mode 100644 index 5da620d9..00000000 --- a/data/emotefilter.json +++ /dev/null @@ -1,1371 +0,0 @@ -{ - "blacklist": [ - "8Bit", - "AAAAAH", - "AESTHETIC", - "AHEM", - "AK47", - "ALAN", - "ALEC", - "ALERT", - "ALIENS", - "ARMY", - "ASCENSION", - "ATOMIC", - "AVERAGE", - "AWAKENED", - "Abaddon", - "Abnegation", - "Abra", - "Aces", - "Advice", - "Affe", - "Afire", - "Aftertaste", - "Agility", - "Ahaha", - "Ahead", - "Aids", - "Ak47", - "Akame", - "Akeno", - "Akuma", - "Alamo", - "Aldo", - "Alexis", - "Alien", - "Alienated", - "Alkaline", - "Amazed", - "Amazing!", - "Amicable", - "Amine", - "Anarchy", - "Anders", - "Anonymous", - "Antichrist", - "Aped", - "Apollo", - "Arduous", - "Ares", - "Aries", - "Arma3", - "Armos", - "Arnold", - "Ashley", - "Asteroid", - "Astor", - "Attack", - "Attempt", - "Avon", - "Awww", - "Ayaya", - "Ayes", - "AyyLmao", - "Azimuth", - "B1nzy", - "BABE", - "BADGE", - "BALLET", - "BANGER", - "BANISH", - "BANNED", - "BANZAI", - "BAPTIZED", - "BEARS", - "BEER", - "BERSERK", - "BINGO", - "BLESS", - "BLESSED", - "BLURB", - "BOMB", - "BOMB", - "BONK", - "BONUS", - "BOOBS", - "BOOM", - "BOOTY", - "BORK", - "BOTCH", - "BREAKFAST", - "BRUCE", - "BUBBLE", - "BUFFER", - "BULLY", - "BURN", - "BURT", - "BUSTER", - "Baby", - "BabyRoll", - "Bagels", - "Baguette", - "Balance", - "Ballin", - "Bamboozle", - "Bamboozled", - "Banana", - "Bandits", - "Bane", - "Banger", - "Banter", - "Barb", - "Barf", - "Barracks", - "Bear", - "Beard", - "Beardless", - "Beer", - "Behemoth", - "Behind", - "Bella", - "Benito", - "Benny", - "Benson", - "Bepis", - "Berber", - "Berge", - "Bern", - "Berserk", - "Besen", - "BetterDiscord", - "Bighorn", - "Bijou", - "Bilbo", - "Bile", - "Bilingual", - "Billy", - "Binary", - "Birb", - "Birb", - "Birdie", - "Bjorn", - "Blanking", - "Blazed", - "Bleaching", - "Bleeder", - "Blip", - "Blubber", - "Blurred", - "Boeuf", - "Bolts", - "Bomb", - "Bombastic", - "Bombshell", - "Boned", - "Boner", - "Bonfire", - "Bonito", - "Bonnie", - "Booby", - "Boosted", - "Bork", - "Boss", - "Bourbon", - "Brille", - "Brimstone", - "Brittany", - "Brofist", - "Bruh", - "Bryce", - "Bubbles", - "Bubs", - "Bumpy", - "Bunny", - "Burg", - "Burial", - "Busted", - "Bustin", - "Butterfly", - "Butterfly", - "CAKE", - "CASH", - "CAUTION", - "CHAOS", - "CHOCOLATE", - "CLAM", - "CLASSIC", - "CLENCH", - "CLIP", - "COAST", - "COFFIN", - "COGS", - "CONFUSION", - "CONSUME", - "COSMOS", - "COVENANT", - "CRETIN", - "CROSSED", - "CSS3", - "CURSE", - "Cactus", - "Cadence", - "Canard", - "Candlelight", - "Cannon", - "Cara", - "Card", - "Caret", - "Carina", - "Carlton", - "Cart", - "Castle", - "Caveman", - "Chains", - "Challenge", - "Champ", - "Charged", - "Charizard", - "Charlie", - "Charlotte", - "Cheeky", - "Cheese", - "Chelsea", - "Chester", - "Chet", - "Chew", - "Chickpea", - "Chinking", - "Chocobo", - "Chocolate", - "Choke", - "Chris", - "Chrysalis", - "Chuckles", - "Cinder", - "Cinnamon", - "Claire", - "Clank", - "Clarisse", - "Cleft", - "Clinton", - "Clive", - "Coagulant", - "Coaxial", - "Coco", - "Coffee", - "Coldsteel", - "Colonel", - "Commenter", - "Conga", - "Conniptions", - "Constipation", - "Constitution", - "Construction", - "Content", - "Convicted", - "Conway", - "Cookie", - "Cooking", - "Coolest", - "Coop", - "Cooper", - "Corny", - "Costanza", - "Covered", - "Crabby", - "Crafting", - "Credo", - "Creeper", - "Crock", - "Crossover", - "Crown", - "Cruising", - "Crumbs", - "Crusaders", - "Cube", - "Cucumber", - "Cupcake", - "Cupid", - "Cuppa", - "Custard", - "DAMAGE", - "DAVE", - "DEAD", - "DEATH", - "DEEM", - "DEER", - "DELETE", - "DELETED", - "DELUXE", - "DENIAL", - "DENIED", - "DESTROYED", - "DESU", - "DETECTED", - "DETERMINATION", - "DEVIL", - "DICKS", - "DODGE", - "DODO", - "DOGS", - "DRINK", - "DUCKS", - "DUDE", - "DUNKED", - "DUST", - "DWAYNE", - "Dale", - "Dallas", - "Damnit", - "Damnit", - "Dance", - "Dante", - "Darnell", - "Dashing", - "Dauntless", - "Davy", - "Dayum", - "Deadass", - "Deadass", - "Dearest", - "Dedede", - "Dehumidify", - "Demon", - "Denis", - "Dentures", - "Derp", - "Derp", - "Deud", - "Dickered", - "Digestive", - "Ding", - "Dingle", - "Dirk", - "Disgusted", - "Ditto", - "Divination", - "Dodo", - "Doge", - "Dogging", - "Donnie", - "Dons", - "Donuts", - "Dood", - "Dropsy", - "Drunken", - "Duane", - "Dunked", - "Dunno", - "Dyson", - "EDGE", - "EMOTE", - "ENERGY", - "ESCAPE", - "EXACTLY", - "EXCELLENT", - "EXPLOSION", - "EXTERMINATE", - "Earl", - "Eduardo", - "Egging", - "Eggman", - "Eggplant", - "Electrode", - "Elementals", - "Envision", - "Ernie", - "Errol", - "Ethereal", - "Eugh", - "Everyone!", - "Evil", - "Excalibur", - "Exclaim", - "Exclamation", - "Exclusive", - "Exited", - "Exploded", - "Exposed", - "Eyes", - "Eyyy", - "Eyyyyyy", - "Ezreal", - "FAIL", - "FAIL", - "FAILURE", - "FAKER", - "FANCY", - "FAST", - "FEESH", - "FERVOR", - "FEVER", - "FFXV", - "FIREBALL", - "FIVE", - "FOCUS", - "FORESHADOWING", - "FORKED", - "FOUR", - "FOXHOUND", - "FRIG", - "FRISK", - "FUMBLES", - "Face", - "Faded", - "Fail", - "Faithless", - "Farming", - "Faro", - "Fatality", - "Faust", - "Faye", - "Fennec", - "Feuer", - "Fidget", - "Fighting", - "Fire", - "Firefly", - "First", - "Fishing", - "Fishy", - "Fist", - "Fluffy", - "Follow", - "Forman", - "Foxhound", - "Franck", - "Frankie", - "Freddie", - "Freddy", - "Fritz", - "Frog", - "GAME!", - "GAMING", - "GANG", - "GERARD", - "GET!", - "GIBE", - "GLITCH", - "GOBBLE", - "GRIN", - "GROWL", - "GUILE", - "Gaben", - "Gabriel", - "Gambler", - "Gamest", - "Gandalf", - "Garble", - "Gary", - "Gaseous", - "Gasp", - "Gauche", - "Gazorpazorpfield", - "Generated", - "Gengar", - "Genius", - "George!", - "Gerald", - "Gerard", - "Ghastly", - "Ghost", - "Gilbert", - "Gilliam", - "Gilligan", - "Gimpy", - "Glenn", - "Glob", - "Gloom", - "Gloomy", - "Glottis", - "Gnome", - "Goat", - "Godhead", - "Godspeed", - "Gold", - "Goldberg", - "Goldblum", - "Golden", - "Gondola", - "Goofballs", - "Gosu", - "Gramps", - "Greened", - "Greener", - "Gregg", - "Greninja", - "Grimace", - "Grunge", - "Guacamole", - "Guardianship", - "Guff", - "Guile", - "Guinness", - "Gunny", - "Gunter", - "Gunther", - "Gustav", - "H3H3", - "HAAA", - "HAHAHA", - "HALLO", - "HANK", - "HEART", - "HEEHAW", - "HELGA", - "HELLO", - "HELLO!", - "HELP", - "HELP", - "HEY!", - "HOFF", - "HOLLOW", - "HONK", - "HORNLESS", - "HUGO", - "HUNK", - "HUZZAH", - "HYPE", - "HYPE", - "Hacked", - "Hai!", - "Halp", - "Halp", - "Handsome", - "Hannibal", - "Hanzo", - "Happy", - "Harambe", - "Hartz", - "Haste", - "Headliner", - "Heart", - "Heartless", - "Heated", - "Heave", - "Heels", - "Hello", - "Heresy", - "Herz", - "Hey!", - "Heya!", - "Heyo", - "Hiccup", - "Hiccups", - "Hideout", - "Hiho", - "Hillary", - "Hina", - "Hobbes", - "Homeboy", - "Honda", - "Horace", - "Horde", - "Huck", - "Huehue", - "Huffs", - "Hullo", - "Humm", - "Hunter", - "Husk", - "Hydro", - "Hype", - "Hypnosis", - "INFIDEL", - "INHALE", - "INIT", - "INSANE", - "Ibiza", - "Ibuki", - "Icebox", - "Iceman", - "Ichigo", - "Illusionist", - "Imperil", - "Implications", - "Impulse", - "Impulsion", - "Inconceivable", - "Infiltrated", - "Inkling", - "Integrity", - "Interrogation", - "Iota", - "Isaac", - "Isadora", - "Isis", - "JAPANNED", - "JESSE", - "JOBLESS", - "JUMP", - "JUNES", - "JUST", - "JUSTICE", - "Jabba", - "Jace", - "Jaffa", - "Jago", - "James", - "Jazz", - "Jeanne", - "Jeremiah", - "Jerry", - "Jill", - "John", - "Johnny", - "Joint", - "Josef", - "Josuke", - "Juliana", - "Julius", - "KAPPA", - "KAPPA", - "KARL", - "KISS", - "KOBE", - "Kaede", - "Kali", - "Kaput", - "Karma", - "Katze", - "Keith", - "Kenny", - "Kevin", - "Kian", - "Kierkegaard", - "Kill", - "Kipper", - "Kirby", - "Kisses", - "Kitsune", - "Kitsunemimi", - "Kitty", - "Kkkkk", - "Knotty", - "Koba", - "Kola", - "Kona", - "Koopa", - "Krill", - "LEFT", - "LEGENDARY", - "LEVELED", - "LEWD", - "LEXINGTON", - "LIES", - "LINDA", - "LINK", - "LOADS", - "LOCO", - "LOGO", - "LOOT", - "LOST", - "LOUD", - "LOVE", - "LULUL", - "LURK", - "Lamb", - "Lank", - "Lanky", - "Lantern", - "Lapin", - "Lapras", - "Lars", - "Launch", - "Leffen", - "Lemons", - "Lemony", - "Lenny", - "Leon", - "Leonardo", - "Lester", - "LetsPlay", - "Lewd", - "Licky", - "Limpa", - "Link", - "Lion", - "Liquid", - "Lockjaw", - "Loge", - "Logo", - "Lonk", - "Loool", - "Loots", - "Lorraine", - "Lose", - "Lotus", - "Loudness", - "Louie", - "Love", - "Lucarionite", - "Lucas", - "Lucky", - "Ludicrous", - "Lulu", - "Lunch", - "Lunk", - "Lurk", - "Lurking", - "MANGER", - "MART", - "MAST", - "MASTERPIECE", - "MEGA", - "MENACING", - "MERICA", - "MMMMM", - "MODERNS", - "MOIST", - "MONEY", - "MONICA", - "MSPaint", - "MUSTACHE", - "Machine", - "Mackey", - "Macron", - "Mafia", - "Magic", - "Magikarp", - "Mahatma", - "MainMenu", - "Malice", - "Manes", - "Mangle", - "Mania", - "Manny", - "Marco", - "Mari", - "Marius", - "Martin", - "Massage", - "Massey", - "Master", - "Matty", - "Maxi", - "Maya", - "Maytag", - "Memelord", - "Menacing", - "Meow", - "Mercenary", - "Mettaton", - "Mhmm", - "Mick", - "Midair", - "Midna", - "Milk", - "Milkman", - "Millie", - "Mindy", - "Minecraft", - "Minigames", - "Mining", - "Minion", - "Mirabelle", - "Miriam", - "Missile", - "Mitchell", - "Miura", - "Mmmm", - "Moddb", - "Moin", - "Momma", - "Moneybags", - "Mongrels", - "Monkey", - "Monolith", - "Montezuma", - "Moomin", - "Moon", - "Mopping", - "Morreu", - "Mouton", - "Muffin", - "Mustachio", - "Myers", - "NADINE", - "NANA", - "NEAT", - "NERD", - "NICE", - "NICK", - "NICK", - "NITS", - "NOOK", - "NOOOO", - "NOVA", - "NUTS", - "Narwhal", - "Nathaniel", - "Nati", - "Nebby", - "Nein", - "Nemesis", - "Nerdy", - "Nice!", - "Nimbus", - "Nita", - "Nixon", - "Noel", - "Nolan", - "Nono", - "Noodle", - "Nooo", - "Noot", - "Norway", - "Nugget", - "Numb", - "Nuuu", - "OBJECTION", - "OBJECTION", - "OHNO", - "OKAY", - "OMG!", - "OMGG", - "OMGGG", - "OPieOP", - "OUCH", - "OVERTIME", - "OVERWATCH", - "Objection", - "Odin", - "Ohok", - "Okay", - "Okie", - "Oldies", - "Ollie", - "Omniscient", - "OneShot", - "Oooh", - "Ooooohhh", - "Oops", - "Orange", - "Overlord", - "PACE", - "PAINTBALL", - "PANDA", - "PARDON", - "PAUSE", - "PEACE", - "PELICANS", - "PENETRATING", - "PENETRATION", - "PEPPY", - "PERFECT", - "PERFECT", - "PHALANX", - "PINHEAD", - "PINKIE", - "PLANTED", - "PLAYS", - "PLAYSTATION", - "PLEASE", - "PLEASE", - "POISON", - "PONCHO", - "PROPHET", - "PUKE", - "PURE", - "Pachimari", - "Paladins", - "Panther", - "Panzer", - "Parappa", - "Parker", - "Pastor", - "Patches", - "Pathetic", - "Patty", - "Peacemaker", - "Peach", - "Peachy", - "Pedro", - "Peeps", - "Penguin", - "Penn", - "Pent", - "Pera", - "Perez", - "Pfft", - "Phenylalanine", - "Phil", - "Phyllis", - "Picnic", - "Pigeon", - "Piggy", - "Pikachu", - "Pinch", - "Pineapple", - "Pinker", - "Pinkie", - "Pinking", - "Pinto", - "Piper", - "Plating", - "Pleure", - "Plop", - "Plotting", - "Plugged", - "Point", - "PokeBall", - "Pokeball", - "Poked", - "Police", - "Pomeranian", - "Pooch", - "Poodles", - "Poopy", - "Poppa", - "Poppins", - "Port", - "Possessor", - "Potassium", - "Potato", - "Potsdam", - "Poyo", - "Practice", - "Praetor", - "Praise", - "Prayer", - "Prince", - "Profi", - "Pudding", - "Puffball", - "Punchy", - "Puppy", - "Pursed", - "QUAD", - "Quack", - "Question", - "Quickie", - "RAGE", - "RAGE", - "REALLY", - "REGRET", - "REKTangle", - "REMIX", - "RENAULT", - "RENT", - "RESET", - "RICHARD", - "RIDER", - "RIGGED", - "RIGHT", - "RIPPER", - "RITA", - "RNGesus", - "ROMA", - "ROOSTER", - "RUDY", - "RUNAWAY", - "Rabbits", - "Rage", - "Rages", - "Rainbow", - "Rainmaker", - "Ranged", - "Rash", - "Ravage", - "Raven", - "Rebirth", - "Reblochon", - "Reface", - "Reggie", - "Rein", - "Rejected", - "Rengar", - "Reported", - "Reporting", - "Resetti", - "Reuben", - "Ribombee", - "Ricky", - "Riddles", - "Rifleman", - "Rigged", - "Ringside", - "Riolu", - "Ripper", - "Ripperoni", - "Roasted", - "Rocky", - "Roderick", - "Roscoe", - "Roshan", - "Rosy", - "Roxanne", - "Rubicon", - "Rucksack", - "Ruckus", - "Runescape", - "Rusted", - "Rusty", - "SALAMI", - "SARDINES", - "SAWYER", - "SEGA", - "SEXY", - "SHENANIGANS", - "SHINE", - "SHOCKED", - "SHOOT", - "SHOTS", - "SHOVEL", - "SHUCKS", - "SLAM", - "SLAPPED", - "SLAT", - "SLEEPY", - "SLOW", - "SMASH", - "SMASHING", - "SMILE", - "SMILEY", - "SMITHY", - "SMOL", - "SMUG", - "SNACKS", - "SNAIL", - "SONIC", - "SPENCER", - "SPHERICAL", - "SPIKED", - "SPLAT", - "SPLIT", - "SPONGEBOB", - "SPOOK", - "SPOONED", - "SQUAD", - "SQUIRREL", - "SQUIRT", - "SQUISH", - "SQUISHY", - "SRS!", - "STEAK", - "STEAK", - "STEVE", - "STOP", - "SWAG", - "Saber", - "Safari", - "Sally", - "Salty", - "Sammich", - "Samus", - "Sandbag", - "Sans", - "Sartre", - "Schaf", - "Schizoid", - "Schopenhauer", - "Scorpio", - "Scout", - "Screw", - "Scrub", - "Scuffed", - "Scum", - "Sectoid", - "Sense", - "Serena", - "Serious", - "Seriously", - "Serperior", - "Servine", - "Shadow", - "Shaman", - "Shambled", - "Sharpshooter", - "Shat", - "Shawn", - "Shaymin", - "Shearing", - "Shelly", - "Sherlock", - "Shield", - "Shiny", - "Shirase", - "Shitpost", - "Shoryuken", - "Shuffled", - "Shulk", - "Sigh", - "Sizzle", - "Skippy", - "Skitters", - "Skylarked", - "Slapstick", - "Slayer", - "Slicker", - "Slime", - "Slowbro", - "Slowpoke", - "Slums", - "Smashed", - "Smashing", - "Smithing", - "Smokey", - "Smooch", - "Smooches", - "Snake", - "Snapdragon", - "Snapshot", - "Snek", - "Sniped", - "Sniper", - "Snivy", - "Snore", - "Snorlax", - "Snowman", - "Sonya", - "Soraka", - "Sourpuss", - "Soviet", - "Spaghetti", - "Sparta", - "Speared", - "Specialist", - "Spider", - "Spikes", - "Spitz", - "Spla2n", - "Splattered", - "Spongebob", - "Spooky", - "Springfield", - "Spud", - "Spume", - "Squad", - "Squall", - "Squint", - "Stalin", - "Stanchion", - "Stanza", - "Starboard", - "Starr", - "Stellar", - "Steve", - "Strength", - "Stuffs", - "Stygian", - "Success", - "Succubus", - "Summoning", - "Sunshine", - "Survivor", - "Swag", - "Swanky", - "Sweating", - "Sword", - "Swordplay", - "Sylveon", - "TANGO", - "TARA", - "TECHNOLOGY", - "THANKS", - "THIEF", - "THIRST", - "THREE", - "THUD", - "TITS", - "TOASTY", - "TOMFOOLERY", - "TOOTHY", - "TOOTS", - "TORPEDO", - "TOUCHDOWN", - "TRAILERS", - "TRASH", - "TRIGGERED", - "TROLL", - "TRUTH", - "TUNA", - "TURTLE", - "Taiga", - "Takeoff", - "Tambourine", - "Tanner", - "Targets", - "Taro", - "Tater", - "Teatime", - "Technical", - "Teddy", - "Terminated", - "Thanking", - "Thieving", - "Thinking", - "Thirsty", - "Thor", - "Thoughts", - "Thump", - "Tiara", - "Tillie", - "Tilted", - "Tingly", - "Tipsy", - "Toad", - "Toasty", - "Toby", - "Toejam", - "Toils", - "Token", - "Tone", - "Tong", - "Tonne", - "Topkek", - "Torpedo", - "Toss", - "Touchdown", - "Tower", - "Tr4sh", - "Tracy", - "Trap", - "Trash", - "Tree", - "Trekt", - "Tricky", - "Troll", - "Trooper", - "Truelove", - "Trumped", - "Turkey", - "Twink", - "Twinning", - "Twitch", - "Tyrael", - "Tyro", - "UNRELENTING", - "Uhhh", - "Uhhhhh", - "Ulgrim", - "Unacceptable", - "Unbearable", - "Underling", - "Unquenchable", - "Unrivaled", - "Unstoppable", - "Untitled", - "Unwise", - "Uranus", - "VANQUISH", - "VICTORY", - "VIEWS", - "VOLVO", - "Vayne", - "Versus", - "Viking", - "Villager", - "Volcano", - "Voli", - "Volt", - "Voltage", - "WAIT", - "WAOW", - "WARPING", - "WARTIME", - "WELCOME", - "WHACK", - "WHAT!", - "WHOOPS", - "WIGWAM", - "WINNER", - "WONDERFUL", - "WOODS", - "Wacko", - "Waddles", - "Waive", - "Walled", - "Wanker", - "Warning", - "Warwick", - "Wassup", - "Watching", - "Wayne", - "Wdym", - "Weenie", - "Welcome", - "Wendy", - "Whaaa", - "Whaaaat", - "Whalecum", - "Whammy", - "Whelps", - "Whew", - "Whoa!", - "Whoosh", - "Wicket", - "Widespread", - "Wildcat", - "Willy", - "Winder", - "Wining", - "Winston", - "Wolf", - "Woodcutting", - "Woodman", - "Woodpecker", - "Woooh", - "Word", - "Wurst", - "YAHOO", - "YEAH", - "YES!", - "YIPPEE", - "YOUTUBE", - "YUMMY", - "Yalta", - "Yawn", - "Yay!", - "Yikes", - "Yolo", - "Yosh", - "You!", - "YouTube", - "Yup!", - "ZERO", - "ZOMBIE", - "Zangief", - "Zappa", - "Zappy", - "Zerker", - "Zero", - "Ziggs", - "Zilean", - "Zinger", - "Zippy", - "Zombie", - "Zorua", - "Zyra", - "babb", - "bonbon", - "broken", - "chemo", - "conan", - "constructor", - "dale", - "dank", - "dodo", - "double", - "eHHH", - "elgato", - "emote", - "facepalm", - "ghillie", - "hasOwnProperty", - "heartless", - "inverse", - "lamer", - "lenny", - "limburger", - "linger", - "meditation", - "morph", - "nuke", - "objection", - "osu!", - "pebbles", - "penta", - "quadra", - "rekt", - "sinner", - "snap", - "triple", - "unnamed", - "userID", - "vogel", - "xDDDD", - "yukon" - ] -} \ No newline at end of file diff --git a/data/emotes/blacklist.json b/data/emotes/blacklist.json new file mode 100644 index 00000000..0493a27f --- /dev/null +++ b/data/emotes/blacklist.json @@ -0,0 +1,1369 @@ +[ + "8Bit", + "AAAAAH", + "AESTHETIC", + "AHEM", + "AK47", + "ALAN", + "ALEC", + "ALERT", + "ALIENS", + "ARMY", + "ASCENSION", + "ATOMIC", + "AVERAGE", + "AWAKENED", + "Abaddon", + "Abnegation", + "Abra", + "Aces", + "Advice", + "Affe", + "Afire", + "Aftertaste", + "Agility", + "Ahaha", + "Ahead", + "Aids", + "Ak47", + "Akame", + "Akeno", + "Akuma", + "Alamo", + "Aldo", + "Alexis", + "Alien", + "Alienated", + "Alkaline", + "Amazed", + "Amazing!", + "Amicable", + "Amine", + "Anarchy", + "Anders", + "Anonymous", + "Antichrist", + "Aped", + "Apollo", + "Arduous", + "Ares", + "Aries", + "Arma3", + "Armos", + "Arnold", + "Ashley", + "Asteroid", + "Astor", + "Attack", + "Attempt", + "Avon", + "Awww", + "Ayaya", + "Ayes", + "AyyLmao", + "Azimuth", + "B1nzy", + "BABE", + "BADGE", + "BALLET", + "BANGER", + "BANISH", + "BANNED", + "BANZAI", + "BAPTIZED", + "BEARS", + "BEER", + "BERSERK", + "BINGO", + "BLESS", + "BLESSED", + "BLURB", + "BOMB", + "BOMB", + "BONK", + "BONUS", + "BOOBS", + "BOOM", + "BOOTY", + "BORK", + "BOTCH", + "BREAKFAST", + "BRUCE", + "BUBBLE", + "BUFFER", + "BULLY", + "BURN", + "BURT", + "BUSTER", + "Baby", + "BabyRoll", + "Bagels", + "Baguette", + "Balance", + "Ballin", + "Bamboozle", + "Bamboozled", + "Banana", + "Bandits", + "Bane", + "Banger", + "Banter", + "Barb", + "Barf", + "Barracks", + "Bear", + "Beard", + "Beardless", + "Beer", + "Behemoth", + "Behind", + "Bella", + "Benito", + "Benny", + "Benson", + "Bepis", + "Berber", + "Berge", + "Bern", + "Berserk", + "Besen", + "BetterDiscord", + "Bighorn", + "Bijou", + "Bilbo", + "Bile", + "Bilingual", + "Billy", + "Binary", + "Birb", + "Birb", + "Birdie", + "Bjorn", + "Blanking", + "Blazed", + "Bleaching", + "Bleeder", + "Blip", + "Blubber", + "Blurred", + "Boeuf", + "Bolts", + "Bomb", + "Bombastic", + "Bombshell", + "Boned", + "Boner", + "Bonfire", + "Bonito", + "Bonnie", + "Booby", + "Boosted", + "Bork", + "Boss", + "Bourbon", + "Brille", + "Brimstone", + "Brittany", + "Brofist", + "Bruh", + "Bryce", + "Bubbles", + "Bubs", + "Bumpy", + "Bunny", + "Burg", + "Burial", + "Busted", + "Bustin", + "Butterfly", + "Butterfly", + "CAKE", + "CASH", + "CAUTION", + "CHAOS", + "CHOCOLATE", + "CLAM", + "CLASSIC", + "CLENCH", + "CLIP", + "COAST", + "COFFIN", + "COGS", + "CONFUSION", + "CONSUME", + "COSMOS", + "COVENANT", + "CRETIN", + "CROSSED", + "CSS3", + "CURSE", + "Cactus", + "Cadence", + "Canard", + "Candlelight", + "Cannon", + "Cara", + "Card", + "Caret", + "Carina", + "Carlton", + "Cart", + "Castle", + "Caveman", + "Chains", + "Challenge", + "Champ", + "Charged", + "Charizard", + "Charlie", + "Charlotte", + "Cheeky", + "Cheese", + "Chelsea", + "Chester", + "Chet", + "Chew", + "Chickpea", + "Chinking", + "Chocobo", + "Chocolate", + "Choke", + "Chris", + "Chrysalis", + "Chuckles", + "Cinder", + "Cinnamon", + "Claire", + "Clank", + "Clarisse", + "Cleft", + "Clinton", + "Clive", + "Coagulant", + "Coaxial", + "Coco", + "Coffee", + "Coldsteel", + "Colonel", + "Commenter", + "Conga", + "Conniptions", + "Constipation", + "Constitution", + "Construction", + "Content", + "Convicted", + "Conway", + "Cookie", + "Cooking", + "Coolest", + "Coop", + "Cooper", + "Corny", + "Costanza", + "Covered", + "Crabby", + "Crafting", + "Credo", + "Creeper", + "Crock", + "Crossover", + "Crown", + "Cruising", + "Crumbs", + "Crusaders", + "Cube", + "Cucumber", + "Cupcake", + "Cupid", + "Cuppa", + "Custard", + "DAMAGE", + "DAVE", + "DEAD", + "DEATH", + "DEEM", + "DEER", + "DELETE", + "DELETED", + "DELUXE", + "DENIAL", + "DENIED", + "DESTROYED", + "DESU", + "DETECTED", + "DETERMINATION", + "DEVIL", + "DICKS", + "DODGE", + "DODO", + "DOGS", + "DRINK", + "DUCKS", + "DUDE", + "DUNKED", + "DUST", + "DWAYNE", + "Dale", + "Dallas", + "Damnit", + "Damnit", + "Dance", + "Dante", + "Darnell", + "Dashing", + "Dauntless", + "Davy", + "Dayum", + "Deadass", + "Deadass", + "Dearest", + "Dedede", + "Dehumidify", + "Demon", + "Denis", + "Dentures", + "Derp", + "Derp", + "Deud", + "Dickered", + "Digestive", + "Ding", + "Dingle", + "Dirk", + "Disgusted", + "Ditto", + "Divination", + "Dodo", + "Doge", + "Dogging", + "Donnie", + "Dons", + "Donuts", + "Dood", + "Dropsy", + "Drunken", + "Duane", + "Dunked", + "Dunno", + "Dyson", + "EDGE", + "EMOTE", + "ENERGY", + "ESCAPE", + "EXACTLY", + "EXCELLENT", + "EXPLOSION", + "EXTERMINATE", + "Earl", + "Eduardo", + "Egging", + "Eggman", + "Eggplant", + "Electrode", + "Elementals", + "Envision", + "Ernie", + "Errol", + "Ethereal", + "Eugh", + "Everyone!", + "Evil", + "Excalibur", + "Exclaim", + "Exclamation", + "Exclusive", + "Exited", + "Exploded", + "Exposed", + "Eyes", + "Eyyy", + "Eyyyyyy", + "Ezreal", + "FAIL", + "FAIL", + "FAILURE", + "FAKER", + "FANCY", + "FAST", + "FEESH", + "FERVOR", + "FEVER", + "FFXV", + "FIREBALL", + "FIVE", + "FOCUS", + "FORESHADOWING", + "FORKED", + "FOUR", + "FOXHOUND", + "FRIG", + "FRISK", + "FUMBLES", + "Face", + "Faded", + "Fail", + "Faithless", + "Farming", + "Faro", + "Fatality", + "Faust", + "Faye", + "Fennec", + "Feuer", + "Fidget", + "Fighting", + "Fire", + "Firefly", + "First", + "Fishing", + "Fishy", + "Fist", + "Fluffy", + "Follow", + "Forman", + "Foxhound", + "Franck", + "Frankie", + "Freddie", + "Freddy", + "Fritz", + "Frog", + "GAME!", + "GAMING", + "GANG", + "GERARD", + "GET!", + "GIBE", + "GLITCH", + "GOBBLE", + "GRIN", + "GROWL", + "GUILE", + "Gaben", + "Gabriel", + "Gambler", + "Gamest", + "Gandalf", + "Garble", + "Gary", + "Gaseous", + "Gasp", + "Gauche", + "Gazorpazorpfield", + "Generated", + "Gengar", + "Genius", + "George!", + "Gerald", + "Gerard", + "Ghastly", + "Ghost", + "Gilbert", + "Gilliam", + "Gilligan", + "Gimpy", + "Glenn", + "Glob", + "Gloom", + "Gloomy", + "Glottis", + "Gnome", + "Goat", + "Godhead", + "Godspeed", + "Gold", + "Goldberg", + "Goldblum", + "Golden", + "Gondola", + "Goofballs", + "Gosu", + "Gramps", + "Greened", + "Greener", + "Gregg", + "Greninja", + "Grimace", + "Grunge", + "Guacamole", + "Guardianship", + "Guff", + "Guile", + "Guinness", + "Gunny", + "Gunter", + "Gunther", + "Gustav", + "H3H3", + "HAAA", + "HAHAHA", + "HALLO", + "HANK", + "HEART", + "HEEHAW", + "HELGA", + "HELLO", + "HELLO!", + "HELP", + "HELP", + "HEY!", + "HOFF", + "HOLLOW", + "HONK", + "HORNLESS", + "HUGO", + "HUNK", + "HUZZAH", + "HYPE", + "HYPE", + "Hacked", + "Hai!", + "Halp", + "Halp", + "Handsome", + "Hannibal", + "Hanzo", + "Happy", + "Harambe", + "Hartz", + "Haste", + "Headliner", + "Heart", + "Heartless", + "Heated", + "Heave", + "Heels", + "Hello", + "Heresy", + "Herz", + "Hey!", + "Heya!", + "Heyo", + "Hiccup", + "Hiccups", + "Hideout", + "Hiho", + "Hillary", + "Hina", + "Hobbes", + "Homeboy", + "Honda", + "Horace", + "Horde", + "Huck", + "Huehue", + "Huffs", + "Hullo", + "Humm", + "Hunter", + "Husk", + "Hydro", + "Hype", + "Hypnosis", + "INFIDEL", + "INHALE", + "INIT", + "INSANE", + "Ibiza", + "Ibuki", + "Icebox", + "Iceman", + "Ichigo", + "Illusionist", + "Imperil", + "Implications", + "Impulse", + "Impulsion", + "Inconceivable", + "Infiltrated", + "Inkling", + "Integrity", + "Interrogation", + "Iota", + "Isaac", + "Isadora", + "Isis", + "JAPANNED", + "JESSE", + "JOBLESS", + "JUMP", + "JUNES", + "JUST", + "JUSTICE", + "Jabba", + "Jace", + "Jaffa", + "Jago", + "James", + "Jazz", + "Jeanne", + "Jeremiah", + "Jerry", + "Jill", + "John", + "Johnny", + "Joint", + "Josef", + "Josuke", + "Juliana", + "Julius", + "KAPPA", + "KAPPA", + "KARL", + "KISS", + "KOBE", + "Kaede", + "Kali", + "Kaput", + "Karma", + "Katze", + "Keith", + "Kenny", + "Kevin", + "Kian", + "Kierkegaard", + "Kill", + "Kipper", + "Kirby", + "Kisses", + "Kitsune", + "Kitsunemimi", + "Kitty", + "Kkkkk", + "Knotty", + "Koba", + "Kola", + "Kona", + "Koopa", + "Krill", + "LEFT", + "LEGENDARY", + "LEVELED", + "LEWD", + "LEXINGTON", + "LIES", + "LINDA", + "LINK", + "LOADS", + "LOCO", + "LOGO", + "LOOT", + "LOST", + "LOUD", + "LOVE", + "LULUL", + "LURK", + "Lamb", + "Lank", + "Lanky", + "Lantern", + "Lapin", + "Lapras", + "Lars", + "Launch", + "Leffen", + "Lemons", + "Lemony", + "Lenny", + "Leon", + "Leonardo", + "Lester", + "LetsPlay", + "Lewd", + "Licky", + "Limpa", + "Link", + "Lion", + "Liquid", + "Lockjaw", + "Loge", + "Logo", + "Lonk", + "Loool", + "Loots", + "Lorraine", + "Lose", + "Lotus", + "Loudness", + "Louie", + "Love", + "Lucarionite", + "Lucas", + "Lucky", + "Ludicrous", + "Lulu", + "Lunch", + "Lunk", + "Lurk", + "Lurking", + "MANGER", + "MART", + "MAST", + "MASTERPIECE", + "MEGA", + "MENACING", + "MERICA", + "MMMMM", + "MODERNS", + "MOIST", + "MONEY", + "MONICA", + "MSPaint", + "MUSTACHE", + "Machine", + "Mackey", + "Macron", + "Mafia", + "Magic", + "Magikarp", + "Mahatma", + "MainMenu", + "Malice", + "Manes", + "Mangle", + "Mania", + "Manny", + "Marco", + "Mari", + "Marius", + "Martin", + "Massage", + "Massey", + "Master", + "Matty", + "Maxi", + "Maya", + "Maytag", + "Memelord", + "Menacing", + "Meow", + "Mercenary", + "Mettaton", + "Mhmm", + "Mick", + "Midair", + "Midna", + "Milk", + "Milkman", + "Millie", + "Mindy", + "Minecraft", + "Minigames", + "Mining", + "Minion", + "Mirabelle", + "Miriam", + "Missile", + "Mitchell", + "Miura", + "Mmmm", + "Moddb", + "Moin", + "Momma", + "Moneybags", + "Mongrels", + "Monkey", + "Monolith", + "Montezuma", + "Moomin", + "Moon", + "Mopping", + "Morreu", + "Mouton", + "Muffin", + "Mustachio", + "Myers", + "NADINE", + "NANA", + "NEAT", + "NERD", + "NICE", + "NICK", + "NICK", + "NITS", + "NOOK", + "NOOOO", + "NOVA", + "NUTS", + "Narwhal", + "Nathaniel", + "Nati", + "Nebby", + "Nein", + "Nemesis", + "Nerdy", + "Nice!", + "Nimbus", + "Nita", + "Nixon", + "Noel", + "Nolan", + "Nono", + "Noodle", + "Nooo", + "Noot", + "Norway", + "Nugget", + "Numb", + "Nuuu", + "OBJECTION", + "OBJECTION", + "OHNO", + "OKAY", + "OMG!", + "OMGG", + "OMGGG", + "OPieOP", + "OUCH", + "OVERTIME", + "OVERWATCH", + "Objection", + "Odin", + "Ohok", + "Okay", + "Okie", + "Oldies", + "Ollie", +"Omniscient", + "OneShot", + "Oooh", + "Ooooohhh", + "Oops", + "Orange", + "Overlord", + "PACE", + "PAINTBALL", + "PANDA", + "PARDON", + "PAUSE", + "PEACE", + "PELICANS", + "PENETRATING", + "PENETRATION", + "PEPPY", + "PERFECT", + "PERFECT", + "PHALANX", + "PINHEAD", + "PINKIE", + "PLANTED", + "PLAYS", + "PLAYSTATION", + "PLEASE", + "PLEASE", + "POISON", + "PONCHO", + "PROPHET", + "PUKE", + "PURE", + "Pachimari", + "Paladins", + "Panther", + "Panzer", + "Parappa", + "Parker", + "Pastor", + "Patches", + "Pathetic", + "Patty", + "Peacemaker", + "Peach", + "Peachy", + "Pedro", + "Peeps", + "Penguin", + "Penn", + "Pent", + "Pera", + "Perez", + "Pfft", + "Phenylalanine", + "Phil", + "Phyllis", + "Picnic", + "Pigeon", + "Piggy", + "Pikachu", + "Pinch", + "Pineapple", + "Pinker", + "Pinkie", + "Pinking", + "Pinto", + "Piper", + "Plating", + "Pleure", + "Plop", + "Plotting", + "Plugged", + "Point", + "PokeBall", + "Pokeball", + "Poked", + "Police", + "Pomeranian", + "Pooch", + "Poodles", + "Poopy", + "Poppa", + "Poppins", + "Port", + "Possessor", + "Potassium", + "Potato", + "Potsdam", + "Poyo", + "Practice", + "Praetor", + "Praise", + "Prayer", + "Prince", + "Profi", + "Pudding", + "Puffball", + "Punchy", + "Puppy", + "Pursed", + "QUAD", + "Quack", + "Question", + "Quickie", + "RAGE", + "RAGE", + "REALLY", + "REGRET", + "REKTangle", + "REMIX", + "RENAULT", + "RENT", + "RESET", + "RICHARD", + "RIDER", + "RIGGED", + "RIGHT", + "RIPPER", + "RITA", + "RNGesus", + "ROMA", + "ROOSTER", + "RUDY", + "RUNAWAY", + "Rabbits", + "Rage", + "Rages", + "Rainbow", + "Rainmaker", + "Ranged", + "Rash", + "Ravage", + "Raven", + "Rebirth", + "Reblochon", + "Reface", + "Reggie", + "Rein", + "Rejected", + "Rengar", + "Reported", + "Reporting", + "Resetti", + "Reuben", + "Ribombee", + "Ricky", + "Riddles", + "Rifleman", + "Rigged", + "Ringside", + "Riolu", + "Ripper", + "Ripperoni", + "Roasted", + "Rocky", + "Roderick", + "Roscoe", + "Roshan", + "Rosy", + "Roxanne", + "Rubicon", + "Rucksack", + "Ruckus", + "Runescape", + "Rusted", + "Rusty", + "SALAMI", + "SARDINES", + "SAWYER", + "SEGA", + "SEXY", + "SHENANIGANS", + "SHINE", + "SHOCKED", + "SHOOT", + "SHOTS", + "SHOVEL", + "SHUCKS", + "SLAM", + "SLAPPED", + "SLAT", + "SLEEPY", + "SLOW", + "SMASH", + "SMASHING", + "SMILE", + "SMILEY", + "SMITHY", + "SMOL", + "SMUG", + "SNACKS", + "SNAIL", + "SONIC", + "SPENCER", + "SPHERICAL", + "SPIKED", + "SPLAT", + "SPLIT", + "SPONGEBOB", + "SPOOK", + "SPOONED", + "SQUAD", + "SQUIRREL", + "SQUIRT", + "SQUISH", + "SQUISHY", + "SRS!", + "STEAK", + "STEAK", + "STEVE", + "STOP", + "SWAG", + "Saber", + "Safari", + "Sally", + "Salty", + "Sammich", + "Samus", + "Sandbag", + "Sans", + "Sartre", + "Schaf", + "Schizoid", + "Schopenhauer", + "Scorpio", + "Scout", + "Screw", + "Scrub", + "Scuffed", + "Scum", + "Sectoid", + "Sense", + "Serena", + "Serious", + "Seriously", + "Serperior", + "Servine", + "Shadow", + "Shaman", + "Shambled", + "Sharpshooter", + "Shat", + "Shawn", + "Shaymin", + "Shearing", + "Shelly", + "Sherlock", + "Shield", + "Shiny", + "Shirase", + "Shitpost", + "Shoryuken", + "Shuffled", + "Shulk", + "Sigh", + "Sizzle", + "Skippy", + "Skitters", + "Skylarked", + "Slapstick", + "Slayer", + "Slicker", + "Slime", + "Slowbro", + "Slowpoke", + "Slums", + "Smashed", + "Smashing", + "Smithing", + "Smokey", + "Smooch", + "Smooches", + "Snake", + "Snapdragon", + "Snapshot", + "Snek", + "Sniped", + "Sniper", + "Snivy", + "Snore", + "Snorlax", + "Snowman", + "Sonya", + "Soraka", + "Sourpuss", + "Soviet", + "Spaghetti", + "Sparta", + "Speared", + "Specialist", + "Spider", + "Spikes", + "Spitz", + "Spla2n", + "Splattered", + "Spongebob", + "Spooky", + "Springfield", + "Spud", + "Spume", + "Squad", + "Squall", + "Squint", + "Stalin", + "Stanchion", + "Stanza", + "Starboard", + "Starr", + "Stellar", + "Steve", + "Strength", + "Stuffs", + "Stygian", + "Success", + "Succubus", + "Summoning", + "Sunshine", + "Survivor", + "Swag", + "Swanky", + "Sweating", + "Sword", + "Swordplay", + "Sylveon", + "TANGO", + "TARA", + "TECHNOLOGY", + "THANKS", + "THIEF", + "THIRST", + "THREE", + "THUD", + "TITS", + "TOASTY", + "TOMFOOLERY", + "TOOTHY", + "TOOTS", + "TORPEDO", + "TOUCHDOWN", + "TRAILERS", + "TRASH", + "TRIGGERED", + "TROLL", + "TRUTH", + "TUNA", + "TURTLE", + "Taiga", + "Takeoff", + "Tambourine", + "Tanner", + "Targets", + "Taro", + "Tater", + "Teatime", + "Technical", + "Teddy", + "Terminated", + "Thanking", + "Thieving", + "Thinking", + "Thirsty", + "Thor", + "Thoughts", + "Thump", + "Tiara", + "Tillie", + "Tilted", + "Tingly", + "Tipsy", + "Toad", + "Toasty", + "Toby", + "Toejam", + "Toils", + "Token", + "Tone", + "Tong", + "Tonne", + "Topkek", + "Torpedo", + "Toss", + "Touchdown", + "Tower", + "Tr4sh", + "Tracy", + "Trap", + "Trash", + "Tree", + "Trekt", + "Tricky", + "Troll", + "Trooper", + "Truelove", + "Trumped", + "Turkey", + "Twink", + "Twinning", + "Twitch", + "Tyrael", + "Tyro", + "UNRELENTING", + "Uhhh", + "Uhhhhh", + "Ulgrim", + "Unacceptable", + "Unbearable", + "Underling", + "Unquenchable", + "Unrivaled", + "Unstoppable", + "Untitled", + "Unwise", + "Uranus", + "VANQUISH", + "VICTORY", + "VIEWS", + "VOLVO", + "Vayne", + "Versus", + "Viking", + "Villager", + "Volcano", + "Voli", + "Volt", + "Voltage", + "WAIT", + "WAOW", + "WARPING", + "WARTIME", + "WELCOME", + "WHACK", + "WHAT!", + "WHOOPS", + "WIGWAM", + "WINNER", + "WONDERFUL", + "WOODS", + "Wacko", + "Waddles", + "Waive", + "Walled", + "Wanker", + "Warning", + "Warwick", + "Wassup", + "Watching", + "Wayne", + "Wdym", + "Weenie", + "Welcome", + "Wendy", + "Whaaa", + "Whaaaat", + "Whalecum", + "Whammy", + "Whelps", + "Whew", + "Whoa!", + "Whoosh", + "Wicket", + "Widespread", + "Wildcat", + "Willy", + "Winder", + "Wining", + "Winston", + "Wolf", + "Woodcutting", + "Woodman", + "Woodpecker", + "Woooh", + "Word", + "Wurst", + "YAHOO", + "YEAH", + "YES!", + "YIPPEE", + "YOUTUBE", + "YUMMY", + "Yalta", + "Yawn", + "Yay!", + "Yikes", + "Yolo", + "Yosh", + "You!", + "YouTube", + "Yup!", + "ZERO", + "ZOMBIE", + "Zangief", + "Zappa", + "Zappy", + "Zerker", + "Zero", + "Ziggs", + "Zilean", + "Zinger", + "Zippy", + "Zombie", + "Zorua", + "Zyra", + "babb", + "bonbon", + "broken", + "chemo", + "conan", + "constructor", + "dale", + "dank", + "dodo", + "double", + "eHHH", + "elgato", + "emote", + "facepalm", + "ghillie", + "hasOwnProperty", + "heartless", + "inverse", + "lamer", + "lenny", + "limburger", + "linger", + "meditation", + "morph", + "nuke", + "objection", + "osu!", + "pebbles", + "penta", + "quadra", + "rekt", + "sinner", + "snap", + "triple", + "unnamed", + "userID", + "vogel", + "xDDDD", + "yukon" +] \ No newline at end of file diff --git a/data/emotes/bttv.json b/data/emotes/bttv.json new file mode 100644 index 00000000..6d46a1df --- /dev/null +++ b/data/emotes/bttv.json @@ -0,0 +1 @@ +{"OhMyGoodness":"//cdn.betterttv.net/emote/54fa925e01e468494b85b54d/1x","PancakeMix":"//cdn.betterttv.net/emote/54fa927801e468494b85b54e/1x","PedoBear":"//cdn.betterttv.net/emote/54fa928f01e468494b85b54f/1x","PokerFace":"//cdn.betterttv.net/emote/54fa92a701e468494b85b550/1x","RageFace":"//cdn.betterttv.net/emote/54fa92d701e468494b85b552/1x","RebeccaBlack":"//cdn.betterttv.net/emote/54fa92ee01e468494b85b553/1x",":tf:":"//cdn.betterttv.net/emote/54fa8f1401e468494b85b537/1x","aPliS":"//cdn.betterttv.net/emote/54fa8f4201e468494b85b538/1x","CiGrip":"//cdn.betterttv.net/emote/54fa8fce01e468494b85b53c/1x","CHAccepted":"//cdn.betterttv.net/emote/54fa8fb201e468494b85b53b/1x","FuckYea":"//cdn.betterttv.net/emote/54fa90d601e468494b85b544/1x","DatSauce":"//cdn.betterttv.net/emote/54fa903b01e468494b85b53f/1x","ForeverAlone":"//cdn.betterttv.net/emote/54fa909b01e468494b85b542/1x","GabeN":"//cdn.betterttv.net/emote/54fa90ba01e468494b85b543/1x","HailHelix":"//cdn.betterttv.net/emote/54fa90f201e468494b85b545/1x","HerbPerve":"//cdn.betterttv.net/emote/54fa913701e468494b85b546/1x","iDog":"//cdn.betterttv.net/emote/54fa919901e468494b85b548/1x","rStrike":"//cdn.betterttv.net/emote/54fa930801e468494b85b554/1x","ShoopDaWhoop":"//cdn.betterttv.net/emote/54fa932201e468494b85b555/1x","SwedSwag":"//cdn.betterttv.net/emote/54fa9cc901e468494b85b565/1x","M&Mjc":"//cdn.betterttv.net/emote/54fab45f633595ca4c713abc/1x","bttvNice":"//cdn.betterttv.net/emote/54fab7d2633595ca4c713abf/1x","TopHam":"//cdn.betterttv.net/emote/54fa934001e468494b85b556/1x","TwaT":"//cdn.betterttv.net/emote/54fa935601e468494b85b557/1x","WhatAYolk":"//cdn.betterttv.net/emote/54fa93d001e468494b85b559/1x","WatChuSay":"//cdn.betterttv.net/emote/54fa99b601e468494b85b55d/1x","Blackappa":"//cdn.betterttv.net/emote/54faa50d01e468494b85b578/1x","DogeWitIt":"//cdn.betterttv.net/emote/54faa52f01e468494b85b579/1x","BadAss":"//cdn.betterttv.net/emote/54faa4f101e468494b85b577/1x","SavageJerky":"//cdn.betterttv.net/emote/54fb603201abde735115ddb5/1x","Zappa":"//cdn.betterttv.net/emote/5622aaef3286c42e57d8e4ab/1x","tehPoleCat":"//cdn.betterttv.net/emote/566ca11a65dbbdab32ec0558/1x","AngelThump":"//cdn.betterttv.net/emote/566ca1a365dbbdab32ec055b/1x","Kaged":"//cdn.betterttv.net/emote/54fbf11001abde735115de66/1x","HHydro":"//cdn.betterttv.net/emote/54fbef6601abde735115de57/1x","TaxiBro":"//cdn.betterttv.net/emote/54fbefeb01abde735115de5b/1x","BroBalt":"//cdn.betterttv.net/emote/54fbf00a01abde735115de5c/1x","ButterSauce":"//cdn.betterttv.net/emote/54fbf02f01abde735115de5d/1x","BaconEffect":"//cdn.betterttv.net/emote/54fbf05a01abde735115de5e/1x","SuchFraud":"//cdn.betterttv.net/emote/54fbf07e01abde735115de5f/1x","CandianRage":"//cdn.betterttv.net/emote/54fbf09c01abde735115de61/1x","She'llBeRight":"//cdn.betterttv.net/emote/54fbefc901abde735115de5a/1x","OhhhKee":"//cdn.betterttv.net/emote/54fbefa901abde735115de59/1x","D:":"//cdn.betterttv.net/emote/55028cd2135896936880fdd7/1x","SexPanda":"//cdn.betterttv.net/emote/5502874d135896936880fdd2/1x","(poolparty)":"//cdn.betterttv.net/emote/5502883d135896936880fdd3/1x",":'(":"//cdn.betterttv.net/emote/55028923135896936880fdd5/1x","(puke)":"//cdn.betterttv.net/emote/550288fe135896936880fdd4/1x","bttvWink":"//cdn.betterttv.net/emote/550292c0135896936880fdef/1x","bttvAngry":"//cdn.betterttv.net/emote/550291a3135896936880fde3/1x","bttvConfused":"//cdn.betterttv.net/emote/550291be135896936880fde4/1x","bttvCool":"//cdn.betterttv.net/emote/550291d4135896936880fde5/1x","bttvHappy":"//cdn.betterttv.net/emote/55029200135896936880fde7/1x","bttvSad":"//cdn.betterttv.net/emote/5502925d135896936880fdea/1x","bttvSleep":"//cdn.betterttv.net/emote/55029272135896936880fdeb/1x","bttvSurprised":"//cdn.betterttv.net/emote/55029288135896936880fdec/1x","bttvTongue":"//cdn.betterttv.net/emote/5502929b135896936880fded/1x","bttvUnsure":"//cdn.betterttv.net/emote/550292ad135896936880fdee/1x","bttvGrin":"//cdn.betterttv.net/emote/550291ea135896936880fde6/1x","bttvHeart":"//cdn.betterttv.net/emote/55029215135896936880fde8/1x","bttvTwink":"//cdn.betterttv.net/emote/55029247135896936880fde9/1x","VisLaud":"//cdn.betterttv.net/emote/550352766f86a5b26c281ba2/1x","(chompy)":"//cdn.betterttv.net/emote/550b225fff8ecee922d2a3b2/1x","SoSerious":"//cdn.betterttv.net/emote/5514afe362e6bd0027aede8a/1x","BatKappa":"//cdn.betterttv.net/emote/550b6b07ff8ecee922d2a3e7/1x","KaRappa":"//cdn.betterttv.net/emote/550b344bff8ecee922d2a3c1/1x","YetiZ":"//cdn.betterttv.net/emote/55189a5062e6bd0027aee082/1x","miniJulia":"//cdn.betterttv.net/emote/552d2fc2236a1aa17a996c5b/1x","FishMoley":"//cdn.betterttv.net/emote/566ca00f65dbbdab32ec0544/1x","Hhhehehe":"//cdn.betterttv.net/emote/566ca02865dbbdab32ec0547/1x","KKona":"//cdn.betterttv.net/emote/566ca04265dbbdab32ec054a/1x","OhGod":"//cdn.betterttv.net/emote/566ca07965dbbdab32ec0552/1x","PoleDoge":"//cdn.betterttv.net/emote/566ca09365dbbdab32ec0555/1x","motnahP":"//cdn.betterttv.net/emote/55288e390fa35376704a4c7a/1x","sosGame":"//cdn.betterttv.net/emote/553b48a21f145f087fc15ca6/1x","CruW":"//cdn.betterttv.net/emote/55471c2789d53f2d12781713/1x","RarePepe":"//cdn.betterttv.net/emote/555015b77676617e17dd2e8e/1x","iamsocal":"//cdn.betterttv.net/emote/54fbef8701abde735115de58/1x","haHAA":"//cdn.betterttv.net/emote/555981336ba1901877765555/1x","FeelsBirthdayMan":"//cdn.betterttv.net/emote/55b6524154eefd53777b2580/1x","RonSmug":"//cdn.betterttv.net/emote/55f324c47f08be9f0a63cce0/1x","KappaCool":"//cdn.betterttv.net/emote/560577560874de34757d2dc0/1x","FeelsBadMan":"//cdn.betterttv.net/emote/566c9fc265dbbdab32ec053b/1x","BasedGod":"//cdn.betterttv.net/emote/566c9eeb65dbbdab32ec052b/1x","bUrself":"//cdn.betterttv.net/emote/566c9f3b65dbbdab32ec052e/1x","ConcernDoge":"//cdn.betterttv.net/emote/566c9f6365dbbdab32ec0532/1x","FapFapFap":"//cdn.betterttv.net/emote/566c9f9265dbbdab32ec0538/1x","FeelsGoodMan":"//cdn.betterttv.net/emote/566c9fde65dbbdab32ec053e/1x","FireSpeed":"//cdn.betterttv.net/emote/566c9ff365dbbdab32ec0541/1x","NaM":"//cdn.betterttv.net/emote/566ca06065dbbdab32ec054e/1x","SourPls":"//cdn.betterttv.net/emote/566ca38765dbbdab32ec0560/1x","LuL":"//cdn.betterttv.net/emote/567b00c61ddbe1786688a633/1x","SaltyCorn":"//cdn.betterttv.net/emote/56901914991f200c34ffa656/1x","FCreep":"//cdn.betterttv.net/emote/56d937f7216793c63ec140cb/1x","monkaS":"//cdn.betterttv.net/emote/56e9f494fff3cc5c35e5287e/1x","VapeNation":"//cdn.betterttv.net/emote/56f5be00d48006ba34f530a4/1x","ariW":"//cdn.betterttv.net/emote/56fa09f18eff3b595e93ac26/1x","notsquishY":"//cdn.betterttv.net/emote/5709ab688eff3b595e93c595/1x","FeelsAmazingMan":"//cdn.betterttv.net/emote/5733ff12e72c3c0814233e20/1x","DuckerZ":"//cdn.betterttv.net/emote/573d38b50ffbf6cc5cc38dc9/1x","SqShy":"//cdn.betterttv.net/emote/59cf182fcbe2693d59d7bf46/1x","Wowee":"//cdn.betterttv.net/emote/58d2e73058d8950a875ad027/1x"} \ No newline at end of file diff --git a/data/emotedata_bttv.json b/data/emotes/bttv2.json similarity index 100% rename from data/emotedata_bttv.json rename to data/emotes/bttv2.json diff --git a/data/emotedata_ffz.json b/data/emotes/ffz.json similarity index 100% rename from data/emotedata_ffz.json rename to data/emotes/ffz.json diff --git a/data/emotedata_twitch_global.json b/data/emotes/twitchglobal.json similarity index 100% rename from data/emotedata_twitch_global.json rename to data/emotes/twitchglobal.json diff --git a/data/emotedata_twitch_subscriber.json b/data/emotes/twitchsubscriber.json similarity index 100% rename from data/emotedata_twitch_subscriber.json rename to data/emotes/twitchsubscriber.json diff --git a/data/locales/en.json b/data/locales/en.json index f8cbf1d0..6d832b97 100644 --- a/data/locales/en.json +++ b/data/locales/en.json @@ -168,7 +168,7 @@ "title": "{{name}} v{{version}} by {{author}}", "openFolder": "Open {{type}} Folder", "reload": "Reload", - "pluginSettings": "Settings", + "addonSettings": "Settings", "website": "Website", "source": "Source", "server": "Support Server", diff --git a/data/updater.json b/data/updater.json index 31318cc1..d1303021 100644 --- a/data/updater.json +++ b/data/updater.json @@ -1,4 +1,3 @@ { - "LatestVersion":"0.3.2", - "CDN":"cdn.rawgit.com" + "injectorVersion": "0.3.2" } \ No newline at end of file diff --git a/js/main.js b/js/main.js index 92d87909..12478f69 100644 --- a/js/main.js +++ b/js/main.js @@ -203,7 +203,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _str /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _structs_builtin__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../structs/builtin */ \"./src/structs/builtin.js\");\n/* harmony import */ var data__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! data */ \"./src/data/data.js\");\n/* harmony import */ var modules__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! modules */ \"./src/modules/modules.js\");\n/* harmony import */ var _ui_emote__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../ui/emote */ \"./src/ui/emote.js\");\n/* harmony import */ var _ui_toasts__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../ui/toasts */ \"./src/ui/toasts.js\");\n\n\n\n\n // import EmoteMenu from \"./emotemenu\";\n\nconst Emotes = {\n TwitchGlobal: {},\n TwitchSubscriber: {},\n BTTV: {},\n FrankerFaceZ: {},\n BTTV2: {}\n};\nconst bdEmoteSettingIDs = {\n TwitchGlobal: \"twitch\",\n TwitchSubscriber: \"twitch\",\n BTTV: \"bttv\",\n FrankerFaceZ: \"ffz\",\n BTTV2: \"bttv\"\n};\nconst blacklist = [];\nconst overrides = [\"twitch\", \"bttv\", \"ffz\"];\nconst modifiers = [\"flip\", \"spin\", \"pulse\", \"spin2\", \"spin3\", \"1spin\", \"2spin\", \"3spin\", \"tr\", \"bl\", \"br\", \"shake\", \"shake2\", \"shake3\", \"flap\"];\n/* harmony default export */ __webpack_exports__[\"default\"] = (new class EmoteModule extends _structs_builtin__WEBPACK_IMPORTED_MODULE_0__[\"default\"] {\n get name() {\n return \"Emotes\";\n }\n\n get collection() {\n return \"settings\";\n }\n\n get category() {\n return \"general\";\n }\n\n get id() {\n return \"emotes\";\n }\n\n get categories() {\n return Object.keys(bdEmoteSettingIDs).filter(k => this.isCategoryEnabled(bdEmoteSettingIDs[k]));\n }\n\n isCategoryEnabled(id) {\n return super.get(\"emotes\", \"categories\", id);\n }\n\n get(id) {\n return super.get(\"emotes\", \"general\", id);\n }\n\n get MessageContentComponent() {\n return modules__WEBPACK_IMPORTED_MODULE_2__[\"WebpackModules\"].getModule(m => m.defaultProps && m.defaultProps.hasOwnProperty(\"disableButtons\"));\n }\n\n get Emotes() {\n return Emotes;\n }\n\n get TwitchGlobal() {\n return Emotes.TwitchGlobal;\n }\n\n get TwitchSubscriber() {\n return Emotes.TwitchSubscriber;\n }\n\n get BTTV() {\n return Emotes.BTTV;\n }\n\n get FrankerFaceZ() {\n return Emotes.FrankerFaceZ;\n }\n\n get BTTV2() {\n return Emotes.BTTV2;\n }\n\n get blacklist() {\n return blacklist;\n }\n\n get favorites() {\n return this.favoriteEmotes;\n }\n\n getCategory(category) {\n return Emotes[category];\n }\n\n initialize() {\n super.initialize();\n this.favoriteEmotes = {};\n const fe = modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].getBDData(\"bdfavemotes\");\n if (fe !== \"\" && fe !== null) this.favoriteEmotes = JSON.parse(window.atob(fe));\n this.saveFavorites();\n this.addFavorite = this.addFavorite.bind(this);\n this.removeFavorite = this.removeFavorite.bind(this); // EmoteConfig;\n // emoteCollection.button = {title: \"Clear Emote Cache\", onClick: () => { this.clearEmoteData(); this.loadEmoteData(EmoteInfo); }};\n }\n\n async enabled() {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].registerCollection(\"emotes\", \"Emotes\", data__WEBPACK_IMPORTED_MODULE_1__[\"EmoteConfig\"], {\n title: modules__WEBPACK_IMPORTED_MODULE_2__[\"Strings\"].Emotes.clearEmotes,\n onClick: () => {\n this.clearEmoteData();\n this.loadEmoteData(data__WEBPACK_IMPORTED_MODULE_1__[\"EmoteInfo\"]);\n }\n }); // Disable emote module for now because it's annoying and slow\n // await this.getBlacklist();\n // await this.loadEmoteData(EmoteInfo);\n // while (!this.MessageContentComponent) await new Promise(resolve => setTimeout(resolve, 100));\n // this.patchMessageContent();\n\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].on(\"emotes-favorite-added\", this.addFavorite);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].on(\"emotes-favorite-removed\", this.removeFavorite);\n }\n\n disabled() {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].off(\"emotes-favorite-added\", this.addFavorite);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].off(\"emotes-favorite-removed\", this.removeFavorite);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].removeCollection(\"emotes\");\n this.emptyEmotes();\n if (!this.cancelEmoteRender) return;\n this.cancelEmoteRender();\n delete this.cancelEmoteRender;\n }\n\n addFavorite(name, url) {\n if (!this.favoriteEmotes.hasOwnProperty(name)) this.favoriteEmotes[name] = url;\n this.saveFavorites();\n }\n\n removeFavorite(name) {\n if (!this.favoriteEmotes.hasOwnProperty(name)) return;\n delete this.favoriteEmotes[name];\n this.saveFavorites();\n }\n\n isFavorite(name) {\n return this.favoriteEmotes.hasOwnProperty(name);\n }\n\n saveFavorites() {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"bdfavemotes\", window.btoa(JSON.stringify(this.favoriteEmotes)));\n }\n\n emptyEmotes() {\n for (const cat in Emotes) Object.assign(Emotes, {\n [cat]: {}\n });\n }\n\n patchMessageContent() {\n if (this.cancelEmoteRender) return;\n this.cancelEmoteRender = this.after(this.MessageContentComponent.prototype, \"render\", (thisObj, args, retVal) => {\n this.after(retVal.props, \"children\", (t, a, returnValue) => {\n if (this.categories.length == 0) return;\n const markup = returnValue.props.children[1];\n if (!markup.props.children) return;\n const nodes = markup.props.children[1];\n if (!nodes || !nodes.length) return;\n\n for (let n = 0; n < nodes.length; n++) {\n const node = nodes[n];\n if (typeof node !== \"string\") continue;\n const words = node.split(/([^\\s]+)([\\s]|$)/g);\n\n for (let c = 0, clen = this.categories.length; c < clen; c++) {\n for (let w = 0, wlen = words.length; w < wlen; w++) {\n const emote = words[w];\n const emoteSplit = emote.split(\":\");\n const emoteName = emoteSplit[0];\n let emoteModifier = emoteSplit[1] ? emoteSplit[1] : \"\";\n let emoteOverride = emoteModifier.slice(0);\n if (emoteName.length < 4 || blacklist.includes(emoteName)) continue;\n if (!modifiers.includes(emoteModifier) || !modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].get(this.category, \"general\", \"modifiers\")) emoteModifier = \"\";\n if (!overrides.includes(emoteOverride)) emoteOverride = \"\";else emoteModifier = emoteOverride;\n let current = this.categories[c];\n\n if (emoteOverride === \"twitch\") {\n if (Emotes.TwitchGlobal[emoteName]) current = \"TwitchGlobal\";else if (Emotes.TwitchSubscriber[emoteName]) current = \"TwitchSubscriber\";\n } else if (emoteOverride === \"bttv\") {\n if (Emotes.BTTV[emoteName]) current = \"BTTV\";else if (Emotes.BTTV2[emoteName]) current = \"BTTV2\";\n } else if (emoteOverride === \"ffz\") {\n if (Emotes.FrankerFaceZ[emoteName]) current = \"FrankerFaceZ\";\n }\n\n if (!Emotes[current][emoteName] || !modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].get(this.category, \"categories\", bdEmoteSettingIDs[current])) continue;\n const results = nodes[n].match(new RegExp(`([\\\\s]|^)${modules__WEBPACK_IMPORTED_MODULE_2__[\"Utilities\"].escape(emoteModifier ? emoteName + \":\" + emoteModifier : emoteName)}([\\\\s]|$)`));\n if (!results) continue;\n const pre = nodes[n].substring(0, results.index + results[1].length);\n const post = nodes[n].substring(results.index + results[0].length - results[2].length);\n nodes[n] = pre;\n const emoteComponent = modules__WEBPACK_IMPORTED_MODULE_2__[\"DiscordModules\"].React.createElement(_ui_emote__WEBPACK_IMPORTED_MODULE_3__[\"default\"], {\n name: emoteName,\n url: Emotes[current][emoteName],\n modifier: emoteModifier,\n isFavorite: this.isFavorite(emoteName)\n });\n nodes.splice(n + 1, 0, post);\n nodes.splice(n + 1, 0, emoteComponent);\n }\n }\n }\n\n const onlyEmotes = nodes.every(r => {\n if (typeof r == \"string\" && r.replace(/\\s*/, \"\") == \"\") return true;else if (r.type && r.type.name == \"BDEmote\") return true;else if (r.props && r.props.children && r.props.children.props && r.props.children.props.emojiName) return true;\n return false;\n });\n if (!onlyEmotes) return;\n\n for (const node of nodes) {\n if (typeof node != \"object\") continue;\n if (node.type.name == \"BDEmote\") node.props.jumboable = true;else if (node.props && node.props.children && node.props.children.props && node.props.children.props.emojiName) node.props.children.props.jumboable = true;\n }\n });\n });\n }\n\n async loadEmoteData(emoteInfo) {\n this.emotesLoaded = false;\n\n const _fs = __webpack_require__(/*! fs */ \"fs\");\n\n const emoteFile = \"emote_data.json\";\n const file = data__WEBPACK_IMPORTED_MODULE_1__[\"Config\"].dataPath + emoteFile;\n\n const exists = _fs.existsSync(file);\n\n if (exists && this.isCacheValid()) {\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(\"Loading emotes from cache.\", {\n type: \"info\"\n });\n this.log(\"Loading emotes from local cache.\");\n const data = await new Promise(resolve => {\n _fs.readFile(file, \"utf8\", (err, content) => {\n this.log(\"Emotes loaded from cache.\");\n if (err) content = {};\n resolve(content);\n });\n });\n const parsed = modules__WEBPACK_IMPORTED_MODULE_2__[\"Utilities\"].testJSON(data);\n let isValid = !!parsed;\n if (isValid) Object.assign(Emotes, parsed);\n\n for (const e in emoteInfo) {\n isValid = Object.keys(Emotes[emoteInfo[e].variable]).length > 0;\n }\n\n if (isValid) {\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(\"Emotes successfully loaded.\", {\n type: \"success\"\n });\n this.emotesLoaded = true;\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].dispatch(\"emotes-loaded\");\n return;\n }\n\n this.log(\"Cache was corrupt, downloading...\");\n\n _fs.unlinkSync(file);\n }\n\n if (!modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].get(this.category, \"general\", \"download\")) return;\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(modules__WEBPACK_IMPORTED_MODULE_2__[\"Strings\"].Emotes.downloading, {\n type: \"info\"\n });\n\n for (const e in emoteInfo) {\n await new Promise(r => setTimeout(r, 1000));\n const data = await this.downloadEmotes(emoteInfo[e]);\n Emotes[emoteInfo[e].variable] = data;\n }\n\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(modules__WEBPACK_IMPORTED_MODULE_2__[\"Strings\"].Emotes.downloaded, {\n type: \"success\"\n });\n\n try {\n _fs.writeFileSync(file, JSON.stringify(Emotes), \"utf8\");\n } catch (err) {\n this.stacktrace(\"Could not save emote data.\", err);\n }\n\n this.emotesLoaded = true;\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].dispatch(\"emotes-loaded\");\n }\n\n downloadEmotes(emoteMeta) {\n const request = __webpack_require__(/*! request */ \"request\");\n\n const options = {\n url: emoteMeta.url,\n timeout: emoteMeta.timeout ? emoteMeta.timeout : 5000,\n json: true\n };\n this.log(`Downloading: ${emoteMeta.variable} (${emoteMeta.url})`);\n return new Promise((resolve, reject) => {\n request(options, (error, response, parsedData) => {\n if (error) {\n this.stacktrace(\"Could not download \" + emoteMeta.variable, error);\n\n if (emoteMeta.backup) {\n emoteMeta.url = emoteMeta.backup;\n emoteMeta.backup = null;\n if (emoteMeta.backupParser) emoteMeta.parser = emoteMeta.backupParser;\n return resolve(this.downloadEmotes(emoteMeta));\n }\n\n return reject({});\n }\n\n if (typeof emoteMeta.parser === \"function\") parsedData = emoteMeta.parser(parsedData);\n\n for (const emote in parsedData) {\n if (emote.length < 4 || blacklist.includes(emote)) {\n delete parsedData[emote];\n continue;\n }\n\n parsedData[emote] = emoteMeta.getEmoteURL(parsedData[emote]);\n }\n\n resolve(parsedData);\n this.log(\"Downloaded: \" + emoteMeta.variable);\n });\n });\n }\n\n getBlacklist() {\n return new Promise(resolve => {\n $.getJSON(`https://rauenzi.github.io/BetterDiscordApp/data/emotefilter.json`, function (data) {\n resolve(blacklist.push(...data.blacklist));\n });\n });\n }\n\n isCacheValid() {\n const cacheLength = modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].getBDData(\"emoteCacheDays\") || modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"emoteCacheDays\", 7) || 7;\n const cacheDate = new Date(modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].getBDData(\"emoteCacheDate\") || null);\n const currentDate = new Date();\n const daysBetween = Math.round(Math.abs((currentDate.getTime() - cacheDate.getTime()) / (24 * 60 * 60 * 1000)));\n\n if (daysBetween > cacheLength) {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"emoteCacheDate\", currentDate.toJSON());\n return false;\n }\n\n return true;\n }\n\n clearEmoteData() {\n const _fs = __webpack_require__(/*! fs */ \"fs\");\n\n const emoteFile = \"emote_data.json\";\n const file = data__WEBPACK_IMPORTED_MODULE_1__[\"Config\"].dataPath + emoteFile;\n\n const exists = _fs.existsSync(file);\n\n if (exists) _fs.unlinkSync(file);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"emoteCacheDate\", new Date().toJSON());\n\n for (const category in Emotes) Object.assign(Emotes, {\n [category]: {}\n });\n }\n\n}());//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/builtins/emotes.js?6a04"],"names":["Emotes","TwitchGlobal","TwitchSubscriber","BTTV","FrankerFaceZ","BTTV2","bdEmoteSettingIDs","blacklist","overrides","modifiers","EmoteModule","Builtin","name","collection","category","id","categories","Object","keys","filter","k","isCategoryEnabled","get","MessageContentComponent","WebpackModules","getModule","m","defaultProps","hasOwnProperty","favorites","favoriteEmotes","getCategory","initialize","fe","DataStore","getBDData","JSON","parse","window","atob","saveFavorites","addFavorite","bind","removeFavorite","enabled","Settings","registerCollection","EmoteConfig","title","Strings","clearEmotes","onClick","clearEmoteData","loadEmoteData","EmoteInfo","Events","on","disabled","off","removeCollection","emptyEmotes","cancelEmoteRender","url","isFavorite","setBDData","btoa","stringify","cat","assign","patchMessageContent","after","prototype","thisObj","args","retVal","props","t","a","returnValue","length","markup","children","nodes","n","node","words","split","c","clen","w","wlen","emote","emoteSplit","emoteName","emoteModifier","emoteOverride","slice","includes","current","results","match","RegExp","Utilities","escape","pre","substring","index","post","emoteComponent","DiscordModules","React","createElement","BDEmote","modifier","splice","onlyEmotes","every","r","replace","type","emojiName","jumboable","emoteInfo","emotesLoaded","_fs","require","emoteFile","file","Config","dataPath","exists","existsSync","isCacheValid","Toasts","show","log","data","Promise","resolve","readFile","err","content","parsed","testJSON","isValid","e","variable","dispatch","unlinkSync","downloading","setTimeout","downloadEmotes","downloaded","writeFileSync","stacktrace","emoteMeta","request","options","timeout","json","reject","error","response","parsedData","backup","backupParser","parser","getEmoteURL","getBlacklist","$","getJSON","push","cacheLength","cacheDate","Date","currentDate","daysBetween","Math","round","abs","getTime","toJSON"],"mappings":"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","file":"./src/builtins/emotes.js.js","sourcesContent":["import Builtin from \"../structs/builtin\";\r\n\r\nimport {Config, EmoteInfo, EmoteConfig} from \"data\";\r\nimport {Utilities, WebpackModules, DataStore, DiscordModules, Events, Settings, Strings} from \"modules\";\r\nimport BDEmote from \"../ui/emote\";\r\nimport Toasts from \"../ui/toasts\";\r\n// import EmoteMenu from \"./emotemenu\";\r\n\r\nconst Emotes = {\r\n    TwitchGlobal: {},\r\n    TwitchSubscriber: {},\r\n    BTTV: {},\r\n    FrankerFaceZ: {},\r\n    BTTV2: {}\r\n};\r\n\r\nconst bdEmoteSettingIDs = {\r\n    TwitchGlobal: \"twitch\",\r\n    TwitchSubscriber: \"twitch\",\r\n    BTTV: \"bttv\",\r\n    FrankerFaceZ: \"ffz\",\r\n    BTTV2: \"bttv\"\r\n};\r\n\r\nconst blacklist = [];\r\nconst overrides = [\"twitch\", \"bttv\", \"ffz\"];\r\nconst modifiers = [\"flip\", \"spin\", \"pulse\", \"spin2\", \"spin3\", \"1spin\", \"2spin\", \"3spin\", \"tr\", \"bl\", \"br\", \"shake\", \"shake2\", \"shake3\", \"flap\"];\r\n\r\nexport default new class EmoteModule extends Builtin {\r\n    get name() {return \"Emotes\";}\r\n    get collection() {return \"settings\";}\r\n    get category() {return \"general\";}\r\n    get id() {return \"emotes\";}\r\n    get categories() { return Object.keys(bdEmoteSettingIDs).filter(k => this.isCategoryEnabled(bdEmoteSettingIDs[k])); }\r\n\r\n    isCategoryEnabled(id) {return super.get(\"emotes\", \"categories\", id);}\r\n\r\n    get(id) {return super.get(\"emotes\", \"general\", id);}\r\n\r\n    get MessageContentComponent() {return WebpackModules.getModule(m => m.defaultProps && m.defaultProps.hasOwnProperty(\"disableButtons\"));}\r\n\r\n    get Emotes() {return Emotes;}\r\n    get TwitchGlobal() {return Emotes.TwitchGlobal;}\r\n    get TwitchSubscriber() {return Emotes.TwitchSubscriber;}\r\n    get BTTV() {return Emotes.BTTV;}\r\n    get FrankerFaceZ() {return Emotes.FrankerFaceZ;}\r\n    get BTTV2() {return Emotes.BTTV2;}\r\n    get blacklist() {return blacklist;}\r\n    get favorites() {return this.favoriteEmotes;}\r\n\r\n    getCategory(category) {\r\n        return Emotes[category];\r\n    }\r\n\r\n    initialize() {\r\n        super.initialize();\r\n        this.favoriteEmotes = {};\r\n        const fe = DataStore.getBDData(\"bdfavemotes\");\r\n        if (fe !== \"\" && fe !== null) this.favoriteEmotes = JSON.parse(window.atob(fe));\r\n        this.saveFavorites();\r\n        this.addFavorite = this.addFavorite.bind(this);\r\n        this.removeFavorite = this.removeFavorite.bind(this);\r\n        // EmoteConfig;\r\n        // emoteCollection.button = {title: \"Clear Emote Cache\", onClick: () => { this.clearEmoteData(); this.loadEmoteData(EmoteInfo); }};\r\n    }\r\n\r\n    async enabled() {\r\n        Settings.registerCollection(\"emotes\", \"Emotes\", EmoteConfig, {title: Strings.Emotes.clearEmotes, onClick: () => { this.clearEmoteData(); this.loadEmoteData(EmoteInfo); }});\r\n        // Disable emote module for now because it's annoying and slow\r\n        // await this.getBlacklist();\r\n        // await this.loadEmoteData(EmoteInfo);\r\n\r\n        // while (!this.MessageContentComponent) await new Promise(resolve => setTimeout(resolve, 100));\r\n        // this.patchMessageContent();\r\n        Events.on(\"emotes-favorite-added\", this.addFavorite);\r\n        Events.on(\"emotes-favorite-removed\", this.removeFavorite);\r\n    }\r\n\r\n    disabled() {\r\n        Events.off(\"emotes-favorite-added\", this.addFavorite);\r\n        Events.off(\"emotes-favorite-removed\", this.removeFavorite);\r\n        Settings.removeCollection(\"emotes\");\r\n        this.emptyEmotes();\r\n        if (!this.cancelEmoteRender) return;\r\n        this.cancelEmoteRender();\r\n        delete this.cancelEmoteRender;\r\n    }\r\n\r\n    addFavorite(name, url) {\r\n        if (!this.favoriteEmotes.hasOwnProperty(name)) this.favoriteEmotes[name] = url;\r\n        this.saveFavorites();\r\n    }\r\n\r\n    removeFavorite(name) {\r\n        if (!this.favoriteEmotes.hasOwnProperty(name)) return;\r\n        delete this.favoriteEmotes[name];\r\n        this.saveFavorites();\r\n    }\r\n\r\n    isFavorite(name) {\r\n        return this.favoriteEmotes.hasOwnProperty(name);\r\n    }\r\n\r\n    saveFavorites() {\r\n        DataStore.setBDData(\"bdfavemotes\", window.btoa(JSON.stringify(this.favoriteEmotes)));\r\n    }\r\n\r\n    emptyEmotes() {\r\n        for (const cat in Emotes) Object.assign(Emotes, {[cat]: {}});\r\n    }\r\n\r\n    patchMessageContent() {\r\n        if (this.cancelEmoteRender) return;\r\n        this.cancelEmoteRender = this.after(this.MessageContentComponent.prototype, \"render\", (thisObj, args, retVal) => {\r\n            this.after(retVal.props, \"children\", (t, a, returnValue) => {\r\n                if (this.categories.length == 0) return;\r\n                const markup = returnValue.props.children[1];\r\n                if (!markup.props.children) return;\r\n                const nodes = markup.props.children[1];\r\n                if (!nodes || !nodes.length) return;\r\n                for (let n = 0; n < nodes.length; n++) {\r\n                    const node = nodes[n];\r\n                    if (typeof(node) !== \"string\") continue;\r\n                    const words = node.split(/([^\\s]+)([\\s]|$)/g);\r\n                    for (let c = 0, clen = this.categories.length; c < clen; c++) {\r\n                        for (let w = 0, wlen = words.length; w < wlen; w++) {\r\n                            const emote = words[w];\r\n                            const emoteSplit = emote.split(\":\");\r\n                            const emoteName = emoteSplit[0];\r\n                            let emoteModifier = emoteSplit[1] ? emoteSplit[1] : \"\";\r\n                            let emoteOverride = emoteModifier.slice(0);\r\n\r\n                            if (emoteName.length < 4 || blacklist.includes(emoteName)) continue;\r\n                            if (!modifiers.includes(emoteModifier) || !Settings.get(this.category, \"general\", \"modifiers\")) emoteModifier = \"\";\r\n                            if (!overrides.includes(emoteOverride)) emoteOverride = \"\";\r\n                            else emoteModifier = emoteOverride;\r\n\r\n                            let current = this.categories[c];\r\n                            if (emoteOverride === \"twitch\") {\r\n                                if (Emotes.TwitchGlobal[emoteName]) current = \"TwitchGlobal\";\r\n                                else if (Emotes.TwitchSubscriber[emoteName]) current = \"TwitchSubscriber\";\r\n                            }\r\n                            else if (emoteOverride === \"bttv\") {\r\n                                if (Emotes.BTTV[emoteName]) current = \"BTTV\";\r\n                                else if (Emotes.BTTV2[emoteName]) current = \"BTTV2\";\r\n                            }\r\n                            else if (emoteOverride === \"ffz\") {\r\n                                if (Emotes.FrankerFaceZ[emoteName]) current = \"FrankerFaceZ\";\r\n                            }\r\n\r\n                            if (!Emotes[current][emoteName] || !Settings.get(this.category, \"categories\", bdEmoteSettingIDs[current])) continue;\r\n                            const results = nodes[n].match(new RegExp(`([\\\\s]|^)${Utilities.escape(emoteModifier ? emoteName + \":\" + emoteModifier : emoteName)}([\\\\s]|$)`));\r\n                            if (!results) continue;\r\n                            const pre = nodes[n].substring(0, results.index + results[1].length);\r\n                            const post = nodes[n].substring(results.index + results[0].length - results[2].length);\r\n                            nodes[n] = pre;\r\n                            const emoteComponent = DiscordModules.React.createElement(BDEmote, {name: emoteName, url: Emotes[current][emoteName], modifier: emoteModifier, isFavorite: this.isFavorite(emoteName)});\r\n                            nodes.splice(n + 1, 0, post);\r\n                            nodes.splice(n + 1, 0, emoteComponent);\r\n                        }\r\n                    }\r\n                }\r\n                const onlyEmotes = nodes.every(r => {\r\n                    if (typeof(r) == \"string\" && r.replace(/\\s*/, \"\") == \"\") return true;\r\n                    else if (r.type && r.type.name == \"BDEmote\") return true;\r\n                    else if (r.props && r.props.children && r.props.children.props && r.props.children.props.emojiName) return true;\r\n                    return false;\r\n                });\r\n                if (!onlyEmotes) return;\r\n\r\n                for (const node of nodes) {\r\n                    if (typeof(node) != \"object\") continue;\r\n                    if (node.type.name == \"BDEmote\") node.props.jumboable = true;\r\n                    else if (node.props && node.props.children && node.props.children.props && node.props.children.props.emojiName) node.props.children.props.jumboable = true;\r\n                }\r\n            });\r\n        });\r\n    }\r\n\r\n    async loadEmoteData(emoteInfo) {\r\n        this.emotesLoaded = false;\r\n        const _fs = require(\"fs\");\r\n        const emoteFile = \"emote_data.json\";\r\n        const file = Config.dataPath + emoteFile;\r\n        const exists = _fs.existsSync(file);\r\n\r\n        if (exists && this.isCacheValid()) {\r\n            Toasts.show(\"Loading emotes from cache.\", {type: \"info\"});\r\n            this.log(\"Loading emotes from local cache.\");\r\n\r\n            const data = await new Promise(resolve => {\r\n                _fs.readFile(file, \"utf8\", (err, content) => {\r\n                    this.log(\"Emotes loaded from cache.\");\r\n                    if (err) content = {};\r\n                    resolve(content);\r\n                });\r\n            });\r\n\r\n            const parsed = Utilities.testJSON(data);\r\n            let isValid = !!parsed;\r\n            if (isValid) Object.assign(Emotes, parsed);\r\n\r\n            for (const e in emoteInfo) {\r\n                isValid = Object.keys(Emotes[emoteInfo[e].variable]).length > 0;\r\n            }\r\n\r\n            if (isValid) {\r\n                Toasts.show(\"Emotes successfully loaded.\", {type: \"success\"});\r\n                this.emotesLoaded = true;\r\n                Events.dispatch(\"emotes-loaded\");\r\n                return;\r\n            }\r\n\r\n            this.log(\"Cache was corrupt, downloading...\");\r\n            _fs.unlinkSync(file);\r\n        }\r\n\r\n        if (!Settings.get(this.category, \"general\", \"download\")) return;\r\n        Toasts.show(Strings.Emotes.downloading, {type: \"info\"});\r\n\r\n        for (const e in emoteInfo) {\r\n            await new Promise(r => setTimeout(r, 1000));\r\n            const data = await this.downloadEmotes(emoteInfo[e]);\r\n            Emotes[emoteInfo[e].variable] = data;\r\n        }\r\n\r\n        Toasts.show(Strings.Emotes.downloaded, {type: \"success\"});\r\n\r\n        try { _fs.writeFileSync(file, JSON.stringify(Emotes), \"utf8\"); }\r\n        catch (err) { this.stacktrace(\"Could not save emote data.\", err); }\r\n\r\n        this.emotesLoaded = true;\r\n        Events.dispatch(\"emotes-loaded\");\r\n    }\r\n\r\n    downloadEmotes(emoteMeta) {\r\n        const request = require(\"request\");\r\n        const options = {\r\n            url: emoteMeta.url,\r\n            timeout: emoteMeta.timeout ? emoteMeta.timeout : 5000,\r\n            json: true\r\n        };\r\n\r\n        this.log(`Downloading: ${emoteMeta.variable} (${emoteMeta.url})`);\r\n\r\n        return new Promise((resolve, reject) => {\r\n            request(options, (error, response, parsedData) => {\r\n                if (error) {\r\n                    this.stacktrace(\"Could not download \" + emoteMeta.variable, error);\r\n                    if (emoteMeta.backup) {\r\n                        emoteMeta.url = emoteMeta.backup;\r\n                        emoteMeta.backup = null;\r\n                        if (emoteMeta.backupParser) emoteMeta.parser = emoteMeta.backupParser;\r\n                        return resolve(this.downloadEmotes(emoteMeta));\r\n                    }\r\n                    return reject({});\r\n                }\r\n\r\n                if (typeof(emoteMeta.parser) === \"function\") parsedData = emoteMeta.parser(parsedData);\r\n\r\n                for (const emote in parsedData) {\r\n                    if (emote.length < 4 || blacklist.includes(emote)) {\r\n                        delete parsedData[emote];\r\n                        continue;\r\n                    }\r\n                    parsedData[emote] = emoteMeta.getEmoteURL(parsedData[emote]);\r\n                }\r\n                resolve(parsedData);\r\n                this.log(\"Downloaded: \" + emoteMeta.variable);\r\n            });\r\n        });\r\n    }\r\n\r\n    getBlacklist() {\r\n        return new Promise(resolve => {\r\n            $.getJSON(`https://rauenzi.github.io/BetterDiscordApp/data/emotefilter.json`, function (data) {\r\n                resolve(blacklist.push(...data.blacklist));\r\n            });\r\n        });\r\n    }\r\n\r\n    isCacheValid() {\r\n        const cacheLength = DataStore.getBDData(\"emoteCacheDays\") || DataStore.setBDData(\"emoteCacheDays\", 7) || 7;\r\n        const cacheDate = new Date(DataStore.getBDData(\"emoteCacheDate\") || null);\r\n        const currentDate = new Date();\r\n        const daysBetween = Math.round(Math.abs((currentDate.getTime() - cacheDate.getTime()) / (24 * 60 * 60 * 1000)));\r\n        if (daysBetween > cacheLength) {\r\n            DataStore.setBDData(\"emoteCacheDate\", currentDate.toJSON());\r\n            return false;\r\n        }\r\n        return true;\r\n    }\r\n\r\n    clearEmoteData() {\r\n        const _fs = require(\"fs\");\r\n        const emoteFile = \"emote_data.json\";\r\n        const file = Config.dataPath + emoteFile;\r\n        const exists = _fs.existsSync(file);\r\n        if (exists) _fs.unlinkSync(file);\r\n        DataStore.setBDData(\"emoteCacheDate\", (new Date()).toJSON());\r\n        for (const category in Emotes) Object.assign(Emotes, {[category]: {}});\r\n    }\r\n};"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/builtins/emotes.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _structs_builtin__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../structs/builtin */ \"./src/structs/builtin.js\");\n/* harmony import */ var data__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! data */ \"./src/data/data.js\");\n/* harmony import */ var modules__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! modules */ \"./src/modules/modules.js\");\n/* harmony import */ var _ui_emote__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../ui/emote */ \"./src/ui/emote.js\");\n/* harmony import */ var _ui_toasts__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../ui/toasts */ \"./src/ui/toasts.js\");\n\n\n\n\n // import EmoteMenu from \"./emotemenu\";\n\nconst request = __webpack_require__(/*! request */ \"request\");\n\nconst Emotes = {\n TwitchGlobal: {},\n TwitchSubscriber: {},\n BTTV: {},\n FrankerFaceZ: {},\n BTTV2: {}\n};\nconst bdEmoteSettingIDs = {\n TwitchGlobal: \"twitch\",\n TwitchSubscriber: \"twitch\",\n BTTV: \"bttv\",\n FrankerFaceZ: \"ffz\",\n BTTV2: \"bttv\"\n};\nconst blacklist = [];\nconst overrides = [\"twitch\", \"bttv\", \"ffz\"];\nconst modifiers = [\"flip\", \"spin\", \"pulse\", \"spin2\", \"spin3\", \"1spin\", \"2spin\", \"3spin\", \"tr\", \"bl\", \"br\", \"shake\", \"shake2\", \"shake3\", \"flap\"];\n/* harmony default export */ __webpack_exports__[\"default\"] = (new class EmoteModule extends _structs_builtin__WEBPACK_IMPORTED_MODULE_0__[\"default\"] {\n get name() {\n return \"Emotes\";\n }\n\n get collection() {\n return \"settings\";\n }\n\n get category() {\n return \"general\";\n }\n\n get id() {\n return \"emotes\";\n }\n\n get categories() {\n return Object.keys(bdEmoteSettingIDs).filter(k => this.isCategoryEnabled(bdEmoteSettingIDs[k]));\n }\n\n isCategoryEnabled(id) {\n return super.get(\"emotes\", \"categories\", id);\n }\n\n get(id) {\n return super.get(\"emotes\", \"general\", id);\n }\n\n get MessageContentComponent() {\n return modules__WEBPACK_IMPORTED_MODULE_2__[\"WebpackModules\"].getModule(m => m.defaultProps && m.defaultProps.hasOwnProperty(\"disableButtons\"));\n }\n\n get Emotes() {\n return Emotes;\n }\n\n get TwitchGlobal() {\n return Emotes.TwitchGlobal;\n }\n\n get TwitchSubscriber() {\n return Emotes.TwitchSubscriber;\n }\n\n get BTTV() {\n return Emotes.BTTV;\n }\n\n get FrankerFaceZ() {\n return Emotes.FrankerFaceZ;\n }\n\n get BTTV2() {\n return Emotes.BTTV2;\n }\n\n get blacklist() {\n return blacklist;\n }\n\n get favorites() {\n return this.favoriteEmotes;\n }\n\n getCategory(category) {\n return Emotes[category];\n }\n\n initialize() {\n super.initialize();\n this.favoriteEmotes = {};\n const fe = modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].getBDData(\"bdfavemotes\");\n if (fe !== \"\" && fe !== null) this.favoriteEmotes = JSON.parse(window.atob(fe));\n this.saveFavorites();\n this.addFavorite = this.addFavorite.bind(this);\n this.removeFavorite = this.removeFavorite.bind(this); // EmoteConfig;\n // emoteCollection.button = {title: \"Clear Emote Cache\", onClick: () => { this.clearEmoteData(); this.loadEmoteData(EmoteInfo); }};\n }\n\n async enabled() {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].registerCollection(\"emotes\", \"Emotes\", data__WEBPACK_IMPORTED_MODULE_1__[\"EmoteConfig\"], {\n title: modules__WEBPACK_IMPORTED_MODULE_2__[\"Strings\"].Emotes.clearEmotes,\n onClick: () => {\n this.clearEmoteData();\n this.loadEmoteData(data__WEBPACK_IMPORTED_MODULE_1__[\"EmoteInfo\"]);\n }\n }); // Disable emote module for now because it's annoying and slow\n\n await this.getBlacklist();\n await this.loadEmoteData(data__WEBPACK_IMPORTED_MODULE_1__[\"EmoteInfo\"]);\n\n while (!this.MessageContentComponent) await new Promise(resolve => setTimeout(resolve, 100));\n\n this.patchMessageContent();\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].on(\"emotes-favorite-added\", this.addFavorite);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].on(\"emotes-favorite-removed\", this.removeFavorite);\n }\n\n disabled() {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].off(\"emotes-favorite-added\", this.addFavorite);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].off(\"emotes-favorite-removed\", this.removeFavorite);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].removeCollection(\"emotes\");\n this.emptyEmotes();\n if (!this.cancelEmoteRender) return;\n this.cancelEmoteRender();\n delete this.cancelEmoteRender;\n }\n\n addFavorite(name, url) {\n if (!this.favoriteEmotes.hasOwnProperty(name)) this.favoriteEmotes[name] = url;\n this.saveFavorites();\n }\n\n removeFavorite(name) {\n if (!this.favoriteEmotes.hasOwnProperty(name)) return;\n delete this.favoriteEmotes[name];\n this.saveFavorites();\n }\n\n isFavorite(name) {\n return this.favoriteEmotes.hasOwnProperty(name);\n }\n\n saveFavorites() {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"bdfavemotes\", window.btoa(JSON.stringify(this.favoriteEmotes)));\n }\n\n emptyEmotes() {\n for (const cat in Emotes) Object.assign(Emotes, {\n [cat]: {}\n });\n }\n\n patchMessageContent() {\n if (this.cancelEmoteRender) return;\n this.cancelEmoteRender = this.after(this.MessageContentComponent.prototype, \"render\", (thisObj, args, retVal) => {\n this.after(retVal.props, \"children\", (t, a, returnValue) => {\n if (this.categories.length == 0) return;\n const markup = returnValue.props.children[1];\n if (!markup.props.children) return;\n const nodes = markup.props.children[1];\n if (!nodes || !nodes.length) return;\n\n for (let n = 0; n < nodes.length; n++) {\n const node = nodes[n];\n if (typeof node !== \"string\") continue;\n const words = node.split(/([^\\s]+)([\\s]|$)/g);\n\n for (let c = 0, clen = this.categories.length; c < clen; c++) {\n for (let w = 0, wlen = words.length; w < wlen; w++) {\n const emote = words[w];\n const emoteSplit = emote.split(\":\");\n const emoteName = emoteSplit[0];\n let emoteModifier = emoteSplit[1] ? emoteSplit[1] : \"\";\n let emoteOverride = emoteModifier.slice(0);\n if (emoteName.length < 4 || blacklist.includes(emoteName)) continue;\n if (!modifiers.includes(emoteModifier) || !modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].get(this.category, \"general\", \"modifiers\")) emoteModifier = \"\";\n if (!overrides.includes(emoteOverride)) emoteOverride = \"\";else emoteModifier = emoteOverride;\n let current = this.categories[c];\n\n if (emoteOverride === \"twitch\") {\n if (Emotes.TwitchGlobal[emoteName]) current = \"TwitchGlobal\";else if (Emotes.TwitchSubscriber[emoteName]) current = \"TwitchSubscriber\";\n } else if (emoteOverride === \"bttv\") {\n if (Emotes.BTTV[emoteName]) current = \"BTTV\";else if (Emotes.BTTV2[emoteName]) current = \"BTTV2\";\n } else if (emoteOverride === \"ffz\") {\n if (Emotes.FrankerFaceZ[emoteName]) current = \"FrankerFaceZ\";\n }\n\n if (!Emotes[current][emoteName] || !modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].get(this.category, \"categories\", bdEmoteSettingIDs[current])) continue;\n const results = nodes[n].match(new RegExp(`([\\\\s]|^)${modules__WEBPACK_IMPORTED_MODULE_2__[\"Utilities\"].escape(emoteModifier ? emoteName + \":\" + emoteModifier : emoteName)}([\\\\s]|$)`));\n if (!results) continue;\n const pre = nodes[n].substring(0, results.index + results[1].length);\n const post = nodes[n].substring(results.index + results[0].length - results[2].length);\n nodes[n] = pre;\n const emoteComponent = modules__WEBPACK_IMPORTED_MODULE_2__[\"DiscordModules\"].React.createElement(_ui_emote__WEBPACK_IMPORTED_MODULE_3__[\"default\"], {\n name: emoteName,\n url: Emotes[current][emoteName],\n modifier: emoteModifier,\n isFavorite: this.isFavorite(emoteName)\n });\n nodes.splice(n + 1, 0, post);\n nodes.splice(n + 1, 0, emoteComponent);\n }\n }\n }\n\n const onlyEmotes = nodes.every(r => {\n if (typeof r == \"string\" && r.replace(/\\s*/, \"\") == \"\") return true;else if (r.type && r.type.name == \"BDEmote\") return true;else if (r.props && r.props.children && r.props.children.props && r.props.children.props.emojiName) return true;\n return false;\n });\n if (!onlyEmotes) return;\n\n for (const node of nodes) {\n if (typeof node != \"object\") continue;\n if (node.type.name == \"BDEmote\") node.props.jumboable = true;else if (node.props && node.props.children && node.props.children.props && node.props.children.props.emojiName) node.props.children.props.jumboable = true;\n }\n });\n });\n }\n\n async loadEmoteData(emoteInfo) {\n this.emotesLoaded = false;\n\n const _fs = __webpack_require__(/*! fs */ \"fs\");\n\n const emoteFile = \"emote_data.json\";\n const file = data__WEBPACK_IMPORTED_MODULE_1__[\"Config\"].dataPath + emoteFile;\n\n const exists = _fs.existsSync(file);\n\n if (exists && this.isCacheValid()) {\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(\"Loading emotes from cache.\", {\n type: \"info\"\n });\n this.log(\"Loading emotes from local cache.\");\n const data = await new Promise(resolve => {\n _fs.readFile(file, \"utf8\", (err, content) => {\n this.log(\"Emotes loaded from cache.\");\n if (err) content = {};\n resolve(content);\n });\n });\n const parsed = modules__WEBPACK_IMPORTED_MODULE_2__[\"Utilities\"].testJSON(data);\n let isValid = !!parsed;\n if (isValid) Object.assign(Emotes, parsed);\n\n for (const e in emoteInfo) {\n isValid = Object.keys(Emotes[emoteInfo[e].variable]).length > 0;\n }\n\n if (isValid) {\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(\"Emotes successfully loaded.\", {\n type: \"success\"\n });\n this.emotesLoaded = true;\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].dispatch(\"emotes-loaded\");\n return;\n }\n\n this.log(\"Cache was corrupt, downloading...\");\n\n _fs.unlinkSync(file);\n }\n\n if (!modules__WEBPACK_IMPORTED_MODULE_2__[\"Settings\"].get(this.category, \"general\", \"download\")) return;\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(modules__WEBPACK_IMPORTED_MODULE_2__[\"Strings\"].Emotes.downloading, {\n type: \"info\"\n });\n\n for (const e in emoteInfo) {\n await new Promise(r => setTimeout(r, 1000));\n const data = await this.downloadEmotes(emoteInfo[e]);\n Emotes[emoteInfo[e].variable] = data;\n }\n\n _ui_toasts__WEBPACK_IMPORTED_MODULE_4__[\"default\"].show(modules__WEBPACK_IMPORTED_MODULE_2__[\"Strings\"].Emotes.downloaded, {\n type: \"success\"\n });\n\n try {\n _fs.writeFileSync(file, JSON.stringify(Emotes), \"utf8\");\n } catch (err) {\n this.stacktrace(\"Could not save emote data.\", err);\n }\n\n this.emotesLoaded = true;\n modules__WEBPACK_IMPORTED_MODULE_2__[\"Events\"].dispatch(\"emotes-loaded\");\n }\n\n downloadEmotes(emoteMeta) {\n const repoFile = modules__WEBPACK_IMPORTED_MODULE_2__[\"Utilities\"].repoUrl(`data/emotes/${emoteMeta.variable.toLowerCase()}.json`);\n if (emoteMeta.url && !emoteMeta.backup) emoteMeta.backup = repoFile;\n if (!emoteMeta.url) emoteMeta.url = repoFile;\n const options = {\n url: emoteMeta.url,\n timeout: emoteMeta.timeout ? emoteMeta.timeout : 5000,\n json: true\n };\n this.log(`Downloading: ${emoteMeta.variable} (${emoteMeta.url})`);\n return new Promise((resolve, reject) => {\n request.get(options, (error, response, parsedData) => {\n if (error) {\n this.stacktrace(\"Could not download \" + emoteMeta.variable, error);\n\n if (emoteMeta.backup || emoteMeta.url) {\n emoteMeta.url = emoteMeta.backup;\n emoteMeta.backup = null;\n if (emoteMeta.backupParser) emoteMeta.parser = emoteMeta.backupParser;\n return resolve(this.downloadEmotes(emoteMeta));\n }\n\n return reject({});\n }\n\n if (typeof emoteMeta.parser === \"function\") parsedData = emoteMeta.parser(parsedData);\n\n for (const emote in parsedData) {\n if (emote.length < 4 || blacklist.includes(emote)) {\n delete parsedData[emote];\n continue;\n }\n\n parsedData[emote] = emoteMeta.getEmoteURL(parsedData[emote]);\n }\n\n resolve(parsedData);\n this.log(\"Downloaded: \" + emoteMeta.variable);\n });\n });\n }\n\n getBlacklist() {\n return new Promise(resolve => {\n request.get({\n url: modules__WEBPACK_IMPORTED_MODULE_2__[\"Utilities\"].repoUrl(`data/emotes/blacklist.json`),\n json: true\n }, (err, resp, data) => {\n if (err || resp.statusCode != 200) return resolve();\n resolve(blacklist.push(...data));\n });\n });\n }\n\n isCacheValid() {\n const cacheLength = modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].getBDData(\"emoteCacheDays\") || modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"emoteCacheDays\", 7) || 7;\n const cacheDate = new Date(modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].getBDData(\"emoteCacheDate\") || null);\n const currentDate = new Date();\n const daysBetween = Math.round(Math.abs((currentDate.getTime() - cacheDate.getTime()) / (24 * 60 * 60 * 1000)));\n\n if (daysBetween > cacheLength) {\n modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"emoteCacheDate\", currentDate.toJSON());\n return false;\n }\n\n return true;\n }\n\n clearEmoteData() {\n const _fs = __webpack_require__(/*! fs */ \"fs\");\n\n const emoteFile = \"emote_data.json\";\n const file = data__WEBPACK_IMPORTED_MODULE_1__[\"Config\"].dataPath + emoteFile;\n\n const exists = _fs.existsSync(file);\n\n if (exists) _fs.unlinkSync(file);\n modules__WEBPACK_IMPORTED_MODULE_2__[\"DataStore\"].setBDData(\"emoteCacheDate\", new Date().toJSON());\n\n for (const category in Emotes) Object.assign(Emotes, {\n [category]: {}\n });\n }\n\n}());//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/builtins/emotes.js?6a04"],"names":["request","require","Emotes","TwitchGlobal","TwitchSubscriber","BTTV","FrankerFaceZ","BTTV2","bdEmoteSettingIDs","blacklist","overrides","modifiers","EmoteModule","Builtin","name","collection","category","id","categories","Object","keys","filter","k","isCategoryEnabled","get","MessageContentComponent","WebpackModules","getModule","m","defaultProps","hasOwnProperty","favorites","favoriteEmotes","getCategory","initialize","fe","DataStore","getBDData","JSON","parse","window","atob","saveFavorites","addFavorite","bind","removeFavorite","enabled","Settings","registerCollection","EmoteConfig","title","Strings","clearEmotes","onClick","clearEmoteData","loadEmoteData","EmoteInfo","getBlacklist","Promise","resolve","setTimeout","patchMessageContent","Events","on","disabled","off","removeCollection","emptyEmotes","cancelEmoteRender","url","isFavorite","setBDData","btoa","stringify","cat","assign","after","prototype","thisObj","args","retVal","props","t","a","returnValue","length","markup","children","nodes","n","node","words","split","c","clen","w","wlen","emote","emoteSplit","emoteName","emoteModifier","emoteOverride","slice","includes","current","results","match","RegExp","Utilities","escape","pre","substring","index","post","emoteComponent","DiscordModules","React","createElement","BDEmote","modifier","splice","onlyEmotes","every","r","replace","type","emojiName","jumboable","emoteInfo","emotesLoaded","_fs","emoteFile","file","Config","dataPath","exists","existsSync","isCacheValid","Toasts","show","log","data","readFile","err","content","parsed","testJSON","isValid","e","variable","dispatch","unlinkSync","downloading","downloadEmotes","downloaded","writeFileSync","stacktrace","emoteMeta","repoFile","repoUrl","toLowerCase","backup","options","timeout","json","reject","error","response","parsedData","backupParser","parser","getEmoteURL","resp","statusCode","push","cacheLength","cacheDate","Date","currentDate","daysBetween","Math","round","abs","getTime","toJSON"],"mappings":"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","file":"./src/builtins/emotes.js.js","sourcesContent":["import Builtin from \"../structs/builtin\";\r\n\r\nimport {Config, EmoteInfo, EmoteConfig} from \"data\";\r\nimport {Utilities, WebpackModules, DataStore, DiscordModules, Events, Settings, Strings} from \"modules\";\r\nimport BDEmote from \"../ui/emote\";\r\nimport Toasts from \"../ui/toasts\";\r\n// import EmoteMenu from \"./emotemenu\";\r\nconst request = require(\"request\");\r\n\r\nconst Emotes = {\r\n    TwitchGlobal: {},\r\n    TwitchSubscriber: {},\r\n    BTTV: {},\r\n    FrankerFaceZ: {},\r\n    BTTV2: {}\r\n};\r\n\r\nconst bdEmoteSettingIDs = {\r\n    TwitchGlobal: \"twitch\",\r\n    TwitchSubscriber: \"twitch\",\r\n    BTTV: \"bttv\",\r\n    FrankerFaceZ: \"ffz\",\r\n    BTTV2: \"bttv\"\r\n};\r\n\r\nconst blacklist = [];\r\nconst overrides = [\"twitch\", \"bttv\", \"ffz\"];\r\nconst modifiers = [\"flip\", \"spin\", \"pulse\", \"spin2\", \"spin3\", \"1spin\", \"2spin\", \"3spin\", \"tr\", \"bl\", \"br\", \"shake\", \"shake2\", \"shake3\", \"flap\"];\r\n\r\nexport default new class EmoteModule extends Builtin {\r\n    get name() {return \"Emotes\";}\r\n    get collection() {return \"settings\";}\r\n    get category() {return \"general\";}\r\n    get id() {return \"emotes\";}\r\n    get categories() { return Object.keys(bdEmoteSettingIDs).filter(k => this.isCategoryEnabled(bdEmoteSettingIDs[k])); }\r\n\r\n    isCategoryEnabled(id) {return super.get(\"emotes\", \"categories\", id);}\r\n\r\n    get(id) {return super.get(\"emotes\", \"general\", id);}\r\n\r\n    get MessageContentComponent() {return WebpackModules.getModule(m => m.defaultProps && m.defaultProps.hasOwnProperty(\"disableButtons\"));}\r\n\r\n    get Emotes() {return Emotes;}\r\n    get TwitchGlobal() {return Emotes.TwitchGlobal;}\r\n    get TwitchSubscriber() {return Emotes.TwitchSubscriber;}\r\n    get BTTV() {return Emotes.BTTV;}\r\n    get FrankerFaceZ() {return Emotes.FrankerFaceZ;}\r\n    get BTTV2() {return Emotes.BTTV2;}\r\n    get blacklist() {return blacklist;}\r\n    get favorites() {return this.favoriteEmotes;}\r\n\r\n    getCategory(category) {\r\n        return Emotes[category];\r\n    }\r\n\r\n    initialize() {\r\n        super.initialize();\r\n        this.favoriteEmotes = {};\r\n        const fe = DataStore.getBDData(\"bdfavemotes\");\r\n        if (fe !== \"\" && fe !== null) this.favoriteEmotes = JSON.parse(window.atob(fe));\r\n        this.saveFavorites();\r\n        this.addFavorite = this.addFavorite.bind(this);\r\n        this.removeFavorite = this.removeFavorite.bind(this);\r\n        // EmoteConfig;\r\n        // emoteCollection.button = {title: \"Clear Emote Cache\", onClick: () => { this.clearEmoteData(); this.loadEmoteData(EmoteInfo); }};\r\n    }\r\n\r\n    async enabled() {\r\n        Settings.registerCollection(\"emotes\", \"Emotes\", EmoteConfig, {title: Strings.Emotes.clearEmotes, onClick: () => { this.clearEmoteData(); this.loadEmoteData(EmoteInfo); }});\r\n        // Disable emote module for now because it's annoying and slow\r\n        await this.getBlacklist();\r\n        await this.loadEmoteData(EmoteInfo);\r\n\r\n        while (!this.MessageContentComponent) await new Promise(resolve => setTimeout(resolve, 100));\r\n        this.patchMessageContent();\r\n        Events.on(\"emotes-favorite-added\", this.addFavorite);\r\n        Events.on(\"emotes-favorite-removed\", this.removeFavorite);\r\n    }\r\n\r\n    disabled() {\r\n        Events.off(\"emotes-favorite-added\", this.addFavorite);\r\n        Events.off(\"emotes-favorite-removed\", this.removeFavorite);\r\n        Settings.removeCollection(\"emotes\");\r\n        this.emptyEmotes();\r\n        if (!this.cancelEmoteRender) return;\r\n        this.cancelEmoteRender();\r\n        delete this.cancelEmoteRender;\r\n    }\r\n\r\n    addFavorite(name, url) {\r\n        if (!this.favoriteEmotes.hasOwnProperty(name)) this.favoriteEmotes[name] = url;\r\n        this.saveFavorites();\r\n    }\r\n\r\n    removeFavorite(name) {\r\n        if (!this.favoriteEmotes.hasOwnProperty(name)) return;\r\n        delete this.favoriteEmotes[name];\r\n        this.saveFavorites();\r\n    }\r\n\r\n    isFavorite(name) {\r\n        return this.favoriteEmotes.hasOwnProperty(name);\r\n    }\r\n\r\n    saveFavorites() {\r\n        DataStore.setBDData(\"bdfavemotes\", window.btoa(JSON.stringify(this.favoriteEmotes)));\r\n    }\r\n\r\n    emptyEmotes() {\r\n        for (const cat in Emotes) Object.assign(Emotes, {[cat]: {}});\r\n    }\r\n\r\n    patchMessageContent() {\r\n        if (this.cancelEmoteRender) return;\r\n        this.cancelEmoteRender = this.after(this.MessageContentComponent.prototype, \"render\", (thisObj, args, retVal) => {\r\n            this.after(retVal.props, \"children\", (t, a, returnValue) => {\r\n                if (this.categories.length == 0) return;\r\n                const markup = returnValue.props.children[1];\r\n                if (!markup.props.children) return;\r\n                const nodes = markup.props.children[1];\r\n                if (!nodes || !nodes.length) return;\r\n                for (let n = 0; n < nodes.length; n++) {\r\n                    const node = nodes[n];\r\n                    if (typeof(node) !== \"string\") continue;\r\n                    const words = node.split(/([^\\s]+)([\\s]|$)/g);\r\n                    for (let c = 0, clen = this.categories.length; c < clen; c++) {\r\n                        for (let w = 0, wlen = words.length; w < wlen; w++) {\r\n                            const emote = words[w];\r\n                            const emoteSplit = emote.split(\":\");\r\n                            const emoteName = emoteSplit[0];\r\n                            let emoteModifier = emoteSplit[1] ? emoteSplit[1] : \"\";\r\n                            let emoteOverride = emoteModifier.slice(0);\r\n\r\n                            if (emoteName.length < 4 || blacklist.includes(emoteName)) continue;\r\n                            if (!modifiers.includes(emoteModifier) || !Settings.get(this.category, \"general\", \"modifiers\")) emoteModifier = \"\";\r\n                            if (!overrides.includes(emoteOverride)) emoteOverride = \"\";\r\n                            else emoteModifier = emoteOverride;\r\n\r\n                            let current = this.categories[c];\r\n                            if (emoteOverride === \"twitch\") {\r\n                                if (Emotes.TwitchGlobal[emoteName]) current = \"TwitchGlobal\";\r\n                                else if (Emotes.TwitchSubscriber[emoteName]) current = \"TwitchSubscriber\";\r\n                            }\r\n                            else if (emoteOverride === \"bttv\") {\r\n                                if (Emotes.BTTV[emoteName]) current = \"BTTV\";\r\n                                else if (Emotes.BTTV2[emoteName]) current = \"BTTV2\";\r\n                            }\r\n                            else if (emoteOverride === \"ffz\") {\r\n                                if (Emotes.FrankerFaceZ[emoteName]) current = \"FrankerFaceZ\";\r\n                            }\r\n\r\n                            if (!Emotes[current][emoteName] || !Settings.get(this.category, \"categories\", bdEmoteSettingIDs[current])) continue;\r\n                            const results = nodes[n].match(new RegExp(`([\\\\s]|^)${Utilities.escape(emoteModifier ? emoteName + \":\" + emoteModifier : emoteName)}([\\\\s]|$)`));\r\n                            if (!results) continue;\r\n                            const pre = nodes[n].substring(0, results.index + results[1].length);\r\n                            const post = nodes[n].substring(results.index + results[0].length - results[2].length);\r\n                            nodes[n] = pre;\r\n                            const emoteComponent = DiscordModules.React.createElement(BDEmote, {name: emoteName, url: Emotes[current][emoteName], modifier: emoteModifier, isFavorite: this.isFavorite(emoteName)});\r\n                            nodes.splice(n + 1, 0, post);\r\n                            nodes.splice(n + 1, 0, emoteComponent);\r\n                        }\r\n                    }\r\n                }\r\n                const onlyEmotes = nodes.every(r => {\r\n                    if (typeof(r) == \"string\" && r.replace(/\\s*/, \"\") == \"\") return true;\r\n                    else if (r.type && r.type.name == \"BDEmote\") return true;\r\n                    else if (r.props && r.props.children && r.props.children.props && r.props.children.props.emojiName) return true;\r\n                    return false;\r\n                });\r\n                if (!onlyEmotes) return;\r\n\r\n                for (const node of nodes) {\r\n                    if (typeof(node) != \"object\") continue;\r\n                    if (node.type.name == \"BDEmote\") node.props.jumboable = true;\r\n                    else if (node.props && node.props.children && node.props.children.props && node.props.children.props.emojiName) node.props.children.props.jumboable = true;\r\n                }\r\n            });\r\n        });\r\n    }\r\n\r\n    async loadEmoteData(emoteInfo) {\r\n        this.emotesLoaded = false;\r\n        const _fs = require(\"fs\");\r\n        const emoteFile = \"emote_data.json\";\r\n        const file = Config.dataPath + emoteFile;\r\n        const exists = _fs.existsSync(file);\r\n\r\n        if (exists && this.isCacheValid()) {\r\n            Toasts.show(\"Loading emotes from cache.\", {type: \"info\"});\r\n            this.log(\"Loading emotes from local cache.\");\r\n\r\n            const data = await new Promise(resolve => {\r\n                _fs.readFile(file, \"utf8\", (err, content) => {\r\n                    this.log(\"Emotes loaded from cache.\");\r\n                    if (err) content = {};\r\n                    resolve(content);\r\n                });\r\n            });\r\n\r\n            const parsed = Utilities.testJSON(data);\r\n            let isValid = !!parsed;\r\n            if (isValid) Object.assign(Emotes, parsed);\r\n\r\n            for (const e in emoteInfo) {\r\n                isValid = Object.keys(Emotes[emoteInfo[e].variable]).length > 0;\r\n            }\r\n\r\n            if (isValid) {\r\n                Toasts.show(\"Emotes successfully loaded.\", {type: \"success\"});\r\n                this.emotesLoaded = true;\r\n                Events.dispatch(\"emotes-loaded\");\r\n                return;\r\n            }\r\n\r\n            this.log(\"Cache was corrupt, downloading...\");\r\n            _fs.unlinkSync(file);\r\n        }\r\n\r\n        if (!Settings.get(this.category, \"general\", \"download\")) return;\r\n        Toasts.show(Strings.Emotes.downloading, {type: \"info\"});\r\n\r\n        for (const e in emoteInfo) {\r\n            await new Promise(r => setTimeout(r, 1000));\r\n            const data = await this.downloadEmotes(emoteInfo[e]);\r\n            Emotes[emoteInfo[e].variable] = data;\r\n        }\r\n\r\n        Toasts.show(Strings.Emotes.downloaded, {type: \"success\"});\r\n\r\n        try { _fs.writeFileSync(file, JSON.stringify(Emotes), \"utf8\"); }\r\n        catch (err) { this.stacktrace(\"Could not save emote data.\", err); }\r\n\r\n        this.emotesLoaded = true;\r\n        Events.dispatch(\"emotes-loaded\");\r\n    }\r\n\r\n    downloadEmotes(emoteMeta) {\r\n        const repoFile = Utilities.repoUrl(`data/emotes/${emoteMeta.variable.toLowerCase()}.json`);\r\n        if (emoteMeta.url && !emoteMeta.backup) emoteMeta.backup = repoFile;\r\n        if (!emoteMeta.url) emoteMeta.url = repoFile;\r\n\r\n        const options = {\r\n            url: emoteMeta.url,\r\n            timeout: emoteMeta.timeout ? emoteMeta.timeout : 5000,\r\n            json: true\r\n        };\r\n\r\n        this.log(`Downloading: ${emoteMeta.variable} (${emoteMeta.url})`);\r\n\r\n        return new Promise((resolve, reject) => {\r\n            request.get(options, (error, response, parsedData) => {\r\n                if (error) {\r\n                    this.stacktrace(\"Could not download \" + emoteMeta.variable, error);\r\n                    if (emoteMeta.backup || emoteMeta.url) {\r\n                        emoteMeta.url = emoteMeta.backup;\r\n                        emoteMeta.backup = null;\r\n                        if (emoteMeta.backupParser) emoteMeta.parser = emoteMeta.backupParser;\r\n                        return resolve(this.downloadEmotes(emoteMeta));\r\n                    }\r\n                    return reject({});\r\n                }\r\n\r\n                if (typeof(emoteMeta.parser) === \"function\") parsedData = emoteMeta.parser(parsedData);\r\n\r\n                for (const emote in parsedData) {\r\n                    if (emote.length < 4 || blacklist.includes(emote)) {\r\n                        delete parsedData[emote];\r\n                        continue;\r\n                    }\r\n                    parsedData[emote] = emoteMeta.getEmoteURL(parsedData[emote]);\r\n                }\r\n                resolve(parsedData);\r\n                this.log(\"Downloaded: \" + emoteMeta.variable);\r\n            });\r\n        });\r\n    }\r\n\r\n    getBlacklist() {\r\n        return new Promise(resolve => {\r\n            request.get({url: Utilities.repoUrl(`data/emotes/blacklist.json`), json: true}, (err, resp, data) => {\r\n                if (err || resp.statusCode != 200) return resolve();\r\n                resolve(blacklist.push(...data));\r\n            });\r\n        });\r\n    }\r\n\r\n    isCacheValid() {\r\n        const cacheLength = DataStore.getBDData(\"emoteCacheDays\") || DataStore.setBDData(\"emoteCacheDays\", 7) || 7;\r\n        const cacheDate = new Date(DataStore.getBDData(\"emoteCacheDate\") || null);\r\n        const currentDate = new Date();\r\n        const daysBetween = Math.round(Math.abs((currentDate.getTime() - cacheDate.getTime()) / (24 * 60 * 60 * 1000)));\r\n        if (daysBetween > cacheLength) {\r\n            DataStore.setBDData(\"emoteCacheDate\", currentDate.toJSON());\r\n            return false;\r\n        }\r\n        return true;\r\n    }\r\n\r\n    clearEmoteData() {\r\n        const _fs = require(\"fs\");\r\n        const emoteFile = \"emote_data.json\";\r\n        const file = Config.dataPath + emoteFile;\r\n        const exists = _fs.existsSync(file);\r\n        if (exists) _fs.unlinkSync(file);\r\n        DataStore.setBDData(\"emoteCacheDate\", (new Date()).toJSON());\r\n        for (const category in Emotes) Object.assign(Emotes, {[category]: {}});\r\n    }\r\n};"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/builtins/emotes.js\n"); /***/ }), @@ -311,7 +311,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony default export */ /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony default export */ __webpack_exports__[\"default\"] = ({\n TwitchGlobal: {\n url: \"https://twitchemotes.com/api_cache/v3/global.json\",\n backup: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_twitch_global.json`,\n variable: \"TwitchGlobal\",\n getEmoteURL: e => `https://static-cdn.jtvnw.net/emoticons/v1/${e.id}/1.0`,\n getOldData: (url, name) => {\n return {\n id: url.match(/\\/([0-9]+)\\//)[1],\n code: name,\n emoticon_set: 0,\n description: null\n };\n }\n },\n TwitchSubscriber: {\n url: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_twitch_subscriber.json`,\n variable: \"TwitchSubscriber\",\n getEmoteURL: e => `https://static-cdn.jtvnw.net/emoticons/v1/${e}/1.0`,\n getOldData: url => url.match(/\\/([0-9]+)\\//)[1]\n },\n FrankerFaceZ: {\n url: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_ffz.json`,\n variable: \"FrankerFaceZ\",\n getEmoteURL: e => `https://cdn.frankerfacez.com/emoticon/${e}/1`,\n getOldData: url => url.match(/\\/([0-9]+)\\//)[1]\n },\n BTTV: {\n url: \"https://api.betterttv.net/emotes\",\n variable: \"BTTV\",\n parser: data => {\n const emotes = {};\n\n for (let e = 0, len = data.emotes.length; e < len; e++) {\n const emote = data.emotes[e];\n emotes[emote.regex] = emote.url;\n }\n\n return emotes;\n },\n getEmoteURL: e => `${e}`,\n getOldData: url => url\n },\n BTTV2: {\n url: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_bttv.json`,\n variable: \"BTTV2\",\n oldVariable: \"emotesBTTV2\",\n getEmoteURL: e => `https://cdn.betterttv.net/emote/${e}/1x`,\n getOldData: url => url.match(/emote\\/(.+)\\//)[1]\n }\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9Db3JlLy4vc3JjL2RhdGEvZW1vdGVzL2luZm8uanM/OWY4ZSJdLCJuYW1lcyI6WyJUd2l0Y2hHbG9iYWwiLCJ1cmwiLCJiYWNrdXAiLCJ2YXJpYWJsZSIsImdldEVtb3RlVVJMIiwiZSIsImlkIiwiZ2V0T2xkRGF0YSIsIm5hbWUiLCJtYXRjaCIsImNvZGUiLCJlbW90aWNvbl9zZXQiLCJkZXNjcmlwdGlvbiIsIlR3aXRjaFN1YnNjcmliZXIiLCJGcmFua2VyRmFjZVoiLCJCVFRWIiwicGFyc2VyIiwiZGF0YSIsImVtb3RlcyIsImxlbiIsImxlbmd0aCIsImVtb3RlIiwicmVnZXgiLCJCVFRWMiIsIm9sZFZhcmlhYmxlIl0sIm1hcHBpbmdzIjoiQUFBQTtBQUFlO0FBQ1hBLGNBQVksRUFBRTtBQUNWQyxPQUFHLEVBQUUsbURBREs7QUFFVkMsVUFBTSxFQUFHLDhFQUZDO0FBR1ZDLFlBQVEsRUFBRSxjQUhBO0FBSVZDLGVBQVcsRUFBR0MsQ0FBRCxJQUFRLDZDQUE0Q0EsQ0FBQyxDQUFDQyxFQUFHLE1BSjVEO0FBS1ZDLGNBQVUsRUFBRSxDQUFDTixHQUFELEVBQU1PLElBQU4sS0FBZTtBQUFFLGFBQU87QUFBQ0YsVUFBRSxFQUFFTCxHQUFHLENBQUNRLEtBQUosQ0FBVSxjQUFWLEVBQTBCLENBQTFCLENBQUw7QUFBbUNDLFlBQUksRUFBRUYsSUFBekM7QUFBK0NHLG9CQUFZLEVBQUUsQ0FBN0Q7QUFBZ0VDLG1CQUFXLEVBQUU7QUFBN0UsT0FBUDtBQUE0RjtBQUwvRyxHQURIO0FBUVhDLGtCQUFnQixFQUFFO0FBQ2RaLE9BQUcsRUFBRyxrRkFEUTtBQUVkRSxZQUFRLEVBQUUsa0JBRkk7QUFHZEMsZUFBVyxFQUFHQyxDQUFELElBQVEsNkNBQTRDQSxDQUFFLE1BSHJEO0FBSWRFLGNBQVUsRUFBR04sR0FBRCxJQUFTQSxHQUFHLENBQUNRLEtBQUosQ0FBVSxjQUFWLEVBQTBCLENBQTFCO0FBSlAsR0FSUDtBQWNYSyxjQUFZLEVBQUU7QUFDVmIsT0FBRyxFQUFHLG9FQURJO0FBRVZFLFlBQVEsRUFBRSxjQUZBO0FBR1ZDLGVBQVcsRUFBR0MsQ0FBRCxJQUFRLHlDQUF3Q0EsQ0FBRSxJQUhyRDtBQUlWRSxjQUFVLEVBQUdOLEdBQUQsSUFBU0EsR0FBRyxDQUFDUSxLQUFKLENBQVUsY0FBVixFQUEwQixDQUExQjtBQUpYLEdBZEg7QUFvQlhNLE1BQUksRUFBRTtBQUNGZCxPQUFHLEVBQUUsa0NBREg7QUFFRkUsWUFBUSxFQUFFLE1BRlI7QUFHRmEsVUFBTSxFQUFHQyxJQUFELElBQVU7QUFDZCxZQUFNQyxNQUFNLEdBQUcsRUFBZjs7QUFDQSxXQUFLLElBQUliLENBQUMsR0FBRyxDQUFSLEVBQVdjLEdBQUcsR0FBR0YsSUFBSSxDQUFDQyxNQUFMLENBQVlFLE1BQWxDLEVBQTBDZixDQUFDLEdBQUdjLEdBQTlDLEVBQW1EZCxDQUFDLEVBQXBELEVBQXdEO0FBQ3BELGNBQU1nQixLQUFLLEdBQUdKLElBQUksQ0FBQ0MsTUFBTCxDQUFZYixDQUFaLENBQWQ7QUFDQWEsY0FBTSxDQUFDRyxLQUFLLENBQUNDLEtBQVAsQ0FBTixHQUFzQkQsS0FBSyxDQUFDcEIsR0FBNUI7QUFDSDs7QUFDRCxhQUFPaUIsTUFBUDtBQUNILEtBVkM7QUFXRmQsZUFBVyxFQUFHQyxDQUFELElBQVEsR0FBRUEsQ0FBRSxFQVh2QjtBQVlGRSxjQUFVLEVBQUdOLEdBQUQsSUFBU0E7QUFabkIsR0FwQks7QUFrQ1hzQixPQUFLLEVBQUU7QUFDSHRCLE9BQUcsRUFBRyxxRUFESDtBQUVIRSxZQUFRLEVBQUUsT0FGUDtBQUdIcUIsZUFBVyxFQUFFLGFBSFY7QUFJSHBCLGVBQVcsRUFBR0MsQ0FBRCxJQUFRLG1DQUFrQ0EsQ0FBRSxLQUp0RDtBQUtIRSxjQUFVLEVBQUdOLEdBQUQsSUFBU0EsR0FBRyxDQUFDUSxLQUFKLENBQVUsZUFBVixFQUEyQixDQUEzQjtBQUxsQjtBQWxDSSxDQUFmIiwiZmlsZSI6Ii4vc3JjL2RhdGEvZW1vdGVzL2luZm8uanMuanMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgZGVmYXVsdCB7XHJcbiAgICBUd2l0Y2hHbG9iYWw6IHtcclxuICAgICAgICB1cmw6IFwiaHR0cHM6Ly90d2l0Y2hlbW90ZXMuY29tL2FwaV9jYWNoZS92My9nbG9iYWwuanNvblwiLFxyXG4gICAgICAgIGJhY2t1cDogYGh0dHBzOi8vcmF1ZW56aS5naXRodWIuaW8vQmV0dGVyRGlzY29yZEFwcC9kYXRhL2Vtb3RlZGF0YV90d2l0Y2hfZ2xvYmFsLmpzb25gLFxyXG4gICAgICAgIHZhcmlhYmxlOiBcIlR3aXRjaEdsb2JhbFwiLFxyXG4gICAgICAgIGdldEVtb3RlVVJMOiAoZSkgPT4gYGh0dHBzOi8vc3RhdGljLWNkbi5qdHZudy5uZXQvZW1vdGljb25zL3YxLyR7ZS5pZH0vMS4wYCxcclxuICAgICAgICBnZXRPbGREYXRhOiAodXJsLCBuYW1lKSA9PiB7IHJldHVybiB7aWQ6IHVybC5tYXRjaCgvXFwvKFswLTldKylcXC8vKVsxXSwgY29kZTogbmFtZSwgZW1vdGljb25fc2V0OiAwLCBkZXNjcmlwdGlvbjogbnVsbH07IH1cclxuICAgIH0sXHJcbiAgICBUd2l0Y2hTdWJzY3JpYmVyOiB7XHJcbiAgICAgICAgdXJsOiBgaHR0cHM6Ly9yYXVlbnppLmdpdGh1Yi5pby9CZXR0ZXJEaXNjb3JkQXBwL2RhdGEvZW1vdGVkYXRhX3R3aXRjaF9zdWJzY3JpYmVyLmpzb25gLFxyXG4gICAgICAgIHZhcmlhYmxlOiBcIlR3aXRjaFN1YnNjcmliZXJcIixcclxuICAgICAgICBnZXRFbW90ZVVSTDogKGUpID0+IGBodHRwczovL3N0YXRpYy1jZG4uanR2bncubmV0L2Vtb3RpY29ucy92MS8ke2V9LzEuMGAsXHJcbiAgICAgICAgZ2V0T2xkRGF0YTogKHVybCkgPT4gdXJsLm1hdGNoKC9cXC8oWzAtOV0rKVxcLy8pWzFdXHJcbiAgICB9LFxyXG4gICAgRnJhbmtlckZhY2VaOiB7XHJcbiAgICAgICAgdXJsOiBgaHR0cHM6Ly9yYXVlbnppLmdpdGh1Yi5pby9CZXR0ZXJEaXNjb3JkQXBwL2RhdGEvZW1vdGVkYXRhX2Zmei5qc29uYCxcclxuICAgICAgICB2YXJpYWJsZTogXCJGcmFua2VyRmFjZVpcIixcclxuICAgICAgICBnZXRFbW90ZVVSTDogKGUpID0+IGBodHRwczovL2Nkbi5mcmFua2VyZmFjZXouY29tL2Vtb3RpY29uLyR7ZX0vMWAsXHJcbiAgICAgICAgZ2V0T2xkRGF0YTogKHVybCkgPT4gdXJsLm1hdGNoKC9cXC8oWzAtOV0rKVxcLy8pWzFdXHJcbiAgICB9LFxyXG4gICAgQlRUVjoge1xyXG4gICAgICAgIHVybDogXCJodHRwczovL2FwaS5iZXR0ZXJ0dHYubmV0L2Vtb3Rlc1wiLFxyXG4gICAgICAgIHZhcmlhYmxlOiBcIkJUVFZcIixcclxuICAgICAgICBwYXJzZXI6IChkYXRhKSA9PiB7XHJcbiAgICAgICAgICAgIGNvbnN0IGVtb3RlcyA9IHt9O1xyXG4gICAgICAgICAgICBmb3IgKGxldCBlID0gMCwgbGVuID0gZGF0YS5lbW90ZXMubGVuZ3RoOyBlIDwgbGVuOyBlKyspIHtcclxuICAgICAgICAgICAgICAgIGNvbnN0IGVtb3RlID0gZGF0YS5lbW90ZXNbZV07XHJcbiAgICAgICAgICAgICAgICBlbW90ZXNbZW1vdGUucmVnZXhdID0gZW1vdGUudXJsO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIHJldHVybiBlbW90ZXM7XHJcbiAgICAgICAgfSxcclxuICAgICAgICBnZXRFbW90ZVVSTDogKGUpID0+IGAke2V9YCxcclxuICAgICAgICBnZXRPbGREYXRhOiAodXJsKSA9PiB1cmxcclxuICAgIH0sXHJcbiAgICBCVFRWMjoge1xyXG4gICAgICAgIHVybDogYGh0dHBzOi8vcmF1ZW56aS5naXRodWIuaW8vQmV0dGVyRGlzY29yZEFwcC9kYXRhL2Vtb3RlZGF0YV9idHR2Lmpzb25gLFxyXG4gICAgICAgIHZhcmlhYmxlOiBcIkJUVFYyXCIsXHJcbiAgICAgICAgb2xkVmFyaWFibGU6IFwiZW1vdGVzQlRUVjJcIixcclxuICAgICAgICBnZXRFbW90ZVVSTDogKGUpID0+IGBodHRwczovL2Nkbi5iZXR0ZXJ0dHYubmV0L2Vtb3RlLyR7ZX0vMXhgLFxyXG4gICAgICAgIGdldE9sZERhdGE6ICh1cmwpID0+IHVybC5tYXRjaCgvZW1vdGVcXC8oLispXFwvLylbMV1cclxuICAgIH1cclxufTsiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/data/emotes/info.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony default export */ __webpack_exports__[\"default\"] = ({\n TwitchGlobal: {\n url: \"https://twitchemotes.com/api_cache/v3/global.json\",\n variable: \"TwitchGlobal\",\n getEmoteURL: e => `https://static-cdn.jtvnw.net/emoticons/v1/${e.id}/1.0`,\n getOldData: (url, name) => {\n return {\n id: url.match(/\\/([0-9]+)\\//)[1],\n code: name,\n emoticon_set: 0,\n description: null\n };\n }\n },\n TwitchSubscriber: {\n variable: \"TwitchSubscriber\",\n getEmoteURL: e => `https://static-cdn.jtvnw.net/emoticons/v1/${e}/1.0`,\n getOldData: url => url.match(/\\/([0-9]+)\\//)[1]\n },\n FrankerFaceZ: {\n variable: \"FrankerFaceZ\",\n getEmoteURL: e => `https://cdn.frankerfacez.com/emoticon/${e}/1`,\n getOldData: url => url.match(/\\/([0-9]+)\\//)[1]\n },\n BTTV: {\n url: \"https://api.betterttv.net/emotes\",\n variable: \"BTTV\",\n parser: data => {\n const emotes = {};\n\n for (let e = 0, len = data.emotes.length; e < len; e++) {\n const emote = data.emotes[e];\n emotes[emote.regex] = emote.url;\n }\n\n return emotes;\n },\n getEmoteURL: e => `${e}`,\n getOldData: url => url\n },\n BTTV2: {\n variable: \"BTTV2\",\n oldVariable: \"emotesBTTV2\",\n getEmoteURL: e => `https://cdn.betterttv.net/emote/${e}/1x`,\n getOldData: url => url.match(/emote\\/(.+)\\//)[1]\n }\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9Db3JlLy4vc3JjL2RhdGEvZW1vdGVzL2luZm8uanM/OWY4ZSJdLCJuYW1lcyI6WyJUd2l0Y2hHbG9iYWwiLCJ1cmwiLCJ2YXJpYWJsZSIsImdldEVtb3RlVVJMIiwiZSIsImlkIiwiZ2V0T2xkRGF0YSIsIm5hbWUiLCJtYXRjaCIsImNvZGUiLCJlbW90aWNvbl9zZXQiLCJkZXNjcmlwdGlvbiIsIlR3aXRjaFN1YnNjcmliZXIiLCJGcmFua2VyRmFjZVoiLCJCVFRWIiwicGFyc2VyIiwiZGF0YSIsImVtb3RlcyIsImxlbiIsImxlbmd0aCIsImVtb3RlIiwicmVnZXgiLCJCVFRWMiIsIm9sZFZhcmlhYmxlIl0sIm1hcHBpbmdzIjoiQUFBQTtBQUFlO0FBQ1hBLGNBQVksRUFBRTtBQUNWQyxPQUFHLEVBQUUsbURBREs7QUFFVkMsWUFBUSxFQUFFLGNBRkE7QUFHVkMsZUFBVyxFQUFHQyxDQUFELElBQVEsNkNBQTRDQSxDQUFDLENBQUNDLEVBQUcsTUFINUQ7QUFJVkMsY0FBVSxFQUFFLENBQUNMLEdBQUQsRUFBTU0sSUFBTixLQUFlO0FBQUMsYUFBTztBQUFDRixVQUFFLEVBQUVKLEdBQUcsQ0FBQ08sS0FBSixDQUFVLGNBQVYsRUFBMEIsQ0FBMUIsQ0FBTDtBQUFtQ0MsWUFBSSxFQUFFRixJQUF6QztBQUErQ0csb0JBQVksRUFBRSxDQUE3RDtBQUFnRUMsbUJBQVcsRUFBRTtBQUE3RSxPQUFQO0FBQTJGO0FBSjdHLEdBREg7QUFPWEMsa0JBQWdCLEVBQUU7QUFDZFYsWUFBUSxFQUFFLGtCQURJO0FBRWRDLGVBQVcsRUFBR0MsQ0FBRCxJQUFRLDZDQUE0Q0EsQ0FBRSxNQUZyRDtBQUdkRSxjQUFVLEVBQUdMLEdBQUQsSUFBU0EsR0FBRyxDQUFDTyxLQUFKLENBQVUsY0FBVixFQUEwQixDQUExQjtBQUhQLEdBUFA7QUFZWEssY0FBWSxFQUFFO0FBQ1ZYLFlBQVEsRUFBRSxjQURBO0FBRVZDLGVBQVcsRUFBR0MsQ0FBRCxJQUFRLHlDQUF3Q0EsQ0FBRSxJQUZyRDtBQUdWRSxjQUFVLEVBQUdMLEdBQUQsSUFBU0EsR0FBRyxDQUFDTyxLQUFKLENBQVUsY0FBVixFQUEwQixDQUExQjtBQUhYLEdBWkg7QUFpQlhNLE1BQUksRUFBRTtBQUNGYixPQUFHLEVBQUUsa0NBREg7QUFFRkMsWUFBUSxFQUFFLE1BRlI7QUFHRmEsVUFBTSxFQUFHQyxJQUFELElBQVU7QUFDZCxZQUFNQyxNQUFNLEdBQUcsRUFBZjs7QUFDQSxXQUFLLElBQUliLENBQUMsR0FBRyxDQUFSLEVBQVdjLEdBQUcsR0FBR0YsSUFBSSxDQUFDQyxNQUFMLENBQVlFLE1BQWxDLEVBQTBDZixDQUFDLEdBQUdjLEdBQTlDLEVBQW1EZCxDQUFDLEVBQXBELEVBQXdEO0FBQ3BELGNBQU1nQixLQUFLLEdBQUdKLElBQUksQ0FBQ0MsTUFBTCxDQUFZYixDQUFaLENBQWQ7QUFDQWEsY0FBTSxDQUFDRyxLQUFLLENBQUNDLEtBQVAsQ0FBTixHQUFzQkQsS0FBSyxDQUFDbkIsR0FBNUI7QUFDSDs7QUFDRCxhQUFPZ0IsTUFBUDtBQUNILEtBVkM7QUFXRmQsZUFBVyxFQUFHQyxDQUFELElBQVEsR0FBRUEsQ0FBRSxFQVh2QjtBQVlGRSxjQUFVLEVBQUdMLEdBQUQsSUFBU0E7QUFabkIsR0FqQks7QUErQlhxQixPQUFLLEVBQUU7QUFDSHBCLFlBQVEsRUFBRSxPQURQO0FBRUhxQixlQUFXLEVBQUUsYUFGVjtBQUdIcEIsZUFBVyxFQUFHQyxDQUFELElBQVEsbUNBQWtDQSxDQUFFLEtBSHREO0FBSUhFLGNBQVUsRUFBR0wsR0FBRCxJQUFTQSxHQUFHLENBQUNPLEtBQUosQ0FBVSxlQUFWLEVBQTJCLENBQTNCO0FBSmxCO0FBL0JJLENBQWYiLCJmaWxlIjoiLi9zcmMvZGF0YS9lbW90ZXMvaW5mby5qcy5qcyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBkZWZhdWx0IHtcclxuICAgIFR3aXRjaEdsb2JhbDoge1xyXG4gICAgICAgIHVybDogXCJodHRwczovL3R3aXRjaGVtb3Rlcy5jb20vYXBpX2NhY2hlL3YzL2dsb2JhbC5qc29uXCIsXHJcbiAgICAgICAgdmFyaWFibGU6IFwiVHdpdGNoR2xvYmFsXCIsXHJcbiAgICAgICAgZ2V0RW1vdGVVUkw6IChlKSA9PiBgaHR0cHM6Ly9zdGF0aWMtY2RuLmp0dm53Lm5ldC9lbW90aWNvbnMvdjEvJHtlLmlkfS8xLjBgLFxyXG4gICAgICAgIGdldE9sZERhdGE6ICh1cmwsIG5hbWUpID0+IHtyZXR1cm4ge2lkOiB1cmwubWF0Y2goL1xcLyhbMC05XSspXFwvLylbMV0sIGNvZGU6IG5hbWUsIGVtb3RpY29uX3NldDogMCwgZGVzY3JpcHRpb246IG51bGx9O31cclxuICAgIH0sXHJcbiAgICBUd2l0Y2hTdWJzY3JpYmVyOiB7XHJcbiAgICAgICAgdmFyaWFibGU6IFwiVHdpdGNoU3Vic2NyaWJlclwiLFxyXG4gICAgICAgIGdldEVtb3RlVVJMOiAoZSkgPT4gYGh0dHBzOi8vc3RhdGljLWNkbi5qdHZudy5uZXQvZW1vdGljb25zL3YxLyR7ZX0vMS4wYCxcclxuICAgICAgICBnZXRPbGREYXRhOiAodXJsKSA9PiB1cmwubWF0Y2goL1xcLyhbMC05XSspXFwvLylbMV1cclxuICAgIH0sXHJcbiAgICBGcmFua2VyRmFjZVo6IHtcclxuICAgICAgICB2YXJpYWJsZTogXCJGcmFua2VyRmFjZVpcIixcclxuICAgICAgICBnZXRFbW90ZVVSTDogKGUpID0+IGBodHRwczovL2Nkbi5mcmFua2VyZmFjZXouY29tL2Vtb3RpY29uLyR7ZX0vMWAsXHJcbiAgICAgICAgZ2V0T2xkRGF0YTogKHVybCkgPT4gdXJsLm1hdGNoKC9cXC8oWzAtOV0rKVxcLy8pWzFdXHJcbiAgICB9LFxyXG4gICAgQlRUVjoge1xyXG4gICAgICAgIHVybDogXCJodHRwczovL2FwaS5iZXR0ZXJ0dHYubmV0L2Vtb3Rlc1wiLFxyXG4gICAgICAgIHZhcmlhYmxlOiBcIkJUVFZcIixcclxuICAgICAgICBwYXJzZXI6IChkYXRhKSA9PiB7XHJcbiAgICAgICAgICAgIGNvbnN0IGVtb3RlcyA9IHt9O1xyXG4gICAgICAgICAgICBmb3IgKGxldCBlID0gMCwgbGVuID0gZGF0YS5lbW90ZXMubGVuZ3RoOyBlIDwgbGVuOyBlKyspIHtcclxuICAgICAgICAgICAgICAgIGNvbnN0IGVtb3RlID0gZGF0YS5lbW90ZXNbZV07XHJcbiAgICAgICAgICAgICAgICBlbW90ZXNbZW1vdGUucmVnZXhdID0gZW1vdGUudXJsO1xyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIHJldHVybiBlbW90ZXM7XHJcbiAgICAgICAgfSxcclxuICAgICAgICBnZXRFbW90ZVVSTDogKGUpID0+IGAke2V9YCxcclxuICAgICAgICBnZXRPbGREYXRhOiAodXJsKSA9PiB1cmxcclxuICAgIH0sXHJcbiAgICBCVFRWMjoge1xyXG4gICAgICAgIHZhcmlhYmxlOiBcIkJUVFYyXCIsXHJcbiAgICAgICAgb2xkVmFyaWFibGU6IFwiZW1vdGVzQlRUVjJcIixcclxuICAgICAgICBnZXRFbW90ZVVSTDogKGUpID0+IGBodHRwczovL2Nkbi5iZXR0ZXJ0dHYubmV0L2Vtb3RlLyR7ZX0vMXhgLFxyXG4gICAgICAgIGdldE9sZERhdGE6ICh1cmwpID0+IHVybC5tYXRjaCgvZW1vdGVcXC8oLispXFwvLylbMV1cclxuICAgIH1cclxufTsiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/data/emotes/info.js\n"); /***/ }), @@ -335,7 +335,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony default export */ /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony default export */ __webpack_exports__[\"default\"] = ({\n Panels: {\n plugins: \"Plugins\",\n themes: \"Themes\",\n customcss: \"Custom CSS\"\n },\n Collections: {\n settings: {\n name: \"Settings\",\n general: {\n name: \"General\",\n emotes: {\n name: \"Emote System\",\n note: \"Enables BD's emote system\"\n },\n publicServers: {\n name: \"Public Servers\",\n note: \"Display public servers button\"\n },\n voiceDisconnect: {\n name: \"Voice Disconnect\",\n note: \"Disconnect from voice server when closing Discord\"\n },\n twentyFourHour: {\n name: \"24-Hour Timestamps\",\n note: \"Hides channels when in minimal mode\"\n },\n classNormalizer: {\n name: \"Normalize Classes\",\n note: \"Adds stable classes to elements to help themes. (e.g. adds .da-channels to .channels-Ie2l6A)\"\n },\n showToasts: {\n name: \"Show Toasts\",\n note: \"Shows a small notification for important information\"\n }\n },\n appearance: {\n name: \"Appearance\",\n voiceMode: {\n name: \"Voice Mode\",\n note: \"Hides everything that isn't voice chat\"\n },\n minimalMode: {\n name: \"Minimal Mode\",\n note: \"Hide elements and reduce the size of elements\"\n },\n hideChannels: {\n name: \"Hide Channels\",\n note: \"Hides channels when in minimal mode\"\n },\n darkMode: {\n name: \"Dark Mode\",\n note: \"Make certain elements dark by default\"\n },\n coloredText: {\n name: \"Colored Text\",\n note: \"Make text colour the same as role color\"\n }\n },\n addons: {\n name: \"Addon Manager\",\n addonErrors: {\n name: \"Show Addon Errors\",\n note: \"Shows a modal with plugin/theme errors\"\n },\n autoScroll: {\n name: \"Scroll To Settings\",\n note: \"Auto-scrolls to a plugin's settings when the button is clicked (only if out of view)\"\n },\n autoReload: {\n name: \"Automatic Loading\",\n note: \"Automatically loads, reloads, and unloads plugins and themes\"\n }\n },\n customcss: {\n name: \"Custom CSS\",\n customcss: {\n name: \"Custom CSS\",\n note: \"Enables the Custom CSS tab\"\n },\n liveUpdate: {\n name: \"Live Update\",\n note: \"Updates the css as you type\"\n },\n startDetached: {\n name: \"Start Detached\",\n note: \"Clicking the Custom CSS tab opens the editor in a separate window\"\n },\n nativeOpen: {\n name: \"Open in Native Editor\",\n note: \"Clicking the Custom CSS tab opens your custom css in your native editor\"\n }\n },\n developer: {\n name: \"Developer Settings\",\n developerMode: {\n name: \"Developer Mode\",\n note: \"Allows activating debugger when pressing F8\"\n },\n copySelector: {\n name: \"Copy Selector\",\n note: \"Adds a \\\"Copy Selector\\\" option to context menus when developer mode is active\"\n }\n },\n window: {\n name: \"Window Preferences\",\n transparency: {\n name: \"Enable Transparency\",\n note: \"Enables the main window to be see-through (requires restart)\"\n },\n frame: {\n name: \"Window Frame\",\n note: \"Adds the native os window frame to the main window\"\n }\n }\n },\n emotes: {\n name: \"Emotes\",\n general: {\n name: \"General\",\n download: {\n name: \"Download Emotes\",\n note: \"Download emotes once a week to stay up to date\"\n },\n emoteMenu: {\n name: \"Emote Menu\",\n note: \"Show Twitch/Favourite emotes in emote menu\"\n },\n hideEmojiMenu: {\n name: \"Hide Emoji Menu\",\n note: \"Hides Discord's emoji menu when using emote menu\"\n },\n autoCaps: {\n name: \"Emote Autocapitalization\",\n note: \"Autocapitalize emote commands\"\n },\n showNames: {\n name: \"Show Names\",\n note: \"Show emote names on hover\"\n },\n modifiers: {\n name: \"Show Emote Modifiers\",\n note: \"Enable emote mods (flip, spin, pulse, spin2, spin3, 1spin, 2spin, 3spin, tr, bl, br, shake, shake2, shake3, flap)\"\n },\n animateOnHover: {\n name: \"Animate On Hover\",\n note: \"Only animate the emote modifiers on hover\"\n }\n },\n categories: {\n name: \"Categories\",\n twitch: {\n name: \"Twitch\",\n note: \"Show Twitch global & subscriber emotes\"\n },\n ffz: {\n name: \"FrankerFaceZ\",\n note: \"Show emotes from FFZ\"\n },\n bttv: {\n name: \"BetterTTV\",\n note: \"Show emotes from BTTV\"\n }\n }\n }\n },\n Addons: {\n title: \"{{name}} v{{version}} by {{author}}\",\n openFolder: \"Open {{type}} Folder\",\n reload: \"Reload\",\n pluginSettings: \"Settings\",\n website: \"Website\",\n source: \"Source\",\n server: \"Support Server\",\n donate: \"Donate\"\n },\n Emotes: {\n downloading: \"Downloading emotes in the background do not reload.\",\n downloaded: \"All emotes successfully downloaded.\",\n clearEmotes: \"Clear Emote Data\",\n favoriteAction: \"Favorite!\"\n },\n CustomCSS: {\n confirmationText: \"You have unsaved changes to your Custom CSS. Closing this window will lose all those changes.\",\n update: \"Update\",\n save: \"Save\",\n openNative: \"Open in System Editor\",\n openDetached: \"Detach Window\",\n settings: \"Editor Settings\",\n editorTitle: \"Custom CSS Editor\"\n },\n PublicServers: {\n button: \"public\",\n join: \"Join\",\n joining: \"Joining\",\n joined: \"Joined\",\n loading: \"Loading\",\n loadMore: \"Load More\",\n notConnected: \"Not connected to DiscordServers.com!\",\n search: \"Search\",\n connect: \"Connect\",\n reconnect: \"Reconnect\",\n categories: \"Categories\",\n connection: \"Connected as: {{username}}#{{discriminator}}\",\n results: \"Showing {{start}}-{{end}} of {{total}} results in {{category}}\",\n query: \"for {{query}}\"\n },\n Modals: {\n confirmClose: \"Are You Sure?\",\n okay: \"Okay\",\n cancel: \"Cancel\",\n name: \"Name\",\n message: \"Message\",\n error: \"Error\",\n addonErrors: \"Addon Errors\"\n }\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/data/strings.js?bc71"],"names":["Panels","plugins","themes","customcss","Collections","settings","name","general","emotes","note","publicServers","voiceDisconnect","twentyFourHour","classNormalizer","showToasts","appearance","voiceMode","minimalMode","hideChannels","darkMode","coloredText","addons","addonErrors","autoScroll","autoReload","liveUpdate","startDetached","nativeOpen","developer","developerMode","copySelector","window","transparency","frame","download","emoteMenu","hideEmojiMenu","autoCaps","showNames","modifiers","animateOnHover","categories","twitch","ffz","bttv","Addons","title","openFolder","reload","pluginSettings","website","source","server","donate","Emotes","downloading","downloaded","clearEmotes","favoriteAction","CustomCSS","confirmationText","update","save","openNative","openDetached","editorTitle","PublicServers","button","join","joining","joined","loading","loadMore","notConnected","search","connect","reconnect","connection","results","query","Modals","confirmClose","okay","cancel","message","error"],"mappings":"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","file":"./src/data/strings.js.js","sourcesContent":["export default {\r\n    Panels: {\r\n        plugins: \"Plugins\",\r\n        themes: \"Themes\",\r\n        customcss: \"Custom CSS\"\r\n    },\r\n    Collections: {\r\n        settings: {\r\n            name: \"Settings\",\r\n            general: {\r\n                name: \"General\",\r\n                emotes: {\r\n                    name: \"Emote System\",\r\n                    note: \"Enables BD's emote system\"\r\n                },\r\n                publicServers: {\r\n                    name: \"Public Servers\",\r\n                    note: \"Display public servers button\"\r\n                },\r\n                voiceDisconnect: {\r\n                    name: \"Voice Disconnect\",\r\n                    note: \"Disconnect from voice server when closing Discord\"\r\n                },\r\n                twentyFourHour: {\r\n                    name: \"24-Hour Timestamps\",\r\n                    note: \"Hides channels when in minimal mode\"\r\n                },\r\n                classNormalizer: {\r\n                    name: \"Normalize Classes\",\r\n                    note: \"Adds stable classes to elements to help themes. (e.g. adds .da-channels to .channels-Ie2l6A)\"\r\n                },\r\n                showToasts: {\r\n                    name: \"Show Toasts\",\r\n                    note: \"Shows a small notification for important information\"\r\n                }\r\n            },\r\n            appearance: {\r\n                name: \"Appearance\",\r\n                voiceMode: {\r\n                    name: \"Voice Mode\",\r\n                    note: \"Hides everything that isn't voice chat\"\r\n                },\r\n                minimalMode: {\r\n                    name: \"Minimal Mode\",\r\n                    note: \"Hide elements and reduce the size of elements\"\r\n                },\r\n                hideChannels: {\r\n                    name: \"Hide Channels\",\r\n                    note: \"Hides channels when in minimal mode\"\r\n                },\r\n                darkMode: {\r\n                    name: \"Dark Mode\",\r\n                    note: \"Make certain elements dark by default\"\r\n                },\r\n                coloredText: {\r\n                    name: \"Colored Text\",\r\n                    note: \"Make text colour the same as role color\"\r\n                }\r\n            },\r\n            addons: {\r\n                name: \"Addon Manager\",\r\n                addonErrors: {\r\n                    name: \"Show Addon Errors\",\r\n                    note: \"Shows a modal with plugin/theme errors\"\r\n                },\r\n                autoScroll: {\r\n                    name: \"Scroll To Settings\",\r\n                    note: \"Auto-scrolls to a plugin's settings when the button is clicked (only if out of view)\"\r\n                },\r\n                autoReload: {\r\n                    name: \"Automatic Loading\",\r\n                    note: \"Automatically loads, reloads, and unloads plugins and themes\"\r\n                },\r\n            },\r\n            customcss: {\r\n                name: \"Custom CSS\",\r\n                customcss: {\r\n                    name: \"Custom CSS\",\r\n                    note: \"Enables the Custom CSS tab\"\r\n                },\r\n                liveUpdate: {\r\n                    name: \"Live Update\",\r\n                    note: \"Updates the css as you type\"\r\n                },\r\n                startDetached: {\r\n                    name: \"Start Detached\",\r\n                    note: \"Clicking the Custom CSS tab opens the editor in a separate window\",\r\n                },\r\n                nativeOpen: {\r\n                    name: \"Open in Native Editor\",\r\n                    note: \"Clicking the Custom CSS tab opens your custom css in your native editor\"\r\n                }\r\n            },\r\n            developer: {\r\n                name: \"Developer Settings\",\r\n                developerMode: {\r\n                    name: \"Developer Mode\",\r\n                    note: \"Allows activating debugger when pressing F8\"\r\n                },\r\n                copySelector: {\r\n                    name: \"Copy Selector\",\r\n                    note: \"Adds a \\\"Copy Selector\\\" option to context menus when developer mode is active\"\r\n                }\r\n            },\r\n            window: {\r\n                name: \"Window Preferences\",\r\n                transparency: {\r\n                    name: \"Enable Transparency\",\r\n                    note: \"Enables the main window to be see-through (requires restart)\"\r\n                },\r\n                frame: {\r\n                    name: \"Window Frame\",\r\n                    note: \"Adds the native os window frame to the main window\"\r\n                }\r\n            }\r\n        },\r\n        emotes: {\r\n            name: \"Emotes\",\r\n            general: {\r\n                name: \"General\",\r\n                download: {\r\n                    name: \"Download Emotes\",\r\n                    note: \"Download emotes once a week to stay up to date\"\r\n                },\r\n                emoteMenu: {\r\n                    name: \"Emote Menu\",\r\n                    note: \"Show Twitch/Favourite emotes in emote menu\"\r\n                },\r\n                hideEmojiMenu: {\r\n                    name: \"Hide Emoji Menu\",\r\n                    note: \"Hides Discord's emoji menu when using emote menu\"\r\n                },\r\n                autoCaps: {\r\n                    name: \"Emote Autocapitalization\",\r\n                    note: \"Autocapitalize emote commands\"\r\n                },\r\n                showNames: {\r\n                    name: \"Show Names\",\r\n                    note: \"Show emote names on hover\"\r\n                },\r\n                modifiers: {\r\n                    name: \"Show Emote Modifiers\",\r\n                    note: \"Enable emote mods (flip, spin, pulse, spin2, spin3, 1spin, 2spin, 3spin, tr, bl, br, shake, shake2, shake3, flap)\"\r\n                },\r\n                animateOnHover: {\r\n                    name: \"Animate On Hover\",\r\n                    note: \"Only animate the emote modifiers on hover\"\r\n                }\r\n            },\r\n            categories: {\r\n                name: \"Categories\",\r\n                twitch: {\r\n                    name: \"Twitch\",\r\n                    note: \"Show Twitch global & subscriber emotes\"\r\n                },\r\n                ffz: {\r\n                    name: \"FrankerFaceZ\",\r\n                    note: \"Show emotes from FFZ\"\r\n                },\r\n                bttv: {\r\n                    name: \"BetterTTV\",\r\n                    note: \"Show emotes from BTTV\"\r\n                }\r\n            }\r\n        }\r\n    },\r\n    Addons: {\r\n        title: \"{{name}} v{{version}} by {{author}}\",\r\n        openFolder: \"Open {{type}} Folder\",\r\n        reload: \"Reload\",\r\n        pluginSettings: \"Settings\",\r\n        website: \"Website\",\r\n        source: \"Source\",\r\n        server: \"Support Server\",\r\n        donate: \"Donate\"\r\n    },\r\n    Emotes: {\r\n        downloading: \"Downloading emotes in the background do not reload.\",\r\n        downloaded: \"All emotes successfully downloaded.\",\r\n        clearEmotes: \"Clear Emote Data\",\r\n        favoriteAction: \"Favorite!\"\r\n    },\r\n    CustomCSS: {\r\n        confirmationText: \"You have unsaved changes to your Custom CSS. Closing this window will lose all those changes.\",\r\n        update: \"Update\",\r\n        save: \"Save\",\r\n        openNative: \"Open in System Editor\",\r\n        openDetached: \"Detach Window\",\r\n        settings: \"Editor Settings\",\r\n        editorTitle: \"Custom CSS Editor\"\r\n    },\r\n    PublicServers: {\r\n        button: \"public\",\r\n        join: \"Join\",\r\n        joining: \"Joining\",\r\n        joined: \"Joined\",\r\n        loading: \"Loading\",\r\n        loadMore: \"Load More\",\r\n        notConnected: \"Not connected to DiscordServers.com!\",\r\n        search: \"Search\",\r\n        connect: \"Connect\",\r\n        reconnect: \"Reconnect\",\r\n        categories: \"Categories\",\r\n        connection: \"Connected as: {{username}}#{{discriminator}}\",\r\n        results: \"Showing {{start}}-{{end}} of {{total}} results in {{category}}\",\r\n        query: \"for {{query}}\"\r\n    },\r\n    Modals: {\r\n        confirmClose: \"Are You Sure?\",\r\n        okay: \"Okay\",\r\n        cancel: \"Cancel\",\r\n        name: \"Name\",\r\n        message: \"Message\",\r\n        error: \"Error\",\r\n        addonErrors: \"Addon Errors\"\r\n    }\r\n};\r\n"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/data/strings.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony default export */ __webpack_exports__[\"default\"] = ({\n Panels: {\n plugins: \"Plugins\",\n themes: \"Themes\",\n customcss: \"Custom CSS\"\n },\n Collections: {\n settings: {\n name: \"Settings\",\n general: {\n name: \"General\",\n emotes: {\n name: \"Emote System\",\n note: \"Enables BD's emote system\"\n },\n publicServers: {\n name: \"Public Servers\",\n note: \"Display public servers button\"\n },\n voiceDisconnect: {\n name: \"Voice Disconnect\",\n note: \"Disconnect from voice server when closing Discord\"\n },\n twentyFourHour: {\n name: \"24-Hour Timestamps\",\n note: \"Hides channels when in minimal mode\"\n },\n classNormalizer: {\n name: \"Normalize Classes\",\n note: \"Adds stable classes to elements to help themes. (e.g. adds .da-channels to .channels-Ie2l6A)\"\n },\n showToasts: {\n name: \"Show Toasts\",\n note: \"Shows a small notification for important information\"\n }\n },\n appearance: {\n name: \"Appearance\",\n voiceMode: {\n name: \"Voice Mode\",\n note: \"Hides everything that isn't voice chat\"\n },\n minimalMode: {\n name: \"Minimal Mode\",\n note: \"Hide elements and reduce the size of elements\"\n },\n hideChannels: {\n name: \"Hide Channels\",\n note: \"Hides channels when in minimal mode\"\n },\n darkMode: {\n name: \"Dark Mode\",\n note: \"Make certain elements dark by default\"\n },\n coloredText: {\n name: \"Colored Text\",\n note: \"Make text colour the same as role color\"\n }\n },\n addons: {\n name: \"Addon Manager\",\n addonErrors: {\n name: \"Show Addon Errors\",\n note: \"Shows a modal with plugin/theme errors\"\n },\n autoScroll: {\n name: \"Scroll To Settings\",\n note: \"Auto-scrolls to a plugin's settings when the button is clicked (only if out of view)\"\n },\n autoReload: {\n name: \"Automatic Loading\",\n note: \"Automatically loads, reloads, and unloads plugins and themes\"\n }\n },\n customcss: {\n name: \"Custom CSS\",\n customcss: {\n name: \"Custom CSS\",\n note: \"Enables the Custom CSS tab\"\n },\n liveUpdate: {\n name: \"Live Update\",\n note: \"Updates the css as you type\"\n },\n startDetached: {\n name: \"Start Detached\",\n note: \"Clicking the Custom CSS tab opens the editor in a separate window\"\n },\n nativeOpen: {\n name: \"Open in Native Editor\",\n note: \"Clicking the Custom CSS tab opens your custom css in your native editor\"\n }\n },\n developer: {\n name: \"Developer Settings\",\n developerMode: {\n name: \"Developer Mode\",\n note: \"Allows activating debugger when pressing F8\"\n },\n copySelector: {\n name: \"Copy Selector\",\n note: \"Adds a \\\"Copy Selector\\\" option to context menus when developer mode is active\"\n }\n },\n window: {\n name: \"Window Preferences\",\n transparency: {\n name: \"Enable Transparency\",\n note: \"Enables the main window to be see-through (requires restart)\"\n },\n frame: {\n name: \"Window Frame\",\n note: \"Adds the native os window frame to the main window\"\n }\n }\n },\n emotes: {\n name: \"Emotes\",\n general: {\n name: \"General\",\n download: {\n name: \"Download Emotes\",\n note: \"Download emotes once a week to stay up to date\"\n },\n emoteMenu: {\n name: \"Emote Menu\",\n note: \"Show Twitch/Favourite emotes in emote menu\"\n },\n hideEmojiMenu: {\n name: \"Hide Emoji Menu\",\n note: \"Hides Discord's emoji menu when using emote menu\"\n },\n autoCaps: {\n name: \"Emote Autocapitalization\",\n note: \"Autocapitalize emote commands\"\n },\n showNames: {\n name: \"Show Names\",\n note: \"Show emote names on hover\"\n },\n modifiers: {\n name: \"Show Emote Modifiers\",\n note: \"Enable emote mods (flip, spin, pulse, spin2, spin3, 1spin, 2spin, 3spin, tr, bl, br, shake, shake2, shake3, flap)\"\n },\n animateOnHover: {\n name: \"Animate On Hover\",\n note: \"Only animate the emote modifiers on hover\"\n }\n },\n categories: {\n name: \"Categories\",\n twitch: {\n name: \"Twitch\",\n note: \"Show Twitch global & subscriber emotes\"\n },\n ffz: {\n name: \"FrankerFaceZ\",\n note: \"Show emotes from FFZ\"\n },\n bttv: {\n name: \"BetterTTV\",\n note: \"Show emotes from BTTV\"\n }\n }\n }\n },\n Addons: {\n title: \"{{name}} v{{version}} by {{author}}\",\n openFolder: \"Open {{type}} Folder\",\n reload: \"Reload\",\n addonSettings: \"Settings\",\n website: \"Website\",\n source: \"Source\",\n server: \"Support Server\",\n donate: \"Donate\"\n },\n Emotes: {\n downloading: \"Downloading emotes in the background do not reload.\",\n downloaded: \"All emotes successfully downloaded.\",\n clearEmotes: \"Clear Emote Data\",\n favoriteAction: \"Favorite!\"\n },\n CustomCSS: {\n confirmationText: \"You have unsaved changes to your Custom CSS. Closing this window will lose all those changes.\",\n update: \"Update\",\n save: \"Save\",\n openNative: \"Open in System Editor\",\n openDetached: \"Detach Window\",\n settings: \"Editor Settings\",\n editorTitle: \"Custom CSS Editor\"\n },\n PublicServers: {\n button: \"public\",\n join: \"Join\",\n joining: \"Joining\",\n joined: \"Joined\",\n loading: \"Loading\",\n loadMore: \"Load More\",\n notConnected: \"Not connected to DiscordServers.com!\",\n search: \"Search\",\n connect: \"Connect\",\n reconnect: \"Reconnect\",\n categories: \"Categories\",\n connection: \"Connected as: {{username}}#{{discriminator}}\",\n results: \"Showing {{start}}-{{end}} of {{total}} results in {{category}}\",\n query: \"for {{query}}\"\n },\n Modals: {\n confirmClose: \"Are You Sure?\",\n okay: \"Okay\",\n cancel: \"Cancel\",\n name: \"Name\",\n message: \"Message\",\n error: \"Error\",\n addonErrors: \"Addon Errors\"\n }\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/data/strings.js?bc71"],"names":["Panels","plugins","themes","customcss","Collections","settings","name","general","emotes","note","publicServers","voiceDisconnect","twentyFourHour","classNormalizer","showToasts","appearance","voiceMode","minimalMode","hideChannels","darkMode","coloredText","addons","addonErrors","autoScroll","autoReload","liveUpdate","startDetached","nativeOpen","developer","developerMode","copySelector","window","transparency","frame","download","emoteMenu","hideEmojiMenu","autoCaps","showNames","modifiers","animateOnHover","categories","twitch","ffz","bttv","Addons","title","openFolder","reload","addonSettings","website","source","server","donate","Emotes","downloading","downloaded","clearEmotes","favoriteAction","CustomCSS","confirmationText","update","save","openNative","openDetached","editorTitle","PublicServers","button","join","joining","joined","loading","loadMore","notConnected","search","connect","reconnect","connection","results","query","Modals","confirmClose","okay","cancel","message","error"],"mappings":"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","file":"./src/data/strings.js.js","sourcesContent":["export default {\r\n    Panels: {\r\n        plugins: \"Plugins\",\r\n        themes: \"Themes\",\r\n        customcss: \"Custom CSS\"\r\n    },\r\n    Collections: {\r\n        settings: {\r\n            name: \"Settings\",\r\n            general: {\r\n                name: \"General\",\r\n                emotes: {\r\n                    name: \"Emote System\",\r\n                    note: \"Enables BD's emote system\"\r\n                },\r\n                publicServers: {\r\n                    name: \"Public Servers\",\r\n                    note: \"Display public servers button\"\r\n                },\r\n                voiceDisconnect: {\r\n                    name: \"Voice Disconnect\",\r\n                    note: \"Disconnect from voice server when closing Discord\"\r\n                },\r\n                twentyFourHour: {\r\n                    name: \"24-Hour Timestamps\",\r\n                    note: \"Hides channels when in minimal mode\"\r\n                },\r\n                classNormalizer: {\r\n                    name: \"Normalize Classes\",\r\n                    note: \"Adds stable classes to elements to help themes. (e.g. adds .da-channels to .channels-Ie2l6A)\"\r\n                },\r\n                showToasts: {\r\n                    name: \"Show Toasts\",\r\n                    note: \"Shows a small notification for important information\"\r\n                }\r\n            },\r\n            appearance: {\r\n                name: \"Appearance\",\r\n                voiceMode: {\r\n                    name: \"Voice Mode\",\r\n                    note: \"Hides everything that isn't voice chat\"\r\n                },\r\n                minimalMode: {\r\n                    name: \"Minimal Mode\",\r\n                    note: \"Hide elements and reduce the size of elements\"\r\n                },\r\n                hideChannels: {\r\n                    name: \"Hide Channels\",\r\n                    note: \"Hides channels when in minimal mode\"\r\n                },\r\n                darkMode: {\r\n                    name: \"Dark Mode\",\r\n                    note: \"Make certain elements dark by default\"\r\n                },\r\n                coloredText: {\r\n                    name: \"Colored Text\",\r\n                    note: \"Make text colour the same as role color\"\r\n                }\r\n            },\r\n            addons: {\r\n                name: \"Addon Manager\",\r\n                addonErrors: {\r\n                    name: \"Show Addon Errors\",\r\n                    note: \"Shows a modal with plugin/theme errors\"\r\n                },\r\n                autoScroll: {\r\n                    name: \"Scroll To Settings\",\r\n                    note: \"Auto-scrolls to a plugin's settings when the button is clicked (only if out of view)\"\r\n                },\r\n                autoReload: {\r\n                    name: \"Automatic Loading\",\r\n                    note: \"Automatically loads, reloads, and unloads plugins and themes\"\r\n                },\r\n            },\r\n            customcss: {\r\n                name: \"Custom CSS\",\r\n                customcss: {\r\n                    name: \"Custom CSS\",\r\n                    note: \"Enables the Custom CSS tab\"\r\n                },\r\n                liveUpdate: {\r\n                    name: \"Live Update\",\r\n                    note: \"Updates the css as you type\"\r\n                },\r\n                startDetached: {\r\n                    name: \"Start Detached\",\r\n                    note: \"Clicking the Custom CSS tab opens the editor in a separate window\",\r\n                },\r\n                nativeOpen: {\r\n                    name: \"Open in Native Editor\",\r\n                    note: \"Clicking the Custom CSS tab opens your custom css in your native editor\"\r\n                }\r\n            },\r\n            developer: {\r\n                name: \"Developer Settings\",\r\n                developerMode: {\r\n                    name: \"Developer Mode\",\r\n                    note: \"Allows activating debugger when pressing F8\"\r\n                },\r\n                copySelector: {\r\n                    name: \"Copy Selector\",\r\n                    note: \"Adds a \\\"Copy Selector\\\" option to context menus when developer mode is active\"\r\n                }\r\n            },\r\n            window: {\r\n                name: \"Window Preferences\",\r\n                transparency: {\r\n                    name: \"Enable Transparency\",\r\n                    note: \"Enables the main window to be see-through (requires restart)\"\r\n                },\r\n                frame: {\r\n                    name: \"Window Frame\",\r\n                    note: \"Adds the native os window frame to the main window\"\r\n                }\r\n            }\r\n        },\r\n        emotes: {\r\n            name: \"Emotes\",\r\n            general: {\r\n                name: \"General\",\r\n                download: {\r\n                    name: \"Download Emotes\",\r\n                    note: \"Download emotes once a week to stay up to date\"\r\n                },\r\n                emoteMenu: {\r\n                    name: \"Emote Menu\",\r\n                    note: \"Show Twitch/Favourite emotes in emote menu\"\r\n                },\r\n                hideEmojiMenu: {\r\n                    name: \"Hide Emoji Menu\",\r\n                    note: \"Hides Discord's emoji menu when using emote menu\"\r\n                },\r\n                autoCaps: {\r\n                    name: \"Emote Autocapitalization\",\r\n                    note: \"Autocapitalize emote commands\"\r\n                },\r\n                showNames: {\r\n                    name: \"Show Names\",\r\n                    note: \"Show emote names on hover\"\r\n                },\r\n                modifiers: {\r\n                    name: \"Show Emote Modifiers\",\r\n                    note: \"Enable emote mods (flip, spin, pulse, spin2, spin3, 1spin, 2spin, 3spin, tr, bl, br, shake, shake2, shake3, flap)\"\r\n                },\r\n                animateOnHover: {\r\n                    name: \"Animate On Hover\",\r\n                    note: \"Only animate the emote modifiers on hover\"\r\n                }\r\n            },\r\n            categories: {\r\n                name: \"Categories\",\r\n                twitch: {\r\n                    name: \"Twitch\",\r\n                    note: \"Show Twitch global & subscriber emotes\"\r\n                },\r\n                ffz: {\r\n                    name: \"FrankerFaceZ\",\r\n                    note: \"Show emotes from FFZ\"\r\n                },\r\n                bttv: {\r\n                    name: \"BetterTTV\",\r\n                    note: \"Show emotes from BTTV\"\r\n                }\r\n            }\r\n        }\r\n    },\r\n    Addons: {\r\n        title: \"{{name}} v{{version}} by {{author}}\",\r\n        openFolder: \"Open {{type}} Folder\",\r\n        reload: \"Reload\",\r\n        addonSettings: \"Settings\",\r\n        website: \"Website\",\r\n        source: \"Source\",\r\n        server: \"Support Server\",\r\n        donate: \"Donate\"\r\n    },\r\n    Emotes: {\r\n        downloading: \"Downloading emotes in the background do not reload.\",\r\n        downloaded: \"All emotes successfully downloaded.\",\r\n        clearEmotes: \"Clear Emote Data\",\r\n        favoriteAction: \"Favorite!\"\r\n    },\r\n    CustomCSS: {\r\n        confirmationText: \"You have unsaved changes to your Custom CSS. Closing this window will lose all those changes.\",\r\n        update: \"Update\",\r\n        save: \"Save\",\r\n        openNative: \"Open in System Editor\",\r\n        openDetached: \"Detach Window\",\r\n        settings: \"Editor Settings\",\r\n        editorTitle: \"Custom CSS Editor\"\r\n    },\r\n    PublicServers: {\r\n        button: \"public\",\r\n        join: \"Join\",\r\n        joining: \"Joining\",\r\n        joined: \"Joined\",\r\n        loading: \"Loading\",\r\n        loadMore: \"Load More\",\r\n        notConnected: \"Not connected to DiscordServers.com!\",\r\n        search: \"Search\",\r\n        connect: \"Connect\",\r\n        reconnect: \"Reconnect\",\r\n        categories: \"Categories\",\r\n        connection: \"Connected as: {{username}}#{{discriminator}}\",\r\n        results: \"Showing {{start}}-{{end}} of {{total}} results in {{category}}\",\r\n        query: \"for {{query}}\"\r\n    },\r\n    Modals: {\r\n        confirmClose: \"Are You Sure?\",\r\n        okay: \"Okay\",\r\n        cancel: \"Cancel\",\r\n        name: \"Name\",\r\n        message: \"Message\",\r\n        error: \"Error\",\r\n        addonErrors: \"Addon Errors\"\r\n    }\r\n};\r\n"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/data/strings.js\n"); /***/ }), @@ -395,7 +395,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _loc /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var data__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! data */ \"./src/data/data.js\");\n/* harmony import */ var _utilities__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./utilities */ \"./src/modules/utilities.js\");\n\n\n\nconst fs = __webpack_require__(/*! fs */ \"fs\");\n\nconst path = __webpack_require__(/*! path */ \"path\");\n\nconst releaseChannel = DiscordNative.globals.releaseChannel; // Schema\n// =======================\n// %appdata%\\BetterDiscord\n// -> data\n// -> [releaseChannel]\\ (stable/canary/ptb)\n// -> settings.json\n// -> plugins.json\n// -> themes.json\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (new class DataStore {\n constructor() {\n this.data = {\n misc: {}\n };\n this.pluginData = {};\n }\n\n initialize() {\n if (!fs.existsSync(this.baseFolder)) fs.mkdirSync(this.baseFolder);\n if (!fs.existsSync(this.dataFolder)) fs.mkdirSync(this.dataFolder);\n if (!fs.existsSync(this.localeFolder)) fs.mkdirSync(this.localeFolder);\n if (!fs.existsSync(this.BDFile)) fs.writeFileSync(this.BDFile, JSON.stringify(this.data.misc, null, 4));\n if (!fs.existsSync(this.customCSS)) fs.writeFileSync(this.customCSS, \"\");\n const dataFiles = fs.readdirSync(this.dataFolder).filter(f => !fs.statSync(path.resolve(this.dataFolder, f)).isDirectory() && f.endsWith(\".json\"));\n\n for (const file of dataFiles) {\n this.data[file.split(\".\")[0]] = require(path.resolve(this.dataFolder, file));\n } // this.data = __non_webpack_require__(this.BDFile);\n // if (data.hasOwnProperty(\"settings\")) this.data = data;\n // if (!fs.existsSync(this.settingsFile)) return;\n // let settings = __non_webpack_require__(this.settingsFile);\n // fs.unlinkSync(this.settingsFile);\n // if (settings.hasOwnProperty(\"settings\")) settings = Object.assign({stable: {}, canary: {}, ptb: {}}, {[releaseChannel]: settings});\n // else settings = Object.assign({stable: {}, canary: {}, ptb: {}}, settings);\n // this.setBDData(\"settings\", settings);\n\n }\n\n get customCSS() {\n return this._customCSS || (this._customCSS = path.resolve(this.dataFolder, \"custom.css\"));\n }\n\n get baseFolder() {\n return this._baseFolder || (this._baseFolder = path.resolve(data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].dataPath, \"data\"));\n }\n\n get dataFolder() {\n return this._dataFolder || (this._dataFolder = path.resolve(this.baseFolder, `${releaseChannel}`));\n }\n\n get localeFolder() {\n return this._localeFolder || (this._localeFolder = path.resolve(this.baseFolder, `locales`));\n }\n\n get BDFile() {\n return this._BDFile || (this._BDFile = path.resolve(data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].dataPath, \"data\", `${releaseChannel}.json`));\n } // get settingsFile() {return this._settingsFile || (this._settingsFile = path.resolve(Config.dataPath, \"bdsettings.json\"));}\n\n\n getPluginFile(pluginName) {\n return path.resolve(data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].dataPath, \"plugins\", pluginName + \".config.json\");\n } // getSettingGroup(key) {\n // return this.data.settings[key] || null;\n // }\n // setSettingGroup(key, data) {\n // this.data.settings[key] = data;\n // fs.writeFileSync(this.BDFile, JSON.stringify(this.data, null, 4));\n // }\n\n\n _getFile(key) {\n if (key == \"settings\" || key == \"plugins\" || key == \"themes\") return path.resolve(this.dataFolder, `${key}.json`);\n return path.resolve(this.dataFolder, `misc.json`);\n }\n\n getBDData(key) {\n return this.data.misc[key] || \"\";\n }\n\n setBDData(key, value) {\n this.data.misc[key] = value;\n fs.writeFileSync(path.resolve(this.dataFolder, `misc.json`), JSON.stringify(this.data.misc, null, 4));\n }\n\n getLocale(locale) {\n const file = path.resolve(this.localeFolder, `${locale}.json`);\n if (!fs.existsSync(file)) return null;\n return _utilities__WEBPACK_IMPORTED_MODULE_1__[\"default\"].testJSON(fs.readFileSync(file).toString());\n }\n\n saveLocale(locale, strings) {\n fs.writeFileSync(path.resolve(this.localeFolder, `${locale}.json`), JSON.stringify(strings, null, 4));\n }\n\n getData(key) {\n return this.data[key] || \"\";\n }\n\n setData(key, value) {\n this.data[key] = value;\n fs.writeFileSync(path.resolve(this.dataFolder, `${key}.json`), JSON.stringify(value, null, 4));\n }\n\n loadCustomCSS() {\n return fs.readFileSync(this.customCSS).toString();\n }\n\n saveCustomCSS(css) {\n return fs.writeFileSync(this.customCSS, css);\n }\n\n getPluginData(pluginName, key) {\n if (this.pluginData[pluginName] !== undefined) return this.pluginData[pluginName][key] || undefined;\n if (!fs.existsSync(this.getPluginFile(pluginName))) return undefined;\n this.pluginData[pluginName] = JSON.parse(fs.readFileSync(this.getPluginFile(pluginName)));\n return this.pluginData[pluginName][key] || undefined;\n }\n\n setPluginData(pluginName, key, value) {\n if (value === undefined) return;\n if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\n this.pluginData[pluginName][key] = value;\n fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\n }\n\n deletePluginData(pluginName, key) {\n if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\n delete this.pluginData[pluginName][key];\n fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\n }\n\n}());//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/modules/datastore.js?1052"],"names":["fs","require","path","releaseChannel","DiscordNative","globals","DataStore","constructor","data","misc","pluginData","initialize","existsSync","baseFolder","mkdirSync","dataFolder","localeFolder","BDFile","writeFileSync","JSON","stringify","customCSS","dataFiles","readdirSync","filter","f","statSync","resolve","isDirectory","endsWith","file","split","__non_webpack_require__","_customCSS","_baseFolder","Config","dataPath","_dataFolder","_localeFolder","_BDFile","getPluginFile","pluginName","_getFile","key","getBDData","setBDData","value","getLocale","locale","Utilities","testJSON","readFileSync","toString","saveLocale","strings","getData","setData","loadCustomCSS","saveCustomCSS","css","getPluginData","undefined","parse","setPluginData","deletePluginData"],"mappings":"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","file":"./src/modules/datastore.js.js","sourcesContent":["import {Config} from \"data\";\r\nimport Utilities from \"./utilities\";\r\nconst fs = require(\"fs\");\r\nconst path = require(\"path\");\r\nconst releaseChannel = DiscordNative.globals.releaseChannel;\r\n\r\n// Schema\r\n// =======================\r\n// %appdata%\\BetterDiscord\r\n//     -> data\r\n//         -> [releaseChannel]\\ (stable/canary/ptb)\r\n//             -> settings.json\r\n//             -> plugins.json\r\n//             -> themes.json\r\n\r\nexport default new class DataStore {\r\n    constructor() {\r\n        this.data = {misc: {}};\r\n        this.pluginData = {};\r\n    }\r\n\r\n    initialize() {\r\n        if (!fs.existsSync(this.baseFolder)) fs.mkdirSync(this.baseFolder);\r\n        if (!fs.existsSync(this.dataFolder)) fs.mkdirSync(this.dataFolder);\r\n        if (!fs.existsSync(this.localeFolder)) fs.mkdirSync(this.localeFolder);\r\n        if (!fs.existsSync(this.BDFile)) fs.writeFileSync(this.BDFile, JSON.stringify(this.data.misc, null, 4));\r\n        if (!fs.existsSync(this.customCSS)) fs.writeFileSync(this.customCSS, \"\");\r\n        const dataFiles = fs.readdirSync(this.dataFolder).filter(f => !fs.statSync(path.resolve(this.dataFolder, f)).isDirectory() && f.endsWith(\".json\"));\r\n        for (const file of dataFiles) {\r\n            this.data[file.split(\".\")[0]] = __non_webpack_require__(path.resolve(this.dataFolder, file));\r\n        }\r\n        // this.data = __non_webpack_require__(this.BDFile);\r\n        // if (data.hasOwnProperty(\"settings\")) this.data = data;\r\n        // if (!fs.existsSync(this.settingsFile)) return;\r\n        // let settings = __non_webpack_require__(this.settingsFile);\r\n        // fs.unlinkSync(this.settingsFile);\r\n        // if (settings.hasOwnProperty(\"settings\")) settings = Object.assign({stable: {}, canary: {}, ptb: {}}, {[releaseChannel]: settings});\r\n        // else settings = Object.assign({stable: {}, canary: {}, ptb: {}}, settings);\r\n        // this.setBDData(\"settings\", settings);\r\n    }\r\n\r\n    get customCSS() {return this._customCSS || (this._customCSS = path.resolve(this.dataFolder, \"custom.css\"));}\r\n    get baseFolder() {return this._baseFolder || (this._baseFolder = path.resolve(Config.dataPath, \"data\"));}\r\n    get dataFolder() {return this._dataFolder || (this._dataFolder = path.resolve(this.baseFolder, `${releaseChannel}`));}\r\n    get localeFolder() {return this._localeFolder || (this._localeFolder = path.resolve(this.baseFolder, `locales`));}\r\n    get BDFile() {return this._BDFile || (this._BDFile = path.resolve(Config.dataPath, \"data\", `${releaseChannel}.json`));}\r\n    // get settingsFile() {return this._settingsFile || (this._settingsFile = path.resolve(Config.dataPath, \"bdsettings.json\"));}\r\n    getPluginFile(pluginName) {return path.resolve(Config.dataPath, \"plugins\", pluginName + \".config.json\");}\r\n\r\n    // getSettingGroup(key) {\r\n    //     return this.data.settings[key] || null;\r\n    // }\r\n\r\n    // setSettingGroup(key, data) {\r\n    //     this.data.settings[key] = data;\r\n    //     fs.writeFileSync(this.BDFile, JSON.stringify(this.data, null, 4));\r\n    // }\r\n\r\n    _getFile(key) {\r\n        if (key == \"settings\" || key == \"plugins\" || key == \"themes\") return path.resolve(this.dataFolder, `${key}.json`);\r\n        return path.resolve(this.dataFolder, `misc.json`);\r\n    }\r\n\r\n    getBDData(key) {\r\n        return this.data.misc[key] || \"\";\r\n    }\r\n\r\n    setBDData(key, value) {\r\n        this.data.misc[key] = value;\r\n        fs.writeFileSync(path.resolve(this.dataFolder, `misc.json`), JSON.stringify(this.data.misc, null, 4));\r\n    }\r\n\r\n    getLocale(locale) {\r\n        const file = path.resolve(this.localeFolder, `${locale}.json`);\r\n        if (!fs.existsSync(file)) return null;\r\n        return Utilities.testJSON(fs.readFileSync(file).toString());\r\n    }\r\n\r\n    saveLocale(locale, strings) {\r\n        fs.writeFileSync(path.resolve(this.localeFolder, `${locale}.json`), JSON.stringify(strings, null, 4));\r\n    }\r\n\r\n    getData(key) {\r\n        return this.data[key] || \"\";\r\n    }\r\n\r\n    setData(key, value) {\r\n        this.data[key] = value;\r\n        fs.writeFileSync(path.resolve(this.dataFolder, `${key}.json`), JSON.stringify(value, null, 4));\r\n    }\r\n\r\n    loadCustomCSS() {\r\n        return fs.readFileSync(this.customCSS).toString();\r\n    }\r\n\r\n    saveCustomCSS(css) {\r\n        return fs.writeFileSync(this.customCSS, css);\r\n    }\r\n\r\n    getPluginData(pluginName, key) {\r\n        if (this.pluginData[pluginName] !== undefined) return this.pluginData[pluginName][key] || undefined;\r\n        if (!fs.existsSync(this.getPluginFile(pluginName))) return undefined;\r\n        this.pluginData[pluginName] = JSON.parse(fs.readFileSync(this.getPluginFile(pluginName)));\r\n        return this.pluginData[pluginName][key] || undefined;\r\n    }\r\n\r\n    setPluginData(pluginName, key, value) {\r\n        if (value === undefined) return;\r\n        if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\r\n        this.pluginData[pluginName][key] = value;\r\n        fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\r\n    }\r\n\r\n    deletePluginData(pluginName, key) {\r\n        if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\r\n        delete this.pluginData[pluginName][key];\r\n        fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\r\n    }\r\n};"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/modules/datastore.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var data__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! data */ \"./src/data/data.js\");\n/* harmony import */ var _utilities__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./utilities */ \"./src/modules/utilities.js\");\n\n\n\nconst fs = __webpack_require__(/*! fs */ \"fs\");\n\nconst path = __webpack_require__(/*! path */ \"path\");\n\nconst releaseChannel = DiscordNative.globals.releaseChannel; // Schema\n// =======================\n// %appdata%\\BetterDiscord\n// -> data\n// -> [releaseChannel]\\ (stable/canary/ptb)\n// -> settings.json\n// -> plugins.json\n// -> themes.json\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (new class DataStore {\n constructor() {\n this.data = {\n misc: {}\n };\n this.pluginData = {};\n this.localeHashes = {};\n }\n\n initialize() {\n if (!fs.existsSync(this.baseFolder)) fs.mkdirSync(this.baseFolder);\n if (!fs.existsSync(this.dataFolder)) fs.mkdirSync(this.dataFolder);\n if (!fs.existsSync(this.localeFolder)) fs.mkdirSync(this.localeFolder);\n if (!fs.existsSync(this.localeCache)) fs.writeFileSync(this.localeCache, JSON.stringify({}));\n if (!fs.existsSync(this.BDFile)) fs.writeFileSync(this.BDFile, JSON.stringify(this.data.misc, null, 4));\n if (!fs.existsSync(this.customCSS)) fs.writeFileSync(this.customCSS, \"\");\n const dataFiles = fs.readdirSync(this.dataFolder).filter(f => !fs.statSync(path.resolve(this.dataFolder, f)).isDirectory() && f.endsWith(\".json\"));\n\n for (const file of dataFiles) {\n this.data[file.split(\".\")[0]] = require(path.resolve(this.dataFolder, file));\n }\n\n this.localeHashes = JSON.parse(fs.readFileSync(this.localeCache).toString());\n }\n\n get customCSS() {\n return this._customCSS || (this._customCSS = path.resolve(this.dataFolder, \"custom.css\"));\n }\n\n get baseFolder() {\n return this._baseFolder || (this._baseFolder = path.resolve(data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].dataPath, \"data\"));\n }\n\n get dataFolder() {\n return this._dataFolder || (this._dataFolder = path.resolve(this.baseFolder, `${releaseChannel}`));\n }\n\n get localeFolder() {\n return this._localeFolder || (this._localeFolder = path.resolve(this.baseFolder, `locales`));\n }\n\n get localeCache() {\n return this._localeCache || (this._localeCache = path.resolve(this.localeFolder, `.cache`));\n }\n\n get BDFile() {\n return this._BDFile || (this._BDFile = path.resolve(data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].dataPath, \"data\", `${releaseChannel}.json`));\n }\n\n getPluginFile(pluginName) {\n return path.resolve(data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].dataPath, \"plugins\", pluginName + \".config.json\");\n }\n\n _getFile(key) {\n if (key == \"settings\" || key == \"plugins\" || key == \"themes\") return path.resolve(this.dataFolder, `${key}.json`);\n return path.resolve(this.dataFolder, `misc.json`);\n }\n\n getBDData(key) {\n return this.data.misc[key] || \"\";\n }\n\n setBDData(key, value) {\n this.data.misc[key] = value;\n fs.writeFileSync(path.resolve(this.dataFolder, `misc.json`), JSON.stringify(this.data.misc, null, 4));\n }\n\n getLocale(locale) {\n const file = path.resolve(this.localeFolder, `${locale}.json`);\n if (!fs.existsSync(file)) return null;\n return _utilities__WEBPACK_IMPORTED_MODULE_1__[\"default\"].testJSON(fs.readFileSync(file).toString());\n }\n\n saveLocale(locale, strings) {\n fs.writeFileSync(path.resolve(this.localeFolder, `${locale}.json`), JSON.stringify(strings, null, 4));\n }\n\n getLocaleHash(locale) {\n return this.localeHashes[locale] || \"\";\n }\n\n saveLocaleHash(locale, hash) {\n this.localeHashes[locale] = hash;\n fs.writeFileSync(this.localeCache, JSON.stringify(this.localeHashes, null, 4));\n }\n\n getData(key) {\n return this.data[key] || \"\";\n }\n\n setData(key, value) {\n this.data[key] = value;\n fs.writeFileSync(path.resolve(this.dataFolder, `${key}.json`), JSON.stringify(value, null, 4));\n }\n\n loadCustomCSS() {\n return fs.readFileSync(this.customCSS).toString();\n }\n\n saveCustomCSS(css) {\n return fs.writeFileSync(this.customCSS, css);\n }\n\n getPluginData(pluginName, key) {\n if (this.pluginData[pluginName] !== undefined) return this.pluginData[pluginName][key] || undefined;\n if (!fs.existsSync(this.getPluginFile(pluginName))) return undefined;\n this.pluginData[pluginName] = JSON.parse(fs.readFileSync(this.getPluginFile(pluginName)));\n return this.pluginData[pluginName][key] || undefined;\n }\n\n setPluginData(pluginName, key, value) {\n if (value === undefined) return;\n if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\n this.pluginData[pluginName][key] = value;\n fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\n }\n\n deletePluginData(pluginName, key) {\n if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\n delete this.pluginData[pluginName][key];\n fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\n }\n\n}());//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/modules/datastore.js?1052"],"names":["fs","require","path","releaseChannel","DiscordNative","globals","DataStore","constructor","data","misc","pluginData","localeHashes","initialize","existsSync","baseFolder","mkdirSync","dataFolder","localeFolder","localeCache","writeFileSync","JSON","stringify","BDFile","customCSS","dataFiles","readdirSync","filter","f","statSync","resolve","isDirectory","endsWith","file","split","__non_webpack_require__","parse","readFileSync","toString","_customCSS","_baseFolder","Config","dataPath","_dataFolder","_localeFolder","_localeCache","_BDFile","getPluginFile","pluginName","_getFile","key","getBDData","setBDData","value","getLocale","locale","Utilities","testJSON","saveLocale","strings","getLocaleHash","saveLocaleHash","hash","getData","setData","loadCustomCSS","saveCustomCSS","css","getPluginData","undefined","setPluginData","deletePluginData"],"mappings":"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","file":"./src/modules/datastore.js.js","sourcesContent":["import {Config} from \"data\";\r\nimport Utilities from \"./utilities\";\r\nconst fs = require(\"fs\");\r\nconst path = require(\"path\");\r\nconst releaseChannel = DiscordNative.globals.releaseChannel;\r\n\r\n// Schema\r\n// =======================\r\n// %appdata%\\BetterDiscord\r\n//     -> data\r\n//         -> [releaseChannel]\\ (stable/canary/ptb)\r\n//             -> settings.json\r\n//             -> plugins.json\r\n//             -> themes.json\r\n\r\nexport default new class DataStore {\r\n    constructor() {\r\n        this.data = {misc: {}};\r\n        this.pluginData = {};\r\n        this.localeHashes = {};\r\n    }\r\n\r\n    initialize() {\r\n        if (!fs.existsSync(this.baseFolder)) fs.mkdirSync(this.baseFolder);\r\n        if (!fs.existsSync(this.dataFolder)) fs.mkdirSync(this.dataFolder);\r\n        if (!fs.existsSync(this.localeFolder)) fs.mkdirSync(this.localeFolder);\r\n        if (!fs.existsSync(this.localeCache)) fs.writeFileSync(this.localeCache, JSON.stringify({}));\r\n        if (!fs.existsSync(this.BDFile)) fs.writeFileSync(this.BDFile, JSON.stringify(this.data.misc, null, 4));\r\n        if (!fs.existsSync(this.customCSS)) fs.writeFileSync(this.customCSS, \"\");\r\n        const dataFiles = fs.readdirSync(this.dataFolder).filter(f => !fs.statSync(path.resolve(this.dataFolder, f)).isDirectory() && f.endsWith(\".json\"));\r\n        for (const file of dataFiles) {\r\n            this.data[file.split(\".\")[0]] = __non_webpack_require__(path.resolve(this.dataFolder, file));\r\n        }\r\n        this.localeHashes = JSON.parse(fs.readFileSync(this.localeCache).toString());\r\n    }\r\n\r\n    get customCSS() {return this._customCSS || (this._customCSS = path.resolve(this.dataFolder, \"custom.css\"));}\r\n    get baseFolder() {return this._baseFolder || (this._baseFolder = path.resolve(Config.dataPath, \"data\"));}\r\n    get dataFolder() {return this._dataFolder || (this._dataFolder = path.resolve(this.baseFolder, `${releaseChannel}`));}\r\n    get localeFolder() {return this._localeFolder || (this._localeFolder = path.resolve(this.baseFolder, `locales`));}\r\n    get localeCache() {return this._localeCache || (this._localeCache = path.resolve(this.localeFolder, `.cache`));}\r\n    get BDFile() {return this._BDFile || (this._BDFile = path.resolve(Config.dataPath, \"data\", `${releaseChannel}.json`));}\r\n    getPluginFile(pluginName) {return path.resolve(Config.dataPath, \"plugins\", pluginName + \".config.json\");}\r\n\r\n\r\n    _getFile(key) {\r\n        if (key == \"settings\" || key == \"plugins\" || key == \"themes\") return path.resolve(this.dataFolder, `${key}.json`);\r\n        return path.resolve(this.dataFolder, `misc.json`);\r\n    }\r\n\r\n    getBDData(key) {\r\n        return this.data.misc[key] || \"\";\r\n    }\r\n\r\n    setBDData(key, value) {\r\n        this.data.misc[key] = value;\r\n        fs.writeFileSync(path.resolve(this.dataFolder, `misc.json`), JSON.stringify(this.data.misc, null, 4));\r\n    }\r\n\r\n    getLocale(locale) {\r\n        const file = path.resolve(this.localeFolder, `${locale}.json`);\r\n        if (!fs.existsSync(file)) return null;\r\n        return Utilities.testJSON(fs.readFileSync(file).toString());\r\n    }\r\n\r\n    saveLocale(locale, strings) {\r\n        fs.writeFileSync(path.resolve(this.localeFolder, `${locale}.json`), JSON.stringify(strings, null, 4));\r\n    }\r\n\r\n    getLocaleHash(locale) {\r\n        return this.localeHashes[locale] || \"\";\r\n    }\r\n\r\n    saveLocaleHash(locale, hash) {\r\n        this.localeHashes[locale] = hash;\r\n        fs.writeFileSync(this.localeCache, JSON.stringify(this.localeHashes, null, 4));\r\n    }\r\n\r\n    getData(key) {\r\n        return this.data[key] || \"\";\r\n    }\r\n\r\n    setData(key, value) {\r\n        this.data[key] = value;\r\n        fs.writeFileSync(path.resolve(this.dataFolder, `${key}.json`), JSON.stringify(value, null, 4));\r\n    }\r\n\r\n    loadCustomCSS() {\r\n        return fs.readFileSync(this.customCSS).toString();\r\n    }\r\n\r\n    saveCustomCSS(css) {\r\n        return fs.writeFileSync(this.customCSS, css);\r\n    }\r\n\r\n    getPluginData(pluginName, key) {\r\n        if (this.pluginData[pluginName] !== undefined) return this.pluginData[pluginName][key] || undefined;\r\n        if (!fs.existsSync(this.getPluginFile(pluginName))) return undefined;\r\n        this.pluginData[pluginName] = JSON.parse(fs.readFileSync(this.getPluginFile(pluginName)));\r\n        return this.pluginData[pluginName][key] || undefined;\r\n    }\r\n\r\n    setPluginData(pluginName, key, value) {\r\n        if (value === undefined) return;\r\n        if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\r\n        this.pluginData[pluginName][key] = value;\r\n        fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\r\n    }\r\n\r\n    deletePluginData(pluginName, key) {\r\n        if (this.pluginData[pluginName] === undefined) this.pluginData[pluginName] = {};\r\n        delete this.pluginData[pluginName][key];\r\n        fs.writeFileSync(this.getPluginFile(pluginName), JSON.stringify(this.pluginData[pluginName], null, 4));\r\n    }\r\n};"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/modules/datastore.js\n"); /***/ }), @@ -443,7 +443,7 @@ eval("__webpack_require__.r(__webpack_exports__);\nconst EventEmitter = __webpac /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _data_strings__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../data/strings */ \"./src/data/strings.js\");\n/* harmony import */ var _discordmodules__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./discordmodules */ \"./src/modules/discordmodules.js\");\n/* harmony import */ var _utilities__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./utilities */ \"./src/modules/utilities.js\");\n/* harmony import */ var _emitter__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./emitter */ \"./src/modules/emitter.js\");\n/* harmony import */ var _datastore__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./datastore */ \"./src/modules/datastore.js\");\n\n\n\n\n\n\nconst request = __webpack_require__(/*! request */ \"request\");\n\nconst {\n Dispatcher,\n DiscordConstants,\n UserSettingsStore\n} = _discordmodules__WEBPACK_IMPORTED_MODULE_1__[\"default\"];\n/* harmony default export */ __webpack_exports__[\"default\"] = (new class LocaleManager {\n get discordLocale() {\n return UserSettingsStore.locale.split(\"-\")[0];\n }\n\n get defaultLocale() {\n return \"en\";\n }\n\n constructor() {\n this.locale = \"\";\n this.strings = {};\n }\n\n async initialize() {\n await this.setLocale(this.discordLocale);\n Dispatcher.subscribe(DiscordConstants.ActionTypes.USER_SETTINGS_UPDATE, ({\n settings\n }) => {\n const newLocale = settings.locale;\n if (newLocale && newLocale != this.locale) this.setLocale(newLocale.split(\"-\")[0]);\n });\n }\n\n async setLocale(newLocale) {\n let newStrings;\n\n if (newLocale != this.defaultLocale) {\n const savedStrings = _datastore__WEBPACK_IMPORTED_MODULE_4__[\"default\"].getLocale(newLocale);\n newStrings = savedStrings || (await this.downloadLocale(newLocale));\n if (!newStrings) return this.setLocale(this.defaultLocale);\n } else {\n newStrings = _data_strings__WEBPACK_IMPORTED_MODULE_0__[\"default\"];\n }\n\n this.locale = newLocale;\n _utilities__WEBPACK_IMPORTED_MODULE_2__[\"default\"].extend(this.strings, newStrings);\n _emitter__WEBPACK_IMPORTED_MODULE_3__[\"default\"].emit(\"strings-updated\");\n }\n\n downloadLocale(locale) {\n return new Promise(resolve => {\n const options = {\n url: `https://raw.githubusercontent.com/rauenzi/BetterDiscordApp/development/data/locales/${locale}.json`,\n //`https://rauenzi.github.io/BetterDiscordApp/data/locales/${discordLocale}.json`,\n timeout: 2000,\n json: true\n };\n request.get(options, (err, resp, newStrings) => {\n if (err || resp.statusCode !== 200) return resolve(null);\n _datastore__WEBPACK_IMPORTED_MODULE_4__[\"default\"].saveLocale(locale, newStrings);\n resolve(newStrings);\n });\n });\n }\n\n}());//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9Db3JlLy4vc3JjL21vZHVsZXMvbG9jYWxlbWFuYWdlci5qcz8yMzlkIl0sIm5hbWVzIjpbInJlcXVlc3QiLCJyZXF1aXJlIiwiRGlzcGF0Y2hlciIsIkRpc2NvcmRDb25zdGFudHMiLCJVc2VyU2V0dGluZ3NTdG9yZSIsIkRpc2NvcmRNb2R1bGVzIiwiTG9jYWxlTWFuYWdlciIsImRpc2NvcmRMb2NhbGUiLCJsb2NhbGUiLCJzcGxpdCIsImRlZmF1bHRMb2NhbGUiLCJjb25zdHJ1Y3RvciIsInN0cmluZ3MiLCJpbml0aWFsaXplIiwic2V0TG9jYWxlIiwic3Vic2NyaWJlIiwiQWN0aW9uVHlwZXMiLCJVU0VSX1NFVFRJTkdTX1VQREFURSIsInNldHRpbmdzIiwibmV3TG9jYWxlIiwibmV3U3RyaW5ncyIsInNhdmVkU3RyaW5ncyIsIkRhdGFTdG9yZSIsImdldExvY2FsZSIsImRvd25sb2FkTG9jYWxlIiwiRGVmYXVsdFN0cmluZ3MiLCJVdGlsaXRpZXMiLCJleHRlbmQiLCJFdmVudHMiLCJlbWl0IiwiUHJvbWlzZSIsInJlc29sdmUiLCJvcHRpb25zIiwidXJsIiwidGltZW91dCIsImpzb24iLCJnZXQiLCJlcnIiLCJyZXNwIiwic3RhdHVzQ29kZSIsInNhdmVMb2NhbGUiXSwibWFwcGluZ3MiOiJBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBQ0EsTUFBTUEsT0FBTyxHQUFHQyxtQkFBTyxDQUFDLHdCQUFELENBQXZCOztBQUVBLE1BQU07QUFBQ0MsWUFBRDtBQUFhQyxrQkFBYjtBQUErQkM7QUFBL0IsSUFBb0RDLHVEQUExRDtBQUVlLG1FQUFJLE1BQU1DLGFBQU4sQ0FBb0I7QUFDdEMsTUFBSUMsYUFBSixHQUFvQjtBQUFDLFdBQU9ILGlCQUFpQixDQUFDSSxNQUFsQixDQUF5QkMsS0FBekIsQ0FBK0IsR0FBL0IsRUFBb0MsQ0FBcEMsQ0FBUDtBQUErQzs7QUFDcEUsTUFBSUMsYUFBSixHQUFvQjtBQUFDLFdBQU8sSUFBUDtBQUFhOztBQUVsQ0MsYUFBVyxHQUFHO0FBQ1AsU0FBS0gsTUFBTCxHQUFjLEVBQWQ7QUFDQSxTQUFLSSxPQUFMLEdBQWUsRUFBZjtBQUNOOztBQUVELFFBQU1DLFVBQU4sR0FBbUI7QUFDWixVQUFNLEtBQUtDLFNBQUwsQ0FBZSxLQUFLUCxhQUFwQixDQUFOO0FBQ0FMLGNBQVUsQ0FBQ2EsU0FBWCxDQUFxQlosZ0JBQWdCLENBQUNhLFdBQWpCLENBQTZCQyxvQkFBbEQsRUFBd0UsQ0FBQztBQUFDQztBQUFELEtBQUQsS0FBZ0I7QUFDcEYsWUFBTUMsU0FBUyxHQUFHRCxRQUFRLENBQUNWLE1BQTNCO0FBQ0EsVUFBSVcsU0FBUyxJQUFJQSxTQUFTLElBQUksS0FBS1gsTUFBbkMsRUFBMkMsS0FBS00sU0FBTCxDQUFlSyxTQUFTLENBQUNWLEtBQVYsQ0FBZ0IsR0FBaEIsRUFBcUIsQ0FBckIsQ0FBZjtBQUM5QyxLQUhEO0FBSU47O0FBRUQsUUFBTUssU0FBTixDQUFnQkssU0FBaEIsRUFBMkI7QUFDcEIsUUFBSUMsVUFBSjs7QUFDQSxRQUFJRCxTQUFTLElBQUksS0FBS1QsYUFBdEIsRUFBcUM7QUFDakMsWUFBTVcsWUFBWSxHQUFHQyxrREFBUyxDQUFDQyxTQUFWLENBQW9CSixTQUFwQixDQUFyQjtBQUNBQyxnQkFBVSxHQUFHQyxZQUFZLEtBQUksTUFBTSxLQUFLRyxjQUFMLENBQW9CTCxTQUFwQixDQUFWLENBQXpCO0FBQ0EsVUFBSSxDQUFDQyxVQUFMLEVBQWlCLE9BQU8sS0FBS04sU0FBTCxDQUFlLEtBQUtKLGFBQXBCLENBQVA7QUFDcEIsS0FKRCxNQUtLO0FBQ0RVLGdCQUFVLEdBQUdLLHFEQUFiO0FBQ0g7O0FBQ1AsU0FBS2pCLE1BQUwsR0FBY1csU0FBZDtBQUNBTyxzREFBUyxDQUFDQyxNQUFWLENBQWlCLEtBQUtmLE9BQXRCLEVBQStCUSxVQUEvQjtBQUNBUSxvREFBTSxDQUFDQyxJQUFQLENBQVksaUJBQVo7QUFDQTs7QUFFREwsZ0JBQWMsQ0FBQ2hCLE1BQUQsRUFBUztBQUN0QixXQUFPLElBQUlzQixPQUFKLENBQVlDLE9BQU8sSUFBSTtBQUM3QixZQUFNQyxPQUFPLEdBQUc7QUFDZkMsV0FBRyxFQUFHLHVGQUFzRnpCLE1BQU8sT0FEcEY7QUFDMkY7QUFDMUcwQixlQUFPLEVBQUUsSUFGTTtBQUdmQyxZQUFJLEVBQUU7QUFIUyxPQUFoQjtBQUtBbkMsYUFBTyxDQUFDb0MsR0FBUixDQUFZSixPQUFaLEVBQXFCLENBQUNLLEdBQUQsRUFBTUMsSUFBTixFQUFZbEIsVUFBWixLQUEyQjtBQUNuQyxZQUFJaUIsR0FBRyxJQUFJQyxJQUFJLENBQUNDLFVBQUwsS0FBb0IsR0FBL0IsRUFBb0MsT0FBT1IsT0FBTyxDQUFDLElBQUQsQ0FBZDtBQUNwQ1QsMERBQVMsQ0FBQ2tCLFVBQVYsQ0FBcUJoQyxNQUFyQixFQUE2QlksVUFBN0I7QUFDWlcsZUFBTyxDQUFDWCxVQUFELENBQVA7QUFDQSxPQUpEO0FBS0EsS0FYTSxDQUFQO0FBWUE7O0FBN0NxQyxDQUF4QixFQUFmIiwiZmlsZSI6Ii4vc3JjL21vZHVsZXMvbG9jYWxlbWFuYWdlci5qcy5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBEZWZhdWx0U3RyaW5ncyBmcm9tIFwiLi4vZGF0YS9zdHJpbmdzXCI7XHJcbmltcG9ydCBEaXNjb3JkTW9kdWxlcyBmcm9tIFwiLi9kaXNjb3JkbW9kdWxlc1wiO1xyXG5pbXBvcnQgVXRpbGl0aWVzIGZyb20gXCIuL3V0aWxpdGllc1wiO1xyXG5pbXBvcnQgRXZlbnRzIGZyb20gXCIuL2VtaXR0ZXJcIjtcclxuaW1wb3J0IERhdGFTdG9yZSBmcm9tIFwiLi9kYXRhc3RvcmVcIjtcclxuY29uc3QgcmVxdWVzdCA9IHJlcXVpcmUoXCJyZXF1ZXN0XCIpO1xyXG5cclxuY29uc3Qge0Rpc3BhdGNoZXIsIERpc2NvcmRDb25zdGFudHMsIFVzZXJTZXR0aW5nc1N0b3JlfSA9IERpc2NvcmRNb2R1bGVzO1xyXG5cclxuZXhwb3J0IGRlZmF1bHQgbmV3IGNsYXNzIExvY2FsZU1hbmFnZXIge1xyXG5cdGdldCBkaXNjb3JkTG9jYWxlKCkge3JldHVybiBVc2VyU2V0dGluZ3NTdG9yZS5sb2NhbGUuc3BsaXQoXCItXCIpWzBdO31cclxuXHRnZXQgZGVmYXVsdExvY2FsZSgpIHtyZXR1cm4gXCJlblwiO31cclxuXHJcblx0Y29uc3RydWN0b3IoKSB7XHJcbiAgICAgICAgdGhpcy5sb2NhbGUgPSBcIlwiO1xyXG4gICAgICAgIHRoaXMuc3RyaW5ncyA9IHt9O1xyXG5cdH1cclxuXHRcclxuXHRhc3luYyBpbml0aWFsaXplKCkge1xyXG4gICAgICAgIGF3YWl0IHRoaXMuc2V0TG9jYWxlKHRoaXMuZGlzY29yZExvY2FsZSk7XHJcbiAgICAgICAgRGlzcGF0Y2hlci5zdWJzY3JpYmUoRGlzY29yZENvbnN0YW50cy5BY3Rpb25UeXBlcy5VU0VSX1NFVFRJTkdTX1VQREFURSwgKHtzZXR0aW5nc30pID0+IHtcclxuICAgICAgICAgICAgY29uc3QgbmV3TG9jYWxlID0gc2V0dGluZ3MubG9jYWxlO1xyXG4gICAgICAgICAgICBpZiAobmV3TG9jYWxlICYmIG5ld0xvY2FsZSAhPSB0aGlzLmxvY2FsZSkgdGhpcy5zZXRMb2NhbGUobmV3TG9jYWxlLnNwbGl0KFwiLVwiKVswXSk7XHJcbiAgICAgICAgfSk7XHJcblx0fVxyXG5cclxuXHRhc3luYyBzZXRMb2NhbGUobmV3TG9jYWxlKSB7XHJcbiAgICAgICAgbGV0IG5ld1N0cmluZ3M7XHJcbiAgICAgICAgaWYgKG5ld0xvY2FsZSAhPSB0aGlzLmRlZmF1bHRMb2NhbGUpIHtcclxuICAgICAgICAgICAgY29uc3Qgc2F2ZWRTdHJpbmdzID0gRGF0YVN0b3JlLmdldExvY2FsZShuZXdMb2NhbGUpO1xyXG4gICAgICAgICAgICBuZXdTdHJpbmdzID0gc2F2ZWRTdHJpbmdzIHx8IGF3YWl0IHRoaXMuZG93bmxvYWRMb2NhbGUobmV3TG9jYWxlKTtcclxuICAgICAgICAgICAgaWYgKCFuZXdTdHJpbmdzKSByZXR1cm4gdGhpcy5zZXRMb2NhbGUodGhpcy5kZWZhdWx0TG9jYWxlKTtcclxuICAgICAgICB9XHJcbiAgICAgICAgZWxzZSB7XHJcbiAgICAgICAgICAgIG5ld1N0cmluZ3MgPSBEZWZhdWx0U3RyaW5ncztcclxuICAgICAgICB9XHJcblx0XHR0aGlzLmxvY2FsZSA9IG5ld0xvY2FsZTtcclxuXHRcdFV0aWxpdGllcy5leHRlbmQodGhpcy5zdHJpbmdzLCBuZXdTdHJpbmdzKTtcclxuXHRcdEV2ZW50cy5lbWl0KFwic3RyaW5ncy11cGRhdGVkXCIpO1xyXG5cdH1cclxuXHJcblx0ZG93bmxvYWRMb2NhbGUobG9jYWxlKSB7XHJcblx0XHRyZXR1cm4gbmV3IFByb21pc2UocmVzb2x2ZSA9PiB7XHJcblx0XHRcdGNvbnN0IG9wdGlvbnMgPSB7XHJcblx0XHRcdFx0dXJsOiBgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3JhdWVuemkvQmV0dGVyRGlzY29yZEFwcC9kZXZlbG9wbWVudC9kYXRhL2xvY2FsZXMvJHtsb2NhbGV9Lmpzb25gLC8vYGh0dHBzOi8vcmF1ZW56aS5naXRodWIuaW8vQmV0dGVyRGlzY29yZEFwcC9kYXRhL2xvY2FsZXMvJHtkaXNjb3JkTG9jYWxlfS5qc29uYCxcclxuXHRcdFx0XHR0aW1lb3V0OiAyMDAwLFxyXG5cdFx0XHRcdGpzb246IHRydWVcclxuXHRcdFx0fTtcclxuXHRcdFx0cmVxdWVzdC5nZXQob3B0aW9ucywgKGVyciwgcmVzcCwgbmV3U3RyaW5ncykgPT4ge1xyXG4gICAgICAgICAgICAgICAgaWYgKGVyciB8fCByZXNwLnN0YXR1c0NvZGUgIT09IDIwMCkgcmV0dXJuIHJlc29sdmUobnVsbCk7XHJcbiAgICAgICAgICAgICAgICBEYXRhU3RvcmUuc2F2ZUxvY2FsZShsb2NhbGUsIG5ld1N0cmluZ3MpO1xyXG5cdFx0XHRcdHJlc29sdmUobmV3U3RyaW5ncyk7XHJcblx0XHRcdH0pO1xyXG5cdFx0fSk7XHJcblx0fVxyXG59OyJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/modules/localemanager.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _data_strings__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../data/strings */ \"./src/data/strings.js\");\n/* harmony import */ var _discordmodules__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./discordmodules */ \"./src/modules/discordmodules.js\");\n/* harmony import */ var _utilities__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./utilities */ \"./src/modules/utilities.js\");\n/* harmony import */ var _emitter__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./emitter */ \"./src/modules/emitter.js\");\n/* harmony import */ var _datastore__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./datastore */ \"./src/modules/datastore.js\");\n\n\n\n\n\n\nconst request = __webpack_require__(/*! request */ \"request\");\n\nconst {\n Dispatcher,\n DiscordConstants,\n UserSettingsStore\n} = _discordmodules__WEBPACK_IMPORTED_MODULE_1__[\"default\"];\n/* harmony default export */ __webpack_exports__[\"default\"] = (new class LocaleManager {\n get discordLocale() {\n return UserSettingsStore.locale.split(\"-\")[0];\n }\n\n get defaultLocale() {\n return \"en\";\n }\n\n constructor() {\n this.locale = \"\";\n this.strings = {};\n }\n\n async initialize() {\n await this.setLocale(this.discordLocale);\n Dispatcher.subscribe(DiscordConstants.ActionTypes.USER_SETTINGS_UPDATE, ({\n settings\n }) => {\n const newLocale = settings.locale;\n if (newLocale && newLocale != this.locale) this.setLocale(newLocale.split(\"-\")[0]);\n });\n }\n\n async setLocale(newLocale) {\n let newStrings;\n\n if (newLocale != this.defaultLocale) {\n newStrings = await this.getLocaleStrings(newLocale);\n if (!newStrings) return this.setLocale(this.defaultLocale);\n } else {\n newStrings = _data_strings__WEBPACK_IMPORTED_MODULE_0__[\"default\"];\n }\n\n this.locale = newLocale;\n _utilities__WEBPACK_IMPORTED_MODULE_2__[\"default\"].extend(this.strings, newStrings);\n _emitter__WEBPACK_IMPORTED_MODULE_3__[\"default\"].emit(\"strings-updated\");\n }\n\n async getLocaleStrings(locale) {\n const hash = _datastore__WEBPACK_IMPORTED_MODULE_4__[\"default\"].getLocaleHash(locale);\n if (!hash) return await this.downloadLocale(locale);\n const invalid = await this.downloadLocale(locale, hash);\n if (!invalid) return _datastore__WEBPACK_IMPORTED_MODULE_4__[\"default\"].getLocale(locale);\n return invalid;\n }\n\n downloadLocale(locale, hash = \"\") {\n return new Promise(resolve => {\n const options = {\n url: _utilities__WEBPACK_IMPORTED_MODULE_2__[\"default\"].repoUrl(`data/locales/${locale}.json`),\n timeout: 2000,\n json: true\n };\n if (hash) options.headers = {\n \"If-None-Match\": hash\n };\n console.log(options.headers);\n request.get(options, (err, resp, newStrings) => {\n if (err || resp.statusCode !== 200) return resolve(null);\n console.log(resp);\n _datastore__WEBPACK_IMPORTED_MODULE_4__[\"default\"].saveLocale(locale, newStrings);\n _datastore__WEBPACK_IMPORTED_MODULE_4__[\"default\"].saveLocaleHash(locale, resp.headers.etag);\n resolve(newStrings);\n });\n });\n }\n\n}());//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9Db3JlLy4vc3JjL21vZHVsZXMvbG9jYWxlbWFuYWdlci5qcz8yMzlkIl0sIm5hbWVzIjpbInJlcXVlc3QiLCJyZXF1aXJlIiwiRGlzcGF0Y2hlciIsIkRpc2NvcmRDb25zdGFudHMiLCJVc2VyU2V0dGluZ3NTdG9yZSIsIkRpc2NvcmRNb2R1bGVzIiwiTG9jYWxlTWFuYWdlciIsImRpc2NvcmRMb2NhbGUiLCJsb2NhbGUiLCJzcGxpdCIsImRlZmF1bHRMb2NhbGUiLCJjb25zdHJ1Y3RvciIsInN0cmluZ3MiLCJpbml0aWFsaXplIiwic2V0TG9jYWxlIiwic3Vic2NyaWJlIiwiQWN0aW9uVHlwZXMiLCJVU0VSX1NFVFRJTkdTX1VQREFURSIsInNldHRpbmdzIiwibmV3TG9jYWxlIiwibmV3U3RyaW5ncyIsImdldExvY2FsZVN0cmluZ3MiLCJEZWZhdWx0U3RyaW5ncyIsIlV0aWxpdGllcyIsImV4dGVuZCIsIkV2ZW50cyIsImVtaXQiLCJoYXNoIiwiRGF0YVN0b3JlIiwiZ2V0TG9jYWxlSGFzaCIsImRvd25sb2FkTG9jYWxlIiwiaW52YWxpZCIsImdldExvY2FsZSIsIlByb21pc2UiLCJyZXNvbHZlIiwib3B0aW9ucyIsInVybCIsInJlcG9VcmwiLCJ0aW1lb3V0IiwianNvbiIsImhlYWRlcnMiLCJjb25zb2xlIiwibG9nIiwiZ2V0IiwiZXJyIiwicmVzcCIsInN0YXR1c0NvZGUiLCJzYXZlTG9jYWxlIiwic2F2ZUxvY2FsZUhhc2giLCJldGFnIl0sIm1hcHBpbmdzIjoiQUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUNBLE1BQU1BLE9BQU8sR0FBR0MsbUJBQU8sQ0FBQyx3QkFBRCxDQUF2Qjs7QUFFQSxNQUFNO0FBQUNDLFlBQUQ7QUFBYUMsa0JBQWI7QUFBK0JDO0FBQS9CLElBQW9EQyx1REFBMUQ7QUFFZSxtRUFBSSxNQUFNQyxhQUFOLENBQW9CO0FBQ3RDLE1BQUlDLGFBQUos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LENBQXFCdkMsTUFBckIsRUFBNkJZLFVBQTdCO0FBQ0FRLDBEQUFTLENBQUNvQixjQUFWLENBQXlCeEMsTUFBekIsRUFBaUNxQyxJQUFJLENBQUNMLE9BQUwsQ0FBYVMsSUFBOUM7QUFDQWYsZUFBTyxDQUFDZCxVQUFELENBQVA7QUFDQSxPQU5EO0FBT0EsS0FmTSxDQUFQO0FBZ0JBOztBQXhEcUMsQ0FBeEIsRUFBZiIsImZpbGUiOiIuL3NyYy9tb2R1bGVzL2xvY2FsZW1hbmFnZXIuanMuanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgRGVmYXVsdFN0cmluZ3MgZnJvbSBcIi4uL2RhdGEvc3RyaW5nc1wiO1xyXG5pbXBvcnQgRGlzY29yZE1vZHVsZXMgZnJvbSBcIi4vZGlzY29yZG1vZHVsZXNcIjtcclxuaW1wb3J0IFV0aWxpdGllcyBmcm9tIFwiLi91dGlsaXRpZXNcIjtcclxuaW1wb3J0IEV2ZW50cyBmcm9tIFwiLi9lbWl0dGVyXCI7XHJcbmltcG9ydCBEYXRhU3RvcmUgZnJvbSBcIi4vZGF0YXN0b3JlXCI7XHJcbmNvbnN0IHJlcXVlc3QgPSByZXF1aXJlKFwicmVxdWVzdFwiKTtcclxuXHJcbmNvbnN0IHtEaXNwYXRjaGVyLCBEaXNjb3JkQ29uc3RhbnRzLCBVc2VyU2V0dGluZ3NTdG9yZX0gPSBEaXNjb3JkTW9kdWxlcztcclxuXHJcbmV4cG9ydCBkZWZhdWx0IG5ldyBjbGFzcyBMb2NhbGVNYW5hZ2VyIHtcclxuXHRnZXQgZGlzY29yZExvY2FsZSgpIHtyZXR1cm4gVXNlclNldHRpbmdzU3RvcmUubG9jYWxlLnNwbGl0KFwiLVwiKVswXTt9XHJcblx0Z2V0IGRlZmF1bHRMb2NhbGUoKSB7cmV0dXJuIFwiZW5cIjt9XHJcblxyXG5cdGNvbnN0cnVjdG9yKCkge1xyXG4gICAgICAgIHRoaXMubG9jYWxlID0gXCJcIjtcclxuICAgICAgICB0aGlzLnN0cmluZ3MgPSB7fTtcclxuXHR9XHJcblxyXG5cdGFzeW5jIGluaXRpYWxpemUoKSB7XHJcbiAgICAgICAgYXdhaXQgdGhpcy5zZXRMb2NhbGUodGhpcy5kaXNjb3JkTG9jYWxlKTtcclxuICAgICAgICBEaXNwYXRjaGVyLnN1YnNjcmliZShEaXNjb3JkQ29uc3RhbnRzLkFjdGlvblR5cGVzLlVTRVJfU0VUVElOR1NfVVBEQVRFLCAoe3NldHRpbmdzfSkgPT4ge1xyXG4gICAgICAgICAgICBjb25zdCBuZXdMb2NhbGUgPSBzZXR0aW5ncy5sb2NhbGU7XHJcbiAgICAgICAgICAgIGlmIChuZXdMb2NhbGUgJiYgbmV3TG9jYWxlICE9IHRoaXMubG9jYWxlKSB0aGlzLnNldExvY2FsZShuZXdMb2NhbGUuc3BsaXQoXCItXCIpWzBdKTtcclxuICAgICAgICB9KTtcclxuXHR9XHJcblxyXG5cdGFzeW5jIHNldExvY2FsZShuZXdMb2NhbGUpIHtcclxuICAgICAgICBsZXQgbmV3U3RyaW5ncztcclxuICAgICAgICBpZiAobmV3TG9jYWxlICE9IHRoaXMuZGVmYXVsdExvY2FsZSkge1xyXG5cdFx0XHRuZXdTdHJpbmdzID0gYXdhaXQgdGhpcy5nZXRMb2NhbGVTdHJpbmdzKG5ld0xvY2FsZSk7XHJcbiAgICAgICAgICAgIGlmICghbmV3U3RyaW5ncykgcmV0dXJuIHRoaXMuc2V0TG9jYWxlKHRoaXMuZGVmYXVsdExvY2FsZSk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIGVsc2Uge1xyXG4gICAgICAgICAgICBuZXdTdHJpbmdzID0gRGVmYXVsdFN0cmluZ3M7XHJcbiAgICAgICAgfVxyXG5cdFx0dGhpcy5sb2NhbGUgPSBuZXdMb2NhbGU7XHJcblx0XHRVdGlsaXRpZXMuZXh0ZW5kKHRoaXMuc3RyaW5ncywgbmV3U3RyaW5ncyk7XHJcblx0XHRFdmVudHMuZW1pdChcInN0cmluZ3MtdXBkYXRlZFwiKTtcclxuXHR9XHJcblxyXG5cdGFzeW5jIGdldExvY2FsZVN0cmluZ3MobG9jYWxlKSB7XHJcblx0XHRjb25zdCBoYXNoID0gRGF0YVN0b3JlLmdldExvY2FsZUhhc2gobG9jYWxlKTtcclxuXHRcdGlmICghaGFzaCkgcmV0dXJuIGF3YWl0IHRoaXMuZG93bmxvYWRMb2NhbGUobG9jYWxlKTtcclxuXHRcdGNvbnN0IGludmFsaWQgPSBhd2FpdCB0aGlzLmRvd25sb2FkTG9jYWxlKGxvY2FsZSwgaGFzaCk7XHJcblx0XHRpZiAoIWludmFsaWQpIHJldHVybiBEYXRhU3RvcmUuZ2V0TG9jYWxlKGxvY2FsZSk7XHJcblx0XHRyZXR1cm4gaW52YWxpZDtcclxuXHR9XHJcblxyXG5cdGRvd25sb2FkTG9jYWxlKGxvY2FsZSwgaGFzaCA9IFwiXCIpIHtcclxuXHRcdHJldHVybiBuZXcgUHJvbWlzZShyZXNvbHZlID0+IHtcclxuXHRcdFx0Y29uc3Qgb3B0aW9ucyA9IHtcclxuXHRcdFx0XHR1cmw6IFV0aWxpdGllcy5yZXBvVXJsKGBkYXRhL2xvY2FsZXMvJHtsb2NhbGV9Lmpzb25gKSxcclxuXHRcdFx0XHR0aW1lb3V0OiAyMDAwLFxyXG5cdFx0XHRcdGpzb246IHRydWVcclxuXHRcdFx0fTtcclxuXHRcdFx0aWYgKGhhc2gpIG9wdGlvbnMuaGVhZGVycyA9IHtcIklmLU5vbmUtTWF0Y2hcIjogaGFzaH07XHJcblx0XHRcdGNvbnNvbGUubG9nKG9wdGlvbnMuaGVhZGVycyk7XHJcblx0XHRcdHJlcXVlc3QuZ2V0KG9wdGlvbnMsIChlcnIsIHJlc3AsIG5ld1N0cmluZ3MpID0+IHtcclxuXHRcdFx0XHRpZiAoZXJyIHx8IHJlc3Auc3RhdHVzQ29kZSAhPT0gMjAwKSByZXR1cm4gcmVzb2x2ZShudWxsKTtcclxuXHRcdFx0XHRjb25zb2xlLmxvZyhyZXNwKTtcclxuXHRcdFx0XHREYXRhU3RvcmUuc2F2ZUxvY2FsZShsb2NhbGUsIG5ld1N0cmluZ3MpO1xyXG5cdFx0XHRcdERhdGFTdG9yZS5zYXZlTG9jYWxlSGFzaChsb2NhbGUsIHJlc3AuaGVhZGVycy5ldGFnKTtcclxuXHRcdFx0XHRyZXNvbHZlKG5ld1N0cmluZ3MpO1xyXG5cdFx0XHR9KTtcclxuXHRcdH0pO1xyXG5cdH1cclxufTsiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/modules/localemanager.js\n"); /***/ }), @@ -563,7 +563,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var data /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"default\", function() { return Utilities; });\n/* harmony import */ var _logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./logger */ \"./src/modules/logger.js\");\n\nclass Utilities {\n /**\r\n * Parses a string of HTML and returns the results. If the second parameter is true,\r\n * the parsed HTML will be returned as a document fragment {@see https://developer.mozilla.org/en-US/docs/Web/API/DocumentFragment}.\r\n * This is extremely useful if you have a list of elements at the top level, they can then be appended all at once to another node.\r\n *\r\n * If the second parameter is false, then the return value will be the list of parsed\r\n * nodes and there were multiple top level nodes, otherwise the single node is returned.\r\n * @param {string} html - HTML to be parsed\r\n * @param {boolean} [fragment=false] - Whether or not the return should be the raw `DocumentFragment`\r\n * @returns {(DocumentFragment|NodeList|HTMLElement)} - The result of HTML parsing\r\n */\n static parseHTML(html, fragment = false) {\n const template = document.createElement(\"template\");\n template.innerHTML = html;\n const node = template.content.cloneNode(true);\n if (fragment) return node;\n return node.childNodes.length > 1 ? node.childNodes : node.childNodes[0];\n }\n\n static getTextArea() {\n return $(\".channelTextArea-1LDbYG textarea\");\n }\n\n static insertText(textarea, text) {\n textarea.focus();\n textarea.selectionStart = 0;\n textarea.selectionEnd = textarea.value.length;\n document.execCommand(\"insertText\", false, text);\n }\n\n static escape(s) {\n return s.replace(/[-/\\\\^$*+?.()|[\\]{}]/g, \"\\\\$&\");\n }\n\n static testJSON(data) {\n try {\n return JSON.parse(data);\n } catch (err) {\n return false;\n }\n }\n\n static suppressErrors(method, message) {\n return (...params) => {\n try {\n return method(...params);\n } catch (e) {\n _logger__WEBPACK_IMPORTED_MODULE_0__[\"default\"].stacktrace(\"SuppressedError\", \"Error occurred in \" + message, e);\n }\n };\n }\n\n static monkeyPatch(what, methodName, options) {\n const {\n before,\n after,\n instead,\n once = false,\n silent = false,\n force = false\n } = options;\n const displayName = options.displayName || what.displayName || what.name || what.constructor.displayName || what.constructor.name;\n if (!silent) console.log(\"patch\", methodName, \"of\", displayName); // eslint-disable-line no-console\n\n if (!what[methodName]) {\n if (force) what[methodName] = function () {};else return console.error(methodName, \"does not exist for\", displayName); // eslint-disable-line no-console\n }\n\n const origMethod = what[methodName];\n\n const cancel = () => {\n if (!silent) console.log(\"unpatch\", methodName, \"of\", displayName); // eslint-disable-line no-console\n\n what[methodName] = origMethod;\n };\n\n what[methodName] = function () {\n const data = {\n thisObject: this,\n methodArguments: arguments,\n cancelPatch: cancel,\n originalMethod: origMethod,\n callOriginalMethod: () => data.returnValue = data.originalMethod.apply(data.thisObject, data.methodArguments)\n };\n\n if (instead) {\n const tempRet = Utilities.suppressErrors(instead, \"`instead` callback of \" + what[methodName].displayName)(data);\n if (tempRet !== undefined) data.returnValue = tempRet;\n } else {\n if (before) Utilities.suppressErrors(before, \"`before` callback of \" + what[methodName].displayName)(data);\n data.callOriginalMethod();\n if (after) Utilities.suppressErrors(after, \"`after` callback of \" + what[methodName].displayName)(data);\n }\n\n if (once) cancel();\n return data.returnValue;\n };\n\n what[methodName].__monkeyPatched = true;\n if (!what[methodName].__originalMethod) what[methodName].__originalMethod = origMethod;\n what[methodName].displayName = \"patched \" + (what[methodName].displayName || methodName);\n return cancel;\n }\n\n static onRemoved(node, callback) {\n const observer = new MutationObserver(mutations => {\n for (let m = 0; m < mutations.length; m++) {\n const mutation = mutations[m];\n const nodes = Array.from(mutation.removedNodes);\n const directMatch = nodes.indexOf(node) > -1;\n const parentMatch = nodes.some(parent => parent.contains(node));\n\n if (directMatch || parentMatch) {\n observer.disconnect();\n callback();\n }\n }\n });\n observer.observe(document.body, {\n subtree: true,\n childList: true\n });\n }\n\n static isEmpty(obj) {\n if (obj == null || obj == undefined || obj == \"\") return true;\n if (typeof obj !== \"object\") return false;\n if (Array.isArray(obj)) return obj.length == 0;\n\n for (const key in obj) {\n if (obj.hasOwnProperty(key)) return false;\n }\n\n return true;\n }\n /**\r\n * Generates an automatically memoizing version of an object.\r\n * @author Zerebos\r\n * @param {Object} object - object to memoize\r\n * @returns {Proxy} the proxy to the object that memoizes properties\r\n */\n\n\n static memoizeObject(object) {\n const proxy = new Proxy(object, {\n get: function (obj, mod) {\n if (!obj.hasOwnProperty(mod)) return undefined;\n\n if (Object.getOwnPropertyDescriptor(obj, mod).get) {\n const value = obj[mod];\n delete obj[mod];\n obj[mod] = value;\n }\n\n return obj[mod];\n },\n set: function (obj, mod, value) {\n if (obj.hasOwnProperty(mod)) return _logger__WEBPACK_IMPORTED_MODULE_0__[\"default\"].error(\"MemoizedObject\", \"Trying to overwrite existing property\");\n obj[mod] = value;\n return obj[mod];\n }\n });\n Object.defineProperty(proxy, \"hasOwnProperty\", {\n value: function (prop) {\n return this[prop] !== undefined;\n }\n });\n return proxy;\n }\n /**\r\n * Deep extends an object with a set of other objects. Objects later in the list\r\n * of `extenders` have priority, that is to say if one sets a key to be a primitive,\r\n * it will be overwritten with the next one with the same key. If it is an object,\r\n * and the keys match, the object is extended. This happens recursively.\r\n * @param {object} extendee - Object to be extended\r\n * @param {...object} extenders - Objects to extend with\r\n * @returns {object} - A reference to `extendee`\r\n */\n\n\n static extend(extendee, ...extenders) {\n for (let i = 0; i < extenders.length; i++) {\n for (const key in extenders[i]) {\n if (extenders[i].hasOwnProperty(key)) {\n if (typeof extendee[key] === \"object\" && typeof extenders[i][key] === \"object\") this.extend(extendee[key], extenders[i][key]);else if (typeof extenders[i][key] === \"object\") extendee[key] = {}, this.extend(extendee[key], extenders[i][key]);else extendee[key] = extenders[i][key];\n }\n }\n }\n\n return extendee;\n }\n /**\r\n * Format strings with placeholders (`{{placeholder}}`) into full strings.\r\n * Quick example: `PluginUtilities.formatString(\"Hello, {{user}}\", {user: \"Zerebos\"})`\r\n * would return \"Hello, Zerebos\".\r\n * @param {string} string - string to format\r\n * @param {object} values - object literal of placeholders to replacements\r\n * @returns {string} the properly formatted string\r\n */\n\n\n static formatString(string, values) {\n for (const val in values) {\n let replacement = values[val];\n if (Array.isArray(replacement)) replacement = JSON.stringify(replacement);\n if (typeof replacement === \"object\" && replacement !== null) replacement = replacement.toString();\n string = string.replace(new RegExp(`{{${val}}}`, \"g\"), replacement);\n }\n\n return string;\n }\n /**\r\n * Finds a value, subobject, or array from a tree that matches a specific filter.\r\n * @param {object} tree Tree that should be walked\r\n * @param {callable} searchFilter Filter to check against each object and subobject\r\n * @param {object} options Additional options to customize the search\r\n * @param {Array|null} [options.walkable=null] Array of strings to use as keys that are allowed to be walked on. Null value indicates all keys are walkable\r\n * @param {Array} [options.ignore=[]] Array of strings to use as keys to exclude from the search, most helpful when `walkable = null`.\r\n */\n\n\n static findInTree(tree, searchFilter, {\n walkable = null,\n ignore = []\n } = {}) {\n if (typeof searchFilter === \"string\") {\n if (tree.hasOwnProperty(searchFilter)) return tree[searchFilter];\n } else if (searchFilter(tree)) {\n return tree;\n }\n\n if (typeof tree !== \"object\" || tree == null) return undefined;\n let tempReturn = undefined;\n\n if (tree instanceof Array) {\n for (const value of tree) {\n tempReturn = this.findInTree(value, searchFilter, {\n walkable,\n ignore\n });\n if (typeof tempReturn != \"undefined\") return tempReturn;\n }\n } else {\n const toWalk = walkable == null ? Object.keys(tree) : walkable;\n\n for (const key of toWalk) {\n if (!tree.hasOwnProperty(key) || ignore.includes(key)) continue;\n tempReturn = this.findInTree(tree[key], searchFilter, {\n walkable,\n ignore\n });\n if (typeof tempReturn != \"undefined\") return tempReturn;\n }\n }\n\n return tempReturn;\n }\n /**\r\n * Gets a nested property (if it exists) safely. Path should be something like `prop.prop2.prop3`.\r\n * Numbers can be used for arrays as well like `prop.prop2.array.0.id`.\r\n * @param {Object} obj - object to get nested property of\r\n * @param {string} path - representation of the property to obtain\r\n */\n\n\n static getNestedProp(obj, path) {\n return path.split(/\\s?\\.\\s?/).reduce(function (currentObj, prop) {\n return currentObj && currentObj[prop];\n }, obj);\n }\n /**\r\n * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n * @param {callable} searchFilter Filter function to check subobjects against.\r\n */\n\n\n static findInRenderTree(tree, searchFilter, {\n walkable = [\"props\", \"children\", \"child\", \"sibling\"],\n ignore = []\n } = {}) {\n return this.findInTree(tree, searchFilter, {\n walkable,\n ignore\n });\n }\n /**\r\n * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n * @param {callable} searchFilter Filter function to check subobjects against.\r\n */\n\n\n static findInReactTree(tree, searchFilter) {\n return this.findInTree(tree, searchFilter, {\n walkable: [\"props\", \"children\", \"return\", \"stateNode\"]\n });\n }\n\n static getReactInstance(node) {\n if (node.__reactInternalInstance$) return node.__reactInternalInstance$;\n return node[Object.keys(node).find(k => k.startsWith(\"__reactInternalInstance\"))] || null;\n }\n /**\r\n * Grabs a value from the react internal instance. Allows you to grab\r\n * long depth values safely without accessing no longer valid properties.\r\n * @param {HTMLElement} node - node to obtain react instance of\r\n * @param {object} options - options for the search\r\n * @param {array} [options.include] - list of items to include from the search\r\n * @param {array} [options.exclude=[\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"]] - list of items to exclude from the search\r\n * @param {callable} [options.filter=_=>_] - filter to check the current instance with (should return a boolean)\r\n * @return {(*|null)} the owner instance or undefined if not found.\r\n */\n\n\n static getOwnerInstance(node, {\n include,\n exclude = [\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"],\n filter = _ => _\n } = {}) {\n if (node === undefined) return undefined;\n const excluding = include === undefined;\n const nameFilter = excluding ? exclude : include;\n\n function getDisplayName(owner) {\n const type = owner.type;\n if (!type) return null;\n return type.displayName || type.name || null;\n }\n\n function classFilter(owner) {\n const name = getDisplayName(owner);\n return name !== null && !!(nameFilter.includes(name) ^ excluding);\n }\n\n let curr = this.getReactInstance(node);\n\n for (curr = curr && curr.return; curr !== null; curr = curr.return) {\n if (curr === null) continue;\n const owner = curr.stateNode;\n if (curr !== null && !(owner instanceof HTMLElement) && classFilter(curr) && filter(owner)) return owner;\n }\n\n return null;\n }\n\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/modules/utilities.js?9769"],"names":["Utilities","parseHTML","html","fragment","template","document","createElement","innerHTML","node","content","cloneNode","childNodes","length","getTextArea","$","insertText","textarea","text","focus","selectionStart","selectionEnd","value","execCommand","escape","s","replace","testJSON","data","JSON","parse","err","suppressErrors","method","message","params","e","Logger","stacktrace","monkeyPatch","what","methodName","options","before","after","instead","once","silent","force","displayName","name","constructor","console","log","error","origMethod","cancel","thisObject","methodArguments","arguments","cancelPatch","originalMethod","callOriginalMethod","returnValue","apply","tempRet","undefined","__monkeyPatched","__originalMethod","onRemoved","callback","observer","MutationObserver","mutations","m","mutation","nodes","Array","from","removedNodes","directMatch","indexOf","parentMatch","some","parent","contains","disconnect","observe","body","subtree","childList","isEmpty","obj","isArray","key","hasOwnProperty","memoizeObject","object","proxy","Proxy","get","mod","Object","getOwnPropertyDescriptor","set","defineProperty","prop","extend","extendee","extenders","i","formatString","string","values","val","replacement","stringify","toString","RegExp","findInTree","tree","searchFilter","walkable","ignore","tempReturn","toWalk","keys","includes","getNestedProp","path","split","reduce","currentObj","findInRenderTree","findInReactTree","getReactInstance","__reactInternalInstance$","find","k","startsWith","getOwnerInstance","include","exclude","filter","_","excluding","nameFilter","getDisplayName","owner","type","classFilter","curr","return","stateNode","HTMLElement"],"mappings":"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","file":"./src/modules/utilities.js.js","sourcesContent":["import Logger from \"./logger\";\r\n\r\nexport default class Utilities {\r\n\r\n    /**\r\n     * Parses a string of HTML and returns the results. If the second parameter is true,\r\n     * the parsed HTML will be returned as a document fragment {@see https://developer.mozilla.org/en-US/docs/Web/API/DocumentFragment}.\r\n     * This is extremely useful if you have a list of elements at the top level, they can then be appended all at once to another node.\r\n     *\r\n     * If the second parameter is false, then the return value will be the list of parsed\r\n     * nodes and there were multiple top level nodes, otherwise the single node is returned.\r\n     * @param {string} html - HTML to be parsed\r\n     * @param {boolean} [fragment=false] - Whether or not the return should be the raw `DocumentFragment`\r\n     * @returns {(DocumentFragment|NodeList|HTMLElement)} - The result of HTML parsing\r\n     */\r\n    static parseHTML(html, fragment = false) {\r\n        const template = document.createElement(\"template\");\r\n        template.innerHTML = html;\r\n        const node = template.content.cloneNode(true);\r\n        if (fragment) return node;\r\n        return node.childNodes.length > 1 ? node.childNodes : node.childNodes[0];\r\n    }\r\n\r\n    static getTextArea() {\r\n        return $(\".channelTextArea-1LDbYG textarea\");\r\n    }\r\n\r\n    static insertText(textarea, text) {\r\n        textarea.focus();\r\n        textarea.selectionStart = 0;\r\n        textarea.selectionEnd = textarea.value.length;\r\n        document.execCommand(\"insertText\", false, text);\r\n    }\r\n\r\n    static escape(s) {\r\n        return s.replace(/[-/\\\\^$*+?.()|[\\]{}]/g, \"\\\\$&\");\r\n    }\r\n\r\n    static testJSON(data) {\r\n        try {\r\n            return JSON.parse(data);\r\n        }\r\n        catch (err) {\r\n            return false;\r\n        }\r\n    }\r\n\r\n    static suppressErrors(method, message) {\r\n        return (...params) => {\r\n            try { return method(...params);\t}\r\n            catch (e) { Logger.stacktrace(\"SuppressedError\", \"Error occurred in \" + message, e); }\r\n        };\r\n    }\r\n\r\n    static monkeyPatch(what, methodName, options) {\r\n        const {before, after, instead, once = false, silent = false, force = false} = options;\r\n        const displayName = options.displayName || what.displayName || what.name || what.constructor.displayName || what.constructor.name;\r\n        if (!silent) console.log(\"patch\", methodName, \"of\", displayName); // eslint-disable-line no-console\r\n        if (!what[methodName]) {\r\n            if (force) what[methodName] = function() {};\r\n            else return console.error(methodName, \"does not exist for\", displayName); // eslint-disable-line no-console\r\n        }\r\n        const origMethod = what[methodName];\r\n        const cancel = () => {\r\n            if (!silent) console.log(\"unpatch\", methodName, \"of\", displayName); // eslint-disable-line no-console\r\n            what[methodName] = origMethod;\r\n        };\r\n        what[methodName] = function() {\r\n            const data = {\r\n                thisObject: this,\r\n                methodArguments: arguments,\r\n                cancelPatch: cancel,\r\n                originalMethod: origMethod,\r\n                callOriginalMethod: () => data.returnValue = data.originalMethod.apply(data.thisObject, data.methodArguments)\r\n            };\r\n            if (instead) {\r\n                const tempRet = Utilities.suppressErrors(instead, \"`instead` callback of \" + what[methodName].displayName)(data);\r\n                if (tempRet !== undefined) data.returnValue = tempRet;\r\n            }\r\n            else {\r\n                if (before) Utilities.suppressErrors(before, \"`before` callback of \" + what[methodName].displayName)(data);\r\n                data.callOriginalMethod();\r\n                if (after) Utilities.suppressErrors(after, \"`after` callback of \" + what[methodName].displayName)(data);\r\n            }\r\n            if (once) cancel();\r\n            return data.returnValue;\r\n        };\r\n        what[methodName].__monkeyPatched = true;\r\n        if (!what[methodName].__originalMethod) what[methodName].__originalMethod = origMethod;\r\n        what[methodName].displayName = \"patched \" + (what[methodName].displayName || methodName);\r\n        return cancel;\r\n    }\r\n\r\n    static onRemoved(node, callback) {\r\n        const observer = new MutationObserver((mutations) => {\r\n            for (let m = 0; m < mutations.length; m++) {\r\n                const mutation = mutations[m];\r\n                const nodes = Array.from(mutation.removedNodes);\r\n                const directMatch = nodes.indexOf(node) > -1;\r\n                const parentMatch = nodes.some(parent => parent.contains(node));\r\n                if (directMatch || parentMatch) {\r\n                    observer.disconnect();\r\n                    callback();\r\n                }\r\n            }\r\n        });\r\n\r\n        observer.observe(document.body, {subtree: true, childList: true});\r\n    }\r\n\r\n    static isEmpty(obj) {\r\n        if (obj == null || obj == undefined || obj == \"\") return true;\r\n        if (typeof(obj) !== \"object\") return false;\r\n        if (Array.isArray(obj)) return obj.length == 0;\r\n        for (const key in obj) {\r\n            if (obj.hasOwnProperty(key)) return false;\r\n        }\r\n        return true;\r\n    }\r\n\r\n    /**\r\n     * Generates an automatically memoizing version of an object.\r\n     * @author Zerebos\r\n     * @param {Object} object - object to memoize\r\n     * @returns {Proxy} the proxy to the object that memoizes properties\r\n     */\r\n    static memoizeObject(object) {\r\n        const proxy = new Proxy(object, {\r\n            get: function(obj, mod) {\r\n                if (!obj.hasOwnProperty(mod)) return undefined;\r\n                if (Object.getOwnPropertyDescriptor(obj, mod).get) {\r\n                    const value = obj[mod];\r\n                    delete obj[mod];\r\n                    obj[mod] = value;\r\n                }\r\n                return obj[mod];\r\n            },\r\n            set: function(obj, mod, value) {\r\n                if (obj.hasOwnProperty(mod)) return Logger.error(\"MemoizedObject\", \"Trying to overwrite existing property\");\r\n                obj[mod] = value;\r\n                return obj[mod];\r\n            }\r\n        });\r\n\r\n        Object.defineProperty(proxy, \"hasOwnProperty\", {value: function(prop) {\r\n            return this[prop] !== undefined;\r\n        }});\r\n\r\n        return proxy;\r\n    }\r\n\r\n    /**\r\n     * Deep extends an object with a set of other objects. Objects later in the list\r\n     * of `extenders` have priority, that is to say if one sets a key to be a primitive,\r\n     * it will be overwritten with the next one with the same key. If it is an object,\r\n     * and the keys match, the object is extended. This happens recursively.\r\n     * @param {object} extendee - Object to be extended\r\n     * @param {...object} extenders - Objects to extend with\r\n     * @returns {object} - A reference to `extendee`\r\n     */\r\n    static extend(extendee, ...extenders) {\r\n        for (let i = 0; i < extenders.length; i++) {\r\n            for (const key in extenders[i]) {\r\n                if (extenders[i].hasOwnProperty(key)) {\r\n                    if (typeof extendee[key] === \"object\" && typeof extenders[i][key] === \"object\") this.extend(extendee[key], extenders[i][key]);\r\n                    else if (typeof extenders[i][key] === \"object\") extendee[key] = {}, this.extend(extendee[key], extenders[i][key]);\r\n                    else extendee[key] = extenders[i][key];\r\n                }\r\n            }\r\n        }\r\n        return extendee;\r\n    }\r\n\r\n    /**\r\n     * Format strings with placeholders (`{{placeholder}}`) into full strings.\r\n     * Quick example: `PluginUtilities.formatString(\"Hello, {{user}}\", {user: \"Zerebos\"})`\r\n     * would return \"Hello, Zerebos\".\r\n     * @param {string} string - string to format\r\n     * @param {object} values - object literal of placeholders to replacements\r\n     * @returns {string} the properly formatted string\r\n     */\r\n    static formatString(string, values) {\r\n        for (const val in values) {\r\n            let replacement = values[val];\r\n            if (Array.isArray(replacement)) replacement = JSON.stringify(replacement);\r\n            if (typeof(replacement) === \"object\" && replacement !== null) replacement = replacement.toString();\r\n            string = string.replace(new RegExp(`{{${val}}}`, \"g\"), replacement);\r\n        }\r\n        return string;\r\n    }\r\n\r\n    /**\r\n     * Finds a value, subobject, or array from a tree that matches a specific filter.\r\n     * @param {object} tree Tree that should be walked\r\n     * @param {callable} searchFilter Filter to check against each object and subobject\r\n     * @param {object} options Additional options to customize the search\r\n     * @param {Array<string>|null} [options.walkable=null] Array of strings to use as keys that are allowed to be walked on. Null value indicates all keys are walkable\r\n     * @param {Array<string>} [options.ignore=[]] Array of strings to use as keys to exclude from the search, most helpful when `walkable = null`.\r\n     */\r\n    static findInTree(tree, searchFilter, {walkable = null, ignore = []} = {}) {\r\n        if (typeof searchFilter === \"string\") {\r\n            if (tree.hasOwnProperty(searchFilter)) return tree[searchFilter];\r\n        }\r\n        else if (searchFilter(tree)) {\r\n            return tree;\r\n        }\r\n\r\n        if (typeof tree !== \"object\" || tree == null) return undefined;\r\n\r\n        let tempReturn = undefined;\r\n        if (tree instanceof Array) {\r\n            for (const value of tree) {\r\n                tempReturn = this.findInTree(value, searchFilter, {walkable, ignore});\r\n                if (typeof tempReturn != \"undefined\") return tempReturn;\r\n            }\r\n        }\r\n        else {\r\n            const toWalk = walkable == null ? Object.keys(tree) : walkable;\r\n            for (const key of toWalk) {\r\n                if (!tree.hasOwnProperty(key) || ignore.includes(key)) continue;\r\n                tempReturn = this.findInTree(tree[key], searchFilter, {walkable, ignore});\r\n                if (typeof tempReturn != \"undefined\") return tempReturn;\r\n            }\r\n        }\r\n        return tempReturn;\r\n    }\r\n\r\n    /**\r\n     * Gets a nested property (if it exists) safely. Path should be something like `prop.prop2.prop3`.\r\n     * Numbers can be used for arrays as well like `prop.prop2.array.0.id`.\r\n     * @param {Object} obj - object to get nested property of\r\n     * @param {string} path - representation of the property to obtain\r\n     */\r\n    static getNestedProp(obj, path) {\r\n        return path.split(/\\s?\\.\\s?/).reduce(function(currentObj, prop) {\r\n            return currentObj && currentObj[prop];\r\n        }, obj);\r\n    }\r\n\r\n    /**\r\n     * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n     * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n     * @param {callable} searchFilter Filter function to check subobjects against.\r\n     */\r\n    static findInRenderTree(tree, searchFilter, {walkable = [\"props\", \"children\", \"child\", \"sibling\"], ignore = []} = {}) {\r\n        return this.findInTree(tree, searchFilter, {walkable, ignore});\r\n    }\r\n\r\n    /**\r\n     * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n     * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n     * @param {callable} searchFilter Filter function to check subobjects against.\r\n     */\r\n    static findInReactTree(tree, searchFilter) {\r\n        return this.findInTree(tree, searchFilter, {walkable: [\"props\", \"children\", \"return\", \"stateNode\"]});\r\n    }\r\n\r\n    static getReactInstance(node) {\r\n        if (node.__reactInternalInstance$) return node.__reactInternalInstance$;\r\n        return node[Object.keys(node).find(k => k.startsWith(\"__reactInternalInstance\"))] || null;\r\n    }\r\n\r\n    /**\r\n\t * Grabs a value from the react internal instance. Allows you to grab\r\n\t * long depth values safely without accessing no longer valid properties.\r\n\t * @param {HTMLElement} node - node to obtain react instance of\r\n\t * @param {object} options - options for the search\r\n\t * @param {array} [options.include] - list of items to include from the search\r\n\t * @param {array} [options.exclude=[\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"]] - list of items to exclude from the search\r\n\t * @param {callable} [options.filter=_=>_] - filter to check the current instance with (should return a boolean)\r\n\t * @return {(*|null)} the owner instance or undefined if not found.\r\n\t */\r\n\tstatic getOwnerInstance(node, {include, exclude = [\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"], filter = _ => _} = {}) {\r\n\t\tif (node === undefined) return undefined;\r\n\t\tconst excluding = include === undefined;\r\n\t\tconst nameFilter = excluding ? exclude : include;\r\n\t\tfunction getDisplayName(owner) {\r\n\t\t\tconst type = owner.type;\r\n\t\t\tif (!type) return null;\r\n\t\t\treturn type.displayName || type.name || null;\r\n\t\t}\r\n\t\tfunction classFilter(owner) {\r\n\t\t\tconst name = getDisplayName(owner);\r\n\t\t\treturn (name !== null && !!(nameFilter.includes(name) ^ excluding));\r\n\t\t}\r\n\r\n\t\tlet curr = this.getReactInstance(node);\r\n\t\tfor (curr = curr && curr.return; curr !== null; curr = curr.return) {\r\n\t\t\tif (curr === null) continue;\r\n\t\t\tconst owner = curr.stateNode;\r\n\t\t\tif (curr !== null && !(owner instanceof HTMLElement) && classFilter(curr) && filter(owner)) return owner;\r\n\t\t}\r\n\r\n\t\treturn null;\r\n\t}\r\n}"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/modules/utilities.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"default\", function() { return Utilities; });\n/* harmony import */ var data__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! data */ \"./src/data/data.js\");\n/* harmony import */ var _logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./logger */ \"./src/modules/logger.js\");\n\n\nclass Utilities {\n static repoUrl(path) {\n return `https://cdn.staticaly.com/gh/${data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].repo}/BetterDiscordApp/${data__WEBPACK_IMPORTED_MODULE_0__[\"Config\"].hash}/${path}`;\n }\n /**\r\n * Parses a string of HTML and returns the results. If the second parameter is true,\r\n * the parsed HTML will be returned as a document fragment {@see https://developer.mozilla.org/en-US/docs/Web/API/DocumentFragment}.\r\n * This is extremely useful if you have a list of elements at the top level, they can then be appended all at once to another node.\r\n *\r\n * If the second parameter is false, then the return value will be the list of parsed\r\n * nodes and there were multiple top level nodes, otherwise the single node is returned.\r\n * @param {string} html - HTML to be parsed\r\n * @param {boolean} [fragment=false] - Whether or not the return should be the raw `DocumentFragment`\r\n * @returns {(DocumentFragment|NodeList|HTMLElement)} - The result of HTML parsing\r\n */\n\n\n static parseHTML(html, fragment = false) {\n const template = document.createElement(\"template\");\n template.innerHTML = html;\n const node = template.content.cloneNode(true);\n if (fragment) return node;\n return node.childNodes.length > 1 ? node.childNodes : node.childNodes[0];\n }\n\n static getTextArea() {\n return $(\".channelTextArea-1LDbYG textarea\");\n }\n\n static insertText(textarea, text) {\n textarea.focus();\n textarea.selectionStart = 0;\n textarea.selectionEnd = textarea.value.length;\n document.execCommand(\"insertText\", false, text);\n }\n\n static escape(s) {\n return s.replace(/[-/\\\\^$*+?.()|[\\]{}]/g, \"\\\\$&\");\n }\n\n static testJSON(data) {\n try {\n return JSON.parse(data);\n } catch (err) {\n return false;\n }\n }\n\n static suppressErrors(method, message) {\n return (...params) => {\n try {\n return method(...params);\n } catch (e) {\n _logger__WEBPACK_IMPORTED_MODULE_1__[\"default\"].stacktrace(\"SuppressedError\", \"Error occurred in \" + message, e);\n }\n };\n }\n\n static monkeyPatch(what, methodName, options) {\n const {\n before,\n after,\n instead,\n once = false,\n silent = false,\n force = false\n } = options;\n const displayName = options.displayName || what.displayName || what.name || what.constructor.displayName || what.constructor.name;\n if (!silent) console.log(\"patch\", methodName, \"of\", displayName); // eslint-disable-line no-console\n\n if (!what[methodName]) {\n if (force) what[methodName] = function () {};else return console.error(methodName, \"does not exist for\", displayName); // eslint-disable-line no-console\n }\n\n const origMethod = what[methodName];\n\n const cancel = () => {\n if (!silent) console.log(\"unpatch\", methodName, \"of\", displayName); // eslint-disable-line no-console\n\n what[methodName] = origMethod;\n };\n\n what[methodName] = function () {\n const data = {\n thisObject: this,\n methodArguments: arguments,\n cancelPatch: cancel,\n originalMethod: origMethod,\n callOriginalMethod: () => data.returnValue = data.originalMethod.apply(data.thisObject, data.methodArguments)\n };\n\n if (instead) {\n const tempRet = Utilities.suppressErrors(instead, \"`instead` callback of \" + what[methodName].displayName)(data);\n if (tempRet !== undefined) data.returnValue = tempRet;\n } else {\n if (before) Utilities.suppressErrors(before, \"`before` callback of \" + what[methodName].displayName)(data);\n data.callOriginalMethod();\n if (after) Utilities.suppressErrors(after, \"`after` callback of \" + what[methodName].displayName)(data);\n }\n\n if (once) cancel();\n return data.returnValue;\n };\n\n what[methodName].__monkeyPatched = true;\n if (!what[methodName].__originalMethod) what[methodName].__originalMethod = origMethod;\n what[methodName].displayName = \"patched \" + (what[methodName].displayName || methodName);\n return cancel;\n }\n\n static onRemoved(node, callback) {\n const observer = new MutationObserver(mutations => {\n for (let m = 0; m < mutations.length; m++) {\n const mutation = mutations[m];\n const nodes = Array.from(mutation.removedNodes);\n const directMatch = nodes.indexOf(node) > -1;\n const parentMatch = nodes.some(parent => parent.contains(node));\n\n if (directMatch || parentMatch) {\n observer.disconnect();\n callback();\n }\n }\n });\n observer.observe(document.body, {\n subtree: true,\n childList: true\n });\n }\n\n static isEmpty(obj) {\n if (obj == null || obj == undefined || obj == \"\") return true;\n if (typeof obj !== \"object\") return false;\n if (Array.isArray(obj)) return obj.length == 0;\n\n for (const key in obj) {\n if (obj.hasOwnProperty(key)) return false;\n }\n\n return true;\n }\n /**\r\n * Generates an automatically memoizing version of an object.\r\n * @author Zerebos\r\n * @param {Object} object - object to memoize\r\n * @returns {Proxy} the proxy to the object that memoizes properties\r\n */\n\n\n static memoizeObject(object) {\n const proxy = new Proxy(object, {\n get: function (obj, mod) {\n if (!obj.hasOwnProperty(mod)) return undefined;\n\n if (Object.getOwnPropertyDescriptor(obj, mod).get) {\n const value = obj[mod];\n delete obj[mod];\n obj[mod] = value;\n }\n\n return obj[mod];\n },\n set: function (obj, mod, value) {\n if (obj.hasOwnProperty(mod)) return _logger__WEBPACK_IMPORTED_MODULE_1__[\"default\"].error(\"MemoizedObject\", \"Trying to overwrite existing property\");\n obj[mod] = value;\n return obj[mod];\n }\n });\n Object.defineProperty(proxy, \"hasOwnProperty\", {\n value: function (prop) {\n return this[prop] !== undefined;\n }\n });\n return proxy;\n }\n /**\r\n * Deep extends an object with a set of other objects. Objects later in the list\r\n * of `extenders` have priority, that is to say if one sets a key to be a primitive,\r\n * it will be overwritten with the next one with the same key. If it is an object,\r\n * and the keys match, the object is extended. This happens recursively.\r\n * @param {object} extendee - Object to be extended\r\n * @param {...object} extenders - Objects to extend with\r\n * @returns {object} - A reference to `extendee`\r\n */\n\n\n static extend(extendee, ...extenders) {\n for (let i = 0; i < extenders.length; i++) {\n for (const key in extenders[i]) {\n if (extenders[i].hasOwnProperty(key)) {\n if (typeof extendee[key] === \"object\" && typeof extenders[i][key] === \"object\") this.extend(extendee[key], extenders[i][key]);else if (typeof extenders[i][key] === \"object\") extendee[key] = {}, this.extend(extendee[key], extenders[i][key]);else extendee[key] = extenders[i][key];\n }\n }\n }\n\n return extendee;\n }\n /**\r\n * Format strings with placeholders (`{{placeholder}}`) into full strings.\r\n * Quick example: `PluginUtilities.formatString(\"Hello, {{user}}\", {user: \"Zerebos\"})`\r\n * would return \"Hello, Zerebos\".\r\n * @param {string} string - string to format\r\n * @param {object} values - object literal of placeholders to replacements\r\n * @returns {string} the properly formatted string\r\n */\n\n\n static formatString(string, values) {\n for (const val in values) {\n let replacement = values[val];\n if (Array.isArray(replacement)) replacement = JSON.stringify(replacement);\n if (typeof replacement === \"object\" && replacement !== null) replacement = replacement.toString();\n string = string.replace(new RegExp(`{{${val}}}`, \"g\"), replacement);\n }\n\n return string;\n }\n /**\r\n * Finds a value, subobject, or array from a tree that matches a specific filter.\r\n * @param {object} tree Tree that should be walked\r\n * @param {callable} searchFilter Filter to check against each object and subobject\r\n * @param {object} options Additional options to customize the search\r\n * @param {Array|null} [options.walkable=null] Array of strings to use as keys that are allowed to be walked on. Null value indicates all keys are walkable\r\n * @param {Array} [options.ignore=[]] Array of strings to use as keys to exclude from the search, most helpful when `walkable = null`.\r\n */\n\n\n static findInTree(tree, searchFilter, {\n walkable = null,\n ignore = []\n } = {}) {\n if (typeof searchFilter === \"string\") {\n if (tree.hasOwnProperty(searchFilter)) return tree[searchFilter];\n } else if (searchFilter(tree)) {\n return tree;\n }\n\n if (typeof tree !== \"object\" || tree == null) return undefined;\n let tempReturn = undefined;\n\n if (tree instanceof Array) {\n for (const value of tree) {\n tempReturn = this.findInTree(value, searchFilter, {\n walkable,\n ignore\n });\n if (typeof tempReturn != \"undefined\") return tempReturn;\n }\n } else {\n const toWalk = walkable == null ? Object.keys(tree) : walkable;\n\n for (const key of toWalk) {\n if (!tree.hasOwnProperty(key) || ignore.includes(key)) continue;\n tempReturn = this.findInTree(tree[key], searchFilter, {\n walkable,\n ignore\n });\n if (typeof tempReturn != \"undefined\") return tempReturn;\n }\n }\n\n return tempReturn;\n }\n /**\r\n * Gets a nested property (if it exists) safely. Path should be something like `prop.prop2.prop3`.\r\n * Numbers can be used for arrays as well like `prop.prop2.array.0.id`.\r\n * @param {Object} obj - object to get nested property of\r\n * @param {string} path - representation of the property to obtain\r\n */\n\n\n static getNestedProp(obj, path) {\n return path.split(/\\s?\\.\\s?/).reduce(function (currentObj, prop) {\n return currentObj && currentObj[prop];\n }, obj);\n }\n /**\r\n * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n * @param {callable} searchFilter Filter function to check subobjects against.\r\n */\n\n\n static findInRenderTree(tree, searchFilter, {\n walkable = [\"props\", \"children\", \"child\", \"sibling\"],\n ignore = []\n } = {}) {\n return this.findInTree(tree, searchFilter, {\n walkable,\n ignore\n });\n }\n /**\r\n * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n * @param {callable} searchFilter Filter function to check subobjects against.\r\n */\n\n\n static findInReactTree(tree, searchFilter) {\n return this.findInTree(tree, searchFilter, {\n walkable: [\"props\", \"children\", \"return\", \"stateNode\"]\n });\n }\n\n static getReactInstance(node) {\n if (node.__reactInternalInstance$) return node.__reactInternalInstance$;\n return node[Object.keys(node).find(k => k.startsWith(\"__reactInternalInstance\"))] || null;\n }\n /**\r\n * Grabs a value from the react internal instance. Allows you to grab\r\n * long depth values safely without accessing no longer valid properties.\r\n * @param {HTMLElement} node - node to obtain react instance of\r\n * @param {object} options - options for the search\r\n * @param {array} [options.include] - list of items to include from the search\r\n * @param {array} [options.exclude=[\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"]] - list of items to exclude from the search\r\n * @param {callable} [options.filter=_=>_] - filter to check the current instance with (should return a boolean)\r\n * @return {(*|null)} the owner instance or undefined if not found.\r\n */\n\n\n static getOwnerInstance(node, {\n include,\n exclude = [\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"],\n filter = _ => _\n } = {}) {\n if (node === undefined) return undefined;\n const excluding = include === undefined;\n const nameFilter = excluding ? exclude : include;\n\n function getDisplayName(owner) {\n const type = owner.type;\n if (!type) return null;\n return type.displayName || type.name || null;\n }\n\n function classFilter(owner) {\n const name = getDisplayName(owner);\n return name !== null && !!(nameFilter.includes(name) ^ excluding);\n }\n\n let curr = this.getReactInstance(node);\n\n for (curr = curr && curr.return; curr !== null; curr = curr.return) {\n if (curr === null) continue;\n const owner = curr.stateNode;\n if (curr !== null && !(owner instanceof HTMLElement) && classFilter(curr) && filter(owner)) return owner;\n }\n\n return null;\n }\n\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/modules/utilities.js?9769"],"names":["Utilities","repoUrl","path","Config","repo","hash","parseHTML","html","fragment","template","document","createElement","innerHTML","node","content","cloneNode","childNodes","length","getTextArea","$","insertText","textarea","text","focus","selectionStart","selectionEnd","value","execCommand","escape","s","replace","testJSON","data","JSON","parse","err","suppressErrors","method","message","params","e","Logger","stacktrace","monkeyPatch","what","methodName","options","before","after","instead","once","silent","force","displayName","name","constructor","console","log","error","origMethod","cancel","thisObject","methodArguments","arguments","cancelPatch","originalMethod","callOriginalMethod","returnValue","apply","tempRet","undefined","__monkeyPatched","__originalMethod","onRemoved","callback","observer","MutationObserver","mutations","m","mutation","nodes","Array","from","removedNodes","directMatch","indexOf","parentMatch","some","parent","contains","disconnect","observe","body","subtree","childList","isEmpty","obj","isArray","key","hasOwnProperty","memoizeObject","object","proxy","Proxy","get","mod","Object","getOwnPropertyDescriptor","set","defineProperty","prop","extend","extendee","extenders","i","formatString","string","values","val","replacement","stringify","toString","RegExp","findInTree","tree","searchFilter","walkable","ignore","tempReturn","toWalk","keys","includes","getNestedProp","split","reduce","currentObj","findInRenderTree","findInReactTree","getReactInstance","__reactInternalInstance$","find","k","startsWith","getOwnerInstance","include","exclude","filter","_","excluding","nameFilter","getDisplayName","owner","type","classFilter","curr","return","stateNode","HTMLElement"],"mappings":"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","file":"./src/modules/utilities.js.js","sourcesContent":["import {Config} from \"data\";\r\nimport Logger from \"./logger\";\r\n\r\nexport default class Utilities {\r\n\r\n    static repoUrl(path) {\r\n        return `https://cdn.staticaly.com/gh/${Config.repo}/BetterDiscordApp/${Config.hash}/${path}`;\r\n    }\r\n\r\n    /**\r\n     * Parses a string of HTML and returns the results. If the second parameter is true,\r\n     * the parsed HTML will be returned as a document fragment {@see https://developer.mozilla.org/en-US/docs/Web/API/DocumentFragment}.\r\n     * This is extremely useful if you have a list of elements at the top level, they can then be appended all at once to another node.\r\n     *\r\n     * If the second parameter is false, then the return value will be the list of parsed\r\n     * nodes and there were multiple top level nodes, otherwise the single node is returned.\r\n     * @param {string} html - HTML to be parsed\r\n     * @param {boolean} [fragment=false] - Whether or not the return should be the raw `DocumentFragment`\r\n     * @returns {(DocumentFragment|NodeList|HTMLElement)} - The result of HTML parsing\r\n     */\r\n    static parseHTML(html, fragment = false) {\r\n        const template = document.createElement(\"template\");\r\n        template.innerHTML = html;\r\n        const node = template.content.cloneNode(true);\r\n        if (fragment) return node;\r\n        return node.childNodes.length > 1 ? node.childNodes : node.childNodes[0];\r\n    }\r\n\r\n    static getTextArea() {\r\n        return $(\".channelTextArea-1LDbYG textarea\");\r\n    }\r\n\r\n    static insertText(textarea, text) {\r\n        textarea.focus();\r\n        textarea.selectionStart = 0;\r\n        textarea.selectionEnd = textarea.value.length;\r\n        document.execCommand(\"insertText\", false, text);\r\n    }\r\n\r\n    static escape(s) {\r\n        return s.replace(/[-/\\\\^$*+?.()|[\\]{}]/g, \"\\\\$&\");\r\n    }\r\n\r\n    static testJSON(data) {\r\n        try {\r\n            return JSON.parse(data);\r\n        }\r\n        catch (err) {\r\n            return false;\r\n        }\r\n    }\r\n\r\n    static suppressErrors(method, message) {\r\n        return (...params) => {\r\n            try { return method(...params);\t}\r\n            catch (e) { Logger.stacktrace(\"SuppressedError\", \"Error occurred in \" + message, e); }\r\n        };\r\n    }\r\n\r\n    static monkeyPatch(what, methodName, options) {\r\n        const {before, after, instead, once = false, silent = false, force = false} = options;\r\n        const displayName = options.displayName || what.displayName || what.name || what.constructor.displayName || what.constructor.name;\r\n        if (!silent) console.log(\"patch\", methodName, \"of\", displayName); // eslint-disable-line no-console\r\n        if (!what[methodName]) {\r\n            if (force) what[methodName] = function() {};\r\n            else return console.error(methodName, \"does not exist for\", displayName); // eslint-disable-line no-console\r\n        }\r\n        const origMethod = what[methodName];\r\n        const cancel = () => {\r\n            if (!silent) console.log(\"unpatch\", methodName, \"of\", displayName); // eslint-disable-line no-console\r\n            what[methodName] = origMethod;\r\n        };\r\n        what[methodName] = function() {\r\n            const data = {\r\n                thisObject: this,\r\n                methodArguments: arguments,\r\n                cancelPatch: cancel,\r\n                originalMethod: origMethod,\r\n                callOriginalMethod: () => data.returnValue = data.originalMethod.apply(data.thisObject, data.methodArguments)\r\n            };\r\n            if (instead) {\r\n                const tempRet = Utilities.suppressErrors(instead, \"`instead` callback of \" + what[methodName].displayName)(data);\r\n                if (tempRet !== undefined) data.returnValue = tempRet;\r\n            }\r\n            else {\r\n                if (before) Utilities.suppressErrors(before, \"`before` callback of \" + what[methodName].displayName)(data);\r\n                data.callOriginalMethod();\r\n                if (after) Utilities.suppressErrors(after, \"`after` callback of \" + what[methodName].displayName)(data);\r\n            }\r\n            if (once) cancel();\r\n            return data.returnValue;\r\n        };\r\n        what[methodName].__monkeyPatched = true;\r\n        if (!what[methodName].__originalMethod) what[methodName].__originalMethod = origMethod;\r\n        what[methodName].displayName = \"patched \" + (what[methodName].displayName || methodName);\r\n        return cancel;\r\n    }\r\n\r\n    static onRemoved(node, callback) {\r\n        const observer = new MutationObserver((mutations) => {\r\n            for (let m = 0; m < mutations.length; m++) {\r\n                const mutation = mutations[m];\r\n                const nodes = Array.from(mutation.removedNodes);\r\n                const directMatch = nodes.indexOf(node) > -1;\r\n                const parentMatch = nodes.some(parent => parent.contains(node));\r\n                if (directMatch || parentMatch) {\r\n                    observer.disconnect();\r\n                    callback();\r\n                }\r\n            }\r\n        });\r\n\r\n        observer.observe(document.body, {subtree: true, childList: true});\r\n    }\r\n\r\n    static isEmpty(obj) {\r\n        if (obj == null || obj == undefined || obj == \"\") return true;\r\n        if (typeof(obj) !== \"object\") return false;\r\n        if (Array.isArray(obj)) return obj.length == 0;\r\n        for (const key in obj) {\r\n            if (obj.hasOwnProperty(key)) return false;\r\n        }\r\n        return true;\r\n    }\r\n\r\n    /**\r\n     * Generates an automatically memoizing version of an object.\r\n     * @author Zerebos\r\n     * @param {Object} object - object to memoize\r\n     * @returns {Proxy} the proxy to the object that memoizes properties\r\n     */\r\n    static memoizeObject(object) {\r\n        const proxy = new Proxy(object, {\r\n            get: function(obj, mod) {\r\n                if (!obj.hasOwnProperty(mod)) return undefined;\r\n                if (Object.getOwnPropertyDescriptor(obj, mod).get) {\r\n                    const value = obj[mod];\r\n                    delete obj[mod];\r\n                    obj[mod] = value;\r\n                }\r\n                return obj[mod];\r\n            },\r\n            set: function(obj, mod, value) {\r\n                if (obj.hasOwnProperty(mod)) return Logger.error(\"MemoizedObject\", \"Trying to overwrite existing property\");\r\n                obj[mod] = value;\r\n                return obj[mod];\r\n            }\r\n        });\r\n\r\n        Object.defineProperty(proxy, \"hasOwnProperty\", {value: function(prop) {\r\n            return this[prop] !== undefined;\r\n        }});\r\n\r\n        return proxy;\r\n    }\r\n\r\n    /**\r\n     * Deep extends an object with a set of other objects. Objects later in the list\r\n     * of `extenders` have priority, that is to say if one sets a key to be a primitive,\r\n     * it will be overwritten with the next one with the same key. If it is an object,\r\n     * and the keys match, the object is extended. This happens recursively.\r\n     * @param {object} extendee - Object to be extended\r\n     * @param {...object} extenders - Objects to extend with\r\n     * @returns {object} - A reference to `extendee`\r\n     */\r\n    static extend(extendee, ...extenders) {\r\n        for (let i = 0; i < extenders.length; i++) {\r\n            for (const key in extenders[i]) {\r\n                if (extenders[i].hasOwnProperty(key)) {\r\n                    if (typeof extendee[key] === \"object\" && typeof extenders[i][key] === \"object\") this.extend(extendee[key], extenders[i][key]);\r\n                    else if (typeof extenders[i][key] === \"object\") extendee[key] = {}, this.extend(extendee[key], extenders[i][key]);\r\n                    else extendee[key] = extenders[i][key];\r\n                }\r\n            }\r\n        }\r\n        return extendee;\r\n    }\r\n\r\n    /**\r\n     * Format strings with placeholders (`{{placeholder}}`) into full strings.\r\n     * Quick example: `PluginUtilities.formatString(\"Hello, {{user}}\", {user: \"Zerebos\"})`\r\n     * would return \"Hello, Zerebos\".\r\n     * @param {string} string - string to format\r\n     * @param {object} values - object literal of placeholders to replacements\r\n     * @returns {string} the properly formatted string\r\n     */\r\n    static formatString(string, values) {\r\n        for (const val in values) {\r\n            let replacement = values[val];\r\n            if (Array.isArray(replacement)) replacement = JSON.stringify(replacement);\r\n            if (typeof(replacement) === \"object\" && replacement !== null) replacement = replacement.toString();\r\n            string = string.replace(new RegExp(`{{${val}}}`, \"g\"), replacement);\r\n        }\r\n        return string;\r\n    }\r\n\r\n    /**\r\n     * Finds a value, subobject, or array from a tree that matches a specific filter.\r\n     * @param {object} tree Tree that should be walked\r\n     * @param {callable} searchFilter Filter to check against each object and subobject\r\n     * @param {object} options Additional options to customize the search\r\n     * @param {Array<string>|null} [options.walkable=null] Array of strings to use as keys that are allowed to be walked on. Null value indicates all keys are walkable\r\n     * @param {Array<string>} [options.ignore=[]] Array of strings to use as keys to exclude from the search, most helpful when `walkable = null`.\r\n     */\r\n    static findInTree(tree, searchFilter, {walkable = null, ignore = []} = {}) {\r\n        if (typeof searchFilter === \"string\") {\r\n            if (tree.hasOwnProperty(searchFilter)) return tree[searchFilter];\r\n        }\r\n        else if (searchFilter(tree)) {\r\n            return tree;\r\n        }\r\n\r\n        if (typeof tree !== \"object\" || tree == null) return undefined;\r\n\r\n        let tempReturn = undefined;\r\n        if (tree instanceof Array) {\r\n            for (const value of tree) {\r\n                tempReturn = this.findInTree(value, searchFilter, {walkable, ignore});\r\n                if (typeof tempReturn != \"undefined\") return tempReturn;\r\n            }\r\n        }\r\n        else {\r\n            const toWalk = walkable == null ? Object.keys(tree) : walkable;\r\n            for (const key of toWalk) {\r\n                if (!tree.hasOwnProperty(key) || ignore.includes(key)) continue;\r\n                tempReturn = this.findInTree(tree[key], searchFilter, {walkable, ignore});\r\n                if (typeof tempReturn != \"undefined\") return tempReturn;\r\n            }\r\n        }\r\n        return tempReturn;\r\n    }\r\n\r\n    /**\r\n     * Gets a nested property (if it exists) safely. Path should be something like `prop.prop2.prop3`.\r\n     * Numbers can be used for arrays as well like `prop.prop2.array.0.id`.\r\n     * @param {Object} obj - object to get nested property of\r\n     * @param {string} path - representation of the property to obtain\r\n     */\r\n    static getNestedProp(obj, path) {\r\n        return path.split(/\\s?\\.\\s?/).reduce(function(currentObj, prop) {\r\n            return currentObj && currentObj[prop];\r\n        }, obj);\r\n    }\r\n\r\n    /**\r\n     * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n     * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n     * @param {callable} searchFilter Filter function to check subobjects against.\r\n     */\r\n    static findInRenderTree(tree, searchFilter, {walkable = [\"props\", \"children\", \"child\", \"sibling\"], ignore = []} = {}) {\r\n        return this.findInTree(tree, searchFilter, {walkable, ignore});\r\n    }\r\n\r\n    /**\r\n     * Finds a value, subobject, or array from a tree that matches a specific filter. Great for patching render functions.\r\n     * @param {object} tree React tree to look through. Can be a rendered object or an internal instance.\r\n     * @param {callable} searchFilter Filter function to check subobjects against.\r\n     */\r\n    static findInReactTree(tree, searchFilter) {\r\n        return this.findInTree(tree, searchFilter, {walkable: [\"props\", \"children\", \"return\", \"stateNode\"]});\r\n    }\r\n\r\n    static getReactInstance(node) {\r\n        if (node.__reactInternalInstance$) return node.__reactInternalInstance$;\r\n        return node[Object.keys(node).find(k => k.startsWith(\"__reactInternalInstance\"))] || null;\r\n    }\r\n\r\n    /**\r\n\t * Grabs a value from the react internal instance. Allows you to grab\r\n\t * long depth values safely without accessing no longer valid properties.\r\n\t * @param {HTMLElement} node - node to obtain react instance of\r\n\t * @param {object} options - options for the search\r\n\t * @param {array} [options.include] - list of items to include from the search\r\n\t * @param {array} [options.exclude=[\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"]] - list of items to exclude from the search\r\n\t * @param {callable} [options.filter=_=>_] - filter to check the current instance with (should return a boolean)\r\n\t * @return {(*|null)} the owner instance or undefined if not found.\r\n\t */\r\n\tstatic getOwnerInstance(node, {include, exclude = [\"Popout\", \"Tooltip\", \"Scroller\", \"BackgroundFlash\"], filter = _ => _} = {}) {\r\n\t\tif (node === undefined) return undefined;\r\n\t\tconst excluding = include === undefined;\r\n\t\tconst nameFilter = excluding ? exclude : include;\r\n\t\tfunction getDisplayName(owner) {\r\n\t\t\tconst type = owner.type;\r\n\t\t\tif (!type) return null;\r\n\t\t\treturn type.displayName || type.name || null;\r\n\t\t}\r\n\t\tfunction classFilter(owner) {\r\n\t\t\tconst name = getDisplayName(owner);\r\n\t\t\treturn (name !== null && !!(nameFilter.includes(name) ^ excluding));\r\n\t\t}\r\n\r\n\t\tlet curr = this.getReactInstance(node);\r\n\t\tfor (curr = curr && curr.return; curr !== null; curr = curr.return) {\r\n\t\t\tif (curr === null) continue;\r\n\t\t\tconst owner = curr.stateNode;\r\n\t\t\tif (curr !== null && !(owner instanceof HTMLElement) && classFilter(curr) && filter(owner)) return owner;\r\n\t\t}\r\n\r\n\t\treturn null;\r\n\t}\r\n}"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/modules/utilities.js\n"); /***/ }), @@ -855,6 +855,18 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var modu /***/ }), +/***/ "./src/ui/settings/addoncard.jsx": +/*!***************************************!*\ + !*** ./src/ui/settings/addoncard.jsx ***! + \***************************************/ +/*! exports provided: default */ +/***/ (function(module, __webpack_exports__, __webpack_require__) { + +"use strict"; +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"default\", function() { return AddonCard; });\n/* harmony import */ var modules__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! modules */ \"./src/modules/modules.js\");\n/* harmony import */ var _icons_close__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../icons/close */ \"./src/ui/icons/close.jsx\");\n/* harmony import */ var _icons_reload__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../icons/reload */ \"./src/ui/icons/reload.jsx\");\n\n\n\nclass AddonCard extends modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].Component {\n constructor(props) {\n super(props);\n this.state = {\n checked: this.props.enabled,\n settingsOpen: false\n };\n this.settingsPanel = \"\";\n this.panelRef = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createRef();\n this.onChange = this.onChange.bind(this);\n this.reload = this.reload.bind(this);\n this.showSettings = this.showSettings.bind(this);\n this.closeSettings = this.closeSettings.bind(this);\n }\n\n reload() {\n if (!this.props.reload) return;\n this.props.addon = this.props.reload(this.props.addon.id);\n this.forceUpdate();\n }\n\n componentDidUpdate() {\n if (!this.state.settingsOpen) return;\n if (this.settingsPanel instanceof Node) this.panelRef.current.appendChild(this.settingsPanel); // if (!SettingsCookie[\"fork-ps-3\"]) return;\n\n const isHidden = (container, element) => {\n const cTop = container.scrollTop;\n const cBottom = cTop + container.clientHeight;\n const eTop = element.offsetTop;\n const eBottom = eTop + element.clientHeight;\n return eTop < cTop || eBottom > cBottom;\n };\n\n const panel = $(this.panelRef.current);\n const container = panel.parents(\".scroller-2FKFPG\");\n if (!isHidden(container[0], panel[0])) return;\n container.animate({\n scrollTop: panel.offset().top - container.offset().top + container.scrollTop() - 30\n }, 300);\n }\n\n getString(value) {\n return typeof value == \"string\" ? value : value.toString();\n }\n\n onChange() {\n this.setState({\n checked: !this.state.checked\n });\n this.props.onChange && this.props.onChange(this.props.addon.id);\n }\n\n showSettings() {\n if (!this.props.hasSettings) return;\n this.setState({\n settingsOpen: true\n });\n }\n\n closeSettings() {\n this.panelRef.current.innerHTML = \"\";\n this.setState({\n settingsOpen: false\n });\n }\n\n buildTitle(name, version, author) {\n const title = modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons.title.split(/({{[A-Za-z]+}})/);\n const nameIndex = title.findIndex(s => s == \"{{name}}\");\n if (nameIndex) title[nameIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-name\"\n }, name);\n const versionIndex = title.findIndex(s => s == \"{{version}}\");\n if (nameIndex) title[versionIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-version\"\n }, version);\n const authorIndex = title.findIndex(s => s == \"{{author}}\");\n if (nameIndex) title[authorIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-author\"\n }, author);\n return title.flat();\n }\n\n get settingsComponent() {\n const addon = this.props.addon;\n const name = this.getString(addon.name);\n\n try {\n this.settingsPanel = this.props.getSettingsPanel();\n } catch (err) {\n modules__WEBPACK_IMPORTED_MODULE_0__[\"Logger\"].stacktrace(\"Addon Settings\", \"Unable to get settings panel for \" + name + \".\", err);\n }\n\n const props = {\n id: `${name}-settings`,\n className: \"addon-settings\",\n ref: this.panelRef\n };\n if (typeof settingsPanel == \"string\") props.dangerouslySetInnerHTML = this.settingsPanel;\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"li\", {\n className: \"settings-open bd-switch-item\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-close\",\n onClick: this.closeSettings\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_icons_close__WEBPACK_IMPORTED_MODULE_1__[\"default\"], null)), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", props, this.settingsPanel instanceof modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].Component ? this.settingsPanel : null));\n }\n\n buildLink(which) {\n const url = this.props.addon[which];\n if (!url) return null;\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"a\", {\n className: \"bd-link bd-link-website\",\n href: url,\n target: \"_blank\",\n rel: \"noopener noreferrer\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons[which]);\n }\n\n get footer() {\n const links = [\"website\", \"source\"];\n if (!links.some(l => this.props.addon[l]) && !this.props.hasSettings) return null;\n const linkComponents = links.map(this.buildLink.bind(this)).filter(c => c);\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-footer\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-links\"\n }, linkComponents.map((comp, i) => i < linkComponents.length - 1 ? [comp, \" | \"] : [comp]).flat()), this.props.hasSettings && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"button\", {\n onClick: this.showSettings,\n className: \"bd-button bd-button-addon-settings\",\n disabled: !this.state.checked\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons.addonSettings));\n }\n\n render() {\n if (this.state.settingsOpen) return this.settingsComponent;\n const addon = this.props.addon;\n const name = this.getString(addon.name);\n const author = this.getString(addon.author);\n const description = this.getString(addon.description);\n const version = this.getString(addon.version);\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"li\", {\n dataName: name,\n dataVersion: version,\n className: \"settings-closed bd-switch-item\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-header\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-header-title\"\n }, this.buildTitle(name, version, author)), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-controls\"\n }, this.props.showReloadIcon && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_icons_reload__WEBPACK_IMPORTED_MODULE_2__[\"default\"], {\n className: \"bd-reload bd-reload-card\",\n onClick: this.reload\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"label\", {\n className: \"bd-switch-wrapper\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"input\", {\n className: \"bd-switch-checkbox\",\n checked: this.state.checked,\n onChange: this.onChange,\n type: \"checkbox\"\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: this.state.checked ? \"bd-switch checked\" : \"bd-switch\"\n })))), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-description-wrap scroller-wrap fade\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-description scroller\"\n }, description)), this.footer);\n }\n\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/ui/settings/addoncard.jsx?6373"],"names":["AddonCard","React","Component","constructor","props","state","checked","enabled","settingsOpen","settingsPanel","panelRef","createRef","onChange","bind","reload","showSettings","closeSettings","addon","id","forceUpdate","componentDidUpdate","Node","current","appendChild","isHidden","container","element","cTop","scrollTop","cBottom","clientHeight","eTop","offsetTop","eBottom","panel","$","parents","animate","offset","top","getString","value","toString","setState","hasSettings","innerHTML","buildTitle","name","version","author","title","Strings","Addons","split","nameIndex","findIndex","s","createElement","className","versionIndex","authorIndex","flat","settingsComponent","getSettingsPanel","err","Logger","stacktrace","ref","dangerouslySetInnerHTML","buildLink","which","url","footer","links","some","l","linkComponents","map","filter","c","comp","i","length","addonSettings","render","description","showReloadIcon"],"mappings":"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","file":"./src/ui/settings/addoncard.jsx.js","sourcesContent":["import {React, Logger, Strings} from \"modules\";\r\nimport CloseButton from \"../icons/close\";\r\nimport ReloadIcon from \"../icons/reload\";\r\n\r\nexport default class AddonCard extends React.Component {\r\n\r\n    constructor(props) {\r\n        super(props);\r\n        this.state = {\r\n            checked: this.props.enabled,\r\n            settingsOpen: false\r\n        };\r\n\r\n        this.settingsPanel = \"\";\r\n        this.panelRef = React.createRef();\r\n\r\n        this.onChange = this.onChange.bind(this);\r\n        this.reload = this.reload.bind(this);\r\n        this.showSettings = this.showSettings.bind(this);\r\n        this.closeSettings = this.closeSettings.bind(this);\r\n    }\r\n\r\n    reload() {\r\n        if (!this.props.reload) return;\r\n        this.props.addon = this.props.reload(this.props.addon.id);\r\n        this.forceUpdate();\r\n    }\r\n\r\n    componentDidUpdate() {\r\n        if (!this.state.settingsOpen) return;\r\n        if (this.settingsPanel instanceof Node) this.panelRef.current.appendChild(this.settingsPanel);\r\n\r\n        // if (!SettingsCookie[\"fork-ps-3\"]) return;\r\n        const isHidden = (container, element) => {\r\n            const cTop = container.scrollTop;\r\n            const cBottom = cTop + container.clientHeight;\r\n            const eTop = element.offsetTop;\r\n            const eBottom = eTop + element.clientHeight;\r\n            return  (eTop < cTop || eBottom > cBottom);\r\n        };\r\n\r\n        const panel = $(this.panelRef.current);\r\n        const container = panel.parents(\".scroller-2FKFPG\");\r\n        if (!isHidden(container[0], panel[0])) return;\r\n        container.animate({\r\n            scrollTop: panel.offset().top - container.offset().top + container.scrollTop() - 30\r\n        }, 300);\r\n    }\r\n\r\n    getString(value) {return typeof value == \"string\" ? value : value.toString();}\r\n\r\n    onChange() {\r\n        this.setState({checked: !this.state.checked});\r\n        this.props.onChange && this.props.onChange(this.props.addon.id);\r\n    }\r\n\r\n    showSettings() {\r\n        if (!this.props.hasSettings) return;\r\n        this.setState({settingsOpen: true});\r\n    }\r\n\r\n    closeSettings() {\r\n        this.panelRef.current.innerHTML = \"\";\r\n        this.setState({settingsOpen: false});\r\n    }\r\n\r\n    buildTitle(name, version, author) {\r\n        const title = Strings.Addons.title.split(/({{[A-Za-z]+}})/);\r\n        const nameIndex = title.findIndex(s => s == \"{{name}}\");\r\n        if (nameIndex) title[nameIndex] = React.createElement(\"span\", {className: \"bd-name\"}, name);\r\n        const versionIndex = title.findIndex(s => s == \"{{version}}\");\r\n        if (nameIndex) title[versionIndex] = React.createElement(\"span\", {className: \"bd-version\"}, version);\r\n        const authorIndex = title.findIndex(s => s == \"{{author}}\");\r\n        if (nameIndex) title[authorIndex] = React.createElement(\"span\", {className: \"bd-author\"}, author);\r\n        return title.flat();\r\n    }\r\n\r\n    get settingsComponent() {\r\n        const addon = this.props.addon;\r\n        const name = this.getString(addon.name);\r\n        try { this.settingsPanel = this.props.getSettingsPanel(); }\r\n        catch (err) { Logger.stacktrace(\"Addon Settings\", \"Unable to get settings panel for \" + name + \".\", err); }\r\n\r\n        const props = {id: `${name}-settings`, className: \"addon-settings\", ref: this.panelRef};\r\n        if (typeof(settingsPanel) == \"string\") props.dangerouslySetInnerHTML = this.settingsPanel;\r\n\r\n        return <li className=\"settings-open bd-switch-item\">\r\n                    <div className=\"bd-close\" onClick={this.closeSettings}><CloseButton /></div>\r\n                    <div {...props}>{this.settingsPanel instanceof React.Component ? this.settingsPanel : null}</div>\r\n                </li>;\r\n    }\r\n\r\n    buildLink(which) {\r\n        const url = this.props.addon[which];\r\n        if (!url) return null;\r\n        return <a className=\"bd-link bd-link-website\" href={url} target=\"_blank\" rel=\"noopener noreferrer\">{Strings.Addons[which]}</a>;\r\n    }\r\n\r\n    get footer() {\r\n        const links = [\"website\", \"source\"];\r\n        if (!links.some(l => this.props.addon[l]) && !this.props.hasSettings) return null;\r\n        const linkComponents = links.map(this.buildLink.bind(this)).filter(c => c);\r\n        return <div className=\"bd-footer\">\r\n                    <span className=\"bd-links\">{linkComponents.map((comp, i) => i < linkComponents.length - 1 ? [comp, \" | \"] : [comp]).flat()}</span>\r\n                    {this.props.hasSettings && <button onClick={this.showSettings} className=\"bd-button bd-button-addon-settings\" disabled={!this.state.checked}>{Strings.Addons.addonSettings}</button>}\r\n                </div>;\r\n    }\r\n\r\n    render() {\r\n        if (this.state.settingsOpen) return this.settingsComponent;\r\n\r\n        const addon = this.props.addon;\r\n        const name = this.getString(addon.name);\r\n        const author = this.getString(addon.author);\r\n        const description = this.getString(addon.description);\r\n        const version = this.getString(addon.version);\r\n\r\n        return <li dataName={name} dataVersion={version} className=\"settings-closed bd-switch-item\">\r\n                    <div className=\"bd-header\">\r\n                            <span className=\"bd-header-title\">{this.buildTitle(name, version, author)}</span>\r\n                            <div className=\"bd-controls\">\r\n                                {this.props.showReloadIcon && <ReloadIcon className=\"bd-reload bd-reload-card\" onClick={this.reload} />}\r\n                                <label className=\"bd-switch-wrapper\">\r\n                                    <input className=\"bd-switch-checkbox\" checked={this.state.checked} onChange={this.onChange} type=\"checkbox\" />\r\n                                    <div className={this.state.checked ? \"bd-switch checked\" : \"bd-switch\"} />\r\n                                </label>\r\n                            </div>\r\n                    </div>\r\n                    <div className=\"bd-description-wrap scroller-wrap fade\"><div className=\"bd-description scroller\">{description}</div></div>\r\n                    {this.footer}\r\n                </li>;\r\n    }\r\n}\r\n"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/ui/settings/addoncard.jsx\n"); + +/***/ }), + /***/ "./src/ui/settings/addonlist.jsx": /*!***************************************!*\ !*** ./src/ui/settings/addonlist.jsx ***! @@ -863,7 +875,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var modu /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"default\", function() { return AddonList; });\n/* harmony import */ var modules__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! modules */ \"./src/modules/modules.js\");\n/* harmony import */ var _title__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./title */ \"./src/ui/settings/title.jsx\");\n/* harmony import */ var _plugincard__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./plugincard */ \"./src/ui/settings/plugincard.js\");\n/* harmony import */ var _themecard__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./themecard */ \"./src/ui/settings/themecard.js\");\n/* harmony import */ var _icons_reload__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../icons/reload */ \"./src/ui/icons/reload.jsx\");\n\n\n\n\n\nclass AddonList extends modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].Component {\n reload() {\n if (this.props.refreshList) this.props.refreshList();\n this.forceUpdate();\n }\n\n render() {\n const {\n title,\n folder,\n addonList,\n addonState,\n onChange,\n reload\n } = this.props;\n const showReloadIcon = !modules__WEBPACK_IMPORTED_MODULE_0__[\"Settings\"].get(\"settings\", \"addons\", \"autoReload\");\n const button = folder ? {\n title: modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons.openFolder.format({\n type: title\n }),\n onClick: () => {\n __webpack_require__(/*! electron */ \"electron\").shell.openItem(folder);\n }\n } : null;\n return [modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_title__WEBPACK_IMPORTED_MODULE_1__[\"default\"], {\n key: \"title\",\n text: title,\n button: button,\n otherChildren: showReloadIcon && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_icons_reload__WEBPACK_IMPORTED_MODULE_4__[\"default\"], {\n className: \"bd-reload\",\n onClick: this.reload.bind(this)\n })\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"ul\", {\n key: \"addonList\",\n className: \"bd-slist\"\n }, addonList.sort((a, b) => a.name.toLowerCase().localeCompare(b.name.toLowerCase())).map(addon => {\n const CardType = addon.type ? _plugincard__WEBPACK_IMPORTED_MODULE_2__[\"default\"] : _themecard__WEBPACK_IMPORTED_MODULE_3__[\"default\"];\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(CardType, {\n showReloadIcon: showReloadIcon,\n key: addon.id,\n enabled: addonState[addon.id],\n addon: addon,\n onChange: onChange,\n reload: reload\n });\n }))];\n }\n\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9Db3JlLy4vc3JjL3VpL3NldHRpbmdzL2FkZG9ubGlzdC5qc3g/NjRkMiJdLCJuYW1lcyI6WyJBZGRvbkxpc3QiLCJSZWFjdCIsIkNvbXBvbmVudCIsInJlbG9hZCIsInByb3BzIiwicmVmcmVzaExpc3QiLCJmb3JjZVVwZGF0ZSIsInJlbmRlciIsInRpdGxlIiwiZm9sZGVyIiwiYWRkb25MaXN0IiwiYWRkb25TdGF0ZSIsIm9uQ2hhbmdlIiwic2hvd1JlbG9hZEljb24iLCJTZXR0aW5ncyIsImdldCIsImJ1dHRvbiIsIlN0cmluZ3MiLCJBZGRvbnMiLCJvcGVuRm9sZGVyIiwiZm9ybWF0IiwidHlwZSIsIm9uQ2xpY2siLCJyZXF1aXJlIiwic2hlbGwiLCJvcGVuSXRlbSIsImJpbmQiLCJzb3J0IiwiYSIsImIiLCJuYW1lIiwidG9Mb3dlckNhc2UiLCJsb2NhbGVDb21wYXJlIiwibWFwIiwiYWRkb24iLCJDYXJkVHlwZSIsIlBsdWdpbkNhcmQiLCJUaGVtZUNhcmQiLCJpZCJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBRWUsTUFBTUEsU0FBTixTQUF3QkMsNkNBQUssQ0FBQ0MsU0FBOUIsQ0FBd0M7QUFFbkRDLFFBQU0sR0FBRztBQUNMLFFBQUksS0FBS0MsS0FBTCxDQUFXQyxXQUFmLEVBQTRCLEtBQUtELEtBQUwsQ0FBV0MsV0FBWDtBQUM1QixTQUFLQyxXQUFMO0FBQ0g7O0FBRURDLFFBQU0sR0FBRztBQUNMLFVBQU07QUFBQ0MsV0FBRDtBQUFRQyxZQUFSO0FBQWdCQyxlQUFoQjtBQUEyQkMsZ0JBQTNCO0FBQXVDQyxjQUF2QztBQUFpRFQ7QUFBakQsUUFBMkQsS0FBS0MsS0FBdEU7QUFDQSxVQUFNUyxjQUFjLEdBQUcsQ0FBQ0MsZ0RBQVEsQ0FBQ0MsR0FBVCxDQUFhLFVBQWIsRUFBeUIsUUFBekIsRUFBbUMsWUFBbkMsQ0FBeEI7QUFDQSxVQUFNQyxNQUFNLEdBQUdQLE1BQU0sR0FBRztBQUFDRCxXQUFLLEVBQUVTLCtDQUFPLENBQUNDLE1BQVIsQ0FBZUMsVUFBZixDQUEwQkMsTUFBMUIsQ0FBaUM7QUFBQ0MsWUFBSSxFQUFFYjtBQUFQLE9BQWpDLENBQVI7QUFBeURjLGFBQU8sRUFBRSxNQUFNO0FBQUNDLDJCQUFPLENBQUMsMEJBQUQsQ0FBUCxDQUFvQkMsS0FBcEIsQ0FBMEJDLFFBQTFCLENBQW1DaEIsTUFBbkM7QUFBNEM7QUFBckgsS0FBSCxHQUE0SCxJQUFqSjtBQUNBLFdBQU8sQ0FDSCw0REFBQyw4Q0FBRDtBQUFlLFNBQUcsRUFBQyxPQUFuQjtBQUEyQixVQUFJLEVBQUVELEtBQWpDO0FBQXdDLFlBQU0sRUFBRVEsTUFBaEQ7QUFBd0QsbUJBQWEsRUFBRUgsY0FBYyxJQUFJLDREQUFDLHFEQUFEO0FBQVksaUJBQVMsRUFBQyxXQUF0QjtBQUFrQyxlQUFPLEVBQUUsS0FBS1YsTUFBTCxDQUFZdUIsSUFBWixDQUFpQixJQUFqQjtBQUEzQztBQUF6RixNQURHLEVBRUg7QUFBSSxTQUFHLEVBQUMsV0FBUjtBQUFvQixlQUFTLEVBQUU7QUFBL0IsT0FDQ2hCLFNBQVMsQ0FBQ2lCLElBQVYsQ0FBZSxDQUFDQyxDQUFELEVBQUlDLENBQUosS0FBVUQsQ0FBQyxDQUFDRSxJQUFGLENBQU9DLFdBQVAsR0FBcUJDLGFBQXJCLENBQW1DSCxDQUFDLENBQUNDLElBQUYsQ0FBT0MsV0FBUCxFQUFuQyxDQUF6QixFQUFtRkUsR0FBbkYsQ0FBdUZDLEtBQUssSUFBSTtBQUM3RixZQUFNQyxRQUFRLEdBQUdELEtBQUssQ0FBQ2IsSUFBTixHQUFhZSxtREFBYixHQUEwQkMsa0RBQTNDO0FBQ0EsYUFBTyw0REFBQyxRQUFEO0FBQVUsc0JBQWMsRUFBRXhCLGNBQTFCO0FBQTBDLFdBQUcsRUFBRXFCLEtBQUssQ0FBQ0ksRUFBckQ7QUFBeUQsZUFBTyxFQUFFM0IsVUFBVSxDQUFDdUIsS0FBSyxDQUFDSSxFQUFQLENBQTVFO0FBQXdGLGFBQUssRUFBRUosS0FBL0Y7QUFBc0csZ0JBQVEsRUFBRXRCLFFBQWhIO0FBQTBILGNBQU0sRUFBRVQ7QUFBbEksUUFBUDtBQUNILEtBSEEsQ0FERCxDQUZHLENBQVA7QUFTSDs7QUFwQmtEIiwiZmlsZSI6Ii4vc3JjL3VpL3NldHRpbmdzL2FkZG9ubGlzdC5qc3guanMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1JlYWN0LCBTZXR0aW5ncywgU3RyaW5nc30gZnJvbSBcIm1vZHVsZXNcIjtcclxuXHJcbmltcG9ydCBTZXR0aW5nc1RpdGxlIGZyb20gXCIuL3RpdGxlXCI7XHJcbmltcG9ydCBQbHVnaW5DYXJkIGZyb20gXCIuL3BsdWdpbmNhcmRcIjtcclxuaW1wb3J0IFRoZW1lQ2FyZCBmcm9tIFwiLi90aGVtZWNhcmRcIjtcclxuaW1wb3J0IFJlbG9hZEljb24gZnJvbSBcIi4uL2ljb25zL3JlbG9hZFwiO1xyXG5cclxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgQWRkb25MaXN0IGV4dGVuZHMgUmVhY3QuQ29tcG9uZW50IHtcclxuXHJcbiAgICByZWxvYWQoKSB7XHJcbiAgICAgICAgaWYgKHRoaXMucHJvcHMucmVmcmVzaExpc3QpIHRoaXMucHJvcHMucmVmcmVzaExpc3QoKTtcclxuICAgICAgICB0aGlzLmZvcmNlVXBkYXRlKCk7XHJcbiAgICB9XHJcblxyXG4gICAgcmVuZGVyKCkge1xyXG4gICAgICAgIGNvbnN0IHt0aXRsZSwgZm9sZGVyLCBhZGRvbkxpc3QsIGFkZG9uU3RhdGUsIG9uQ2hhbmdlLCByZWxvYWR9ID0gdGhpcy5wcm9wcztcclxuICAgICAgICBjb25zdCBzaG93UmVsb2FkSWNvbiA9ICFTZXR0aW5ncy5nZXQoXCJzZXR0aW5nc1wiLCBcImFkZG9uc1wiLCBcImF1dG9SZWxvYWRcIik7XHJcbiAgICAgICAgY29uc3QgYnV0dG9uID0gZm9sZGVyID8ge3RpdGxlOiBTdHJpbmdzLkFkZG9ucy5vcGVuRm9sZGVyLmZvcm1hdCh7dHlwZTogdGl0bGV9KSwgb25DbGljazogKCkgPT4ge3JlcXVpcmUoXCJlbGVjdHJvblwiKS5zaGVsbC5vcGVuSXRlbShmb2xkZXIpO319IDogbnVsbDtcclxuICAgICAgICByZXR1cm4gW1xyXG4gICAgICAgICAgICA8U2V0dGluZ3NUaXRsZSBrZXk9XCJ0aXRsZVwiIHRleHQ9e3RpdGxlfSBidXR0b249e2J1dHRvbn0gb3RoZXJDaGlsZHJlbj17c2hvd1JlbG9hZEljb24gJiYgPFJlbG9hZEljb24gY2xhc3NOYW1lPVwiYmQtcmVsb2FkXCIgb25DbGljaz17dGhpcy5yZWxvYWQuYmluZCh0aGlzKX0gLz59IC8+LFxyXG4gICAgICAgICAgICA8dWwga2V5PVwiYWRkb25MaXN0XCIgY2xhc3NOYW1lPXtcImJkLXNsaXN0XCJ9PlxyXG4gICAgICAgICAgICB7YWRkb25MaXN0LnNvcnQoKGEsIGIpID0+IGEubmFtZS50b0xvd2VyQ2FzZSgpLmxvY2FsZUNvbXBhcmUoYi5uYW1lLnRvTG93ZXJDYXNlKCkpKS5tYXAoYWRkb24gPT4ge1xyXG4gICAgICAgICAgICAgICAgY29uc3QgQ2FyZFR5cGUgPSBhZGRvbi50eXBlID8gUGx1Z2luQ2FyZCA6IFRoZW1lQ2FyZDtcclxuICAgICAgICAgICAgICAgIHJldHVybiA8Q2FyZFR5cGUgc2hvd1JlbG9hZEljb249e3Nob3dSZWxvYWRJY29ufSBrZXk9e2FkZG9uLmlkfSBlbmFibGVkPXthZGRvblN0YXRlW2FkZG9uLmlkXX0gYWRkb249e2FkZG9ufSBvbkNoYW5nZT17b25DaGFuZ2V9IHJlbG9hZD17cmVsb2FkfSAvPjtcclxuICAgICAgICAgICAgfSl9XHJcbiAgICAgICAgICAgIDwvdWw+XHJcbiAgICAgICAgXTtcclxuICAgIH1cclxufSJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/ui/settings/addonlist.jsx\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"default\", function() { return AddonList; });\n/* harmony import */ var modules__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! modules */ \"./src/modules/modules.js\");\n/* harmony import */ var _title__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./title */ \"./src/ui/settings/title.jsx\");\n/* harmony import */ var _icons_reload__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../icons/reload */ \"./src/ui/icons/reload.jsx\");\n/* harmony import */ var _addoncard__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./addoncard */ \"./src/ui/settings/addoncard.jsx\");\n\n\n\n\nclass AddonList extends modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].Component {\n reload() {\n if (this.props.refreshList) this.props.refreshList();\n this.forceUpdate();\n }\n\n render() {\n const {\n title,\n folder,\n addonList,\n addonState,\n onChange,\n reload\n } = this.props;\n const showReloadIcon = !modules__WEBPACK_IMPORTED_MODULE_0__[\"Settings\"].get(\"settings\", \"addons\", \"autoReload\");\n const button = folder ? {\n title: modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons.openFolder.format({\n type: title\n }),\n onClick: () => {\n __webpack_require__(/*! electron */ \"electron\").shell.openItem(folder);\n }\n } : null;\n return [modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_title__WEBPACK_IMPORTED_MODULE_1__[\"default\"], {\n key: \"title\",\n text: title,\n button: button,\n otherChildren: showReloadIcon && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_icons_reload__WEBPACK_IMPORTED_MODULE_2__[\"default\"], {\n className: \"bd-reload\",\n onClick: this.reload.bind(this)\n })\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"ul\", {\n key: \"addonList\",\n className: \"bd-slist\"\n }, addonList.sort((a, b) => a.name.toLowerCase().localeCompare(b.name.toLowerCase())).map(addon => {\n const hasSettings = addon.type && typeof addon.plugin.getSettingsPanel === \"function\";\n const getSettings = hasSettings && addon.plugin.getSettingsPanel.bind(addon.plugin);\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_addoncard__WEBPACK_IMPORTED_MODULE_3__[\"default\"], {\n showReloadIcon: showReloadIcon,\n key: addon.id,\n enabled: addonState[addon.id],\n addon: addon,\n onChange: onChange,\n reload: reload,\n hasSettings: hasSettings,\n getSettingsPanel: getSettings\n });\n }))];\n }\n\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9Db3JlLy4vc3JjL3VpL3NldHRpbmdzL2FkZG9ubGlzdC5qc3g/NjRkMiJdLCJuYW1lcyI6WyJBZGRvbkxpc3QiLCJSZWFjdCIsIkNvbXBvbmVudCIsInJlbG9hZCIsInByb3BzIiwicmVmcmVzaExpc3QiLCJmb3JjZVVwZGF0ZSIsInJlbmRlciIsInRpdGxlIiwiZm9sZGVyIiwiYWRkb25MaXN0IiwiYWRkb25TdGF0ZSIsIm9uQ2hhbmdlIiwic2hvd1JlbG9hZEljb24iLCJTZXR0aW5ncyIsImdldCIsImJ1dHRvbiIsIlN0cmluZ3MiLCJBZGRvbnMiLCJvcGVuRm9sZGVyIiwiZm9ybWF0IiwidHlwZSIsIm9uQ2xpY2siLCJyZXF1aXJlIiwic2hlbGwiLCJvcGVuSXRlbSIsImJpbmQiLCJzb3J0IiwiYSIsImIiLCJuYW1lIiwidG9Mb3dlckNhc2UiLCJsb2NhbGVDb21wYXJlIiwibWFwIiwiYWRkb24iLCJoYXNTZXR0aW5ncyIsInBsdWdpbiIsImdldFNldHRpbmdzUGFuZWwiLCJnZXRTZXR0aW5ncyIsImlkIl0sIm1hcHBpbmdzIjoiQUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUVBO0FBQ0E7QUFDQTtBQUVlLE1BQU1BLFNBQU4sU0FBd0JDLDZDQUFLLENBQUNDLFNBQTlCLENBQXdDO0FBRW5EQyxRQUFNLEdBQUc7QUFDTCxRQUFJLEtBQUtDLEtBQUwsQ0FBV0MsV0FBZixFQUE0QixLQUFLRCxLQUFMLENBQVdDLFdBQVg7QUFDNUIsU0FBS0MsV0FBTDtBQUNIOztBQUVEQyxRQUFNLEdBQUc7QUFDTCxVQUFNO0FBQUNDLFdBQUQ7QUFBUUMsWUFBUjtBQUFnQkMsZUFBaEI7QUFBMkJDLGdCQUEzQjtBQUF1Q0MsY0FBdkM7QUFBaURUO0FBQWpELFFBQTJELEtBQUtDLEtBQXRFO0FBQ0EsVUFBTVMsY0FBYyxHQUFHLENBQUNDLGdEQUFRLENBQUNDLEdBQVQsQ0FBYSxVQUFiLEVBQXlCLFFBQXpCLEVBQW1DLFlBQW5DLENBQXhCO0FBQ0EsVUFBTUMsTUFBTSxHQUFHUCxNQUFNLEdBQUc7QUFBQ0QsV0FBSyxFQUFFUywrQ0FBTyxDQUFDQyxNQUFSLENBQWVDLFVBQWYsQ0FBMEJDLE1BQTFCLENBQWlDO0FBQUNDLFlBQUksRUFBRWI7QUFBUCxPQUFqQyxDQUFSO0FBQXlEYyxhQUFPLEVBQUUsTUFBTTtBQUFDQywyQkFBTyxDQUFDLDBCQUFELENBQVAsQ0FBb0JDLEtBQXBCLENBQTBCQyxRQUExQixDQUFtQ2hCLE1BQW5DO0FBQTRDO0FBQXJILEtBQUgsR0FBNEgsSUFBako7QUFDQSxXQUFPLENBQ0gsNERBQUMsOENBQUQ7QUFBZSxTQUFHLEVBQUMsT0FBbkI7QUFBMkIsVUFBSSxFQUFFRCxLQUFqQztBQUF3QyxZQUFNLEVBQUVRLE1BQWhEO0FBQXdELG1CQUFhLEVBQUVILGNBQWMsSUFBSSw0REFBQyxxREFBRDtBQUFZLGlCQUFTLEVBQUMsV0FBdEI7QUFBa0MsZUFBTyxFQUFFLEtBQUtWLE1BQUwsQ0FBWXVCLElBQVosQ0FBaUIsSUFBakI7QUFBM0M7QUFBekYsTUFERyxFQUVIO0FBQUksU0FBRyxFQUFDLFdBQVI7QUFBb0IsZUFBUyxFQUFFO0FBQS9CLE9BQ0NoQixTQUFTLENBQUNpQixJQUFWLENBQWUsQ0FBQ0MsQ0FBRCxFQUFJQyxDQUFKLEtBQVVELENBQUMsQ0FBQ0UsSUFBRixDQUFPQyxXQUFQLEdBQXFCQyxhQUFyQixDQUFtQ0gsQ0FBQyxDQUFDQyxJQUFGLENBQU9DLFdBQVAsRUFBbkMsQ0FBekIsRUFBbUZFLEdBQW5GLENBQXVGQyxLQUFLLElBQUk7QUFDN0YsWUFBTUMsV0FBVyxHQUFHRCxLQUFLLENBQUNiLElBQU4sSUFBYyxPQUFPYSxLQUFLLENBQUNFLE1BQU4sQ0FBYUMsZ0JBQXBCLEtBQTBDLFVBQTVFO0FBQ0EsWUFBTUMsV0FBVyxHQUFHSCxXQUFXLElBQUlELEtBQUssQ0FBQ0UsTUFBTixDQUFhQyxnQkFBYixDQUE4QlgsSUFBOUIsQ0FBbUNRLEtBQUssQ0FBQ0UsTUFBekMsQ0FBbkM7QUFDQSxhQUFPLDREQUFDLGtEQUFEO0FBQVcsc0JBQWMsRUFBRXZCLGNBQTNCO0FBQTJDLFdBQUcsRUFBRXFCLEtBQUssQ0FBQ0ssRUFBdEQ7QUFBMEQsZUFBTyxFQUFFNUIsVUFBVSxDQUFDdUIsS0FBSyxDQUFDSyxFQUFQLENBQTdFO0FBQXlGLGFBQUssRUFBRUwsS0FBaEc7QUFBdUcsZ0JBQVEsRUFBRXRCLFFBQWpIO0FBQTJILGNBQU0sRUFBRVQsTUFBbkk7QUFBMkksbUJBQVcsRUFBRWdDLFdBQXhKO0FBQXFLLHdCQUFnQixFQUFFRztBQUF2TCxRQUFQO0FBQ0gsS0FKQSxDQURELENBRkcsQ0FBUDtBQVVIOztBQXJCa0QiLCJmaWxlIjoiLi9zcmMvdWkvc2V0dGluZ3MvYWRkb25saXN0LmpzeC5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7UmVhY3QsIFNldHRpbmdzLCBTdHJpbmdzfSBmcm9tIFwibW9kdWxlc1wiO1xyXG5cclxuaW1wb3J0IFNldHRpbmdzVGl0bGUgZnJvbSBcIi4vdGl0bGVcIjtcclxuaW1wb3J0IFJlbG9hZEljb24gZnJvbSBcIi4uL2ljb25zL3JlbG9hZFwiO1xyXG5pbXBvcnQgQWRkb25DYXJkIGZyb20gXCIuL2FkZG9uY2FyZFwiO1xyXG5cclxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgQWRkb25MaXN0IGV4dGVuZHMgUmVhY3QuQ29tcG9uZW50IHtcclxuXHJcbiAgICByZWxvYWQoKSB7XHJcbiAgICAgICAgaWYgKHRoaXMucHJvcHMucmVmcmVzaExpc3QpIHRoaXMucHJvcHMucmVmcmVzaExpc3QoKTtcclxuICAgICAgICB0aGlzLmZvcmNlVXBkYXRlKCk7XHJcbiAgICB9XHJcblxyXG4gICAgcmVuZGVyKCkge1xyXG4gICAgICAgIGNvbnN0IHt0aXRsZSwgZm9sZGVyLCBhZGRvbkxpc3QsIGFkZG9uU3RhdGUsIG9uQ2hhbmdlLCByZWxvYWR9ID0gdGhpcy5wcm9wcztcclxuICAgICAgICBjb25zdCBzaG93UmVsb2FkSWNvbiA9ICFTZXR0aW5ncy5nZXQoXCJzZXR0aW5nc1wiLCBcImFkZG9uc1wiLCBcImF1dG9SZWxvYWRcIik7XHJcbiAgICAgICAgY29uc3QgYnV0dG9uID0gZm9sZGVyID8ge3RpdGxlOiBTdHJpbmdzLkFkZG9ucy5vcGVuRm9sZGVyLmZvcm1hdCh7dHlwZTogdGl0bGV9KSwgb25DbGljazogKCkgPT4ge3JlcXVpcmUoXCJlbGVjdHJvblwiKS5zaGVsbC5vcGVuSXRlbShmb2xkZXIpO319IDogbnVsbDtcclxuICAgICAgICByZXR1cm4gW1xyXG4gICAgICAgICAgICA8U2V0dGluZ3NUaXRsZSBrZXk9XCJ0aXRsZVwiIHRleHQ9e3RpdGxlfSBidXR0b249e2J1dHRvbn0gb3RoZXJDaGlsZHJlbj17c2hvd1JlbG9hZEljb24gJiYgPFJlbG9hZEljb24gY2xhc3NOYW1lPVwiYmQtcmVsb2FkXCIgb25DbGljaz17dGhpcy5yZWxvYWQuYmluZCh0aGlzKX0gLz59IC8+LFxyXG4gICAgICAgICAgICA8dWwga2V5PVwiYWRkb25MaXN0XCIgY2xhc3NOYW1lPXtcImJkLXNsaXN0XCJ9PlxyXG4gICAgICAgICAgICB7YWRkb25MaXN0LnNvcnQoKGEsIGIpID0+IGEubmFtZS50b0xvd2VyQ2FzZSgpLmxvY2FsZUNvbXBhcmUoYi5uYW1lLnRvTG93ZXJDYXNlKCkpKS5tYXAoYWRkb24gPT4ge1xyXG4gICAgICAgICAgICAgICAgY29uc3QgaGFzU2V0dGluZ3MgPSBhZGRvbi50eXBlICYmIHR5cGVvZihhZGRvbi5wbHVnaW4uZ2V0U2V0dGluZ3NQYW5lbCkgPT09IFwiZnVuY3Rpb25cIjtcclxuICAgICAgICAgICAgICAgIGNvbnN0IGdldFNldHRpbmdzID0gaGFzU2V0dGluZ3MgJiYgYWRkb24ucGx1Z2luLmdldFNldHRpbmdzUGFuZWwuYmluZChhZGRvbi5wbHVnaW4pO1xyXG4gICAgICAgICAgICAgICAgcmV0dXJuIDxBZGRvbkNhcmQgc2hvd1JlbG9hZEljb249e3Nob3dSZWxvYWRJY29ufSBrZXk9e2FkZG9uLmlkfSBlbmFibGVkPXthZGRvblN0YXRlW2FkZG9uLmlkXX0gYWRkb249e2FkZG9ufSBvbkNoYW5nZT17b25DaGFuZ2V9IHJlbG9hZD17cmVsb2FkfSBoYXNTZXR0aW5ncz17aGFzU2V0dGluZ3N9IGdldFNldHRpbmdzUGFuZWw9e2dldFNldHRpbmdzfSAvPjtcclxuICAgICAgICAgICAgfSl9XHJcbiAgICAgICAgICAgIDwvdWw+XHJcbiAgICAgICAgXTtcclxuICAgIH1cclxufSJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/ui/settings/addonlist.jsx\n"); /***/ }), @@ -903,18 +915,6 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ }), -/***/ "./src/ui/settings/plugincard.js": -/*!***************************************!*\ - !*** ./src/ui/settings/plugincard.js ***! - \***************************************/ -/*! exports provided: default */ -/***/ (function(module, __webpack_exports__, __webpack_require__) { - -"use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"default\", function() { return PluginCard; });\n/* harmony import */ var modules__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! modules */ \"./src/modules/modules.js\");\n/* harmony import */ var _icons_close__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../icons/close */ \"./src/ui/icons/close.jsx\");\n/* harmony import */ var _icons_reload__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../icons/reload */ \"./src/ui/icons/reload.jsx\");\n\n\n\nclass PluginCard extends modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].Component {\n constructor(props) {\n super(props);\n this.onChange = this.onChange.bind(this);\n this.showSettings = this.showSettings.bind(this);\n this.state = {\n checked: this.props.enabled,\n settingsOpen: false\n };\n this.hasSettings = typeof this.props.addon.plugin.getSettingsPanel === \"function\";\n this.settingsPanel = \"\";\n this.panelRef = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createRef();\n this.reload = this.reload.bind(this); // this.onReload = this.onReload.bind(this);\n\n this.closeSettings = this.closeSettings.bind(this);\n }\n\n reload() {\n if (!this.props.reload) return;\n this.props.addon = this.props.reload(this.props.addon.id);\n this.forceUpdate();\n }\n\n componentDidUpdate() {\n if (!this.state.settingsOpen) return;\n if (this.settingsPanel instanceof Node) this.panelRef.current.appendChild(this.settingsPanel); // if (!SettingsCookie[\"fork-ps-3\"]) return;\n\n const isHidden = (container, element) => {\n const cTop = container.scrollTop;\n const cBottom = cTop + container.clientHeight;\n const eTop = element.offsetTop;\n const eBottom = eTop + element.clientHeight;\n return eTop < cTop || eBottom > cBottom;\n };\n\n const panel = $(this.panelRef.current);\n const container = panel.parents(\".scroller-2FKFPG\");\n if (!isHidden(container[0], panel[0])) return;\n container.animate({\n scrollTop: panel.offset().top - container.offset().top + container.scrollTop() - 30\n }, 300);\n }\n\n getString(value) {\n return typeof value == \"string\" ? value : value.toString();\n }\n\n closeSettings() {\n this.panelRef.current.innerHTML = \"\";\n this.setState({\n settingsOpen: false\n });\n }\n\n buildTitle(name, version, author) {\n const title = modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons.title.split(/({{[A-Za-z]+}})/);\n const nameIndex = title.findIndex(s => s == \"{{name}}\");\n if (nameIndex) title[nameIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-name\"\n }, name);\n const versionIndex = title.findIndex(s => s == \"{{version}}\");\n if (nameIndex) title[versionIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-version\"\n }, version);\n const authorIndex = title.findIndex(s => s == \"{{author}}\");\n if (nameIndex) title[authorIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-author\"\n }, author);\n return title.flat();\n }\n\n get settingsComponent() {\n const addon = this.props.addon;\n const name = this.getString(addon.name);\n\n try {\n this.settingsPanel = addon.plugin.getSettingsPanel();\n } catch (err) {\n modules__WEBPACK_IMPORTED_MODULE_0__[\"Logger\"].stacktrace(\"Plugin Settings\", \"Unable to get settings panel for \" + name + \".\", err);\n }\n\n const props = {\n id: `plugin-settings-${name}`,\n className: \"plugin-settings\",\n ref: this.panelRef\n };\n if (typeof settingsPanel == \"string\") props.dangerouslySetInnerHTML = this.settingsPanel;\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"li\", {\n className: \"settings-open bd-switch-item\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-close\",\n onClick: this.closeSettings\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_icons_close__WEBPACK_IMPORTED_MODULE_1__[\"default\"], null)), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", props, this.settingsPanel instanceof modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].Component ? this.settingsPanel : null));\n }\n\n buildLink(which) {\n const url = this.props.addon[which];\n if (!url) return null;\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"a\", {\n className: \"bd-link bd-link-website\",\n href: url,\n target: \"_blank\",\n rel: \"noopener noreferrer\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons[which]);\n }\n\n get footer() {\n const links = [\"website\", \"source\"];\n if (!links.some(l => this.props.addon[l]) && !this.hasSettings) return null;\n const linkComponents = links.map(this.buildLink.bind(this)).filter(c => c);\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-footer\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-links\"\n }, linkComponents.map((comp, i) => i < linkComponents.length - 1 ? [comp, \" | \"] : [comp]).flat()), this.hasSettings && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"button\", {\n onClick: this.showSettings,\n className: \"bd-button bd-button-plugin-settings\",\n disabled: !this.state.checked\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons.pluginSettings));\n }\n\n render() {\n if (this.state.settingsOpen) return this.settingsComponent;\n const {\n addon\n } = this.props;\n const name = this.getString(addon.name);\n const author = this.getString(addon.author);\n const description = this.getString(addon.description);\n const version = this.getString(addon.version);\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"li\", {\n dataName: name,\n dataVersion: version,\n className: \"settings-closed bd-switch-item\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-header\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-header-title\"\n }, this.buildTitle(name, version, author)), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-controls\"\n }, this.props.showReloadIcon && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_icons_reload__WEBPACK_IMPORTED_MODULE_2__[\"default\"], {\n className: \"bd-reload bd-reload-card\",\n onClick: this.reload\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"label\", {\n className: \"bd-switch-wrapper bd-flex-child\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"input\", {\n className: \"bd-switch-checkbox\",\n checked: this.state.checked,\n onChange: this.onChange,\n type: \"checkbox\"\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: this.state.checked ? \"bd-switch checked\" : \"bd-switch\"\n })))), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-description-wrap scroller-wrap fade\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-description scroller\"\n }, description)), this.footer);\n }\n\n onChange() {\n this.setState({\n checked: !this.state.checked\n });\n this.props.onChange && this.props.onChange(this.props.addon.id);\n }\n\n showSettings() {\n if (!this.hasSettings) return;\n this.setState({\n settingsOpen: true\n });\n }\n\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/ui/settings/plugincard.js?773d"],"names":["PluginCard","React","Component","constructor","props","onChange","bind","showSettings","state","checked","enabled","settingsOpen","hasSettings","addon","plugin","getSettingsPanel","settingsPanel","panelRef","createRef","reload","closeSettings","id","forceUpdate","componentDidUpdate","Node","current","appendChild","isHidden","container","element","cTop","scrollTop","cBottom","clientHeight","eTop","offsetTop","eBottom","panel","$","parents","animate","offset","top","getString","value","toString","innerHTML","setState","buildTitle","name","version","author","title","Strings","Addons","split","nameIndex","findIndex","s","createElement","className","versionIndex","authorIndex","flat","settingsComponent","err","Logger","stacktrace","ref","dangerouslySetInnerHTML","buildLink","which","url","footer","links","some","l","linkComponents","map","filter","c","comp","i","length","pluginSettings","render","description","showReloadIcon"],"mappings":"AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AAEe,MAAMA,UAAN,SAAyBC,6CAAK,CAACC,SAA/B,CAAyC;AAEpDC,aAAW,CAACC,KAAD,EAAQ;AACf,UAAMA,KAAN;AACA,SAAKC,QAAL,GAAgB,KAAKA,QAAL,CAAcC,IAAd,CAAmB,IAAnB,CAAhB;AACA,SAAKC,YAAL,GAAoB,KAAKA,YAAL,CAAkBD,IAAlB,CAAuB,IAAvB,CAApB;AACA,SAAKE,KAAL,GAAa;AACTC,aAAO,EAAE,KAAKL,KAAL,CAAWM,OADX;AAETC,kBAAY,EAAE;AAFL,KAAb;AAIA,SAAKC,WAAL,GAAmB,OAAO,KAAKR,KAAL,CAAWS,KAAX,CAAiBC,MAAjB,CAAwBC,gBAA/B,KAAoD,UAAvE;AACA,SAAKC,aAAL,GAAqB,EAArB;AACA,SAAKC,QAAL,GAAgBhB,6CAAK,CAACiB,SAAN,EAAhB;AAEA,SAAKC,MAAL,GAAc,KAAKA,MAAL,CAAYb,IAAZ,CAAiB,IAAjB,CAAd,CAZe,CAaf;;AACA,SAAKc,aAAL,GAAqB,KAAKA,aAAL,CAAmBd,IAAnB,CAAwB,IAAxB,CAArB;AACH;;AAEDa,QAAM,GAAG;AACL,QAAI,CAAC,KAAKf,KAAL,CAAWe,MAAhB,EAAwB;AACxB,SAAKf,KAAL,CAAWS,KAAX,GAAmB,KAAKT,KAAL,CAAWe,MAAX,CAAkB,KAAKf,KAAL,CAAWS,KAAX,CAAiBQ,EAAnC,CAAnB;AACA,SAAKC,WAAL;AACH;;AAEDC,oBAAkB,GAAG;AACjB,QAAI,CAAC,KAAKf,KAAL,CAAWG,YAAhB,EAA8B;AAC9B,QAAI,KAAKK,aAAL,YAA8BQ,IAAlC,EAAwC,KAAKP,QAAL,CAAcQ,OAAd,CAAsBC,WAAtB,CAAkC,KAAKV,aAAvC,EAFvB,CAIjB;;AACA,UAAMW,QAAQ,GAAG,CAACC,SAAD,EAAYC,OAAZ,KAAwB;AACrC,YAAMC,IAAI,GAAGF,SAAS,CAACG,SAAvB;AACA,YAAMC,OAAO,GAAGF,IAAI,GAAGF,SAAS,CAACK,YAAjC;AACA,YAAMC,IAAI,GAAGL,OAAO,CAACM,SAArB;AACA,YAAMC,OAAO,GAAGF,IAAI,GAAGL,OAAO,CAACI,YAA/B;AACA,aAASC,IAAI,GAAGJ,IAAP,IAAeM,OAAO,GAAGJ,OAAlC;AACH,KAND;;AAQA,UAAMK,KAAK,GAAGC,CAAC,CAAC,KAAKrB,QAAL,CAAcQ,OAAf,CAAf;AACA,UAAMG,SAAS,GAAGS,KAAK,CAACE,OAAN,CAAc,kBAAd,CAAlB;AACA,QAAI,CAACZ,QAAQ,CAACC,SAAS,CAAC,CAAD,CAAV,EAAeS,KAAK,CAAC,CAAD,CAApB,CAAb,EAAuC;AACvCT,aAAS,CAACY,OAAV,CAAkB;AACdT,eAAS,EAAEM,KAAK,CAACI,MAAN,GAAeC,GAAf,GAAqBd,SAAS,CAACa,MAAV,GAAmBC,GAAxC,GAA8Cd,SAAS,CAACG,SAAV,EAA9C,GAAsE;AADnE,KAAlB,EAEG,GAFH;AAGH;;AAEDY,WAAS,CAACC,KAAD,EAAQ;AAAC,WAAO,OAAOA,KAAP,IAAgB,QAAhB,GAA2BA,KAA3B,GAAmCA,KAAK,CAACC,QAAN,EAA1C;AAA4D;;AAE9EzB,eAAa,GAAG;AACZ,SAAKH,QAAL,CAAcQ,OAAd,CAAsBqB,SAAtB,GAAkC,EAAlC;AACA,SAAKC,QAAL,CAAc;AAACpC,kBAAY,EAAE;AAAf,KAAd;AACH;;AAEDqC,YAAU,CAACC,IAAD,EAAOC,OAAP,EAAgBC,MAAhB,EAAwB;AAC9B,UAAMC,KAAK,GAAGC,+CAAO,CAACC,MAAR,CAAeF,KAAf,CAAqBG,KAArB,CAA2B,iBAA3B,CAAd;AACA,UAAMC,SAAS,GAAGJ,KAAK,CAACK,SAAN,CAAgBC,CAAC,IAAIA,CAAC,IAAI,UAA1B,CAAlB;AACA,QAAIF,SAAJ,EAAeJ,KAAK,CAACI,SAAD,CAAL,GAAmBvD,6CAAK,CAAC0D,aAAN,CAAoB,MAApB,EAA4B;AAACC,eAAS,EAAE;AAAZ,KAA5B,EAAoDX,IAApD,CAAnB;AACf,UAAMY,YAAY,GAAGT,KAAK,CAACK,SAAN,CAAgBC,CAAC,IAAIA,CAAC,IAAI,aAA1B,CAArB;AACA,QAAIF,SAAJ,EAAeJ,KAAK,CAACS,YAAD,CAAL,GAAsB5D,6CAAK,CAAC0D,aAAN,CAAoB,MAApB,EAA4B;AAACC,eAAS,EAAE;AAAZ,KAA5B,EAAuDV,OAAvD,CAAtB;AACf,UAAMY,WAAW,GAAGV,KAAK,CAACK,SAAN,CAAgBC,CAAC,IAAIA,CAAC,IAAI,YAA1B,CAApB;AACA,QAAIF,SAAJ,EAAeJ,KAAK,CAACU,WAAD,CAAL,GAAqB7D,6CAAK,CAAC0D,aAAN,CAAoB,MAApB,EAA4B;AAACC,eAAS,EAAE;AAAZ,KAA5B,EAAsDT,MAAtD,CAArB;AACf,WAAOC,KAAK,CAACW,IAAN,EAAP;AACH;;AAED,MAAIC,iBAAJ,GAAwB;AACpB,UAAMnD,KAAK,GAAG,KAAKT,KAAL,CAAWS,KAAzB;AACA,UAAMoC,IAAI,GAAG,KAAKN,SAAL,CAAe9B,KAAK,CAACoC,IAArB,CAAb;;AACA,QAAI;AAAE,WAAKjC,aAAL,GAAqBH,KAAK,CAACC,MAAN,CAAaC,gBAAb,EAArB;AAAuD,KAA7D,CACA,OAAOkD,GAAP,EAAY;AAAEC,oDAAM,CAACC,UAAP,CAAkB,iBAAlB,EAAqC,sCAAsClB,IAAtC,GAA6C,GAAlF,EAAuFgB,GAAvF;AAA8F;;AAE5G,UAAM7D,KAAK,GAAG;AAACiB,QAAE,EAAG,mBAAkB4B,IAAK,EAA7B;AAAgCW,eAAS,EAAE,iBAA3C;AAA8DQ,SAAG,EAAE,KAAKnD;AAAxE,KAAd;AACA,QAAI,OAAOD,aAAP,IAAyB,QAA7B,EAAuCZ,KAAK,CAACiE,uBAAN,GAAgC,KAAKrD,aAArC;AAEvC,WAAO;AAAI,eAAS,EAAC;AAAd,OACK;AAAK,eAAS,EAAC,UAAf;AAA0B,aAAO,EAAE,KAAKI;AAAxC,OAAuD,4DAAC,oDAAD,OAAvD,CADL,EAEK,mEAAShB,KAAT,EAAiB,KAAKY,aAAL,YAA8Bf,6CAAK,CAACC,SAApC,GAAgD,KAAKc,aAArD,GAAqE,IAAtF,CAFL,CAAP;AAIH;;AAEDsD,WAAS,CAACC,KAAD,EAAQ;AACb,UAAMC,GAAG,GAAG,KAAKpE,KAAL,CAAWS,KAAX,CAAiB0D,KAAjB,CAAZ;AACA,QAAI,CAACC,GAAL,EAAU,OAAO,IAAP;AACV,WAAO;AAAG,eAAS,EAAC,yBAAb;AAAuC,UAAI,EAAEA,GAA7C;AAAkD,YAAM,EAAC,QAAzD;AAAkE,SAAG,EAAC;AAAtE,OAA6FnB,+CAAO,CAACC,MAAR,CAAeiB,KAAf,CAA7F,CAAP;AACH;;AAED,MAAIE,MAAJ,GAAa;AACT,UAAMC,KAAK,GAAG,CAAC,SAAD,EAAY,QAAZ,CAAd;AACA,QAAI,CAACA,KAAK,CAACC,IAAN,CAAWC,CAAC,IAAI,KAAKxE,KAAL,CAAWS,KAAX,CAAiB+D,CAAjB,CAAhB,CAAD,IAAyC,CAAC,KAAKhE,WAAnD,EAAgE,OAAO,IAAP;AAChE,UAAMiE,cAAc,GAAGH,KAAK,CAACI,GAAN,CAAU,KAAKR,SAAL,CAAehE,IAAf,CAAoB,IAApB,CAAV,EAAqCyE,MAArC,CAA4CC,CAAC,IAAIA,CAAjD,CAAvB;AACA,WAAO;AAAK,eAAS,EAAC;AAAf,OACK;AAAM,eAAS,EAAC;AAAhB,OAA4BH,cAAc,CAACC,GAAf,CAAmB,CAACG,IAAD,EAAOC,CAAP,KAAaA,CAAC,GAAGL,cAAc,CAACM,MAAf,GAAwB,CAA5B,GAAgC,CAACF,IAAD,EAAO,KAAP,CAAhC,GAAgD,CAACA,IAAD,CAAhF,EAAwFlB,IAAxF,EAA5B,CADL,EAEM,KAAKnD,WAAL,IAAoB;AAAQ,aAAO,EAAE,KAAKL,YAAtB;AAAoC,eAAS,EAAC,qCAA9C;AAAoF,cAAQ,EAAE,CAAC,KAAKC,KAAL,CAAWC;AAA1G,OAAoH4C,+CAAO,CAACC,MAAR,CAAe8B,cAAnI,CAF1B,CAAP;AAIH;;AAEDC,QAAM,GAAG;AACL,QAAI,KAAK7E,KAAL,CAAWG,YAAf,EAA6B,OAAO,KAAKqD,iBAAZ;AAE7B,UAAM;AAACnD;AAAD,QAAU,KAAKT,KAArB;AACA,UAAM6C,IAAI,GAAG,KAAKN,SAAL,CAAe9B,KAAK,CAACoC,IAArB,CAAb;AACA,UAAME,MAAM,GAAG,KAAKR,SAAL,CAAe9B,KAAK,CAACsC,MAArB,CAAf;AACA,UAAMmC,WAAW,GAAG,KAAK3C,SAAL,CAAe9B,KAAK,CAACyE,WAArB,CAApB;AACA,UAAMpC,OAAO,GAAG,KAAKP,SAAL,CAAe9B,KAAK,CAACqC,OAArB,CAAhB;AAEA,WAAO;AAAI,cAAQ,EAAED,IAAd;AAAoB,iBAAW,EAAEC,OAAjC;AAA0C,eAAS,EAAC;AAApD,OACK;AAAK,eAAS,EAAC;AAAf,OACQ;AAAM,eAAS,EAAC;AAAhB,OAAmC,KAAKF,UAAL,CAAgBC,IAAhB,EAAsBC,OAAtB,EAA+BC,MAA/B,CAAnC,CADR,EAEQ;AAAK,eAAS,EAAC;AAAf,OACK,KAAK/C,KAAL,CAAWmF,cAAX,IAA6B,4DAAC,qDAAD;AAAY,eAAS,EAAC,0BAAtB;AAAiD,aAAO,EAAE,KAAKpE;AAA/D,MADlC,EAEI;AAAO,eAAS,EAAC;AAAjB,OACI;AAAO,eAAS,EAAC,oBAAjB;AAAsC,aAAO,EAAE,KAAKX,KAAL,CAAWC,OAA1D;AAAmE,cAAQ,EAAE,KAAKJ,QAAlF;AAA4F,UAAI,EAAC;AAAjG,MADJ,EAEI;AAAK,eAAS,EAAE,KAAKG,KAAL,CAAWC,OAAX,GAAqB,mBAArB,GAA2C;AAA3D,MAFJ,CAFJ,CAFR,CADL,EAWK;AAAK,eAAS,EAAC;AAAf,OAAwD;AAAK,eAAS,EAAC;AAAf,OAA0C6E,WAA1C,CAAxD,CAXL,EAYM,KAAKb,MAZX,CAAP;AAcH;;AAEDpE,UAAQ,GAAG;AACP,SAAK0C,QAAL,CAAc;AAACtC,aAAO,EAAE,CAAC,KAAKD,KAAL,CAAWC;AAAtB,KAAd;AACA,SAAKL,KAAL,CAAWC,QAAX,IAAuB,KAAKD,KAAL,CAAWC,QAAX,CAAoB,KAAKD,KAAL,CAAWS,KAAX,CAAiBQ,EAArC,CAAvB;AACH;;AAEDd,cAAY,GAAG;AACX,QAAI,CAAC,KAAKK,WAAV,EAAuB;AACvB,SAAKmC,QAAL,CAAc;AAACpC,kBAAY,EAAE;AAAf,KAAd;AACH;;AAhImD","file":"./src/ui/settings/plugincard.js.js","sourcesContent":["import {React, Logger, Strings} from \"modules\";\r\nimport CloseButton from \"../icons/close\";\r\nimport ReloadIcon from \"../icons/reload\";\r\n\r\nexport default class PluginCard extends React.Component {\r\n\r\n    constructor(props) {\r\n        super(props);\r\n        this.onChange = this.onChange.bind(this);\r\n        this.showSettings = this.showSettings.bind(this);\r\n        this.state = {\r\n            checked: this.props.enabled,\r\n            settingsOpen: false\r\n        };\r\n        this.hasSettings = typeof this.props.addon.plugin.getSettingsPanel === \"function\";\r\n        this.settingsPanel = \"\";\r\n        this.panelRef = React.createRef();\r\n\r\n        this.reload = this.reload.bind(this);\r\n        // this.onReload = this.onReload.bind(this);\r\n        this.closeSettings = this.closeSettings.bind(this);\r\n    }\r\n\r\n    reload() {\r\n        if (!this.props.reload) return;\r\n        this.props.addon = this.props.reload(this.props.addon.id);\r\n        this.forceUpdate();\r\n    }\r\n\r\n    componentDidUpdate() {\r\n        if (!this.state.settingsOpen) return;\r\n        if (this.settingsPanel instanceof Node) this.panelRef.current.appendChild(this.settingsPanel);\r\n\r\n        // if (!SettingsCookie[\"fork-ps-3\"]) return;\r\n        const isHidden = (container, element) => {\r\n            const cTop = container.scrollTop;\r\n            const cBottom = cTop + container.clientHeight;\r\n            const eTop = element.offsetTop;\r\n            const eBottom = eTop + element.clientHeight;\r\n            return  (eTop < cTop || eBottom > cBottom);\r\n        };\r\n\r\n        const panel = $(this.panelRef.current);\r\n        const container = panel.parents(\".scroller-2FKFPG\");\r\n        if (!isHidden(container[0], panel[0])) return;\r\n        container.animate({\r\n            scrollTop: panel.offset().top - container.offset().top + container.scrollTop() - 30\r\n        }, 300);\r\n    }\r\n\r\n    getString(value) {return typeof value == \"string\" ? value : value.toString();}\r\n\r\n    closeSettings() {\r\n        this.panelRef.current.innerHTML = \"\";\r\n        this.setState({settingsOpen: false});\r\n    }\r\n\r\n    buildTitle(name, version, author) {\r\n        const title = Strings.Addons.title.split(/({{[A-Za-z]+}})/);\r\n        const nameIndex = title.findIndex(s => s == \"{{name}}\");\r\n        if (nameIndex) title[nameIndex] = React.createElement(\"span\", {className: \"bd-name\"}, name);\r\n        const versionIndex = title.findIndex(s => s == \"{{version}}\");\r\n        if (nameIndex) title[versionIndex] = React.createElement(\"span\", {className: \"bd-version\"}, version);\r\n        const authorIndex = title.findIndex(s => s == \"{{author}}\");\r\n        if (nameIndex) title[authorIndex] = React.createElement(\"span\", {className: \"bd-author\"}, author);\r\n        return title.flat();\r\n    }\r\n\r\n    get settingsComponent() {\r\n        const addon = this.props.addon;\r\n        const name = this.getString(addon.name);\r\n        try { this.settingsPanel = addon.plugin.getSettingsPanel(); }\r\n        catch (err) { Logger.stacktrace(\"Plugin Settings\", \"Unable to get settings panel for \" + name + \".\", err); }\r\n\r\n        const props = {id: `plugin-settings-${name}`, className: \"plugin-settings\", ref: this.panelRef};\r\n        if (typeof(settingsPanel) == \"string\") props.dangerouslySetInnerHTML = this.settingsPanel;\r\n\r\n        return <li className=\"settings-open bd-switch-item\">\r\n                    <div className=\"bd-close\" onClick={this.closeSettings}><CloseButton /></div>\r\n                    <div {...props}>{this.settingsPanel instanceof React.Component ? this.settingsPanel : null}</div>\r\n                </li>;\r\n    }\r\n\r\n    buildLink(which) {\r\n        const url = this.props.addon[which];\r\n        if (!url) return null;\r\n        return <a className=\"bd-link bd-link-website\" href={url} target=\"_blank\" rel=\"noopener noreferrer\">{Strings.Addons[which]}</a>;\r\n    }\r\n\r\n    get footer() {\r\n        const links = [\"website\", \"source\"];\r\n        if (!links.some(l => this.props.addon[l]) && !this.hasSettings) return null;\r\n        const linkComponents = links.map(this.buildLink.bind(this)).filter(c => c);\r\n        return <div className=\"bd-footer\">\r\n                    <span className=\"bd-links\">{linkComponents.map((comp, i) => i < linkComponents.length - 1 ? [comp, \" | \"] : [comp]).flat()}</span>\r\n                    {this.hasSettings && <button onClick={this.showSettings} className=\"bd-button bd-button-plugin-settings\" disabled={!this.state.checked}>{Strings.Addons.pluginSettings}</button>}\r\n                </div>;\r\n    }\r\n\r\n    render() {\r\n        if (this.state.settingsOpen) return this.settingsComponent;\r\n\r\n        const {addon} = this.props;\r\n        const name = this.getString(addon.name);\r\n        const author = this.getString(addon.author);\r\n        const description = this.getString(addon.description);\r\n        const version = this.getString(addon.version);\r\n\r\n        return <li dataName={name} dataVersion={version} className=\"settings-closed bd-switch-item\">\r\n                    <div className=\"bd-header\">\r\n                            <span className=\"bd-header-title\">{this.buildTitle(name, version, author)}</span>\r\n                            <div className=\"bd-controls\">\r\n                                {this.props.showReloadIcon && <ReloadIcon className=\"bd-reload bd-reload-card\" onClick={this.reload} />}\r\n                                <label className=\"bd-switch-wrapper bd-flex-child\">\r\n                                    <input className=\"bd-switch-checkbox\" checked={this.state.checked} onChange={this.onChange} type=\"checkbox\" />\r\n                                    <div className={this.state.checked ? \"bd-switch checked\" : \"bd-switch\"} />\r\n                                </label>\r\n                            </div>\r\n                    </div>\r\n                    <div className=\"bd-description-wrap scroller-wrap fade\"><div className=\"bd-description scroller\">{description}</div></div>\r\n                    {this.footer}\r\n                </li>;\r\n    }\r\n\r\n    onChange() {\r\n        this.setState({checked: !this.state.checked});\r\n        this.props.onChange && this.props.onChange(this.props.addon.id);\r\n    }\r\n\r\n    showSettings() {\r\n        if (!this.hasSettings) return;\r\n        this.setState({settingsOpen: true});\r\n    }\r\n}"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/ui/settings/plugincard.js\n"); - -/***/ }), - /***/ "./src/ui/settings/switch.jsx": /*!************************************!*\ !*** ./src/ui/settings/switch.jsx ***! @@ -927,18 +927,6 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) * /***/ }), -/***/ "./src/ui/settings/themecard.js": -/*!**************************************!*\ - !*** ./src/ui/settings/themecard.js ***! - \**************************************/ -/*! exports provided: default */ -/***/ (function(module, __webpack_exports__, __webpack_require__) { - -"use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, \"default\", function() { return ThemeCard; });\n/* harmony import */ var modules__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! modules */ \"./src/modules/modules.js\");\n/* harmony import */ var _icons_reload__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../icons/reload */ \"./src/ui/icons/reload.jsx\");\n\n // import Toasts from \"../toasts\";\n\nclass ThemeCard extends modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].Component {\n constructor(props) {\n super(props);\n this.state = {\n checked: this.props.enabled,\n reloads: 0\n };\n this.onChange = this.onChange.bind(this);\n this.reload = this.reload.bind(this);\n }\n\n reload() {\n if (!this.props.reload) return;\n this.props.addon = this.props.reload(this.props.addon.id);\n this.forceUpdate();\n }\n\n buildTitle(name, version, author) {\n const title = modules__WEBPACK_IMPORTED_MODULE_0__[\"Strings\"].Addons.title.split(/({{[A-Za-z]+}})/);\n const nameIndex = title.findIndex(s => s == \"{{name}}\");\n if (nameIndex) title[nameIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-name\"\n }, name);\n const versionIndex = title.findIndex(s => s == \"{{version}}\");\n if (nameIndex) title[versionIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-version\"\n }, version);\n const authorIndex = title.findIndex(s => s == \"{{author}}\");\n if (nameIndex) title[authorIndex] = modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-author\"\n }, author);\n return title.flat();\n }\n\n render() {\n const {\n addon\n } = this.props;\n const name = addon.name;\n const description = addon.description;\n const version = addon.version;\n const author = addon.author;\n const website = addon.website;\n const source = addon.source;\n return modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"li\", {\n \"data-name\": name,\n \"data-version\": version,\n \"className\": \"settings-closed bd-switch-item\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-header\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-header-title\"\n }, this.buildTitle(name, version, author)), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-controls\"\n }, this.props.showReloadIcon && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(_icons_reload__WEBPACK_IMPORTED_MODULE_1__[\"default\"], {\n className: \"bd-reload bd-reload-card\",\n onClick: this.reload\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"label\", {\n className: \"bd-switch-wrapper bd-flex-child\",\n style: {\n flex: \"0 0 auto\"\n }\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"input\", {\n checked: this.state.checked,\n onChange: this.onChange,\n className: \"bd-switch-checkbox\",\n type: \"checkbox\"\n }), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: this.state.checked ? \"bd-switch checked\" : \"bd-switch\"\n })))), modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-description-wrap scroller-wrap fade\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-description scroller\"\n }, description)), (website || source) && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"div\", {\n className: \"bd-footer\"\n }, modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"span\", {\n className: \"bd-links\"\n }, website && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"a\", {\n className: \"bd-link\",\n href: website,\n target: \"_blank\"\n }, \"Website\"), website && source && \" | \", source && modules__WEBPACK_IMPORTED_MODULE_0__[\"React\"].createElement(\"a\", {\n className: \"bd-link\",\n href: source,\n target: \"_blank\"\n }, \"Source\"))));\n }\n\n onChange() {\n this.setState({\n checked: !this.state.checked\n });\n this.props.onChange && this.props.onChange(this.props.addon.id);\n }\n\n}//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack://Core/./src/ui/settings/themecard.js?c05c"],"names":["ThemeCard","React","Component","constructor","props","state","checked","enabled","reloads","onChange","bind","reload","addon","id","forceUpdate","buildTitle","name","version","author","title","Strings","Addons","split","nameIndex","findIndex","s","createElement","className","versionIndex","authorIndex","flat","render","description","website","source","showReloadIcon","ReloadIcon","onClick","style","flex","type","href","target","setState"],"mappings":"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","file":"./src/ui/settings/themecard.js.js","sourcesContent":["import {React, Strings} from \"modules\";\r\nimport ReloadIcon from \"../icons/reload\";\r\n// import Toasts from \"../toasts\";\r\n\r\nexport default class ThemeCard extends React.Component {\r\n\r\n    constructor(props) {\r\n        super(props);\r\n        this.state = {\r\n            checked: this.props.enabled,\r\n            reloads: 0\r\n        };\r\n        this.onChange = this.onChange.bind(this);\r\n        this.reload = this.reload.bind(this);\r\n    }\r\n\r\n    reload() {\r\n        if (!this.props.reload) return;\r\n        this.props.addon = this.props.reload(this.props.addon.id);\r\n        this.forceUpdate();\r\n    }\r\n\r\n    buildTitle(name, version, author) {\r\n        const title = Strings.Addons.title.split(/({{[A-Za-z]+}})/);\r\n        const nameIndex = title.findIndex(s => s == \"{{name}}\");\r\n        if (nameIndex) title[nameIndex] = React.createElement(\"span\", {className: \"bd-name\"}, name);\r\n        const versionIndex = title.findIndex(s => s == \"{{version}}\");\r\n        if (nameIndex) title[versionIndex] = React.createElement(\"span\", {className: \"bd-version\"}, version);\r\n        const authorIndex = title.findIndex(s => s == \"{{author}}\");\r\n        if (nameIndex) title[authorIndex] = React.createElement(\"span\", {className: \"bd-author\"}, author);\r\n        return title.flat();\r\n    }\r\n\r\n    render() {\r\n        const {addon} = this.props;\r\n        const name = addon.name;\r\n        const description = addon.description;\r\n        const version = addon.version;\r\n        const author = addon.author;\r\n        const website = addon.website;\r\n        const source = addon.source;\r\n\r\n        return React.createElement(\"li\", {\"data-name\": name, \"data-version\": version, \"className\": \"settings-closed bd-switch-item\"},\r\n            React.createElement(\"div\", {className: \"bd-header\"},\r\n                    React.createElement(\"span\", {className: \"bd-header-title\"},\r\n                        this.buildTitle(name, version, author)\r\n                    ),\r\n                    React.createElement(\"div\", {className: \"bd-controls\"},\r\n                    this.props.showReloadIcon && React.createElement(ReloadIcon, {className: \"bd-reload bd-reload-card\", onClick: this.reload}),\r\n                        React.createElement(\"label\", {className: \"bd-switch-wrapper bd-flex-child\", style: {flex: \"0 0 auto\"}},\r\n                            React.createElement(\"input\", {checked: this.state.checked, onChange: this.onChange, className: \"bd-switch-checkbox\", type: \"checkbox\"}),\r\n                            React.createElement(\"div\", {className: this.state.checked ? \"bd-switch checked\" : \"bd-switch\"})\r\n                        )\r\n                    )\r\n            ),\r\n            React.createElement(\"div\", {className: \"bd-description-wrap scroller-wrap fade\"},\r\n                React.createElement(\"div\", {className: \"bd-description scroller\"}, description)\r\n            ),\r\n            (website || source) && React.createElement(\"div\", {className: \"bd-footer\"},\r\n                React.createElement(\"span\", {className: \"bd-links\"},\r\n                    website && React.createElement(\"a\", {className: \"bd-link\", href: website, target: \"_blank\"}, \"Website\"),\r\n                    website && source && \" | \",\r\n                    source && React.createElement(\"a\", {className: \"bd-link\", href: source, target: \"_blank\"}, \"Source\")\r\n                )\r\n            )\r\n        );\r\n    }\r\n\r\n    onChange() {\r\n        this.setState({checked: !this.state.checked});\r\n        this.props.onChange && this.props.onChange(this.props.addon.id);\r\n    }\r\n}"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/ui/settings/themecard.js\n"); - -/***/ }), - /***/ "./src/ui/settings/title.jsx": /*!***********************************!*\ !*** ./src/ui/settings/title.jsx ***! diff --git a/src/builtins/emotes.js b/src/builtins/emotes.js index 9423c7c3..d4e2aee7 100644 --- a/src/builtins/emotes.js +++ b/src/builtins/emotes.js @@ -5,6 +5,7 @@ import {Utilities, WebpackModules, DataStore, DiscordModules, Events, Settings, import BDEmote from "../ui/emote"; import Toasts from "../ui/toasts"; // import EmoteMenu from "./emotemenu"; +const request = require("request"); const Emotes = { TwitchGlobal: {}, @@ -67,11 +68,11 @@ export default new class EmoteModule extends Builtin { async enabled() { Settings.registerCollection("emotes", "Emotes", EmoteConfig, {title: Strings.Emotes.clearEmotes, onClick: () => { this.clearEmoteData(); this.loadEmoteData(EmoteInfo); }}); // Disable emote module for now because it's annoying and slow - // await this.getBlacklist(); - // await this.loadEmoteData(EmoteInfo); + await this.getBlacklist(); + await this.loadEmoteData(EmoteInfo); - // while (!this.MessageContentComponent) await new Promise(resolve => setTimeout(resolve, 100)); - // this.patchMessageContent(); + while (!this.MessageContentComponent) await new Promise(resolve => setTimeout(resolve, 100)); + this.patchMessageContent(); Events.on("emotes-favorite-added", this.addFavorite); Events.on("emotes-favorite-removed", this.removeFavorite); } @@ -234,7 +235,10 @@ export default new class EmoteModule extends Builtin { } downloadEmotes(emoteMeta) { - const request = require("request"); + const repoFile = Utilities.repoUrl(`data/emotes/${emoteMeta.variable.toLowerCase()}.json`); + if (emoteMeta.url && !emoteMeta.backup) emoteMeta.backup = repoFile; + if (!emoteMeta.url) emoteMeta.url = repoFile; + const options = { url: emoteMeta.url, timeout: emoteMeta.timeout ? emoteMeta.timeout : 5000, @@ -244,10 +248,10 @@ export default new class EmoteModule extends Builtin { this.log(`Downloading: ${emoteMeta.variable} (${emoteMeta.url})`); return new Promise((resolve, reject) => { - request(options, (error, response, parsedData) => { + request.get(options, (error, response, parsedData) => { if (error) { this.stacktrace("Could not download " + emoteMeta.variable, error); - if (emoteMeta.backup) { + if (emoteMeta.backup || emoteMeta.url) { emoteMeta.url = emoteMeta.backup; emoteMeta.backup = null; if (emoteMeta.backupParser) emoteMeta.parser = emoteMeta.backupParser; @@ -273,8 +277,9 @@ export default new class EmoteModule extends Builtin { getBlacklist() { return new Promise(resolve => { - $.getJSON(`https://rauenzi.github.io/BetterDiscordApp/data/emotefilter.json`, function (data) { - resolve(blacklist.push(...data.blacklist)); + request.get({url: Utilities.repoUrl(`data/emotes/blacklist.json`), json: true}, (err, resp, data) => { + if (err || resp.statusCode != 200) return resolve(); + resolve(blacklist.push(...data)); }); }); } diff --git a/src/data/emotes/info.js b/src/data/emotes/info.js index baa3c1ca..240adf23 100644 --- a/src/data/emotes/info.js +++ b/src/data/emotes/info.js @@ -1,19 +1,16 @@ export default { TwitchGlobal: { url: "https://twitchemotes.com/api_cache/v3/global.json", - backup: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_twitch_global.json`, variable: "TwitchGlobal", getEmoteURL: (e) => `https://static-cdn.jtvnw.net/emoticons/v1/${e.id}/1.0`, - getOldData: (url, name) => { return {id: url.match(/\/([0-9]+)\//)[1], code: name, emoticon_set: 0, description: null}; } + getOldData: (url, name) => {return {id: url.match(/\/([0-9]+)\//)[1], code: name, emoticon_set: 0, description: null};} }, TwitchSubscriber: { - url: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_twitch_subscriber.json`, variable: "TwitchSubscriber", getEmoteURL: (e) => `https://static-cdn.jtvnw.net/emoticons/v1/${e}/1.0`, getOldData: (url) => url.match(/\/([0-9]+)\//)[1] }, FrankerFaceZ: { - url: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_ffz.json`, variable: "FrankerFaceZ", getEmoteURL: (e) => `https://cdn.frankerfacez.com/emoticon/${e}/1`, getOldData: (url) => url.match(/\/([0-9]+)\//)[1] @@ -33,7 +30,6 @@ export default { getOldData: (url) => url }, BTTV2: { - url: `https://rauenzi.github.io/BetterDiscordApp/data/emotedata_bttv.json`, variable: "BTTV2", oldVariable: "emotesBTTV2", getEmoteURL: (e) => `https://cdn.betterttv.net/emote/${e}/1x`, diff --git a/src/data/strings.js b/src/data/strings.js index de0c15d2..2bebf08c 100644 --- a/src/data/strings.js +++ b/src/data/strings.js @@ -168,7 +168,7 @@ export default { title: "{{name}} v{{version}} by {{author}}", openFolder: "Open {{type}} Folder", reload: "Reload", - pluginSettings: "Settings", + addonSettings: "Settings", website: "Website", source: "Source", server: "Support Server", diff --git a/src/modules/datastore.js b/src/modules/datastore.js index 2a4f4da2..6ad5d558 100644 --- a/src/modules/datastore.js +++ b/src/modules/datastore.js @@ -17,44 +17,31 @@ export default new class DataStore { constructor() { this.data = {misc: {}}; this.pluginData = {}; + this.localeHashes = {}; } initialize() { if (!fs.existsSync(this.baseFolder)) fs.mkdirSync(this.baseFolder); if (!fs.existsSync(this.dataFolder)) fs.mkdirSync(this.dataFolder); if (!fs.existsSync(this.localeFolder)) fs.mkdirSync(this.localeFolder); + if (!fs.existsSync(this.localeCache)) fs.writeFileSync(this.localeCache, JSON.stringify({})); if (!fs.existsSync(this.BDFile)) fs.writeFileSync(this.BDFile, JSON.stringify(this.data.misc, null, 4)); if (!fs.existsSync(this.customCSS)) fs.writeFileSync(this.customCSS, ""); const dataFiles = fs.readdirSync(this.dataFolder).filter(f => !fs.statSync(path.resolve(this.dataFolder, f)).isDirectory() && f.endsWith(".json")); for (const file of dataFiles) { this.data[file.split(".")[0]] = __non_webpack_require__(path.resolve(this.dataFolder, file)); } - // this.data = __non_webpack_require__(this.BDFile); - // if (data.hasOwnProperty("settings")) this.data = data; - // if (!fs.existsSync(this.settingsFile)) return; - // let settings = __non_webpack_require__(this.settingsFile); - // fs.unlinkSync(this.settingsFile); - // if (settings.hasOwnProperty("settings")) settings = Object.assign({stable: {}, canary: {}, ptb: {}}, {[releaseChannel]: settings}); - // else settings = Object.assign({stable: {}, canary: {}, ptb: {}}, settings); - // this.setBDData("settings", settings); + this.localeHashes = JSON.parse(fs.readFileSync(this.localeCache).toString()); } get customCSS() {return this._customCSS || (this._customCSS = path.resolve(this.dataFolder, "custom.css"));} get baseFolder() {return this._baseFolder || (this._baseFolder = path.resolve(Config.dataPath, "data"));} get dataFolder() {return this._dataFolder || (this._dataFolder = path.resolve(this.baseFolder, `${releaseChannel}`));} get localeFolder() {return this._localeFolder || (this._localeFolder = path.resolve(this.baseFolder, `locales`));} + get localeCache() {return this._localeCache || (this._localeCache = path.resolve(this.localeFolder, `.cache`));} get BDFile() {return this._BDFile || (this._BDFile = path.resolve(Config.dataPath, "data", `${releaseChannel}.json`));} - // get settingsFile() {return this._settingsFile || (this._settingsFile = path.resolve(Config.dataPath, "bdsettings.json"));} getPluginFile(pluginName) {return path.resolve(Config.dataPath, "plugins", pluginName + ".config.json");} - // getSettingGroup(key) { - // return this.data.settings[key] || null; - // } - - // setSettingGroup(key, data) { - // this.data.settings[key] = data; - // fs.writeFileSync(this.BDFile, JSON.stringify(this.data, null, 4)); - // } _getFile(key) { if (key == "settings" || key == "plugins" || key == "themes") return path.resolve(this.dataFolder, `${key}.json`); @@ -80,6 +67,15 @@ export default new class DataStore { fs.writeFileSync(path.resolve(this.localeFolder, `${locale}.json`), JSON.stringify(strings, null, 4)); } + getLocaleHash(locale) { + return this.localeHashes[locale] || ""; + } + + saveLocaleHash(locale, hash) { + this.localeHashes[locale] = hash; + fs.writeFileSync(this.localeCache, JSON.stringify(this.localeHashes, null, 4)); + } + getData(key) { return this.data[key] || ""; } diff --git a/src/modules/localemanager.js b/src/modules/localemanager.js index 44a8bc71..03cd863a 100644 --- a/src/modules/localemanager.js +++ b/src/modules/localemanager.js @@ -15,7 +15,7 @@ export default new class LocaleManager { this.locale = ""; this.strings = {}; } - + async initialize() { await this.setLocale(this.discordLocale); Dispatcher.subscribe(DiscordConstants.ActionTypes.USER_SETTINGS_UPDATE, ({settings}) => { @@ -27,8 +27,7 @@ export default new class LocaleManager { async setLocale(newLocale) { let newStrings; if (newLocale != this.defaultLocale) { - const savedStrings = DataStore.getLocale(newLocale); - newStrings = savedStrings || await this.downloadLocale(newLocale); + newStrings = await this.getLocaleStrings(newLocale); if (!newStrings) return this.setLocale(this.defaultLocale); } else { @@ -39,16 +38,28 @@ export default new class LocaleManager { Events.emit("strings-updated"); } - downloadLocale(locale) { + async getLocaleStrings(locale) { + const hash = DataStore.getLocaleHash(locale); + if (!hash) return await this.downloadLocale(locale); + const invalid = await this.downloadLocale(locale, hash); + if (!invalid) return DataStore.getLocale(locale); + return invalid; + } + + downloadLocale(locale, hash = "") { return new Promise(resolve => { const options = { - url: `https://raw.githubusercontent.com/rauenzi/BetterDiscordApp/development/data/locales/${locale}.json`,//`https://rauenzi.github.io/BetterDiscordApp/data/locales/${discordLocale}.json`, + url: Utilities.repoUrl(`data/locales/${locale}.json`), timeout: 2000, json: true }; + if (hash) options.headers = {"If-None-Match": hash}; + console.log(options.headers); request.get(options, (err, resp, newStrings) => { - if (err || resp.statusCode !== 200) return resolve(null); - DataStore.saveLocale(locale, newStrings); + if (err || resp.statusCode !== 200) return resolve(null); + console.log(resp); + DataStore.saveLocale(locale, newStrings); + DataStore.saveLocaleHash(locale, resp.headers.etag); resolve(newStrings); }); }); diff --git a/src/modules/utilities.js b/src/modules/utilities.js index 684abff4..79123da0 100644 --- a/src/modules/utilities.js +++ b/src/modules/utilities.js @@ -1,7 +1,12 @@ +import {Config} from "data"; import Logger from "./logger"; export default class Utilities { + static repoUrl(path) { + return `https://cdn.staticaly.com/gh/${Config.repo}/BetterDiscordApp/${Config.hash}/${path}`; + } + /** * Parses a string of HTML and returns the results. If the second parameter is true, * the parsed HTML will be returned as a document fragment {@see https://developer.mozilla.org/en-US/docs/Web/API/DocumentFragment}. diff --git a/src/ui/settings/addoncard.jsx b/src/ui/settings/addoncard.jsx index fb22f86f..f82ef4b0 100644 --- a/src/ui/settings/addoncard.jsx +++ b/src/ui/settings/addoncard.jsx @@ -2,22 +2,21 @@ import {React, Logger, Strings} from "modules"; import CloseButton from "../icons/close"; import ReloadIcon from "../icons/reload"; -export default class PluginCard extends React.Component { +export default class AddonCard extends React.Component { constructor(props) { super(props); - this.onChange = this.onChange.bind(this); - this.showSettings = this.showSettings.bind(this); this.state = { checked: this.props.enabled, settingsOpen: false }; - this.hasSettings = typeof this.props.addon.plugin.getSettingsPanel === "function"; + this.settingsPanel = ""; this.panelRef = React.createRef(); + this.onChange = this.onChange.bind(this); this.reload = this.reload.bind(this); - // this.onReload = this.onReload.bind(this); + this.showSettings = this.showSettings.bind(this); this.closeSettings = this.closeSettings.bind(this); } @@ -50,6 +49,16 @@ export default class PluginCard extends React.Component { getString(value) {return typeof value == "string" ? value : value.toString();} + onChange() { + this.setState({checked: !this.state.checked}); + this.props.onChange && this.props.onChange(this.props.addon.id); + } + + showSettings() { + if (!this.props.hasSettings) return; + this.setState({settingsOpen: true}); + } + closeSettings() { this.panelRef.current.innerHTML = ""; this.setState({settingsOpen: false}); @@ -69,10 +78,10 @@ export default class PluginCard extends React.Component { get settingsComponent() { const addon = this.props.addon; const name = this.getString(addon.name); - try { this.settingsPanel = addon.plugin.getSettingsPanel(); } - catch (err) { Logger.stacktrace("Plugin Settings", "Unable to get settings panel for " + name + ".", err); } + try { this.settingsPanel = this.props.getSettingsPanel(); } + catch (err) { Logger.stacktrace("Addon Settings", "Unable to get settings panel for " + name + ".", err); } - const props = {id: `plugin-settings-${name}`, className: "plugin-settings", ref: this.panelRef}; + const props = {id: `${name}-settings`, className: "addon-settings", ref: this.panelRef}; if (typeof(settingsPanel) == "string") props.dangerouslySetInnerHTML = this.settingsPanel; return
  • @@ -89,18 +98,18 @@ export default class PluginCard extends React.Component { get footer() { const links = ["website", "source"]; - if (!links.some(l => this.props.addon[l]) && !this.hasSettings) return null; + if (!links.some(l => this.props.addon[l]) && !this.props.hasSettings) return null; const linkComponents = links.map(this.buildLink.bind(this)).filter(c => c); return
    {linkComponents.map((comp, i) => i < linkComponents.length - 1 ? [comp, " | "] : [comp]).flat()} - {this.hasSettings && } + {this.props.hasSettings && }
    ; } render() { if (this.state.settingsOpen) return this.settingsComponent; - const {addon} = this.props; + const addon = this.props.addon; const name = this.getString(addon.name); const author = this.getString(addon.author); const description = this.getString(addon.description); @@ -111,7 +120,7 @@ export default class PluginCard extends React.Component { {this.buildTitle(name, version, author)}
    {this.props.showReloadIcon && } -
  • ; } - - onChange() { - this.setState({checked: !this.state.checked}); - this.props.onChange && this.props.onChange(this.props.addon.id); - } - - showSettings() { - if (!this.hasSettings) return; - this.setState({settingsOpen: true}); - } -} \ No newline at end of file +} diff --git a/src/ui/settings/addonlist.jsx b/src/ui/settings/addonlist.jsx index 29d567cb..b1f79de3 100644 --- a/src/ui/settings/addonlist.jsx +++ b/src/ui/settings/addonlist.jsx @@ -1,9 +1,8 @@ import {React, Settings, Strings} from "modules"; import SettingsTitle from "./title"; -import PluginCard from "./plugincard"; -import ThemeCard from "./themecard"; import ReloadIcon from "../icons/reload"; +import AddonCard from "./addoncard"; export default class AddonList extends React.Component { @@ -20,8 +19,9 @@ export default class AddonList extends React.Component { } />,
      {addonList.sort((a, b) => a.name.toLowerCase().localeCompare(b.name.toLowerCase())).map(addon => { - const CardType = addon.type ? PluginCard : ThemeCard; - return ; + const hasSettings = addon.type && typeof(addon.plugin.getSettingsPanel) === "function"; + const getSettings = hasSettings && addon.plugin.getSettingsPanel.bind(addon.plugin); + return ; })}
    ]; diff --git a/src/ui/settings/plugincard.js b/src/ui/settings/plugincard.js deleted file mode 100644 index fb22f86f..00000000 --- a/src/ui/settings/plugincard.js +++ /dev/null @@ -1,134 +0,0 @@ -import {React, Logger, Strings} from "modules"; -import CloseButton from "../icons/close"; -import ReloadIcon from "../icons/reload"; - -export default class PluginCard extends React.Component { - - constructor(props) { - super(props); - this.onChange = this.onChange.bind(this); - this.showSettings = this.showSettings.bind(this); - this.state = { - checked: this.props.enabled, - settingsOpen: false - }; - this.hasSettings = typeof this.props.addon.plugin.getSettingsPanel === "function"; - this.settingsPanel = ""; - this.panelRef = React.createRef(); - - this.reload = this.reload.bind(this); - // this.onReload = this.onReload.bind(this); - this.closeSettings = this.closeSettings.bind(this); - } - - reload() { - if (!this.props.reload) return; - this.props.addon = this.props.reload(this.props.addon.id); - this.forceUpdate(); - } - - componentDidUpdate() { - if (!this.state.settingsOpen) return; - if (this.settingsPanel instanceof Node) this.panelRef.current.appendChild(this.settingsPanel); - - // if (!SettingsCookie["fork-ps-3"]) return; - const isHidden = (container, element) => { - const cTop = container.scrollTop; - const cBottom = cTop + container.clientHeight; - const eTop = element.offsetTop; - const eBottom = eTop + element.clientHeight; - return (eTop < cTop || eBottom > cBottom); - }; - - const panel = $(this.panelRef.current); - const container = panel.parents(".scroller-2FKFPG"); - if (!isHidden(container[0], panel[0])) return; - container.animate({ - scrollTop: panel.offset().top - container.offset().top + container.scrollTop() - 30 - }, 300); - } - - getString(value) {return typeof value == "string" ? value : value.toString();} - - closeSettings() { - this.panelRef.current.innerHTML = ""; - this.setState({settingsOpen: false}); - } - - buildTitle(name, version, author) { - const title = Strings.Addons.title.split(/({{[A-Za-z]+}})/); - const nameIndex = title.findIndex(s => s == "{{name}}"); - if (nameIndex) title[nameIndex] = React.createElement("span", {className: "bd-name"}, name); - const versionIndex = title.findIndex(s => s == "{{version}}"); - if (nameIndex) title[versionIndex] = React.createElement("span", {className: "bd-version"}, version); - const authorIndex = title.findIndex(s => s == "{{author}}"); - if (nameIndex) title[authorIndex] = React.createElement("span", {className: "bd-author"}, author); - return title.flat(); - } - - get settingsComponent() { - const addon = this.props.addon; - const name = this.getString(addon.name); - try { this.settingsPanel = addon.plugin.getSettingsPanel(); } - catch (err) { Logger.stacktrace("Plugin Settings", "Unable to get settings panel for " + name + ".", err); } - - const props = {id: `plugin-settings-${name}`, className: "plugin-settings", ref: this.panelRef}; - if (typeof(settingsPanel) == "string") props.dangerouslySetInnerHTML = this.settingsPanel; - - return
  • -
    -
    {this.settingsPanel instanceof React.Component ? this.settingsPanel : null}
    -
  • ; - } - - buildLink(which) { - const url = this.props.addon[which]; - if (!url) return null; - return {Strings.Addons[which]}; - } - - get footer() { - const links = ["website", "source"]; - if (!links.some(l => this.props.addon[l]) && !this.hasSettings) return null; - const linkComponents = links.map(this.buildLink.bind(this)).filter(c => c); - return
    - {linkComponents.map((comp, i) => i < linkComponents.length - 1 ? [comp, " | "] : [comp]).flat()} - {this.hasSettings && } -
    ; - } - - render() { - if (this.state.settingsOpen) return this.settingsComponent; - - const {addon} = this.props; - const name = this.getString(addon.name); - const author = this.getString(addon.author); - const description = this.getString(addon.description); - const version = this.getString(addon.version); - - return
  • -
    - {this.buildTitle(name, version, author)} -
    - {this.props.showReloadIcon && } -
    -
    {description}
    - {this.footer} -
  • ; - } - - onChange() { - this.setState({checked: !this.state.checked}); - this.props.onChange && this.props.onChange(this.props.addon.id); - } - - showSettings() { - if (!this.hasSettings) return; - this.setState({settingsOpen: true}); - } -} \ No newline at end of file diff --git a/src/ui/settings/themecard.js b/src/ui/settings/themecard.js deleted file mode 100644 index c98f906b..00000000 --- a/src/ui/settings/themecard.js +++ /dev/null @@ -1,73 +0,0 @@ -import {React, Strings} from "modules"; -import ReloadIcon from "../icons/reload"; -// import Toasts from "../toasts"; - -export default class ThemeCard extends React.Component { - - constructor(props) { - super(props); - this.state = { - checked: this.props.enabled, - reloads: 0 - }; - this.onChange = this.onChange.bind(this); - this.reload = this.reload.bind(this); - } - - reload() { - if (!this.props.reload) return; - this.props.addon = this.props.reload(this.props.addon.id); - this.forceUpdate(); - } - - buildTitle(name, version, author) { - const title = Strings.Addons.title.split(/({{[A-Za-z]+}})/); - const nameIndex = title.findIndex(s => s == "{{name}}"); - if (nameIndex) title[nameIndex] = React.createElement("span", {className: "bd-name"}, name); - const versionIndex = title.findIndex(s => s == "{{version}}"); - if (nameIndex) title[versionIndex] = React.createElement("span", {className: "bd-version"}, version); - const authorIndex = title.findIndex(s => s == "{{author}}"); - if (nameIndex) title[authorIndex] = React.createElement("span", {className: "bd-author"}, author); - return title.flat(); - } - - render() { - const {addon} = this.props; - const name = addon.name; - const description = addon.description; - const version = addon.version; - const author = addon.author; - const website = addon.website; - const source = addon.source; - - return React.createElement("li", {"data-name": name, "data-version": version, "className": "settings-closed bd-switch-item"}, - React.createElement("div", {className: "bd-header"}, - React.createElement("span", {className: "bd-header-title"}, - this.buildTitle(name, version, author) - ), - React.createElement("div", {className: "bd-controls"}, - this.props.showReloadIcon && React.createElement(ReloadIcon, {className: "bd-reload bd-reload-card", onClick: this.reload}), - React.createElement("label", {className: "bd-switch-wrapper bd-flex-child", style: {flex: "0 0 auto"}}, - React.createElement("input", {checked: this.state.checked, onChange: this.onChange, className: "bd-switch-checkbox", type: "checkbox"}), - React.createElement("div", {className: this.state.checked ? "bd-switch checked" : "bd-switch"}) - ) - ) - ), - React.createElement("div", {className: "bd-description-wrap scroller-wrap fade"}, - React.createElement("div", {className: "bd-description scroller"}, description) - ), - (website || source) && React.createElement("div", {className: "bd-footer"}, - React.createElement("span", {className: "bd-links"}, - website && React.createElement("a", {className: "bd-link", href: website, target: "_blank"}, "Website"), - website && source && " | ", - source && React.createElement("a", {className: "bd-link", href: source, target: "_blank"}, "Source") - ) - ) - ); - } - - onChange() { - this.setState({checked: !this.state.checked}); - this.props.onChange && this.props.onChange(this.props.addon.id); - } -} \ No newline at end of file