RemoteVideoTrack.kt
3.2 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
package io.livekit.android.room.track
import android.view.View
import io.livekit.android.dagger.InjectionNames
import io.livekit.android.events.TrackEvent
import io.livekit.android.room.track.video.VideoSinkVisibility
import io.livekit.android.room.track.video.ViewVisibility
import io.livekit.android.util.LKLog
import kotlinx.coroutines.*
import org.webrtc.VideoSink
import javax.inject.Named
import kotlin.math.max
class RemoteVideoTrack(
name: String,
rtcTrack: org.webrtc.VideoTrack,
val autoManageVideo: Boolean = false,
@Named(InjectionNames.DISPATCHER_DEFAULT)
private val dispatcher: CoroutineDispatcher,
) : VideoTrack(name, rtcTrack) {
private var coroutineScope = CoroutineScope(dispatcher + SupervisorJob())
private val sinkVisibilityMap = mutableMapOf<VideoSink, VideoSinkVisibility>()
private val visibilities = sinkVisibilityMap.values
private var lastVisibility = false
private var lastDimensions: Dimensions = Dimensions(0, 0)
override fun addRenderer(renderer: VideoSink) {
if (autoManageVideo && renderer is View) {
addRenderer(renderer, ViewVisibility(renderer))
} else {
super.addRenderer(renderer)
}
}
fun addRenderer(renderer: VideoSink, visibility: VideoSinkVisibility) {
super.addRenderer(renderer)
if (autoManageVideo) {
sinkVisibilityMap[renderer] = visibility
visibility.addObserver { _, _ -> recalculateVisibility() }
recalculateVisibility()
} else {
LKLog.w { "attempted to tracking video sink visibility on an non auto managed video track." }
}
}
override fun removeRenderer(renderer: VideoSink) {
super.removeRenderer(renderer)
val visibility = sinkVisibilityMap.remove(renderer)
visibility?.close()
}
override fun stop() {
super.stop()
sinkVisibilityMap.values.forEach { it.close() }
sinkVisibilityMap.clear()
}
private fun hasVisibleSinks(): Boolean {
return visibilities.any { it.isVisible() }
}
private fun largestVideoViewSize(): Dimensions {
var maxWidth = 0
var maxHeight = 0
visibilities.forEach { visibility ->
val size = visibility.size()
maxWidth = max(maxWidth, size.width)
maxHeight = max(maxHeight, size.height)
}
return Dimensions(maxWidth, maxHeight)
}
private fun recalculateVisibility() {
val isVisible = hasVisibleSinks()
val newDimensions = largestVideoViewSize()
val eventsToPost = mutableListOf<TrackEvent>()
if (isVisible != lastVisibility) {
lastVisibility = isVisible
eventsToPost.add(TrackEvent.VisibilityChanged(this, isVisible))
}
if (newDimensions != lastDimensions) {
lastDimensions = newDimensions
eventsToPost.add(TrackEvent.VideoDimensionsChanged(this, newDimensions))
}
if (eventsToPost.any()) {
coroutineScope.launch {
eventBus.postEvents(eventsToPost)
}
}
}
override fun dispose() {
super.dispose()
coroutineScope.cancel()
}
}