Add: modify review
This commit is contained in:
@@ -95,6 +95,8 @@ func NewAPI(config Config) (*API, error) {
|
|||||||
// review
|
// review
|
||||||
apiMux.HandleFunc("/insert_review", api.HandleInsertReview)
|
apiMux.HandleFunc("/insert_review", api.HandleInsertReview)
|
||||||
apiMux.HandleFunc("/get_reviews_on_file", api.HandleGetReviewsOnFile)
|
apiMux.HandleFunc("/get_reviews_on_file", api.HandleGetReviewsOnFile)
|
||||||
|
apiMux.HandleFunc("/get_review", api.HandleGetReview)
|
||||||
|
apiMux.HandleFunc("/update_review", api.HandleUpdateReview)
|
||||||
// below needs token
|
// below needs token
|
||||||
apiMux.HandleFunc("/walk", api.HandleWalk)
|
apiMux.HandleFunc("/walk", api.HandleWalk)
|
||||||
apiMux.HandleFunc("/reset", api.HandleReset)
|
apiMux.HandleFunc("/reset", api.HandleReset)
|
||||||
|
|||||||
@@ -67,3 +67,57 @@ func (api *API) HandleGetReviewsOnFile(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type GetReviewRequest struct {
|
||||||
|
ID int64 `json:"id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type GetReviewResponse struct {
|
||||||
|
Review *database.Review `json:"review"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (api *API) HandleGetReview(w http.ResponseWriter, r *http.Request) {
|
||||||
|
req := &GetReviewRequest{}
|
||||||
|
|
||||||
|
err := json.NewDecoder(r.Body).Decode(req)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
review, err := api.Db.GetReview(req.ID)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ret := &GetReviewResponse{
|
||||||
|
Review: review,
|
||||||
|
}
|
||||||
|
|
||||||
|
err = json.NewEncoder(w).Encode(ret)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (api *API) HandleUpdateReview(w http.ResponseWriter, r *http.Request) {
|
||||||
|
req := &database.Review{}
|
||||||
|
|
||||||
|
err := json.NewDecoder(r.Body).Decode(req)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
req.UpdatedAt = time.Now().Unix()
|
||||||
|
|
||||||
|
err = api.Db.UpdateReview(req)
|
||||||
|
if err != nil {
|
||||||
|
api.HandleError(w, r, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
api.HandleOK(w, r)
|
||||||
|
}
|
||||||
|
|||||||
@@ -40,3 +40,29 @@ func (database *Database) GetReviewsOnFile(fileId int64) ([]*Review, error) {
|
|||||||
}
|
}
|
||||||
return reviews, nil
|
return reviews, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (database *Database) GetReview(reviewId int64) (*Review, error) {
|
||||||
|
row := database.stmt.getReview.QueryRow(reviewId)
|
||||||
|
|
||||||
|
review := &Review{}
|
||||||
|
err := row.Scan(
|
||||||
|
&review.ID,
|
||||||
|
&review.FileId,
|
||||||
|
&review.UserId,
|
||||||
|
&review.CreatedAt,
|
||||||
|
&review.UpdatedAt,
|
||||||
|
&review.Content)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return review, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (database *Database) UpdateReview(review *Review) error {
|
||||||
|
_, err := database.stmt.updateReview.Exec(
|
||||||
|
review.Content,
|
||||||
|
review.UpdatedAt,
|
||||||
|
review.ID)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|||||||
@@ -218,6 +218,10 @@ WHERE reviews.file_id = ?
|
|||||||
ORDER BY reviews.created_at
|
ORDER BY reviews.created_at
|
||||||
;`
|
;`
|
||||||
|
|
||||||
|
var getReviewQuery = `SELECT id, file_id, user_id, created_at, updated_at, content FROM reviews WHERE id = ? LIMIT 1;`
|
||||||
|
|
||||||
|
var updateReviewQuery = `UPDATE reviews SET content = ?, updated_at = ? WHERE id = ?;`
|
||||||
|
|
||||||
type Stmt struct {
|
type Stmt struct {
|
||||||
initFilesTable *sql.Stmt
|
initFilesTable *sql.Stmt
|
||||||
initFoldersTable *sql.Stmt
|
initFoldersTable *sql.Stmt
|
||||||
@@ -260,6 +264,8 @@ type Stmt struct {
|
|||||||
updateFoldername *sql.Stmt
|
updateFoldername *sql.Stmt
|
||||||
insertReview *sql.Stmt
|
insertReview *sql.Stmt
|
||||||
getReviewsOnFile *sql.Stmt
|
getReviewsOnFile *sql.Stmt
|
||||||
|
getReview *sql.Stmt
|
||||||
|
updateReview *sql.Stmt
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewPreparedStatement(sqlConn *sql.DB) (*Stmt, error) {
|
func NewPreparedStatement(sqlConn *sql.DB) (*Stmt, error) {
|
||||||
@@ -576,5 +582,17 @@ func NewPreparedStatement(sqlConn *sql.DB) (*Stmt, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// init getReview
|
||||||
|
stmt.getReview, err = sqlConn.Prepare(getReviewQuery)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// init updateReview
|
||||||
|
stmt.updateReview, err = sqlConn.Prepare(updateReviewQuery)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
return stmt, err
|
return stmt, err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import Login from "./component/Login";
|
|||||||
import Register from "./component/Register";
|
import Register from "./component/Register";
|
||||||
import Tags from "./component/Tags";
|
import Tags from "./component/Tags";
|
||||||
import EditTag from "./component/EditTag";
|
import EditTag from "./component/EditTag";
|
||||||
|
import EditReview from "./component/EditReview";
|
||||||
import AudioPlayer from "./component/AudioPlayer";
|
import AudioPlayer from "./component/AudioPlayer";
|
||||||
import UserStatus from "./component/UserStatus";
|
import UserStatus from "./component/UserStatus";
|
||||||
import ReviewPage from "./component/ReviewPage";
|
import ReviewPage from "./component/ReviewPage";
|
||||||
@@ -83,6 +84,10 @@ function App() {
|
|||||||
path="/manage/tags/:id"
|
path="/manage/tags/:id"
|
||||||
element={<EditTag user={user} />}
|
element={<EditTag user={user} />}
|
||||||
/>
|
/>
|
||||||
|
<Route
|
||||||
|
path="/manage/reviews/:id"
|
||||||
|
element={<EditReview user={user} />}
|
||||||
|
/>
|
||||||
<Route
|
<Route
|
||||||
path="/files/:id"
|
path="/files/:id"
|
||||||
element={<FileInfo setPlayingFile={setPlayingFile} />}
|
element={<FileInfo setPlayingFile={setPlayingFile} />}
|
||||||
@@ -93,7 +98,7 @@ function App() {
|
|||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path="/files/:id/review"
|
path="/files/:id/review"
|
||||||
element={<ReviewPage setPlayingFile={setPlayingFile} />}
|
element={<ReviewPage user={user} setPlayingFile={setPlayingFile} />}
|
||||||
/>
|
/>
|
||||||
</Routes>
|
</Routes>
|
||||||
</main>
|
</main>
|
||||||
|
|||||||
75
web/src/component/EditReview.js
Normal file
75
web/src/component/EditReview.js
Normal file
@@ -0,0 +1,75 @@
|
|||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { useParams, useNavigate } from "react-router";
|
||||||
|
|
||||||
|
function SingleReview() {
|
||||||
|
let params = useParams();
|
||||||
|
let navigate = useNavigate();
|
||||||
|
|
||||||
|
const [review, setReview] = useState({
|
||||||
|
id: "",
|
||||||
|
user_id: "",
|
||||||
|
file_id: "",
|
||||||
|
content: "",
|
||||||
|
created_at: "",
|
||||||
|
updated_at: "",
|
||||||
|
});
|
||||||
|
|
||||||
|
function refresh() {
|
||||||
|
fetch("/api/v1/get_review", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
id: parseInt(params.id),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
.then((response) => response.json())
|
||||||
|
.then((data) => {
|
||||||
|
if (data.error) {
|
||||||
|
alert(data.error);
|
||||||
|
} else {
|
||||||
|
setReview(data.review);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function save() {
|
||||||
|
fetch("/api/v1/update_review", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
id: parseInt(params.id),
|
||||||
|
content: review.content,
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
.then((response) => response.json())
|
||||||
|
.then((data) => {
|
||||||
|
if (data.error) {
|
||||||
|
alert(data.error);
|
||||||
|
} else {
|
||||||
|
alert("Review updated!");
|
||||||
|
navigate(-1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
refresh();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="page">
|
||||||
|
<h3>Edit Review</h3>
|
||||||
|
<textarea
|
||||||
|
value={review.content}
|
||||||
|
onChange={(e) => setReview({ ...review, content: e.target.value })}
|
||||||
|
></textarea>
|
||||||
|
<button onClick={() => save()}>Save</button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default SingleReview;
|
||||||
@@ -1,10 +1,11 @@
|
|||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { useParams } from "react-router";
|
import { useParams, useNavigate } from "react-router";
|
||||||
import { Link } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
import { convertIntToDateTime } from "./Common";
|
import { convertIntToDateTime } from "./Common";
|
||||||
|
|
||||||
function ReviewPage() {
|
function ReviewPage(props) {
|
||||||
let params = useParams();
|
let params = useParams();
|
||||||
|
let navigate = useNavigate();
|
||||||
const [newReview, setNewReview] = useState("");
|
const [newReview, setNewReview] = useState("");
|
||||||
const [reviews, setReviews] = useState([]);
|
const [reviews, setReviews] = useState([]);
|
||||||
|
|
||||||
@@ -61,10 +62,22 @@ function ReviewPage() {
|
|||||||
{reviews.map((review) => (
|
{reviews.map((review) => (
|
||||||
<div key={review.id}>
|
<div key={review.id}>
|
||||||
<h4>
|
<h4>
|
||||||
<Link to={`/manage/users/${review.user.id}`}>@{review.user.username}</Link> wrote on{" "}
|
<Link to={`/manage/users/${review.user.id}`}>
|
||||||
{convertIntToDateTime(review.created_at)}{" "}
|
@{review.user.username}
|
||||||
|
</Link>{" "}
|
||||||
|
wrote on {convertIntToDateTime(review.created_at)}{" "}
|
||||||
</h4>
|
</h4>
|
||||||
<p>{review.content}</p>
|
<p>{review.content}</p>
|
||||||
|
{(props.user.role === 1 || review.user.id === props.user.id) &&
|
||||||
|
props.user.role != 0 && (
|
||||||
|
<button
|
||||||
|
onClick={() => {
|
||||||
|
navigate(`/manage/reviews/${review.id}`);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Edit
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user