OpenTok SDK applied and method channels written to native
parent
57a3c2a04c
commit
d0986fcaf6
@ -0,0 +1,170 @@
|
||||
package com.example.basic_video_chat_flutter
|
||||
|
||||
import android.content.Context
|
||||
import android.os.Handler
|
||||
import android.os.Looper
|
||||
import android.util.AttributeSet
|
||||
import android.util.Log
|
||||
import android.view.LayoutInflater
|
||||
import android.view.View
|
||||
import android.view.ViewGroup
|
||||
import android.widget.FrameLayout
|
||||
import android.widget.LinearLayout
|
||||
import com.opentok.android.*
|
||||
import io.flutter.embedding.engine.FlutterEngine
|
||||
import io.flutter.plugin.common.MethodCall
|
||||
import io.flutter.plugin.common.MethodChannel
|
||||
import io.flutter.plugin.common.StandardMessageCodec
|
||||
import io.flutter.plugin.platform.PlatformView
|
||||
import io.flutter.plugin.platform.PlatformViewFactory
|
||||
|
||||
|
||||
enum class SdkState {
|
||||
LOGGED_OUT,
|
||||
LOGGED_IN,
|
||||
WAIT,
|
||||
ERROR
|
||||
}
|
||||
|
||||
class OpenTok(private var context: Context, private var flutterEngine: FlutterEngine){
|
||||
private lateinit var opentokVideoPlatformView: OpentokVideoPlatformView
|
||||
|
||||
init {
|
||||
opentokVideoPlatformView = OpentokVideoFactory.getViewInstance(context)
|
||||
flutterEngine
|
||||
.platformViewsController
|
||||
.registry
|
||||
// opentok-video-container is a custom platform-view-type
|
||||
.registerViewFactory("opentok-video-container", OpentokVideoFactory())
|
||||
}
|
||||
|
||||
private var session: Session? = null
|
||||
private var publisher: Publisher? = null
|
||||
private var subscriber: Subscriber? = null
|
||||
|
||||
|
||||
|
||||
private val sessionListener: Session.SessionListener = object: Session.SessionListener {
|
||||
override fun onConnected(session: Session) {
|
||||
// Connected to session
|
||||
Log.d("MainActivity", "Connected to session ${session.sessionId}")
|
||||
|
||||
publisher = Publisher.Builder(context).build().apply {
|
||||
setPublisherListener(publisherListener)
|
||||
renderer?.setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL)
|
||||
|
||||
view.layoutParams = ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT)
|
||||
opentokVideoPlatformView.publisherContainer.addView(view)
|
||||
}
|
||||
|
||||
notifyFlutter(SdkState.LOGGED_IN)
|
||||
session.publish(publisher)
|
||||
}
|
||||
|
||||
override fun onDisconnected(session: Session) {
|
||||
notifyFlutter(SdkState.LOGGED_OUT)
|
||||
}
|
||||
|
||||
override fun onStreamReceived(session: Session, stream: Stream) {
|
||||
Log.d(
|
||||
"MainActivity",
|
||||
"onStreamReceived: New Stream Received " + stream.streamId + " in session: " + session.sessionId
|
||||
)
|
||||
if (subscriber == null) {
|
||||
subscriber = Subscriber.Builder(context, stream).build().apply {
|
||||
renderer?.setStyle(BaseVideoRenderer.STYLE_VIDEO_SCALE, BaseVideoRenderer.STYLE_VIDEO_FILL)
|
||||
setSubscriberListener(subscriberListener)
|
||||
session.subscribe(this)
|
||||
|
||||
opentokVideoPlatformView.subscriberContainer.addView(view)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
override fun onStreamDropped(session: Session, stream: Stream) {
|
||||
Log.d(
|
||||
"MainActivity",
|
||||
"onStreamDropped: Stream Dropped: " + stream.streamId + " in session: " + session.sessionId
|
||||
)
|
||||
|
||||
if (subscriber != null) {
|
||||
subscriber = null
|
||||
|
||||
opentokVideoPlatformView.subscriberContainer.removeAllViews()
|
||||
}
|
||||
}
|
||||
|
||||
override fun onError(session: Session, opentokError: OpentokError) {
|
||||
Log.d("MainActivity", "Session error: " + opentokError.message)
|
||||
notifyFlutter(SdkState.ERROR)
|
||||
}
|
||||
}
|
||||
|
||||
private val publisherListener: PublisherKit.PublisherListener = object : PublisherKit.PublisherListener {
|
||||
override fun onStreamCreated(publisherKit: PublisherKit, stream: Stream) {
|
||||
Log.d("MainActivity", "onStreamCreated: Publisher Stream Created. Own stream " + stream.streamId)
|
||||
}
|
||||
|
||||
override fun onStreamDestroyed(publisherKit: PublisherKit, stream: Stream) {
|
||||
Log.d("MainActivity", "onStreamDestroyed: Publisher Stream Destroyed. Own stream " + stream.streamId)
|
||||
}
|
||||
|
||||
override fun onError(publisherKit: PublisherKit, opentokError: OpentokError) {
|
||||
Log.d("MainActivity", "PublisherKit onError: " + opentokError.message)
|
||||
notifyFlutter(SdkState.ERROR)
|
||||
}
|
||||
}
|
||||
|
||||
var subscriberListener: SubscriberKit.SubscriberListener = object : SubscriberKit.SubscriberListener {
|
||||
override fun onConnected(subscriberKit: SubscriberKit) {
|
||||
Log.d("MainActivity", "onConnected: Subscriber connected. Stream: " + subscriberKit.stream.streamId)
|
||||
}
|
||||
|
||||
override fun onDisconnected(subscriberKit: SubscriberKit) {
|
||||
Log.d("MainActivity", "onDisconnected: Subscriber disconnected. Stream: " + subscriberKit.stream.streamId)
|
||||
notifyFlutter(SdkState.LOGGED_OUT)
|
||||
}
|
||||
|
||||
override fun onError(subscriberKit: SubscriberKit, opentokError: OpentokError) {
|
||||
Log.d("MainActivity", "SubscriberKit onError: " + opentokError.message)
|
||||
notifyFlutter(SdkState.ERROR)
|
||||
}
|
||||
}
|
||||
|
||||
fun initSession(call: MethodCall, result: MethodChannel.Result) {
|
||||
|
||||
val apiKey = requireNotNull(call.argument<String>("apiKey"))
|
||||
val sessionId = requireNotNull(call.argument<String>("sessionId"))
|
||||
val token = requireNotNull(call.argument<String>("token"))
|
||||
|
||||
notifyFlutter(SdkState.WAIT)
|
||||
session = Session.Builder(context, apiKey, sessionId).build()
|
||||
session?.setSessionListener(sessionListener)
|
||||
session?.connect(token)
|
||||
result.success("")
|
||||
}
|
||||
|
||||
fun swapCamera(call: MethodCall, result: MethodChannel.Result) {
|
||||
publisher?.cycleCamera()
|
||||
result.success("")
|
||||
}
|
||||
|
||||
fun toggleAudio(call: MethodCall, result: MethodChannel.Result) {
|
||||
val publishAudio = requireNotNull(call.argument<Boolean>("publishAudio"))
|
||||
publisher?.publishAudio = publishAudio
|
||||
result.success("")
|
||||
}
|
||||
|
||||
fun toggleVideo(call: MethodCall, result: MethodChannel.Result) {
|
||||
val publishVideo = requireNotNull(call.argument<Boolean>("publishVideo"))
|
||||
publisher?.publishVideo = publishVideo
|
||||
result.success("")
|
||||
}
|
||||
|
||||
private fun notifyFlutter(state: SdkState) {
|
||||
Handler(Looper.getMainLooper()).post {
|
||||
MethodChannel(flutterEngine.dartExecutor.binaryMessenger, "OpenTok-Platform-Bridge")
|
||||
.invokeMethod("updateState", state.toString())
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,63 @@
|
||||
package com.example.basic_video_chat_flutter
|
||||
|
||||
import android.content.Context
|
||||
import android.util.AttributeSet
|
||||
import android.view.LayoutInflater
|
||||
import android.view.View
|
||||
import android.widget.FrameLayout
|
||||
import android.widget.LinearLayout
|
||||
import io.flutter.plugin.common.StandardMessageCodec
|
||||
import io.flutter.plugin.platform.PlatformView
|
||||
import io.flutter.plugin.platform.PlatformViewFactory
|
||||
|
||||
class OpentokVideoFactory : PlatformViewFactory(StandardMessageCodec.INSTANCE) {
|
||||
|
||||
companion object {
|
||||
private lateinit var view: OpentokVideoPlatformView
|
||||
|
||||
fun getViewInstance(context: Context): OpentokVideoPlatformView {
|
||||
if(!this::view.isInitialized) {
|
||||
view = OpentokVideoPlatformView(context)
|
||||
}
|
||||
|
||||
return view
|
||||
}
|
||||
}
|
||||
|
||||
override fun create(context: Context, viewId: Int, args: Any?): PlatformView {
|
||||
return getViewInstance(context)
|
||||
}
|
||||
}
|
||||
|
||||
class OpentokVideoPlatformView(context: Context) : PlatformView {
|
||||
private val videoContainer: OpenTokVideoContainer = OpenTokVideoContainer(context)
|
||||
|
||||
val subscriberContainer get() = videoContainer.subscriberContainer
|
||||
val publisherContainer get() = videoContainer.publisherContainer
|
||||
|
||||
override fun getView(): View {
|
||||
return videoContainer
|
||||
}
|
||||
|
||||
override fun dispose() {}
|
||||
}
|
||||
|
||||
class OpenTokVideoContainer @JvmOverloads constructor(
|
||||
context: Context,
|
||||
attrs: AttributeSet? = null,
|
||||
defStyle: Int = 0,
|
||||
defStyleRes: Int = 0
|
||||
) : LinearLayout(context, attrs, defStyle, defStyleRes) {
|
||||
|
||||
var subscriberContainer: FrameLayout
|
||||
private set
|
||||
|
||||
var publisherContainer: FrameLayout
|
||||
private set
|
||||
|
||||
init {
|
||||
val view = LayoutInflater.from(context).inflate(R.layout.view_video, this, true)
|
||||
subscriberContainer = view.findViewById(R.id.subscriber_container)
|
||||
publisherContainer = view.findViewById(R.id.publisher_container)
|
||||
}
|
||||
}
|
@ -0,0 +1,50 @@
|
||||
package com.ejada.hmg.utils
|
||||
|
||||
import android.content.Context
|
||||
import android.net.wifi.WifiManager
|
||||
import android.util.Log
|
||||
import com.ejada.hmg.MainActivity
|
||||
import com.ejada.hmg.hmgwifi.HMG_Guest
|
||||
import com.ejada.hmg.hmgwifi.HMG_Internet
|
||||
import com.ejada.hmg.geofence.GeoZoneModel
|
||||
import com.ejada.hmg.geofence.HMG_Geofence
|
||||
import com.example.basic_video_chat_flutter.OpenTok
|
||||
import io.flutter.embedding.engine.FlutterEngine
|
||||
import io.flutter.plugin.common.BinaryMessenger
|
||||
import io.flutter.plugin.common.MethodCall
|
||||
import io.flutter.plugin.common.MethodChannel
|
||||
|
||||
class OpenTokPlatformBridge(private var flutterEngine: FlutterEngine, private var mainActivity: MainActivity) {
|
||||
|
||||
private lateinit var channel: MethodChannel
|
||||
private lateinit var openTok: OpenTok
|
||||
|
||||
companion object {
|
||||
private const val CHANNEL = "OpenTok-Platform-Bridge"
|
||||
}
|
||||
|
||||
fun create(){
|
||||
openTok = OpenTok(mainActivity, flutterEngine)
|
||||
channel = MethodChannel(flutterEngine.dartExecutor.binaryMessenger, CHANNEL)
|
||||
channel.setMethodCallHandler { call: MethodCall, result: MethodChannel.Result ->
|
||||
when (call.method) {
|
||||
"initSession" -> {
|
||||
openTok.initSession(call, result)
|
||||
}
|
||||
"swapCamera" -> {
|
||||
openTok.swapCamera(call, result)
|
||||
}
|
||||
"toggleAudio" -> {
|
||||
openTok.toggleAudio(call, result)
|
||||
}
|
||||
"toggleVideo" -> {
|
||||
openTok.toggleVideo(call, result)
|
||||
}
|
||||
else -> {
|
||||
result.notImplemented()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -1 +1 @@
|
||||
e52eba3667a38bec777870899c15ae7d
|
||||
3f3d14a0ae775b56806906c2cb14a1f0
|
@ -0,0 +1,59 @@
|
||||
//
|
||||
// HMGPlatformBridge.swift
|
||||
// Runner
|
||||
//
|
||||
// Created by ZiKambrani on 14/12/2020.
|
||||
//
|
||||
|
||||
import UIKit
|
||||
import NetworkExtension
|
||||
import SystemConfiguration.CaptiveNetwork
|
||||
import OpenTok
|
||||
|
||||
|
||||
fileprivate var openTok:OpenTok?
|
||||
|
||||
class OpenTokPlatformBridge : NSObject{
|
||||
private var methodChannel:FlutterMethodChannel? = nil
|
||||
private var mainViewController:MainFlutterVC!
|
||||
private static var shared_:OpenTokPlatformBridge?
|
||||
|
||||
class func initialize(flutterViewController:MainFlutterVC, registrar:FlutterPluginRegistrar?){
|
||||
shared_ = OpenTokPlatformBridge()
|
||||
shared_?.mainViewController = flutterViewController
|
||||
|
||||
shared_?.openChannel()
|
||||
openTok = OpenTok(mainViewController: flutterViewController, registrar: registrar)
|
||||
}
|
||||
|
||||
func shared() -> OpenTokPlatformBridge{
|
||||
assert((OpenTokPlatformBridge.shared_ != nil), "OpenTokPlatformBridge is not initialized, call initialize(mainViewController:MainFlutterVC) function first.")
|
||||
return OpenTokPlatformBridge.shared_!
|
||||
}
|
||||
|
||||
private func openChannel(){
|
||||
methodChannel = FlutterMethodChannel(name: "OpenTok-Platform-Bridge", binaryMessenger: mainViewController.binaryMessenger)
|
||||
methodChannel?.setMethodCallHandler { (call, result) in
|
||||
print("Called function \(call.method)")
|
||||
|
||||
switch(call.method) {
|
||||
case "initSession":
|
||||
openTok?.initSession(call: call, result: result)
|
||||
|
||||
case "swapCamera":
|
||||
openTok?.swapCamera(call: call, result: result)
|
||||
|
||||
case "toggleAudio":
|
||||
openTok?.toggleAudio(call: call, result: result)
|
||||
|
||||
case "toggleVideo":
|
||||
openTok?.toggleVideo(call: call, result: result)
|
||||
|
||||
default:
|
||||
result(FlutterMethodNotImplemented)
|
||||
}
|
||||
|
||||
print("")
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,273 @@
|
||||
//
|
||||
// OpenTok.swift
|
||||
// Runner
|
||||
//
|
||||
// Created by Zohaib Iqbal Kambrani on 18/10/2021.
|
||||
//
|
||||
|
||||
import Foundation
|
||||
import OpenTok
|
||||
|
||||
enum SdkState: String {
|
||||
case loggedOut = "LOGGED_OUT"
|
||||
case loggedIn = "LOGGED_IN"
|
||||
case wait = "WAIT"
|
||||
case error = "ERROR"
|
||||
}
|
||||
|
||||
class OpenTok : NSObject{
|
||||
private var mainViewController:MainFlutterVC!
|
||||
private var registrar:FlutterPluginRegistrar?
|
||||
init(mainViewController:MainFlutterVC, registrar:FlutterPluginRegistrar?){
|
||||
self.mainViewController = mainViewController
|
||||
self.registrar = registrar
|
||||
|
||||
let factory = OpentokVideoFactory(messenger: registrar!.messenger())
|
||||
registrar?.register(factory, withId: "opentok-video-container")
|
||||
}
|
||||
|
||||
var otSession: OTSession?
|
||||
var vonageChannel: FlutterMethodChannel?
|
||||
|
||||
var subscriber: OTSubscriber?
|
||||
lazy var publisher: OTPublisher = {
|
||||
let settings = OTPublisherSettings()
|
||||
settings.name = UIDevice.current.name
|
||||
return OTPublisher(delegate: self, settings: settings)!
|
||||
}()
|
||||
|
||||
func initSession(call:FlutterMethodCall, result: @escaping FlutterResult){
|
||||
if let arguments = call.arguments as? [String: String],
|
||||
let apiKey = arguments["apiKey"],
|
||||
let sessionId = arguments["sessionId"],
|
||||
let token = arguments["token"]{
|
||||
|
||||
var error: OTError?
|
||||
defer {
|
||||
// todo
|
||||
}
|
||||
|
||||
notifyFlutter(state: SdkState.wait)
|
||||
otSession = OTSession(apiKey: apiKey, sessionId: sessionId, delegate: self)!
|
||||
otSession?.connect(withToken: token, error: &error)
|
||||
|
||||
result("")
|
||||
}else{
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
func swapCamera(call:FlutterMethodCall, result: @escaping FlutterResult) {
|
||||
if publisher.cameraPosition == .front {
|
||||
publisher.cameraPosition = .back
|
||||
} else {
|
||||
publisher.cameraPosition = .front
|
||||
}
|
||||
result("")
|
||||
}
|
||||
|
||||
func toggleAudio(call:FlutterMethodCall, result: @escaping FlutterResult) {
|
||||
if let arguments = call.arguments as? [String: Bool],
|
||||
let publishAudio = arguments["publishAudio"] {
|
||||
publisher.publishAudio = !publisher.publishAudio
|
||||
}
|
||||
result("")
|
||||
}
|
||||
|
||||
func toggleVideo(call:FlutterMethodCall, result: @escaping FlutterResult) {
|
||||
if let arguments = call.arguments as? [String: Bool],
|
||||
let publishVideo = arguments["publishVideo"] {
|
||||
publisher.publishVideo = !publisher.publishVideo
|
||||
}
|
||||
result("")
|
||||
}
|
||||
|
||||
|
||||
func notifyFlutter(state: SdkState) {
|
||||
vonageChannel?.invokeMethod("updateState", arguments: state.rawValue)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
extension OpenTok: OTSessionDelegate {
|
||||
func sessionDidConnect(_ sessionDelegate: OTSession) {
|
||||
print("The client connected to the session.")
|
||||
notifyFlutter(state: SdkState.loggedIn)
|
||||
|
||||
var error: OTError?
|
||||
defer {
|
||||
// todo
|
||||
}
|
||||
|
||||
self.otSession?.publish(self.publisher, error: &error)
|
||||
|
||||
if let pubView = self.publisher.view {
|
||||
pubView.frame = CGRect(x: 0, y: 0, width: 200, height: 300)
|
||||
|
||||
if OpentokVideoFactory.view == nil {
|
||||
OpentokVideoFactory.viewToAddPub = pubView
|
||||
} else {
|
||||
OpentokVideoFactory.view?.addPublisherView(pubView)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func sessionDidDisconnect(_ session: OTSession) {
|
||||
print("The client disconnected from the session.")
|
||||
notifyFlutter(state: SdkState.loggedOut)
|
||||
}
|
||||
|
||||
func session(_ session: OTSession, didFailWithError error: OTError) {
|
||||
print("The client failed to connect to the session: \(error).")
|
||||
}
|
||||
|
||||
func session(_ session: OTSession, streamCreated stream: OTStream) {
|
||||
print("A stream was created in the session.")
|
||||
var error: OTError?
|
||||
defer {
|
||||
// todo
|
||||
}
|
||||
subscriber = OTSubscriber(stream: stream, delegate: self)
|
||||
|
||||
session.subscribe(subscriber!, error: &error)
|
||||
}
|
||||
|
||||
func session(_ session: OTSession, streamDestroyed stream: OTStream) {
|
||||
print("A stream was destroyed in the session.")
|
||||
}
|
||||
}
|
||||
|
||||
extension OpenTok: OTPublisherDelegate {
|
||||
func publisher(_ publisher: OTPublisherKit, streamCreated stream: OTStream) {
|
||||
}
|
||||
|
||||
func publisher(_ publisher: OTPublisherKit, streamDestroyed stream: OTStream) {
|
||||
}
|
||||
|
||||
func publisher(_ publisher: OTPublisherKit, didFailWithError error: OTError) {
|
||||
print("Publisher failed: \(error.localizedDescription)")
|
||||
}
|
||||
}
|
||||
|
||||
extension OpenTok: OTSubscriberDelegate {
|
||||
func subscriberDidConnect(toStream subscriberKit: OTSubscriberKit) {
|
||||
print("Subscriber connected")
|
||||
|
||||
if let subView = self.subscriber?.view {
|
||||
subView.frame = CGRect(x: 0, y: 0, width: 200, height: 300)
|
||||
|
||||
if OpentokVideoFactory.view == nil {
|
||||
OpentokVideoFactory.viewToAddSub = subView
|
||||
} else {
|
||||
OpentokVideoFactory.view?.addSubscriberView(subView)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func subscriber(_ subscriber: OTSubscriberKit, didFailWithError error: OTError) {
|
||||
print("Subscriber failed: \(error.localizedDescription)")
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
class OpentokVideoFactory: NSObject, FlutterPlatformViewFactory {
|
||||
static var view: OpentokVideoPlatformView?
|
||||
|
||||
static var viewToAddSub: UIView?
|
||||
static var viewToAddPub: UIView?
|
||||
|
||||
static func getViewInstance(
|
||||
frame: CGRect,
|
||||
viewId: Int64,
|
||||
args: Any?,
|
||||
messenger: FlutterBinaryMessenger?
|
||||
) -> OpentokVideoPlatformView{
|
||||
if(view == nil) {
|
||||
view = OpentokVideoPlatformView()
|
||||
if viewToAddSub != nil {
|
||||
view?.addSubscriberView(viewToAddSub!)
|
||||
}
|
||||
if viewToAddPub != nil {
|
||||
view?.addPublisherView(viewToAddPub!)
|
||||
}
|
||||
}
|
||||
|
||||
return view!
|
||||
}
|
||||
|
||||
private var messenger: FlutterBinaryMessenger
|
||||
|
||||
init(messenger: FlutterBinaryMessenger) {
|
||||
self.messenger = messenger
|
||||
super.init()
|
||||
}
|
||||
|
||||
func create(
|
||||
withFrame frame: CGRect,
|
||||
viewIdentifier viewId: Int64,
|
||||
arguments args: Any?
|
||||
) -> FlutterPlatformView {
|
||||
return OpentokVideoFactory.getViewInstance(
|
||||
frame: frame,
|
||||
viewId: viewId,
|
||||
args: args,
|
||||
messenger: messenger)
|
||||
}
|
||||
}
|
||||
|
||||
class OpentokVideoPlatformView: NSObject, FlutterPlatformView {
|
||||
private let videoContainer: OpenTokVideoContainer
|
||||
|
||||
override init() {
|
||||
videoContainer = OpenTokVideoContainer()
|
||||
super.init()
|
||||
}
|
||||
|
||||
public func addSubscriberView(_ view: UIView) {
|
||||
videoContainer.addSubscriberView(view)
|
||||
}
|
||||
|
||||
public func addPublisherView(_ view: UIView) {
|
||||
videoContainer.addPublisherView(view)
|
||||
}
|
||||
|
||||
func view() -> UIView {
|
||||
return videoContainer
|
||||
}
|
||||
}
|
||||
|
||||
final class OpenTokVideoContainer: UIView {
|
||||
private let subscriberContainer = UIView()
|
||||
private let publisherContainer = UIView()
|
||||
|
||||
init() {
|
||||
super.init(frame: .zero)
|
||||
addSubview(subscriberContainer)
|
||||
addSubview(publisherContainer)
|
||||
}
|
||||
|
||||
|
||||
public func addSubscriberView(_ view: UIView) {
|
||||
subscriberContainer.addSubview(view)
|
||||
}
|
||||
|
||||
public func addPublisherView(_ view: UIView) {
|
||||
publisherContainer.addSubview(view)
|
||||
}
|
||||
|
||||
required init?(coder: NSCoder) {
|
||||
fatalError("init(coder:) has not been implemented")
|
||||
}
|
||||
|
||||
override func layoutSubviews() {
|
||||
super.layoutSubviews()
|
||||
let width = frame.width
|
||||
let height = frame.height
|
||||
|
||||
let videoWidth = width / 2
|
||||
subscriberContainer.frame = CGRect(x: 0, y: 0, width: videoWidth, height: height)
|
||||
publisherContainer.frame = CGRect(x: videoWidth, y: 0, width: videoWidth, height: height)
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue