2020-10-20 23:25:34 +02:00
/ * *
* @ name BetterFriendList
* @ authorId 278543574059057154
* @ invite Jx3TjNS
2020-11-19 16:45:36 +01:00
* @ donate https : //www.paypal.me/MircoWittrien
* @ patreon https : //www.patreon.com/MircoWittrien
* @ website https : //github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/BetterFriendList
* @ source https : //raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/BetterFriendList/BetterFriendList.plugin.js
* @ updateUrl https : //raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/BetterFriendList/BetterFriendList.plugin.js
2020-10-20 23:25:34 +02:00
* /
2020-08-26 12:10:17 +02:00
2020-09-19 20:49:33 +02:00
module . exports = ( _ => {
2020-10-09 21:09:35 +02:00
const config = {
2020-09-19 20:49:33 +02:00
"info" : {
"name" : "BetterFriendList" ,
"author" : "DevilBro" ,
2020-11-27 17:53:50 +01:00
"version" : "1.3.2" ,
2020-10-16 10:25:30 +02:00
"description" : "Add extra controls to the friends page, like sort by name/status, search and all/request/blocked amount"
2020-11-16 09:51:45 +01:00
} ,
"changeLog" : {
"fixed" : {
2020-11-27 17:53:50 +01:00
"Mutual Guilds" : "Visible again"
2020-11-16 09:51:45 +01:00
}
2020-08-26 20:07:49 +02:00
}
2020-09-19 20:49:33 +02:00
} ;
2020-11-13 19:47:44 +01:00
2020-10-09 21:09:35 +02:00
return ! window . BDFDB _Global || ( ! window . BDFDB _Global . loaded && ! window . BDFDB _Global . started ) ? class {
2021-01-06 12:38:36 +01:00
getName ( ) { return config . info . name ; }
getAuthor ( ) { return config . info . author ; }
getVersion ( ) { return config . info . version ; }
2021-02-01 17:13:13 +01:00
getDescription ( ) { return ` The Library Plugin needed for ${ config . info . name } is missing. Open the Plugin Settings to download it. \n \n ${ config . info . description } ` ; }
downloadLibrary ( ) {
require ( "request" ) . get ( "https://mwittrien.github.io/BetterDiscordAddons/Library/0BDFDB.plugin.js" , ( e , r , b ) => {
2021-02-02 11:22:26 +01:00
if ( ! e && b && b . indexOf ( ` * @name BDFDB ` ) > - 1 ) require ( "fs" ) . writeFile ( require ( "path" ) . join ( BdApi . Plugins . folder , "0BDFDB.plugin.js" ) , b , _ => BdApi . showToast ( "Finished downloading BDFDB Library" , { type : "success" } ) ) ;
2021-02-01 17:13:13 +01:00
else BdApi . alert ( "Error" , "Could not download BDFDB Library Plugin, try again later or download it manually from GitHub: https://github.com/mwittrien/BetterDiscordAddons/tree/master/Library/" ) ;
} ) ;
}
2020-08-26 20:07:49 +02:00
2021-01-06 12:38:36 +01:00
load ( ) {
2020-11-19 16:51:14 +01:00
if ( ! window . BDFDB _Global || ! Array . isArray ( window . BDFDB _Global . pluginQueue ) ) window . BDFDB _Global = Object . assign ( { } , window . BDFDB _Global , { pluginQueue : [ ] } ) ;
2020-09-19 20:49:33 +02:00
if ( ! window . BDFDB _Global . downloadModal ) {
window . BDFDB _Global . downloadModal = true ;
2021-01-14 16:14:44 +01:00
BdApi . showConfirmationModal ( "Library Missing" , ` The Library Plugin needed for ${ config . info . name } is missing. Please click "Download Now" to install it. ` , {
2020-09-19 20:49:33 +02:00
confirmText : "Download Now" ,
cancelText : "Cancel" ,
onCancel : _ => { delete window . BDFDB _Global . downloadModal ; } ,
2020-09-20 08:15:13 +02:00
onConfirm : _ => {
delete window . BDFDB _Global . downloadModal ;
2021-02-01 17:13:13 +01:00
this . downloadLibrary ( ) ;
2020-09-20 08:15:13 +02:00
}
2020-09-19 20:49:33 +02:00
} ) ;
2020-08-26 12:10:17 +02:00
}
2020-09-19 20:49:33 +02:00
if ( ! window . BDFDB _Global . pluginQueue . includes ( config . info . name ) ) window . BDFDB _Global . pluginQueue . push ( config . info . name ) ;
2020-10-09 21:09:35 +02:00
}
2021-01-06 12:38:36 +01:00
start ( ) { this . load ( ) ; }
stop ( ) { }
getSettingsPanel ( ) {
2020-11-28 23:12:09 +01:00
let template = document . createElement ( "template" ) ;
2021-01-14 16:14:44 +01:00
template . innerHTML = ` <div style="color: var(--header-primary); font-size: 16px; font-weight: 300; white-space: pre; line-height: 22px;">The Library Plugin needed for ${ config . info . name } is missing. \n Please click <a style="font-weight: 500;">Download Now</a> to install it.</div> ` ;
2021-02-01 17:13:13 +01:00
template . content . firstElementChild . querySelector ( "a" ) . addEventListener ( "click" , this . downloadLibrary ) ;
2020-11-28 23:12:09 +01:00
return template . content . firstElementChild ;
}
2020-10-09 21:09:35 +02:00
} : ( ( [ Plugin , BDFDB ] ) => {
2020-09-19 20:49:33 +02:00
var rerenderTimeout , sortKey , sortReversed , searchQuery , searchTimeout ;
var settings = { } ;
const placeHolderId = "PLACEHOLDER_BETTERFRIENDLIST" ;
const statusSortOrder = {
online : 0 ,
streaming : 1 ,
idle : 2 ,
dnd : 3 ,
offline : 4 ,
invisible : 5 ,
unknown : 6
} ;
2020-10-09 21:09:35 +02:00
return class BetterFriendList extends Plugin {
2021-01-06 12:38:36 +01:00
onLoad ( ) {
2020-09-19 20:49:33 +02:00
this . defaults = {
settings : {
2020-11-19 16:51:14 +01:00
addTotalAmount : { value : true , description : "Add total amount for all/requested/blocked" } ,
addSortOptions : { value : true , description : "Add sort options" } ,
addSearchbar : { value : true , description : "Add searchbar" } ,
addMutualGuild : { value : true , description : "Add mutual servers in friend list" }
2020-09-19 20:49:33 +02:00
}
} ;
2020-08-26 12:10:17 +02:00
2020-09-19 20:49:33 +02:00
this . patchedModules = {
before : {
PeopleListSectionedLazy : "default" ,
2020-10-25 18:15:18 +01:00
PeopleListSectionedNonLazy : "default"
2020-09-19 20:49:33 +02:00
} ,
after : {
TabBar : "render" ,
PeopleListSectionedLazy : "default" ,
2020-10-25 18:15:18 +01:00
PeopleListSectionedNonLazy : "default" ,
2020-09-19 20:49:33 +02:00
FriendRow : "render" ,
PendingRow : "default" ,
BlockedRow : "render" ,
PeopleListItem : [ "render" , "componentDidMount" , "componentWillUnmount" ]
}
} ;
this . css = `
$ { BDFDB . dotCN . peoplewrapper } > * {
justify - content : unset ;
}
$ { BDFDB . dotCNS . peoplewrapper + BDFDB . dotCN . userinfo } {
flex : 1 1 auto ;
}
$ { BDFDB . dotCN . _betterfriendlistmutualguilds } {
margin - left : 13 px ;
width : 200 px ;
}
$ { BDFDB . dotCN . _betterfriendlisttitle } {
width : 200 px ;
}
$ { BDFDB . dotCN . _betterfriendlistnamecell } {
width : 150 px ;
}
` ;
}
2021-01-06 12:38:36 +01:00
onStart ( ) {
2020-09-19 20:49:33 +02:00
sortKey = null ;
sortReversed = false ;
searchQuery = "" ;
2020-08-26 12:10:17 +02:00
2020-08-26 15:21:49 +02:00
this . forceUpdateAll ( ) ;
2020-08-26 12:10:17 +02:00
}
2020-09-19 20:49:33 +02:00
2021-01-06 12:38:36 +01:00
onStop ( ) {
2020-08-26 15:21:49 +02:00
this . forceUpdateAll ( ) ;
2020-08-26 12:10:17 +02:00
}
2020-09-19 20:49:33 +02:00
getSettingsPanel ( collapseStates = { } ) {
let settingsPanel , settingsItems = [ ] ;
for ( let key in settings ) settingsItems . push ( BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . SettingsSaveItem , {
type : "Switch" ,
plugin : this ,
keys : [ "settings" , key ] ,
label : this . defaults . settings [ key ] . description ,
value : settings [ key ]
} ) ) ;
return settingsPanel = BDFDB . PluginUtils . createSettingsPanel ( this , settingsItems ) ;
}
2020-08-26 12:10:17 +02:00
2021-01-06 12:38:36 +01:00
onSettingsClosed ( ) {
2020-09-19 20:49:33 +02:00
if ( this . SettingsUpdated ) {
delete this . SettingsUpdated ;
this . forceUpdateAll ( ) ;
}
}
2021-01-06 12:38:36 +01:00
forceUpdateAll ( ) {
2020-09-19 20:49:33 +02:00
settings = BDFDB . DataUtils . get ( this , "settings" ) ;
2020-08-26 12:10:17 +02:00
2020-09-19 20:49:33 +02:00
BDFDB . PatchUtils . forceAllUpdates ( this ) ;
this . rerenderList ( ) ;
2020-08-26 15:21:49 +02:00
}
2020-09-19 20:49:33 +02:00
processTabBar ( e ) {
if ( settings . addTotalAmount && e . returnvalue . props . children ) for ( let checkChild of e . returnvalue . props . children ) if ( checkChild && checkChild . props . id == "ADD_FRIEND" ) {
let relationships = BDFDB . LibraryModules . FriendUtils . getRelationships ( ) , relationshipCount = { } ;
for ( let type in BDFDB . DiscordConstants . RelationshipTypes ) relationshipCount [ type ] = 0 ;
for ( let id in relationships ) relationshipCount [ relationships [ id ] ] ++ ;
for ( let child of e . returnvalue . props . children ) if ( child && child . props . id != "ADD_FRIEND" ) {
let newChildren = [ child . props . children ] . flat ( ) . filter ( child => BDFDB . ObjectUtils . get ( child , "type.displayName" ) != "NumberBadge" ) ;
switch ( child . props . id ) {
case "ALL" :
newChildren . push ( this . createBadge ( relationshipCount [ BDFDB . DiscordConstants . RelationshipTypes . FRIEND ] ) ) ;
break ;
case "ONLINE" :
newChildren . push ( this . createBadge ( BDFDB . LibraryModules . StatusMetaUtils . getOnlineFriendCount ( ) ) ) ;
break ;
case "PENDING" :
newChildren . push ( this . createBadge ( relationshipCount [ BDFDB . DiscordConstants . RelationshipTypes . PENDING _INCOMING ] ) ) ;
newChildren . push ( this . createBadge ( relationshipCount [ BDFDB . DiscordConstants . RelationshipTypes . PENDING _OUTGOING ] ) ) ;
break ;
case "BLOCKED" :
newChildren . push ( this . createBadge ( relationshipCount [ BDFDB . DiscordConstants . RelationshipTypes . BLOCKED ] ) ) ;
break ;
}
child . props . children = newChildren ;
2020-08-26 12:10:17 +02:00
}
2020-09-19 20:49:33 +02:00
break ;
2020-08-26 12:10:17 +02:00
}
}
2020-09-19 20:49:33 +02:00
processPeopleListSectionedLazy ( e ) {
if ( sortKey || searchQuery ) {
e . instance . props . statusSections = [ ] . concat ( e . instance . props . statusSections ) . map ( section => {
let newSection = [ ] . concat ( section ) ;
if ( searchQuery ) {
let usedSearchQuery = searchQuery . toLowerCase ( ) ;
newSection = newSection . filter ( user => user && typeof user . usernameLower == "string" && user . usernameLower . indexOf ( usedSearchQuery ) > - 1 ) ;
}
if ( sortKey ) {
newSection = BDFDB . ArrayUtils . keySort ( newSection . map ( user => Object . assign ( { } , user , { statusIndex : statusSortOrder [ user . status ] } ) ) , sortKey ) ;
if ( sortReversed ) newSection . reverse ( ) ;
}
if ( ! newSection . length ) {
let placeholder = new BDFDB . DiscordObjects . User ( {
id : placeHolderId ,
username : placeHolderId
} ) ;
if ( placeholder ) newSection . push ( new BDFDB . DiscordObjects . Relationship ( {
activities : [ ] ,
applicationStream : null ,
isMobile : false ,
key : placeHolderId ,
mutualGuilds : [ ] ,
mutualGuildsLength : 0 ,
status : "offline" ,
type : BDFDB . DiscordConstants . RelationshipTypes . NONE ,
user : placeholder ,
usernameLower : placeholder . usernameNormalized
} ) ) ;
}
return newSection ;
} ) ;
}
2020-11-16 09:51:45 +01:00
if ( ! BDFDB . PatchUtils . isPatched ( this , e . instance . props , "getSectionTitle" ) ) BDFDB . PatchUtils . patch ( this , e . instance . props , "getSectionTitle" , { after : e2 => {
if ( typeof e2 . returnValue == "string" ) {
let users = e . instance . props . statusSections . flat ( 10 ) ;
return BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . Flex , {
align : BDFDB . LibraryComponents . Flex . Align . CENTER ,
children : [
BDFDB . ReactUtils . createElement ( "div" , {
className : BDFDB . disCN . _betterfriendlisttitle ,
children : e2 . returnValue . replace ( users . length , users . filter ( u => u && u . key != placeHolderId ) . length )
2020-09-19 20:49:33 +02:00
} ) ,
2020-11-16 09:51:45 +01:00
settings . addSortOptions && [
{ key : "usernameLower" , label : BDFDB . LanguageUtils . LanguageStrings . FRIENDS _COLUMN _NAME } ,
{ key : "statusIndex" , label : BDFDB . LanguageUtils . LanguageStrings . FRIENDS _COLUMN _STATUS }
] . filter ( n => n ) . map ( data => BDFDB . ReactUtils . createElement ( "div" , {
className : BDFDB . DOMUtils . formatClassName ( BDFDB . disCN . tableheadercellwrapper , BDFDB . disCN . tableheadercell , BDFDB . disCN . _betterfriendlistnamecell , sortKey == data . key && BDFDB . disCN . tableheadercellsorted , BDFDB . disCN . tableheadercellclickable ) ,
children : BDFDB . ReactUtils . createElement ( "div" , {
className : BDFDB . disCN . tableheadercellcontent ,
children : [
data . label ,
sortKey == data . key && BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . SvgIcon , {
className : BDFDB . disCN . tableheadersorticon ,
name : BDFDB . LibraryComponents . SvgIcon . Names [ sortReversed ? "ARROW_UP" : "ARROW_DOWN" ]
} )
] . filter ( n => n )
} ) ,
onClick : event => {
if ( sortKey == data . key ) {
if ( ! sortReversed ) sortReversed = true ;
else {
sortKey = null ;
sortReversed = false ;
}
}
2020-09-19 20:49:33 +02:00
else {
2020-11-16 09:51:45 +01:00
sortKey = data . key ;
2020-09-19 20:49:33 +02:00
sortReversed = false ;
}
this . rerenderList ( ) ;
}
2020-11-16 09:51:45 +01:00
} ) ) ,
settings . addSearchbar && BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . Flex . Child , {
children : BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . SearchBar , {
query : searchQuery ,
onChange : value => {
BDFDB . TimeUtils . clear ( searchTimeout ) ;
searchTimeout = BDFDB . TimeUtils . timeout ( _ => {
searchQuery = value ;
this . rerenderList ( ) ;
} , 1000 ) ;
} ,
onClear : _ => {
searchQuery = "" ;
this . rerenderList ( ) ;
}
} )
2020-09-19 20:49:33 +02:00
} )
2020-11-16 09:51:45 +01:00
] . flat ( 10 ) . filter ( n => n )
} ) ;
}
} } , { force : true , noCache : true } ) ;
2020-09-19 20:49:33 +02:00
}
2020-10-25 18:15:18 +01:00
processPeopleListSectionedNonLazy ( e ) {
this . processPeopleListSectionedLazy ( e ) ;
}
2020-09-19 20:49:33 +02:00
processFriendRow ( e ) {
e . returnvalue . props . mutualGuilds = e . instance . props . mutualGuilds ;
}
processPendingRow ( e ) {
this . processFriendRow ( e ) ;
}
processBlockedRow ( e ) {
this . processFriendRow ( e ) ;
2020-08-29 00:48:29 +02:00
}
2020-09-19 20:49:33 +02:00
processPeopleListItem ( e ) {
if ( e . node ) {
BDFDB . TimeUtils . clear ( rerenderTimeout ) ;
rerenderTimeout = BDFDB . TimeUtils . timeout ( _ => { BDFDB . PatchUtils . forceAllUpdates ( this , "TabBar" ) ; } , 1000 ) ;
}
else {
if ( e . instance . props . user . id == placeHolderId ) return null ;
else if ( settings . addMutualGuild && e . instance . props . mutualGuilds && e . instance . props . mutualGuilds . length ) {
2020-11-27 17:53:50 +01:00
if ( typeof e . returnvalue . props . children == "function" ) {
let childrenRender = e . returnvalue . props . children ;
e . returnvalue . props . children = ( ... args ) => {
let children = childrenRender ( ... args ) ;
this . injectMutualGuilds ( children , e . instance . props . mutualGuilds ) ;
return children ;
} ;
}
else this . injectMutualGuilds ( e . returnvalue , e . instance . props . mutualGuilds ) ;
2020-09-19 20:49:33 +02:00
}
2020-08-29 08:55:29 +02:00
}
2020-08-29 00:48:29 +02:00
}
2020-09-19 20:49:33 +02:00
2020-11-27 17:53:50 +01:00
injectMutualGuilds ( returnvalue , mutualGuilds ) {
let [ children , index ] = BDFDB . ReactUtils . findParent ( returnvalue , { name : "UserInfo" } ) ;
if ( index > - 1 ) children . splice ( index + 1 , 0 , BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . GuildSummaryItem , {
className : BDFDB . disCN . _betterfriendlistmutualguilds ,
guilds : mutualGuilds ,
showTooltip : true ,
max : 10
} , true ) ) ;
}
2020-09-19 20:49:33 +02:00
createBadge ( amount ) {
return BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . Badges . NumberBadge , {
className : BDFDB . disCN . peoplesbadge ,
count : amount ,
style : { marginLeft : 6 }
} ) ;
}
2021-01-06 12:38:36 +01:00
rerenderList ( ) {
2020-09-19 20:49:33 +02:00
let selectedButton = document . querySelector ( BDFDB . dotCNS . peoplestabbar + BDFDB . dotCN . settingsitemselected ) ;
if ( selectedButton ) selectedButton . click ( ) ;
}
} ;
2020-10-09 21:09:35 +02:00
} ) ( window . BDFDB _Global . PluginUtils . buildPlugin ( config ) ) ;
2020-09-19 20:49:33 +02:00
} ) ( ) ;