Feedback
Default feedback form
The information on the GDPR should be obligatorily shown to the users from Europe. See the component's guide for the styles and its content.
tsx
import React from 'react';
import FeedbackForm from '@semcore/ui/feedback-form';
import Input from '@semcore/ui/input';
import { Box, Flex } from '@semcore/ui/flex-box';
import Link from '@semcore/ui/link';
import Dropdown from '@semcore/ui/dropdown';
import ChatM from '@semcore/ui/icon/Chat/m';
import Textarea from '@semcore/ui/textarea';
import { Text } from '@semcore/ui/typography';
import { ButtonLink } from '@semcore/ui/button';
const validate = {
description: (value = '') => {
const splitText = value.split(' ');
const numberSpaces = splitText.reduce((acc, item) => {
if (!item.length) {
acc += 1;
}
return acc;
}, 0);
if ([...value].length - numberSpaces < 10) {
return 'Your feedback must contain at least 10 characters.';
}
},
email: (value = '') => {
validate.description(value);
if (!/.+@.+\..+/i.test(String(value).toLowerCase())) {
return 'Please enter valid email.\t';
}
},
};
type FeedbackProps = {
status: string;
onSubmit: (data: any) => void;
onCancel: () => void;
onChange: (event: any, trigger: string) => void;
value: { description: string; email: string };
};
class Feedback extends React.PureComponent<FeedbackProps> {
ref = React.createRef<HTMLDivElement>();
componentDidMount(): void {
// need this timeout because Feedback component used in function component
setTimeout(() => {
this.ref.current?.focus();
}, 0);
}
componentDidUpdate(prevProps: Readonly<FeedbackProps>): void {
if (prevProps.status !== 'success' && this.props.status === 'success') {
this.ref.current?.focus();
}
}
handleChange = (fn) => (_, e) => {
fn(e);
this.props.onChange(e, e.currentTarget.id);
};
render() {
const { status, onSubmit, onCancel, value } = this.props;
if (status === 'success') {
return (
<FeedbackForm.Success ref={this.ref}>Thank you for your feedback!</FeedbackForm.Success>
);
}
return (
<FeedbackForm onSubmit={onSubmit} loading={status === 'loading'}>
<Box p={4}>
<Flex direction='column'>
<Text mb={2} size={200} tag='label' htmlFor='description'>
Tell us your suggestion or report an issue
</Text>
<FeedbackForm.Item
name='description'
validate={validate.description}
initialValue={''}
placement='left-start'
flip={{
fallbackPlacements: ['right-start', 'bottom'],
}}
validateOnBlur={value.description === '' ? false : true}
>
{({ input }) => (
<Textarea
{...input}
autoFocus
h={80}
onChange={this.handleChange(input.onChange)}
id='description'
/>
)}
</FeedbackForm.Item>
</Flex>
<Flex mt={4} direction='column'>
<Text mb={2} size={200} tag='label' htmlFor='email'>
Reply-to email
</Text>
<FeedbackForm.Item
name='email'
validate={validate.email}
initialValue={''}
validateOnBlur={value.email === '' ? false : true}
>
{({ input }) => (
<Input state={input.state}>
<Input.Value
{...input}
onChange={this.handleChange(input.onChange)}
id='email'
aria-describedby='email-description'
/>
</Input>
)}
</FeedbackForm.Item>
</Flex>
<Box mt={2}>
<Text size={200} color='text-secondary' id='email-description'>
We will only use this email to respond to you on your feedback.{' '}
<Link href='https://www.semrush.com/company/legal/privacy-policy/'>
Privacy Policy
</Link>
</Text>
</Box>
<Flex mt={4}>
<FeedbackForm.Submit>Send feedback</FeedbackForm.Submit>
<FeedbackForm.Cancel onClick={onCancel}>Cancel</FeedbackForm.Cancel>
</Flex>
</Box>
<FeedbackForm.Notice hidden={status === 'failed'}>
You can also send us an email to <Link>backlink.audit@semrush.com</Link>
</FeedbackForm.Notice>
<FeedbackForm.Notice hidden={status !== 'failed'} theme='danger'>
Your message has not been sent.
</FeedbackForm.Notice>
</FeedbackForm>
);
}
}
class FeedbackLink extends React.PureComponent {
state = { status: 'default', value: { description: '', email: '' } };
timeout: any;
onSubmit = () => {
this.requestServer('success', 1000);
this.setState({ status: 'loading' });
};
onChange = (e, trigger) => {
const { value } = e.currentTarget;
this.setState({ value: { ...this.state.value, [trigger]: value } });
};
requestServer = (status, time = 500, cb = () => {}) => {
this.timeout = setTimeout(() => {
this.setState({ status });
cb();
}, time);
};
componentWillUnmount() {
clearTimeout(this.timeout);
}
render() {
const { status, value } = this.state;
return (
<Dropdown>
<Dropdown.Trigger tag={ButtonLink} addonLeft={ChatM}>
Send feedback
</Dropdown.Trigger>
<Dropdown.Popper aria-label={'Feedback form'} tabIndex={-1}>
{(_props, { visible }) => (
<Feedback
status={status}
onCancel={() => visible(false)}
onSubmit={() => this.onSubmit()}
value={value}
onChange={this.onChange}
/>
)}
</Dropdown.Popper>
</Dropdown>
);
}
}
const Demo = FeedbackLink;
export default Demo;
Feedback form example
Refer for more details about forms in the Form.
tsx
import React from 'react';
import FeedbackForm from '@semcore/ui/feedback-form';
import Input from '@semcore/ui/input';
import InputNumber from '@semcore/ui/input-number';
import Radio, { RadioGroup } from '@semcore/ui/radio';
import Select from '@semcore/ui/select';
import { Text } from '@semcore/ui/typography';
import { Flex, Box } from '@semcore/ui/flex-box';
type Data = {
title: string;
campaign: string;
call: boolean;
day: number;
};
const validate = (values: Data) => {
if (!values) return {};
const errors: Partial<Record<keyof Data, string>> = {};
if (!values.title) {
errors.title = 'Title is required';
}
if (!values.campaign) {
errors.campaign = 'Campaign is required';
}
if (!values.call) {
errors.call = 'To pick to call or not is required';
}
if (!values.day || values.day <= 0) {
errors.day = 'Invalid day value';
}
return errors;
};
const Demo = () => (
<FeedbackForm validate={validate} p={1} onSubmit={() => ({})}>
<Box tag={'label'} htmlFor='acitivty'>
<Text mb={2} tag='p' size={200}>
Activity
</Text>
<FeedbackForm.Item name='title'>
{({ input }) => {
const { state, className, ...other } = input;
return (
<Input state={state} className={className}>
<Input.Value
{...other}
placeholder='Activity title'
id='acitivty'
autoComplete='off'
/>
</Input>
);
}}
</FeedbackForm.Item>
</Box>
<Box tag={'label'} htmlFor='campaign' mt={4}>
<Text mb={2} tag='p' size={200}>
Campaign
</Text>
<FeedbackForm.Item name='campaign'>
{({ input }) => (
<Select onChange={input.onChange} state={input.state} placeholder='Select campaign'>
<Select.Trigger id='campaign' {...input} />
<Select.Menu>
{Array(4)
.fill(0)
.map((item, ind) => (
<Select.Option
value={`Company ${ind}`}
key={ind}
>{`Company ${ind}`}</Select.Option>
))}
</Select.Menu>
</Select>
)}
</FeedbackForm.Item>
</Box>
<Box tag={'label'} htmlFor='day' mt={4}>
<Text mb={2} tag='p' size={200}>
Day
</Text>
<FeedbackForm.Item name='day'>
{({ input }) => {
const { state, className, ...other } = input;
return (
<InputNumber state={state} className={className}>
<InputNumber.Value id='day' {...other} placeholder='Enter day' autoComplete='off' />
</InputNumber>
);
}}
</FeedbackForm.Item>
</Box>
<FeedbackForm.Item name='call'>
{({ input }) => (
<RadioGroup {...input}>
<Flex direction='column' gap={1} my={4}>
<Radio>
<Radio.Value value='yes' />
<Radio.Text>Call me</Radio.Text>
</Radio>
<Radio>
<Radio.Value value='no' />
<Radio.Text>Don't call me!</Radio.Text>
</Radio>
</Flex>
</RadioGroup>
)}
</FeedbackForm.Item>
<FeedbackForm.Submit>Submit this strange form</FeedbackForm.Submit>
</FeedbackForm>
);
export default Demo;