2020-02-27 08:44:03 +01:00
//META{"name":"SpellCheck","authorId":"278543574059057154","invite":"Jx3TjNS","donate":"https://www.paypal.me/MircoWittrien","patreon":"https://www.patreon.com/MircoWittrien","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/SpellCheck","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/SpellCheck/SpellCheck.plugin.js"}*//
2018-10-11 10:21:26 +02:00
2020-02-20 17:23:49 +01:00
var SpellCheck = ( _ => {
2020-06-09 19:42:00 +02:00
var languages , dictionaries , langDictionaries , languageToasts , checkTimeout , currentText ;
2020-06-09 09:30:21 +02:00
var settings = { } , choices = { } , amounts = { } ;
2020-02-20 17:23:49 +01:00
return class SpellCheck {
getName ( ) { return "SpellCheck" ; }
2019-01-17 23:48:29 +01:00
2020-06-22 18:22:07 +02:00
getVersion ( ) { return "1.5.3" ; }
2019-01-17 23:48:29 +01:00
2020-02-20 17:23:49 +01:00
getAuthor ( ) { return "DevilBro" ; }
2019-01-17 23:48:29 +01:00
2020-02-20 17:23:49 +01:00
getDescription ( ) { return "Adds a spellcheck to all textareas. Select a word and rightclick it to add it to your dictionary." ; }
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
constructor ( ) {
2020-05-19 19:24:52 +02:00
this . changelog = {
2020-06-22 18:22:07 +02:00
"improved" : [ [ "Special Character / Symbols" , "If half or more of the characters in a word are symbols, the word will automatically be ignored by the dictionary check to avoid stuff like 'v1.2.3' being marked as incorrect" ] ]
2020-05-19 19:24:52 +02:00
} ;
2020-02-20 17:23:49 +01:00
this . patchedModules = {
after : {
SlateChannelTextArea : [ "componentDidMount" , "componentDidUpdate" ]
}
} ;
}
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
initConstructor ( ) {
languages = { } ;
2020-05-23 10:59:15 +02:00
dictionaries = { } ;
langDictionaries = { } ;
languageToasts = { } ;
2020-02-20 17:23:49 +01:00
this . css = `
$ { BDFDB . dotCNS . _spellcheckoverlay + BDFDB . dotCN . _spellcheckerror } {
background - image : url ( '' ) ;
background - repeat : repeat - x ;
background - position : bottom ;
} ` ;
this . defaults = {
2020-05-25 21:03:56 +02:00
settings : {
downloadDictionary : { value : false , description : "Use local dictionary file (downloads dictionary on first useage)" }
} ,
2020-02-20 17:23:49 +01:00
choices : {
2020-05-23 10:59:15 +02:00
dictionaryLanguage : { value : "en" , force : true , description : "Primary Language:" } ,
secondaryLanguage : { value : "-" , force : false , description : "Secondary Language:" }
2020-02-20 17:23:49 +01:00
} ,
amounts : {
2020-05-23 10:59:15 +02:00
maxSimilarAmount : { value : 6 , min : 1 , max : 30 , description : "Maximal Amount of suggested Words:" }
2020-02-20 17:23:49 +01:00
}
} ;
}
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
getSettingsPanel ( ) {
if ( ! window . BDFDB || typeof BDFDB != "object" || ! BDFDB . loaded || ! this . started ) return ;
2020-05-25 21:03:56 +02:00
let settings = BDFDB . DataUtils . get ( this , "settings" ) ;
2020-02-20 17:23:49 +01:00
let choices = BDFDB . DataUtils . get ( this , "choices" ) ;
let amounts = BDFDB . DataUtils . get ( this , "amounts" ) ;
let ownDictionary = BDFDB . DataUtils . load ( this , "owndics" , choices . dictionaryLanguage ) || [ ] ;
2020-03-28 07:55:39 +01:00
let settingsPanel , settingsItems = [ ] ;
2020-02-20 17:23:49 +01:00
2020-05-25 21:03:56 +02:00
for ( let key in settings ) settingsItems . push ( BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . SettingsSaveItem , {
className : BDFDB . disCN . marginbottom8 ,
type : "Switch" ,
plugin : this ,
keys : [ "settings" , key ] ,
label : this . defaults . settings [ key ] . description ,
value : settings [ key ]
} ) ) ;
2020-03-28 07:55:39 +01:00
for ( let key in choices ) settingsItems . push ( BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . SettingsSaveItem , {
2020-02-20 17:23:49 +01:00
className : BDFDB . disCN . marginbottom8 ,
type : "Select" ,
plugin : this ,
keys : [ "choices" , key ] ,
label : this . defaults . choices [ key ] . description ,
basis : "70%" ,
value : choices [ key ] ,
2020-05-23 10:59:15 +02:00
options : ( this . defaults . choices [ key ] . force ? [ ] : [ { value : "-" , label : BDFDB . LanguageUtils . LanguageStrings . FORM _LABEL _NOTHING } ] ) . concat ( BDFDB . ObjectUtils . toArray ( BDFDB . ObjectUtils . map ( languages , ( lang , id ) => ( { value : id , label : this . getLanguageName ( lang ) } ) ) ) ) ,
2020-02-20 17:23:49 +01:00
searchable : true ,
onChange : value => {
2020-05-23 10:59:15 +02:00
this . setDictionary ( key , value ) ;
2020-03-28 07:55:39 +01:00
BDFDB . PluginUtils . refreshSettingsPanel ( this , settingsPanel ) ;
2020-01-21 12:24:42 +01:00
}
2020-02-20 17:23:49 +01:00
} ) ) ;
2020-03-28 07:55:39 +01:00
for ( let key in amounts ) settingsItems . push ( BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . SettingsSaveItem , {
2020-02-20 17:23:49 +01:00
className : BDFDB . disCN . marginbottom8 ,
type : "TextInput" ,
childProps : {
type : "number"
} ,
plugin : this ,
keys : [ "amounts" , key ] ,
label : this . defaults . amounts [ key ] . description ,
basis : "20%" ,
min : this . defaults . amounts [ key ] . min ,
max : this . defaults . amounts [ key ] . max ,
value : amounts [ key ]
} ) ) ;
2020-03-28 07:55:39 +01:00
if ( ownDictionary . length ) settingsItems . push ( BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . SettingsPanelInner , {
2020-02-20 17:23:49 +01:00
title : "Your own Dictionary:" ,
2020-03-28 07:55:39 +01:00
first : settingsItems . length == 0 ,
2020-02-20 17:23:49 +01:00
last : true ,
children : ownDictionary . map ( word => BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . Card , {
children : word . toLowerCase ( ) ,
onRemove : _ => {
BDFDB . ArrayUtils . remove ( ownDictionary , word ) ;
BDFDB . DataUtils . save ( ownDictionary , this , "owndics" , choices . dictionaryLanguage ) ;
2020-05-25 17:03:57 +02:00
dictionaries . dictionaryLanguage = this . formatDictionary ( langDictionaries . dictionaryLanguage . concat ( ownDictionary ) ) ;
2020-03-28 07:55:39 +01:00
BDFDB . PluginUtils . refreshSettingsPanel ( this , settingsPanel ) ;
2020-02-20 17:23:49 +01:00
}
} ) )
} ) ) ;
2020-03-28 07:55:39 +01:00
return settingsPanel = BDFDB . PluginUtils . createSettingsPanel ( this , settingsItems ) ;
2020-02-20 17:23:49 +01:00
}
2018-10-11 10:21:26 +02:00
2020-04-11 19:32:58 +02:00
// Legacy
2020-02-20 17:23:49 +01:00
load ( ) { }
start ( ) {
if ( ! window . BDFDB ) window . BDFDB = { myPlugins : { } } ;
if ( window . BDFDB && window . BDFDB . myPlugins && typeof window . BDFDB . myPlugins == "object" ) window . BDFDB . myPlugins [ this . getName ( ) ] = this ;
let libraryScript = document . querySelector ( "head script#BDFDBLibraryScript" ) ;
if ( ! libraryScript || ( performance . now ( ) - libraryScript . getAttribute ( "date" ) ) > 600000 ) {
if ( libraryScript ) libraryScript . remove ( ) ;
libraryScript = document . createElement ( "script" ) ;
libraryScript . setAttribute ( "id" , "BDFDBLibraryScript" ) ;
libraryScript . setAttribute ( "type" , "text/javascript" ) ;
libraryScript . setAttribute ( "src" , "https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.min.js" ) ;
libraryScript . setAttribute ( "date" , performance . now ( ) ) ;
libraryScript . addEventListener ( "load" , _ => { this . initialize ( ) ; } ) ;
document . head . appendChild ( libraryScript ) ;
}
else if ( window . BDFDB && typeof BDFDB === "object" && BDFDB . loaded ) this . initialize ( ) ;
this . startTimeout = setTimeout ( _ => {
try { return this . initialize ( ) ; }
catch ( err ) { console . error ( ` %c[ ${ this . getName ( ) } ]%c ` , "color: #3a71c1; font-weight: 700;" , "" , "Fatal Error: Could not initiate plugin! " + err ) ; }
} , 30000 ) ;
2018-10-11 10:21:26 +02:00
}
2020-02-20 17:23:49 +01:00
initialize ( ) {
if ( window . BDFDB && typeof BDFDB === "object" && BDFDB . loaded ) {
if ( this . started ) return ;
BDFDB . PluginUtils . init ( this ) ;
2020-06-05 20:03:21 +02:00
BDFDB . LibraryRequires . request ( "https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/SpellCheck/dic" , ( error , response , body ) => {
let dictionaryLanguageIds = Array . from ( BDFDB . DOMUtils . create ( body ) . querySelectorAll ( ` [href*="/mwittrien/BetterDiscordAddons/blob/master/Plugins/SpellCheck/dic/"] ` ) ) . map ( n => n . innerText . split ( "." ) [ 0 ] ) . filter ( n => n ) ;
languages = BDFDB . ObjectUtils . filter ( BDFDB . LanguageUtils . languages , langId => dictionaryLanguageIds . includes ( langId ) , true ) ;
2020-06-09 09:30:21 +02:00
if ( ( BDFDB . LibraryModules . StoreChangeUtils && BDFDB . LibraryModules . StoreChangeUtils . get ( "SpellcheckStore" ) || { } ) . enabled ) BDFDB . LibraryModules . SpellCheckUtils . toggleSpellcheck ( ) ;
this . forceUpdateAll ( ) ;
2020-06-05 20:03:21 +02:00
for ( let key in choices ) {
2020-06-16 09:20:23 +02:00
if ( key == "dictionaryLanguage" && ! languages [ choices [ key ] ] ) {
2020-06-05 20:03:21 +02:00
choices [ key ] = "en" ;
2020-06-09 09:30:21 +02:00
BDFDB . DataUtils . save ( choices [ key ] , this , "choices" , key ) ;
2020-06-05 20:03:21 +02:00
}
this . setDictionary ( key , choices [ key ] ) ;
}
} ) ;
2020-02-20 17:23:49 +01:00
}
else console . error ( ` %c[ ${ this . getName ( ) } ]%c ` , "color: #3a71c1; font-weight: 700;" , "" , "Fatal Error: Could not load BD functions!" ) ;
2018-10-11 10:21:26 +02:00
}
2020-02-20 17:23:49 +01:00
stop ( ) {
if ( window . BDFDB && typeof BDFDB === "object" && BDFDB . loaded ) {
this . stopping = true ;
BDFDB . DOMUtils . remove ( BDFDB . dotCN . _spellcheckoverlay ) ;
2019-01-26 22:45:19 +01:00
2020-05-23 10:59:15 +02:00
for ( let key in languageToasts ) this . killLanguageToast ( key ) ;
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
BDFDB . PluginUtils . clear ( this ) ;
}
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2020-04-11 19:32:58 +02:00
// Begin of own functions
2018-10-11 10:21:26 +02:00
2020-02-20 17:23:49 +01:00
onSlateContextMenu ( e ) {
2020-05-23 16:35:29 +02:00
let [ SCparent , SCindex ] = BDFDB . ContextMenuUtils . findItem ( e . returnvalue , { id : "spellcheck" , group : true } ) ;
2020-02-20 17:23:49 +01:00
if ( SCindex > - 1 ) SCparent . splice ( SCindex , 1 ) ;
let textarea = BDFDB . DOMUtils . getParent ( BDFDB . dotCN . textarea , e . instance . props . target ) , word = null ;
if ( textarea ) for ( let error of textarea . parentElement . querySelectorAll ( BDFDB . dotCN . _spellcheckerror ) ) {
let rects = BDFDB . DOMUtils . getRects ( error ) ;
if ( BDFDB . InternalData . mousePosition . pageX > rects . x && BDFDB . InternalData . mousePosition . pageX < ( rects . x + rects . width ) && BDFDB . InternalData . mousePosition . pageY > rects . y && BDFDB . InternalData . mousePosition . pageY < ( rects . y + rects . height ) ) {
word = error . innerText ;
break ;
}
2019-02-19 13:34:09 +01:00
}
2020-02-20 17:23:49 +01:00
if ( word && this . isWordNotInDictionary ( word ) ) {
let similarWords = this . getSimilarWords ( word . toLowerCase ( ) . trim ( ) ) ;
2020-05-20 14:40:43 +02:00
let [ children , index ] = BDFDB . ContextMenuUtils . findItem ( e . returnvalue , { id : "devmode-copy-id" , group : true } ) ;
2020-05-20 11:55:46 +02:00
children . splice ( index > - 1 ? index : children . length , 0 , BDFDB . ContextMenuUtils . createItem ( BDFDB . LibraryComponents . MenuItems . MenuGroup , {
children : BDFDB . ContextMenuUtils . createItem ( BDFDB . LibraryComponents . MenuItems . MenuItem , {
2020-02-20 17:23:49 +01:00
label : BDFDB . LanguageUtils . LanguageStrings . SPELLCHECK ,
2020-05-19 19:24:52 +02:00
id : BDFDB . ContextMenuUtils . createItemId ( this . name , "spellcheck" ) ,
children : [
2020-05-20 11:55:46 +02:00
BDFDB . ContextMenuUtils . createItem ( BDFDB . LibraryComponents . MenuItems . MenuItem , {
2020-02-20 17:23:49 +01:00
label : this . labels . context _spellcheck _text ,
2020-05-19 19:24:52 +02:00
id : BDFDB . ContextMenuUtils . createItemId ( this . name , "add-to-spellcheck" ) ,
hint : _ => {
2020-05-20 14:40:43 +02:00
return BDFDB . ReactUtils . createElement ( BDFDB . LibraryComponents . MenuItems . MenuHint , {
2020-05-19 19:24:52 +02:00
hint : word
} ) ;
} ,
2019-11-21 11:38:04 +01:00
action : _ => {
2019-12-10 13:59:44 +01:00
BDFDB . ContextMenuUtils . close ( e . instance ) ;
2020-02-20 17:23:49 +01:00
this . addToOwnDictionary ( word ) ;
2019-10-05 18:58:18 +02:00
}
2020-02-20 17:23:49 +01:00
} ) ,
2020-05-28 14:33:01 +02:00
BDFDB . ContextMenuUtils . createItem ( BDFDB . LibraryComponents . MenuItems . MenuSeparator , { } ) ,
! similarWords . length ? BDFDB . ContextMenuUtils . createItem ( BDFDB . LibraryComponents . MenuItems . MenuItem , {
label : this . labels . context _nosimilarwords _text ,
id : BDFDB . ContextMenuUtils . createItemId ( this . name , "no-suggestions" ) ,
disabled : true
} ) : similarWords . sort ( ) . map ( suggestion => BDFDB . ContextMenuUtils . createItem ( BDFDB . LibraryComponents . MenuItems . MenuItem , {
label : suggestion ,
id : BDFDB . ContextMenuUtils . createItemId ( this . name , "suggestion" , suggestion ) ,
action : _ => {
BDFDB . ContextMenuUtils . close ( e . instance ) ;
this . replaceWord ( e . instance . props . editor , word , suggestion ) ;
}
} ) )
] . flat ( 10 ) . filter ( n => n )
2020-02-20 17:23:49 +01:00
} )
} ) ) ;
}
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2020-06-09 09:30:21 +02:00
onSettingsClosed ( ) {
if ( this . SettingsUpdated ) {
delete this . SettingsUpdated ;
this . forceUpdateAll ( ) ;
}
}
2020-02-20 17:23:49 +01:00
processSlateChannelTextArea ( e ) {
2020-06-09 19:42:00 +02:00
let newText = BDFDB . LibraryModules . SlateUtils . serialize ( e . instance . props . value ) ;
if ( newText != currentText ) {
currentText = newText ;
BDFDB . DOMUtils . remove ( e . node . parentElement . querySelectorAll ( BDFDB . dotCN . _spellcheckoverlay ) ) ;
BDFDB . TimeUtils . clear ( checkTimeout ) ;
checkTimeout = BDFDB . TimeUtils . timeout ( _ => {
let overlay = e . node . cloneNode ( true ) , wrapper = BDFDB . DOMUtils . getParent ( BDFDB . dotCN . textareainner , e . node ) ;
BDFDB . DOMUtils . addClass ( overlay , BDFDB . disCN . _spellcheckoverlay ) ;
let style = Object . assign ( { } , getComputedStyle ( e . node ) ) ;
for ( let i in style ) if ( i . indexOf ( "webkit" ) == - 1 && isNaN ( parseInt ( i ) ) ) overlay . style [ i ] = style [ i ] ;
overlay . style . setProperty ( "color" , "transparent" , "important" ) ;
overlay . style . setProperty ( "background" , "none" , "important" ) ;
overlay . style . setProperty ( "mask" , "none" , "important" ) ;
overlay . style . setProperty ( "pointer-events" , "none" , "important" ) ;
overlay . style . setProperty ( "position" , "absolute" , "important" ) ;
overlay . style . setProperty ( "left" , BDFDB . DOMUtils . getRects ( e . node ) . left - BDFDB . DOMUtils . getRects ( wrapper ) . left + "px" , "important" ) ;
overlay . style . setProperty ( "width" , BDFDB . DOMUtils . getRects ( e . node ) . width - style . paddingLeft - style . paddingRight + "px" , "important" ) ;
overlay . style . setProperty ( "height" , style . height , "important" ) ;
for ( let child of overlay . querySelectorAll ( "*" ) ) {
child . style . setProperty ( "color" , "transparent" , "important" ) ;
child . style . setProperty ( "background-color" , "transparent" , "important" ) ;
child . style . setProperty ( "border-color" , "transparent" , "important" ) ;
child . style . setProperty ( "text-shadow" , "none" , "important" ) ;
child . style . setProperty ( "pointer-events" , "none" , "important" ) ;
if ( child . getAttribute ( "data-slate-string" ) && child . parentElement . getAttribute ( "data-slate-leaf" ) ) {
let newline = child . querySelector ( "br" ) ;
if ( newline ) newline . remove ( ) ;
child . innerHTML = this . spellCheckText ( child . textContent ) ;
if ( newline ) child . appendChild ( newline ) ;
}
2020-06-09 09:30:21 +02:00
}
2020-06-09 19:42:00 +02:00
e . node . parentElement . appendChild ( overlay ) ;
} , 300 ) ;
}
2019-01-17 23:48:29 +01:00
}
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
spellCheckText ( string ) {
let htmlString = [ ] ;
string . replace ( /\n/g , "\n " ) . split ( " " ) . forEach ( word => {
let hasNewline = word . endsWith ( "\n" ) ;
word = word . replace ( /\n/g , "" ) ;
htmlString . push ( ` <label class=" ${ this . isWordNotInDictionary ( word ) ? BDFDB . disCN . _spellcheckerror : "" } " style="color: transparent !important; text-shadow: none !important;"> ${ BDFDB . StringUtils . htmlEscape ( word ) } </label> ${ hasNewline ? "\n" : "" } ` ) ;
} ) ;
return htmlString . join ( " " ) . replace ( /\n /g , "\n" ) ;
}
2020-01-15 21:46:41 +01:00
2020-02-20 17:23:49 +01:00
replaceWord ( editor , toBeReplaced , replacement ) {
2020-05-28 14:33:01 +02:00
let editorContainer = BDFDB . ReactUtils . findOwner ( editor , { name : "ChannelEditorContainer" , up : true } ) ;
if ( ! editor || ! editorContainer || ! editorContainer . props || ! editorContainer . props . textValue ) return ;
2020-02-20 17:23:49 +01:00
toBeReplaced = toBeReplaced . toUpperCase ( ) ;
let newString = [ ] ;
2020-05-28 14:33:01 +02:00
editorContainer . props . textValue . replace ( /\n/g , "\n " ) . split ( " " ) . forEach ( word => {
2020-02-20 17:23:49 +01:00
let hasNewline = word . endsWith ( "\n" ) ;
word = word . replace ( /\n/g , "" ) ;
if ( word . toUpperCase ( ) == toBeReplaced ) {
let firstLetter = word . charAt ( 0 ) ;
let isCapitalised = firstLetter . toUpperCase ( ) == firstLetter && firstLetter . toLowerCase ( ) != firstLetter ;
newString . push ( ( isCapitalised ? replacement . charAt ( 0 ) . toUpperCase ( ) + replacement . slice ( 1 ) : replacement ) + ( hasNewline ? "\n" : "" ) ) ;
}
else newString . push ( word + ( hasNewline ? "\n" : "" ) ) ;
} ) ;
editor . setValue ( BDFDB . SlateUtils . copyRichValue ( newString . join ( " " ) . replace ( /\n /g , "\n" ) , editor . props . value ) ) ;
}
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
addToOwnDictionary ( word ) {
word = word . split ( " " ) [ 0 ] . split ( "\n" ) [ 0 ] . split ( "\r" ) [ 0 ] . split ( "\t" ) [ 0 ] ;
if ( word ) {
2020-05-23 10:59:15 +02:00
let wordLow = word . toLowerCase ( ) ;
2020-06-09 09:30:21 +02:00
if ( languages [ choices . dictionaryLanguage ] ) {
let ownDictionary = BDFDB . DataUtils . load ( this , "owndics" , choices . dictionaryLanguage ) || [ ] ;
2020-05-23 10:59:15 +02:00
if ( ! ownDictionary . includes ( wordLow ) ) {
ownDictionary . push ( wordLow ) ;
2020-06-09 09:30:21 +02:00
BDFDB . DataUtils . save ( ownDictionary , this , "owndics" , choices . dictionaryLanguage ) ;
BDFDB . NotificationUtils . toast ( this . labels . toast _wordadd _text . replace ( "${word}" , word ) . replace ( "${dicName}" , this . getLanguageName ( languages [ choices . dictionaryLanguage ] ) ) , { type : "success" } ) ;
2020-05-25 17:03:57 +02:00
dictionaries . dictionaryLanguage = this . formatDictionary ( langDictionaries . dictionaryLanguage . concat ( ownDictionary ) ) ;
2020-05-23 10:59:15 +02:00
}
2020-02-20 17:23:49 +01:00
}
2018-10-11 10:21:26 +02:00
}
}
2019-01-26 22:45:19 +01:00
2020-05-23 10:59:15 +02:00
setDictionary ( key , lang ) {
this . killLanguageToast ( key ) ;
if ( languages [ lang ] ) {
2020-05-25 17:03:57 +02:00
let ownDictionary = BDFDB . DataUtils . load ( this , "owndics" , lang ) || [ ] ;
2020-05-23 10:59:15 +02:00
languageToasts [ key ] = BDFDB . NotificationUtils . toast ( "Grabbing dictionary (" + this . getLanguageName ( languages [ lang ] ) + "). Please wait" , { timeout : 0 } ) ;
languageToasts [ key ] . interval = BDFDB . TimeUtils . interval ( _ => {
languageToasts [ key ] . textContent = languageToasts [ key ] . textContent . indexOf ( "....." ) > - 1 ? "Grabbing dictionary (" + this . getLanguageName ( languages [ lang ] ) + "). Please wait" : languageToasts [ key ] . textContent + "." ;
} , 500 ) ;
languageToasts [ key ] . lang = lang
2020-05-25 21:03:56 +02:00
let folder = BDFDB . LibraryRequires . path . join ( BDFDB . BDUtils . getPluginsFolder ( ) , "dictionaries" ) ;
let filePath = BDFDB . LibraryRequires . path . join ( folder , lang + ".dic" ) ;
let parse = ( error , response , body , download ) => {
this . killLanguageToast ( key ) ;
2020-05-25 18:35:25 +02:00
if ( error || ( response && body . toLowerCase ( ) . indexOf ( "<!doctype html>" ) > - 1 ) ) {
2020-05-23 10:59:15 +02:00
BDFDB . NotificationUtils . toast ( "Failed to grab dictionary (" + this . getLanguageName ( languages [ lang ] ) + ")." , { type : "error" } ) ;
}
else if ( response && languageToasts [ key ] . lang == lang ) {
2020-05-25 21:03:56 +02:00
if ( download ) {
if ( ! BDFDB . LibraryRequires . fs . existsSync ( folder ) ) BDFDB . LibraryRequires . fs . mkdirSync ( folder ) ;
BDFDB . LibraryRequires . fs . writeFile ( filePath , body , _ => { } ) ;
}
2020-05-25 18:35:25 +02:00
langDictionaries [ key ] = body . toLowerCase ( ) . replace ( /\r/g , "" ) . split ( "\n" ) ;
dictionaries [ key ] = this . formatDictionary ( langDictionaries [ key ] . concat ( ownDictionary ) ) ;
2020-05-23 10:59:15 +02:00
BDFDB . NotificationUtils . toast ( "Successfully grabbed dictionary (" + this . getLanguageName ( languages [ lang ] ) + ")." , { type : "success" } ) ;
}
2020-05-25 21:03:56 +02:00
} ;
2020-06-09 09:30:21 +02:00
if ( settings . downloadDictionary && BDFDB . LibraryRequires . fs . existsSync ( filePath ) ) BDFDB . LibraryRequires . fs . readFile ( filePath , ( error , buffer ) => {
2020-05-25 21:03:56 +02:00
parse ( error , buffer , buffer . toString ( ) , false ) ;
} ) ;
else BDFDB . LibraryRequires . request ( "https://mwittrien.github.io/BetterDiscordAddons/Plugins/SpellCheck/dic/" + lang + ".dic" , ( error , response , body ) => {
2020-06-09 09:30:21 +02:00
parse ( error , response , body , settings . downloadDictionary ) ;
2020-05-23 10:59:15 +02:00
} ) ;
}
else {
delete dictionaries [ key ] ;
delete langDictionaries [ key ] ;
}
2020-02-20 17:23:49 +01:00
}
2020-05-25 17:03:57 +02:00
formatDictionary ( words ) {
2020-05-25 18:35:25 +02:00
let i = 0 ;
return words . reduce ( ( dictionary , word ) => {
2020-05-25 17:03:57 +02:00
let firstLetterLower = word . charAt ( 0 ) . toLowerCase ( ) ;
if ( ! dictionary [ firstLetterLower ] ) dictionary [ firstLetterLower ] = { } ;
if ( ! dictionary [ firstLetterLower ] [ word . length ] ) dictionary [ firstLetterLower ] [ word . length ] = [ ] ;
2020-05-25 18:35:25 +02:00
dictionary [ firstLetterLower ] [ word . length ] . push ( word ) ;
return dictionary ;
} , { } ) ;
2020-05-25 17:03:57 +02:00
}
2019-01-26 22:45:19 +01:00
2020-05-23 10:59:15 +02:00
killLanguageToast ( key ) {
if ( languageToasts [ key ] && typeof languageToasts [ key ] . close == "function" ) {
BDFDB . TimeUtils . clear ( languageToasts [ key ] . interval ) ;
languageToasts [ key ] . close ( ) ;
2020-02-20 17:23:49 +01:00
}
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2020-05-25 17:03:57 +02:00
isWordNotInDictionary ( unformatedWord ) {
let wordLow = unformatedWord . toLowerCase ( ) ;
2020-05-28 14:33:01 +02:00
let wordWithoutSymbols = wordLow . replace ( /[0-9\µ\@\$\£\€\¥\¢\²\³\>\<\|\,\;\.\:\-\_\#\+\*\~\?\¿\\\´ \`\}\=\]\)\[\(\{\/\&\%\§\"\!\¡\^\°\n\t\r]/g , "" ) ;
2020-06-22 18:22:07 +02:00
if ( wordLow . indexOf ( "http://" ) != 0 && wordLow . indexOf ( "https://" ) != 0 && wordWithoutSymbols && wordWithoutSymbols . length > wordLow . length / 2 ) {
2020-06-06 08:23:30 +02:00
let wordStartingPos = /^.{1}'/ . test ( wordWithoutSymbols ) ? wordWithoutSymbols . split ( "'" ) [ 1 ] : "" ;
let wordEndingPos = /'.{1}$/ . test ( wordWithoutSymbols ) ? wordWithoutSymbols . split ( "'" ) . reverse ( ) [ 1 ] : "" ;
for ( let key in dictionaries ) for ( let word of BDFDB . ArrayUtils . removeCopies ( [ wordLow , wordWithoutSymbols , wordStartingPos , wordEndingPos ] . filter ( n => n ) ) ) {
2020-05-28 14:33:01 +02:00
let firstLetterLower = word . charAt ( 0 ) ;
if ( dictionaries [ key ] && dictionaries [ key ] [ firstLetterLower ] && dictionaries [ key ] [ firstLetterLower ] [ word . length ] && dictionaries [ key ] [ firstLetterLower ] [ word . length ] . includes ( word ) ) return false ;
2020-05-23 10:59:15 +02:00
}
return true ;
}
return false ;
2020-02-20 17:23:49 +01:00
}
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
getSimilarWords ( word ) {
2020-06-09 09:30:21 +02:00
let similarWords = [ ] ;
if ( amounts . maxSimilarAmount > 0 ) {
2020-05-25 17:03:57 +02:00
let firstLetterLower = word . charAt ( 0 ) . toLowerCase ( ) ;
let possibilities = [ ] ;
for ( let key in dictionaries ) if ( dictionaries [ key ] && dictionaries [ key ] [ firstLetterLower ] ) possibilities = possibilities . concat ( BDFDB . ObjectUtils . toArray ( dictionaries [ key ] [ firstLetterLower ] ) . flat ( ) ) ;
possibilities = BDFDB . ArrayUtils . removeCopies ( possibilities ) ;
2020-02-20 17:23:49 +01:00
let similarities = { } ;
2020-05-25 17:03:57 +02:00
for ( let string of possibilities ) {
2020-02-20 17:23:49 +01:00
let value = this . wordSimilarity ( word , string ) ;
if ( ! similarities [ value ] ) similarities [ value ] = [ ] ;
similarities [ value ] . push ( string ) ;
}
let amount = 0 ;
for ( let value of Object . keys ( similarities ) . sort ( ) . reverse ( ) ) {
for ( let similarWord of similarities [ value ] ) {
2020-06-09 19:42:00 +02:00
if ( amount < amounts . maxSimilarAmount && ! similarWords . includes ( similarWord ) ) {
2020-02-20 17:23:49 +01:00
similarWords . push ( similarWord ) ;
amount ++ ;
}
2020-06-09 19:42:00 +02:00
if ( amount >= amounts . maxSimilarAmount ) break ;
2018-10-11 10:21:26 +02:00
}
2020-06-09 19:42:00 +02:00
if ( amount >= amounts . maxSimilarAmount ) break ;
2018-10-11 10:21:26 +02:00
}
}
2020-02-20 17:23:49 +01:00
return similarWords ;
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
wordSimilarity ( a , b ) {
let temp ;
if ( a . length === 0 || b . length === 0 || a . length - b . length > 3 || b . length - a . length > 3 ) return 0 ;
if ( a . length > b . length ) {
temp = a ;
a = b ;
b = temp ;
2018-10-11 10:21:26 +02:00
}
2020-02-20 17:23:49 +01:00
let result = 0 , row = [ ... Array ( a . length + 1 ) . keys ( ) ] ;
for ( let i = 1 ; i <= b . length ; i ++ ) {
result = i ;
for ( let j = 1 ; j <= a . length ; j ++ ) {
temp = row [ j - 1 ] ;
row [ j - 1 ] = result ;
result = b [ i - 1 ] === a [ j - 1 ] ? temp : Math . min ( temp + 1 , Math . min ( result + 1 , row [ j ] + 1 ) ) ;
}
}
return ( b . length - result ) / b . length ;
2018-10-11 10:21:26 +02:00
}
2020-05-23 10:59:15 +02:00
getLanguageName ( language ) {
if ( language . name . startsWith ( "Discord" ) ) return language . name . slice ( 0 , - 1 ) + ( language . ownlang && languages [ language . id ] . name != language . ownlang ? ` / ${ language . ownlang } ` : "" ) + ")" ;
else return language . name + ( language . ownlang && language . name != language . ownlang ? ` / ${ language . ownlang } ` : "" ) ;
}
2020-06-09 09:30:21 +02:00
forceUpdateAll ( ) {
settings = BDFDB . DataUtils . get ( this , "settings" ) ;
choices = BDFDB . DataUtils . get ( this , "choices" ) ;
amounts = BDFDB . DataUtils . get ( this , "amounts" ) ;
BDFDB . ModuleUtils . forceAllUpdates ( this ) ;
}
2019-01-26 22:45:19 +01:00
2020-02-20 17:23:49 +01:00
setLabelsByLanguage ( ) {
switch ( BDFDB . LanguageUtils . getLanguage ( ) . id ) {
case "hr" : //croatian
return {
context _spellcheck _text : "Dodaj u rječnik" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Nema sličnih riječi" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Riječ ${word} dodana je u rječnik ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "da" : //danish
return {
context _spellcheck _text : "Tilføj til ordbog" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Ingen lignende ord" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Ord ${word} tilføjet til ordbog ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "de" : //german
return {
context _spellcheck _text : "Zum Wörterbuch hinzufügen" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Keine ähnlichen Wörter" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Wort ${word} wurde zum Wörterbuch ${dicName} hinzugefügt."
2020-02-20 17:23:49 +01:00
} ;
case "es" : //spanish
return {
context _spellcheck _text : "Agregar al diccionario" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "No hay palabras similares" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Se agregó la palabra ${word} al diccionario ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "fr" : //french
return {
context _spellcheck _text : "Ajouter au dictionnaire" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Pas de mots similaires" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Le mot ${word} a été ajouté au dictionnaire ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "it" : //italian
return {
context _spellcheck _text : "Aggiungi al dizionario" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Nessuna parola simile" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Parola ${word} aggiunta al dizionario ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "nl" : //dutch
return {
context _spellcheck _text : "Toevoegen aan woordenboek" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Geen vergelijkbare woorden" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Word ${word} toegevoegd aan woordenboek ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "no" : //norwegian
return {
context _spellcheck _text : "Legg til i ordbok" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Ingen lignende ord" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Ord ${word} legges til ordbok ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "pl" : //polish
return {
context _spellcheck _text : "Dodaj do słownika" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Brak podobnych słów" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Słowo ${word} dodane do słownika ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "pt-BR" : //portuguese (brazil)
return {
context _spellcheck _text : "Adicionar ao dicionário" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Sem palavras semelhantes" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Palavra ${word} adicionado ao dicionário ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "fi" : //finnish
return {
context _spellcheck _text : "Lisää sanakirjaan" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Ei vastaavia sanoja" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Sana ${word} lisättiin sanakirjaan ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "sv" : //swedish
return {
context _spellcheck _text : "Lägg till i ordbok" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Inga liknande ord" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Ord ${word} läggs till ordbok ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "tr" : //turkish
return {
context _spellcheck _text : "Sözlükye Ekle" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Benzer kelime yoktur" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Sözcük ${word} sözlük ${dicName}'ye eklendi."
2020-02-20 17:23:49 +01:00
} ;
case "cs" : //czech
return {
context _spellcheck _text : "Přidat do slovníku" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Žádné podobné slova" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Slovo ${word} bylo přidáno do slovníku ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "bg" : //bulgarian
return {
context _spellcheck _text : "Добави в речника" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Няма подобни думи" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Думата ${word} е добавена към речника ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "ru" : //russian
return {
context _spellcheck _text : "Добавить в словарь" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Нет похожих слов" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Слово ${word} добавлено в словарь ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "uk" : //ukrainian
return {
context _spellcheck _text : "Додати до словника" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "Немає подібних слів" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Словник ${word} додається до словника ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
case "ja" : //japanese
return {
context _spellcheck _text : "辞書に追加" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "類似の単語はありません" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "単語 ${word} が辞書 ${dicName} に追加されました。"
2020-02-20 17:23:49 +01:00
} ;
case "zh-TW" : //chinese (traditional)
return {
context _spellcheck _text : "添加到詞典" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "沒有類似的詞" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "單詞 ${word} 添加到字典 ${dicName}。"
2020-02-20 17:23:49 +01:00
} ;
case "ko" : //korean
return {
context _spellcheck _text : "사전에 추가" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "유사한 단어 없음" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "단어 ${word} 사전 ${dicName} 에 추가되었습니다."
2020-02-20 17:23:49 +01:00
} ;
default : //default: english
return {
context _spellcheck _text : "Add to Dictionary" ,
2020-05-28 14:33:01 +02:00
context _nosimilarwords _text : "No similar Words" ,
2020-05-23 10:59:15 +02:00
toast _wordadd _text : "Word ${word} added to dictionary ${dicName}."
2020-02-20 17:23:49 +01:00
} ;
}
2018-10-11 10:21:26 +02:00
}
}
2020-02-20 17:23:49 +01:00
} ) ( ) ;