From 937348a501a41f348fafc8fdaca69e2cfbaeded3 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Sun, 28 Nov 2021 06:32:32 +0700 Subject: [PATCH 01/19] feat(survey) - init survey page comp. - add link to navbar - add route --- src/components/survey/AddChoice.jsx | 0 src/components/survey/CreateSurvey.jsx | 128 +++++++++++++++++++++++++ src/components/utility/Navbar.jsx | 10 ++ src/routes/App.jsx | 2 + 4 files changed, 140 insertions(+) create mode 100644 src/components/survey/AddChoice.jsx create mode 100644 src/components/survey/CreateSurvey.jsx diff --git a/src/components/survey/AddChoice.jsx b/src/components/survey/AddChoice.jsx new file mode 100644 index 0000000..e69de29 diff --git a/src/components/survey/CreateSurvey.jsx b/src/components/survey/CreateSurvey.jsx new file mode 100644 index 0000000..e72c4ef --- /dev/null +++ b/src/components/survey/CreateSurvey.jsx @@ -0,0 +1,128 @@ +import axios from 'axios'; +import React, { useState, useEffect } from 'react'; +import { useInput } from '../../helpers/hooks/input-hook'; +import { authHeader } from '../../helpers/services/auth.service'; +import { API_URL } from '../../config/keys'; +import '../../styles/thread/Form.css'; + +export default function CreateSurvey(props) { + const { history } = props; + + const back = () => { + history.goBack(); + }; + + const redirect = (path) => { + history.push(path); + }; + + const { value: title, bind: bindTitle, reset: resetTitle } = useInput(''); + const { value: body, bind: bindBody, reset: resetBody } = useInput(''); + const { value: topicId, bind: bindTopicId, reset: resetTopicId } = useInput( + 0 + ); + + const [listTopic, setlistTopic] = useState([]); + const [message, setMessage] = useState(''); + + useEffect(() => { + async function getListTopic() { + const allTopic = await (await axios.get(`${API_URL}/topics`))?.data?.data; + setlistTopic(allTopic); + } + getListTopic(); + }, []); + + const handleSubmit = async (event) => { + event.preventDefault(); + try { + console.log(body); + const thisThread = await axios.post( + `${API_URL}/surveys`, + { + thread: { + content: body, + points: 0, + title: title, + topic_id: topicId, + }, + }, + { + headers: authHeader(), + } + ); + const { data } = thisThread?.data; + redirect(`/topic/${data?.topic_name}/${data?.id}/page/1`); + } catch (error) { + setMessage("Thread's title already exist in the selected topic"); + } + resetBody(); + resetTitle(); + resetTopicId(); + }; + + return ( + <div className="formComponentContainer"> + <div className="back" onClick={back}> + <i className="fas fa-angle-left"></i> + <h5>Back</h5> + </div> + <div className="header"> + <h1> + <b>Create a Survey</b> + </h1> + </div> + <div className="formSection"> + <div> + {message && ( + <div> + <div className="alert alert-danger" role="alert"> + <p>{message}</p> + </div> + </div> + )} + <form onSubmit={handleSubmit}> + <div className="formContainer"> + <label for="title">Title</label> + <input + type="text" + className="title" + name="title" + placeholder="Your survey's title" + required="false" + {...bindTitle} + /> + <label>Topic</label> + <select + className="topic" + name="topic" + required="false" + {...bindTopicId} + > + <option value="" /> + {listTopic.map((topic) => ( + <option value={topic.id}>{topic.name}</option> + ))} + </select> + <label for="body">Body</label> + <textarea + className="body" + name="body" + placeholder="Your thread's body" + required="false" + {...bindBody} + /> + <div className="buttonContainer"> + <input + type="submit" + className="buttonSubmit" + value="Create Thread" + /> + </div> + </div> + </form> + </div> + </div> + </div> + ); +} diff --git a/src/components/utility/Navbar.jsx b/src/components/utility/Navbar.jsx index a6f8038..3d0a64b 100644 --- a/src/components/utility/Navbar.jsx +++ b/src/components/utility/Navbar.jsx @@ -67,6 +67,16 @@ const Navbar = (props) => { <b>Threads</b> </NavLink> </li> + <li className="nav-item mr-auto"> + <NavLink + exact + activeClassName="navbar--active" + className="nav-link" + to="survey/page/1" + > + <b>Surveys</b> + </NavLink> + </li> <li className="nav-item mr-auto"> <NavLink exact diff --git a/src/routes/App.jsx b/src/routes/App.jsx index 6f72f0c..a0172f7 100644 --- a/src/routes/App.jsx +++ b/src/routes/App.jsx @@ -13,6 +13,7 @@ import EditThread from '../components/thread/EditThread'; import Footer from '../components/utility/Footer'; import UpdateProfileForm from '../components/profile/UpdateProfileForm.jsx'; import EditComment from '../components/thread/EditComment'; +import CreateSurvey from '../components/survey/CreateSurvey'; import { BrowserRouter as Router, Route, @@ -52,6 +53,7 @@ function App() { <Route exact path="/login" component={LoginForm} /> <Route exact path="/create/topic" component={CreateTopicForm} /> <Route exact path="/create/thread" component={CreateThread} /> + <Route exact path="/create/survey" component={CreateSurvey} /> <Route exact path="/search/:input/page/:pageNumber" component={Search} /> <Route exact -- GitLab From 690d6f347e9acae927d2be79435f30917f40ab55 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 29 Nov 2021 07:32:26 +0700 Subject: [PATCH 02/19] feat(create-survey) - add duration widget - add html datetime to UTC converter util method --- .../{AddChoice.jsx => AddChoiceInput.jsx} | 0 src/components/survey/CreateSurvey.jsx | 34 +++++++++++++++++-- src/helpers/time-util.js | 7 +++- src/styles/survey/Form.css | 16 +++++++++ src/styles/thread/Form.css | 2 +- 5 files changed, 55 insertions(+), 4 deletions(-) rename src/components/survey/{AddChoice.jsx => AddChoiceInput.jsx} (100%) create mode 100644 src/styles/survey/Form.css diff --git a/src/components/survey/AddChoice.jsx b/src/components/survey/AddChoiceInput.jsx similarity index 100% rename from src/components/survey/AddChoice.jsx rename to src/components/survey/AddChoiceInput.jsx diff --git a/src/components/survey/CreateSurvey.jsx b/src/components/survey/CreateSurvey.jsx index e72c4ef..63edf5f 100644 --- a/src/components/survey/CreateSurvey.jsx +++ b/src/components/survey/CreateSurvey.jsx @@ -3,7 +3,10 @@ import React, { useState, useEffect } from 'react'; import { useInput } from '../../helpers/hooks/input-hook'; import { authHeader } from '../../helpers/services/auth.service'; import { API_URL } from '../../config/keys'; + import '../../styles/thread/Form.css'; +import '../../styles/survey/Form.css'; +import { datetimeToUTC } from '../../helpers/time-util'; export default function CreateSurvey(props) { const { history } = props; @@ -21,6 +24,8 @@ export default function CreateSurvey(props) { const { value: topicId, bind: bindTopicId, reset: resetTopicId } = useInput( 0 ); + const { value: endDate, bind: bindEndDate, reset: resetEndDate } = useInput(0) + const { value: endTime, bind: bindEndTime, reset: resetEndTime } = useInput(0) const [listTopic, setlistTopic] = useState([]); const [message, setMessage] = useState(''); @@ -33,6 +38,9 @@ export default function CreateSurvey(props) { getListTopic(); }, []); + useEffect(() => { + }, [endDate, endTime]) + const handleSubmit = async (event) => { event.preventDefault(); try { @@ -45,6 +53,7 @@ export default function CreateSurvey(props) { points: 0, title: title, topic_id: topicId, + datetime: datetimeToUTC(endDate, endTime), }, }, { @@ -59,6 +68,8 @@ export default function CreateSurvey(props) { resetBody(); resetTitle(); resetTopicId(); + resetEndDate(); + resetEndTime(); }; return ( @@ -104,14 +115,33 @@ export default function CreateSurvey(props) { <option value={topic.id}>{topic.name}</option> ))} </select> - <label for="body">Body</label> + <label for="choices">Choices</label> <textarea className="body" name="body" - placeholder="Your thread's body" + placeholder="Your thread's body"t required="false" {...bindBody} /> + <label for="duration">Duration</label> + <div className="d-flex justify-content-between"> + <input + className="end-date w-100" + name="end-date" + placeholder="End Date" + required="false" + type="date" + {...bindEndDate} + /> + <input + className="end-time w-100" + name="end-time" + placeholder="End Time" + required="false" + type="time" + {...bindEndTime} + /> + </div> <div className="buttonContainer"> <input type="submit" diff --git a/src/helpers/time-util.js b/src/helpers/time-util.js index 73b16ed..ca432dc 100644 --- a/src/helpers/time-util.js +++ b/src/helpers/time-util.js @@ -10,4 +10,9 @@ function translate(time) { } catch (error) {} } -export { translate }; +const datetimeToUTC = (endDate, endTime) => { + const date = new Date(`${endDate} ${endTime}`); + return date.toISOString(); +} + +export { translate, datetimeToUTC}; diff --git a/src/styles/survey/Form.css b/src/styles/survey/Form.css new file mode 100644 index 0000000..4e56ee4 --- /dev/null +++ b/src/styles/survey/Form.css @@ -0,0 +1,16 @@ +input.end-date, +input.end-time { + padding: 12px 16px; + border-radius: 8px; + border: 0; + box-shadow: 0px 0px 8px 1px rgba(0,0,0,0.25); + margin-bottom: 32px; +} + +input.end-date { + margin-right: 12px; +} + +input.end-time { + margin-left: 12px; +} \ No newline at end of file diff --git a/src/styles/thread/Form.css b/src/styles/thread/Form.css index a8909a5..7d1aa25 100644 --- a/src/styles/thread/Form.css +++ b/src/styles/thread/Form.css @@ -57,7 +57,7 @@ label { .topic { min-width: 303px; - max-width: 400px; + max-width: 3696px; font-family: "DM Sans"; height: 45px; padding: 12px 16px; -- GitLab From 404042c011c3179f3bc235bc03754e4f9351df4e Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 29 Nov 2021 12:41:51 +0700 Subject: [PATCH 03/19] feat(create-survey): finish ui impl. --- src/components/survey/CreateSurvey.jsx | 103 ++++++++++++++++++++----- src/components/survey/dummy.js | 21 +++++ src/images/add-choice.svg | 21 +++++ src/images/del-choice.svg | 21 +++++ src/styles/survey/Form.css | 40 +++++++++- 5 files changed, 187 insertions(+), 19 deletions(-) create mode 100644 src/components/survey/dummy.js create mode 100644 src/images/add-choice.svg create mode 100644 src/images/del-choice.svg diff --git a/src/components/survey/CreateSurvey.jsx b/src/components/survey/CreateSurvey.jsx index 63edf5f..2667650 100644 --- a/src/components/survey/CreateSurvey.jsx +++ b/src/components/survey/CreateSurvey.jsx @@ -8,6 +8,24 @@ import '../../styles/thread/Form.css'; import '../../styles/survey/Form.css'; import { datetimeToUTC } from '../../helpers/time-util'; +import AddChoiceSvg from '../../images/add-choice.svg'; +import DelChoiceSvg from '../../images/del-choice.svg'; + +const INITIAL_CHOICES = [ + { + "choiceNum": 1, + "choice": "", + }, + { + "choiceNum": 2, + "choice": "", + }, + { + "choiceNum": 3, + "choice": "", + } +] + export default function CreateSurvey(props) { const { history } = props; @@ -20,13 +38,11 @@ export default function CreateSurvey(props) { }; const { value: title, bind: bindTitle, reset: resetTitle } = useInput(''); - const { value: body, bind: bindBody, reset: resetBody } = useInput(''); - const { value: topicId, bind: bindTopicId, reset: resetTopicId } = useInput( - 0 - ); - const { value: endDate, bind: bindEndDate, reset: resetEndDate } = useInput(0) - const { value: endTime, bind: bindEndTime, reset: resetEndTime } = useInput(0) + const { value: topicId, bind: bindTopicId, reset: resetTopicId } = useInput(0); + const { value: endDate, bind: bindEndDate, reset: resetEndDate } = useInput(null); + const { value: endTime, bind: bindEndTime, reset: resetEndTime } = useInput(null); + const [choices, setChoices] = useState(INITIAL_CHOICES); const [listTopic, setlistTopic] = useState([]); const [message, setMessage] = useState(''); @@ -38,18 +54,34 @@ export default function CreateSurvey(props) { getListTopic(); }, []); - useEffect(() => { - }, [endDate, endTime]) + const handleChoiceInput = id => e => { + let updatedChoices = [...choices]; + updatedChoices[id].choice = e.target.value; + setChoices(updatedChoices); + } + + const handleAddChoiceBelow = id => { + let updatedChoices = [...choices]; + updatedChoices.splice(id+1, 0, { + "choiceNum": id+2, + "choice": "" + }); + setChoices(updatedChoices); + } + + const handleDeleteChoice = id => { + let updatedChoices = [...choices]; + updatedChoices.splice(id, 1); + setChoices(updatedChoices); + } const handleSubmit = async (event) => { event.preventDefault(); try { - console.log(body); const thisThread = await axios.post( `${API_URL}/surveys`, { thread: { - content: body, points: 0, title: title, topic_id: topicId, @@ -65,7 +97,6 @@ export default function CreateSurvey(props) { } catch (error) { setMessage("Thread's title already exist in the selected topic"); } - resetBody(); resetTitle(); resetTopicId(); resetEndDate(); @@ -116,13 +147,49 @@ export default function CreateSurvey(props) { ))} </select> <label for="choices">Choices</label> - <textarea - className="body" - name="body" - placeholder="Your thread's body"t - required="false" - {...bindBody} - /> + <div className="choices"> + {choices.length ? ( + choices.map((_, id) => ( + <div className="choice"> + <input + className="choice__input" + name="choice" + placeholder={`Choice ${id + 1}`} + required="false" + onChange={handleChoiceInput(id)} + value={choices[id].choice} + /> + <div className="choice__navs"> + <div + className="choice__navs-item add" + onClick={() => handleAddChoiceBelow(id)} + > + <img src={AddChoiceSvg} alt="add-choice" /> + </div> + <div + className="choice__navs-item del" + onClick={() => handleDeleteChoice(id)} + > + <img src={DelChoiceSvg} alt="del-choice" /> + </div> + </div> + </div> + )) + ) : ( + <div className="choice__navs"> + <div + className="choice__navs-item add" + > + <img src={AddChoiceSvg} alt="add-choice" /> + </div> + <div + className="choice__navs-item del" + > + <img src={DelChoiceSvg} alt="del-choice" /> + </div> + </div> + )} + </div> <label for="duration">Duration</label> <div className="d-flex justify-content-between"> <input diff --git a/src/components/survey/dummy.js b/src/components/survey/dummy.js new file mode 100644 index 0000000..29ac95f --- /dev/null +++ b/src/components/survey/dummy.js @@ -0,0 +1,21 @@ +const DUMMY_SURVEYS = [ + { + "id": 251, + "title": "You snap your finger and one of the following are erased for ever, what do you choose ?", + "choices": [ + { + "id": 1, + "choice": "World Hunger" + } + ], + "inserted_at": "2021-11-26T12:34:22", + "updated_at": "2021-11-26T12:34:22", + "points": 0, + "topic_id": 22, + "topic_name": "Ok Google", + "user_id": 15, + "username": "Ahmad Haulian Yoga Pratama" + } +] + +export default DUMMY_SURVEYS; \ No newline at end of file diff --git a/src/images/add-choice.svg b/src/images/add-choice.svg new file mode 100644 index 0000000..0a054ca --- /dev/null +++ b/src/images/add-choice.svg @@ -0,0 +1,21 @@ +<svg width="34" height="34" viewBox="0 0 34 34" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<g filter="url(#filter0_d_226_946)"> +<rect x="2" width="30" height="30" fill="url(#pattern0)" shape-rendering="crispEdges"/> +</g> +<defs> +<filter id="filter0_d_226_946" x="0" y="0" width="34" height="34" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB"> +<feFlood flood-opacity="0" result="BackgroundImageFix"/> +<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/> +<feOffset dy="2"/> +<feGaussianBlur stdDeviation="1"/> +<feComposite in2="hardAlpha" operator="out"/> +<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.25 0"/> +<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_226_946"/> +<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_226_946" result="shape"/> +</filter> +<pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_226_946" transform="scale(0.0333333)"/> +</pattern> +<image id="image0_226_946" width="30" height="30" xlink:href=""/> +</defs> +</svg> diff --git a/src/images/del-choice.svg b/src/images/del-choice.svg new file mode 100644 index 0000000..e956724 --- /dev/null +++ b/src/images/del-choice.svg @@ -0,0 +1,21 @@ +<svg width="34" height="34" viewBox="0 0 34 34" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> +<g filter="url(#filter0_d_226_947)"> +<rect x="2" width="30" height="30" fill="url(#pattern0)" shape-rendering="crispEdges"/> +</g> +<defs> +<filter id="filter0_d_226_947" x="0" y="0" width="34" height="34" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB"> +<feFlood flood-opacity="0" result="BackgroundImageFix"/> +<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/> +<feOffset dy="2"/> +<feGaussianBlur stdDeviation="1"/> +<feComposite in2="hardAlpha" operator="out"/> +<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.25 0"/> +<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_226_947"/> +<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_226_947" result="shape"/> +</filter> +<pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1"> +<use xlink:href="#image0_226_947" transform="scale(0.0333333)"/> +</pattern> +<image id="image0_226_947" width="30" height="30" xlink:href=""/> +</defs> +</svg> diff --git a/src/styles/survey/Form.css b/src/styles/survey/Form.css index 4e56ee4..374180d 100644 --- a/src/styles/survey/Form.css +++ b/src/styles/survey/Form.css @@ -5,6 +5,7 @@ input.end-time { border: 0; box-shadow: 0px 0px 8px 1px rgba(0,0,0,0.25); margin-bottom: 32px; + font-family: "DM Sans"; } input.end-date { @@ -13,4 +14,41 @@ input.end-date { input.end-time { margin-left: 12px; -} \ No newline at end of file +} + +.choices { + padding: 12px 24px; + border-radius: 8px; + border: 0; + box-shadow: 0px 0px 8px 1px rgba(0,0,0,0.25); + margin-bottom: 32px; + font-family: "DM Sans"; +} + +.choice { + width: 100%; + display: flex; + align-items: center; +} + +input.choice__input { + width: 100%; + padding: 12px 16px; + margin: 12px 0px; + border-radius: 8px; + border: 0; + box-shadow: 0px 0px 8px 1px rgba(0,0,0,0.25); +} + +.choice__navs { + display: flex; + margin-left: 24px; +} + +.choice__navs-item { + cursor: pointer; +} + +.choice__navs-item:last-child { + margin-left: 7px; +} -- GitLab From fa5e337631c73b28b912daddd69ca27a7e8a7250 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 29 Nov 2021 12:49:00 +0700 Subject: [PATCH 04/19] feat(create-survey): add empty placeholder --- src/components/survey/CreateSurvey.jsx | 17 ++++------------- src/styles/survey/Form.css | 17 +++++++++++++++++ 2 files changed, 21 insertions(+), 13 deletions(-) diff --git a/src/components/survey/CreateSurvey.jsx b/src/components/survey/CreateSurvey.jsx index 2667650..31dc83f 100644 --- a/src/components/survey/CreateSurvey.jsx +++ b/src/components/survey/CreateSurvey.jsx @@ -13,15 +13,12 @@ import DelChoiceSvg from '../../images/del-choice.svg'; const INITIAL_CHOICES = [ { - "choiceNum": 1, "choice": "", }, { - "choiceNum": 2, "choice": "", }, { - "choiceNum": 3, "choice": "", } ] @@ -62,10 +59,7 @@ export default function CreateSurvey(props) { const handleAddChoiceBelow = id => { let updatedChoices = [...choices]; - updatedChoices.splice(id+1, 0, { - "choiceNum": id+2, - "choice": "" - }); + updatedChoices.splice(id+1, 0, {"choice": ""}); setChoices(updatedChoices); } @@ -176,17 +170,14 @@ export default function CreateSurvey(props) { </div> )) ) : ( - <div className="choice__navs"> + <div className="choice-empty"> + Add Choices <div className="choice__navs-item add" + onClick={() => handleAddChoiceBelow(-1)} > <img src={AddChoiceSvg} alt="add-choice" /> </div> - <div - className="choice__navs-item del" - > - <img src={DelChoiceSvg} alt="del-choice" /> - </div> </div> )} </div> diff --git a/src/styles/survey/Form.css b/src/styles/survey/Form.css index 374180d..64b937f 100644 --- a/src/styles/survey/Form.css +++ b/src/styles/survey/Form.css @@ -31,6 +31,23 @@ input.end-time { align-items: center; } +.choice-empty { + text-align: center; + font-family: "DM Sans"; + font-size: 18px; + color: #000000; + font-weight: bold; +} + +.choice-empty > .choice__navs-item.add { + width: fit-content; + padding: 5px; + margin: 12px auto; + border-radius: 8px; + border: 0; + box-shadow: 0px 0px 8px 1px rgba(0,0,0,0.25); +} + input.choice__input { width: 100%; padding: 12px 16px; -- GitLab From f700e6c01f513d0cb4373006977a9436efea1ac7 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 29 Nov 2021 14:09:17 +0700 Subject: [PATCH 05/19] feat(list-survey): add files, using thread components as reference --- src/components/survey/CreateSurvey.jsx | 2 +- src/components/survey/ListSurveys.jsx | 58 +++++++++++++++++++++++++ src/components/survey/PreviewThread.jsx | 29 +++++++++++++ src/components/survey/RecentSurveys.jsx | 55 +++++++++++++++++++++++ src/components/survey/SurveyList.jsx | 18 ++++++++ src/components/survey/TopSurveys.jsx | 54 +++++++++++++++++++++++ src/components/survey/dummy.js | 39 +++++++++++++++-- src/routes/App.jsx | 8 ++-- 8 files changed, 256 insertions(+), 7 deletions(-) create mode 100644 src/components/survey/ListSurveys.jsx create mode 100644 src/components/survey/PreviewThread.jsx create mode 100644 src/components/survey/RecentSurveys.jsx create mode 100644 src/components/survey/SurveyList.jsx create mode 100644 src/components/survey/TopSurveys.jsx diff --git a/src/components/survey/CreateSurvey.jsx b/src/components/survey/CreateSurvey.jsx index 31dc83f..ee60635 100644 --- a/src/components/survey/CreateSurvey.jsx +++ b/src/components/survey/CreateSurvey.jsx @@ -204,7 +204,7 @@ export default function CreateSurvey(props) { <input type="submit" className="buttonSubmit" - value="Create Thread" + value="Create Survey" /> </div> </div> diff --git a/src/components/survey/ListSurveys.jsx b/src/components/survey/ListSurveys.jsx new file mode 100644 index 0000000..426435d --- /dev/null +++ b/src/components/survey/ListSurveys.jsx @@ -0,0 +1,58 @@ +import React from 'react'; +import '../../styles/thread/ListThreads.css'; +import Button from '../utility/Button'; +import TopSurveys from './TopSurveys'; +import RecentSurveys from './RecentSurveys'; +import { loggedIn } from '../../helpers/services/auth.service'; +import { Link } from 'react-router-dom'; + +export default function ListSurveys(props) { + return ( + <div className="listThreadsContainer"> + <div className="header"> + <h1> + <b>Threads</b> + </h1> + </div> + + {props.isRecent ? ( + <div className="listThreadsSection"> + <div className="subHeaderListThreads"> + <div className="tab"> + <Link to="/page/1" style={{ textDecoration: 'none' }}> + <h3 className="active">Recent</h3> + </Link> + <Link to="/top/page/1" style={{ textDecoration: 'none' }}> + <h3>Top</h3> + </Link> + </div> + {loggedIn && ( + <Button text="Create Thread" color="orange" url="create/thread" /> + )} + </div> + <RecentSurveys + pageNumber={props.pageNumber} + history={props.history} + /> + </div> + ) : ( + <div className="listThreadsSection"> + <div className="subHeaderListThreads"> + <div className="tab"> + <Link to="/page/1" style={{ textDecoration: 'none' }}> + <h3>Recent</h3> + </Link> + <Link to="/top/page/1" style={{ textDecoration: 'none' }}> + <h3 className="active">Top</h3> + </Link> + </div> + {loggedIn && ( + <Button text="Create Thread" color="orange" url="create/thread" /> + )} + </div> + <TopSurveys pageNumber={props.pageNumber} history={props.history} /> + </div> + )} + </div> + ); +} diff --git a/src/components/survey/PreviewThread.jsx b/src/components/survey/PreviewThread.jsx new file mode 100644 index 0000000..7891c1a --- /dev/null +++ b/src/components/survey/PreviewThread.jsx @@ -0,0 +1,29 @@ +import '../../styles/thread/PreviewThread.css'; +import { translate } from '../../helpers/time-util'; +import { Link } from 'react-router-dom'; + +export default function PreviewThread(props) { + const { content } = props; + const time = translate(content.inserted_at); + return ( + <div className="threadCard"> + <div className="threadCardHeader"> + <h2 className="previewThreadTitle"> + <b>{content.title}</b> + </h2> + </div> + <p className="previewThreadTopic">{content.topic_name}</p> + <div className="threadCardContent"> + <p> + By{' '} + <Link to={`/profile/${content.username}/1`}>{content.username}</Link>{' '} + {', '} + {window.innerWidth < 780 && ( + <br></br> + )} + {time} - <i className="far fa-thumbs-up" /> {content.points} + </p> + </div> + </div> + ); +} diff --git a/src/components/survey/RecentSurveys.jsx b/src/components/survey/RecentSurveys.jsx new file mode 100644 index 0000000..4db3329 --- /dev/null +++ b/src/components/survey/RecentSurveys.jsx @@ -0,0 +1,55 @@ +/* eslint-disable eqeqeq */ +import React, { useEffect, useState, Fragment } from 'react'; +import axios from 'axios'; +import SurveyList from './SurveyList'; +import { API_URL } from '../../config/keys'; +import Pagination from '../thread/Pagination'; + +export default function RecentSurveys(props) { + const [threads, setThreads] = useState([]); + const [totalItems, setTotalItems] = useState(0); + + const currentPage = props.pageNumber; + + if (currentPage != props.pageNumber) { + window.location.reload(); + } + + useEffect(() => { + const fetch = async () => { + const responseThreads = await axios.get( + `${API_URL}/threads/pages/recent/?page=${currentPage}` + ); + const { data } = responseThreads; + setThreads(data?.data); + setTotalItems(data?.metadata?.total_data); + }; + fetch(); + }, [currentPage]); + + function switchPage(pageNumber) { + props.history.push(`/page/${pageNumber}`); + window.location.reload(); + } + + return ( + <div className="recentThreads"> + {totalItems == 0 ? ( + <div className="noThreadsMessage"> + <p>There is no threads yet.</p> + </div> + ) : ( + <Fragment> + <SurveyList thread={threads} /> + <div className="paginationContainer"> + <Pagination + activePage={currentPage} + totalItems={totalItems} + switchPage={switchPage} + /> + </div> + </Fragment> + )} + </div> + ); +} diff --git a/src/components/survey/SurveyList.jsx b/src/components/survey/SurveyList.jsx new file mode 100644 index 0000000..00cb2e2 --- /dev/null +++ b/src/components/survey/SurveyList.jsx @@ -0,0 +1,18 @@ +import PreviewThread from './PreviewThread'; +import { Link } from 'react-router-dom'; + +export default function SurveyList(props) { + return ( + <div className="list_threads"> + {props.thread.map((value) => ( + <Link + key={value.id} + to={`/topic/${value.topic_name}/${value.id}/page/1`} + style={{ textDecoration: 'none' }} + > + <PreviewThread content={value} /> + </Link> + ))} + </div> + ); +} diff --git a/src/components/survey/TopSurveys.jsx b/src/components/survey/TopSurveys.jsx new file mode 100644 index 0000000..b45b91d --- /dev/null +++ b/src/components/survey/TopSurveys.jsx @@ -0,0 +1,54 @@ +/* eslint-disable eqeqeq */ +import React, { useEffect, useState, Fragment } from 'react'; +import axios from 'axios'; +import SurveyList from './SurveyList'; +import { API_URL } from '../../config/keys'; +import Pagination from './Pagination'; + +export default function TopSurveys(props) { + const [threads, setThreads] = useState([]); + const currentPage = props.pageNumber; + const [totalItems, setTotalItems] = useState(0); + + if (currentPage != props.pageNumber) { + window.location.reload(); + } + + useEffect(() => { + const fetch = async () => { + const responseThreads = await axios.get( + `${API_URL}/threads/pages/top/?page=${currentPage}` + ); + const { data } = responseThreads; + setThreads(data?.data); + setTotalItems(data?.metadata?.total_data); + }; + fetch(); + }, [currentPage]); + + function switchPage(pageNumber) { + props.history.push(`/top/page/${pageNumber}`); + window.location.reload(); + } + + return ( + <div className="topThreads"> + {totalItems == 0 ? ( + <div className="noThreadsMessage"> + <p>There is no threads yet.</p> + </div> + ) : ( + <Fragment> + <SurveyList thread={threads} /> + <div className="paginationContainer"> + <Pagination + activePage={currentPage} + totalItems={totalItems} + switchPage={switchPage} + /> + </div> + </Fragment> + )} + </div> + ); +} diff --git a/src/components/survey/dummy.js b/src/components/survey/dummy.js index 29ac95f..425c406 100644 --- a/src/components/survey/dummy.js +++ b/src/components/survey/dummy.js @@ -1,12 +1,45 @@ const DUMMY_SURVEYS = [ { "id": 251, - "title": "You snap your finger and one of the following are erased for ever, what do you choose ?", + "title": "You snap your finger and one of the following are erased for ever, what do you choose?", "choices": [ { - "id": 1, "choice": "World Hunger" - } + }, + { + "choice": "Poverty" + }, + { + "choice": "Famine" + }, + ], + "inserted_at": "2021-11-26T12:34:22", + "updated_at": "2021-11-26T12:34:22", + "points": 0, + "topic_id": 22, + "topic_name": "Ok Google", + "user_id": 15, + "username": "Ahmad Haulian Yoga Pratama" + }, + { + "id": 252, + "title": "Kamu suka pakai bahasa pemrograman fungsional yang mana?", + "choices": [ + { + "choice": "Scala" + }, + { + "choice": "Haskell" + }, + { + "choice": "Mercury" + }, + { + "choice": "Clojure" + }, + { + "choice": "OCaml" + }, ], "inserted_at": "2021-11-26T12:34:22", "updated_at": "2021-11-26T12:34:22", diff --git a/src/routes/App.jsx b/src/routes/App.jsx index a0172f7..e15902a 100644 --- a/src/routes/App.jsx +++ b/src/routes/App.jsx @@ -20,6 +20,7 @@ import { Redirect, Switch, } from 'react-router-dom'; +import ListSurveys from '../components/survey/ListSurveys'; function App() { return ( @@ -28,7 +29,7 @@ function App() { <Switch> <Route exact - path="/page/:pageNumber" + path="/threads/page/:pageNumber" component={(props) => ( <ListThreads isRecent={true} @@ -39,7 +40,7 @@ function App() { /> <Route exact - path="/top/page/:pageNumber" + path="/threads/top/page/:pageNumber" component={(props) => ( <ListThreads isRecent={false} @@ -62,10 +63,11 @@ function App() { /> <Route exact path="/topic/:topic/page/:pageNumber" component={Topic} /> <Route exact path="/threads" component={ListThreads} /> + <Route exact path="/surveys" component={ListSurveys} /> <Route exact path="/thread/edit/:thread" component={EditThread} /> <Route exact path="/profile/update" component={UpdateProfileForm} /> <Route exact path="/comment/edit/:comment" component={EditComment} /> - <Redirect exact from="" to="page/1" /> + <Redirect exact from="" to="/threads/page/1" /> </Switch> <Footer /> </Router> -- GitLab From 6b6deeba3cc4774d6ba94b9a3cacbea11671a6a4 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 29 Nov 2021 14:48:07 +0700 Subject: [PATCH 06/19] feat(list-survey): init PreviewChoices comp. --- src/components/survey/ListSurveys.jsx | 22 +++--- src/components/survey/PreviewChoices.jsx | 10 +++ .../{PreviewThread.jsx => PreviewSurvey.jsx} | 19 +++-- src/components/survey/RecentSurveys.jsx | 11 ++- src/components/survey/SurveyList.jsx | 4 +- src/components/survey/TopSurveys.jsx | 9 ++- src/components/survey/dummy.js | 2 + src/components/utility/Navbar.jsx | 4 +- src/styles/survey/ListSurveys.css | 78 ++++++++++++++++++ src/styles/survey/PreviewChoices.css | 0 src/styles/survey/PreviewSurvey.css | 79 +++++++++++++++++++ 11 files changed, 209 insertions(+), 29 deletions(-) create mode 100644 src/components/survey/PreviewChoices.jsx rename src/components/survey/{PreviewThread.jsx => PreviewSurvey.jsx} (54%) create mode 100644 src/styles/survey/ListSurveys.css create mode 100644 src/styles/survey/PreviewChoices.css create mode 100644 src/styles/survey/PreviewSurvey.css diff --git a/src/components/survey/ListSurveys.jsx b/src/components/survey/ListSurveys.jsx index 426435d..d55b144 100644 --- a/src/components/survey/ListSurveys.jsx +++ b/src/components/survey/ListSurveys.jsx @@ -1,5 +1,5 @@ import React from 'react'; -import '../../styles/thread/ListThreads.css'; +import '../../styles/survey/ListSurveys.css'; import Button from '../utility/Button'; import TopSurveys from './TopSurveys'; import RecentSurveys from './RecentSurveys'; @@ -8,26 +8,26 @@ import { Link } from 'react-router-dom'; export default function ListSurveys(props) { return ( - <div className="listThreadsContainer"> + <div className="listSurveysContainer"> <div className="header"> <h1> - <b>Threads</b> + <b>Surveys</b> </h1> </div> {props.isRecent ? ( - <div className="listThreadsSection"> - <div className="subHeaderListThreads"> + <div className="listSurveysSection"> + <div className="subHeaderListSurveys"> <div className="tab"> - <Link to="/page/1" style={{ textDecoration: 'none' }}> + <Link to="surveys/page/1" style={{ textDecoration: 'none' }}> <h3 className="active">Recent</h3> </Link> - <Link to="/top/page/1" style={{ textDecoration: 'none' }}> + <Link to="surveys/top/page/1" style={{ textDecoration: 'none' }}> <h3>Top</h3> </Link> </div> {loggedIn && ( - <Button text="Create Thread" color="orange" url="create/thread" /> + <Button text="Create Surveys" color="orange" url="create/survey" /> )} </div> <RecentSurveys @@ -36,8 +36,8 @@ export default function ListSurveys(props) { /> </div> ) : ( - <div className="listThreadsSection"> - <div className="subHeaderListThreads"> + <div className="listSurveysSection"> + <div className="subHeaderListSurveys"> <div className="tab"> <Link to="/page/1" style={{ textDecoration: 'none' }}> <h3>Recent</h3> @@ -47,7 +47,7 @@ export default function ListSurveys(props) { </Link> </div> {loggedIn && ( - <Button text="Create Thread" color="orange" url="create/thread" /> + <Button text="Create Surveys" color="orange" url="create/survey" /> )} </div> <TopSurveys pageNumber={props.pageNumber} history={props.history} /> diff --git a/src/components/survey/PreviewChoices.jsx b/src/components/survey/PreviewChoices.jsx new file mode 100644 index 0000000..c297330 --- /dev/null +++ b/src/components/survey/PreviewChoices.jsx @@ -0,0 +1,10 @@ +import '../../styles/survey/PreviewChoices.css'; + +export default function PreviewChoices({ choices, endTime }) { + + return ( + <div className=""> + Component + </div> + ) +} diff --git a/src/components/survey/PreviewThread.jsx b/src/components/survey/PreviewSurvey.jsx similarity index 54% rename from src/components/survey/PreviewThread.jsx rename to src/components/survey/PreviewSurvey.jsx index 7891c1a..bc430c8 100644 --- a/src/components/survey/PreviewThread.jsx +++ b/src/components/survey/PreviewSurvey.jsx @@ -1,19 +1,24 @@ -import '../../styles/thread/PreviewThread.css'; +import '../../styles/survey/PreviewSurvey.css'; import { translate } from '../../helpers/time-util'; import { Link } from 'react-router-dom'; +import PreviewChoices from './PreviewChoices'; -export default function PreviewThread(props) { +export default function PreviewSurvey(props) { const { content } = props; const time = translate(content.inserted_at); return ( - <div className="threadCard"> - <div className="threadCardHeader"> - <h2 className="previewThreadTitle"> + <div className="surveyCard"> + <div className="surveyCardHeader"> + <h2 className="previewSurveyTitle"> <b>{content.title}</b> </h2> </div> - <p className="previewThreadTopic">{content.topic_name}</p> - <div className="threadCardContent"> + <PreviewChoices + choices={content.choices} + endTime={content.endTime} + /> + <p className="previewSurveyTopic">{content.topic_name}</p> + <div className="surveyCardContent"> <p> By{' '} <Link to={`/profile/${content.username}/1`}>{content.username}</Link>{' '} diff --git a/src/components/survey/RecentSurveys.jsx b/src/components/survey/RecentSurveys.jsx index 4db3329..277b53d 100644 --- a/src/components/survey/RecentSurveys.jsx +++ b/src/components/survey/RecentSurveys.jsx @@ -4,6 +4,7 @@ import axios from 'axios'; import SurveyList from './SurveyList'; import { API_URL } from '../../config/keys'; import Pagination from '../thread/Pagination'; +import DUMMY_SURVEYS from './dummy'; export default function RecentSurveys(props) { const [threads, setThreads] = useState([]); @@ -21,8 +22,10 @@ export default function RecentSurveys(props) { `${API_URL}/threads/pages/recent/?page=${currentPage}` ); const { data } = responseThreads; - setThreads(data?.data); - setTotalItems(data?.metadata?.total_data); + console.log(data) + + setThreads(DUMMY_SURVEYS); + setTotalItems(DUMMY_SURVEYS.length); }; fetch(); }, [currentPage]); @@ -33,9 +36,9 @@ export default function RecentSurveys(props) { } return ( - <div className="recentThreads"> + <div className="recentSurveys"> {totalItems == 0 ? ( - <div className="noThreadsMessage"> + <div className="noSurveysMessage"> <p>There is no threads yet.</p> </div> ) : ( diff --git a/src/components/survey/SurveyList.jsx b/src/components/survey/SurveyList.jsx index 00cb2e2..436c587 100644 --- a/src/components/survey/SurveyList.jsx +++ b/src/components/survey/SurveyList.jsx @@ -1,4 +1,4 @@ -import PreviewThread from './PreviewThread'; +import PreviewSurvey from './PreviewSurvey'; import { Link } from 'react-router-dom'; export default function SurveyList(props) { @@ -10,7 +10,7 @@ export default function SurveyList(props) { to={`/topic/${value.topic_name}/${value.id}/page/1`} style={{ textDecoration: 'none' }} > - <PreviewThread content={value} /> + <PreviewSurvey content={value} /> </Link> ))} </div> diff --git a/src/components/survey/TopSurveys.jsx b/src/components/survey/TopSurveys.jsx index b45b91d..b5bab50 100644 --- a/src/components/survey/TopSurveys.jsx +++ b/src/components/survey/TopSurveys.jsx @@ -3,7 +3,8 @@ import React, { useEffect, useState, Fragment } from 'react'; import axios from 'axios'; import SurveyList from './SurveyList'; import { API_URL } from '../../config/keys'; -import Pagination from './Pagination'; +import Pagination from '../thread/Pagination'; +import DUMMY_SURVEYS from './dummy'; export default function TopSurveys(props) { const [threads, setThreads] = useState([]); @@ -20,8 +21,10 @@ export default function TopSurveys(props) { `${API_URL}/threads/pages/top/?page=${currentPage}` ); const { data } = responseThreads; - setThreads(data?.data); - setTotalItems(data?.metadata?.total_data); + console.log(data) + + setThreads(DUMMY_SURVEYS); + setTotalItems(DUMMY_SURVEYS.length); }; fetch(); }, [currentPage]); diff --git a/src/components/survey/dummy.js b/src/components/survey/dummy.js index 425c406..f1ecba9 100644 --- a/src/components/survey/dummy.js +++ b/src/components/survey/dummy.js @@ -13,6 +13,7 @@ const DUMMY_SURVEYS = [ "choice": "Famine" }, ], + "end_time": "2021-11-30T12:34:22", "inserted_at": "2021-11-26T12:34:22", "updated_at": "2021-11-26T12:34:22", "points": 0, @@ -41,6 +42,7 @@ const DUMMY_SURVEYS = [ "choice": "OCaml" }, ], + "end_time": "2021-11-30T12:34:22", "inserted_at": "2021-11-26T12:34:22", "updated_at": "2021-11-26T12:34:22", "points": 0, diff --git a/src/components/utility/Navbar.jsx b/src/components/utility/Navbar.jsx index 3d0a64b..8971267 100644 --- a/src/components/utility/Navbar.jsx +++ b/src/components/utility/Navbar.jsx @@ -62,7 +62,7 @@ const Navbar = (props) => { exact activeClassName="navbar--active" className="nav-link" - to="/page/1" + to="/threads/page/1" > <b>Threads</b> </NavLink> @@ -72,7 +72,7 @@ const Navbar = (props) => { exact activeClassName="navbar--active" className="nav-link" - to="survey/page/1" + to="/surveys" > <b>Surveys</b> </NavLink> diff --git a/src/styles/survey/ListSurveys.css b/src/styles/survey/ListSurveys.css new file mode 100644 index 0000000..ad41acb --- /dev/null +++ b/src/styles/survey/ListSurveys.css @@ -0,0 +1,78 @@ +.listSurveysSection { + display: flex; + flex-direction: column; + justify-content: center; + margin-top: 56px; +} + +.subHeaderListSurveys { + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 28px; +} + +.tab { + display: flex; + align-items: center; +} + +.tab h3 { + font-family: "DM Sans"; + font-size: 18px; + color: #000000; + margin-right: 24px; + margin-top: 4px; + opacity: 0.5; +} + +.tab .active { + color: #DE6600 !important; + font-weight: bold; + opacity: 1!important; +} + +.listSurveysContainer { + margin-top: 36px; + margin-left: 72px; + margin-right: 72px; + min-height: 100vh; + margin-bottom: 72px; +} + +@media only screen and (min-device-width : 320px) and (max-device-width : 540px) { + .listSurveysContainer { + margin-top: 24px; + margin-left: 36px; + margin-right: 36px; + } + + .listSurveysSection { + margin-top: 36px; + } + + .subHeaderListSurveys { + margin-bottom: 28px; + flex-wrap: wrap; + justify-content: center; + flex-direction: column; + justify-content: center; + } + + .tab .active { + color: #DE6600 !important; + font-weight: bold; + opacity: 1!important; + } + + .tab h3 { + font-size: 16px; + margin-right: 28px; + margin-left: 28px; + margin-top: 4px; + } + + .noSurveysMessage p{ + text-align: center; + } +} diff --git a/src/styles/survey/PreviewChoices.css b/src/styles/survey/PreviewChoices.css new file mode 100644 index 0000000..e69de29 diff --git a/src/styles/survey/PreviewSurvey.css b/src/styles/survey/PreviewSurvey.css new file mode 100644 index 0000000..8986fb5 --- /dev/null +++ b/src/styles/survey/PreviewSurvey.css @@ -0,0 +1,79 @@ +.surveyCard { + min-width: 303px; + max-width: 3696px; + margin-bottom: 20px; + padding:20px; + box-shadow: 0px 0px 8px 1px rgba(0,0,0,0.25); + border-radius: 8px; + font-family: "DM Sans"; +} + +.previewSurveyTopic { + background: #DE6600; + color:#ffffff; + padding: 4px; + border-radius: 4px; + width: max-content; + margin: 24px 0px 0px 0px; + border-radius: 4px; + font-size: 14px; +} + +.newSurveyButton { + margin-top: 20px; + margin-left: 20px; +} + +.fa-thumbs-up { + height: 16px; + color: #DE6600; +} + +.previewSurveyTitle { + font-size: 24px; + color: #000000; +} + +.surveyCardContent { + margin-top: 8px; +} + +.surveyCardContent p{ + margin-bottom: 0px; + color: #000000; +} + +@media only screen and (min-device-width : 320px) and (max-device-width : 540px) { + .surveyCard { + min-width: 248px; + max-width: 468px; + margin-bottom: 20px; + padding: 16px; + } + + .previewSurveyTopic { + font-size: 14px; + } + + .newSurveyButton { + margin-top: 20px; + margin-left: 20px; + } + + .previewSurveyTitle { + font-size: 24px; + color: #000000; + } + + .surveyCardContent { + margin-top: 8px; + } + + .surveyCardContent p{ + margin-bottom: 0px; + } + + .surveyCardContent p{ + text-align: left; + } +} \ No newline at end of file -- GitLab From affaf5df8cc656cb126f608d51a6486096cd2684 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 29 Nov 2021 14:50:42 +0700 Subject: [PATCH 07/19] build(deps): install moment-countdown --- package.json | 1 + 1 file changed, 1 insertion(+) diff --git a/package.json b/package.json index 12f9fb0..527dae6 100644 --- a/package.json +++ b/package.json @@ -9,6 +9,7 @@ "axios": "^0.21.1", "http-proxy-middleware": "^1.0.6", "moment": "^2.29.1", + "moment-countdown": "^0.0.3", "nodemon": "^2.0.7", "prettier": "^2.2.1", "react": "^17.0.1", -- GitLab From 068b77b650205285a827b5df53a1ef0df66d4f11 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 29 Nov 2021 16:19:24 +0700 Subject: [PATCH 08/19] feat(list-survey): working on countdown datetime - build: uninstall moment-countdown - utils: separate timestamp & html datetime conversion to UTC format --- package.json | 1 - src/components/survey/PreviewChoices.jsx | 9 ++++++++- src/components/survey/PreviewSurvey.jsx | 2 +- src/components/survey/dummy.js | 2 +- src/helpers/time-util.js | 7 ++++++- 5 files changed, 16 insertions(+), 5 deletions(-) diff --git a/package.json b/package.json index 527dae6..12f9fb0 100644 --- a/package.json +++ b/package.json @@ -9,7 +9,6 @@ "axios": "^0.21.1", "http-proxy-middleware": "^1.0.6", "moment": "^2.29.1", - "moment-countdown": "^0.0.3", "nodemon": "^2.0.7", "prettier": "^2.2.1", "react": "^17.0.1", diff --git a/src/components/survey/PreviewChoices.jsx b/src/components/survey/PreviewChoices.jsx index c297330..1c302c3 100644 --- a/src/components/survey/PreviewChoices.jsx +++ b/src/components/survey/PreviewChoices.jsx @@ -1,10 +1,17 @@ import '../../styles/survey/PreviewChoices.css'; +import moment from 'moment'; +import { timestampToUTC } from '../../helpers/time-util'; export default function PreviewChoices({ choices, endTime }) { + const now = moment(timestampToUTC(new Date())) + const end = moment(new Date(endTime)) + const duration = moment.duration(end.diff(now)) + + console.log(end.fromNow()) return ( <div className=""> - Component + Poll ends in 2 days and 15 hours </div> ) } diff --git a/src/components/survey/PreviewSurvey.jsx b/src/components/survey/PreviewSurvey.jsx index bc430c8..d068636 100644 --- a/src/components/survey/PreviewSurvey.jsx +++ b/src/components/survey/PreviewSurvey.jsx @@ -15,7 +15,7 @@ export default function PreviewSurvey(props) { </div> <PreviewChoices choices={content.choices} - endTime={content.endTime} + endTime={content.end_time} /> <p className="previewSurveyTopic">{content.topic_name}</p> <div className="surveyCardContent"> diff --git a/src/components/survey/dummy.js b/src/components/survey/dummy.js index f1ecba9..a325c19 100644 --- a/src/components/survey/dummy.js +++ b/src/components/survey/dummy.js @@ -42,7 +42,7 @@ const DUMMY_SURVEYS = [ "choice": "OCaml" }, ], - "end_time": "2021-11-30T12:34:22", + "end_time": "2021-12-31T12:34:22", "inserted_at": "2021-11-26T12:34:22", "updated_at": "2021-11-26T12:34:22", "points": 0, diff --git a/src/helpers/time-util.js b/src/helpers/time-util.js index ca432dc..a6e0f0e 100644 --- a/src/helpers/time-util.js +++ b/src/helpers/time-util.js @@ -15,4 +15,9 @@ const datetimeToUTC = (endDate, endTime) => { return date.toISOString(); } -export { translate, datetimeToUTC}; +const timestampToUTC = timestamp => { + const date = new Date(timestamp); + return date.toISOString(); +} + +export { translate, datetimeToUTC, timestampToUTC }; -- GitLab From de0b05270a14d7f447c83499a0ebecb2848ba8a2 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Sat, 11 Dec 2021 14:38:27 +0700 Subject: [PATCH 09/19] feat: add method on displaying poll end countdown --- package.json | 1 + src/components/survey/PreviewChoices.jsx | 11 ++++------- src/helpers/time-util.js | 12 ++++++++---- 3 files changed, 13 insertions(+), 11 deletions(-) diff --git a/package.json b/package.json index 12f9fb0..f70128a 100644 --- a/package.json +++ b/package.json @@ -9,6 +9,7 @@ "axios": "^0.21.1", "http-proxy-middleware": "^1.0.6", "moment": "^2.29.1", + "moment-duration-format": "^2.3.2", "nodemon": "^2.0.7", "prettier": "^2.2.1", "react": "^17.0.1", diff --git a/src/components/survey/PreviewChoices.jsx b/src/components/survey/PreviewChoices.jsx index 1c302c3..6294ced 100644 --- a/src/components/survey/PreviewChoices.jsx +++ b/src/components/survey/PreviewChoices.jsx @@ -1,17 +1,14 @@ import '../../styles/survey/PreviewChoices.css'; -import moment from 'moment'; -import { timestampToUTC } from '../../helpers/time-util'; +import { getCountdown } from '../../helpers/time-util'; + export default function PreviewChoices({ choices, endTime }) { - const now = moment(timestampToUTC(new Date())) - const end = moment(new Date(endTime)) - const duration = moment.duration(end.diff(now)) - console.log(end.fromNow()) + const countdown = getCountdown(endTime) return ( <div className=""> - Poll ends in 2 days and 15 hours + Poll ends in { countdown } </div> ) } diff --git a/src/helpers/time-util.js b/src/helpers/time-util.js index a6e0f0e..1345273 100644 --- a/src/helpers/time-util.js +++ b/src/helpers/time-util.js @@ -1,3 +1,4 @@ +import 'moment-duration-format'; import moment from 'moment'; function translate(time) { @@ -15,9 +16,12 @@ const datetimeToUTC = (endDate, endTime) => { return date.toISOString(); } -const timestampToUTC = timestamp => { - const date = new Date(timestamp); - return date.toISOString(); +const getCountdown = endTime => { + const now = moment() + const end = moment(endTime).local() + const duration = moment.duration(end.diff(now)) + + return duration.format("D [day] H [hour] M [min") } -export { translate, datetimeToUTC, timestampToUTC }; +export { translate, datetimeToUTC, getCountdown }; -- GitLab From 35152dec9229dfb134e9790ecb981a5516819664 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 13 Dec 2021 10:16:25 +0700 Subject: [PATCH 10/19] feat(detail-survey): initialize components --- src/components/survey/Survey.jsx | 158 +++++++++++++++++++++++++++++++ src/routes/App.jsx | 6 ++ src/styles/survey/Survey.css | 116 +++++++++++++++++++++++ 3 files changed, 280 insertions(+) create mode 100644 src/components/survey/Survey.jsx create mode 100644 src/styles/survey/Survey.css diff --git a/src/components/survey/Survey.jsx b/src/components/survey/Survey.jsx new file mode 100644 index 0000000..6760c91 --- /dev/null +++ b/src/components/survey/Survey.jsx @@ -0,0 +1,158 @@ +/* eslint-disable eqeqeq */ +import React, { useState, useEffect, useCallback } from 'react'; +import '../../styles/survey/Survey.css'; +import CommentList from '../thread/CommentList'; +import Post from '../thread/Post'; +import { useInput } from '../../helpers/hooks/input-hook'; +import axios from 'axios'; +import AuthService, { + authHeader, + loggedIn, +} from '../../helpers/services/auth.service'; +import { API_URL } from '../../config/keys'; +import Pagination from '../thread/Pagination'; + +export default function Survey(props) { + const currentUserId = AuthService.getCurrentUserId(); + const redirect = (url) => { + props.history.push(url); + }; + + const [survey, setSurvey] = useState({ + content: '', + id: '', + points: 0, + title: '', + topic_id: '', + user_id: '', + updated_at: '', + inserted_at: '', + username: '', + }); + + const [comment, setComment] = useState([ + { + id: '', + message: '', + points: 0, + survey_id: '', + user_id: '', + updated_at: '', + inserted_at: '', + username: '', + }, + ]); + + const { params } = props.match; + const topicParm = params.topic; + const surveyParm = params.survey; + const currentPage = params.pageNumber; + + const { value: input, bind: bindInput, reset: resetInput } = useInput(''); + const [totalComments, setTotalComments] = useState(0); + + const refreshComment = useCallback(() => { + const fetch = async () => { + const responseComment = await axios.get( + `${API_URL}/post/pages/survey/${surveyParm}/?page=${currentPage}` + ); + const { data } = responseComment; + setComment(data?.data); + setTotalComments(data?.metadata?.total_data); + }; + fetch(); + }, [surveyParm, currentPage]); + + useEffect(() => { + const fetch = async () => { + const responseSurvey = await axios.get( + `${API_URL}/surveys/${surveyParm}` + ); + setSurvey(responseSurvey?.data?.data); + refreshComment(); + }; + fetch(); + }, [surveyParm, refreshComment, currentUserId, survey]); + + const handleSubmit = async (event) => { + event.preventDefault(); + + try { + await axios.post( + `${API_URL}/post`, + { + post: { + message: input, + points: 0, + survey_id: surveyParm, + user_id: localStorage.getItem('id'), + }, + }, + { + headers: authHeader(), + } + ); + refreshComment(); + } catch (error) {} + resetInput(); + }; + + function switchPage(pageNumber) { + props.history.push(`/topic/${topicParm}/${surveyParm}/page/${pageNumber}`); + window.location.reload(); + } + + return ( + <div className="surveyContainer"> + <div className="survey_section"> + <Post type="survey" content={survey} redirect={redirect} /> + </div> + + {loggedIn && ( + <div className="addCommentSection"> + <h3>Write a Comment</h3> + <form onSubmit={handleSubmit}> + <div className="commentFormContainer"> + <textarea + className="commentBox" + placeholder="Write your comment here" + required="false" + {...bindInput} + /> + <div> + <button + className="submitComment addCommentButton" + type="submit" + > + Post Comment + </button> + </div> + </div> + </form> + </div> + )} + <h2 className="commentText">Comments</h2> + + {totalComments == 0 ? ( + <div className="noCommentLabel"> + <p>There are no comments yet.</p> + </div> + ) : ( + <div> + <CommentList + comment={comment} + survey_className={surveyParm} + topic_className={topicParm} + /> + <div className="paginationContainer"> + <Pagination + activePage={currentPage} + totalItems={totalComments} + switchPage={switchPage} + /> + </div> + </div> + )} + </div> + ); +} diff --git a/src/routes/App.jsx b/src/routes/App.jsx index e15902a..905f374 100644 --- a/src/routes/App.jsx +++ b/src/routes/App.jsx @@ -21,6 +21,7 @@ import { Switch, } from 'react-router-dom'; import ListSurveys from '../components/survey/ListSurveys'; +import Survey from '../components/survey/Survey'; function App() { return ( @@ -61,6 +62,11 @@ function App() { path="/topic/:topic/:thread/page/:pageNumber" component={Thread} /> + <Route + exact + path="/topic/:topic/:survey/page/:pageNumber" + component={Survey} + /> <Route exact path="/topic/:topic/page/:pageNumber" component={Topic} /> <Route exact path="/threads" component={ListThreads} /> <Route exact path="/surveys" component={ListSurveys} /> diff --git a/src/styles/survey/Survey.css b/src/styles/survey/Survey.css new file mode 100644 index 0000000..058ae95 --- /dev/null +++ b/src/styles/survey/Survey.css @@ -0,0 +1,116 @@ +.surveyContainer { + display: flex; + flex-direction: column; + margin-top: 36px; + font-family: "DM Sans"; + margin-left: 72px; + margin-right: 72px; + min-height: 100vh; + margin-bottom: 72px; +} + +.commentText { + color: #FEA02F; + font-size: 24px; + font-weight: bold; + margin-bottom: 12px; +} + +.addCommentSection { + margin-top: 40px; + display: flex; + flex-direction: column; + margin-top: 20px; +} + +.addCommentSection h3 { + color: #FEA02F; + font-size: 24px; + font-weight: bold; + margin-bottom: 12px; +} + +.addCommentButton { + margin-top: 20px; + margin-left: auto; + margin-right: auto; + margin-bottom: 40px; +} + +.commentBox { + min-width: 468px; + max-width: 3696px; + font-family: "DM Sans"; + height: 120px; + padding: 12px 16px; + border-radius: 8px; + border: 0; + box-shadow: 0px 0px 8px 1px rgba(0,0,0,0.25); +} + +.submitComment { + border-radius: 4px; + border-width: 0px; + padding: 8px; + font-family: "DM Sans"; + color: #ffffff; + border-style: none; + background-color: #DE6600; + border-color: #DE6600; +} + +.noCommentLabel p { + font-size: 16px; + margin-bottom: 0px; +} + +.commentFormContainer { + display: flex; + flex-direction: column; +} + +@media only screen and (min-device-width : 320px) and (max-device-width : 540px) { + .surveyContainer { + margin-top: 24px; + margin-bottom: 24px; + margin-left: 36px; + margin-right: 36px; + } + + .addCommentSection h3 { + font-size: 18px; + text-align: center; + } + + .addCommentSection div { + display: flex; + justify-content: center; + } + + .commentBox { + min-width: 248px; + max-width: 468px; + height: 120px; + font-size: 14px; + } + + .submitComment { + font-size: 16px; + } + + .commentText { + font-size: 18px; + text-align: center; + } + + .noCommentLabel p { + font-size: 14px; + margin-bottom: 0px; + text-align: center; + } + + .buttonContainer .button { + margin-top: 0px; + } +} + -- GitLab From ac4b2f0a1ebc4e4b0ab3b280a425518e930970b3 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 13 Dec 2021 11:32:29 +0700 Subject: [PATCH 11/19] feat(list-survey): add UI --- src/components/survey/ChoiceBox.jsx | 33 ++++++++++++++++++ src/components/survey/PreviewChoices.jsx | 37 +++++++++++++++++--- src/components/survey/PreviewSurvey.jsx | 24 ++++++++----- src/components/survey/dummy.js | 26 +++++++++----- src/helpers/time-util.js | 2 +- src/styles/survey/ChoiceBox.css | 43 ++++++++++++++++++++++++ src/styles/survey/PreviewChoices.css | 13 +++++++ 7 files changed, 157 insertions(+), 21 deletions(-) create mode 100644 src/components/survey/ChoiceBox.jsx create mode 100644 src/styles/survey/ChoiceBox.css diff --git a/src/components/survey/ChoiceBox.jsx b/src/components/survey/ChoiceBox.jsx new file mode 100644 index 0000000..5c379f1 --- /dev/null +++ b/src/components/survey/ChoiceBox.jsx @@ -0,0 +1,33 @@ +import '../../styles/survey/ChoiceBox.css'; + +export const ChoiceBoxVoted = ({ choice }) => { + const { name, votes } = choice; + const isUserVoted = Math.random() < 0.5; + + return ( + <div + className="choice-box__voted" + style={{ + background: isUserVoted + ? '#FEA02F' + : '#FFFFFF', + color: isUserVoted + ? '#FFFFFF' + : '#000000', + }} + > + <h6 className="name">{name}</h6> + <h6 className="votes">{votes}</h6> + </div> + ); +}; + +export const ChoiceBoxUnvoted = ({ choice }) => { + const { name } = choice; + + return ( + <div className="choice-box__unvoted"> + <h6 className="name">{name}</h6> + </div> + ); +}; diff --git a/src/components/survey/PreviewChoices.jsx b/src/components/survey/PreviewChoices.jsx index 6294ced..0f1601b 100644 --- a/src/components/survey/PreviewChoices.jsx +++ b/src/components/survey/PreviewChoices.jsx @@ -1,14 +1,43 @@ import '../../styles/survey/PreviewChoices.css'; import { getCountdown } from '../../helpers/time-util'; +import { ChoiceBoxUnvoted, ChoiceBoxVoted } from './ChoiceBox'; export default function PreviewChoices({ choices, endTime }) { - - const countdown = getCountdown(endTime) + const COUNTDOWN_STRING = { + withCountdown: (countdown) => `Poll ends in ${countdown}`, + withoutCountdown: 'Poll has ended' + }; + const isOverEndTime = new Date() > new Date(endTime); + const isUserHasVoted = Math.random() < 0.5; + const countdown = getCountdown(endTime); return ( - <div className=""> - Poll ends in { countdown } + <div className="preview-choices__container"> + <div className="countdown"> + { + isOverEndTime + ? COUNTDOWN_STRING.withoutCountdown + : COUNTDOWN_STRING.withCountdown(countdown) + } + </div> + <div className="choiceboxes"> + {choices.map((choice, idx) => ( + <> + {isUserHasVoted ? ( + <ChoiceBoxVoted + choice={choice} + key={idx} + /> + ) : ( + <ChoiceBoxUnvoted + choice={choice} + key={idx} + /> + )} + </> + ))} + </div> </div> ) } diff --git a/src/components/survey/PreviewSurvey.jsx b/src/components/survey/PreviewSurvey.jsx index d068636..74ba5e8 100644 --- a/src/components/survey/PreviewSurvey.jsx +++ b/src/components/survey/PreviewSurvey.jsx @@ -3,30 +3,38 @@ import { translate } from '../../helpers/time-util'; import { Link } from 'react-router-dom'; import PreviewChoices from './PreviewChoices'; -export default function PreviewSurvey(props) { - const { content } = props; +export default function PreviewSurvey({ content }) { + const { + title, + choices, + end_time, + topic_name, + username, + points, + } = content; const time = translate(content.inserted_at); + return ( <div className="surveyCard"> <div className="surveyCardHeader"> <h2 className="previewSurveyTitle"> - <b>{content.title}</b> + <b>{title}</b> </h2> </div> <PreviewChoices - choices={content.choices} - endTime={content.end_time} + choices={choices} + endTime={end_time} /> - <p className="previewSurveyTopic">{content.topic_name}</p> + <p className="previewSurveyTopic">{topic_name}</p> <div className="surveyCardContent"> <p> By{' '} - <Link to={`/profile/${content.username}/1`}>{content.username}</Link>{' '} + <Link to={`/profile/${username}/1`}>{username}</Link>{' '} {', '} {window.innerWidth < 780 && ( <br></br> )} - {time} - <i className="far fa-thumbs-up" /> {content.points} + {time} - <i className="far fa-thumbs-up" /> {points} </p> </div> </div> diff --git a/src/components/survey/dummy.js b/src/components/survey/dummy.js index a325c19..0156292 100644 --- a/src/components/survey/dummy.js +++ b/src/components/survey/dummy.js @@ -2,15 +2,19 @@ const DUMMY_SURVEYS = [ { "id": 251, "title": "You snap your finger and one of the following are erased for ever, what do you choose?", + "is_user_has_voted": true, "choices": [ { - "choice": "World Hunger" + "name": "World Hunger", + "votes": Math.floor(Math.random() * 1000) + 1 }, { - "choice": "Poverty" + "name": "Poverty", + "votes": Math.floor(Math.random() * 1000) + 1 }, { - "choice": "Famine" + "name": "Famine", + "votes": Math.floor(Math.random() * 1000) + 1 }, ], "end_time": "2021-11-30T12:34:22", @@ -25,21 +29,27 @@ const DUMMY_SURVEYS = [ { "id": 252, "title": "Kamu suka pakai bahasa pemrograman fungsional yang mana?", + "is_user_has_voted": false, "choices": [ { - "choice": "Scala" + "name": "Scala", + "votes": Math.floor(Math.random() * 1000) + 1 }, { - "choice": "Haskell" + "name": "Haskell", + "votes": Math.floor(Math.random() * 1000) + 1 }, { - "choice": "Mercury" + "name": "Mercury", + "votes": Math.floor(Math.random() * 1000) + 1 }, { - "choice": "Clojure" + "name": "Clojure", + "votes": Math.floor(Math.random() * 1000) + 1 }, { - "choice": "OCaml" + "name": "OCaml", + "votes": Math.floor(Math.random() * 1000) + 1 }, ], "end_time": "2021-12-31T12:34:22", diff --git a/src/helpers/time-util.js b/src/helpers/time-util.js index 1345273..02e1a58 100644 --- a/src/helpers/time-util.js +++ b/src/helpers/time-util.js @@ -21,7 +21,7 @@ const getCountdown = endTime => { const end = moment(endTime).local() const duration = moment.duration(end.diff(now)) - return duration.format("D [day] H [hour] M [min") + return duration.format("D [day] H [hour] M [min]") } export { translate, datetimeToUTC, getCountdown }; diff --git a/src/styles/survey/ChoiceBox.css b/src/styles/survey/ChoiceBox.css new file mode 100644 index 0000000..67f2e1b --- /dev/null +++ b/src/styles/survey/ChoiceBox.css @@ -0,0 +1,43 @@ +.choice-box__voted, +.choice-box__unvoted { + display: flex; + align-items: center; + margin: 8px 0px; + padding: 8px; + cursor: default; + + background: #FFFFFF; + border: 2px solid #FEA02F; + box-sizing: border-box; + border-radius: 25px; +} + +.choice-box__voted h6, +.choice-box__unvoted h6 { + cursor: text; + font-family: DM Sans; + font-style: normal; + font-weight: bold; + font-size: 0.75rem; + padding-bottom: 0px; + margin-bottom: 0px; +} + +.choice-box__unvoted:hover { + background: #FEA02F; +} + +.choice-box__unvoted:hover h6 { + color: #FFFFFF; +} + +.choice-box__voted { + justify-content: space-between; + padding-left: 12px; + padding-right: 12px; +} + +.choice-box__unvoted { + justify-content: center; + color: #000000 +} diff --git a/src/styles/survey/PreviewChoices.css b/src/styles/survey/PreviewChoices.css index e69de29..81c544a 100644 --- a/src/styles/survey/PreviewChoices.css +++ b/src/styles/survey/PreviewChoices.css @@ -0,0 +1,13 @@ +.preview-choices__container { + box-shadow: 0px 0px 8px rgba(0, 0, 0, 0.5); + border-radius: 8px; + padding: 0.75rem 1rem; + margin: 1.5rem 0px 1rem; + cursor: pointer; +} + +.preview-choices__container .countdown { + color: rgba(0, 0, 0, 0.5); + font-weight: bold; + font-size: 0.875rem; +} \ No newline at end of file -- GitLab From d26cbb762f4007844c98c7dfc4ab71dab3812130 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 13 Dec 2021 12:27:44 +0700 Subject: [PATCH 12/19] feat(list-survey): add minor improvement to UI --- src/components/survey/PreviewChoices.jsx | 7 ++++++- src/components/survey/dummy.js | 2 +- src/helpers/time-util.js | 12 +++++++++++- src/styles/survey/ChoiceBox.css | 3 --- src/styles/survey/PreviewChoices.css | 12 ++++++++++-- 5 files changed, 28 insertions(+), 8 deletions(-) diff --git a/src/components/survey/PreviewChoices.jsx b/src/components/survey/PreviewChoices.jsx index 0f1601b..f2f771a 100644 --- a/src/components/survey/PreviewChoices.jsx +++ b/src/components/survey/PreviewChoices.jsx @@ -11,6 +11,7 @@ export default function PreviewChoices({ choices, endTime }) { const isOverEndTime = new Date() > new Date(endTime); const isUserHasVoted = Math.random() < 0.5; const countdown = getCountdown(endTime); + const numOfLeftoverChoices = choices.length - 3; return ( <div className="preview-choices__container"> @@ -22,7 +23,7 @@ export default function PreviewChoices({ choices, endTime }) { } </div> <div className="choiceboxes"> - {choices.map((choice, idx) => ( + {choices.slice(0, 3).map((choice, idx) => ( <> {isUserHasVoted ? ( <ChoiceBoxVoted @@ -38,6 +39,10 @@ export default function PreviewChoices({ choices, endTime }) { </> ))} </div> + <h6 className="choices-left"> + {numOfLeftoverChoices > 0 && `${numOfLeftoverChoices} More Choice`} + {numOfLeftoverChoices > 1 && 's'} + </h6> </div> ) } diff --git a/src/components/survey/dummy.js b/src/components/survey/dummy.js index 0156292..455efa7 100644 --- a/src/components/survey/dummy.js +++ b/src/components/survey/dummy.js @@ -52,7 +52,7 @@ const DUMMY_SURVEYS = [ "votes": Math.floor(Math.random() * 1000) + 1 }, ], - "end_time": "2021-12-31T12:34:22", + "end_time": "2021-12-31T18:34:22", "inserted_at": "2021-11-26T12:34:22", "updated_at": "2021-11-26T12:34:22", "points": 0, diff --git a/src/helpers/time-util.js b/src/helpers/time-util.js index 02e1a58..ba03c53 100644 --- a/src/helpers/time-util.js +++ b/src/helpers/time-util.js @@ -21,7 +21,17 @@ const getCountdown = endTime => { const end = moment(endTime).local() const duration = moment.duration(end.diff(now)) - return duration.format("D [day] H [hour] M [min]") + return ( + duration + .format( + [ + moment.duration(1, "minute"), + moment.duration(1, "hour"), + moment.duration(1, "day") + ], + "d [days], h [hours], m [minutes]" + ) + ); } export { translate, datetimeToUTC, getCountdown }; diff --git a/src/styles/survey/ChoiceBox.css b/src/styles/survey/ChoiceBox.css index 67f2e1b..1e4f451 100644 --- a/src/styles/survey/ChoiceBox.css +++ b/src/styles/survey/ChoiceBox.css @@ -4,7 +4,6 @@ align-items: center; margin: 8px 0px; padding: 8px; - cursor: default; background: #FFFFFF; border: 2px solid #FEA02F; @@ -14,11 +13,9 @@ .choice-box__voted h6, .choice-box__unvoted h6 { - cursor: text; font-family: DM Sans; font-style: normal; font-weight: bold; - font-size: 0.75rem; padding-bottom: 0px; margin-bottom: 0px; } diff --git a/src/styles/survey/PreviewChoices.css b/src/styles/survey/PreviewChoices.css index 81c544a..11c635b 100644 --- a/src/styles/survey/PreviewChoices.css +++ b/src/styles/survey/PreviewChoices.css @@ -9,5 +9,13 @@ .preview-choices__container .countdown { color: rgba(0, 0, 0, 0.5); font-weight: bold; - font-size: 0.875rem; -} \ No newline at end of file + margin-left: calc(1rem - 2px); +} + +.preview-choices__container .choices-left { + color: rgba(0, 0, 0, 0.5); + font-weight: bold; + text-align: center; + margin-top: 0.75rem; + margin-bottom: 0rem; +} -- GitLab From e30a0cdf0a2c6f8da853ff417b5f76766efa705a Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 13 Dec 2021 12:35:54 +0700 Subject: [PATCH 13/19] feat(list-survey): disable hover effect on preview --- src/components/survey/ChoiceBox.jsx | 4 ++-- src/components/survey/PreviewChoices.jsx | 1 + src/styles/survey/ChoiceBox.css | 8 ++++++++ 3 files changed, 11 insertions(+), 2 deletions(-) diff --git a/src/components/survey/ChoiceBox.jsx b/src/components/survey/ChoiceBox.jsx index 5c379f1..3d07fb1 100644 --- a/src/components/survey/ChoiceBox.jsx +++ b/src/components/survey/ChoiceBox.jsx @@ -22,11 +22,11 @@ export const ChoiceBoxVoted = ({ choice }) => { ); }; -export const ChoiceBoxUnvoted = ({ choice }) => { +export const ChoiceBoxUnvoted = ({ choice, isPreview=false }) => { const { name } = choice; return ( - <div className="choice-box__unvoted"> + <div className={`choice-box__unvoted ${isPreview && 'is_preview'}`}> <h6 className="name">{name}</h6> </div> ); diff --git a/src/components/survey/PreviewChoices.jsx b/src/components/survey/PreviewChoices.jsx index f2f771a..f5abf34 100644 --- a/src/components/survey/PreviewChoices.jsx +++ b/src/components/survey/PreviewChoices.jsx @@ -34,6 +34,7 @@ export default function PreviewChoices({ choices, endTime }) { <ChoiceBoxUnvoted choice={choice} key={idx} + isPreview /> )} </> diff --git a/src/styles/survey/ChoiceBox.css b/src/styles/survey/ChoiceBox.css index 1e4f451..711fe34 100644 --- a/src/styles/survey/ChoiceBox.css +++ b/src/styles/survey/ChoiceBox.css @@ -28,6 +28,14 @@ color: #FFFFFF; } +.is_preview.choice-box__unvoted:hover { + background: #FFFFFF; +} + +.is_preview.choice-box__unvoted:hover h6 { + color: #000000; +} + .choice-box__voted { justify-content: space-between; padding-left: 12px; -- GitLab From 52da864247258ebcc951c025b1ee659024d9c325 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 13 Dec 2021 17:05:07 +0700 Subject: [PATCH 14/19] feat(detail-survey): adjust routing to detail thread & survey --- src/components/survey/SurveyList.jsx | 2 +- src/components/thread/ThreadList.jsx | 2 +- src/routes/App.jsx | 4 ++-- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/components/survey/SurveyList.jsx b/src/components/survey/SurveyList.jsx index 436c587..88fe5e7 100644 --- a/src/components/survey/SurveyList.jsx +++ b/src/components/survey/SurveyList.jsx @@ -7,7 +7,7 @@ export default function SurveyList(props) { {props.thread.map((value) => ( <Link key={value.id} - to={`/topic/${value.topic_name}/${value.id}/page/1`} + to={`/survey/topic/${value.topic_name}/${value.id}/page/1`} style={{ textDecoration: 'none' }} > <PreviewSurvey content={value} /> diff --git a/src/components/thread/ThreadList.jsx b/src/components/thread/ThreadList.jsx index 574a18c..7d65c15 100644 --- a/src/components/thread/ThreadList.jsx +++ b/src/components/thread/ThreadList.jsx @@ -7,7 +7,7 @@ export default function ThreadList(props) { {props.thread.map((value) => ( <Link key={value.id} - to={`/topic/${value.topic_name}/${value.id}/page/1`} + to={`/thread/topic/${value.topic_name}/${value.id}/page/1`} style={{ textDecoration: 'none' }} > <PreviewThread content={value} /> diff --git a/src/routes/App.jsx b/src/routes/App.jsx index 905f374..71e4933 100644 --- a/src/routes/App.jsx +++ b/src/routes/App.jsx @@ -59,12 +59,12 @@ function App() { <Route exact path="/search/:input/page/:pageNumber" component={Search} /> <Route exact - path="/topic/:topic/:thread/page/:pageNumber" + path="/thread/topic/:topic/:thread/page/:pageNumber" component={Thread} /> <Route exact - path="/topic/:topic/:survey/page/:pageNumber" + path="/survey/topic/:topic/:survey/page/:pageNumber" component={Survey} /> <Route exact path="/topic/:topic/page/:pageNumber" component={Topic} /> -- GitLab From 2a351fb811d4a1fda2e7dda3db3424307474c94f Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 13 Dec 2021 20:40:52 +0700 Subject: [PATCH 15/19] feat(detail-survey): add detail survey UI --- src/components/survey/DetailChoices.jsx | 44 +++++++ src/components/survey/RecentSurveys.jsx | 2 +- src/components/survey/Survey.jsx | 31 +++-- src/components/survey/SurveyPost.jsx | 158 ++++++++++++++++++++++++ src/components/survey/TopSurveys.jsx | 2 +- src/styles/survey/DetailChoices.css | 10 ++ 6 files changed, 229 insertions(+), 18 deletions(-) create mode 100644 src/components/survey/DetailChoices.jsx create mode 100644 src/components/survey/SurveyPost.jsx create mode 100644 src/styles/survey/DetailChoices.css diff --git a/src/components/survey/DetailChoices.jsx b/src/components/survey/DetailChoices.jsx new file mode 100644 index 0000000..170eb0d --- /dev/null +++ b/src/components/survey/DetailChoices.jsx @@ -0,0 +1,44 @@ +import '../../styles/survey/DetailChoices.css'; +import { getCountdown } from '../../helpers/time-util'; +import { ChoiceBoxUnvoted, ChoiceBoxVoted } from './ChoiceBox'; + + +export default function DetailChoices({ choices, endTime }) { + const COUNTDOWN_STRING = { + withCountdown: (countdown) => `Poll ends in ${countdown}`, + withoutCountdown: 'Poll has ended' + }; + const isOverEndTime = new Date() > new Date(endTime); + const isUserHasVoted = Math.random() < 0.5; + const countdown = getCountdown(endTime); + + return ( + <div className="detail-choices__container"> + <div className="countdown"> + { + isOverEndTime + ? COUNTDOWN_STRING.withoutCountdown + : COUNTDOWN_STRING.withCountdown(countdown) + } + </div> + <div className="choiceboxes"> + {choices.map((choice, idx) => ( + <> + {isUserHasVoted ? ( + <ChoiceBoxVoted + choice={choice} + key={idx} + /> + ) : ( + <ChoiceBoxUnvoted + choice={choice} + key={idx} + isPreview + /> + )} + </> + ))} + </div> + </div> + ) +} diff --git a/src/components/survey/RecentSurveys.jsx b/src/components/survey/RecentSurveys.jsx index 277b53d..6b0d481 100644 --- a/src/components/survey/RecentSurveys.jsx +++ b/src/components/survey/RecentSurveys.jsx @@ -39,7 +39,7 @@ export default function RecentSurveys(props) { <div className="recentSurveys"> {totalItems == 0 ? ( <div className="noSurveysMessage"> - <p>There is no threads yet.</p> + <p>There is no surveys yet.</p> </div> ) : ( <Fragment> diff --git a/src/components/survey/Survey.jsx b/src/components/survey/Survey.jsx index 6760c91..5b0a8b4 100644 --- a/src/components/survey/Survey.jsx +++ b/src/components/survey/Survey.jsx @@ -1,16 +1,18 @@ /* eslint-disable eqeqeq */ +import axios from 'axios'; import React, { useState, useEffect, useCallback } from 'react'; + import '../../styles/survey/Survey.css'; import CommentList from '../thread/CommentList'; -import Post from '../thread/Post'; +import SurveyPost from './SurveyPost'; import { useInput } from '../../helpers/hooks/input-hook'; -import axios from 'axios'; import AuthService, { authHeader, loggedIn, } from '../../helpers/services/auth.service'; import { API_URL } from '../../config/keys'; import Pagination from '../thread/Pagination'; +import DUMMY_SURVEYS from './dummy'; export default function Survey(props) { const currentUserId = AuthService.getCurrentUserId(); @@ -18,18 +20,7 @@ export default function Survey(props) { props.history.push(url); }; - const [survey, setSurvey] = useState({ - content: '', - id: '', - points: 0, - title: '', - topic_id: '', - user_id: '', - updated_at: '', - inserted_at: '', - username: '', - }); - + const [survey, setSurvey] = useState(null); const [comment, setComment] = useState([ { id: '', @@ -51,6 +42,10 @@ export default function Survey(props) { const { value: input, bind: bindInput, reset: resetInput } = useInput(''); const [totalComments, setTotalComments] = useState(0); + useEffect(() => { + setSurvey(DUMMY_SURVEYS[1]); + }, []) + const refreshComment = useCallback(() => { const fetch = async () => { const responseComment = await axios.get( @@ -68,7 +63,11 @@ export default function Survey(props) { const responseSurvey = await axios.get( `${API_URL}/surveys/${surveyParm}` ); - setSurvey(responseSurvey?.data?.data); + + const { data } = responseSurvey; + console.log(data.data) + + setSurvey(DUMMY_SURVEYS[0]); refreshComment(); }; fetch(); @@ -105,7 +104,7 @@ export default function Survey(props) { return ( <div className="surveyContainer"> <div className="survey_section"> - <Post type="survey" content={survey} redirect={redirect} /> + {survey && <SurveyPost content={survey} redirect={redirect} />} </div> {loggedIn && ( diff --git a/src/components/survey/SurveyPost.jsx b/src/components/survey/SurveyPost.jsx new file mode 100644 index 0000000..1d33644 --- /dev/null +++ b/src/components/survey/SurveyPost.jsx @@ -0,0 +1,158 @@ +/* eslint-disable eqeqeq */ +import React, { useState, useEffect } from 'react'; +import { Link } from 'react-router-dom'; +import axios from 'axios'; + +import '../../styles/thread/Post.css'; +import AuthService, { + authHeader, + loggedIn, +} from '../../helpers/services/auth.service'; +import { translate } from '../../helpers/time-util'; +import { API_URL } from '../../config/keys'; +import Button from '../utility/Button'; +import DetailChoices from './DetailChoices'; + +export default function SurveyPost({ content }) { + const [points, setPoints] = useState(0); + const [isLiked, setIsLiked] = useState(false); + const [user, setUser] = useState({ + id: '', + name: '', + username: '', + }); + + const deletePost = async () => { + try { + await axios.delete(`${API_URL}/survey/${content.id}`, { + headers: authHeader(), + }); + } catch (error) { + console.log(error); + } + }; + + useEffect(() => { + const fetch = async () => { + try { + setPoints(content.points); + if (content.username != '') { + const getUser = await axios.get( + `${API_URL}/users/name/${content.username}` + ); + setUser(getUser?.data?.data); + } + await axios.get(`${API_URL}/$survey/checklike/${content.id}`, { + headers: authHeader(), + }); + setIsLiked(true); + } catch (error) { + setIsLiked(false); + } + }; + fetch(); + }, [content.id, content.points, content.username]); + + const handleLike = async () => { + try { + await axios.get(`${API_URL}/survey/checklike/${content.id}`, { + headers: authHeader(), + }); + await axios.post( + `${API_URL}/survey/dislike/${content.id}`, + {}, + { + headers: authHeader(), + } + ); + setPoints(points - 1); + setIsLiked(false); + } catch (error) { + await axios.post( + `${API_URL}/survey/like/${content.id}`, + {}, + { + headers: authHeader(), + } + ); + setPoints(points + 1); + setIsLiked(true); + } + }; + + useEffect(() => { + console.log(content) + }, [content]) + + const time = translate(content.inserted_at); + + return ( + <div className="post"> + <div className="postHeader"> + <div className="headerData"> + <div className="userImage"> + {user.picture ? ( + <i className="far fa-user-circle" /> + ) : ( + <img alt="profile" src={user.picture} /> + )} + </div> + <div className="creator"> + <p> + By{' '} + <Link to={`/profile/${content.username}/1`}> + {content.username} + </Link>{', '} + {window.innerWidth < 780 && ( + <br></br> + )} + {time}{' '} + </p> + </div> + </div> + {content.user_id == AuthService.getCurrentUserId() && ( + <div className="headerButton"> + <div className="postButtonContainer"> + <Button + type="button" + text="Edit" + color="none-green" + url={`thread/edit/${content.id}`} + /> + <button + type="button" + className="deleteButton" + onClick={() => deletePost(user.username)} + > + Delete + </button> + </div> + </div> + )} + </div> + <div className="postContent"> + <h1 className="postTitle">{content.title}</h1> + <DetailChoices + choices={content.choices} + endTime={content.end_time} + /> + <div className="likeSection"> + {loggedIn && ( + <button className="likeButton" onClick={handleLike}> + <i + className={`far fa-thumbs-up ${ + isLiked ? 'active' : 'inactive' + }`} + /> + </button> + )} + <div className="pointContainer"> + <div className="point"> + <p>{points} likes</p> + </div> + </div> + </div> + </div> + </div> + ); +} diff --git a/src/components/survey/TopSurveys.jsx b/src/components/survey/TopSurveys.jsx index b5bab50..796a8c0 100644 --- a/src/components/survey/TopSurveys.jsx +++ b/src/components/survey/TopSurveys.jsx @@ -38,7 +38,7 @@ export default function TopSurveys(props) { <div className="topThreads"> {totalItems == 0 ? ( <div className="noThreadsMessage"> - <p>There is no threads yet.</p> + <p>There is no surveys yet.</p> </div> ) : ( <Fragment> diff --git a/src/styles/survey/DetailChoices.css b/src/styles/survey/DetailChoices.css new file mode 100644 index 0000000..61c1848 --- /dev/null +++ b/src/styles/survey/DetailChoices.css @@ -0,0 +1,10 @@ +.detail-choices__container { + border-radius: 8px; + margin: 1.5rem 0px; + cursor: pointer; +} + +.detail-choices__container .countdown { + color: rgba(0, 0, 0, 0.5); + font-weight: bold; +} -- GitLab From 202300163e25f2e9802dbc3704f819b98d59aae6 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Mon, 13 Dec 2021 21:12:45 +0700 Subject: [PATCH 16/19] feat(survey): sort choices by vote count --- src/components/survey/DetailChoices.jsx | 3 ++- src/components/survey/PreviewChoices.jsx | 3 ++- src/helpers/survey-util.js | 3 +++ 3 files changed, 7 insertions(+), 2 deletions(-) create mode 100644 src/helpers/survey-util.js diff --git a/src/components/survey/DetailChoices.jsx b/src/components/survey/DetailChoices.jsx index 170eb0d..dd2d0c0 100644 --- a/src/components/survey/DetailChoices.jsx +++ b/src/components/survey/DetailChoices.jsx @@ -1,6 +1,7 @@ import '../../styles/survey/DetailChoices.css'; import { getCountdown } from '../../helpers/time-util'; import { ChoiceBoxUnvoted, ChoiceBoxVoted } from './ChoiceBox'; +import { sortChoicesByVoteDescending } from '../../helpers/survey-util'; export default function DetailChoices({ choices, endTime }) { @@ -22,7 +23,7 @@ export default function DetailChoices({ choices, endTime }) { } </div> <div className="choiceboxes"> - {choices.map((choice, idx) => ( + {sortChoicesByVoteDescending(choices).map((choice, idx) => ( <> {isUserHasVoted ? ( <ChoiceBoxVoted diff --git a/src/components/survey/PreviewChoices.jsx b/src/components/survey/PreviewChoices.jsx index f5abf34..f403d46 100644 --- a/src/components/survey/PreviewChoices.jsx +++ b/src/components/survey/PreviewChoices.jsx @@ -1,6 +1,7 @@ import '../../styles/survey/PreviewChoices.css'; import { getCountdown } from '../../helpers/time-util'; import { ChoiceBoxUnvoted, ChoiceBoxVoted } from './ChoiceBox'; +import { sortChoicesByVoteDescending } from '../../helpers/survey-util'; export default function PreviewChoices({ choices, endTime }) { @@ -23,7 +24,7 @@ export default function PreviewChoices({ choices, endTime }) { } </div> <div className="choiceboxes"> - {choices.slice(0, 3).map((choice, idx) => ( + {sortChoicesByVoteDescending(choices).slice(0, 3).map((choice, idx) => ( <> {isUserHasVoted ? ( <ChoiceBoxVoted diff --git a/src/helpers/survey-util.js b/src/helpers/survey-util.js new file mode 100644 index 0000000..4b3090d --- /dev/null +++ b/src/helpers/survey-util.js @@ -0,0 +1,3 @@ +export const sortChoicesByVoteDescending = (choices) => ( + choices.sort((a, b) => parseFloat(b.votes) - parseFloat(a.votes)) +); -- GitLab From 819a9e78fb5023b5a8ff0176195e619bdd9468f5 Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Tue, 14 Dec 2021 22:39:26 +0700 Subject: [PATCH 17/19] fix(route): fix url hrefs on list threads & list surveys --- src/components/survey/ListSurveys.jsx | 8 ++++---- src/components/survey/RecentSurveys.jsx | 2 +- src/components/survey/TopSurveys.jsx | 2 +- src/components/thread/ListThreads.jsx | 8 ++++---- src/components/thread/RecentThreads.jsx | 2 +- src/components/thread/TopThreads.jsx | 2 +- src/routes/App.jsx | 22 ++++++++++++++++++++++ 7 files changed, 34 insertions(+), 12 deletions(-) diff --git a/src/components/survey/ListSurveys.jsx b/src/components/survey/ListSurveys.jsx index d55b144..85aea63 100644 --- a/src/components/survey/ListSurveys.jsx +++ b/src/components/survey/ListSurveys.jsx @@ -19,10 +19,10 @@ export default function ListSurveys(props) { <div className="listSurveysSection"> <div className="subHeaderListSurveys"> <div className="tab"> - <Link to="surveys/page/1" style={{ textDecoration: 'none' }}> + <Link to="/surveys/page/1" style={{ textDecoration: 'none' }}> <h3 className="active">Recent</h3> </Link> - <Link to="surveys/top/page/1" style={{ textDecoration: 'none' }}> + <Link to="/surveys/top/page/1" style={{ textDecoration: 'none' }}> <h3>Top</h3> </Link> </div> @@ -39,10 +39,10 @@ export default function ListSurveys(props) { <div className="listSurveysSection"> <div className="subHeaderListSurveys"> <div className="tab"> - <Link to="/page/1" style={{ textDecoration: 'none' }}> + <Link to="/surveys/page/1" style={{ textDecoration: 'none' }}> <h3>Recent</h3> </Link> - <Link to="/top/page/1" style={{ textDecoration: 'none' }}> + <Link to="/surveys/top/page/1" style={{ textDecoration: 'none' }}> <h3 className="active">Top</h3> </Link> </div> diff --git a/src/components/survey/RecentSurveys.jsx b/src/components/survey/RecentSurveys.jsx index 6b0d481..a498d50 100644 --- a/src/components/survey/RecentSurveys.jsx +++ b/src/components/survey/RecentSurveys.jsx @@ -31,7 +31,7 @@ export default function RecentSurveys(props) { }, [currentPage]); function switchPage(pageNumber) { - props.history.push(`/page/${pageNumber}`); + props.history.push(`/surveys/page/${pageNumber}`); window.location.reload(); } diff --git a/src/components/survey/TopSurveys.jsx b/src/components/survey/TopSurveys.jsx index 796a8c0..50c9b34 100644 --- a/src/components/survey/TopSurveys.jsx +++ b/src/components/survey/TopSurveys.jsx @@ -30,7 +30,7 @@ export default function TopSurveys(props) { }, [currentPage]); function switchPage(pageNumber) { - props.history.push(`/top/page/${pageNumber}`); + props.history.push(`/surveys/top/page/${pageNumber}`); window.location.reload(); } diff --git a/src/components/thread/ListThreads.jsx b/src/components/thread/ListThreads.jsx index 665c5ef..b2de847 100644 --- a/src/components/thread/ListThreads.jsx +++ b/src/components/thread/ListThreads.jsx @@ -19,10 +19,10 @@ export default function ListThreads(props) { <div className="listThreadsSection"> <div className="subHeaderListThreads"> <div className="tab"> - <Link to="/page/1" style={{ textDecoration: 'none' }}> + <Link to="/threads/page/1" style={{ textDecoration: 'none' }}> <h3 className="active">Recent</h3> </Link> - <Link to="/top/page/1" style={{ textDecoration: 'none' }}> + <Link to="/threads/top/page/1" style={{ textDecoration: 'none' }}> <h3>Top</h3> </Link> </div> @@ -39,10 +39,10 @@ export default function ListThreads(props) { <div className="listThreadsSection"> <div className="subHeaderListThreads"> <div className="tab"> - <Link to="/page/1" style={{ textDecoration: 'none' }}> + <Link to="/threads/page/1" style={{ textDecoration: 'none' }}> <h3>Recent</h3> </Link> - <Link to="/top/page/1" style={{ textDecoration: 'none' }}> + <Link to="/threads/top/page/1" style={{ textDecoration: 'none' }}> <h3 className="active">Top</h3> </Link> </div> diff --git a/src/components/thread/RecentThreads.jsx b/src/components/thread/RecentThreads.jsx index ea47f67..b1fa230 100644 --- a/src/components/thread/RecentThreads.jsx +++ b/src/components/thread/RecentThreads.jsx @@ -28,7 +28,7 @@ export default function RecentThreads(props) { }, [currentPage]); function switchPage(pageNumber) { - props.history.push(`/page/${pageNumber}`); + props.history.push(`/threads/page/${pageNumber}`); window.location.reload(); } diff --git a/src/components/thread/TopThreads.jsx b/src/components/thread/TopThreads.jsx index c33b302..e075752 100644 --- a/src/components/thread/TopThreads.jsx +++ b/src/components/thread/TopThreads.jsx @@ -27,7 +27,7 @@ export default function TopThreads(props) { }, [currentPage]); function switchPage(pageNumber) { - props.history.push(`/top/page/${pageNumber}`); + props.history.push(`/threads/top/page/${pageNumber}`); window.location.reload(); } diff --git a/src/routes/App.jsx b/src/routes/App.jsx index 71e4933..1c277d6 100644 --- a/src/routes/App.jsx +++ b/src/routes/App.jsx @@ -50,6 +50,28 @@ function App() { /> )} /> + <Route + exact + path="/surveys/page/:pageNumber" + component={(props) => ( + <ListSurveys + isRecent={true} + pageNumber={props.match.params.pageNumber} + history={props.history} + /> + )} + /> + <Route + exact + path="/surveys/top/page/:pageNumber" + component={(props) => ( + <ListSurveys + isRecent={false} + pageNumber={props.match.params.pageNumber} + history={props.history} + /> + )} + /> <Route exact path="/topic" component={TopicList} /> <Route exact path="/profile/:user/page/:pageNumber" component={Profile} /> <Route exact path="/login" component={LoginForm} /> -- GitLab From 98e83c320e31aa4c2a3773c654f2cca6f5ff2e9a Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Tue, 14 Dec 2021 22:42:24 +0700 Subject: [PATCH 18/19] fix(navbar): change survey nav-item href --- src/components/utility/Navbar.jsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/components/utility/Navbar.jsx b/src/components/utility/Navbar.jsx index 8971267..a295ba2 100644 --- a/src/components/utility/Navbar.jsx +++ b/src/components/utility/Navbar.jsx @@ -72,7 +72,7 @@ const Navbar = (props) => { exact activeClassName="navbar--active" className="nav-link" - to="/surveys" + to="/surveys/page/1" > <b>Surveys</b> </NavLink> -- GitLab From 9112d6f7d73e308296c92825c09646d063597b7b Mon Sep 17 00:00:00 2001 From: Fahdii Ajmalal Fikrie <fahdiaf@gmail.com> Date: Tue, 14 Dec 2021 23:08:52 +0700 Subject: [PATCH 19/19] hotfix(netlify): add _redirects file --- public/_redirects | 1 + 1 file changed, 1 insertion(+) create mode 100644 public/_redirects diff --git a/public/_redirects b/public/_redirects new file mode 100644 index 0000000..7797f7c --- /dev/null +++ b/public/_redirects @@ -0,0 +1 @@ +/* /index.html 200 -- GitLab