mirror of
https://github.com/owncast/owncast.git
synced 2024-10-10 19:16:02 +00:00
finalize layout of textfields; add field status component
This commit is contained in:
parent
037e8f25a7
commit
b26b8abb9b
@ -16,7 +16,6 @@ import {
|
|||||||
API_YP_SWITCH,
|
API_YP_SWITCH,
|
||||||
} from './constants';
|
} from './constants';
|
||||||
|
|
||||||
import configStyles from '../../../styles/config-pages.module.scss';
|
|
||||||
import { UpdateArgs } from '../../../types/config-section';
|
import { UpdateArgs } from '../../../types/config-section';
|
||||||
|
|
||||||
export default function EditInstanceDetails() {
|
export default function EditInstanceDetails() {
|
||||||
@ -57,8 +56,8 @@ export default function EditInstanceDetails() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={configStyles.publicDetailsContainer}>
|
<div className={`publicDetailsContainer`}>
|
||||||
<div className={configStyles.textFieldsSection}>
|
<div className={`textFieldsSection`}>
|
||||||
<TextFieldWithSubmit
|
<TextFieldWithSubmit
|
||||||
fieldName="instanceUrl"
|
fieldName="instanceUrl"
|
||||||
{...TEXTFIELD_PROPS_INSTANCE_URL}
|
{...TEXTFIELD_PROPS_INSTANCE_URL}
|
||||||
|
|||||||
@ -13,7 +13,6 @@ import {
|
|||||||
TEXTFIELD_PROPS_WEB_PORT,
|
TEXTFIELD_PROPS_WEB_PORT,
|
||||||
} from './constants';
|
} from './constants';
|
||||||
|
|
||||||
import configStyles from '../../../styles/config-pages.module.scss';
|
|
||||||
import { UpdateArgs } from '../../../types/config-section';
|
import { UpdateArgs } from '../../../types/config-section';
|
||||||
|
|
||||||
export default function EditInstanceDetails() {
|
export default function EditInstanceDetails() {
|
||||||
@ -64,8 +63,8 @@ export default function EditInstanceDetails() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={configStyles.publicDetailsContainer}>
|
<div className={`publicDetailsContainer`}>
|
||||||
<div className={configStyles.textFieldsSection}>
|
<div className={`textFieldsSection`}>
|
||||||
<TextFieldWithSubmit
|
<TextFieldWithSubmit
|
||||||
fieldName="streamKey"
|
fieldName="streamKey"
|
||||||
{...TEXTFIELD_PROPS_STREAM_KEY}
|
{...TEXTFIELD_PROPS_STREAM_KEY}
|
||||||
|
|||||||
@ -16,8 +16,6 @@ import {
|
|||||||
import { SocialHandle } from '../../../types/config-section';
|
import { SocialHandle } from '../../../types/config-section';
|
||||||
import { isValidUrl } from '../../../utils/urls';
|
import { isValidUrl } from '../../../utils/urls';
|
||||||
|
|
||||||
import configStyles from '../../../styles/config-pages.module.scss';
|
|
||||||
|
|
||||||
const { Title } = Typography;
|
const { Title } = Typography;
|
||||||
|
|
||||||
export default function EditSocialLinks() {
|
export default function EditSocialLinks() {
|
||||||
@ -236,7 +234,7 @@ export default function EditSocialLinks() {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={configStyles.socialLinksEditor}>
|
<div className={`socialLinksEditor`}>
|
||||||
<Title level={2}>Social Links</Title>
|
<Title level={2}>Social Links</Title>
|
||||||
<p>Add all your social media handles and links to your other profiles here.</p>
|
<p>Add all your social media handles and links to your other profiles here.</p>
|
||||||
|
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import React, { useEffect, useState, useContext } from 'react';
|
import React, { useEffect, useState, useContext } from 'react';
|
||||||
import { Button } from 'antd';
|
import { Button } from 'antd';
|
||||||
|
import classNames from 'classnames';
|
||||||
import { RESET_TIMEOUT, postConfigUpdateToAPI } from './constants';
|
import { RESET_TIMEOUT, postConfigUpdateToAPI } from './constants';
|
||||||
|
|
||||||
import { ServerStatusContext } from '../../../utils/server-status-context';
|
import { ServerStatusContext } from '../../../utils/server-status-context';
|
||||||
@ -13,6 +13,7 @@ import {
|
|||||||
STATUS_SUCCESS,
|
STATUS_SUCCESS,
|
||||||
} from '../../../utils/input-statuses';
|
} from '../../../utils/input-statuses';
|
||||||
import { UpdateArgs } from '../../../types/config-section';
|
import { UpdateArgs } from '../../../types/config-section';
|
||||||
|
import InputStatusInfo from './input-status-info';
|
||||||
|
|
||||||
export const TEXTFIELD_TYPE_TEXT = 'default';
|
export const TEXTFIELD_TYPE_TEXT = 'default';
|
||||||
export const TEXTFIELD_TYPE_PASSWORD = 'password'; // Input.Password
|
export const TEXTFIELD_TYPE_PASSWORD = 'password'; // Input.Password
|
||||||
@ -43,7 +44,7 @@ export default function TextFieldWithSubmit(props: TextFieldWithSubmitProps) {
|
|||||||
...textFieldProps // rest of props
|
...textFieldProps // rest of props
|
||||||
} = props;
|
} = props;
|
||||||
|
|
||||||
const { fieldName, required, status, value, onChange, onSubmit } = textFieldProps;
|
const { fieldName, required, tip, status, value, onChange, onSubmit } = textFieldProps;
|
||||||
|
|
||||||
// Clear out any validation states and messaging
|
// Clear out any validation states and messaging
|
||||||
const resetStates = () => {
|
const resetStates = () => {
|
||||||
@ -105,8 +106,13 @@ export default function TextFieldWithSubmit(props: TextFieldWithSubmitProps) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const textfieldContainerClass = classNames({
|
||||||
|
'textfield-with-submit-container': true,
|
||||||
|
submittable: hasChanged,
|
||||||
|
});
|
||||||
return (
|
return (
|
||||||
<div className="textfield-with-submit-container">
|
<div className={textfieldContainerClass}>
|
||||||
|
<div className="textfield-component">
|
||||||
<TextField
|
<TextField
|
||||||
{...textFieldProps}
|
{...textFieldProps}
|
||||||
status={status || fieldStatus}
|
status={status || fieldStatus}
|
||||||
@ -114,14 +120,25 @@ export default function TextFieldWithSubmit(props: TextFieldWithSubmitProps) {
|
|||||||
onBlur={handleBlur}
|
onBlur={handleBlur}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
/>
|
/>
|
||||||
|
</div>
|
||||||
{hasChanged ? (
|
<div className="textfield-container lower-container">
|
||||||
|
<p className="label-spacer" />
|
||||||
|
<div className="lower-content">
|
||||||
|
<div className="field-tip">{tip}</div>
|
||||||
|
<InputStatusInfo status={status || fieldStatus} />
|
||||||
<div className="update-button-container">
|
<div className="update-button-container">
|
||||||
<Button type="primary" size="small" className="submit-button" onClick={handleSubmit}>
|
<Button
|
||||||
|
type="primary"
|
||||||
|
size="small"
|
||||||
|
className="submit-button"
|
||||||
|
onClick={handleSubmit}
|
||||||
|
disabled={!hasChanged}
|
||||||
|
>
|
||||||
Update
|
Update
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
) : null}
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,8 +1,10 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import classNames from 'classnames';
|
||||||
import { Input, InputNumber } from 'antd';
|
import { Input, InputNumber } from 'antd';
|
||||||
import { FieldUpdaterFunc } from '../../../types/config-section';
|
import { FieldUpdaterFunc } from '../../../types/config-section';
|
||||||
import InfoTip from '../info-tip';
|
// import InfoTip from '../info-tip';
|
||||||
import { StatusState } from '../../../utils/input-statuses';
|
import { StatusState } from '../../../utils/input-statuses';
|
||||||
|
import InputStatusInfo from './input-status-info';
|
||||||
|
|
||||||
export const TEXTFIELD_TYPE_TEXT = 'default';
|
export const TEXTFIELD_TYPE_TEXT = 'default';
|
||||||
export const TEXTFIELD_TYPE_PASSWORD = 'password'; // Input.Password
|
export const TEXTFIELD_TYPE_PASSWORD = 'password'; // Input.Password
|
||||||
@ -107,15 +109,20 @@ export default function TextField(props: TextFieldProps) {
|
|||||||
|
|
||||||
const fieldId = `field-${fieldName}`;
|
const fieldId = `field-${fieldName}`;
|
||||||
|
|
||||||
const { icon: statusIcon, message: statusMessage } = status || {};
|
const { type: statusType } = status || {};
|
||||||
|
|
||||||
|
const containerClass = classNames({
|
||||||
|
'textfield-container': true,
|
||||||
|
[`type-${type}`]: true,
|
||||||
|
required,
|
||||||
|
[`status-${statusType}`]: status,
|
||||||
|
});
|
||||||
return (
|
return (
|
||||||
<div className={`textfield-container type-${type}`}>
|
<div className={containerClass}>
|
||||||
{label ? (
|
{label ? (
|
||||||
<div className="label-side">
|
<div className="label-side">
|
||||||
<label htmlFor={fieldId} className="textfield-label">
|
<label htmlFor={fieldId} className="textfield-label">
|
||||||
{required ? <span className="required-label">* </span> : null}
|
{label}
|
||||||
{label}:
|
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
) : null}
|
) : null}
|
||||||
@ -135,17 +142,13 @@ export default function TextField(props: TextFieldProps) {
|
|||||||
disabled={disabled}
|
disabled={disabled}
|
||||||
value={value}
|
value={value}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div className="status-container">
|
<InputStatusInfo status={status} />
|
||||||
{status ? <span className="status-icon">{statusIcon}</span> : null}
|
<p className="field-tip">
|
||||||
{status ? <span className="status-message">{statusMessage}</span> : null}
|
{tip}
|
||||||
</div>
|
{/* <InfoTip tip={tip} /> */}
|
||||||
<p className="tip">
|
|
||||||
<InfoTip tip={tip} />
|
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
22
web/pages/components/config/input-status-info.tsx
Normal file
22
web/pages/components/config/input-status-info.tsx
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import classNames from 'classnames';
|
||||||
|
|
||||||
|
import { StatusState } from '../../../utils/input-statuses';
|
||||||
|
|
||||||
|
interface InputStatusInfoProps {
|
||||||
|
status: StatusState;
|
||||||
|
}
|
||||||
|
export default function InputStatusInfo({ status }: InputStatusInfoProps) {
|
||||||
|
const { type, icon, message } = status || {};
|
||||||
|
const classes = classNames({
|
||||||
|
'status-container': true,
|
||||||
|
[`status-${type}`]: type,
|
||||||
|
empty: !message,
|
||||||
|
});
|
||||||
|
return (
|
||||||
|
<div className={classes}>
|
||||||
|
{icon ? <span className="status-icon">{icon}</span> : null}
|
||||||
|
{message ? <span className="status-message">{message}</span> : null}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@ -2,7 +2,6 @@ import React from 'react';
|
|||||||
import { Typography } from 'antd';
|
import { Typography } from 'antd';
|
||||||
import Link from 'next/link';
|
import Link from 'next/link';
|
||||||
|
|
||||||
import configStyles from '../styles/config-pages.module.scss';
|
|
||||||
import EditInstanceDetails from './components/config/edit-instance-details';
|
import EditInstanceDetails from './components/config/edit-instance-details';
|
||||||
|
|
||||||
const { Title } = Typography;
|
const { Title } = Typography;
|
||||||
@ -12,8 +11,8 @@ export default function PublicFacingDetails() {
|
|||||||
<>
|
<>
|
||||||
<Title level={2}>Edit your public facing instance details</Title>
|
<Title level={2}>Edit your public facing instance details</Title>
|
||||||
|
|
||||||
<div className={configStyles.publicDetailsContainer}>
|
<div className={`publicDetailsContainer`}>
|
||||||
<div className={configStyles.textFieldsSection}>
|
<div className={`textFieldsSection`}>
|
||||||
<EditInstanceDetails />
|
<EditInstanceDetails />
|
||||||
|
|
||||||
<Link href="/admin/config-page-content">
|
<Link href="/admin/config-page-content">
|
||||||
|
|||||||
@ -1,35 +1,75 @@
|
|||||||
|
// Base styles for form-textfield, form-textfield-with-submit, and helper components.
|
||||||
|
|
||||||
|
.status-container {
|
||||||
|
&.status-success {
|
||||||
|
color: var(--ant-success);
|
||||||
|
}
|
||||||
|
&.status-error {
|
||||||
|
color: var(--ant-error);
|
||||||
|
}
|
||||||
|
&.status-warning {
|
||||||
|
color: var(--ant-warning);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.empty {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: flex-start;
|
||||||
|
align-items: center;
|
||||||
|
font-size: .75rem;
|
||||||
|
.status-icon {
|
||||||
|
display: inline-block;
|
||||||
|
margin-right: .5em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.field-tip {
|
||||||
|
font-size: .7em;
|
||||||
|
color: rgba(255,255,255,.7)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
.textfield-container {
|
.textfield-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
max-width: 600px;
|
||||||
|
|
||||||
.label-side {
|
.label-side {
|
||||||
padding-right: 1em;
|
padding-right: .75em;
|
||||||
text-align: right;
|
text-align: right;
|
||||||
width: 12rem;
|
width: 12em;
|
||||||
margin: .2em 0;
|
margin: .2em 0;
|
||||||
}
|
}
|
||||||
.textfield-label {
|
.textfield-label {
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
font-size: .85rem;
|
font-size: .85em;
|
||||||
color: var(--owncast-purple);
|
color: var(--owncast-purple);
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
content: ':';
|
||||||
}
|
}
|
||||||
.required-label {
|
}
|
||||||
|
&.required {
|
||||||
|
.textfield-label {
|
||||||
|
&::before {
|
||||||
|
content: '*';
|
||||||
|
display: inline-block;
|
||||||
|
margin-right: .25em;
|
||||||
color: var(--ant-error);
|
color: var(--ant-error);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.input-side {
|
.input-side {
|
||||||
max-width: 500px;
|
max-width: 500px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.input-group {
|
||||||
.input-group,
|
|
||||||
.status-container {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
@ -37,49 +77,88 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.status-container {
|
.status-container {
|
||||||
margin: 0 .25em;
|
margin: .25em;
|
||||||
min-height: 1.5em;
|
width: 100%;
|
||||||
font-size: .75em;
|
display: block;
|
||||||
|
&.empty {
|
||||||
.status-icon {
|
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
margin-right: .5em;
|
visibility: visible;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.tip {
|
.field-tip {
|
||||||
margin: .5em .5em;
|
margin: .5em .5em;
|
||||||
font-size: .75rem;
|
|
||||||
color: rgba(255,255,255,.75);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 800px) {
|
@media (max-width: 800px) {
|
||||||
// flex-direction: column;
|
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
.label-side {
|
.label-side {
|
||||||
|
width: 100%;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.status-message {
|
|
||||||
// margin: 1rem 0;
|
|
||||||
// min-height: 1.4em;
|
|
||||||
// font-size: .75rem;
|
|
||||||
&.success {
|
|
||||||
color: var(--ant-success);
|
|
||||||
}
|
|
||||||
&.error {
|
|
||||||
color: var(--ant-error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.textfield-with-submit-container {
|
.textfield-with-submit-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: flex-start;
|
||||||
|
margin-bottom: 1em;
|
||||||
|
|
||||||
|
.textfield-component {
|
||||||
|
width: 100%;
|
||||||
|
.textfield-container {
|
||||||
|
.field-tip,
|
||||||
|
.status-container {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// for lack of a better name
|
||||||
|
.lower-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
|
align-items: flex-start;
|
||||||
|
justify-content: flex-start;
|
||||||
|
|
||||||
|
.label-spacer {
|
||||||
|
width: 12em;
|
||||||
|
}
|
||||||
|
.lower-content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
align-items: flex-start;
|
||||||
|
justify-content: flex-end;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
.field-tip {
|
||||||
|
margin-right: 1em;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
.status-container {
|
||||||
|
margin: .5em;
|
||||||
|
}
|
||||||
|
}
|
||||||
.update-button-container {
|
.update-button-container {
|
||||||
display: inline-block;
|
visibility: hidden;
|
||||||
margin: .25em;
|
margin: .25em 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.submittable {
|
||||||
|
.lower-container {
|
||||||
|
.update-button-container {
|
||||||
|
visibility: visible;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@media (max-width: 800px) {
|
||||||
|
.label-spacer {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1,33 +0,0 @@
|
|||||||
.publicDetailsContainer {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
align-items: flex-start;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
|
|
||||||
.textFieldsSection {
|
|
||||||
margin-right: 2rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
.socialLinksEditor {
|
|
||||||
width: 20rem;
|
|
||||||
margin: 2em 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
.tag-editor-container,
|
|
||||||
.config-directory-details-form {
|
|
||||||
border-radius: 1em;
|
|
||||||
background-color: rgba(128,99,255,.1);
|
|
||||||
padding: 1.5em;
|
|
||||||
margin-bottom: 1em;
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////
|
|
||||||
// common?
|
|
||||||
.dataTable {
|
|
||||||
|
|
||||||
}
|
|
||||||
@ -13,12 +13,7 @@ export const STATUS_PROCESSING = 'proessing';
|
|||||||
export const STATUS_SUCCESS = 'success';
|
export const STATUS_SUCCESS = 'success';
|
||||||
export const STATUS_WARNING = 'warning';
|
export const STATUS_WARNING = 'warning';
|
||||||
|
|
||||||
export type InputStatusTypes =
|
export type InputStatusTypes = 'error' | 'invalid' | 'proessing' | 'success' | 'warning';
|
||||||
| typeof STATUS_ERROR
|
|
||||||
| typeof STATUS_INVALID
|
|
||||||
| typeof STATUS_PROCESSING
|
|
||||||
| typeof STATUS_SUCCESS
|
|
||||||
| typeof STATUS_WARNING;
|
|
||||||
|
|
||||||
export type StatusState = {
|
export type StatusState = {
|
||||||
type: InputStatusTypes;
|
type: InputStatusTypes;
|
||||||
@ -28,22 +23,27 @@ export type StatusState = {
|
|||||||
|
|
||||||
export const INPUT_STATES = {
|
export const INPUT_STATES = {
|
||||||
[STATUS_SUCCESS]: {
|
[STATUS_SUCCESS]: {
|
||||||
|
type: STATUS_SUCCESS,
|
||||||
icon: <CheckCircleFilled style={{ color: 'green' }} />,
|
icon: <CheckCircleFilled style={{ color: 'green' }} />,
|
||||||
message: 'Success!',
|
message: 'Success!',
|
||||||
},
|
},
|
||||||
[STATUS_ERROR]: {
|
[STATUS_ERROR]: {
|
||||||
|
type: STATUS_ERROR,
|
||||||
icon: <ExclamationCircleFilled style={{ color: 'red' }} />,
|
icon: <ExclamationCircleFilled style={{ color: 'red' }} />,
|
||||||
message: 'An error occurred.',
|
message: 'An error occurred.',
|
||||||
},
|
},
|
||||||
[STATUS_INVALID]: {
|
[STATUS_INVALID]: {
|
||||||
|
type: STATUS_INVALID,
|
||||||
icon: <ExclamationCircleFilled style={{ color: 'red' }} />,
|
icon: <ExclamationCircleFilled style={{ color: 'red' }} />,
|
||||||
message: 'An error occurred.',
|
message: 'An error occurred.',
|
||||||
},
|
},
|
||||||
[STATUS_PROCESSING]: {
|
[STATUS_PROCESSING]: {
|
||||||
|
type: STATUS_PROCESSING,
|
||||||
icon: <LoadingOutlined />,
|
icon: <LoadingOutlined />,
|
||||||
message: '',
|
message: '',
|
||||||
},
|
},
|
||||||
[STATUS_WARNING]: {
|
[STATUS_WARNING]: {
|
||||||
|
type: STATUS_WARNING,
|
||||||
icon: <WarningOutlined style={{ color: '#fc0' }} />,
|
icon: <WarningOutlined style={{ color: '#fc0' }} />,
|
||||||
message: '',
|
message: '',
|
||||||
},
|
},
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user