2023-01-07 21:52:55 +00:00
/ *
* Vencord , a modification for Discord ' s desktop app
* Copyright ( c ) 2023 Vendicated and contributors
*
* This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation , either version 3 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < https : //www.gnu.org/licenses/>.
* /
2024-02-06 15:29:47 +00:00
import { addChatBarButton , ChatBarButton } from "@api/ChatButtons" ;
2023-01-07 21:52:55 +00:00
import { addButton , removeButton } from "@api/MessagePopover" ;
2023-05-05 23:36:00 +00:00
import { definePluginSettings } from "@api/Settings" ;
2023-01-07 21:52:55 +00:00
import ErrorBoundary from "@components/ErrorBoundary" ;
import { Devs } from "@utils/constants" ;
import { getStegCloak } from "@utils/dependencies" ;
2023-02-12 21:10:03 +00:00
import definePlugin , { OptionType } from "@utils/types" ;
2024-02-06 15:29:47 +00:00
import { ChannelStore , FluxDispatcher , RestAPI , Tooltip } from "@webpack/common" ;
2023-02-12 21:10:03 +00:00
import { Message } from "discord-types/general" ;
2023-01-07 21:52:55 +00:00
import { buildDecModal } from "./components/DecryptionModal" ;
import { buildEncModal } from "./components/EncryptionModal" ;
let steggo : any ;
function PopOverIcon() {
return (
< svg
fill = "var(--header-secondary)"
width = { 24 } height = { 24 }
viewBox = { "0 0 64 64" }
>
< path d = "M 32 9 C 24.832 9 19 14.832 19 22 L 19 27.347656 C 16.670659 28.171862 15 30.388126 15 33 L 15 49 C 15 52.314 17.686 55 21 55 L 43 55 C 46.314 55 49 52.314 49 49 L 49 33 C 49 30.388126 47.329341 28.171862 45 27.347656 L 45 22 C 45 14.832 39.168 9 32 9 z M 32 13 C 36.963 13 41 17.038 41 22 L 41 27 L 23 27 L 23 22 C 23 17.038 27.037 13 32 13 z" / >
< / svg >
) ;
}
function Indicator() {
return (
< Tooltip text = "This message has a hidden message! (InvisibleChat)" >
{ ( { onMouseEnter , onMouseLeave } ) = > (
< img
aria - label = "Hidden Message Indicator (InvisibleChat)"
onMouseEnter = { onMouseEnter }
onMouseLeave = { onMouseLeave }
src = "https://github.com/SammCheese/invisible-chat/raw/NewReplugged/src/assets/lock.png"
width = { 20 }
height = { 20 }
style = { { transform : "translateY(4p)" , paddingInline : 4 } }
/ >
) }
< / Tooltip >
) ;
}
2024-02-06 17:12:09 +00:00
const ChatBarIcon : ChatBarButton = ( { isMainChat } ) = > {
2024-02-06 15:29:47 +00:00
if ( ! isMainChat ) return null ;
2023-05-11 17:44:33 +00:00
2023-01-07 21:52:55 +00:00
return (
2024-02-06 15:29:47 +00:00
< ChatBarButton
tooltip = "Encrypt Message"
onClick = { ( ) = > buildEncModal ( ) }
buttonProps = { {
"aria-haspopup" : "dialog" ,
} }
>
< svg
aria - hidden
role = "img"
2024-02-06 16:50:45 +00:00
width = "24"
height = "24"
2024-02-06 15:29:47 +00:00
viewBox = { "0 0 64 64" }
2024-02-06 16:50:45 +00:00
style = { { scale : "1.39" , translate : "0 -1px" } }
2024-02-06 15:29:47 +00:00
>
< path fill = "currentColor" d = "M 32 9 C 24.832 9 19 14.832 19 22 L 19 27.347656 C 16.670659 28.171862 15 30.388126 15 33 L 15 49 C 15 52.314 17.686 55 21 55 L 43 55 C 46.314 55 49 52.314 49 49 L 49 33 C 49 30.388126 47.329341 28.171862 45 27.347656 L 45 22 C 45 14.832 39.168 9 32 9 z M 32 13 C 36.963 13 41 17.038 41 22 L 41 27 L 23 27 L 23 22 C 23 17.038 27.037 13 32 13 z" / >
< / svg >
< / ChatBarButton >
2023-01-07 21:52:55 +00:00
) ;
2024-02-06 15:29:47 +00:00
} ;
2023-01-07 21:52:55 +00:00
2023-02-12 21:10:03 +00:00
const settings = definePluginSettings ( {
savedPasswords : {
type : OptionType . STRING ,
default : "password, Password" ,
description : "Saved Passwords (Seperated with a , )"
}
} ) ;
2023-01-07 21:52:55 +00:00
export default definePlugin ( {
name : "InvisibleChat" ,
2023-05-17 02:38:01 +00:00
description : "Encrypt your Messages in a non-suspicious way!" ,
2023-01-07 21:52:55 +00:00
authors : [ Devs . SammCheese ] ,
2024-02-06 15:29:47 +00:00
dependencies : [ "MessagePopoverAPI" , "ChatInputButtonAPI" ] ,
2023-01-07 21:52:55 +00:00
patches : [
{
// Indicator
find : ".Messages.MESSAGE_EDITED," ,
replacement : {
2023-10-25 18:24:17 +00:00
match : /let\{className:\i,message:\i[^}]*\}=(\i)/ ,
2023-06-27 20:41:33 +00:00
replace : "try {$1 && $self.INV_REGEX.test($1.message.content) ? $1.content.push($self.indicator()) : null } catch {};$&"
2023-01-07 21:52:55 +00:00
}
} ,
] ,
EMBED_API_URL : "https://embed.sammcheese.net" ,
INV_REGEX : new RegExp ( /( \u200c|\u200d |[\u2060-\u2064])[^\u200b]/ ) ,
URL_REGEX : new RegExp (
/(http(s)?:\/\/.)?(www\.)?[-a-zA-Z0-9@:%._+~#=]{2,256}\.[a-z]{2,6}\b([-a-zA-Z0-9@:%_+.~#?&//=]*)/ ,
) ,
2023-02-12 21:10:03 +00:00
settings ,
2023-01-07 21:52:55 +00:00
async start() {
2024-02-06 15:29:47 +00:00
addButton ( "InvisibleChat" , message = > {
2023-01-07 21:52:55 +00:00
return this . INV_REGEX . test ( message ? . content )
? {
label : "Decrypt Message" ,
icon : this.popOverIcon ,
message : message ,
channel : ChannelStore.getChannel ( message . channel_id ) ,
2023-02-12 21:10:03 +00:00
onClick : async ( ) = > {
await iteratePasswords ( message ) . then ( ( res : string | false ) = > {
if ( res ) return void this . buildEmbed ( message , res ) ;
return void buildDecModal ( { message } ) ;
} ) ;
}
2023-01-07 21:52:55 +00:00
}
: null ;
} ) ;
2024-02-06 15:29:47 +00:00
addChatBarButton ( "InvisibleChat" , ChatBarIcon ) ;
2024-02-06 15:50:21 +00:00
const { default : StegCloak } = await getStegCloak ( ) ;
steggo = new StegCloak ( true , false ) ;
2023-01-07 21:52:55 +00:00
} ,
stop() {
2024-02-06 15:29:47 +00:00
removeButton ( "InvisibleChat" ) ;
removeButton ( "InvisibleChat" ) ;
2023-01-07 21:52:55 +00:00
} ,
// Gets the Embed of a Link
async getEmbed ( url : URL ) : Promise < Object | {} > {
2023-05-17 02:38:01 +00:00
const { body } = await RestAPI . post ( {
url : "/unfurler/embed-urls" ,
body : {
urls : [ url ]
}
} ) ;
return await body . embeds [ 0 ] ;
2023-01-07 21:52:55 +00:00
} ,
async buildEmbed ( message : any , revealed : string ) : Promise < void > {
const urlCheck = revealed . match ( this . URL_REGEX ) ;
message . embeds . push ( {
type : "rich" ,
title : "Decrypted Message" ,
color : "0x45f5f5" ,
description : revealed ,
footer : {
text : "Made with ❤️ by c0dine and Sammy!" ,
} ,
} ) ;
2023-06-27 20:41:33 +00:00
if ( urlCheck ? . length ) {
const embed = await this . getEmbed ( new URL ( urlCheck [ 0 ] ) ) ;
if ( embed )
message . embeds . push ( embed ) ;
}
2023-01-07 21:52:55 +00:00
this . updateMessage ( message ) ;
} ,
updateMessage : ( message : any ) = > {
FluxDispatcher . dispatch ( {
type : "MESSAGE_UPDATE" ,
message ,
} ) ;
} ,
popOverIcon : ( ) = > < PopOverIcon / > ,
indicator : ErrorBoundary.wrap ( Indicator , { noop : true } )
} ) ;
export function encrypt ( secret : string , password : string , cover : string ) : string {
return steggo . hide ( secret + "\u200b" , password , cover ) ;
}
2023-09-12 21:04:50 +00:00
export function decrypt ( encrypted : string , password : string , removeIndicator : boolean ) : string {
const decrypted = steggo . reveal ( encrypted , password ) ;
2023-02-12 21:10:03 +00:00
return removeIndicator ? decrypted . replace ( "\u200b" , "" ) : decrypted ;
2023-01-07 21:52:55 +00:00
}
2023-02-12 21:10:03 +00:00
export function isCorrectPassword ( result : string ) : boolean {
return result . endsWith ( "\u200b" ) ;
}
export async function iteratePasswords ( message : Message ) : Promise < string | false > {
const passwords = settings . store . savedPasswords . split ( "," ) . map ( s = > s . trim ( ) ) ;
if ( ! message ? . content || ! passwords ? . length ) return false ;
let { content } = message ;
// we use an extra variable so we dont have to edit the message content directly
if ( /^\W/ . test ( message . content ) ) content = ` d ${ message . content } d ` ;
for ( let i = 0 ; i < passwords . length ; i ++ ) {
const result = decrypt ( content , passwords [ i ] , false ) ;
if ( isCorrectPassword ( result ) ) {
return result ;
}
}
return false ;
}