1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

AWS AmplifyとAWS×フロントエンド #AWSAmplifyJPAdvent Calendar 2024

Day 13

generateされないcustom属性のフォームの書き方

Posted at

Amplifyはフォームを勝手に作ってくれるnpx ampx generate formsコマンドがめちゃくちゃ便利です。

データを定義して、npx ampx generate formsを実行するだけで、いい感じの入力フォームが勝手に作られます。

そんな便利なnpx ampx generate formsコマンドですが、残念なのが、カスタムタイプを作ってくれないことです。

そこでこの記事では、generateform で作成したコンポーネントにカスタムタイプの入力フォームを付け足す方法を紹介します。

対象読者

想定しているのは、webサービスをさくっとつくりたくて、Amplifyを勉強しはじめた人を対象としています。
私も簡単にwebサービスつくりたくてプログラミング勉強しはじめて、個人開発だからなんでもかんでもやる必要があって、インフラ、バックエンド回りの手間を省けないかなといろいろ探した結果Amplifyにたどり着きましたので、それほど詳しくありません。
そもそも職業エンジニアではないので。
TypeScriptに詳しい人はコードみれば、何してるかわかると思うので、あまり意味ないかもしれません。

簡単な自己紹介

webサービスをつくりたくて、プログラミングを勉強しはじめました。普段は機械メーカーの営業をしていますので、本職のエンジニアほど詳しくないと思います。

せっかくアドヴェントカレンダーで書くので、私の個人開発したwebサービスの宣伝を。

製造業向けのバーチャル展示会サービスを運営しております。

サービス開発の背景

一度開催してお役御免となる展示品を有効活用するためのプラットフォームです。バーチャル展示会でしたら、ユーザーのインタラクションも取得できるため、そこからユーザーの興味、関心に基づく提案もできるため営業活動の助けにもなると思います。
BtoBのサービス且つ、製造業向けというかなりユーザーを限定したサービスなので、関わりある人がどれくらいいるかですが、もし興味があれば。

このサービスを作った動機は、簡潔にいうと2つです。
・展示会を開催するのに、準備からフォローまでとてもコストがかかるのに、1回きりになるのはもったいない
・もっと自分の営業活動を効率化して楽をしたかった。

この2点です。まだはじまったばかりなので、どれくらい広まるかわかりませんが暖かく見守ってください。
蛇足ですが、こちらのサービスもAmplifyで作りました。

さて、本題に戻ります。

カスタム属性の作り方に入る前npx ampx generate formsで生成されるフォームに関しては、こちらを参照してください。
https://docs.amplify.aws/react/build-ui/formbuilder/

とりあえず、dataの構造はこんな感じにしておきます。

DummyData:a
      .model({
      name: a.string(),
      // fields can be of custom types
      location: a.customType({
        // fields can be required or optional
        lat: a.float().required(),
        long: a.float().required(),
      }),
      // fields can be enums
      engagementStage: a.enum(["PROSPECT", "INTERESTED", "PURCHASED"]),
      })
      .authorization(allow => [
        allow.authenticated(),
      ]),

とりあえず、こんな感じのモデルにしましょう。

そして、フォームを作成するとこんな感じにフォームが出来上がります。


/* eslint-disable */
"use client";
import * as React from "react";
import {
  Button,
  Flex,
  Grid,
  SelectField,
  TextField,
} from "@aws-amplify/ui-react";
import { fetchByPath, getOverrideProps, validateField } from "./utils";
import { generateClient } from "aws-amplify/api";
import { createDummyData } from "./graphql/mutations";
const client = generateClient();
export default function DummyDataCreateForm(props) {
  const {
    clearOnSuccess = true,
    onSuccess,
    onError,
    onSubmit,
    onValidate,
    onChange,
    overrides,
    ...rest
  } = props;
  const initialValues = {
    name: "",
    engagementStage: "",
  };
  const [name, setName] = React.useState(initialValues.name);
  const [engagementStage, setEngagementStage] = React.useState(
    initialValues.engagementStage
  );
  const [errors, setErrors] = React.useState({});
  const resetStateValues = () => {
    setName(initialValues.name);
    setEngagementStage(initialValues.engagementStage);
    setErrors({});
  };
  const validations = {
    name: [],
    engagementStage: [],
  };
  const runValidationTasks = async (
    fieldName,
    currentValue,
    getDisplayValue
  ) => {
    const value =
      currentValue && getDisplayValue
        ? getDisplayValue(currentValue)
        : currentValue;
    let validationResponse = validateField(value, validations[fieldName]);
    const customValidator = fetchByPath(onValidate, fieldName);
    if (customValidator) {
      validationResponse = await customValidator(value, validationResponse);
    }
    setErrors((errors) => ({ ...errors, [fieldName]: validationResponse }));
    return validationResponse;
  };
  return (
    <Grid
      as="form"
      rowGap="15px"
      columnGap="15px"
      padding="20px"
      onSubmit={async (event) => {
        event.preventDefault();
        let modelFields = {
          name,
          engagementStage,
        };
        const validationResponses = await Promise.all(
          Object.keys(validations).reduce((promises, fieldName) => {
            if (Array.isArray(modelFields[fieldName])) {
              promises.push(
                ...modelFields[fieldName].map((item) =>
                  runValidationTasks(fieldName, item)
                )
              );
              return promises;
            }
            promises.push(
              runValidationTasks(fieldName, modelFields[fieldName])
            );
            return promises;
          }, [])
        );
        if (validationResponses.some((r) => r.hasError)) {
          return;
        }
        if (onSubmit) {
          modelFields = onSubmit(modelFields);
        }
        try {
          Object.entries(modelFields).forEach(([key, value]) => {
            if (typeof value === "string" && value === "") {
              modelFields[key] = null;
            }
          });
          await client.graphql({
            query: createDummyData.replaceAll("__typename", ""),
            variables: {
              input: {
                ...modelFields,
              },
            },
          });
          if (onSuccess) {
            onSuccess(modelFields);
          }
          if (clearOnSuccess) {
            resetStateValues();
          }
        } catch (err) {
          if (onError) {
            const messages = err.errors.map((e) => e.message).join("\n");
            onError(modelFields, messages);
          }
        }
      }}
      {...getOverrideProps(overrides, "DummyDataCreateForm")}
      {...rest}
    >
      <TextField
        label="Name"
        isRequired={false}
        isReadOnly={false}
        value={name}
        onChange={(e) => {
          let { value } = e.target;
          if (onChange) {
            const modelFields = {
              name: value,
              engagementStage,
            };
            const result = onChange(modelFields);
            value = result?.name ?? value;
          }
          if (errors.name?.hasError) {
            runValidationTasks("name", value);
          }
          setName(value);
        }}
        onBlur={() => runValidationTasks("name", name)}
        errorMessage={errors.name?.errorMessage}
        hasError={errors.name?.hasError}
        {...getOverrideProps(overrides, "name")}
      ></TextField>
      <SelectField
        label="Engagement stage"
        placeholder="Please select an option"
        isDisabled={false}
        value={engagementStage}
        onChange={(e) => {
          let { value } = e.target;
          if (onChange) {
            const modelFields = {
              name,
              engagementStage: value,
            };
            const result = onChange(modelFields);
            value = result?.engagementStage ?? value;
          }
          if (errors.engagementStage?.hasError) {
            runValidationTasks("engagementStage", value);
          }
          setEngagementStage(value);
        }}
        onBlur={() => runValidationTasks("engagementStage", engagementStage)}
        errorMessage={errors.engagementStage?.errorMessage}
        hasError={errors.engagementStage?.hasError}
        {...getOverrideProps(overrides, "engagementStage")}
      >
        <option
          children="Prospect"
          value="PROSPECT"
          {...getOverrideProps(overrides, "engagementStageoption0")}
        ></option>
        <option
          children="Interested"
          value="INTERESTED"
          {...getOverrideProps(overrides, "engagementStageoption1")}
        ></option>
        <option
          children="Purchased"
          value="PURCHASED"
          {...getOverrideProps(overrides, "engagementStageoption2")}
        ></option>
      </SelectField>
      <Flex
        justifyContent="space-between"
        {...getOverrideProps(overrides, "CTAFlex")}
      >
        <Button
          children="Clear"
          type="reset"
          onClick={(event) => {
            event.preventDefault();
            resetStateValues();
          }}
          {...getOverrideProps(overrides, "ClearButton")}
        ></Button>
        <Flex
          gap="15px"
          {...getOverrideProps(overrides, "RightAlignCTASubFlex")}
        >
          <Button
            children="Submit"
            type="submit"
            variation="primary"
            isDisabled={Object.values(errors).some((e) => e?.hasError)}
            {...getOverrideProps(overrides, "SubmitButton")}
          ></Button>
        </Flex>
      </Flex>
    </Grid>
  );
}

この単純なデータと一行のコマンドで結構複雑なコンポーネントが作成されています。

ただ、カスタム属性のフォームが用意されておりません。
サポートに確認しましたが、カスタム属性に関しては、npx ampx generate formsのコマンドでは、フォームを作成してくれないそうなので、自前で作るしかありません。それか、カスタム属性を使うことをあきらめるか。

カスタム属性のフォームの作り方を紹介する前に、そもそも、このフォームの構造を見てみましょう。

useState: フォームフィールド (name と engagementStage) の状態管理を行います。

resetStateValues: フォームの入力値を初期状態にリセットする関数。

validations: 各フィールドに対するバリデーションルールの設定。現時点では空の配列になっています。
ここは、モデルにrequiredとかつけると、必須項目にしてくれます。

runValidationTasks: 指定したフィールドに対してバリデーションを実行する関数。

onSubmit:modelFieldsを使ったデータの書き込み

npx ampx generate formsがやってくれること

状態管理(初期値の設定)
バリデーション
フォームの整形
データの書き込み

なので、カスタムタイプ用にそれぞれ付け足していけばいいだけといえばだけなんです。

それではどんな感じに付け足していくかやってみましょう。

const initialValues = {
    name: "",
    engagementStage: "",
    location: { lat: "", long: "" },
  };
  const [name, setName] = React.useState(initialValues.name);
  const [engagementStage, setEngagementStage] = React.useState(
    initialValues.engagementStage
  );
  const [lat, setLat] = React.useState(initialValues.location.lat);
  const [long, setLong] = React.useState(initialValues.location.long);
  const [errors, setErrors] = React.useState({});
  const resetStateValues = () => {
    setName(initialValues.name);
    setEngagementStage(initialValues.engagementStage);
    setLat(initialValues.location.lat);
    setLong(initialValues.location.long);
    setErrors({});
  };

これで状態管理と、初期値の設定とresetStateValuesの設定ができます。

const validations = {
    name: [],
    engagementStage: [],
    lat: [{ type: "required", message: "Latitude is required" }],
    long: [{ type: "required", message: "Longitude is required" }],
  };

requiredをつけているので、バリデーションの条件はこんな感じでやってあげましょう。

<TextField
        label="Latitude"
        value={lat}
        onChange={(e) => setLat(e.target.value)}
        onBlur={() => runValidationTasks("lat", lat)}
        errorMessage={errors.lat?.errorMessage}
        hasError={errors.lat?.hasError}
        {...getOverrideProps(overrides, "lat")}
/>
<TextField
        label="Longitude"
        value={long}
        onChange={(e) => setLong(e.target.value)}
        onBlur={() => runValidationTasks("long", long)}
        errorMessage={errors.long?.errorMessage}
        hasError={errors.long?.hasError}
        {...getOverrideProps(overrides, "long")}
/>

フォームはこんな感じで整形してあげます。

そして、

onSubmitのmodelFieldsのところを

let modelFields = {
          name,
          engagementStage,
          location: { lat: parseFloat(lat), long: parseFloat(long) },
};

こんな感じで付け足せば完成です。

大量のコードが生成されるので、慣れていないと尻込みしますが、
やっていることは大したことないので、一個一個分解しながら理解すれば、
それほど複雑ではないと思います。

Amplifyで快適な開発体験を一緒に楽しみましょう

1
1
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
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?