Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add input component as types, simplify event handlers - emailpassword recipe #752

Merged
merged 52 commits into from
Nov 3, 2023
Merged
Show file tree
Hide file tree
Changes from 22 commits
Commits
Show all changes
52 commits
Select commit Hold shift + click to select a range
a2f1d7e
Add inputComponent to exposed types
amitbadala Oct 20, 2023
d4ab51f
Add inputComponent to normalised fields
amitbadala Oct 20, 2023
2e8c67d
For testing only - use custom type definition for inputComponent
amitbadala Oct 20, 2023
578815a
Input component already present in FormFieldThemeProps
amitbadala Oct 20, 2023
fcf51e1
Testing if git package is getting installed correctly
amitbadala Oct 20, 2023
1b747f5
Run build for previous commits
amitbadala Oct 20, 2023
e2b95de
Remove inputComp from NormalizedFormField
amitbadala Oct 20, 2023
52e7dab
Add tests for custom fields
amitbadala Oct 23, 2023
eb36b36
Remove testing ele
amitbadala Oct 24, 2023
e6fc942
Move the custom fields tests into existing describe
amitbadala Oct 24, 2023
8514575
Update dropdown values to avoid confusion
amitbadala Oct 24, 2023
ccd46cc
Add helper func to set dropdown, better test title, use existing desc…
amitbadala Oct 24, 2023
16b38e5
Use strict equal
amitbadala Oct 24, 2023
9136be9
Update request
amitbadala Oct 24, 2023
e365a13
A seperate func to fetch custom comp not required
amitbadala Oct 24, 2023
c8b31db
Move inputComponent to signup types
amitbadala Oct 24, 2023
31d6e29
Cleanup unwanted imports
amitbadala Oct 24, 2023
8d60068
Move inputComponent to signup types
amitbadala Oct 25, 2023
809d396
Clean types
amitbadala Oct 25, 2023
b3af734
Update build files
amitbadala Oct 25, 2023
fb99461
Use explicit values in validate func
amitbadala Oct 25, 2023
42cba3f
Minor cleanup of types
amitbadala Oct 25, 2023
8bae273
Better type names
amitbadala Oct 25, 2023
2023dce
Props suggestions working for inputComponent
amitbadala Oct 25, 2023
7dc66b8
Enforce strict string check on form values, now onChange function for…
amitbadala Oct 26, 2023
3cc4ec8
Update based on the new onChange func
amitbadala Oct 26, 2023
a13ddc1
Ability to add default value with getDefaultValue prop
amitbadala Oct 26, 2023
2be7cfa
Handle if getDefaultValue is not a function
amitbadala Oct 26, 2023
2480ebd
instead of form submit apply type test within onChange function itself
amitbadala Oct 26, 2023
d574fc0
Add tests for default value
amitbadala Oct 26, 2023
883662f
Remove unwanted abort
amitbadala Oct 27, 2023
494c5c0
Testing email-verification workflow
amitbadala Oct 27, 2023
e759f13
Reverting onChange changes
amitbadala Oct 27, 2023
90a9344
onChange function to accept only values
amitbadala Oct 27, 2023
7e6a3ec
Initialize fieldstates at the start
amitbadala Oct 27, 2023
bcef516
Remove useEffect
amitbadala Oct 27, 2023
7241f6a
Fix race conditions when setting default value
amitbadala Oct 30, 2023
c9df6d9
Add custom default fields to typescript example, plus add tests to sh…
amitbadala Oct 30, 2023
c305fff
Add tests for incorrect default props in formFields
amitbadala Oct 30, 2023
3f8d094
Add tests for incorrect usage of onChange prop
amitbadala Oct 30, 2023
a6d499d
Add change log
amitbadala Oct 30, 2023
284c6fa
Wrap ternary opeators into seperate func for better readibility
amitbadala Oct 30, 2023
d4c53a5
Wrap inputComponent in a serperate component to avoid unecessary rere…
amitbadala Oct 30, 2023
259699a
Add change log feedbacks
amitbadala Oct 31, 2023
492fee9
Better variable names, include formfields directly in typescript example
amitbadala Oct 31, 2023
329aee4
Add more tests for default & onChange func, updated typescript file t…
amitbadala Oct 31, 2023
9aa6b21
Add more test, which intercepts request payload
amitbadala Nov 1, 2023
e23f1d8
Cleanup comments
amitbadala Nov 1, 2023
a50d64b
Minor formatting
amitbadala Nov 1, 2023
1339645
Minor fix
amitbadala Nov 1, 2023
7d1c7a2
Clean up helper
amitbadala Nov 3, 2023
ffd867f
Update change log & versions
amitbadala Nov 3, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 50 additions & 1 deletion examples/for-tests/src/App.js
Original file line number Diff line number Diff line change
Expand Up @@ -168,6 +168,54 @@ const formFields = [
},
];

const customFields = [
{
id: "ratings",
label: "Ratings",
inputComponent: ({ value, name, ...rest }) => (
<select
name={name}
data-supertokens="inputComponent"
placeholder="Add Ratings"
onChange={(e) => rest.onChange({ id: name, value: e.target.value })}>
rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
<option value="" disabled hidden>
Select an option
</option>
<option value="good">Good</option>
<option value="better">Better</option>
<option value="best">Best</option>
</select>
),
optional: true,
},
{
id: "terms",
showLabels: false,
optional: false,
inputComponent: ({ value, name, ...rest }) => (
<div
data-supertokens="inputComponent"
style={{
display: "flex",
alignItems: "center",
justifyContent: "left",
}}>
<input
name={name}
type="checkbox"
onChange={(e) => rest.onChange({ id: name, value: e.target.checked })}></input>
<span style={{ marginLeft: 5 }}>I agree to the terms and conditions</span>
</div>
),
validate: async (value) => {
if (value === true) {
return undefined;
}
return "Please check Terms and conditions";
},
},
];

const testContext = getTestContext();

let recipeList = [
Expand Down Expand Up @@ -637,7 +685,8 @@ function getEmailPasswordConfigs({ disableDefaultUI }) {
style: theme,
privacyPolicyLink: "https://supertokens.com/legal/privacy-policy",
termsOfServiceLink: "https://supertokens.com/legal/terms-and-conditions",
formFields,
formFields:
localStorage.getItem("SHOW_CUSTOM_FIELDS") === "YES" ? formFields.concat(customFields) : formFields,
},
},
});
Expand Down

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

17 changes: 11 additions & 6 deletions lib/build/recipe/emailpassword/types.d.ts

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

28 changes: 12 additions & 16 deletions lib/ts/recipe/emailpassword/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -135,7 +135,7 @@ export type SignUpFormFeatureUserInput = FeatureBaseConfig & {
/*
* Form fields for SignUp.
*/
formFields?: FormFieldSignUpConfig[];
formFields?: (FormField & { inputComponent?: React.FC<InputProps> })[];

/*
* Privacy policy link for sign up form.
Expand All @@ -152,7 +152,7 @@ export type NormalisedSignUpFormFeatureConfig = NormalisedBaseConfig & {
/*
* Normalised form fields for SignUp.
*/
formFields: NormalisedFormField[];
formFields: (NormalisedFormField & { inputComponent?: React.FC<InputProps> })[];

/*
* Privacy policy link for sign up form.
Expand Down Expand Up @@ -181,8 +181,6 @@ export type NormalisedSignInFormFeatureConfig = NormalisedBaseConfig & {

export type FormFieldSignInConfig = FormFieldBaseConfig;

export type FormFieldSignUpConfig = FormField;

export type ResetPasswordUsingTokenUserInput = {
/*
* Disable default implementation with default routes.
Expand Down Expand Up @@ -225,15 +223,11 @@ export type NormalisedEnterEmailForm = FeatureBaseConfig & {
formFields: NormalisedFormField[];
};

/*
* Props Types.
*/

type FormThemeBaseProps = ThemeBaseProps & {
rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
/*
* Form fields to use in the signin form.
* Form fields to use in the signup form.
rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
*/
formFields: FormFieldThemeProps[];
formFields: Omit<FormFieldThemeProps, "inputComponent">[];

error: string | undefined;
};
Expand All @@ -248,13 +242,15 @@ export type SignInThemeProps = FormThemeBaseProps & {
onSuccess: (result: { user: User }) => void;
};

export type SignUpThemeProps = FormThemeBaseProps & {
export type SignUpThemeProps = ThemeBaseProps & {
recipeImplementation: RecipeInterface;
clearError: () => void;
onError: (error: string) => void;
config: NormalisedConfig;
signInClicked?: () => void;
onSuccess: (result: { user: User }) => void;
formFields: FormFieldThemeProps[];
error: string | undefined;
};

export type SignInAndUpThemeProps = {
Expand All @@ -274,11 +270,6 @@ export type FormFieldThemeProps = NormalisedFormField & {
*/
labelComponent?: JSX.Element;

/*
* Custom component that replaces the standard input component
*/
inputComponent?: React.FC<InputProps>;

/*
* Show Is required (*) next to label
*/
Expand All @@ -288,6 +279,11 @@ export type FormFieldThemeProps = NormalisedFormField & {
* Clears the field after calling the API.
*/
clearOnSubmit?: boolean;

/*
* Ability to add custom components
*/
inputComponent?: React.FC<InputProps>;
rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
};

rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
export type FormFieldError = {
Expand Down
101 changes: 101 additions & 0 deletions test/end-to-end/signup.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ import {
getGeneralError,
waitForSTElement,
backendBeforeEach,
setSelectDropdownValue,
} from "../helpers";

import {
Expand Down Expand Up @@ -342,6 +343,106 @@ describe("SuperTokens SignUp", function () {
assert.deepStrictEqual(emailError, "This email already exists. Please sign in instead.");
});
});

// CUSTOM FIELDS TEST

describe("Signup custom fields test", function () {
beforeEach(async function () {
// set cookie and reload which loads the form with custom field
await page.evaluate(() => window.localStorage.setItem("SHOW_CUSTOM_FIELDS", "YES"));

await page.reload({
waitUntil: "domcontentloaded",
});
await toggleSignInSignUp(page);
});

it("Check if the custom fields are loaded", async function () {
let text = await getAuthPageHeaderText(page);
assert.deepStrictEqual(text, "Sign Up");

// check if select dropdown is loaded
const selectDropdownExists = await waitForSTElement(page, "select");
assert.ok(selectDropdownExists, "Select dropdown exists");

// check if checbox is loaded
const checkboxExists = await waitForSTElement(page, 'input[type="checkbox"]');
assert.ok(checkboxExists, "Checkbox exists");
});

it("Should show error messages, based on optional flag", async function () {
rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
await submitForm(page);
let formFieldErrors = await getFieldErrors(page);

// 4 regular form field errors +
// 1 required custom field => terms checkbox
assert.deepStrictEqual(formFieldErrors, [
"Field is not optional",
"Field is not optional",
"Field is not optional",
"Field is not optional",
"Field is not optional",
]);

// supply values for regular-required fields only
await setInputValues(page, [
{ name: "email", value: "[email protected]" },
{ name: "password", value: "Str0ngP@ssw0rd" },
{ name: "name", value: "John Doe" },
{ name: "age", value: "20" },
]);
await submitForm(page);
formFieldErrors = await getFieldErrors(page);
assert.deepStrictEqual(formFieldErrors, ["Field is not optional"]);

// check terms and condition checkbox
let termsCheckbox = await waitForSTElement(page, '[name="terms"]');
await page.evaluate((e) => e.click(), termsCheckbox);
await submitForm(page);
formFieldErrors = await getFieldErrors(page);
assert.deepStrictEqual(formFieldErrors, []);
});

it("Check if custom values are part of the signup payload", async function () {
const customFields = {
terms: true, // checked
rishabhpoddar marked this conversation as resolved.
Show resolved Hide resolved
ratings: "best",
};
const requestHandler = (request) => {
if (request.url().includes(SIGN_UP_API) && request.method() === "POST") {
Object.keys(customFields).every((key) => {
assert.strictEqual(data[key], customFields[key]);
});
// remove it after updating backend to handle the custom fields
return request.abort();
}
return request.continue();
};
await page.setRequestInterception(true);
page.on("request", requestHandler);

// Fill and submit the form with custom fields
try {
await setInputValues(page, [
{ name: "email", value: "[email protected]" },
{ name: "password", value: "Str0ngP@assw0rd" },
{ name: "name", value: "Supertokens" },
{ name: "age", value: "20" },
]);

await setSelectDropdownValue(page, 'select[name="ratings"]', customFields["ratings"]);

// Check terms and condition checkbox
let termsCheckbox = await waitForSTElement(page, '[name="terms"]');
await page.evaluate((e) => e.click(), termsCheckbox);

await submitForm(page);
} finally {
page.off("request", requestHandler);
await page.setRequestInterception(false);
}
});
});
});

describe("SuperTokens SignUp => Server Error", function () {
Expand Down
15 changes: 15 additions & 0 deletions test/helpers.js
Original file line number Diff line number Diff line change
Expand Up @@ -417,6 +417,21 @@ export async function setInputValues(page, fields) {
return await new Promise((r) => setTimeout(r, 300));
}

export async function setSelectDropdownValue(page, selector, optionValue) {
return await page.evaluate(
(selector, optionValue, ST_ROOT_SELECTOR) => {
const dropdownElement = document.querySelector(ST_ROOT_SELECTOR).shadowRoot.querySelector(selector);
if (dropdownElement) {
dropdownElement.value = optionValue;
dropdownElement.dispatchEvent(new Event("change", { bubbles: false }));
}
},
selector,
optionValue,
ST_ROOT_SELECTOR
);
}

export async function clearBrowserCookiesWithoutAffectingConsole(page, console) {
let toReturn = [...console];
const client = await page.target().createCDPSession();
Expand Down