自動化無しに生活無し

WEB開発関係を中心に備忘録をまとめています

DRF+ReactのSPAで簡易掲示板を作る

thumbnail

最近(2018年以降)の開発では関数ベースのコンポーネントが主流のため、一部を書き換えた。

React

App.js

import React, { useState, useEffect } from "react";

import Modal from "./components/Modal";
import axios from "axios";

const App = () => {
    const [topicList, setTopicList] = useState([]);
    const [modal, setModal] = useState(false);
    const [activeTopic, setActiveTopic] = useState({ comment: "" });

    // コンポーネントがレンダリングされるときに実行する
    useEffect(() => {
        refreshList();
    }, []);
    
    // ページロード
    const refreshList = () => {
        axios
            .get("/api/topics/")
            .then((res) => setTopicList(res.data))
            .catch((err) => console.log(err));
    };

    const handleSubmit = (topic) => {
        if (topic.id) {
            axios
                .put(`/api/topics/${topic.id}/`, topic)
                .then(() => refreshList())
                .catch((err) => console.log(err));
        } else {
            axios
                .post("/api/topics/", topic)
                .then(() => refreshList())
                .catch((err) => console.log(err));
        }
        closeModal();
    };

    const handleDelete = (topic) => {
        axios
            .delete(`/api/topics/${topic.id}/`)
            .then(() => refreshList());
    };

    const openModal = (topic) => {
        if (topic.id) {
            setActiveTopic(topic);
        } else {
            setActiveTopic({ comment: "" });
        }
        setModal(true);
    };

    const closeModal = () => {
        setActiveTopic({ comment: "" });
        setModal(false);
    };


    // \n を <br> にする 
    const linebreaksbr = (string) => {
        return string.split('\n').map((topic, index) => (
            <React.Fragment key={index}>
                {topic}
                {index !== string.split('\n').length - 1 && <br />}
            </React.Fragment>
        ));
    };

    const renderItems = () => {
        return topicList.map((topic) => (
            <div className="border" key={topic.id}>
                <div>{topic.id}</div>
                <div>{linebreaksbr(topic.comment)}</div>
                <div className="text-end">
                    <input type="button" className="mx-1 btn btn-success" value="編集" onClick={() => openModal(topic)} />
                    <input type="button" className="mx-1 btn btn-danger" value="削除" onClick={() => handleDelete(topic)} />
                </div>
            </div>
        ));
    };

    return (
        <>
            <h1 className="bg-primary text-white text-center">簡易掲示板</h1>
            <main className="container">
                <input className="btn btn-primary" type="button" onClick={() => openModal(activeTopic)} value="新規作成" />
                {modal ? (
                    <Modal 
                        activeTopic={activeTopic}
                        handleSubmit={handleSubmit}
                        closeModal={closeModal} 
                    />
                ) : null}
                {renderItems()}
            </main>
        </>
    );
};

export default App;

component/Modal.js

import React, { useState, useEffect } from "react";

const Modal = ( { handleSubmit, activeTopic, closeModal } ) => {

    const [topic, setTopic] = useState(activeTopic);

    useEffect(() => {
        setTopic(activeTopic);
    }, [activeTopic]);

    const handleChange    = (e) => {
        let { name , value }    = e.target;
        setTopic({ ...topic, [name]: value });
    }

    return ( 
        <>
            <div className="modal_area" >
                <div className="modal_bg_area" onClick={closeModal}></div>
                <div className="modal_content_area">
                    <form>
                        
                        { activeTopic.id ? ( <h2>編集</h2> ) : ( <h2>新規作成</h2> ) }
                        <textarea className="form-control" name="comment" onChange={handleChange} value={topic.comment}></textarea>
                        <input className="btn btn-success" type="button" onClick={ () => handleSubmit(topic) } value="保存" />
                    </form>
                </div>
            </div>
        </>
    );

}
export default Modal;

ソースコード

https://github.com/seiya0723/react-django-startup-bbs-function

スポンサーリンク