// Updating a variant will post ALL the variants in an array as an update to the API. import React, { useContext, useState } from 'react'; import { Typography, Table, Modal, Button } from 'antd'; import { ColumnsType } from 'antd/lib/table'; import { DeleteOutlined } from '@ant-design/icons'; import { ServerStatusContext } from '../../utils/server-status-context'; import { AlertMessageContext } from '../../utils/alert-message-context'; import { UpdateArgs, VideoVariant } from '../../types/config-section'; import VideoVariantForm from './video-variant-form'; import { API_VIDEO_VARIANTS, DEFAULT_VARIANT_STATE, SUCCESS_STATES, RESET_TIMEOUT, postConfigUpdateToAPI, } from '../../utils/config-constants'; const { Title } = Typography; const CPU_USAGE_LEVEL_MAP = { 1: 'lowest', 2: 'low', 3: 'medium', 4: 'high', 5: 'highest', }; export default function CurrentVariantsTable() { const [displayModal, setDisplayModal] = useState(false); const [modalProcessing, setModalProcessing] = useState(false); const [editId, setEditId] = useState(0); const { setMessage } = useContext(AlertMessageContext); // current data inside modal const [modalDataState, setModalDataState] = useState(DEFAULT_VARIANT_STATE); const [submitStatus, setSubmitStatus] = useState(null); const [submitStatusMessage, setSubmitStatusMessage] = useState(''); const serverStatusData = useContext(ServerStatusContext); const { serverConfig, setFieldInConfigState } = serverStatusData || {}; const { videoSettings } = serverConfig || {}; const { videoQualityVariants } = videoSettings || {}; let resetTimer = null; if (!videoSettings) { return null; } const resetStates = () => { setSubmitStatus(null); setSubmitStatusMessage(''); resetTimer = null; clearTimeout(resetTimer); }; const handleModalCancel = () => { setDisplayModal(false); setEditId(-1); setModalDataState(DEFAULT_VARIANT_STATE); }; // posts all the variants at once as an array obj const postUpdateToAPI = async (postValue: any) => { await postConfigUpdateToAPI({ apiPath: API_VIDEO_VARIANTS, data: { value: postValue }, onSuccess: () => { setFieldInConfigState({ fieldName: 'videoQualityVariants', value: postValue, path: 'videoSettings', }); // close modal setModalProcessing(false); handleModalCancel(); setSubmitStatus('success'); setSubmitStatusMessage('Variants updated.'); resetTimer = setTimeout(resetStates, RESET_TIMEOUT); if (serverStatusData.online) { setMessage( 'Updating your video configuration will take effect the next time you begin a new stream.', ); } }, onError: (message: string) => { setSubmitStatus('error'); setSubmitStatusMessage(message); resetTimer = setTimeout(resetStates, RESET_TIMEOUT); }, }); }; // on Ok, send all of dataState to api // show loading // close modal when api is done const handleModalOk = () => { setModalProcessing(true); const postData = [...videoQualityVariants]; if (editId === -1) { postData.push(modalDataState); } else { postData.splice(editId, 1, modalDataState); } postUpdateToAPI(postData); }; const handleDeleteVariant = index => { const postData = [...videoQualityVariants]; postData.splice(index, 1); postUpdateToAPI(postData); }; const handleUpdateField = ({ fieldName, value }: UpdateArgs) => { setModalDataState({ ...modalDataState, [fieldName]: value, }); }; const { icon: newStatusIcon = null, message: newStatusMessage = '' } = SUCCESS_STATES[submitStatus] || {}; const videoQualityColumns: ColumnsType = [ { title: 'Video bitrate', dataIndex: 'videoBitrate', key: 'videoBitrate', render: (bitrate: number) => (!bitrate ? 'Same as source' : `${bitrate} kbps`), }, { title: 'CPU Usage', dataIndex: 'cpuUsageLevel', key: 'cpuUsageLevel', render: (level: string) => (!level ? 'n/a' : CPU_USAGE_LEVEL_MAP[level]), }, { title: '', dataIndex: '', key: 'edit', render: (data: VideoVariant) => { const index = data.key - 1; return (