mirror of
				https://github.com/owncast/owncast.git
				synced 2024-10-10 19:16:02 +00:00 
			
		
		
		
	Merge branch 'master' of github.com:thilobillerbeck/owncast-admin
This commit is contained in:
		
						commit
						f7a6e75708
					
				
							
								
								
									
										6
									
								
								web/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										6
									
								
								web/package-lock.json
									
									
									
										generated
									
									
									
								
							| @ -1575,6 +1575,12 @@ | ||||
|         "moment": "^2.10.2" | ||||
|       } | ||||
|     }, | ||||
|     "@types/classnames": { | ||||
|       "version": "2.2.11", | ||||
|       "resolved": "https://registry.npmjs.org/@types/classnames/-/classnames-2.2.11.tgz", | ||||
|       "integrity": "sha512-2koNhpWm3DgWRp5tpkiJ8JGc1xTn2q0l+jUNUE7oMKXUf5NpI9AIdC4kbjGNFBdHtcxBD18LAksoudAVhFKCjw==", | ||||
|       "dev": true | ||||
|     }, | ||||
|     "@types/color-name": { | ||||
|       "version": "1.1.1", | ||||
|       "resolved": "https://registry.npmjs.org/@types/color-name/-/color-name-1.1.1.tgz", | ||||
|  | ||||
| @ -24,6 +24,7 @@ | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@types/chart.js": "^2.9.28", | ||||
|     "@types/classnames": "^2.2.11", | ||||
|     "@types/node": "^14.11.2", | ||||
|     "@types/prop-types": "^15.7.3", | ||||
|     "@types/react": "^16.9.49", | ||||
|  | ||||
| @ -4,6 +4,7 @@ import '../styles/globals.scss'; | ||||
| 
 | ||||
| // GW: I can't override ant design styles through components using NextJS's built-in CSS modules. So I'll just import styles here for now and figure out enabling SASS modules later.
 | ||||
| import '../styles/home.scss'; | ||||
| import '../styles/chat.scss'; | ||||
| 
 | ||||
| import { AppProps } from 'next/app'; | ||||
| import ServerStatusProvider from '../utils/server-status-context'; | ||||
|  | ||||
							
								
								
									
										231
									
								
								web/pages/chat.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										231
									
								
								web/pages/chat.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,231 @@ | ||||
| import React, { useState, useEffect } from "react"; | ||||
| import { Table, Typography, Tooltip, Button } from "antd"; | ||||
| import { CheckCircleFilled, ExclamationCircleFilled, StopOutlined } from "@ant-design/icons"; | ||||
| import classNames from 'classnames'; | ||||
| import { ColumnsType } from 'antd/es/table'; | ||||
| import format from 'date-fns/format' | ||||
| 
 | ||||
| import { CHAT_HISTORY, fetchData, UPDATE_CHAT_MESSGAE_VIZ } from "../utils/apis"; | ||||
| import { MessageType } from '../types/chat'; | ||||
| import { isEmptyObject } from "../utils/format"; | ||||
| 
 | ||||
| const { Title } = Typography; | ||||
| 
 | ||||
| function createUserNameFilters(messages: MessageType[]) { | ||||
|   const filtered = messages.reduce((acc, curItem) => { | ||||
|     const curAuthor = curItem.author; | ||||
|     if (!acc.some(item => item.text === curAuthor)) { | ||||
|       acc.push({ text: curAuthor, value: curAuthor }); | ||||
|     } | ||||
|     return acc; | ||||
|   }, []); | ||||
| 
 | ||||
|   // sort by name
 | ||||
|   return filtered.sort((a, b) => { | ||||
|     const nameA = a.text.toUpperCase(); // ignore upper and lowercase
 | ||||
|     const nameB = b.text.toUpperCase(); // ignore upper and lowercase
 | ||||
|     if (nameA < nameB) { | ||||
|       return -1; | ||||
|     } | ||||
|     if (nameA > nameB) { | ||||
|       return 1; | ||||
|     } | ||||
|     // names must be equal
 | ||||
|     return 0; | ||||
|   }); | ||||
| } | ||||
| export const OUTCOME_TIMEOUT = 3000; | ||||
| 
 | ||||
| export default function Chat() { | ||||
|   const [messages, setMessages] = useState([]); | ||||
|   const [selectedRowKeys, setSelectedRows] = useState([]); | ||||
|   const [bulkProcessing, setBulkProcessing] = useState(false); | ||||
|   const [bulkOutcome, setBulkOutcome] = useState(null); | ||||
|   const [bulkAction, setBulkAction] = useState(''); | ||||
|   let outcomeTimeout = null; | ||||
| 
 | ||||
|   const getInfo = async () => { | ||||
|     try { | ||||
|       const result = await fetchData(CHAT_HISTORY, { auth: true }); | ||||
|       if (isEmptyObject(result)) { | ||||
|         setMessages([]); | ||||
|       } else { | ||||
|         setMessages(result); | ||||
|       } | ||||
|     } catch (error) { | ||||
|       console.log("==== error", error); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   useEffect(() => { | ||||
|     getInfo(); | ||||
|     return () => { | ||||
|       clearTimeout(outcomeTimeout); | ||||
|     }; | ||||
|   }, []); | ||||
| 
 | ||||
|   const nameFilters = createUserNameFilters(messages); | ||||
|    | ||||
|   const rowSelection = { | ||||
|     selectedRowKeys, | ||||
|     onChange: (selectedKeys: string[]) => { | ||||
|       setSelectedRows(selectedKeys); | ||||
|     }, | ||||
|   }; | ||||
| 
 | ||||
| 
 | ||||
|   const resetBulkOutcome = () => { | ||||
|     outcomeTimeout = setTimeout(() => { | ||||
|       setBulkOutcome(null); | ||||
|       setBulkAction(''); | ||||
|     }, OUTCOME_TIMEOUT); | ||||
|   }; | ||||
|   const handleSubmitBulk = async (bulkVisibility) => { | ||||
|     setBulkProcessing(true); | ||||
|     const result = await fetchData(UPDATE_CHAT_MESSGAE_VIZ, { | ||||
|       auth: true, | ||||
|       method: 'POST', | ||||
|       data: { | ||||
|         visible: bulkVisibility, | ||||
|         idArray: selectedRowKeys, | ||||
|       }, | ||||
|     }); | ||||
| 
 | ||||
|     if (result.success && result.message === "changed") { | ||||
|       setBulkOutcome(<CheckCircleFilled />); | ||||
|       resetBulkOutcome(); | ||||
| 
 | ||||
|       // update messages
 | ||||
|       const updatedList = [...messages]; | ||||
|       selectedRowKeys.map(key => { | ||||
|         const messageIndex = updatedList.findIndex(m => m.id === key); | ||||
|         const newMessage = {...messages[messageIndex], visible: bulkVisibility }; | ||||
|         updatedList.splice(messageIndex, 1, newMessage); | ||||
|         return null; | ||||
|       }); | ||||
|       setMessages(updatedList); | ||||
|       setSelectedRows([]); | ||||
|     } else { | ||||
|       setBulkOutcome(<ExclamationCircleFilled />); | ||||
|       resetBulkOutcome(); | ||||
|     } | ||||
|     setBulkProcessing(false); | ||||
|   } | ||||
|   const handleSubmitBulkShow = () => { | ||||
|     setBulkAction('show'); | ||||
|     handleSubmitBulk(true); | ||||
|   } | ||||
|   const handleSubmitBulkHide = () => { | ||||
|     setBulkAction('hide'); | ||||
|     handleSubmitBulk(false); | ||||
|   } | ||||
| 
 | ||||
|   const chatColumns: ColumnsType<MessageType> = [ | ||||
|     { | ||||
|       title: 'Time', | ||||
|       dataIndex: 'timestamp', | ||||
|       key: 'timestamp', | ||||
|       className: 'timestamp-col', | ||||
|       defaultSortOrder: 'descend', | ||||
|       render: (timestamp) => { | ||||
|         const dateObject = new Date(timestamp); | ||||
|         return format(dateObject, 'PP pp'); | ||||
|       }, | ||||
|       sorter: (a, b) => new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime(), | ||||
|       width: 90, | ||||
|     }, | ||||
|     { | ||||
|       title: 'User', | ||||
|       dataIndex: 'author', | ||||
|       key: 'author', | ||||
|       className: 'name-col', | ||||
|       filters: nameFilters, | ||||
|       onFilter: (value, record) => record.author === value, | ||||
|       sorter: (a, b) => a.author.localeCompare(b.author), | ||||
|       sortDirections: ['ascend', 'descend'],   | ||||
|       ellipsis: true, | ||||
|       render: author => ( | ||||
|         <Tooltip placement="topLeft" title={author}> | ||||
|           {author} | ||||
|         </Tooltip> | ||||
|       ), | ||||
|       width: 110, | ||||
|     }, | ||||
|     { | ||||
|       title: 'Message', | ||||
|       dataIndex: 'body', | ||||
|       key: 'body', | ||||
|       className: 'message-col', | ||||
|       width: 320, | ||||
|       render: body => ( | ||||
|         <div | ||||
|           className="message-contents" | ||||
|           // eslint-disable-next-line react/no-danger
 | ||||
|           dangerouslySetInnerHTML={{ __html: body }} | ||||
|         /> | ||||
|       ) | ||||
|     }, | ||||
|     { | ||||
|       title: '', | ||||
|       dataIndex: 'visible', | ||||
|       key: 'visible', | ||||
|       className: 'toggle-col', | ||||
|       filters: [{ text: 'Visible messages', value: true }, { text: 'Hidden messages', value: false }], | ||||
|       onFilter: (value, record) => record.visible === value, | ||||
|       render: visible => visible ? null : <StopOutlined title="This message is hidden" />, | ||||
|       width: 30, | ||||
|     }, | ||||
|   ]; | ||||
| 
 | ||||
|   const bulkDivClasses = classNames({ | ||||
|     'bulk-editor': true, | ||||
|     active: selectedRowKeys.length, | ||||
|   }); | ||||
|    | ||||
|   return ( | ||||
|     <div className="chat-messages"> | ||||
|       <Title level={2}>Chat Messages</Title> | ||||
|       <p>Manage the messages from viewers that show up on your stream.</p> | ||||
|       <div className={bulkDivClasses}> | ||||
|         <span className="label">Check multiple messages to change their visibility to: </span> | ||||
| 
 | ||||
|         <Button | ||||
|           type="primary" | ||||
|           size="small" | ||||
|           shape="round" | ||||
|           className="button" | ||||
|           loading={bulkAction === 'show' && bulkProcessing} | ||||
|           icon={bulkAction === 'show' && bulkOutcome} | ||||
|           disabled={!selectedRowKeys.length || (bulkAction && bulkAction !== 'show')} | ||||
|           onClick={handleSubmitBulkShow} | ||||
|         > | ||||
|           Show | ||||
|         </Button> | ||||
|         <Button | ||||
|           type="primary" | ||||
|           size="small" | ||||
|           shape="round" | ||||
|           className="button" | ||||
|           loading={bulkAction === 'hide' && bulkProcessing} | ||||
|           icon={bulkAction === 'hide' && bulkOutcome} | ||||
|           disabled={!selectedRowKeys.length || (bulkAction && bulkAction !== 'hide')} | ||||
|           onClick={handleSubmitBulkHide} | ||||
|         > | ||||
|           Hide | ||||
|         </Button> | ||||
|       </div> | ||||
|       <Table | ||||
|         size="small" | ||||
|         className="messages-table" | ||||
|         pagination={{ pageSize: 100 }}  | ||||
|         scroll={{ y: 540 }} | ||||
|         rowClassName={record => !record.visible ? 'hidden' : ''} | ||||
|         dataSource={messages} | ||||
|         columns={chatColumns} | ||||
|         rowKey={(row) => row.id} | ||||
|         rowSelection={rowSelection} | ||||
|       /> | ||||
|   </div>) | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| @ -1,7 +1,7 @@ | ||||
| import { LineChart } from 'react-chartkick' | ||||
| import styles from '../../styles/styles.module.scss'; | ||||
| import 'chart.js'; | ||||
| import format from 'date-fns/format' | ||||
| import format from 'date-fns/format'; | ||||
| import styles from '../../styles/styles.module.scss'; | ||||
| 
 | ||||
| interface TimedValue { | ||||
|   time: Date; | ||||
|  | ||||
| @ -13,7 +13,8 @@ import { | ||||
|   ToolOutlined, | ||||
|   PlayCircleFilled, | ||||
|   MinusSquareFilled, | ||||
|   QuestionCircleOutlined | ||||
|   QuestionCircleOutlined, | ||||
|   MessageOutlined | ||||
| } from '@ant-design/icons'; | ||||
| import classNames from 'classnames'; | ||||
| import { upgradeVersionAvailable } from "../../utils/apis"; | ||||
| @ -108,6 +109,14 @@ export default function MainLayout(props) { | ||||
|             <Link href="/viewer-info">Viewers</Link> | ||||
|           </Menu.Item> | ||||
| 
 | ||||
|           <Menu.Item | ||||
|             key="chat" | ||||
|             icon={<MessageOutlined />} | ||||
|             title="Chat utilities" | ||||
|           > | ||||
|             <Link href="/chat">Chat</Link> | ||||
|           </Menu.Item> | ||||
|            | ||||
|           <SubMenu | ||||
|             key="configuration" | ||||
|             title="Configuration" | ||||
|  | ||||
| @ -86,6 +86,8 @@ export default function ViewersOverTime() { | ||||
|       dataIndex: "connectedAt", | ||||
|       key: "connectedAt", | ||||
|       render: (time) => formatDistanceToNow(new Date(time)), | ||||
|       sorter: (a, b) => new Date(a.connectedAt).getTime() - new Date(b.connectedAt).getTime(), | ||||
|       sortDirections: ["descend", "ascend"] as SortOrder[], | ||||
|     }, | ||||
|     { | ||||
|       title: "User Agent", | ||||
|  | ||||
							
								
								
									
										83
									
								
								web/styles/chat.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								web/styles/chat.scss
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,83 @@ | ||||
| .chat-messages { | ||||
|   .ant-table-small .ant-table-selection-column { | ||||
|     width: 20px; | ||||
|     min-width: 20px; | ||||
|   } | ||||
|   .ant-table-tbody > tr > td { | ||||
|     transition: background 0.15s; | ||||
|   } | ||||
|   .ant-table-row.hidden { | ||||
|     .ant-table-cell { | ||||
|       color: #444450;  | ||||
|     } | ||||
| 
 | ||||
|   } | ||||
|   .ant-table-cell { | ||||
|     font-size: 12px; | ||||
| 
 | ||||
|     &.name-col { | ||||
|       text-overflow: ellipsis; | ||||
|       overflow: hidden; | ||||
|     } | ||||
|     &.toggle-col { | ||||
|       label { | ||||
|         font-size: 11px; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     .message-contents { | ||||
|       overflow: auto; | ||||
|       max-height: 200px;  | ||||
|       img { | ||||
|         position: relative; | ||||
|         margin-top: -5px; | ||||
|         width: 3rem; | ||||
|         padding: 0.25rem; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   .bulk-editor { | ||||
|     margin: .5rem 0; | ||||
|     padding: .5rem; | ||||
|     border: 1px solid #333; | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     align-items: center; | ||||
|     justify-content: flex-end; | ||||
|     border-radius: 4px; | ||||
| 
 | ||||
|     &.active { | ||||
|       .label { | ||||
|         color: #ccc; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     .label { | ||||
|       font-size: .75rem; | ||||
|       color: #666; | ||||
|       margin-right: .5rem; | ||||
|     } | ||||
| 
 | ||||
|     button { | ||||
|       margin: 0 .2rem; | ||||
|       font-size: .75rem; | ||||
|     } | ||||
| 
 | ||||
|   } | ||||
| } | ||||
| .ant-table-filter-dropdown { | ||||
|   max-width: 250px; | ||||
| } | ||||
| 
 | ||||
| .toggle-switch { | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   align-items: center; | ||||
|   flex-wrap: nowrap; | ||||
|   justify-content: flex-end; | ||||
| 
 | ||||
|   .outcome-icon { | ||||
|     margin-right: .5rem; | ||||
|   } | ||||
| } | ||||
| @ -3,4 +3,10 @@ | ||||
|   --owncast-purple-highlight: #ccd; | ||||
| 
 | ||||
|   --online-color: #73dd3f; | ||||
| 
 | ||||
| 
 | ||||
|   --ant-error: #ff4d4f; | ||||
|   --ant-success: #52c41a; | ||||
|   --ant-warning: #faad14; | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -41,4 +41,4 @@ pre { | ||||
|     background-color: rgb(44, 44, 44); | ||||
|     color:lightgrey; | ||||
|   } | ||||
| } | ||||
| } | ||||
|  | ||||
							
								
								
									
										10
									
								
								web/types/chat.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								web/types/chat.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | ||||
| export interface MessageType { | ||||
|   author: string; | ||||
|   body: string; | ||||
|   id: string; | ||||
|   key: string; | ||||
|   name: string; | ||||
|   timestamp: string; | ||||
|   type: string; | ||||
|   visible: boolean; | ||||
| } | ||||
| @ -34,22 +34,47 @@ export const LOGS_ALL = `${API_LOCATION}logs`; | ||||
| // Get warnings + errors
 | ||||
| export const LOGS_WARN = `${API_LOCATION}logs/warnings`; | ||||
| 
 | ||||
| // Get chat history
 | ||||
| export const CHAT_HISTORY = `${API_LOCATION}chat/messages`; | ||||
| 
 | ||||
| // Get chat history
 | ||||
| export const UPDATE_CHAT_MESSGAE_VIZ = `${NEXT_PUBLIC_API_HOST}api/admin/chat/updatemessagevisibility`; | ||||
| 
 | ||||
| 
 | ||||
| const GITHUB_RELEASE_URL = "https://api.github.com/repos/owncast/owncast/releases/latest"; | ||||
| 
 | ||||
| export async function fetchData(url) { | ||||
|   let options: RequestInit = {}; | ||||
| interface FetchOptions { | ||||
|   data?: any; | ||||
|   method?: string; | ||||
|   auth?: boolean; | ||||
| }; | ||||
| 
 | ||||
|   if (ADMIN_USERNAME && ADMIN_STREAMKEY) { | ||||
| export async function fetchData(url: string, options?: FetchOptions) { | ||||
|   const { | ||||
|     data, | ||||
|     method = 'GET', | ||||
|     auth = true, | ||||
|   } = options || {}; | ||||
| 
 | ||||
|   const requestOptions: RequestInit = { | ||||
|     method, | ||||
|   }; | ||||
| 
 | ||||
|   if (data) { | ||||
|     requestOptions.body = JSON.stringify(data) | ||||
|   } | ||||
| 
 | ||||
|   if (auth && ADMIN_USERNAME && ADMIN_STREAMKEY) { | ||||
|     const encoded = btoa(`${ADMIN_USERNAME}:${ADMIN_STREAMKEY}`); | ||||
|     options.headers = { | ||||
|     requestOptions.headers = { | ||||
|       'Authorization': `Basic ${encoded}` | ||||
|     } | ||||
|     options.mode = 'cors'; | ||||
|     options.credentials = 'include' | ||||
|     requestOptions.mode = 'cors'; | ||||
|     requestOptions.credentials = 'include'; | ||||
|   } | ||||
| 
 | ||||
|   try { | ||||
|     const response = await fetch(url, options); | ||||
|     const response = await fetch(url, requestOptions); | ||||
|     if (!response.ok) { | ||||
|       const message = `An error has occured: ${response.status}`; | ||||
|       throw new Error(message); | ||||
|  | ||||
| @ -15,7 +15,11 @@ export function formatIPAddress(ipAddress: string): string { | ||||
| 
 | ||||
| // check if obj is {}
 | ||||
| export function isEmptyObject(obj) { | ||||
|   return !obj || Object.keys(obj).length === 0; | ||||
|   return !obj || (Object.keys(obj).length === 0 && obj.constructor === Object); | ||||
| } | ||||
| 
 | ||||
| export function padLeft(text, pad, size) { | ||||
|   return String(pad.repeat(size) + text).slice(-size); | ||||
| } | ||||
| 
 | ||||
| export function parseSecondsToDurationString(seconds = 0) { | ||||
| @ -35,7 +39,3 @@ export function parseSecondsToDurationString(seconds = 0) { | ||||
| 
 | ||||
|   return daysString + hoursString + minString + secsString; | ||||
| } | ||||
| 
 | ||||
| export function padLeft(text, pad, size) { | ||||
|   return String(pad.repeat(size) + text).slice(-size); | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Thilo Billerbeck
						Thilo Billerbeck