0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

React 数字のボタンを表示し押下で数字が打ち込まれるコンポーネント作成

Last updated at Posted at 2024-11-17

はじめに

React初心者が頑張って、数字の打ち込みができるコンポーネントを作りました。
他になにか作るときにも使えるので自分用にここに残しておきます。

早速本題

親コンポーネント(Index.jsx)

import React, { useState, useEffect } from 'react';
import { Link, useForm, usePage } from '@inertiajs/react';
import FormInputMath from '@/Components/FormInputMath';

const Index = ({ sortings }) => {

    // 送信するデータ
    const { data, setData, post, processing, errors } = useForm({
        kari_price: '',
        kashi_price: '',
    });

    // 送信ボタン押下で指定のルートにデータを飛ばす
    const handleSubmit = (e) => {
        e.preventDefault();
        post(route('sorting'));
    };

    // setDataでdataを更新
    const handleInputChange = (name, value) => {
        setData(name, value);
    };
    
//(略)

// コンポーネントの使用 nameとonChangeをpropsで渡す
<FormInputMath
    name="kari_price"
    onChange={handleInputChange}
/>

//(略)
<FormInputMath
    name="kashi_price"
    onChange={handleInputChange}
/>

//(略)

子コンポーネント (FormInputMath.jsx)

import React, { useState } from 'react';

const FormInputMath = ({ onChange, name }) => {
    const [inputValue, setInputValue] = useState('');
    const [showButtons, setShowButtons] = useState(false);

    // ボタン押下で数値を追加する関数
    const handleButtonClick = (e, number) => {
        e.preventDefault();  // フォーム送信を防ぐ form内のbuttonはtypeをしてしないとsubmitになるみたい。type=buuttonと記述すればこの一行は必要ない。
        const newValue = inputValue + number;
        // inputValueの更新
        setInputValue(newValue);
        if (onChange) {
            // 親コンポーネントから渡ったonChangeを発火させる
            onChange(name, newValue); 
        }
    };

    // inputが押下されたときにボタン表示を切り替える関数
    const handleInputClick = () => {
        setShowButtons(!showButtons);
    };

    // キーボード入力を処理する関数を追加
    const handleInputChange = (e) => {
        const value = e.target.value;
        // 数字のみを許可する正規表現
        if (/^\d*$/.test(value)) {
            setInputValue(value);
            if (onChange) {
                // 親コンポーネントから渡ったonChangeを発火させる
                onChange(name, value);
            }
        }
    };

    // バックスペース機能を追加
    const handleBackspace = (e) => {
        e.preventDefault();
        const newValue = inputValue.slice(0, -1);
        setInputValue(newValue);
        if (onChange) {
            // 親コンポーネントから渡ったonChangeを発火させる
            onChange(name, newValue);
        }
    };

    return (
        <div>
            <input
                // 親から渡ったnameをはめる
                name={name}
                className='form-control form-control-sm'
                type="text"
                value={inputValue}
                onClick={handleInputClick} // 表示・非表示制御
                onChange={handleInputChange}  // キーボード入力の処理
            />

            {showButtons && (
                <div style={{ display: 'flex', flexWrap: 'wrap' }}>
                    {[...Array(10)].map((_, index) => (
                        // 0~9の数字を展開
                        <button
                            className='button'
                            key={index}
                            onClick={(e) => handleButtonClick(e, index)} // 数字押下で押した数字を末尾に加える処理
                        >
                            {index}
                        </button>
                    ))}
                    <button
                        className='button'
                        onClick={handleBackspace} // ← 押下で末尾の数字を削除
                    ></button>
                </div>
            )}
        </div>
    );
};

export default FormInputMath;

cssは別途用意してくださいね!

追記

更新処理の際の記述と注意点もまとめておきます。
親コンポ―ネント Index.jsx

import React, { useState, useEffect } from 'react';
import { Link, useForm, usePage } from '@inertiajs/react';
import FormInputMath from '@/Components/FormInputMath';

const Index = ({ sortings }) => {

    // 送信するデータ
    const { data, setData, post, processing, errors } = useForm({
-        kari_price: '',
-        kashi_price: '',
+        kari_price: sortings.kari_price,
+        kashi_price: sortings_kashi_price,
    });

    // 送信ボタン押下で指定のルートにデータを飛ばす
    const handleSubmit = (e) => {
        e.preventDefault();
        post(route('sorting'));
    };

    // setDataでdataを更新
    const handleInputChange = (name, value) => {
        setData(name, value);
    };
    
//(略)

// コンポーネントの使用 nameとonChangeをpropsで渡す
<FormInputMath
+   value={data.kari_price}
    name="kari_price"
    onChange={handleInputChange}
/>

//(略)
<FormInputMath
+   value={data.kashi_price}
    name="kashi_price"
    onChange={handleInputChange}
/>

//(略)

子コンポーネント FormInputMath.jsx

- import React, { useState } from 'react';
+ import React, { useState, useEffect } from 'react';

- const FormInputMath = ({ onChange, name }) => {
+ const FormInputMath = ({ onChange, name, value }) => {
 
-    const [inputValue, setInputValue] = useState('');
+    const newValue = inputValue +  String(number); //文字列に変換しないと加算される
    const [showButtons, setShowButtons] = useState(false);

    // ボタン押下で数値を追加する関数
    const handleButtonClick = (e, number) => {
        e.preventDefault();  
-        const newValue = inputValue + number;
+        const newValue = inputValue +  String(number); //文字列に変換しないと加算される

+    // valueの監視、更新
+    useEffect(() => {
+        setInputValue( String(value || '')); //更新するデータをはめ込む
+    }, [value]);
    
        // inputValueの更新
        setInputValue(newValue);
        if (onChange) {
            // 親コンポーネントから渡ったonChangeを発火させる
            onChange(name, newValue); 
        }
    };

    // inputが押下されたときにボタン表示を切り替える関数
    const handleInputClick = () => {
        setShowButtons(!showButtons);
    };

    // キーボード入力を処理する関数を追加
    const handleInputChange = (e) => {
        const value = e.target.value;
        // 数字のみを許可する正規表現
        if (/^\d*$/.test(value)) {
            setInputValue(value);
            if (onChange) {
                // 親コンポーネントから渡ったonChangeを発火させる
                onChange(name, value);
            }
        }
    };

    // バックスペース機能を追加
    const handleBackspace = (e) => {
        e.preventDefault();
        const newValue = inputValue.slice(0, -1);
        setInputValue(newValue);
        if (onChange) {
            // 親コンポーネントから渡ったonChangeを発火させる
            onChange(name, newValue);
        }
    };

    return (
        <div>
            <input
                // 親から渡ったnameをはめる
                name={name}
                className='form-control form-control-sm'
                type="text"
                value={inputValue}
                onClick={handleInputClick} // 表示・非表示制御
                onChange={handleInputChange}  // キーボード入力の処理
            />

            {showButtons && (
                <div style={{ display: 'flex', flexWrap: 'wrap' }}>
                    {[...Array(10)].map((_, index) => (
                        // 0~9の数字を展開
                        <button
                            className='button'
                            key={index}
                            onClick={(e) => handleButtonClick(e, index)} // 数字押下で押した数字を末尾に加える処理
                        >
                            {index}
                        </button>
                    ))}
                    <button
                        className='button'
                        onClick={handleBackspace} // ← 押下で末尾の数字を削除
                    ></button>
                </div>
            )}
        </div>
    );
};

export default FormInputMath;

※注意点として
子コンポーネントのconst newValue = inputValue + String(number); で
numberを今回文字列に変換している。

更新処理の場合、データに入っているのは数値型なので
数字型 + 数字型 になり加算されてしまう。

1+1=2
けどボタンをおしたら末尾に入るようにしたいので文字列に変換する。

1+"1"="11"

ではなぜ更新処理で急に加算になったのか?
今までは動いたのか?
これは最初の登録処理の際
""+1="1" このように最初は空文字で文字列扱いはので結果は文字列の1に変換されるからだ。

これはハマりそう・・・

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?