{"version":3,"sources":["constants/ActionTypes.js","api/index.js","actions/Messages.js","components/Chatfield/Chatfield.jsx","components/Loginfield/Loginfield.jsx","actions/User.js","components/ChatViewField/Message/Message.jsx","components/ChatViewField/ChatViewField.jsx","App.jsx","reducers/messages.js","reducers/user.js","reducers/index.js","index.js"],"names":["MessageTypes","UserTypes","ApiTypes","WHEN","addMessage","messageData","a","client","send","JSON","stringify","Nuke","headers","TYPE","AddMessage","dispatch","api","payload","body","type","console","trace","withStyles","root","background","input","color","classes","user","useSelector","state","useDispatch","HandleSubmit","evt","bIsDesktop","preventDefault","sMsgContent","target","value","length","id","displayName","message","content","date","Date","style","height","onSubmit","TextField","className","inputProps","autoFocus","multiline","fullWidth","onKeyDown","keyCode","shiftKey","HandleEnter","placeholder","Loginfield","userData","log","err","Message","data","padding","display","paddingBottom","split","map","str","i","margin","ChatViewField","divRef","useRef","messages","useEffect","current","scrollIntoView","slice","reverse","index","ref","w3cwebsocket","App","onmessage","msg","RenderNewMessage","parse","Grid","container","item","xs","sm","lg","xl","onClick","Messages","action","temp","pop","User","combineReducers","composeEnhancers","window","__REDUX_DEVTOOLS_EXTENSION_COMPOSE__","compose","store","createStore","reducers","applyMiddleware","thunk","ReactDOM","render","document","getElementById"],"mappings":"wSAAaA,EACA,YADAA,EAEG,eAFHA,EAGE,cAGFC,EACJ,QADIA,EAEH,SAGGC,EAEL,OAFKA,EAGH,SAIGC,EACJ,QADIA,EAEH,SCMGC,EAAU,uCAAG,WAAOC,GAAP,SAAAC,EAAA,sDAExBC,EAAOC,KAAKC,KAAKC,UAAUL,IAFH,2CAAH,sDAQVM,EAAI,uCAAG,sBAAAL,EAAA,sDAElBC,EAAOC,KAAKC,KAAKC,UAAU,CACzBE,QAAS,CACPC,KAAMX,MAJQ,2CAAH,qDCjBXY,EAAa,SAACT,GAAD,8CAAiB,WAAOU,GAAP,SAAAT,EAAA,+EAI1BU,EAAeX,GAJW,OAMhCU,EAAS,CAAEE,QAASZ,EAAYa,KAAMC,KAAMnB,IANZ,+CAUhCoB,QAAQC,MAAR,MAVgC,wDAAjB,uD,8BC2EJC,cArFA,CACbC,KAAM,CACJC,WAAY,WAEdC,MAAO,CACLC,MAAO,YAgFIJ,EA5EG,SAAC,GAClB,IADoBK,EACrB,EADqBA,QAGbC,EAAOC,aAAY,SAACC,GAAD,OAAWA,EAAMF,QACpCb,EAAWgB,cAIXC,EAAe,SAACC,GACrB,IAD0BC,EAC3B,wDAEED,EAAIE,iBAEJ,IAAMC,EAAcF,EAAaD,EAAII,OAAOC,MAAQL,EAAII,OAAO,GAAGC,MAElE,KAAIF,EAAYG,OAAS,GAAzB,CACA,IAAMlC,EAAc,CAClBO,QAAS,CACPC,KAAMX,GAERgB,KAAM,CACJU,KAAM,CACJY,GAAIZ,EAAKY,GACTC,YAAab,EAAKa,aAEpBC,QAAS,CACPC,QAASP,EACTQ,KAAM,IAAIC,QAMhB9B,EAASD,EAAWT,IAEhB6B,GAEFD,EAAII,OAAOC,MAAQ,GACnBL,EAAII,OAAOS,MAAMC,OAAS,SAI1Bd,EAAII,OAAO,GAAGC,MAAQ,GACtBL,EAAII,OAAO,GAAGS,MAAMC,OAAS,UAcjC,OACE,qBAAKP,GAAG,YAAR,SACE,sBAAMQ,SAAU,SAACf,GAAD,OAASD,EAAaC,IAAtC,SACE,cAACgB,EAAA,EAAD,CACEC,UAAWvB,EAAQJ,KACnB4B,WAAY,CACVD,UAAWvB,EAAQF,OAErB2B,WAAW,EACXC,WAAW,EACXC,WAAW,EACXC,UAAW,SAACtB,GAAD,OApBC,SAACA,GAEC,KAAhBA,EAAIuB,SAAmBvB,EAAIwB,WAE7BzB,EAAaC,GAAK,GAClBA,EAAII,OAAOC,MAAQ,IAeKoB,CAAYzB,IAChC0B,YAAY,eC3DPC,EAvBI,WAGjB,IAAM7C,EAAWgB,cACXC,EAAe,SAACC,GCLN,IAAC4B,EDQfzC,QAAQ0C,IAAI7B,EAAII,OAAO,GAAGC,OAC1BvB,GCTe8C,EDSI5B,EAAII,OAAO,GAAGC,MCTnB,uCAAc,WAAOvB,GAAP,SAAAT,EAAA,sDAE9B,IAEES,EAAS,CAAEE,QAAS4C,EAAU1C,KAAMlB,IAEtC,MAAO8D,GAEL3C,QAAQC,MAAM0C,GARc,2CAAd,wDDUd9B,EAAIE,kBAGN,OACE,uFAEE,sBAAMa,SAAU,SAACf,GAAD,OAASD,EAAaC,IAAtC,SACE,uBAAOmB,WAAS,EAACjC,KAAK,OAAOwC,YAAY,cEDlCK,EAhBC,SAAC,GAChB,IADkBC,EACnB,EADmBA,KAIjB,OACE,sBAAKnB,MAAO,CAAEoB,QAAS,kBAAmBC,QAAS,SAAnD,UACE,qBAAK3B,GAAG,WAAWM,MAAO,CAAEsB,cAAe,OAA3C,SACGH,EAAKrC,KAAKa,cAEb,qBAAKD,GAAG,cAAR,SACGyB,EAAKvB,QAAQC,QAAQ0B,MAAM,MAAMC,KAAI,SAACC,EAAKC,GAAN,OAAY,gCAAeD,EAAIhC,OAAkB,GAAT,uBAAY,mBAAGO,MAAO,CAAE2B,OAAQ,GAApB,SAA0BF,MAAxDC,YCwBrDE,EA/BO,WAGpB,IAAMC,EAASC,iBAAO,MAEhBC,EAAWhD,aAAY,SAACC,GAAD,OAAWA,EAAM+C,YAO9C,OALAC,qBAAU,WAERH,EAAOI,QAAQC,eAAe,OAI7BH,EAAStC,OAQN,sBAAKO,MAAO,CAACoB,QAAQ,eAArB,UACGW,EAASI,MAAM,GAAGC,UAAUZ,KAAI,SAAC5B,EAASyC,GAAV,OAC/B,cAAC,EAAD,CAAqBlB,KAAMvB,GAAbyC,MAEhB,qBAAKC,IAAKT,OAXd,qBAAKS,IAAKT,EAAQ7B,MAAO,CAACoB,QAAQ,eAAlC,SACE,0ECNK3D,G,MAAS,IAAI8E,eAAa,uCA0CxBC,EAxCH,WAEV,IAKMvE,EAAWgB,cAEjBxB,EAAOgF,UAAY,SAACC,GAElBzE,ENIqB,SAACyE,GAAD,8CAAS,WAAOzE,GAAP,SAAAT,EAAA,oEAItBkF,EAAI5E,QAJkB,cAMvBT,EANuB,SASvBA,EATuB,wBAO1BY,EAAS,CAAEE,QAASuE,EAAItE,KAAMC,KAAMnB,IAPV,mCAU1Be,EAAS,CAAEE,QAASuE,EAAItE,KAAMC,KAAMnB,IAVV,mCAa1BoB,QAAQ0C,IAAI,iEAbc,+EAmB9B1C,QAAQC,MAAR,MAnB8B,0DAAT,sDMJZoE,CAAiBhF,KAAKiF,MAAMF,EAAIvB,SAG3C,IAAMrC,EAAOC,aAAY,SAACC,GAAD,OAAWA,EAAMF,QAE1C,OACE,8BACIA,EAAKa,YACL,eAACkD,EAAA,EAAD,CAAMC,WAAS,EAAC9C,MAAO,CAAEC,OAAQ,QAAjC,UACE,cAAC4C,EAAA,EAAD,CAAME,MAAI,EAACC,GAAI,EAAGC,GAAI,EAAGC,GAAI,EAAGC,GAAI,EAApC,SACE,8BACE,oDACqB,uBACnB,wBAAQC,QAAS,WApB7BvF,KAoBY,6BAIN,eAACgF,EAAA,EAAD,CAAME,MAAI,EAACC,GAAI,GAAIC,GAAI,GAAIC,GAAI,GAAIC,GAAI,GAAvC,UACE,qBAAKzD,GAAG,cAAR,SACE,cAAC,EAAD,MAEF,cAAC,EAAD,UAde,cAAC,EAAD,O,QCHZ2D,EA9BE,WAChB,IADiBtB,EAClB,uDAD6B,GAAIuB,EACjC,uCAEMC,EAAI,YAAOxB,GAcf,OAXIwB,EAAK9D,QAAU,KAAK8D,EAAKC,MAWrBF,EAAOjF,MAEb,KAAKnB,EACH,MAAM,GAAN,mBAAWoG,EAAOnF,SAAlB,YAA8BoF,IAChC,KAAKrG,EAEL,KAAKA,EACH,MAAM,CAAEoG,EAAOnF,SAAf,mBAA2BoF,IAC7B,QACE,OAAOA,ICVEE,EAfF,WAIZ,IAJa3E,EAId,uDAJqB,CACnBY,GAAI,GACJC,YAAa,IACZ2D,EACH,uCACE,OAAQA,EAAOjF,MAEb,KAAKlB,EACH,MAAQ,CAAEuC,GAAI,IAAKC,YAAa2D,EAAOnF,SACzC,KAAKhB,EACH,MACF,QACE,OAAO2B,ICVE4E,cAAgB,CAAE3B,WAAUjD,SCOrC6E,G,MAAmBC,OAAOC,sCAAwCC,KAE3DC,EAAQC,YAAYC,EAAgCN,EAAiBO,YAAgBC,OAElGC,IAASC,OACP,cAAC,IAAD,CAAUN,MAAOA,EAAjB,SACE,cAAC,EAAD,MAEFO,SAASC,eAAe,W","file":"static/js/main.10743629.chunk.js","sourcesContent":["export const MessageTypes = {\r\n FETCH_ALL: \"FETCH_ALL\",\r\n POST_MESSAGE: \"POST_MESSAGE\",\r\n NEW_MESSAGE: \"NEW_MESSAGE\"\r\n}\r\n\r\nexport const UserTypes = {\r\n LOGIN: \"LOGIN\",\r\n LOGOUT: \"LOGOUT\"\r\n}\r\n\r\nexport const ApiTypes = {\r\n GET: \"GET\",\r\n POST: \"POST\",\r\n DELETE: \"DELETE\",\r\n PATCH: \"PATCH\"\r\n}\r\n\r\nexport const WHEN = {\r\n FIRST: \"FIRST\",\r\n NORMAL: \"NORMAL\"\r\n}","import { client } from \"../App\";\r\n\r\nimport { ApiTypes } from \"./../constants/ActionTypes.js\"\r\n//const url = \"https://scuffedwebsite.com/chat/api/messages/\";\r\n\r\n\r\n// Request recent messages from database using custom header\r\nexport const fetchMessages = async () =>\r\n{\r\n try\r\n {\r\n client.send(JSON.stringify({\r\n headers: {\r\n TYPE: ApiTypes.GET\r\n }\r\n }))\r\n }\r\n catch (err)\r\n {\r\n console.trace(err);\r\n }\r\n}\r\n\r\n\r\n// Add message to database and broadcast to other clients using custom header defined in \r\n// messageData\r\nexport const addMessage = async (messageData) =>\r\n{\r\n client.send(JSON.stringify(messageData));\r\n}\r\n\r\n\r\n// Delete messages in database using custom header\r\n// This should probably be removed but it's useful for testing purposes\r\nexport const Nuke = async () =>\r\n{\r\n client.send(JSON.stringify({\r\n headers: {\r\n TYPE: ApiTypes.DELETE\r\n }\r\n }));\r\n}","import { MessageTypes, WHEN } from \"../constants/ActionTypes\";\r\nimport * as api from \"./../api/index\";\r\n\r\nconst GetMessages = () => async (dispatch) =>\r\n{\r\n try\r\n {\r\n const data = await api.fetchMessages();\r\n console.log(data);\r\n dispatch({ payload: data, type: MessageTypes.FETCH_ALL });\r\n }\r\n catch (err)\r\n {\r\n console.trace(err);\r\n }\r\n}\r\n\r\nconst AddMessage = (messageData) => async (dispatch) =>\r\n{\r\n try\r\n {\r\n await api.addMessage(messageData);\r\n\r\n dispatch({ payload: messageData.body, type: MessageTypes.POST_MESSAGE });\r\n }\r\n catch (err)\r\n {\r\n console.trace(err);\r\n }\r\n}\r\n\r\nconst RenderNewMessage = (msg) => async (dispatch) =>\r\n{\r\n try\r\n {\r\n switch (msg.headers)\r\n {\r\n case WHEN.FIRST:\r\n dispatch({ payload: msg.body, type: MessageTypes.FETCH_ALL });\r\n break;\r\n case WHEN.NORMAL:\r\n dispatch({ payload: msg.body, type: MessageTypes.NEW_MESSAGE });\r\n break;\r\n default:\r\n console.log(\"(´。_。`) that wasn't supposed to happen\");\r\n break;\r\n }\r\n }\r\n catch (err)\r\n {\r\n console.trace(err);\r\n }\r\n}\r\n\r\n\r\n\r\nexport { GetMessages, AddMessage, RenderNewMessage };\r\n\r\n","import { useDispatch, useSelector } from \"react-redux\";\r\nimport { ApiTypes } from \"../../constants/ActionTypes\";\r\n\r\nimport { AddMessage } from \"./../../actions/Messages\"\r\n\r\nimport { TextField, withStyles } from \"@material-ui/core\"\r\nimport \"./chatfield.css\"\r\nconst styles = {\r\n root: {\r\n background: \"#222222\"\r\n },\r\n input: {\r\n color: \"#21D7EA\"\r\n }\r\n}\r\n\r\nconst Chatfield = ({ classes }) =>\r\n{\r\n\r\n const user = useSelector((state) => state.user);\r\n const dispatch = useDispatch();\r\n\r\n\r\n\r\n const HandleSubmit = (evt, bIsDesktop = false) =>\r\n {\r\n\r\n evt.preventDefault();\r\n\r\n const sMsgContent = bIsDesktop ? evt.target.value : evt.target[0].value;\r\n\r\n if (sMsgContent.length < 1) return;\r\n const messageData = {\r\n headers: {\r\n TYPE: ApiTypes.POST\r\n },\r\n body: {\r\n user: {\r\n id: user.id,\r\n displayName: user.displayName\r\n },\r\n message: {\r\n content: sMsgContent,\r\n date: new Date()\r\n }\r\n }\r\n }\r\n\r\n\r\n dispatch(AddMessage(messageData));\r\n\r\n if (bIsDesktop)\r\n {\r\n evt.target.value = \"\";\r\n evt.target.style.height = \"auto\";\r\n }\r\n else\r\n {\r\n evt.target[0].value = \"\";\r\n evt.target[0].style.height = \"auto\";\r\n }\r\n\r\n }\r\n\r\n const HandleEnter = (evt) =>\r\n {\r\n if (evt.keyCode === 13 && !evt.shiftKey)\r\n {\r\n HandleSubmit(evt, true);\r\n evt.target.value = \"\";\r\n };\r\n }\r\n\r\n return (\r\n
\r\n
HandleSubmit(evt)} >\r\n HandleEnter(evt)}\r\n placeholder=\"sus\"\r\n />\r\n \r\n
\r\n )\r\n}\r\n\r\nexport default withStyles(styles)(Chatfield);\r\n","import { useDispatch } from \"react-redux\";\r\nimport { LoginUser } from \"../../actions/User\";\r\n\r\nconst Loginfield = () =>\r\n{\r\n\r\n const dispatch = useDispatch();\r\n const HandleSubmit = (evt) =>\r\n {\r\n\r\n console.log(evt.target[0].value)\r\n dispatch(LoginUser(evt.target[0].value))\r\n evt.preventDefault();\r\n\r\n }\r\n return (\r\n
\r\n This is super fucking scuffed atm just choose a name\r\n
HandleSubmit(evt)}>\r\n \r\n
\r\n\r\n
\r\n )\r\n}\r\n\r\nexport default Loginfield;","import { UserTypes } from \"../constants/ActionTypes\";\r\n\r\nconst LoginUser = (userData) => async (dispatch) =>\r\n{\r\n try\r\n {\r\n dispatch({ payload: userData, type: UserTypes.LOGIN });\r\n }\r\n catch (err)\r\n {\r\n console.trace(err);\r\n }\r\n}\r\n\r\nconst LogoutUser = () => async (dispatch) =>\r\n{\r\n\r\n try\r\n {\r\n\r\n }\r\n catch (err)\r\n {\r\n console.trace(err);\r\n }\r\n}\r\n\r\nexport { LoginUser, LogoutUser }","\r\n\r\nconst Message = ({ data }) =>\r\n{\r\n // I could make this pretty... but eh (ノ◕ヮ◕)ノ*:・゚✧\r\n\r\n return (\r\n
\r\n
\r\n {data.user.displayName}\r\n
\r\n
\r\n {data.message.content.split(\"\\n\").map((str, i) =>
{!str.length ?
: \"\"}

{str}

)}\r\n
\r\n
\r\n );\r\n}\r\n\r\nexport default Message;\r\n","import { useRef, useEffect } from \"react\";\r\nimport { useSelector } from \"react-redux\";\r\nimport Message from \"./Message/Message\";\r\n\r\n\r\nconst ChatViewField = () =>\r\n{\r\n\r\n const divRef = useRef(null);\r\n\r\n const messages = useSelector((state) => state.messages);\r\n\r\n useEffect(() =>\r\n {\r\n divRef.current.scrollIntoView({});\r\n })\r\n\r\n return (\r\n !messages.length ? (\r\n
\r\n

\r\n No messages(*^-^*)\r\n

\r\n
\r\n )\r\n : (\r\n
\r\n {messages.slice(0).reverse().map((message, index) => (\r\n \r\n ))}\r\n
\r\n
\r\n )\r\n )\r\n}\r\n\r\nexport default ChatViewField;","import { useSelector, useDispatch } from \"react-redux\";\r\nimport { w3cwebsocket } from \"websocket\";\r\n\r\nimport { Grid } from \"@material-ui/core\";\r\n\r\nimport { RenderNewMessage } from \"./actions/Messages\";\r\nimport { Nuke } from \"./api/index\";\r\n\r\nimport Chatfield from \"./components/Chatfield/Chatfield\";\r\nimport Loginfield from \"./components/Loginfield/Loginfield\";\r\nimport ChatViewField from \"./components/ChatViewField/ChatViewField\";\r\n\r\nimport \"./app.css\"\r\n\r\nexport const client = new w3cwebsocket(\"wss://scuffedwebsite.com/chat/api/\")\r\n\r\nconst App = () =>\r\n{\r\n const DeleteMessages = () =>\r\n {\r\n Nuke();\r\n }\r\n\r\n const dispatch = useDispatch();\r\n\r\n client.onmessage = (msg) =>\r\n {\r\n dispatch(RenderNewMessage(JSON.parse(msg.data)));\r\n }\r\n\r\n const user = useSelector((state) => state.user);\r\n\r\n return (\r\n
\r\n {!user.displayName ? : (\r\n \r\n \r\n
\r\n

\r\n Sidebar placeholder
\r\n \r\n

\r\n
\r\n
\r\n \r\n
\r\n \r\n
\r\n \r\n
\r\n
\r\n )}\r\n
\r\n )\r\n}\r\n\r\nexport default App;\r\n\r\n","import { MessageTypes } from \"./../constants/ActionTypes\";\r\nconst Messages = (messages = [], action) =>\r\n{\r\n\r\n let temp = [...messages];\r\n\r\n // Arbitrary max length of temporary view\r\n if (temp.length >= 100) temp.pop();\r\n\r\n /* \r\n Literally the only difference between POST_MESSAGE and NEW_MESSAGE\r\n is what causes them to trigger, should probably consolidate them\r\n into one term\r\n\r\n POST_MESSAGE fires when the client itself creates a new message object\r\n while NEW_MESSAGE fires whenever a new message is received from the websocket\r\n connection, meaning someone else has posted a new message object\r\n */\r\n switch (action.type)\r\n {\r\n case MessageTypes.FETCH_ALL:\r\n return [...action.payload, ...temp];\r\n case MessageTypes.POST_MESSAGE:\r\n return [action.payload, ...temp];\r\n case MessageTypes.NEW_MESSAGE:\r\n return [action.payload, ...temp];\r\n default:\r\n return temp;\r\n }\r\n}\r\n\r\nexport default Messages;","import { UserTypes } from \"./../constants/ActionTypes\"\r\n\r\nconst User = (user = {\r\n id: \"\",\r\n displayName: \"\"\r\n}, action) =>\r\n{\r\n switch (action.type)\r\n {\r\n case UserTypes.LOGIN:\r\n return ({ id: \"0\", displayName: action.payload });\r\n case UserTypes.LOGOUT:\r\n break;\r\n default:\r\n return user;\r\n }\r\n}\r\nexport default User;","import { combineReducers } from \"redux\";\r\nimport messages from \"./messages\";\r\nimport user from \"./user\"\r\n\r\nexport default combineReducers({ messages, user });","import React from 'react';\r\nimport ReactDOM from 'react-dom';\r\nimport { createStore, applyMiddleware, compose } from \"redux\";\r\nimport { Provider } from \"react-redux\";\r\nimport thunk from \"redux-thunk\";\r\n\r\nimport App from \"./App\"\r\nimport reducers from \"./reducers/index\"\r\n\r\nimport \"./index.css\";\r\n\r\nconst composeEnhancers = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__ || compose;\r\n\r\nexport const store = createStore(reducers, /* preloadedState, */ composeEnhancers(applyMiddleware(thunk)));\r\n\r\nReactDOM.render(\r\n \r\n \r\n ,\r\n document.getElementById(\"root\")\r\n);\r\n\r\n"],"sourceRoot":""}