@okta/okta-signin-widget
Version:
The Okta Sign-In Widget
507 lines (469 loc) • 14.8 kB
text/typescript
/*
* Copyright (c) 2022-present, Okta, Inc. and/or its affiliates. All rights reserved.
* The Okta software accompanied by this notice is provided pursuant to the Apache License, Version 2.0 (the "License.")
*
* You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0.
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*
* See the License for the specific language governing permissions and limitations under the License.
*/
import {
AppIcon,
DeviceIcon,
LocationIcon,
} from 'src/components/Images';
import { AUTHENTICATOR_KEY } from 'src/constants';
import {
AuthenticatorButtonListElement,
ButtonElement,
ButtonType,
DescriptionElement,
DividerElement,
FieldElement,
HeadingElement,
IdentifierContainerElement,
IdxStepTransformer,
ImageWithTextElement,
InfoboxElement,
LaunchAuthenticatorButtonElement,
LinkElement,
ListElement,
OpenOktaVerifyFPButtonElement,
PasswordMatchesElement,
PasswordRequirementsElement,
QRCodeElement,
ReminderElement,
SpinnerElement,
TextWithActionLinkElement,
TitleElement,
} from '../../../types';
/**
* Special transformer that places UI elements in a single view for testing and demoing components.
*/
export const transformEnumerateComponents: IdxStepTransformer = ({
formBag,
}) => {
const { uischema, dataSchema, data } = formBag;
const reminderElement: ReminderElement = {
type: 'Reminder',
options: {
step: '',
content: 'Reminder content',
timeout: 0,
buttonText: 'Resend',
},
};
uischema.elements.push(reminderElement);
const infoBox: InfoboxElement = {
type: 'InfoBox',
options: {
message: {
message: 'Infobox with widget message',
},
class: 'INFO',
},
};
uischema.elements.push(infoBox);
const titleElement: TitleElement = {
type: 'Title',
options: { content: 'Title' },
};
uischema.elements.push(titleElement);
const subtitleElement: DescriptionElement = {
type: 'Description',
contentType: 'subtitle',
options: {
content: 'Subtitle',
},
};
uischema.elements.push(subtitleElement);
const inputTextElement: FieldElement = {
type: 'Field',
key: 'inputTextElement',
label: 'Required Text Field',
options: {
type: 'string',
inputMeta: {
name: 'input.text',
required: true,
},
},
};
uischema.elements.push(inputTextElement);
const inputTextOptionalElement: FieldElement = {
type: 'Field',
key: 'inputTextOptionalElement',
label: 'Optional Text Field',
options: {
type: 'string',
inputMeta: {
name: 'input.textOptional',
required: false,
},
},
};
uischema.elements.push(inputTextOptionalElement);
const dividerWithText: DividerElement = {
type: 'Divider',
options: {
text: 'OR',
},
};
uischema.elements.push(dividerWithText);
const errorBox: InfoboxElement = {
type: 'InfoBox',
options: {
message: {
message: 'Infobox with error message',
},
class: 'ERROR',
},
};
uischema.elements.push(errorBox);
const headingElement: HeadingElement = {
type: 'Heading',
options: {
level: 3,
visualLevel: 3,
content: 'Heading 3',
},
};
uischema.elements.push(headingElement);
const descriptionElement: DescriptionElement = {
type: 'Description',
options: {
content: 'Description',
},
};
uischema.elements.push(descriptionElement);
const passwordRequirements: PasswordRequirementsElement = {
type: 'PasswordRequirements',
options: {
id: 'password.requirements',
header: 'Password Requirements',
userInfo: {
identifier: 'user@example.com',
profile: {
firstName: 'User',
},
},
settings: {
complexity: {
minLength: 4,
minNumber: 1,
minSymbol: 1,
excludeUsername: true,
},
},
requirements: [
{
ruleKey: 'minLength',
label: 'Minimum length: 4',
},
{
ruleKey: 'minNumber',
label: 'Minimum numbers: 1',
},
{
ruleKey: 'minSymbol',
label: 'Minimum symbols: 1',
},
{
ruleKey: 'excludeUsername',
label: 'Cannot contain username',
},
{
ruleKey: '',
label: 'Should not match your previous password',
},
],
validationDelayMs: 0,
},
};
uischema.elements.push(passwordRequirements);
const inputPasswordElement: FieldElement = {
type: 'Field',
key: 'credentials.passcode',
label: 'Required Password Field',
options: {
type: 'string',
inputMeta: {
secret: true,
name: 'credentials.passcode',
required: true,
},
},
};
uischema.elements.push(inputPasswordElement);
data['credentials.passcode'] = 'a';
const confirmPassword: FieldElement = {
type: 'Field',
key: 'confirmPassword',
label: 'Confirm Password Field',
options: {
type: 'string',
inputMeta: {
secret: true,
name: 'confirmPassword',
required: true,
},
},
};
uischema.elements.push(confirmPassword);
data.confirmPassword = 'a';
const passwordMatches: PasswordMatchesElement = {
type: 'PasswordMatches',
options: {
validationDelayMs: 0,
},
};
uischema.elements.push(passwordMatches);
const divider: DividerElement = {
type: 'Divider',
};
uischema.elements.push(divider);
const phoneNumberElement: FieldElement = {
type: 'Field',
key: 'inputPhoneElement',
label: 'Phone Field',
options: {
type: 'string',
inputMeta: {
name: 'input.phoneNumber',
},
},
};
uischema.elements.push(phoneNumberElement);
dataSchema['input.phoneNumber'] = {};
const checkboxElement: FieldElement = {
type: 'Field',
key: 'checkboxElement',
label: 'Checkbox',
options: {
inputMeta: {
name: 'checkbox',
type: 'checkbox',
},
},
};
uischema.elements.push(checkboxElement);
const radioElement: FieldElement = {
type: 'Field',
key: 'radioElement',
label: 'Radio',
options: {
inputMeta: {
name: 'radio',
type: 'radio',
options: [
{
value: 'option1',
label: 'Option 1',
},
{
value: 'option2',
label: 'Option 2',
},
],
},
},
};
uischema.elements.push(radioElement);
const authenticatorButtonList: AuthenticatorButtonListElement = {
type: 'AuthenticatorButtonList',
options: {
buttons: [
{
type: 'AuthenticatorButton',
label: 'Authenticator Button 1 with long description text',
options: {
step: '',
type: ButtonType.BUTTON,
key: AUTHENTICATOR_KEY.PHONE,
ctaLabel: 'Call-to-action',
Icon: '',
},
},
{
type: 'AuthenticatorButton',
label: 'Authenticator Button 2',
options: {
step: '',
type: ButtonType.BUTTON,
key: AUTHENTICATOR_KEY.OV,
ctaLabel: 'Call-to-action',
description: 'Description with long description text, long enough to go to a second line',
usageDescription: 'Usage description text',
},
},
{
type: 'AuthenticatorButton',
label: 'Authenticator Button 3 with Phone Nickname',
options: {
step: '',
type: ButtonType.BUTTON,
key: AUTHENTICATOR_KEY.PHONE,
ctaLabel: 'Call-to-action',
description: '555-615-8855',
nickname: 'mobile phone',
Icon: '',
},
},
{
type: 'AuthenticatorButton',
label: 'Authenticator Button 4 with extra long Phone Nickname',
options: {
step: '',
type: ButtonType.BUTTON,
key: AUTHENTICATOR_KEY.PHONE,
ctaLabel: 'Call-to-action',
description: '555-615-8855',
nickname: 'This is an extra long nickname that should display an ellipsis in the UI because of how extremely long it is',
Icon: '',
},
},
{
type: 'AuthenticatorButton',
label: 'Authenticator Button 5 for Enroll',
options: {
step: '',
type: ButtonType.BUTTON,
key: AUTHENTICATOR_KEY.CUSTOM_APP,
ctaLabel: 'Set up',
isEnroll: true,
},
},
{
type: 'AuthenticatorButton',
label: 'Authenticator Button 6 for Additional Enroll',
options: {
step: '',
type: ButtonType.BUTTON,
key: AUTHENTICATOR_KEY.WEBAUTHN,
ctaLabel: 'Set up another',
isEnroll: true,
isAdditionalEnroll: true,
},
},
],
dataSe: 'authenticator-button-list',
},
};
uischema.elements.push(authenticatorButtonList);
const launchAuthenticatorButton: LaunchAuthenticatorButtonElement = {
type: 'LaunchAuthenticatorButton',
options: {
step: '',
},
translations: [{
name: 'label',
i18nKey: '',
value: 'Launch Authenticator button with long name and icon',
}],
};
uischema.elements.push(launchAuthenticatorButton);
const openOktaVerifyFPButton: OpenOktaVerifyFPButtonElement = {
type: 'OpenOktaVerifyFPButton',
options: {
step: '',
},
};
uischema.elements.push(openOktaVerifyFPButton);
const secondaryButton: ButtonElement = {
type: 'Button',
label: 'Secondary Button',
options: {
step: '',
type: ButtonType.BUTTON,
variant: 'secondary',
},
};
uischema.elements.push(secondaryButton);
const floatingButton: ButtonElement = {
type: 'Button',
label: 'Floating Button',
options: {
step: '',
type: ButtonType.BUTTON,
variant: 'floating',
},
};
uischema.elements.push(floatingButton);
uischema.elements.push(divider);
const link: LinkElement = {
type: 'Link',
options: {
step: '',
label: 'Link',
},
};
uischema.elements.push(link);
const list: ListElement = {
type: 'List',
options: {
items: [
'List item 1',
'List item 2',
],
type: 'ul',
description: 'List description',
},
};
uischema.elements.push(list);
const qrCode: QRCodeElement = {
type: 'QRCode',
options: {
data: '',
},
};
uischema.elements.push(qrCode);
const spinner: SpinnerElement = {
type: 'Spinner',
};
uischema.elements.push(spinner);
const textWithActionLink: TextWithActionLinkElement = {
type: 'TextWithActionLink',
options: {
step: '',
content: 'Text with <a href="#" class="text-with-action-link">action link</a>',
contentClassname: 'text-with-action-link',
},
};
uischema.elements.push(textWithActionLink);
uischema.elements.push(divider);
const appImageWithText: ImageWithTextElement = {
type: 'ImageWithText',
options: {
id: 'image.withText',
SVGIcon: AppIcon,
textContent: 'Application',
},
};
uischema.elements.push(appImageWithText);
const deviceImageWithText: ImageWithTextElement = {
type: 'ImageWithText',
options: {
id: 'image.withText',
SVGIcon: DeviceIcon,
textContent: 'Device',
},
};
uischema.elements.push(deviceImageWithText);
const locationImageWithText: ImageWithTextElement = {
type: 'ImageWithText',
options: {
id: 'image.withText',
SVGIcon: LocationIcon,
textContent: 'Location',
},
};
uischema.elements.push(locationImageWithText);
const identifierElement: IdentifierContainerElement = {
type: 'IdentifierContainer',
options: { identifier: 'User' },
};
uischema.elements.push(identifierElement);
return formBag;
};