2019-01-31 17:06:48 +01:00
//META{"name":"SpellCheck","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
class SpellCheck {
2019-01-17 23:48:29 +01:00
getName ( ) { return "SpellCheck" ; }
2019-04-27 18:45:01 +02:00
getVersion ( ) { return "1.3.5" ; }
2019-01-17 23:48:29 +01:00
getAuthor ( ) { return "DevilBro" ; }
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
2018-10-11 10:21:26 +02:00
initConstructor ( ) {
2019-04-27 18:45:01 +02:00
this . changelog = {
"fixed" : [ [ "New Select Classes" , "The Dropdown-Select element got new classes on canary, this update will prevent stable from breaking once the class change is pushed to stable" ] ]
} ;
2019-01-17 23:48:29 +01:00
this . patchModules = {
"ChannelTextArea" : "componentDidMount"
} ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . languages = { } ;
this . langDictionary = [ ] ;
this . dictionary = [ ] ;
this . spellCheckContextEntryMarkup =
` <div class=" ${ BDFDB . disCN . contextmenuitemgroup } ">
< div class = "${BDFDB.disCN.contextmenuitem} similarwords-item ${BDFDB.disCN.contextmenuitemsubmenu}" >
2019-04-18 09:28:20 +02:00
< span class = "BDFDB-textscrollwrapper" speed = 3 > < div class = "BDFDB-textscroll" > REPLACE _context _similarwords _text < / d i v > < / s p a n >
2018-10-11 10:21:26 +02:00
< div class = "${BDFDB.disCN.contextmenuhint}" > < / d i v >
< / d i v >
< div class = "${BDFDB.disCN.contextmenuitem} spellcheck-item" >
2019-04-18 09:28:20 +02:00
< span class = "BDFDB-textscrollwrapper" speed = 3 > < div class = "BDFDB-textscroll" > REPLACE _context _spellcheck _text < / d i v > < / s p a n >
2018-10-11 10:21:26 +02:00
< div class = "${BDFDB.disCN.contextmenuhint}" > < / d i v >
< / d i v >
< / d i v > ` ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . similarWordsContextSubMenuMarkup =
` <div class=" ${ BDFDB . disCN . contextmenu } spellcheck-submenu">
< div class = "${BDFDB.disCN.contextmenuitem} nosimilars-item" >
2019-04-18 09:28:20 +02:00
< span class = "BDFDB-textscrollwrapper" speed = 3 > < div class = "BDFDB-textscroll" > REPLACE _similarwordssubmenu _none _text < / d i v > < / s p a n >
2018-10-11 10:21:26 +02:00
< div class = "${BDFDB.disCN.contextmenuhint}" > < / d i v >
< / d i v >
< / d i v > ` ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . spellCheckLayerMarkup =
` <div class="spellcheck-overlay" style="position:absolute !important; pointer-events:none !important; background:transparent !important; color:transparent !important; text-shadow:none !important;"></div> ` ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . css =
2019-01-17 23:48:29 +01:00
` .spellcheck-overlay::-webkit-scrollbar,
2018-10-11 10:21:26 +02:00
. spellcheck - overlay : : - webkit - scrollbar - button ,
. spellcheck - overlay : : - webkit - scrollbar - track ,
. spellcheck - overlay : : - webkit - scrollbar - track - piece ,
. spellcheck - overlay : : - webkit - scrollbar - thumb ,
. spellcheck - overlay : : - webkit - scrollbar - corner ,
. spellcheck - overlay : : - webkit - resizer {
visibility : hidden ! important ;
}
. spellcheck - overlay . spelling - error {
background - image : url ( '' ) ;
background - repeat : repeat - x ;
background - position : bottom ;
} ` ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . defaults = {
choices : {
dictionaryLanguage : { value : "en" , description : "Dictionay Language:" }
} ,
amounts : {
2019-02-26 12:16:08 +01:00
maxSimilarAmount : { value : 6 , min : 1 , max : 30 , description : "Maximal Amount of suggested Words:" }
2018-10-11 10:21:26 +02:00
}
} ;
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
getSettingsPanel ( ) {
2019-01-22 11:28:32 +01:00
if ( ! global . BDFDB || typeof BDFDB != "object" || ! BDFDB . loaded || ! this . started ) return ;
2018-10-11 10:21:26 +02:00
var settings = BDFDB . getAllData ( this , "settings" ) ;
var choices = BDFDB . getAllData ( this , "choices" ) ;
var amounts = BDFDB . getAllData ( this , "amounts" ) ;
2019-04-18 09:28:20 +02:00
var settingshtml = ` <div class=" ${ this . name } -settings BDFDB-settings"><div class=" ${ BDFDB . disCNS . titledefault + BDFDB . disCNS . title + BDFDB . disCNS . size18 + BDFDB . disCNS . height24 + BDFDB . disCNS . weightnormal + BDFDB . disCN . marginbottom8 } "> ${ this . name } </div><div class="BDFDB-settings-inner"> ` ;
2018-10-11 10:21:26 +02:00
for ( let key in settings ) {
2019-01-17 23:48:29 +01:00
settingshtml += ` <div class=" ${ BDFDB . disCNS . flex + BDFDB . disCNS . flex2 + BDFDB . disCNS . horizontal + BDFDB . disCNS . horizontal2 + BDFDB . disCNS . directionrow + BDFDB . disCNS . justifystart + BDFDB . disCNS . aligncenter + BDFDB . disCNS . nowrap + BDFDB . disCN . marginbottom8 } " style="flex: 1 1 auto;"><h3 class=" ${ BDFDB . disCNS . titledefault + BDFDB . disCNS . title + BDFDB . disCNS . marginreset + BDFDB . disCNS . weightmedium + BDFDB . disCNS . size16 + BDFDB . disCNS . height24 + BDFDB . disCN . flexchild } " style="flex: 1 1 auto;"> ${ this . defaults . settings [ key ] . description } </h3><div class=" ${ BDFDB . disCNS . flexchild + BDFDB . disCNS . switchenabled + BDFDB . disCNS . switch + BDFDB . disCNS . switchvalue + BDFDB . disCNS . switchsizedefault + BDFDB . disCNS . switchsize + BDFDB . disCN . switchthemedefault } " style="flex: 0 0 auto;"><input type="checkbox" value="settings ${ key } " class=" ${ BDFDB . disCNS . switchinnerenabled + BDFDB . disCN . switchinner } settings-switch" ${ settings [ key ] ? " checked" : "" } ></div></div> ` ;
2018-10-11 10:21:26 +02:00
}
for ( let key in choices ) {
2019-04-27 18:45:01 +02:00
settingshtml += ` <div class=" ${ BDFDB . disCNS . flex + BDFDB . disCNS . flex2 + BDFDB . disCNS . horizontal + BDFDB . disCNS . horizontal2 + BDFDB . disCNS . directionrow + BDFDB . disCNS . justifystart + BDFDB . disCNS . aligncenter + BDFDB . disCNS . nowrap + BDFDB . disCN . marginbottom8 } " style="flex: 1 1 auto;"><h3 class=" ${ BDFDB . disCNS . titledefault + BDFDB . disCNS . title + BDFDB . disCNS . weightmedium + BDFDB . disCNS . size16 + BDFDB . disCN . flexchild } " style="flex: 0 0 30%;"> ${ this . defaults . choices [ key ] . description } </h3> ${ BDFDB . createSelectMenu ( this . createSelectChoice ( choices [ key ] ) , choices [ key ] , key ) } </div> ` ;
2018-10-11 10:21:26 +02:00
}
for ( let key in amounts ) {
2019-02-26 12:16:08 +01:00
settingshtml += ` <div class=" ${ BDFDB . disCNS . flex + BDFDB . disCNS . flex2 + BDFDB . disCNS . horizontal + BDFDB . disCNS . horizontal2 + BDFDB . disCNS . directionrow + BDFDB . disCNS . justifystart + BDFDB . disCNS . aligncenter + BDFDB . disCNS . nowrap + BDFDB . disCN . marginbottom8 } " style="flex: 1 1 auto;"><h3 class=" ${ BDFDB . disCNS . titledefault + BDFDB . disCNS . title + BDFDB . disCNS . weightmedium + BDFDB . disCNS . size16 + BDFDB . disCN . flexchild } " style="flex: 0 0 50%;"> ${ this . defaults . amounts [ key ] . description } </h3><div class=" ${ BDFDB . disCN . inputwrapper } inputNumberWrapper ${ BDFDB . disCNS . vertical + BDFDB . disCNS . flex + BDFDB . disCNS . directioncolumn } " style="flex: 1 1 auto;"><span class="numberinput-buttons-zone"><span class="numberinput-button-up"></span><span class="numberinput-button-down"></span></span><input type="number" ${ ( ! isNaN ( this . defaults . amounts [ key ] . min ) && this . defaults . amounts [ key ] . min !== null ? ' min="' + this . defaults . amounts [ key ] . min + '"' : '' ) + ( ! isNaN ( this . defaults . amounts [ key ] . max ) && this . defaults . amounts [ key ] . max !== null ? ' max="' + this . defaults . amounts [ key ] . max + '"' : '' ) } option=" ${ key } " value=" ${ amounts [ key ] } " class=" ${ BDFDB . disCNS . inputdefault + BDFDB . disCNS . input + BDFDB . disCN . size16 } amount-input"></div></div> ` ;
2018-10-11 10:21:26 +02:00
}
var ownDictionary = BDFDB . loadData ( choices . dictionaryLanguage , this , "owndics" ) || [ ] ;
2019-04-18 09:28:20 +02:00
settingshtml += ` <h3 class=" ${ BDFDB . disCNS . titledefault + BDFDB . disCNS . title + BDFDB . disCNS . marginreset + BDFDB . disCNS . weightmedium + BDFDB . disCNS . size16 + BDFDB . disCNS . height24 + BDFDB . disCN . flexchild } " style="flex: 1 1 auto;">Your own Dictionary:</h3><div class="BDFDB-settings-inner-list word-list ${ BDFDB . disCN . marginbottom8 } "> ` ;
2018-10-11 10:21:26 +02:00
for ( let word of ownDictionary ) {
settingshtml += ` <div class=" ${ BDFDB . disCNS . flex + BDFDB . disCNS . flex2 + BDFDB . disCNS . vertical + BDFDB . disCNS . directionrow + BDFDB . disCNS . justifystart + BDFDB . disCNS . alignstretch + BDFDB . disCNS . nowrap + BDFDB . disCNS . margintop4 + BDFDB . disCNS . marginbottom4 + BDFDB . disCN . hovercard } "><div class=" ${ BDFDB . disCN . hovercardinner } "><div class=" ${ BDFDB . disCNS . description + BDFDB . disCNS . formtext + BDFDB . disCNS . note + BDFDB . disCNS . margintop4 + BDFDB . disCNS . modedefault + BDFDB . disCNS . primary + BDFDB . disCN . ellipsis } entryword"> ${ word } </div></div><div class=" ${ BDFDB . disCN . hovercardbutton } remove-word"></div></div> ` ;
}
settingshtml += ` </div> ` ;
settingshtml += ` </div></div> ` ;
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
let settingspanel = BDFDB . htmlToElement ( settingshtml ) ;
2018-10-11 10:21:26 +02:00
2019-01-17 23:48:29 +01:00
BDFDB . initElements ( settingspanel , this ) ;
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
BDFDB . addEventListener ( this , settingspanel , "click" , ".remove-word" , e => { this . removeFromOwnDictionarye ; } ) ;
2019-04-27 18:45:01 +02:00
BDFDB . addEventListener ( this , settingspanel , "click" , BDFDB . dotCN . selectcontrol , e => {
BDFDB . openDropdownMenu ( e , this . saveSelectChoice . bind ( this ) , this . createSelectChoice . bind ( this ) , this . languages , "inSettings" ) ;
} ) ;
2018-10-11 10:21:26 +02:00
return settingspanel ;
}
//legacy
load ( ) { }
start ( ) {
2019-02-04 09:13:15 +01:00
if ( ! global . BDFDB ) global . BDFDB = { myPlugins : { } } ;
if ( global . BDFDB && global . BDFDB . myPlugins && typeof global . BDFDB . myPlugins == "object" ) global . BDFDB . myPlugins [ this . getName ( ) ] = this ;
2019-01-17 23:48:29 +01:00
var libraryScript = document . querySelector ( 'head script[src="https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.js"]' ) ;
if ( ! libraryScript || performance . now ( ) - libraryScript . getAttribute ( "date" ) > 600000 ) {
2018-10-11 10:21:26 +02:00
if ( libraryScript ) libraryScript . remove ( ) ;
libraryScript = document . createElement ( "script" ) ;
libraryScript . setAttribute ( "type" , "text/javascript" ) ;
libraryScript . setAttribute ( "src" , "https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.js" ) ;
2019-01-17 23:48:29 +01:00
libraryScript . setAttribute ( "date" , performance . now ( ) ) ;
2019-01-30 21:23:49 +01:00
libraryScript . addEventListener ( "load" , ( ) => { if ( global . BDFDB && typeof BDFDB === "object" && BDFDB . loaded ) this . initialize ( ) ; } ) ;
2018-10-11 10:21:26 +02:00
document . head . appendChild ( libraryScript ) ;
}
2019-01-17 23:48:29 +01:00
else if ( global . BDFDB && typeof BDFDB === "object" && BDFDB . loaded ) this . initialize ( ) ;
2018-10-11 10:21:26 +02:00
this . startTimeout = setTimeout ( ( ) => { this . initialize ( ) ; } , 30000 ) ;
}
initialize ( ) {
2019-01-17 23:48:29 +01:00
if ( global . BDFDB && typeof BDFDB === "object" && BDFDB . loaded ) {
2019-01-22 11:05:54 +01:00
if ( this . started ) return ;
2018-10-11 10:21:26 +02:00
BDFDB . loadMessage ( this ) ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . languages = Object . assign ( { } , BDFDB . languages ) ;
this . languages = BDFDB . filterObject ( this . languages , ( lang ) => { return lang . dic == true ? lang : null } ) ;
this . setDictionary ( BDFDB . getData ( "dictionaryLanguage" , this , "choices" ) ) ;
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
BDFDB . WebModules . forceAllUpdates ( this ) ;
2018-10-11 10:21:26 +02:00
}
else {
2019-02-12 21:56:34 +01:00
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
}
}
stop ( ) {
2019-01-17 23:48:29 +01:00
if ( global . BDFDB && typeof BDFDB === "object" && BDFDB . loaded ) {
BDFDB . removeEles ( ".spellcheck-overlay" ) ;
BDFDB . removeClasses ( "spellcheck-added" ) ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . killLanguageToast ( ) ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
BDFDB . unloadMessage ( this ) ;
}
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
// begin of own functions
changeLanguageStrings ( ) {
this . spellCheckContextEntryMarkup = this . spellCheckContextEntryMarkup . replace ( "REPLACE_context_spellcheck_text" , this . labels . context _spellcheck _text ) ;
this . spellCheckContextEntryMarkup = this . spellCheckContextEntryMarkup . replace ( "REPLACE_context_similarwords_text" , this . labels . context _similarwords _text ) ;
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
this . similarWordsContextSubMenuMarkup = this . similarWordsContextSubMenuMarkup . replace ( "REPLACE_similarwordssubmenu_none_text" , this . labels . similarwordssubmenu _none _text ) ;
}
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
onNativeContextMenu ( instance , menu ) {
2019-02-19 13:34:09 +01:00
if ( instance . props && instance . props . target && instance . props . type == "CHANNEL_TEXT_AREA" && ! menu . querySelector ( ".spellcheck-item" ) ) {
2019-02-22 21:11:02 +01:00
BDFDB . toggleEles ( BDFDB . React . findDOMNodeSafe ( BDFDB . getOwnerInstance ( { node : menu , name : "NativeSpellcheckGroup" } ) ) , false ) ;
2019-02-19 13:34:09 +01:00
var textarea = instance . props . target , word = null , length = 0 ;
if ( textarea . value && ( textarea . selectionStart || textarea . selectionEnd ) ) for ( let splitword of textarea . value . split ( /\s/g ) ) {
length += splitword . length + 1 ;
if ( length > textarea . selectionStart ) {
word = splitword ;
break ;
}
}
if ( true || ! word && textarea . value ) for ( let error of textarea . parentElement . querySelectorAll ( ".spelling-error" ) ) {
let rects = BDFDB . getRects ( error ) ;
if ( BDFDB . mousePosition . pageX > rects . x && BDFDB . mousePosition . pageX < ( rects . x + rects . width ) && BDFDB . mousePosition . pageY > rects . y && BDFDB . mousePosition . pageY < ( rects . y + rects . height ) ) {
word = error . innerText ;
break ;
}
}
2019-01-17 23:48:29 +01:00
if ( word && this . isWordNotInDictionary ( word ) ) {
2019-02-19 13:34:09 +01:00
let pasteentry = BDFDB . React . findDOMNodeSafe ( BDFDB . getOwnerInstance ( { node : menu , props : [ "handlePasteItem" ] } ) ) ;
if ( pasteentry ) {
2019-01-17 23:48:29 +01:00
let spellCheckContextEntry = BDFDB . htmlToElement ( this . spellCheckContextEntryMarkup ) ;
menu . appendChild ( spellCheckContextEntry ) ;
2019-02-19 13:34:09 +01:00
spellCheckContextEntry . addEventListener ( "mouseenter" , ( ) => {
BDFDB . createTooltip ( word , spellCheckContextEntry , { type : "left" } ) ;
} ) ;
2019-01-17 23:48:29 +01:00
spellCheckContextEntry . querySelector ( ".spellcheck-item" ) . addEventListener ( "click" , ( ) => {
2019-04-27 18:45:01 +02:00
BDFDB . closeContextMenu ( menu ) ;
2019-01-17 23:48:29 +01:00
this . addToOwnDictionary ( word ) ;
} ) ;
let similarwordsitem = spellCheckContextEntry . querySelector ( ".similarwords-item" ) ;
similarwordsitem . addEventListener ( "mouseenter" , ( ) => {
let similarWordsContextSubMenu = BDFDB . htmlToElement ( this . similarWordsContextSubMenuMarkup ) ;
let similarWords = this . getSimilarWords ( word . toLowerCase ( ) . trim ( ) ) ;
if ( similarWords . length > 0 ) {
BDFDB . removeEles ( similarWordsContextSubMenu . querySelector ( ".nosimilars-item" ) ) ;
for ( let foundWord of similarWords . sort ( ) ) similarWordsContextSubMenu . appendChild ( BDFDB . htmlToElement ( ` <div value=" ${ foundWord } " class=" ${ BDFDB . disCN . contextmenuitem } similarword-item"><span> ${ foundWord } </span><div class=" ${ BDFDB . disCN . contextmenuhint } "></div></div> ` ) ) ;
BDFDB . addChildEventListener ( similarWordsContextSubMenu , "click" , ".similarword-item" , e => {
2019-04-27 18:45:01 +02:00
BDFDB . closeContextMenu ( menu ) ;
2019-02-19 13:34:09 +01:00
this . replaceWord ( textarea , word , e . currentTarget . getAttribute ( "value" ) ) ;
2019-01-17 23:48:29 +01:00
} ) ;
}
BDFDB . appendSubMenu ( similarwordsitem , similarWordsContextSubMenu ) ;
} ) ;
}
}
2018-10-11 10:21:26 +02:00
}
}
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
processChannelTextArea ( instance , wrapper ) {
if ( instance . props && instance . props . type ) {
var textarea = wrapper . querySelector ( "textarea" ) ;
if ( ! textarea ) return ;
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
var updateSpellcheck = ( ) => {
var style = Object . assign ( { } , getComputedStyle ( textarea ) ) ;
for ( let i in style ) if ( i . indexOf ( "webkit" ) == - 1 ) spellcheck . style [ i ] = style [ i ] ;
spellcheck . style . setProperty ( "color" , "transparent" , "important" ) ;
2019-01-25 09:39:31 +01:00
spellcheck . style . setProperty ( "background" , "none" , "important" ) ;
spellcheck . style . setProperty ( "mask" , "none" , "important" ) ;
2019-01-17 23:48:29 +01:00
spellcheck . style . setProperty ( "pointer-events" , "none" , "important" ) ;
spellcheck . style . setProperty ( "position" , "absolute" , "important" ) ;
spellcheck . style . setProperty ( "left" , BDFDB . getRects ( textarea ) . left - BDFDB . getRects ( wrapper ) . left + "px" , "important" ) ;
2019-02-19 13:34:09 +01:00
spellcheck . style . setProperty ( "width" , BDFDB . getRects ( textarea ) . width - style . paddingLeft - style . paddingRight + "px" , "important" ) ;
spellcheck . style . setProperty ( "height" , style . height , "important" ) ;
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
spellcheck . innerHTML = this . spellCheckText ( textarea . value ) ;
spellcheck . scrollTop = textarea . scrollTop ;
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
var spellcheck = BDFDB . htmlToElement ( this . spellCheckLayerMarkup ) ;
BDFDB . addClass ( spellcheck , textarea . className ) ;
2019-01-26 22:45:19 +01:00
2019-02-19 13:34:09 +01:00
textarea . setAttribute ( "spellcheck" , false ) ;
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
textarea . parentElement . appendChild ( spellcheck ) ;
2019-02-11 10:59:55 +01:00
BDFDB . addClass ( wrapper , "spellcheck-added" ) ;
2019-01-26 22:45:19 +01:00
2019-01-17 23:48:29 +01:00
updateSpellcheck ( ) ;
BDFDB . addEventListener ( this , textarea , "keyup" , e => {
clearTimeout ( textarea . spellchecktimeout ) ;
2019-02-19 13:34:09 +01:00
if ( textarea . value ) textarea . spellchecktimeout = setTimeout ( ( ) => { updateSpellcheck ( ) ; } , 100 ) ;
else updateSpellcheck ( ) ;
2018-10-11 10:21:26 +02:00
} ) ;
2019-01-17 23:48:29 +01:00
BDFDB . addEventListener ( this , textarea , "scroll" , e => {
spellcheck . scrollTop = textarea . scrollTop ;
} ) ;
}
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
replaceWord ( textarea , word , replacement ) {
2019-01-20 19:41:36 +01:00
if ( ! textarea ) return ;
2018-10-11 10:21:26 +02:00
textarea . focus ( ) ;
textarea . selectionStart = 0 ;
textarea . selectionEnd = textarea . value . length ;
2019-01-17 23:48:29 +01:00
var firstLetter = word . charAt ( 0 ) ;
var isCapitalised = firstLetter . toUpperCase ( ) == firstLetter && firstLetter . toLowerCase ( ) != firstLetter ;
replacement = isCapitalised ? replacement . charAt ( 0 ) . toUpperCase ( ) + replacement . slice ( 1 ) : replacement ;
document . execCommand ( "insertText" , false , textarea . value . replace ( new RegExp ( word . trim ( ) , "i" ) , replacement ) ) ;
textarea . dispatchEvent ( new Event ( "input" ) ) ;
2019-01-20 19:41:36 +01:00
textarea . dispatchEvent ( new Event ( "keyup" ) ) ;
textarea . dispatchEvent ( new Event ( "change" ) ) ;
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
addToOwnDictionary ( word ) {
word = word . split ( " " ) [ 0 ] . split ( "\n" ) [ 0 ] . split ( "\r" ) [ 0 ] . split ( "\t" ) [ 0 ] ;
if ( word ) {
var wordlow = word . toLowerCase ( ) ;
var lang = BDFDB . getData ( "dictionaryLanguage" , this , "choices" ) ;
var ownDictionary = BDFDB . loadData ( lang , this , "owndics" ) || [ ] ;
if ( ! ownDictionary . includes ( wordlow ) ) {
ownDictionary . push ( wordlow ) ;
BDFDB . saveData ( lang , ownDictionary , this , "owndics" ) ;
2019-02-19 13:34:09 +01:00
BDFDB . showToast ( this . labels . toast _wordadd _text ? this . labels . toast _wordadd _text . replace ( "${word}" , word ) . replace ( "${dicname}" , this . languages [ lang ] . name ) : "" , { type : "success" } ) ;
2018-10-11 10:21:26 +02:00
this . dictionary = this . langDictionary . concat ( ownDictionary ) ;
}
}
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
removeFromOwnDictionary ( e ) {
var entry = e . currentTarget . parentElement ;
var word = entry . querySelector ( ".entryword" ) . textContent ;
entry . remove ( ) ;
var lang = BDFDB . getData ( "dictionaryLanguage" , this , "choices" ) ;
var ownDictionary = BDFDB . loadData ( lang , this , "owndics" ) || [ ] ;
BDFDB . removeFromArray ( ownDictionary , word ) ;
BDFDB . saveData ( lang , ownDictionary , this , "owndics" ) ;
this . dictionary = this . langDictionary . concat ( ownDictionary ) ;
}
2019-04-27 18:45:01 +02:00
saveSelectChoice ( selectWrap , type , choice ) {
if ( type && choice ) {
selectWrap . querySelector ( BDFDB . dotCN . title ) . innerText = this . languages [ choice ] . name ;
this . setDictionary ( choice ) ;
BDFDB . saveData ( type , choice , this , "choices" ) ;
var settingspanel = BDFDB . getParentEle ( ".BDFDB-settings" , selectWrap ) , listcontainer = settingspanel ? settingspanel . querySelector ( ".word-list" ) : null ;
2018-10-11 10:21:26 +02:00
if ( listcontainer ) {
2019-04-27 18:45:01 +02:00
var ownDictionary = BDFDB . loadData ( choice , this , "owndics" ) || [ ] ;
2018-10-11 10:21:26 +02:00
var containerhtml = ` ` ;
for ( let word of ownDictionary ) {
containerhtml += ` <div class=" ${ BDFDB . disCNS . flex + BDFDB . disCNS . flex2 + BDFDB . disCNS . vertical + BDFDB . disCNS . directionrow + BDFDB . disCNS . justifystart + BDFDB . disCNS . alignstretch + BDFDB . disCNS . nowrap + BDFDB . disCNS . margintop4 + BDFDB . disCNS . marginbottom4 + BDFDB . disCN . hovercard } "><div class=" ${ BDFDB . disCN . hovercardinner } "><div class=" ${ BDFDB . disCNS . description + BDFDB . disCNS . formtext + BDFDB . disCNS . note + BDFDB . disCNS . margintop4 + BDFDB . disCNS . modedefault + BDFDB . disCNS . primary + BDFDB . disCN . ellipsis } entryword"> ${ word } </div></div><div class=" ${ BDFDB . disCN . hovercardbutton } remove-word"></div></div> ` ;
}
listcontainer . innerHTML = containerhtml ;
}
}
2019-04-27 18:45:01 +02:00
}
createSelectChoice ( choice ) {
return ` <div class=" ${ BDFDB . disCNS . title + BDFDB . disCNS . medium + BDFDB . disCNS . size16 + BDFDB . disCNS . height20 + BDFDB . disCNS . primary + BDFDB . disCNS . weightnormal + BDFDB . disCN . cursorpointer } " style="padding:0;"> ${ this . languages [ choice ] . name } </div> ` ;
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
setDictionary ( lang ) {
this . dictionary = BDFDB . loadData ( lang , this , "owndics" ) || [ ] ;
this . killLanguageToast ( ) ;
this . languageToast = BDFDB . showToast ( "Grabbing dictionary (" + this . languages [ lang ] . name + "). Please wait" , { timeout : 0 } ) ;
this . languageToast . interval = setInterval ( ( ) => {
this . languageToast . textContent = this . languageToast . textContent . indexOf ( "....." ) > - 1 ? "Grabbing dictionary (" + this . languages [ lang ] . name + "). Please wait" : this . languageToast . textContent + "." ;
} , 500 ) ;
this . languageToast . lang = lang
require ( "request" ) ( "https://mwittrien.github.io/BetterDiscordAddons/Plugins/SpellCheck/dic/" + lang + ".dic" , ( error , response , result ) => {
if ( error || ( response && result . toLowerCase ( ) . indexOf ( "<!doctype html>" ) > - 1 ) ) {
this . killLanguageToast ( ) ;
BDFDB . showToast ( "Failed to grab dictionary (" + this . languages [ lang ] . name + ")." , { type : "error" } ) ;
}
else if ( response && this . languageToast . lang == lang ) {
this . langDictionary = result . split ( "\n" ) ;
this . dictionary = this . langDictionary . concat ( this . dictionary ) ;
this . dictionary = this . dictionary . map ( word => word . toLowerCase ( ) ) ;
this . killLanguageToast ( ) ;
BDFDB . showToast ( "Successfully grabbed dictionary (" + this . languages [ lang ] . name + ")." , { type : "success" } ) ;
}
} ) ;
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
killLanguageToast ( ) {
if ( this . languageToast && typeof this . languageToast . close == "function" ) {
clearInterval ( this . languageToast . interval ) ;
this . languageToast . close ( ) ;
}
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
spellCheckText ( string ) {
var htmlString = [ ] ;
2019-02-19 13:34:09 +01:00
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 ) ? "spelling-error" : "nospelling-error" } " style="color: transparent !important; text-shadow: none !important;"> ${ BDFDB . encodeToHTML ( word ) } </label> ${ hasnewline ? "\n" : "" } ` ) ;
2018-10-11 10:21:26 +02:00
} ) ;
2019-02-19 13:34:09 +01:00
return htmlString . join ( " " ) . replace ( /\n /g , "\n" ) ;
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
isWordNotInDictionary ( word ) {
var wordLow = word . toLowerCase ( ) ;
var wordWithoutSymbols = wordLow . replace ( /[0-9\µ\@\$\£\€\¥\¢\²\³\>\<\|\,\;\.\:\_\#\+\*\~\?\¿\\\´ \`\}\=\]\)\[\(\{\/\&\%\§\"\!\¡\^\°\n\t\r]/g , "" ) ;
return ( wordLow . indexOf ( "http://" ) != 0 && wordLow . indexOf ( "https://" ) != 0 && wordWithoutSymbols && Array . isArray ( this . dictionary ) && this . dictionary . length > 0 && ! this . dictionary . includes ( wordLow ) && ! this . dictionary . includes ( wordWithoutSymbols ) ) ;
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
getSimilarWords ( word ) {
var maxAmount = BDFDB . getData ( "maxSimilarAmount" , this , "amounts" ) , similarWords = [ ] ;
if ( maxAmount > 0 ) {
var sameLetterDic = this . dictionary . filter ( string => string . indexOf ( word . toLowerCase ( ) . charAt ( 0 ) ) == 0 ? string : null ) ;
var similarities = { } ;
for ( let string of sameLetterDic ) {
let value = this . wordSimilarity ( word , string ) ;
if ( ! similarities [ value ] ) similarities [ value ] = [ ] ;
similarities [ value ] . push ( string ) ;
}
var amount = 0 ;
for ( let value of Object . keys ( similarities ) . sort ( ) . reverse ( ) ) {
for ( let similarWord of similarities [ value ] ) {
if ( amount < maxAmount && ! similarWords . includes ( similarWord ) ) {
similarWords . push ( similarWord ) ;
amount ++ ;
}
if ( amount >= maxAmount ) break ;
}
if ( amount >= maxAmount ) break ;
}
}
return similarWords ;
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
wordSimilarity ( a , b ) {
var 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 ;
}
let result = 0 ;
let 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 ;
}
2019-01-26 22:45:19 +01:00
2018-10-11 10:21:26 +02:00
setLabelsByLanguage ( ) {
switch ( BDFDB . getDiscordLanguage ( ) . id ) {
case "hr" : //croatian
return {
context _spellcheck _text : "Dodaj u rječnik" ,
context _similarwords _text : "Pretraga sličnih riječi..." ,
similarwordssubmenu _none _text : "Nema sličnih riječi" ,
toast _wordadd _text : "Riječ ${word} dodana je u rječnik ${dicname}."
} ;
case "da" : //danish
return {
context _spellcheck _text : "Tilføj til ordbog" ,
context _similarwords _text : "Søg lignende ord..." ,
similarwordssubmenu _none _text : "Ingen lignende ord" ,
toast _wordadd _text : "Ord ${word} tilføjet til ordbog ${dicname}."
} ;
case "de" : //german
return {
context _spellcheck _text : "Zum Wörterbuch hinzufügen" ,
context _similarwords _text : "Ähnliche Wörter suchen..." ,
similarwordssubmenu _none _text : "Keine ähnlichen Wörter" ,
toast _wordadd _text : "Wort ${word} wurde zum Wörterbuch ${dicname} hinzugefügt."
} ;
case "es" : //spanish
return {
context _spellcheck _text : "Agregar al diccionario" ,
context _similarwords _text : "Buscar palabras similares..." ,
similarwordssubmenu _none _text : "No hay palabras similares" ,
toast _wordadd _text : "Se agregó la palabra ${word} al diccionario ${dicname}."
} ;
case "fr" : //french
return {
context _spellcheck _text : "Ajouter au dictionnaire" ,
context _similarwords _text : "Chercher des mots similaires..." ,
similarwordssubmenu _none _text : "Pas de mots similaires" ,
toast _wordadd _text : "Le mot ${word} a été ajouté au dictionnaire ${dicname}."
} ;
case "it" : //italian
return {
context _spellcheck _text : "Aggiungi al dizionario" ,
context _similarwords _text : "Cerca parole simili..." ,
similarwordssubmenu _none _text : "Nessuna parola simile" ,
toast _wordadd _text : "Parola ${word} aggiunta al dizionario ${dicname}."
} ;
case "nl" : //dutch
return {
context _spellcheck _text : "Toevoegen aan woordenboek" ,
context _similarwords _text : "Zoek vergelijkbare woorden..." ,
similarwordssubmenu _none _text : "Geen vergelijkbare woorden" ,
toast _wordadd _text : "Word ${word} toegevoegd aan woordenboek ${dicname}."
} ;
case "no" : //norwegian
return {
context _spellcheck _text : "Legg til i ordbok" ,
context _similarwords _text : "Søk lignende ord..." ,
similarwordssubmenu _none _text : "Ingen lignende ord" ,
toast _wordadd _text : "Ord ${word} legges til ordbok ${dicname}."
} ;
case "pl" : //polish
return {
context _spellcheck _text : "Dodaj do słownika" ,
context _similarwords _text : "Wyszukaj podobne słowa..." ,
similarwordssubmenu _none _text : "Brak podobnych słów" ,
toast _wordadd _text : "Słowo ${word} dodane do słownika ${dicname}."
} ;
case "pt-BR" : //portuguese (brazil)
return {
context _spellcheck _text : "Adicionar ao dicionário" ,
context _similarwords _text : "Pesquisar palavras similares..." ,
similarwordssubmenu _none _text : "Sem palavras semelhantes" ,
toast _wordadd _text : "Palavra ${word} adicionado ao dicionário ${dicname}."
} ;
case "fi" : //finnish
return {
context _spellcheck _text : "Lisää sanakirjaan" ,
context _similarwords _text : "Hae samankaltaisia sanoja..." ,
similarwordssubmenu _none _text : "Ei vastaavia sanoja" ,
toast _wordadd _text : "Sana ${word} lisättiin sanakirjaan ${dicname}."
} ;
case "sv" : //swedish
return {
context _spellcheck _text : "Lägg till i ordbok" ,
context _similarwords _text : "Sök liknande ord..." ,
similarwordssubmenu _none _text : "Inga liknande ord" ,
toast _wordadd _text : "Ord ${word} läggs till ordbok ${dicname}."
} ;
case "tr" : //turkish
return {
context _spellcheck _text : "Sözlükye Ekle" ,
context _similarwords _text : "Benzer Kelimeler Ara..." ,
similarwordssubmenu _none _text : "Benzer kelime yoktur" ,
toast _wordadd _text : "Sözcük ${word} sözlük ${dicname}'ye eklendi."
} ;
case "cs" : //czech
return {
context _spellcheck _text : "Přidat do slovníku" ,
context _similarwords _text : "Hledat podobné výrazy..." ,
similarwordssubmenu _none _text : "Žádné podobné slova" ,
toast _wordadd _text : "Slovo ${word} bylo přidáno do slovníku ${dicname}."
} ;
case "bg" : //bulgarian
return {
context _spellcheck _text : "Добави в речника" ,
context _similarwords _text : "Търсене на подобни думи..." ,
similarwordssubmenu _none _text : "Няма подобни думи" ,
toast _wordadd _text : "Думата ${word} е добавена към речника ${dicname}."
} ;
case "ru" : //russian
return {
context _spellcheck _text : "Добавить в словарь" ,
context _similarwords _text : "Поиск похожих слов..." ,
similarwordssubmenu _none _text : "Нет похожих слов" ,
toast _wordadd _text : "Слово ${word} добавлено в словарь ${dicname}."
} ;
case "uk" : //ukrainian
return {
context _spellcheck _text : "Додати до словника" ,
context _similarwords _text : "Шукати схожі слова..." ,
similarwordssubmenu _none _text : "Немає подібних слів" ,
toast _wordadd _text : "Словник ${word} додається до словника ${dicname}."
} ;
case "ja" : //japanese
return {
context _spellcheck _text : "辞書に追加" ,
context _similarwords _text : "類似のワードを検索..." ,
similarwordssubmenu _none _text : "類似の単語はありません" ,
toast _wordadd _text : "単語 ${word} が辞書 ${dicname} に追加されました。"
} ;
case "zh-TW" : //chinese (traditional)
return {
context _spellcheck _text : "添加到詞典" ,
context _similarwords _text : "搜索類似的單詞..." ,
similarwordssubmenu _none _text : "沒有類似的詞" ,
toast _wordadd _text : "單詞 ${word} 添加到字典 ${dicname}。"
} ;
case "ko" : //korean
return {
context _spellcheck _text : "사전에 추가" ,
context _similarwords _text : "비슷한 단어 검색..." ,
similarwordssubmenu _none _text : "유사한 단어 없음" ,
toast _wordadd _text : "단어 ${word} 사전 ${dicname} 에 추가되었습니다."
} ;
default : //default: english
return {
context _spellcheck _text : "Add to Dictionay" ,
context _similarwords _text : "Search similar Words..." ,
similarwordssubmenu _none _text : "No similar Words" ,
toast _wordadd _text : "Word ${word} added to dictionary ${dicname}."
} ;
}
}
}