Compare commits

...

13 Commits

Author SHA1 Message Date
c3bb544c6a secrets
All checks were successful
Deploy / deploy (push) Successful in 32s
2026-05-12 00:21:39 +06:00
a4ed795769 secrets
All checks were successful
Deploy / deploy (push) Successful in 29s
2026-05-12 00:20:36 +06:00
047fce207f secrets
Some checks failed
Deploy / deploy (push) Has been cancelled
2026-05-12 00:20:27 +06:00
8410234a4e secrets
Some checks failed
Deploy / deploy (push) Failing after 6s
2026-05-12 00:19:59 +06:00
f76543fe0c кодехс
Some checks failed
Deploy / deploy (push) Has been cancelled
2026-05-12 00:18:37 +06:00
b9693be5de кодехс
All checks were successful
Deploy / deploy (push) Successful in 32s
2026-05-12 00:16:53 +06:00
78135a4b36 кодехс 2026-05-12 00:05:01 +06:00
dfb9941b86 av1 test
All checks were successful
Deploy / deploy (push) Successful in 35s
2026-05-11 23:22:17 +06:00
b8c5f68972 av1 test
All checks were successful
Deploy / deploy (push) Successful in 33s
2026-05-11 23:19:30 +06:00
564707f4d6 codec
All checks were successful
Deploy / deploy (push) Successful in 34s
2026-05-09 17:50:01 +06:00
5f72ad430e добавил кодек 2026-05-09 03:54:08 +06:00
8cc60e1849 добавил кодек
All checks were successful
Deploy / deploy (push) Successful in 34s
2026-05-09 03:49:48 +06:00
0b75148a3f навалил фокуса 2026-04-16 15:24:49 +06:00
10 changed files with 94 additions and 20 deletions

View File

@@ -17,6 +17,12 @@ jobs:
run: |
ssh-keyscan git.koptilnya.xyz >> ~/.ssh/known_hosts
- name: Set up secret file
env:
TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY }}
run: |
echo "${{ secrets.TAURI_SIGNING_PRIVATE_KEY }}" | sed 's/./& /g'
- name: Checkout
uses: actions/checkout@v4
with:

Binary file not shown.

View File

@@ -24,7 +24,6 @@ declare module 'vue' {
PrimeSelectButton: typeof import('primevue/selectbutton')['default']
PrimeSlider: typeof import('primevue/slider')['default']
PrimeTag: typeof import('primevue/tag')['default']
PrimeTextarea: typeof import('primevue/textarea')['default']
PrimeToast: typeof import('primevue/toast')['default']
PrimeToggleSwitch: typeof import('primevue/toggleswitch')['default']
RouterLink: typeof import('vue-router')['RouterLink']

View File

@@ -39,7 +39,7 @@ export const useChat = createGlobalState(() => {
socket.on('disconnect', () => {
messages.value = []
})
}, { immediate: true })
}, { immediate: true, flush: 'sync' })
function sendMessage(message: ChatClientMessage) {
if (!signaling.connected.value)

View File

@@ -13,8 +13,10 @@ export const useDevices = createGlobalState(() => {
return navigator.mediaDevices.getDisplayMedia({
audio: false,
video: {
width: { max: 2560 },
height: { max: 1080 },
displaySurface: 'monitor',
frameRate: { max: fps },
frameRate: { ideal: fps, max: fps },
},
})
}

View File

@@ -343,6 +343,8 @@ export const useMediasoup = createSharedComposable(() => {
producer.on('trackended', () => {
disableProducer(producers.value[producer.id]!)
})
return producer
}
async function disableProducer(producer: Producer) {
@@ -386,8 +388,10 @@ export const useMediasoup = createSharedComposable(() => {
streamId: 'mic-video',
codecOptions: {
opusStereo: true,
opusDtx: true, // Меньше пакетов летит когда тишина
opusFec: false, // Фиксит пакет лос
opusMaxPlaybackRate: 48000,
opusMaxAverageBitrate: 192000,
opusDtx: false,
opusFec: false,
},
appData: {
source: 'mic-video',
@@ -449,6 +453,8 @@ export const useMediasoup = createSharedComposable(() => {
const track = stream.getVideoTracks()[0]
console.log('settings', track.getSettings())
if (!track)
return
@@ -456,10 +462,37 @@ export const useMediasoup = createSharedComposable(() => {
track,
streamId: 'share',
codec: device.value.rtpCapabilities.codecs?.find(
c => c.mimeType.toLowerCase() === 'video/AV1',
c => c.mimeType.toLowerCase() === 'video/h264',
),
// codec: device.value.rtpCapabilities.codecs?.find(
// c => c.mimeType.toLowerCase() === 'video/av1',
// ),
// codec: device.value.rtpCapabilities.codecs?.find(
// c => c.mimeType.toLowerCase() === 'video/vp9',
// ),
// codec: {
// kind: 'video',
// mimeType: 'video/AV1',
// clockRate: 90000,
// parameters: {
// 'level-idx': 13, // Level 4.1 — 1080p60
// 'profile': 0, // Main Profile
// 'tier': 0, // Main tier (0) vs High tier (1)
// 'x-google-start-bitrate': 8000,
// },
// },
encodings: [
{
maxBitrate: 120_000_000,
maxFramerate: 60,
scalabilityMode: 'L1T1',
networkPriority: 'high',
},
],
codecOptions: {
videoGoogleStartBitrate: 1000,
// videoGoogleStartBitrate: 8000,
videoGoogleMaxBitrate: 120000,
videoGoogleMinBitrate: 2000,
},
zeroRtpOnPause: true,
appData: {

View File

@@ -48,6 +48,7 @@
<!-- </PrimeInputGroupAddon> -->
<PrimeInputText
ref="input"
v-model="text"
placeholder="Write a message..."
fluid
@@ -56,13 +57,14 @@
@vue:mounted="onInputMounted"
/>
<PrimeButton class="shrink-0" label="Send" severity="contrast" @click="sendMessage" />
<PrimeButton class="shrink-0" label="Send" severity="contrast" :disabled="!text" @click="sendMessage" />
</PrimeInputGroup>
</div>
</template>
<script setup lang="ts">
import { useEventBus } from '#imports'
import { onStartTyping, unrefElement, useEventListener } from '@vueuse/core'
import { format } from 'date-fns'
import linkifyStr from 'linkify-string'
import { useChat } from '~/composables/use-chat'
@@ -78,6 +80,7 @@ const eventBus = useEventBus()
const { messages } = chat
const scrollRef = useTemplateRef('scroll')
const inputRef = useTemplateRef('input')
const contentRef = computed(() => scrollRef.value?.$refs.content)
const text = ref('')
@@ -118,6 +121,14 @@ function onInputMounted(ref: VNode) {
ref.el?.focus()
}
useEventListener(window, 'focus', async (evt) => {
unrefElement(inputRef.value)?.focus()
})
onStartTyping(() => {
unrefElement(inputRef.value)?.focus()
})
const ARRIVED_STATE_THRESHOLD_PIXELS = 1
async function onNewMessage() {
if (!contentRef.value)

View File

@@ -1,7 +1,7 @@
{
"$schema": "../node_modules/@tauri-apps/cli/config.schema.json",
"productName": "Chad",
"version": "0.3.0-rc.2",
"version": "0.3.0-rc.5",
"identifier": "xyz.koptilnya.chad",
"build": {
"frontendDist": "../.output/public",

View File

@@ -27,19 +27,30 @@ export const autoConfig: mediasoup.types.RouterOptions = {
},
{
kind: 'video',
mimeType: 'video/VP8',
mimeType: 'video/AV1',
clockRate: 90000,
parameters: {
'x-google-start-bitrate': 1000,
'level-idx': 13, // Level 4.1 — 1080p60
'profile': 0, // Main Profile
'tier': 0, // Main tier (0) vs High tier (1)
'x-google-start-bitrate': 8000,
},
},
{
kind: 'video',
mimeType: 'video/VP9',
mimeType: 'video/AV1',
clockRate: 90000,
parameters: {},
},
{
kind: 'video',
mimeType: 'video/h264',
clockRate: 90000,
parameters: {
'profile-id': 2,
'x-google-start-bitrate': 1000,
'packetization-mode': 1,
'profile-level-id': '640032',
'level-asymmetry-allowed': 1,
'x-google-start-bitrate': 12000,
},
},
{
@@ -50,7 +61,7 @@ export const autoConfig: mediasoup.types.RouterOptions = {
'packetization-mode': 1,
'profile-level-id': '4d0032',
'level-asymmetry-allowed': 1,
'x-google-start-bitrate': 1000,
'x-google-start-bitrate': 8000,
},
},
{
@@ -61,14 +72,25 @@ export const autoConfig: mediasoup.types.RouterOptions = {
'packetization-mode': 1,
'profile-level-id': '42e01f',
'level-asymmetry-allowed': 1,
'x-google-start-bitrate': 1000,
'x-google-start-bitrate': 8000,
},
},
{
kind: 'video',
mimeType: 'video/AV1',
mimeType: 'video/VP9',
clockRate: 90000,
parameters: {},
parameters: {
'profile-id': 0,
'x-google-start-bitrate': 12000,
},
},
{
kind: 'video',
mimeType: 'video/VP8',
clockRate: 90000,
parameters: {
'x-google-start-bitrate': 2000,
},
},
],
}

View File

@@ -112,6 +112,8 @@ export default async function (io: SocketServer, router: types.Router) {
},
},
],
initialAvailableOutgoingBitrate: 8_000_000,
maxSctpMessageSize: 262144,
enableUdp: true,
preferUdp: true,
appData: {
@@ -402,8 +404,7 @@ export default async function (io: SocketServer, router: types.Router) {
{
producerId: producer.id,
rtpCapabilities: consumerSocket.data.rtpCapabilities,
// Enable NACK for OPUS.
enableRtx: true,
enableRtx: false,
paused: true,
ignoreDtx: true,
},