React 配置小组件

  • DeviceConfig/index.tsx
javascript 复制代码
import React, { useState, useEffect } from 'react';
import { Form, Input, Select, Radio, Button } from 'antd';
import { PlusOutlined, DeleteOutlined } from '@ant-design/icons';
import './index.css';

interface DeviceOption {
  ua: string;
  name: string;
}

interface Device {
  device_type: string;
  version_type: string;
  version_start: string;
  version_end: string;
  version_list: string;
  version_exc: string;
}

interface DeviceConfigProps {
  deviceTypeOptions: DeviceOption[];
  deviceList: Device[];
  disabledDeviceOptions?: boolean;
  maxCount?: number;
  onDeviceTypeListDataChange?: (data: Device[]) => void;
}

const DeviceConfig: React.FC<DeviceConfigProps> = ({
                                                     deviceTypeOptions = [],
                                                     deviceList = [],
                                                     disabledDeviceOptions = false,
                                                     maxCount = 1,
                                                     onDeviceTypeListDataChange,
                                                   }) => {
  const [deviceTypeListData, setDeviceTypeListData] = useState<Device[]>(deviceList);

  // 添加 useEffect,当 deviceList 变化时更新 deviceTypeListData
  useEffect(() => {
    setDeviceTypeListData(deviceList);
  }, [deviceList]);

  useEffect(() => {
    if (onDeviceTypeListDataChange) {
      onDeviceTypeListDataChange(deviceTypeListData);
    }
  }, [deviceTypeListData, onDeviceTypeListDataChange]);

  const deviceStruct: Device = {
    device_type: '',
    version_type: '0',
    version_start: '',
    version_end: '',
    version_list: '',
    version_exc: '',
  };

  const haveAbleChoice = deviceTypeOptions.some(
    option => !deviceTypeListData.some(device => device.device_type === option.ua)
  );

  const getAvailableDeviceOptions = (index: number) => {
    const selectedTypes = deviceTypeListData
      .map((device, idx) => (idx !== index ? device.device_type : null))
      .filter(type => type !== null);
    return deviceTypeOptions.filter(option => !selectedTypes.includes(option.ua));
  };

  const addDevice = () => {
    const selectedTypes = deviceTypeListData.map(device => device.device_type);
    const firstAvailableType = deviceTypeOptions.find(
      option => !selectedTypes.includes(option.ua)
    );
    if (firstAvailableType) {
      const deviceData = { ...deviceStruct, device_type: firstAvailableType.ua };
      setDeviceTypeListData([...deviceTypeListData, deviceData]);
    }
  };

  const deleteDevice = (index: number) => {
    const newData = [...deviceTypeListData];
    newData.splice(index, 1);
    setDeviceTypeListData(newData);
  };

  const handleChange = (index: number, field: string, value: any) => {
    const newData = [...deviceTypeListData];
    (newData[index] as any)[field] = value;
    setDeviceTypeListData(newData);
  };

  const getVersionPlaceholder = (deviceType: string) =>
    '输入示例:' + (deviceType === 'pc_exe_template' ? '5123400' : '5.12.34');

  const getMultiVersionPlaceholder = (deviceType: string) =>
    '输入示例:' + (deviceType === 'pc_exe_template' ? '5123400,5123411' : '5.12.34,5.12.35');

  return (
    <div>
      {deviceTypeListData.map((device, index) => (
        <div key={index} className="device-item">
          <div className="group-row">
            <Form layout="vertical" style={{ width: '100%' }}>
              <Form.Item label="下发终端" required>
                <Select
                  value={device.device_type}
                  placeholder="请选择"
                  allowClear
                  style={{ width: 200 }}
                  onChange={value => handleChange(index, 'device_type', value)}
                >
                  {getAvailableDeviceOptions(index).map(item => (
                    <Select.Option key={item.ua} value={item.ua}>
                      {item.name}
                    </Select.Option>
                  ))}
                </Select>
              </Form.Item>
              <Form.Item label="指定版本" required>
                <Radio.Group
                  value={device.version_type}
                  onChange={e => handleChange(index, 'version_type', e.target.value)}
                >
                  <Radio value="0">全部</Radio>
                  <Radio value="1">范围</Radio>
                  <Radio value="2">包含</Radio>
                </Radio.Group>
              </Form.Item>
              {device.version_type === '1' && (
                <div style={{ display: 'flex', flexDirection: 'row' }}>
                  <Form.Item label="开始版本">
                    <Input
                      value={device.version_start}
                      placeholder={getVersionPlaceholder(device.device_type)}
                      onChange={e => handleChange(index, 'version_start', e.target.value)}
                    />
                  </Form.Item>
                  <Form.Item label="结束版本" style={{ marginLeft: 0 }}>
                    <Input
                      value={device.version_end}
                      placeholder={getVersionPlaceholder(device.device_type)}
                      onChange={e => handleChange(index, 'version_end', e.target.value)}
                    />
                  </Form.Item>
                </div>
              )}
              {device.version_type === '2' && (
                <Form.Item label="指定版本号">
                  <Input.TextArea
                    value={device.version_list}
                    placeholder={getMultiVersionPlaceholder(device.device_type)}
                    onChange={e => handleChange(index, 'version_list', e.target.value)}
                  />
                </Form.Item>
              )}
              <Form.Item label="剔除版本号">
                <Input.TextArea
                  value={device.version_exc}
                  placeholder={getMultiVersionPlaceholder(device.device_type)}
                  onChange={e => handleChange(index, 'version_exc', e.target.value)}
                />
                <span className="extra-info">若输入多个版本号,请使用英文逗号分隔</span>
              </Form.Item>
            </Form>
            <Button
              icon={<DeleteOutlined />}
              type="text"
              style={{ marginLeft: 10 }}
              onClick={() => deleteDevice(index)}
            />
          </div>
        </div>
      ))}
      {haveAbleChoice && (
        <div className="device-btn" onClick={addDevice}>
          <PlusOutlined style={{ fontWeight: 'bold', marginRight: 5 }} /> 增加终端
        </div>
      )}
    </div>
  );
};

export default DeviceConfig;
  • DeviceConfig/index.css
css 复制代码
/* DeviceConfig.css */
.device-btn {
  width: 138px;
  height: 32px;
  display: flex;
  justify-content: center;
  align-items: center;
  border: 1px dashed #d9d9d9;
  background: #ffffff;
  padding: 4px 15px;
}

.device-btn:hover {
  cursor: pointer;
  color: #409EFF;
  border: 1px dashed #409EFF;
}

.device-btn:active {
  background: #f2f0f7;
}

.device-item {
  margin: 0 0 24px;
  background-color: #f4f5ffda;
  padding: 16px;
  border-radius: 8px;
}

.group-row {
  display: flex;
  flex-direction: row;
  align-items: flex-end;
}

.extra-info {
  color: rgba(0, 0, 0, 0.45);
  font-size: 14px;
  line-height: 1.5715;
  transition: color 0.3s cubic-bezier(.215, .61, .355, 1);
}

index.tsx

javascript 复制代码
import React, { useState } from 'react';
import { useModel } from 'umi';
import { PageContainer } from '@ant-design/pro-layout';
import { Button, Modal, Form } from 'antd';
import DeviceConfig from './DeviceConfig'; // 请根据实际路径调整

const IndexPage: React.FC = () => {
  const { initialState } = useModel('@@initialState');
  const appConfigOption = initialState?.appConfigOption;

  const [visible, setVisible] = useState(false);
  const [modalType, setModalType] = useState<'modify' | 'open'>('open');
  const [deviceList, setDeviceList] = useState([
    { device_type: 'pc_exe_template', version_type: '0', version_exc: '5123400' },
  ]);

  const [form] = Form.useForm();

  const deviceTypeOptions = [
    { ua: 'pc_exe_template', name: 'PC EXE' },
    { ua: 'mobile_app', name: '移动应用' },
    // 添加更多设备类型选项
  ];

  const checkDeviceConfig = (rule: any, value: any) => {
    return new Promise<void>((resolve, reject) => {
      if (!value || value.length === 0) {
        reject('请至少配置一条下发的终端');
      } else {
        for (let item of value) {
          if (
            (item.version_type === '1' &&
              (!item.version_start || item.version_start.length === 0) &&
              (!item.version_end || item.version_end.length === 0)) ||
            (item.version_type === '2' && (!item.version_list || item.version_list.length === 0))
          ) {
            reject('请完善终端版本配置');
            return;
          }

          let regex = /^\d+\.\d+\.\d+$/;
          if (item.device_type === 'pc_exe_template') {
            regex = /^\d{7}$/;
          }

          if (item.version_type === '0') {
            if (
              item.version_exc &&
              item.version_exc.length !== 0 &&
              !item.version_exc
                .split(',')
                .every((segment: string) => segment.length === 0 || regex.test(segment))
            ) {
              reject('剔除版本号输入不合法');
              return;
            }
          } else if (item.version_type === '1') {
            if (item.version_start && item.version_start.length !== 0 && !regex.test(item.version_start)) {
              reject('开始版本输入不合法');
              return;
            }
            if (item.version_end && item.version_end.length !== 0 && !regex.test(item.version_end)) {
              reject('结束版本输入不合法');
              return;
            }
            if (
              item.version_exc &&
              item.version_exc.length !== 0 &&
              !item.version_exc
                .split(',')
                .every((segment: string) => segment.length === 0 || regex.test(segment))
            ) {
              reject('剔除版本号输入不合法');
              return;
            }
          } else {
            if (
              item.version_list &&
              item.version_list.length !== 0 &&
              !item.version_list
                .split(',')
                .every((segment: string) => segment.length === 0 || regex.test(segment))
            ) {
              reject('指定版本号输入不合法');
              return;
            }
            if (
              item.version_exc &&
              item.version_exc.length !== 0 &&
              !item.version_exc
                .split(',')
                .every((segment: string) => segment.length === 0 || regex.test(segment))
            ) {
              reject('剔除版本号输入不合法');
              return;
            }
          }
        }
        resolve();
      }
    });
  };

  const showModal = (type: 'modify' | 'open') => {
    setModalType(type);
    form.resetFields();
    if (type === 'modify') {
      form.setFieldsValue({ deviceList: deviceList });
    } else {
      form.setFieldsValue({
        deviceList: [{ device_type: 'pc_exe_template', version_type: '0', version_exc: '' }],
      });
    }
    setVisible(true);
  };

  const handleOk = () => {
    form
      .validateFields()
      .then((values) => {
        if (modalType === 'modify') {
          setDeviceList(values.deviceList); // 更新 deviceList
        }
        setVisible(false);
      })
      .catch((errorInfo) => {
        console.log('校验失败:', errorInfo);
      });
  };

  const handleCancel = () => {
    setVisible(false);
  };

  return (
    <PageContainer title={false}>
      <Button type="primary" onClick={() => showModal('modify')}>
        修改
      </Button>
      <Button type="default" onClick={() => showModal('open')} style={{ marginLeft: 8 }}>
        打开
      </Button>
      <Modal
        title="终端设置"
        visible={visible}
        onOk={handleOk}
        onCancel={handleCancel}
        destroyOnClose
        width={800}
      >
        <Form form={form}>
          <Form.Item
            label="终端设置"
            name="deviceList"
            rules={[{ validator: checkDeviceConfig }]}
          >
            <DeviceConfig
              deviceTypeOptions={deviceTypeOptions}
              deviceList={form.getFieldValue('deviceList')}
              onDeviceTypeListDataChange={(data) => {
                form.setFieldsValue({ deviceList: data });
              }}
            />
          </Form.Item>
        </Form>
      </Modal>
    </PageContainer>
  );
};

export default IndexPage;
相关推荐
傻小胖4 小时前
React 中hooks之useReducer使用场景和方法总结
前端·javascript·react.js
hikktn4 小时前
【开源宝藏】Jeepay VUE和React构建WebSocket通用模板
vue.js·react.js·开源
Mae_cpski6 小时前
【React学习笔记】第三章:React应用
笔记·学习·react.js
不叫猫先生7 小时前
【React】函数组件底层渲染机制
前端·javascript·react.js
练习两年半的工程师8 小时前
如何修改React 项目版本
前端·javascript·react.js
小李老笨了8 小时前
创建react18版本脚手架报错
前端·javascript·react.js
迷雾漫步者14 小时前
React封装倒计时按钮
前端·react.js·前端框架
lryh_1 天前
zustand 切片模式使用,persist 中间件持久化状态
react.js·zustand·persist
代码搬运媛1 天前
react 与 vue 的比较,以及如何选择?
前端·vue.js·react.js