mirror of
https://github.com/Monadical-SAS/reflector.git
synced 2025-12-20 12:19:06 +00:00
* feat: separate page into different component, greatly improving the loading and reactivity
* fix: various fixes
* feat: migrate to Chakra UI v3 - update theme, fix deprecated props
- Add whiteAlpha color palette with semantic tokens
- Update button recipe with fontWeight 600 and hover states
- Move Poppins font from theme to HTML tag className
- Fix deprecated props: isDisabled→disabled, align→alignItems/textAlign
- Remove button.css as styles are now handled by Chakra v3
* fix: complete Chakra UI v3 deprecated prop migrations
- Replace all isDisabled with disabled
- Replace all isChecked with checked
- Replace all isLoading with loading
- Replace all isOpen with open
- Replace all noOfLines with lineClamp
- Replace all align with alignItems on Flex/Stack components
- Replace all justify with justifyContent on Flex/Stack components
- Update temporary Select components to use new prop names
- Update REFACTOR2.md with completion status
* fix: add value prop to Menu.Item for proper hover states in Chakra v3
* fix: update browse page components for Chakra UI v3 compatibility
- Fix FilterSidebar status filter styling and prop usage
- Update Pagination component to use new Chakra v3 props and structure
- Refactor TranscriptTable to use modern Chakra patterns
- Clean up browse page layout and props
- Remove unused import from transcripts API view
- Enhance theme with additional semantic color tokens
* fix: polish browse page UI for Chakra v3
- Add rounded corners to FilterSidebar
- Adjust responsive breakpoints from md to lg for table/card view
- Add consistent font weights to table headers
- Improve card view typography and spacing
- Fix padding and margins for better mobile experience
- Remove unused table recipe from theme
* fix: padding
* fix: rework transcript page
* fix: more tidy layout for topic
* fix: share and privacy using chakra3 select
* fix: fix share and privacy select, now working, with closing dialog
* fix: complete Chakra UI v3 migration for share components and fix all TypeScript errors
- Refactor shareZulip.tsx to integrate modal content directly
- Replace react-select-search with Chakra UI v3 Select components using collection pattern
- Convert all Checkbox components to use v3 composable structure (Checkbox.Root, etc.)
- Fix Card components to use Card.Root and Card.Body
- Replace deprecated textColor prop with color prop
- Update Menu components to use v3 namespace pattern (Menu.Root, Menu.Trigger, etc.)
- Remove unused AlertDialog imports
- Fix useDisclosure hook changes (isOpen -> open)
- Replace UnorderedList with List.Root and ListItem with List.Item
- Fix Skeleton components by removing isLoaded prop and using conditional rendering
- Update Button variants to valid v3 options
- Fix Spinner props (remove thickness, speed, emptyColor)
- Update toast API to use custom toaster component
- Fix Progress components and FormControl to Field.Root
- Update Alert to use compound component pattern
- Remove shareModal.tsx file after integration
* fix: bring back topic list
* fix: normalize menu item
* fix: migrate rooms page to Chakra UI v3 pattern
- Updated layout to match browse page with Flex container and proper spacing
- Migrated add/edit room modal from custom HTML to Chakra UI v3 Dialog component
- Replaced all Select components with Chakra UI v3 Select using createListCollection
- Replaced FormControl/FormLabel/FormHelperText with Field.Root/Field.Label/Field.HelperText
- Removed inline styles and used Chakra props (mr={2} instead of style={{ marginRight: "8px" }})
- Fixed TypeScript interfaces removing OptionBase extension
- Fixed theme.ts accordion anatomy import issue
* refactor: convert rooms list to table view with responsive design
- Create RoomTable component for desktop view showing room details in columns
- Create RoomCards component for mobile/tablet responsive view
- Refactor RoomList to use table/card components based on screen size
- Display Zulip configuration, room size, and recording settings in table
- Remove unused RoomItem component
- Import Room type from API for proper typing
* refactor: extract RoomActionsMenu component to eliminate duplication
- Create RoomActionsMenu component for consistent room action menus
- Update RoomCards and RoomTable to use the new shared component
- Remove duplicated menu code from both components
* feat: add icons to TranscriptActionsMenu for consistency
- Add FaTrash icon for Delete action with red color
- Add FaArrowsRotate icon for Reprocess action
- Matches the pattern established in RoomActionsMenu
* refactor: update icons from Font Awesome to Lucide React
- Replace FaEllipsisVertical with LuMenu in menu triggers
- Replace FaLink with LuLink for copy URL buttons
- Replace FaPencil with LuPen for edit actions
- Replace FaTrash with LuTrash for delete actions
- Replace FaArrowsRotate with LuRotateCw for reprocess action
- Consistent icon library usage across all components
* refactor: little pass on the icons
* fix: lu icon
* fix: primary for button
* fix: recording page with mic selection
* fix: also fix duration
* fix: use combobox for share zulip
* fix: use proper theming for button, variant was not recognized
* fix: room actions menu
* fix: remove other variant primary left.
319 lines
9.2 KiB
TypeScript
319 lines
9.2 KiB
TypeScript
import React, { useRef, useEffect, useState } from "react";
|
|
|
|
import WaveSurfer from "wavesurfer.js";
|
|
import RecordPlugin from "../../lib/custom-plugins/record";
|
|
|
|
import { formatTime, formatTimeMs } from "../../lib/time";
|
|
import { waveSurferStyles } from "../../styles/recorder";
|
|
import { useError } from "../../(errors)/errorContext";
|
|
import FileUploadButton from "./fileUploadButton";
|
|
import useWebRTC from "./useWebRTC";
|
|
import useAudioDevice from "./useAudioDevice";
|
|
import { Box, Flex, IconButton, Menu, RadioGroup } from "@chakra-ui/react";
|
|
import { LuScreenShare, LuMic, LuPlay, LuStopCircle } from "react-icons/lu";
|
|
|
|
type RecorderProps = {
|
|
transcriptId: string;
|
|
status: string;
|
|
};
|
|
|
|
export default function Recorder(props: RecorderProps) {
|
|
const waveformRef = useRef<HTMLDivElement>(null);
|
|
const [record, setRecord] = useState<RecordPlugin | null>(null);
|
|
const [isRecording, setIsRecording] = useState<boolean>(false);
|
|
|
|
const [duration, setDuration] = useState<number>(0);
|
|
const [deviceId, setDeviceId] = useState<string | null>(null);
|
|
const { setError } = useError();
|
|
const [stream, setStream] = useState<MediaStream | null>(null);
|
|
|
|
// Time tracking, iirc it was drifting without this. to be tested again.
|
|
const [startTime, setStartTime] = useState(0);
|
|
const [currentTime, setCurrentTime] = useState<number>(0);
|
|
const [timeInterval, setTimeInterval] = useState<number | null>(null);
|
|
|
|
const webRTC = useWebRTC(stream, props.transcriptId);
|
|
|
|
const { audioDevices, getAudioStream } = useAudioDevice();
|
|
|
|
// Function used to setup keyboard shortcuts for the streamdeck
|
|
const setupProjectorKeys = (): (() => void) => {
|
|
if (!record) return () => {};
|
|
|
|
const handleKeyPress = (event: KeyboardEvent) => {
|
|
switch (event.key) {
|
|
case "~":
|
|
location.href = "";
|
|
break;
|
|
case ",":
|
|
location.href = "/transcripts/new";
|
|
break;
|
|
case "!":
|
|
if (record.isRecording()) return;
|
|
handleRecClick();
|
|
break;
|
|
case "@":
|
|
if (!record.isRecording()) return;
|
|
handleRecClick();
|
|
break;
|
|
case "(":
|
|
location.href = "/login";
|
|
break;
|
|
case ")":
|
|
location.href = "/logout";
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
};
|
|
|
|
document.addEventListener("keydown", handleKeyPress);
|
|
|
|
// Return the cleanup function
|
|
return () => {
|
|
document.removeEventListener("keydown", handleKeyPress);
|
|
};
|
|
};
|
|
|
|
// Setup Shortcuts
|
|
useEffect(() => {
|
|
if (!record) return;
|
|
|
|
return setupProjectorKeys();
|
|
}, [record, deviceId]);
|
|
|
|
// Waveform setup
|
|
useEffect(() => {
|
|
if (waveformRef.current) {
|
|
const _wavesurfer = WaveSurfer.create({
|
|
container: waveformRef.current,
|
|
hideScrollbar: true,
|
|
autoCenter: true,
|
|
barWidth: 2,
|
|
height: "auto",
|
|
|
|
...waveSurferStyles.player,
|
|
});
|
|
|
|
const _wshack: any = _wavesurfer;
|
|
_wshack.renderer.renderSingleCanvas = () => {};
|
|
|
|
// styling
|
|
const wsWrapper = _wavesurfer.getWrapper();
|
|
wsWrapper.style.cursor = waveSurferStyles.playerStyle.cursor;
|
|
wsWrapper.style.backgroundColor =
|
|
waveSurferStyles.playerStyle.backgroundColor;
|
|
wsWrapper.style.borderRadius = waveSurferStyles.playerStyle.borderRadius;
|
|
|
|
_wavesurfer.on("timeupdate", setCurrentTime);
|
|
|
|
setRecord(_wavesurfer.registerPlugin(RecordPlugin.create()));
|
|
|
|
return () => {
|
|
_wavesurfer.destroy();
|
|
setIsRecording(false);
|
|
setCurrentTime(0);
|
|
};
|
|
}
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
if (isRecording) {
|
|
const interval = window.setInterval(() => {
|
|
setCurrentTime(Date.now() - startTime);
|
|
}, 1000);
|
|
setTimeInterval(interval);
|
|
return () => clearInterval(interval);
|
|
} else {
|
|
clearInterval(timeInterval as number);
|
|
setCurrentTime((prev) => {
|
|
setDuration(prev / 1000);
|
|
return 0;
|
|
});
|
|
}
|
|
}, [isRecording]);
|
|
|
|
const handleRecClick = async () => {
|
|
if (!record) return console.log("no record");
|
|
|
|
if (record.isRecording()) {
|
|
setStream(null);
|
|
webRTC?.send(JSON.stringify({ cmd: "STOP" }));
|
|
record.stopRecording();
|
|
if (screenMediaStream) {
|
|
screenMediaStream.getTracks().forEach((t) => t.stop());
|
|
}
|
|
setIsRecording(false);
|
|
setScreenMediaStream(null);
|
|
setDestinationStream(null);
|
|
} else {
|
|
const stream = await getMicrophoneStream();
|
|
setStartTime(Date.now());
|
|
|
|
setStream(stream);
|
|
if (stream) {
|
|
await record.startRecording(stream);
|
|
setIsRecording(true);
|
|
}
|
|
}
|
|
};
|
|
|
|
const [screenMediaStream, setScreenMediaStream] =
|
|
useState<MediaStream | null>(null);
|
|
|
|
const handleRecordTabClick = async () => {
|
|
if (!record) return console.log("no record");
|
|
const stream: MediaStream = await navigator.mediaDevices.getDisplayMedia({
|
|
video: true,
|
|
audio: {
|
|
echoCancellation: true,
|
|
noiseSuppression: true,
|
|
sampleRate: 44100,
|
|
},
|
|
});
|
|
|
|
if (stream.getAudioTracks().length == 0) {
|
|
setError(new Error("No audio track found in screen recording."));
|
|
return;
|
|
}
|
|
setScreenMediaStream(stream);
|
|
};
|
|
|
|
const [destinationStream, setDestinationStream] =
|
|
useState<MediaStream | null>(null);
|
|
|
|
const startTabRecording = async () => {
|
|
if (!screenMediaStream) return;
|
|
if (!record) return;
|
|
if (destinationStream !== null) return console.log("already recording");
|
|
|
|
// connect mic audio (microphone)
|
|
const micStream = await getMicrophoneStream();
|
|
if (!micStream) {
|
|
console.log("no microphone audio");
|
|
return;
|
|
}
|
|
|
|
// Create MediaStreamSource nodes for the microphone and tab
|
|
const audioContext = new AudioContext();
|
|
const micSource = audioContext.createMediaStreamSource(micStream);
|
|
const tabSource = audioContext.createMediaStreamSource(screenMediaStream);
|
|
|
|
// Merge channels
|
|
// XXX If the length is not the same, we do not receive audio in WebRTC.
|
|
// So for now, merge the channels to have only one stereo source
|
|
const channelMerger = audioContext.createChannelMerger(1);
|
|
micSource.connect(channelMerger, 0, 0);
|
|
tabSource.connect(channelMerger, 0, 0);
|
|
|
|
// Create a MediaStreamDestination node
|
|
const destination = audioContext.createMediaStreamDestination();
|
|
channelMerger.connect(destination);
|
|
|
|
// Use the destination's stream for the WebRTC connection
|
|
setDestinationStream(destination.stream);
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (!record) return;
|
|
if (!destinationStream) return;
|
|
setStream(destinationStream);
|
|
if (destinationStream) {
|
|
record.startRecording(destinationStream);
|
|
setIsRecording(true);
|
|
}
|
|
}, [record, destinationStream]);
|
|
|
|
useEffect(() => {
|
|
startTabRecording();
|
|
}, [record, screenMediaStream]);
|
|
|
|
const timeLabel = () => {
|
|
if (isRecording) return formatTimeMs(currentTime);
|
|
if (duration) return `${formatTimeMs(currentTime)}/${formatTime(duration)}`;
|
|
return "";
|
|
};
|
|
|
|
const getMicrophoneStream = async () => {
|
|
return deviceId && getAudioStream ? await getAudioStream(deviceId) : null;
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (audioDevices && audioDevices.length > 0) {
|
|
setDeviceId(audioDevices[0].value);
|
|
}
|
|
}, [audioDevices]);
|
|
|
|
return (
|
|
<Flex className="flex items-center w-full relative">
|
|
<IconButton
|
|
aria-label={isRecording ? "Stop" : "Record"}
|
|
variant={"ghost"}
|
|
colorPalette={"blue"}
|
|
mr={2}
|
|
onClick={handleRecClick}
|
|
>
|
|
{isRecording ? <LuStopCircle /> : <LuPlay />}
|
|
</IconButton>
|
|
{!isRecording && (window as any).chrome && (
|
|
<IconButton
|
|
aria-label={"Record Tab"}
|
|
variant={"ghost"}
|
|
colorPalette={"blue"}
|
|
disabled={isRecording}
|
|
mr={2}
|
|
onClick={handleRecordTabClick}
|
|
size="sm"
|
|
>
|
|
<LuScreenShare />
|
|
</IconButton>
|
|
)}
|
|
{audioDevices && audioDevices?.length > 0 && deviceId && !isRecording && (
|
|
<Menu.Root>
|
|
<Menu.Trigger asChild>
|
|
<IconButton
|
|
aria-label={"Switch microphone"}
|
|
variant={"ghost"}
|
|
disabled={isRecording}
|
|
colorPalette={"blue"}
|
|
mr={2}
|
|
size="sm"
|
|
>
|
|
<LuMic />
|
|
</IconButton>
|
|
</Menu.Trigger>
|
|
<Menu.Positioner>
|
|
<Menu.Content>
|
|
<Menu.RadioItemGroup
|
|
value={deviceId}
|
|
onValueChange={(e) => setDeviceId(e.value)}
|
|
>
|
|
{audioDevices.map((device) => (
|
|
<Menu.RadioItem key={device.value} value={device.value}>
|
|
<Menu.ItemIndicator />
|
|
{device.label}
|
|
</Menu.RadioItem>
|
|
))}
|
|
</Menu.RadioItemGroup>
|
|
</Menu.Content>
|
|
</Menu.Positioner>
|
|
</Menu.Root>
|
|
)}
|
|
<Box position="relative" flex={1}>
|
|
<Box ref={waveformRef} height={14}></Box>
|
|
<Box
|
|
zIndex={50}
|
|
backgroundColor="rgba(255, 255, 255, 0.5)"
|
|
fontSize={"sm"}
|
|
shadow={"0px 0px 4px 0px white"}
|
|
position={"absolute"}
|
|
right={0}
|
|
bottom={0}
|
|
>
|
|
{timeLabel()}
|
|
</Box>
|
|
</Box>
|
|
</Flex>
|
|
);
|
|
}
|