diff --git a/src/apollo/defs.ts b/src/apollo/defs.ts index 0639a2d..4c4abc1 100644 --- a/src/apollo/defs.ts +++ b/src/apollo/defs.ts @@ -55,4 +55,12 @@ const FORMSUBMIT = gql` } ` -export { LOGIN, FORM, USER, FORMSUBMIT } +const CREATEFORM = gql` + mutation CreateForm($title: String!, $questions: String!) { + createForm(title: $title, questions: $questions) { + success + } + } +` + +export { LOGIN, FORM, USER, FORMSUBMIT, CREATEFORM } diff --git a/src/components/App.tsx b/src/components/App.tsx index 6d9c1bf..e5362e8 100644 --- a/src/components/App.tsx +++ b/src/components/App.tsx @@ -6,6 +6,7 @@ import client from '../apollo' import Context from '../context' import { useUser } from '../hooks' import Authorize from './Authorize' +import CreateForm from './CreateForm' import DoForm from './DoForm' import Login from './Login' import UserPage from './UserPage' @@ -23,6 +24,7 @@ const App: React.FC = () => { + diff --git a/src/components/CreateForm/hooks.ts b/src/components/CreateForm/hooks.ts new file mode 100644 index 0000000..3c03052 --- /dev/null +++ b/src/components/CreateForm/hooks.ts @@ -0,0 +1,156 @@ +import { ApolloError, useMutation } from '@apollo/client' +import { ChangeEvent, FormEvent, useState } from 'react' +import { CREATEFORM } from '../../apollo' +import { MutationCreateFormArgs, ServerAnswer } from '../../apollo/typeDefs.gen' + +type FormQuestion = { + title: string + type: T + variants: string[] +} + +type Form = { + title: string + questions: FormQuestion[] +} + +export type FormatQuestionsToSubmitType = ( + questions: FormQuestion[] +) => string + +interface ICreateFormMutation { + createForm: ServerAnswer +} + +type FormSubmitType = (e: FormEvent) => void + +type HandleFormTitleChangeType = (e: ChangeEvent) => void + +type CreateQuestionType = (type: T) => void + +type HandleQuestionTitleChangeType = ( + questionNumber: number, + e: ChangeEvent +) => void + +type AddVariantType = (questionNumber: number) => void + +type HandleAnswerVariantChangeType = ( + questionNumber: number, + variantNumber: number, + e: ChangeEvent +) => void + +type UseFormCreatorHookTurple = [ + Form, + [ + FormSubmitType, + { + submitData: ICreateFormMutation | null | undefined + submitError: ApolloError | undefined + submitLoading: boolean + } + ], + { + handleFormTitleChange: HandleFormTitleChangeType + addQuestion: CreateQuestionType + handleQuestionTitleChange: HandleQuestionTitleChangeType + handleAnswerVariantChange: HandleAnswerVariantChangeType + addVariant: AddVariantType + } +] + +export const useFormCreator = ( + formatQuestionsToSubmit: FormatQuestionsToSubmitType +): UseFormCreatorHookTurple => { + const [form, setState] = useState>({ title: '', questions: [] }) + + const [ + doFormCreation, + { error: submitError, data: submitData, loading: submitLoading }, + ] = useMutation(CREATEFORM, { + variables: { + title: form.title, + questions: formatQuestionsToSubmit(form.questions), + }, + }) + + const formSubmit: FormSubmitType = (e) => { + e.preventDefault() + console.log({ + title: form.title, + questions: formatQuestionsToSubmit(form.questions), + }) + doFormCreation() + } + + const handleFormTitleChange: HandleFormTitleChangeType = (e) => { + const title = e.currentTarget.value + setState((prev) => ({ ...prev, title })) + } + + const createQuestion: CreateQuestionType = (type) => { + setState(({ title, questions }) => ({ + title, + questions: questions.concat({ title: '', type, variants: [] }), + })) + } + + const handleQuestionTitleChange: HandleQuestionTitleChangeType = ( + questionNumber, + e + ) => { + const questionTitle = e.currentTarget.value + setState(({ title, questions }) => ({ + title, + questions: questions.map((el, index) => + index === questionNumber ? { ...el, title: questionTitle } : el + ), + })) + } + + const handleAnswerVariantChange: HandleAnswerVariantChangeType = ( + questionNumber, + variantNumber, + e + ) => { + const variantText = e.currentTarget.value + setState(({ title, questions }) => ({ + title, + questions: questions.map((question, questionIndex) => + questionIndex === questionNumber + ? { + ...question, + variants: question.variants.map((variant, variantIndex) => + variantIndex === variantNumber ? variantText : variant + ), + } + : question + ), + })) + } + + const addVariant: AddVariantType = (questionNumber) => { + console.log() + setState(({ title, questions }) => ({ + title, + questions: questions.map((el, index) => ({ + ...el, + variants: + index === questionNumber ? el.variants.concat('') : el.variants, + })), + })) + } + + return [ + form, + [formSubmit, { submitData, submitError, submitLoading }], + { + handleFormTitleChange, + addQuestion: createQuestion, + handleQuestionTitleChange, + handleAnswerVariantChange, + addVariant, + }, + ] +} diff --git a/src/components/CreateForm/index.tsx b/src/components/CreateForm/index.tsx new file mode 100644 index 0000000..b441526 --- /dev/null +++ b/src/components/CreateForm/index.tsx @@ -0,0 +1,129 @@ +import React from 'react' + +import { FormatQuestionsToSubmitType, useFormCreator } from './hooks' + +const creationsArray = [ + { title: 'Check', type: 'CHECK', enabled: false }, + { title: 'Input', type: 'INPUT', enabled: true }, + { title: 'Choose', type: 'CHOOSE', enabled: true }, + { title: 'Select', type: 'SELECT', enabled: true }, +] as const + +type QuestionTypes = 'CHECK' | 'INPUT' | 'CHOOSE' | 'SELECT' + +const formatQuestionsToSubmit: FormatQuestionsToSubmitType = (questions) => { + return JSON.stringify( + questions.map((question) => + question.type === 'INPUT' + ? { title: question.title } + : { + ...question, + variants: question.variants.map((variant) => ({ + text: variant, + })), + } + ) + ) +} + +const CreateForm: React.FC = () => { + const [ + form, + [formSubmit, { submitData, submitError, submitLoading }], + { + addQuestion, + handleFormTitleChange, + handleQuestionTitleChange, + handleAnswerVariantChange, + addVariant, + }, + ] = useFormCreator(formatQuestionsToSubmit) + + return ( + <> +
+ + +
+ Content +
    + {creationsArray.flatMap((questionType, index) => + questionType.enabled + ? [ +
  • + +
  • , + ] + : [] + )} +
+
    + {form.questions.map((quesstion, questionIndex) => ( +
  • +

    {quesstion.type} question:

    + handleQuestionTitleChange(questionIndex, e)} + /> + {(quesstion.type === 'CHECK' || + quesstion.type === 'CHOOSE' || + quesstion.type === 'SELECT') && ( + <> +
      + {quesstion.variants.map((variant, variantIndex) => ( +
    • + + handleAnswerVariantChange( + questionIndex, + variantIndex, + e + ) + } + /> +
    • + ))} +
    + + + )} +
  • + ))} +
+
+ {submitLoading ? 'Loading...' : } +
+ {submitData && + submitData.createForm && + submitData.createForm.success && + 'Successfully uploaded'} + {submitError && submitError.message} + + ) +} + +export default CreateForm