added full control over which input device to use in voice chat (limited by cpal)
This commit is contained in:
@@ -2,7 +2,7 @@
|
|||||||
"name": "frangipane-client",
|
"name": "frangipane-client",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "1.0.2",
|
"version": "1.0.2",
|
||||||
"backendVersion": "1.0.5",
|
"backendVersion": "1.0.6",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
use cpal::traits::{DeviceTrait, HostTrait, StreamTrait};
|
use cpal::traits::{DeviceTrait, HostTrait, StreamTrait};
|
||||||
use cpal::SampleFormat;
|
use cpal::{Host, HostId, SampleFormat};
|
||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc, Mutex};
|
||||||
use tauri::{AppHandle, Emitter};
|
use tauri::{AppHandle, Emitter};
|
||||||
|
|
||||||
@@ -21,52 +21,85 @@ struct MicConfig {
|
|||||||
channels: usize,
|
channels: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Helper to resolve host string to cpal::Host
|
||||||
|
fn get_host_by_name(name: &str) -> Result<Host, String> {
|
||||||
|
let host_id = cpal::available_hosts()
|
||||||
|
.into_iter()
|
||||||
|
.find(|id| id.name() == name)
|
||||||
|
.ok_or_else(|| format!("Host '{}' not found", name))?;
|
||||||
|
|
||||||
|
cpal::host_from_id(host_id).map_err(|e| e.to_string())
|
||||||
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
pub fn start_microphone(app: AppHandle, state: tauri::State<AudioState>) -> Result<(), String> {
|
pub fn get_audio_hosts() -> Vec<String> {
|
||||||
#[cfg(all(any(
|
|
||||||
target_os = "linux",
|
|
||||||
target_os = "dragonfly",
|
|
||||||
target_os = "freebsd",
|
|
||||||
target_os = "netbsd"
|
|
||||||
)))]
|
|
||||||
let host = cpal::host_from_id(
|
|
||||||
cpal::available_hosts()
|
cpal::available_hosts()
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.find(|id| *id == cpal::HostId::Jack)
|
.map(|id| id.name().to_string())
|
||||||
.expect("jack host unavailable"),
|
.collect()
|
||||||
)
|
}
|
||||||
.unwrap_or(
|
|
||||||
cpal::host_from_id(*cpal::available_hosts().first().expect("no host available"))
|
|
||||||
.expect("host not available"),
|
|
||||||
);
|
|
||||||
|
|
||||||
#[cfg(any(not(any(
|
#[tauri::command]
|
||||||
target_os = "linux",
|
pub fn get_input_devices(host_name: Option<String>) -> Result<Vec<(String, String)>, String> {
|
||||||
target_os = "dragonfly",
|
// If no host provided, use default
|
||||||
target_os = "freebsd",
|
let host = if let Some(name) = host_name {
|
||||||
target_os = "netbsd"
|
get_host_by_name(&name)?
|
||||||
))))]
|
} else {
|
||||||
let host = cpal::default_host();
|
cpal::default_host()
|
||||||
|
};
|
||||||
|
|
||||||
let device = host
|
let devices = host.input_devices().map_err(|e| e.to_string())?;
|
||||||
.default_input_device()
|
|
||||||
.ok_or("No input device available")?;
|
let device_names: Vec<(String, String)> = devices
|
||||||
|
.into_iter()
|
||||||
|
.filter_map(|device| {
|
||||||
|
Some((
|
||||||
|
device.id().ok()?.to_string(),
|
||||||
|
device.description().ok()?.to_string(),
|
||||||
|
))
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
Ok(device_names)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[tauri::command]
|
||||||
|
pub fn start_microphone(
|
||||||
|
app: AppHandle,
|
||||||
|
state: tauri::State<AudioState>,
|
||||||
|
host_name: Option<String>,
|
||||||
|
device_id: Option<String>,
|
||||||
|
) -> Result<(), String> {
|
||||||
|
let host = if let Some(ref h_name) = host_name {
|
||||||
|
get_host_by_name(h_name)?
|
||||||
|
} else {
|
||||||
|
cpal::default_host()
|
||||||
|
};
|
||||||
|
|
||||||
|
let device = if let Some(ref id) = device_id {
|
||||||
|
host.input_devices()
|
||||||
|
.map_err(|e| e.to_string())?
|
||||||
|
.find(|d| d.id().map(|i| i.to_string()) == Ok(id.to_string()))
|
||||||
|
.ok_or_else(|| format!("Device '{}' not found on host '{:?}'", id, host.id()))?
|
||||||
|
} else {
|
||||||
|
host.default_input_device()
|
||||||
|
.ok_or("No input device available")?
|
||||||
|
};
|
||||||
|
|
||||||
let config = device
|
let config = device
|
||||||
.default_input_config()
|
.default_input_config()
|
||||||
.map_err(|e| format!("Failed to get config: {}", e))?;
|
.map_err(|e| format!("Failed to get config: {}", e))?;
|
||||||
|
|
||||||
println!("Microphone Config: {:?}", config);
|
|
||||||
let sample_rate = config.sample_rate();
|
let sample_rate = config.sample_rate();
|
||||||
let device_channels = config.channels() as usize;
|
let device_channels = config.channels() as usize;
|
||||||
let channels = 1; // NOTE: temporary
|
let channels = 1;
|
||||||
|
|
||||||
let buffer_threshold = (sample_rate as usize / 10) * 2;
|
let buffer_threshold = (sample_rate as usize / 10) * 2;
|
||||||
|
|
||||||
app.emit(
|
app.emit(
|
||||||
"microphone-config",
|
"microphone-config",
|
||||||
MicConfig {
|
MicConfig {
|
||||||
sample_rate,
|
sample_rate: sample_rate,
|
||||||
channels,
|
channels,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
@@ -74,39 +107,24 @@ pub fn start_microphone(app: AppHandle, state: tauri::State<AudioState>) -> Resu
|
|||||||
|
|
||||||
let sample_format = config.sample_format();
|
let sample_format = config.sample_format();
|
||||||
let stream_config: cpal::StreamConfig = config.into();
|
let stream_config: cpal::StreamConfig = config.into();
|
||||||
|
|
||||||
let err_fn = |err| eprintln!("Stream error: {}", err);
|
let err_fn = |err| eprintln!("Stream error: {}", err);
|
||||||
let app_handle = app.clone();
|
let app_handle = app.clone();
|
||||||
|
|
||||||
let mut emit_if_full = move |buffer: &mut Vec<u8>| {
|
|
||||||
if buffer.len() >= buffer_threshold {
|
|
||||||
let payload = buffer.clone();
|
|
||||||
let _ = app_handle.emit("microphone-data", payload);
|
|
||||||
buffer.clear();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let app_handle = app.clone();
|
|
||||||
|
|
||||||
let stream = match sample_format {
|
let stream = match sample_format {
|
||||||
SampleFormat::F32 => {
|
cpal::SampleFormat::F32 => {
|
||||||
let mut local_buffer = Vec::with_capacity(buffer_threshold * 2);
|
let mut local_buffer = Vec::with_capacity(buffer_threshold * 2);
|
||||||
|
|
||||||
device.build_input_stream(
|
device.build_input_stream(
|
||||||
&stream_config,
|
&stream_config,
|
||||||
move |data: &[f32], _| {
|
move |data: &[f32], _| {
|
||||||
for frame in data.chunks(device_channels) {
|
for frame in data.chunks(device_channels) {
|
||||||
let sample = frame[0]; // Take first channel
|
let sample = frame[0].clamp(-1.0, 1.0);
|
||||||
let s = sample.clamp(-1.0, 1.0);
|
let v = if sample >= 0.0 {
|
||||||
let v = if s >= 0.0 {
|
(sample * 32767.0) as i16
|
||||||
(s * 32767.0) as i16
|
|
||||||
} else {
|
} else {
|
||||||
(s * 32768.0) as i16
|
(sample * 32768.0) as i16
|
||||||
};
|
};
|
||||||
local_buffer.extend_from_slice(&v.to_le_bytes());
|
local_buffer.extend_from_slice(&v.to_le_bytes());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Only emit if we have enough data
|
|
||||||
if local_buffer.len() >= buffer_threshold {
|
if local_buffer.len() >= buffer_threshold {
|
||||||
let _ = app_handle.emit("microphone-data", &local_buffer);
|
let _ = app_handle.emit("microphone-data", &local_buffer);
|
||||||
local_buffer.clear();
|
local_buffer.clear();
|
||||||
@@ -116,38 +134,14 @@ pub fn start_microphone(app: AppHandle, state: tauri::State<AudioState>) -> Resu
|
|||||||
None,
|
None,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
SampleFormat::I16 => {
|
cpal::SampleFormat::I16 => {
|
||||||
let mut local_buffer = Vec::with_capacity(buffer_threshold * 2);
|
let mut local_buffer = Vec::with_capacity(buffer_threshold * 2);
|
||||||
|
|
||||||
device.build_input_stream(
|
device.build_input_stream(
|
||||||
&stream_config,
|
&stream_config,
|
||||||
move |data: &[i16], _| {
|
move |data: &[i16], _| {
|
||||||
for frame in data.chunks(device_channels) {
|
for frame in data.chunks(device_channels) {
|
||||||
let sample = frame[0];
|
local_buffer.extend_from_slice(&frame[0].to_le_bytes());
|
||||||
local_buffer.extend_from_slice(&sample.to_le_bytes());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if local_buffer.len() >= buffer_threshold {
|
|
||||||
let _ = app_handle.emit("microphone-data", &local_buffer);
|
|
||||||
local_buffer.clear();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
err_fn,
|
|
||||||
None,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
SampleFormat::U16 => {
|
|
||||||
let mut local_buffer = Vec::with_capacity(buffer_threshold * 2);
|
|
||||||
|
|
||||||
device.build_input_stream(
|
|
||||||
&stream_config,
|
|
||||||
move |data: &[u16], _| {
|
|
||||||
for frame in data.chunks(device_channels) {
|
|
||||||
let sample = frame[0];
|
|
||||||
let v = (sample as i32 - 32768) as i16;
|
|
||||||
local_buffer.extend_from_slice(&v.to_le_bytes());
|
|
||||||
}
|
|
||||||
|
|
||||||
if local_buffer.len() >= buffer_threshold {
|
if local_buffer.len() >= buffer_threshold {
|
||||||
let _ = app_handle.emit("microphone-data", &local_buffer);
|
let _ = app_handle.emit("microphone-data", &local_buffer);
|
||||||
local_buffer.clear();
|
local_buffer.clear();
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
mod audio;
|
mod audio;
|
||||||
use audio::{start_microphone, stop_microphone};
|
use audio::{get_audio_hosts, get_input_devices, start_microphone, stop_microphone};
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
fn log(message: &str) {
|
fn log(message: &str) {
|
||||||
@@ -22,7 +22,9 @@ pub fn run() {
|
|||||||
.invoke_handler(tauri::generate_handler![
|
.invoke_handler(tauri::generate_handler![
|
||||||
start_microphone,
|
start_microphone,
|
||||||
stop_microphone,
|
stop_microphone,
|
||||||
log
|
log,
|
||||||
|
get_input_devices,
|
||||||
|
get_audio_hosts
|
||||||
])
|
])
|
||||||
.run(tauri::generate_context!())
|
.run(tauri::generate_context!())
|
||||||
.expect("error while running tauri application");
|
.expect("error while running tauri application");
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"$schema": "https://schema.tauri.app/config/2",
|
"$schema": "https://schema.tauri.app/config/2",
|
||||||
"productName": "frangipane",
|
"productName": "frangipane",
|
||||||
"version": "1.0.1",
|
"version": "1.0.2",
|
||||||
"identifier": "com.strawberries.frangipane",
|
"identifier": "com.strawberries.frangipane",
|
||||||
"build": {
|
"build": {
|
||||||
"beforeDevCommand": "yarn dev",
|
"beforeDevCommand": "yarn dev",
|
||||||
|
|||||||
@@ -7,6 +7,8 @@
|
|||||||
:ownerUuid="currentRoom?.owner_uuid || ''" @close="showDetailsModal = false"
|
:ownerUuid="currentRoom?.owner_uuid || ''" @close="showDetailsModal = false"
|
||||||
@room-changed="handleRoomChanged" />
|
@room-changed="handleRoomChanged" />
|
||||||
|
|
||||||
|
<VoiceDeviceModal v-if="showVoiceModal" @close="showVoiceModal = false" @select="handleVoiceSelect" />
|
||||||
|
|
||||||
<div v-if="uuid === 'none'" class="no-room">
|
<div v-if="uuid === 'none'" class="no-room">
|
||||||
<div class="empty-state">
|
<div class="empty-state">
|
||||||
<i class="fa-solid fa-comments"></i>
|
<i class="fa-solid fa-comments"></i>
|
||||||
@@ -44,8 +46,7 @@
|
|||||||
<i class="fa-solid fa-users"></i>
|
<i class="fa-solid fa-users"></i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button class="invite-btn" @click="toggleVoice" :class="{ 'active-voice': isCurrentRoomVoice }"
|
<button class="invite-btn" @click="toggleVoice" :class="{ 'active-voice': isCurrentRoomVoice }">
|
||||||
title="Join Voice Chat">
|
|
||||||
<i class="fa-solid" :class="isCurrentRoomVoice ? 'fa-phone-slash' : 'fa-phone'"></i>
|
<i class="fa-solid" :class="isCurrentRoomVoice ? 'fa-phone-slash' : 'fa-phone'"></i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
@@ -68,6 +69,7 @@ import MessageList from "./MessageList.vue";
|
|||||||
import MessageInput from "./MessageInput.vue";
|
import MessageInput from "./MessageInput.vue";
|
||||||
import InvitePeopleModal from './InvitePeopleModal.vue';
|
import InvitePeopleModal from './InvitePeopleModal.vue';
|
||||||
import RoomDetailsModal from "./RoomDetailsModal.vue";
|
import RoomDetailsModal from "./RoomDetailsModal.vue";
|
||||||
|
import VoiceDeviceModal from "./VoiceDeviceModal.vue";
|
||||||
import WebSocket from '@tauri-apps/plugin-websocket';
|
import WebSocket from '@tauri-apps/plugin-websocket';
|
||||||
import { getAuthData } from "../store.ts";
|
import { getAuthData } from "../store.ts";
|
||||||
import { fetchRoomInfo } from "../api/rooms.ts";
|
import { fetchRoomInfo } from "../api/rooms.ts";
|
||||||
@@ -96,9 +98,11 @@ const connectionError = ref<string | null>(null);
|
|||||||
// Pagination State
|
// Pagination State
|
||||||
const isLoadingMore = ref(false);
|
const isLoadingMore = ref(false);
|
||||||
const hasMore = ref(true);
|
const hasMore = ref(true);
|
||||||
|
const isInitialLoad = ref(false);
|
||||||
|
|
||||||
const showInviteModal = ref(false);
|
const showInviteModal = ref(false);
|
||||||
const showDetailsModal = ref(false);
|
const showDetailsModal = ref(false);
|
||||||
const isInitialLoad = ref(false);
|
const showVoiceModal = ref(false);
|
||||||
|
|
||||||
// WebSocket State
|
// WebSocket State
|
||||||
let socket: WebSocket | null = null;
|
let socket: WebSocket | null = null;
|
||||||
@@ -324,7 +328,19 @@ async function toggleVoice() {
|
|||||||
if (isCurrentRoomVoice.value) {
|
if (isCurrentRoomVoice.value) {
|
||||||
await voiceActions.leaveRoom();
|
await voiceActions.leaveRoom();
|
||||||
} else {
|
} else {
|
||||||
await voiceActions.joinRoom(props.uuid);
|
showVoiceModal.value = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleVoiceSelect(selection: { host: string | null, device: string | null }) {
|
||||||
|
showVoiceModal.value = false;
|
||||||
|
|
||||||
|
if (!selection) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await voiceActions.joinRoom(props.uuid, selection.host, selection.device);
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Failed to join voice", e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
198
src/components/VoiceDeviceModal.vue
Normal file
198
src/components/VoiceDeviceModal.vue
Normal file
@@ -0,0 +1,198 @@
|
|||||||
|
<template>
|
||||||
|
<div class="backdrop" @click.self="$emit('close')">
|
||||||
|
<div class="modal">
|
||||||
|
|
||||||
|
<div v-if="step === 1" class="step-container">
|
||||||
|
<h3>{{ $t('chat-voice-select-host') || 'Select Audio System' }}</h3>
|
||||||
|
<div v-if="loadingHosts" class="loading">Loading systems...</div>
|
||||||
|
|
||||||
|
<div v-else class="list">
|
||||||
|
<button @click="selectHost(null)" class="item default">
|
||||||
|
<i class="fa-solid fa-wand-magic-sparkles"></i>
|
||||||
|
<span>Automatic (Default)</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button v-for="host in hosts" :key="host" @click="selectHost(host)" class="item">
|
||||||
|
<i class="fa-solid fa-server"></i>
|
||||||
|
<span>{{ host }}</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else class="step-container">
|
||||||
|
<div class="header">
|
||||||
|
<button class="back-btn" @click="step = 1">
|
||||||
|
<i class="fa-solid fa-arrow-left"></i>
|
||||||
|
</button>
|
||||||
|
<h3>{{ selectedHost }} Devices</h3>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="loadingDevices" class="loading">Scanning devices...</div>
|
||||||
|
|
||||||
|
<div v-else class="list">
|
||||||
|
<button @click="selectDevice(null)" class="item default">
|
||||||
|
<i class="fa-solid fa-star"></i>
|
||||||
|
<span>Default Device</span>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button v-for="device in devices" :key="device[0]" @click="selectDevice(device[0])" class="item">
|
||||||
|
<i class="fa-solid fa-microphone"></i>
|
||||||
|
<div class="device-details">
|
||||||
|
<span class="device-name">{{ device[1] }}</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="actions">
|
||||||
|
<button class="secondary" @click="$emit('close')">Cancel</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { ref, onMounted } from 'vue';
|
||||||
|
import { voiceActions } from '../voice';
|
||||||
|
|
||||||
|
const emit = defineEmits(['close', 'select']);
|
||||||
|
|
||||||
|
const step = ref(1);
|
||||||
|
const hosts = ref<string[]>([]);
|
||||||
|
const devices = ref<string[]>([]); // Array of [id, name] tuples
|
||||||
|
const selectedHost = ref<string | null>(null);
|
||||||
|
|
||||||
|
const loadingHosts = ref(false);
|
||||||
|
const loadingDevices = ref(false);
|
||||||
|
|
||||||
|
onMounted(async () => {
|
||||||
|
loadingHosts.value = true;
|
||||||
|
hosts.value = await voiceActions.getHosts();
|
||||||
|
loadingHosts.value = false;
|
||||||
|
});
|
||||||
|
|
||||||
|
async function selectHost(host: string | null) {
|
||||||
|
selectedHost.value = host;
|
||||||
|
|
||||||
|
// Default
|
||||||
|
if (host === null) {
|
||||||
|
emit('select', { host: null, device: null });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
step.value = 2;
|
||||||
|
loadingDevices.value = true;
|
||||||
|
devices.value = await voiceActions.getDevices(host);
|
||||||
|
loadingDevices.value = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectDevice(deviceId: string | null) {
|
||||||
|
emit('select', { host: selectedHost.value, device: deviceId });
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped>
|
||||||
|
.backdrop {
|
||||||
|
position: fixed;
|
||||||
|
inset: 0;
|
||||||
|
background: rgba(0, 0, 0, 0.6);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.modal {
|
||||||
|
background: var(--panel);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: var(--radius);
|
||||||
|
padding: 1.5rem;
|
||||||
|
width: 100%;
|
||||||
|
max-width: 500px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.step-container {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.back-btn {
|
||||||
|
background: transparent;
|
||||||
|
border: none;
|
||||||
|
color: var(--text);
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 1.1rem;
|
||||||
|
padding: 0.5rem;
|
||||||
|
border-radius: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.back-btn:hover {
|
||||||
|
background: var(--panel-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5rem;
|
||||||
|
max-height: 400px;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item {
|
||||||
|
background: none;
|
||||||
|
text-align: left;
|
||||||
|
font-weight: 500;
|
||||||
|
color: var(--text);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: var(--radius);
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0.8rem;
|
||||||
|
gap: 0.8rem;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item:hover {
|
||||||
|
border-color: var(--accent-color);
|
||||||
|
background: var(--panel-hover);
|
||||||
|
}
|
||||||
|
|
||||||
|
.device-details {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.device-name {
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actions {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
margin-top: 0.5rem;
|
||||||
|
border-top: 1px solid var(--border);
|
||||||
|
padding-top: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loading {
|
||||||
|
text-align: center;
|
||||||
|
color: var(--muted);
|
||||||
|
padding: 2rem;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -50,6 +50,7 @@ chat-create-global = Global room
|
|||||||
chat-create-submit = Create
|
chat-create-submit = Create
|
||||||
chat-connecting = Connecting to room...
|
chat-connecting = Connecting to room...
|
||||||
chat-connecting-failed = Could not connect. Check internet connection.
|
chat-connecting-failed = Could not connect. Check internet connection.
|
||||||
|
chat-voice-select-input = Select input device.
|
||||||
|
|
||||||
## User profile
|
## User profile
|
||||||
profile-title = User profile
|
profile-title = User profile
|
||||||
|
|||||||
@@ -50,6 +50,7 @@ chat-create-global = Salon public
|
|||||||
chat-create-submit = Créer
|
chat-create-submit = Créer
|
||||||
chat-connecting = Connexion au salon...
|
chat-connecting = Connexion au salon...
|
||||||
chat-connecting-failed = Impossible d'établir la connexion. Vérifiez votre internet.
|
chat-connecting-failed = Impossible d'établir la connexion. Vérifiez votre internet.
|
||||||
|
chat-voice-select-input = Sélectionnez un périphérique d'entrée.
|
||||||
|
|
||||||
## User profile
|
## User profile
|
||||||
profile-title = Profil d'utilisateur
|
profile-title = Profil d'utilisateur
|
||||||
|
|||||||
@@ -54,9 +54,9 @@ async function init() {
|
|||||||
|
|
||||||
init()
|
init()
|
||||||
|
|
||||||
export const API = 'http://127.0.0.1:8080'
|
// export const API = 'http://127.0.0.1:8080'
|
||||||
// export const API = 'http://192.168.1.183:8080'
|
export const API = 'http://192.168.1.183:8080'
|
||||||
// export const API = 'https://alatreon.org/frangipane'
|
// export const API = 'https://alatreon.org/frangipane'
|
||||||
export const API_WS = 'ws://127.0.0.1:8080/ws'
|
// export const API_WS = 'ws://127.0.0.1:8080/ws'
|
||||||
// export const API_WS = 'ws://192.168.1.183:8080/ws'
|
export const API_WS = 'ws://192.168.1.183:8080/ws'
|
||||||
// export const API_WS = 'wss://alatreon.org/frangipane/ws'
|
// export const API_WS = 'wss://alatreon.org/frangipane/ws'
|
||||||
|
|||||||
@@ -48,11 +48,9 @@ const errorMessage = ref("");
|
|||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
async function submit(event: Event) {
|
async function submit() {
|
||||||
errorMessage.value = "";
|
errorMessage.value = "";
|
||||||
|
|
||||||
const form = event.target as HTMLFormElement;
|
|
||||||
|
|
||||||
// if (!form.checkValidity()) {
|
// if (!form.checkValidity()) {
|
||||||
// if (password.value.length < 8) {
|
// if (password.value.length < 8) {
|
||||||
// errorMessage.value = $t('auth-error-password-length');
|
// errorMessage.value = $t('auth-error-password-length');
|
||||||
|
|||||||
51
src/voice.ts
51
src/voice.ts
@@ -1,5 +1,5 @@
|
|||||||
import { reactive } from 'vue';
|
import { reactive } from 'vue';
|
||||||
import { API_WS, logJS } from './main';
|
import { API_WS } from './main';
|
||||||
import { apiFetch } from './api/client';
|
import { apiFetch } from './api/client';
|
||||||
import WebSocket from '@tauri-apps/plugin-websocket';
|
import WebSocket from '@tauri-apps/plugin-websocket';
|
||||||
import { invoke } from '@tauri-apps/api/core';
|
import { invoke } from '@tauri-apps/api/core';
|
||||||
@@ -53,34 +53,46 @@ export async function initVoiceListeners() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const voiceActions = {
|
export const voiceActions = {
|
||||||
async joinRoom(roomUuid: string) {
|
async getHosts(): Promise<string[]> {
|
||||||
|
try {
|
||||||
|
return await invoke<string[]>('get_audio_hosts');
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async getDevices(hostName: string | null): Promise<string[]> {
|
||||||
|
try {
|
||||||
|
return await invoke('get_input_devices', { hostName });
|
||||||
|
} catch (e) {
|
||||||
|
console.error(e);
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async joinRoom(roomUuid: string, hostName: string | null, deviceName: string | null) {
|
||||||
if (voiceState.status === 'connected') return;
|
if (voiceState.status === 'connected') return;
|
||||||
|
|
||||||
nextStartTime = 0;
|
nextStartTime = 0;
|
||||||
voiceState.status = 'connecting';
|
voiceState.status = 'connecting';
|
||||||
voiceState.currentRoomUuid = roomUuid;
|
voiceState.currentRoomUuid = roomUuid;
|
||||||
|
|
||||||
// Ensure listeners are active
|
|
||||||
await initVoiceListeners();
|
await initVoiceListeners();
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const res = await apiFetch<{ token: string }>('/ws/issue-token');
|
const res = await apiFetch<{ token: string }>('/ws/issue-token');
|
||||||
const url = `${API_WS}/voice/${roomUuid}?token=${res.token}`;
|
const url = `${API_WS}/voice/${roomUuid}?token=${res.token}`;
|
||||||
|
|
||||||
socket = await WebSocket.connect(url);
|
socket = await WebSocket.connect(url);
|
||||||
|
|
||||||
socket.addListener((msg) => {
|
socket.addListener((msg) => {
|
||||||
if (msg.type === 'Binary') {
|
if (msg.type === 'Binary') handleIncomingAudio(msg.data);
|
||||||
handleIncomingAudio(msg.data);
|
else if (msg.type === 'Close') voiceActions.leaveRoom();
|
||||||
} else if (msg.type === 'Close') {
|
|
||||||
voiceActions.leaveRoom();
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
await this.startAudioCapture();
|
await this.startAudioCapture(hostName, deviceName);
|
||||||
voiceState.status = 'connected';
|
voiceState.status = 'connected';
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
logJS(e);
|
|
||||||
console.error("Voice join failed", e);
|
console.error("Voice join failed", e);
|
||||||
voiceActions.leaveRoom();
|
voiceActions.leaveRoom();
|
||||||
}
|
}
|
||||||
@@ -105,34 +117,25 @@ export const voiceActions = {
|
|||||||
voiceState.isMuted = !voiceState.isMuted;
|
voiceState.isMuted = !voiceState.isMuted;
|
||||||
},
|
},
|
||||||
|
|
||||||
async startAudioCapture() {
|
async startAudioCapture(hostName: string | null, deviceName: string | null) {
|
||||||
const hasPermission = await ensureMicrophonePermission();
|
const hasPermission = await ensureMicrophonePermission();
|
||||||
if (!hasPermission) {
|
if (!hasPermission) throw new Error("Microphone permission denied");
|
||||||
throw new Error("Microphone permission denied");
|
|
||||||
}
|
|
||||||
// Give the OS a moment to release the mic resource from the WebView
|
|
||||||
await new Promise(r => setTimeout(r, 200));
|
await new Promise(r => setTimeout(r, 200));
|
||||||
|
|
||||||
audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();
|
audioContext = new (window.AudioContext || (window as any).webkitAudioContext)();
|
||||||
|
|
||||||
unlistenConfig = await listen<{ sample_rate: number, channels: number }>('microphone-config', (event) => {
|
unlistenConfig = await listen<{ sample_rate: number, channels: number }>('microphone-config', (event) => {
|
||||||
console.log("Mic Config Received:", event.payload);
|
|
||||||
voiceState.captureSampleRate = event.payload.sample_rate;
|
voiceState.captureSampleRate = event.payload.sample_rate;
|
||||||
voiceState.captureChannels = event.payload.channels;
|
voiceState.captureChannels = event.payload.channels;
|
||||||
});
|
});
|
||||||
|
|
||||||
unlistenMic = await listen<number[]>('microphone-data', (event) => {
|
unlistenMic = await listen<number[]>('microphone-data', (event) => {
|
||||||
if (voiceState.isMuted || !socket) return;
|
if (voiceState.isMuted || !socket) return;
|
||||||
|
|
||||||
socket.send(event.payload).catch(console.error);
|
socket.send(event.payload).catch(console.error);
|
||||||
});
|
});
|
||||||
|
|
||||||
try {
|
await invoke('start_microphone', { hostName, deviceName });
|
||||||
await invoke('start_microphone');
|
|
||||||
} catch (e) {
|
|
||||||
console.error("Failed to start mic:", e);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
async stopAudioCapture() {
|
async stopAudioCapture() {
|
||||||
|
|||||||
Reference in New Issue
Block a user