Получение пользовательских компонентов для работы в форме реакции-хука с помощью onChange

У меня есть корзина для оформления заказа, в которой у вас есть разные элементы корзины, и для каждого из них вы можете изменить количество перед покупкой.

Вот как выглядит код:

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

import PureInput from "./PureInput";
import { useForm, Controller } from "react-hook-form";

const CartInner = React.forwardRef(
  (
    {
      control,
      element,
      value,
      handleOnChange,
      images,
      name,
      monthlyAmount,
      price,
      projectedGrowth,
      id,
      ...inputProps
    }: any,
    ref: any
  ) => {
    return (
      <div className = "grid gap-8 grid-cols-2 mb-12 py-6 px-8 border-2 border-slate-200">
        <div>
          <PureInput
            min = {200}
            max = {price}
            onChange = {handleOnChange}
            type = "number"
            step = {200}
            defaultValue = {element.price}
            id = {id}
            ref = {ref}
            {...inputProps}
          />
        </div>
      </div>
    );
  }
);

export default function Checkout() {
  const { control, handleSubmit } = useForm();

  const handleOnChange = (index: any, e: any) => {
    console.info(e, "e");
  };

  const onSubmit = async (data: any) => {
    console.info(data, "data from Form.tsx");
  };

  return (
    <form onSubmit = {handleSubmit(onSubmit)} className = "grid gap-8 grid-cols-3">
      <div className = "col-span-2">
        {[0, 2].map((element, index) => {
          return (
            <fieldset key = {index}>
              <Controller
                render = {({ field }) => (
                  <CartInner
                    element = {element}
                    handleOnChange = {(e) => handleOnChange(index, e)}
                    {...field}
                  />
                )}
                name = {`test.${index}.lastName`}
                control = {control}
              />
            </fieldset>
          );
        })}
        <button>Progess to payment</button>
      </div>
    </form>
  );
}

И чистый ввод:

import * as React from "react";

type IProps = any;

const PureInput = React.forwardRef(
  ({ className, id, onChange, ...inputProps }: IProps, ref: any) => {
    return (
      <input
        id = {id}
        ref = {ref}
        onChange = {onChange}
        type = "input"
        className = {`${className} block w-full bg-white text-black rounded-md border-2 font-bold border-grey-200 text-xl px-4 py-4 focus:border-orange-500 focus:ring-orange-500`}
        {...inputProps}
      />
    );
  }
);

export default PureInput;

Все работает нормально с точки зрения отправки формы. Когда я это делаю, я получаю массив любых значений, которые я ввел на вход:

[{lastName: "1600"}
{lastName: "800"}]

Мои версии пакетов:

"react-dom": "18.2.0",
"react-hook-form": "^7.29.0",

Но мой onChange больше не срабатывает. Как я могу заставить onChange срабатывать, чтобы я мог регистрировать значение ввода внутри компонента <Checkout />?

Вот codeandbox если поможет

Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
В настоящее время производительность загрузки веб-сайта имеет решающее значение не только для удобства пользователей, но и для ранжирования в...
Безумие обратных вызовов в javascript [JS]
Безумие обратных вызовов в javascript [JS]
Здравствуйте! Юный падаван 🚀. Присоединяйся ко мне, чтобы разобраться в одной из самых запутанных концепций, когда вы начинаете изучать мир...
Система управления парковками с использованием HTML, CSS и JavaScript
Система управления парковками с использованием HTML, CSS и JavaScript
Веб-сайт по управлению парковками был создан с использованием HTML, CSS и JavaScript. Это простой сайт, ничего вычурного. Основная цель -...
JavaScript Вопросы с множественным выбором и ответы
JavaScript Вопросы с множественным выбором и ответы
Если вы ищете платформу, которая предоставляет вам бесплатный тест JavaScript MCQ (Multiple Choice Questions With Answers) для оценки ваших знаний,...
1
0
105
2
Перейти к ответу Данный вопрос помечен как решенный

Ответы 2

Эта часть документации привела меня к ответу:

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

import PureInput from "./PureInput";
import { useForm, Controller } from "react-hook-form";

const CartInner = React.forwardRef(
  ({ onChange, onBlur, name, label, ...inputProps }: any, ref: any) => {
    return (
      <input
        name = {name}
        ref = {ref}
        onChange = {onChange}
        onBlur = {onBlur}
        type = "number"
      />
    );
  }
);

export default function Checkout() {
  const { control, handleSubmit } = useForm();

  const handleOnChange = (index: any, e: any) => {
    console.info(e.target.value, "e");
  };

  const onSubmit = async (data: any) => {
    console.info(data, "data from Form.tsx");
  };

  return (
    <form onSubmit = {handleSubmit(onSubmit)} className = "grid gap-8 grid-cols-3">
      <div className = "col-span-2">
        {[0, 2].map((element, index) => {
          return (
            <fieldset key = {index}>
              <Controller
                render = {({ field: { onBlur, value, name, ref } }) => (
                  <CartInner
                    key = {index}
                    name = {name}
                    ref = {ref}
                    onChange = {(e) => handleOnChange(index, e)}
                    onBlur = {onBlur}
                  />
                )}
                name = {`test.${index}.lastName`}
                control = {control}
              />
            </fieldset>
          );
        })}
        <button>Progess to payment</button>
      </div>
    </form>
  );
}

// add delete
// total money
// add the cart documents to a history with a timestamp and show it was a BUY ORDER
// delete the documents from the cart

На самом деле это не работает, так как останавливает работу onSubmit!

a7dc 03.12.2022 06:03
Ответ принят как подходящий

Вы можете внести следующие изменения, чтобы подключиться к событию onChange

    // pass an event handler name with different name       
     <PureInput
        min = {200}
        max = {price}
        // pass a handler with different name as inputOptions overrides that prop
        handleOnChange = {handleOnChange}
        type = "number"
        step = {200}
        defaultValue = {element.price}
        id = {id}
        ref = {ref}
        {...inputProps}
     />

//plug into the default onchange to call you handler also
 <input
        id = {id}
        ref = {ref}
        onChange = {(e) => {
          console.info("on change");
          // call react-hook-form onChange
          onChange(e);
          // call your handler
          handleOnChange(e);
        }}
        type = "input"
        className = {`${className} block w-full bg-white text-black rounded-md border-2 font-bold border-grey-200 text-xl px-4 py-4 focus:border-orange-500 focus:ring-orange-500`}
        {...inputProps}
      />

Надеюсь, это поможет вам в решении вашей проблемы,

Ваше здоровье

Спасибо Аззи! Это разблокировало меня, ура

a7dc 03.12.2022 16:18

Другие вопросы по теме