diff --git a/src/apollo/defs.ts b/src/apollo/defs.ts index a9b6bfe..5ebfee5 100644 --- a/src/apollo/defs.ts +++ b/src/apollo/defs.ts @@ -99,4 +99,12 @@ const CREATEFORM = gql` } ` -export { LOGIN, FORM, USER, FORMSUBMIT, CREATEFORM } +const REGISTER = gql` + mutation Register($email: String!, $name: String!) { + register(email: $email, name: $name) { + success + } + } +` + +export { LOGIN, FORM, USER, FORMSUBMIT, CREATEFORM, REGISTER } diff --git a/src/apollo/index.ts b/src/apollo/index.ts index 3343847..e283232 100644 --- a/src/apollo/index.ts +++ b/src/apollo/index.ts @@ -1,11 +1,17 @@ -import { ApolloClient, createHttpLink, InMemoryCache } from '@apollo/client' - +import { + ApolloClient, + createHttpLink, + from, + InMemoryCache, +} from '@apollo/client' import { setContext } from '@apollo/client/link/context' +import { onError } from '@apollo/client/link/error' export * from './defs' const httpLink = createHttpLink({ - uri: process.env.GRAPHQL_URL || process.env.REACT_APP_GRAPHQL_URL || undefined + uri: + process.env.GRAPHQL_URL || process.env.REACT_APP_GRAPHQL_URL || undefined, // fetchOptions: { // mode: 'no-cors' // } @@ -17,14 +23,25 @@ const authLink = setContext((_, { headers }) => { return { headers: { ...headers, - authorization: token ? `Bearer ${token}` : '' - } + authorization: token ? `Bearer ${token}` : '', + }, } }) +const errorLink = onError(({ graphQLErrors, networkError }) => { + if (graphQLErrors) + graphQLErrors.map(({ message, locations, path }) => + console.log( + `[GraphQL error]: Message: ${message}, Location: ${locations}, Path: ${path}` + ) + ) + + if (networkError) console.log(`[Network error]: ${networkError}`) +}) + const client = new ApolloClient({ - link: authLink.concat(httpLink), - cache: new InMemoryCache() + link: from([errorLink, authLink, httpLink]), + cache: new InMemoryCache(), }) export default client diff --git a/src/components/App.tsx b/src/components/App.tsx index e5362e8..ac8193a 100644 --- a/src/components/App.tsx +++ b/src/components/App.tsx @@ -9,6 +9,7 @@ import Authorize from './Authorize' import CreateForm from './CreateForm' import DoForm from './DoForm' import Login from './Login' +import Register from './Register' import UserPage from './UserPage' const App: React.FC = () => { @@ -25,6 +26,7 @@ const App: React.FC = () => { + diff --git a/src/components/CreateForm/hooks.ts b/src/components/CreateForm/hooks.ts index 3c03052..11da114 100644 --- a/src/components/CreateForm/hooks.ts +++ b/src/components/CreateForm/hooks.ts @@ -57,13 +57,16 @@ type UseFormCreatorHookTurple = [ handleQuestionTitleChange: HandleQuestionTitleChangeType handleAnswerVariantChange: HandleAnswerVariantChangeType addVariant: AddVariantType - } + }, + () => void ] +const initialState = { title: '', questions: [] } + export const useFormCreator = ( formatQuestionsToSubmit: FormatQuestionsToSubmitType ): UseFormCreatorHookTurple => { - const [form, setState] = useState>({ title: '', questions: [] }) + const [form, setState] = useState>(initialState) const [ doFormCreation, @@ -75,13 +78,11 @@ export const useFormCreator = ( }, }) - const formSubmit: FormSubmitType = (e) => { + const formSubmit: FormSubmitType = async (e) => { e.preventDefault() - console.log({ - title: form.title, - questions: formatQuestionsToSubmit(form.questions), - }) - doFormCreation() + try { + await doFormCreation() + } catch (err) {} } const handleFormTitleChange: HandleFormTitleChangeType = (e) => { @@ -92,7 +93,7 @@ export const useFormCreator = ( const createQuestion: CreateQuestionType = (type) => { setState(({ title, questions }) => ({ title, - questions: questions.concat({ title: '', type, variants: [] }), + questions: questions.concat({ title: '', type, variants: [''] }), })) } @@ -131,7 +132,6 @@ export const useFormCreator = ( } const addVariant: AddVariantType = (questionNumber) => { - console.log() setState(({ title, questions }) => ({ title, questions: questions.map((el, index) => ({ @@ -142,6 +142,10 @@ export const useFormCreator = ( })) } + const resetForm = () => { + setState(initialState) + } + return [ form, [formSubmit, { submitData, submitError, submitLoading }], @@ -152,5 +156,6 @@ export const useFormCreator = ( handleAnswerVariantChange, addVariant, }, + resetForm, ] } diff --git a/src/components/CreateForm/index.tsx b/src/components/CreateForm/index.tsx index b441526..2df35e8 100644 --- a/src/components/CreateForm/index.tsx +++ b/src/components/CreateForm/index.tsx @@ -37,11 +37,17 @@ const CreateForm: React.FC = () => { handleAnswerVariantChange, addVariant, }, + resetForm, ] = useFormCreator(formatQuestionsToSubmit) return ( <> -
+ { + resetForm() + formSubmit(e) + }} + > @@ -75,6 +82,7 @@ const CreateForm: React.FC = () => {
  • {quesstion.type} question:

    { {quesstion.variants.map((variant, variantIndex) => (
  • { - console.log(initialValue, 'Inside hook') - const [answers, setAnswer] = useState<(InputAnswer | ChoiseAnswer)[]>( initialValue || [] ) diff --git a/src/components/DoForm/index.tsx b/src/components/DoForm/index.tsx index d7d3067..6387bd4 100644 --- a/src/components/DoForm/index.tsx +++ b/src/components/DoForm/index.tsx @@ -8,7 +8,9 @@ import { Form, InputAnswer, InputQuestion, + MutationFormSubmitArgs, QueryFormArgs, + ServerAnswer, } from '../../apollo/typeDefs.gen' import Lists from './Lists' @@ -16,12 +18,19 @@ interface IFormQuery { form: Form } +interface IFormSubmitMutation { + formSubmit: ServerAnswer +} + const DoForm: React.FC = () => { const { id: idString } = useParams<{ id: string }>() const id = parseInt(idString) - const { data, error, loading } = useQuery(FORM, { + const { data, error, loading, refetch: refetchForm } = useQuery< + IFormQuery, + QueryFormArgs + >(FORM, { variables: { id }, skip: isNaN(id), }) @@ -29,7 +38,7 @@ const DoForm: React.FC = () => { const [ doFormSubmit, { error: submitError, data: submitData, loading: submitLoading }, - ] = useMutation(FORMSUBMIT) + ] = useMutation(FORMSUBMIT) const [answers, setAnswer] = useState<(InputAnswer | ChoiseAnswer)[]>([]) @@ -57,8 +66,6 @@ const DoForm: React.FC = () => { } }, [data]) - useEffect(() => console.log(answers), [answers]) - if (isNaN(id)) return if (loading) return
    Loading...
    @@ -66,24 +73,24 @@ const DoForm: React.FC = () => { const { form } = data! - const handleSubmit = (e: FormEvent) => { + const handleSubmit = async (e: FormEvent) => { e.preventDefault() - console.log('Submited form:', answers) answers.forEach((el) => { delete el.__typename }) - const submitAnswers = JSON.stringify(answers) + try { + const submitAnswers = JSON.stringify(answers) - console.log('Filtered answers: ', submitAnswers) - - doFormSubmit({ - variables: { - formId: id, - answers: submitAnswers, - }, - }) + await doFormSubmit({ + variables: { + formId: id, + answers: submitAnswers, + }, + }) + await refetchForm() + } catch (err) {} } const answerChange = (num: number) => { @@ -108,39 +115,43 @@ const DoForm: React.FC = () => {

    {form.dateCreated}

    {form.author?.name || 'No author'}

    {form.submissions ? ( -
    -

    Submission{form.submissions.length > 1 && 's'}:

    -
      - {form.submissions.map((submission, submissionIndex) => ( -
    • -

      - User:{' '} - {submission.user ? submission.user.name : 'No submitter'} -

      -
        - {submission.answers.map( - (answer: InputAnswer | ChoiseAnswer, answerIndex) => ( -
      • -

        {form.questions[answerIndex].title}

        - {answer.__typename === 'ChoiseAnswer' && ( -

        - { - (form.questions[answerIndex] as ChoisesQuestion) - .variants[answer.userChoise].text - } -

        - )} - {answer.__typename === 'InputAnswer' && ( -

        {answer.userInput}

        - )} -
      • - ) - )} -
      -
    • - ))} -
    -
    + form.submissions.length > 0 ? ( +
    +

    Submission{form.submissions.length > 1 && 's'}:

    +
      + {form.submissions.map((submission, submissionIndex) => ( +
    • +

      + User:{' '} + {submission.user ? submission.user.name : 'No submitter'} +

      +
        + {submission.answers.map( + (answer: InputAnswer | ChoiseAnswer, answerIndex) => ( +
      • +

        {form.questions[answerIndex].title}

        + {answer.__typename === 'ChoiseAnswer' && ( +

        + { + (form.questions[answerIndex] as ChoisesQuestion) + .variants[answer.userChoise].text + } +

        + )} + {answer.__typename === 'InputAnswer' && ( +

        {answer.userInput}

        + )} +
      • + ) + )} +
      +
    • + ))} +
    +
    + ) : ( + 'No submissions yet' + ) ) : (
      diff --git a/src/components/Login/index.tsx b/src/components/Login/index.tsx index d2ac539..fe3cea8 100644 --- a/src/components/Login/index.tsx +++ b/src/components/Login/index.tsx @@ -3,11 +3,19 @@ import React, { ChangeEvent, FormEvent, useState } from 'react' import { Redirect } from 'react-router-dom' import { LOGIN } from '../../apollo' +import { MutationLoginArgs, ServerAnswer } from '../../apollo/typeDefs.gen' + +interface ILoginMutation { + login: ServerAnswer +} const Login: React.FC = () => { const [email, setEmail] = useState('') - const [doLogin, { error, data }] = useMutation(LOGIN) + const [doLogin, { error, data }] = useMutation< + ILoginMutation, + MutationLoginArgs + >(LOGIN) const handleFormSubmit = async (e: FormEvent) => { e.preventDefault() diff --git a/src/components/Register/index.tsx b/src/components/Register/index.tsx new file mode 100644 index 0000000..4727df0 --- /dev/null +++ b/src/components/Register/index.tsx @@ -0,0 +1,45 @@ +import { useMutation } from '@apollo/client' +import React, { FormEvent } from 'react' +import { Redirect } from 'react-router-dom' +import { REGISTER } from '../../apollo' +import { MutationRegisterArgs, ServerAnswer } from '../../apollo/typeDefs.gen' + +interface IRegisterMutation { + register: ServerAnswer +} + +const Register: React.FC = () => { + const [doRegister, { data, loading, error }] = useMutation< + IRegisterMutation, + MutationRegisterArgs + >(REGISTER) + + const handleSubmit = async (e: FormEvent) => { + e.preventDefault() + + const formData = new FormData(e.currentTarget) + try { + await doRegister({ + variables: { + email: formData.get('email') as string, + name: formData.get('name') as string, + }, + }) + } catch (err) {} + } + + return ( +
      + Register + + + + {loading ? 'Loading...' : } + {error && error.message} + {data && data.register && data.register.success && } + +
      + ) +} + +export default Register diff --git a/src/components/UserPage/index.tsx b/src/components/UserPage/index.tsx index 7a86e92..9635d05 100644 --- a/src/components/UserPage/index.tsx +++ b/src/components/UserPage/index.tsx @@ -1,14 +1,14 @@ import { useQuery } from '@apollo/client' import React from 'react' import { USER } from '../../apollo' -import { User } from '../../apollo/typeDefs.gen' +import { QueryUserArgs, User } from '../../apollo/typeDefs.gen' -interface UserQuery { +interface IUserQuery { user: User } const UserPage: React.FC = () => { - const { data, error, loading } = useQuery(USER) + const { data, error, loading } = useQuery(USER) if (loading) return

      Loading...

      if (error) return

      {error.message}