2022-02-17 01:53:11 +05:30
|
|
|
import GLKit
|
2024-08-27 02:29:24 +05:30
|
|
|
import Libmpv
|
2022-03-27 17:12:20 +05:30
|
|
|
import Logging
|
2023-09-23 20:12:46 +05:30
|
|
|
import OpenGLES
|
2022-02-17 01:53:11 +05:30
|
|
|
|
|
|
|
final class MPVOGLView: GLKView {
|
2022-03-27 17:12:20 +05:30
|
|
|
private var logger = Logger(label: "stream.yattee.mpv.oglview")
|
2022-02-17 01:53:11 +05:30
|
|
|
private var defaultFBO: GLint?
|
2024-09-08 01:52:09 +05:30
|
|
|
private var displayLink: CADisplayLink?
|
2022-02-17 01:53:11 +05:30
|
|
|
|
|
|
|
var mpvGL: UnsafeMutableRawPointer?
|
2024-09-08 01:52:09 +05:30
|
|
|
var queue = DispatchQueue(label: "stream.yattee.opengl", qos: .userInteractive)
|
2022-02-17 01:53:11 +05:30
|
|
|
var needsDrawing = true
|
2024-09-14 16:00:33 +05:30
|
|
|
private var dirtyRegion: CGRect?
|
2022-02-17 01:53:11 +05:30
|
|
|
|
|
|
|
override init(frame: CGRect) {
|
2022-06-18 18:09:49 +05:30
|
|
|
guard let context = EAGLContext(api: .openGLES2) else {
|
2022-02-17 01:53:11 +05:30
|
|
|
print("Failed to initialize OpenGLES 2.0 context")
|
|
|
|
exit(1)
|
|
|
|
}
|
|
|
|
|
2022-03-27 17:12:20 +05:30
|
|
|
logger.info("frame size: \(frame.width) x \(frame.height)")
|
|
|
|
|
2022-02-17 01:53:11 +05:30
|
|
|
super.init(frame: frame, context: context)
|
|
|
|
|
2022-06-18 18:09:49 +05:30
|
|
|
self.context = context
|
|
|
|
bindDrawable()
|
2022-02-17 01:53:11 +05:30
|
|
|
|
|
|
|
defaultFBO = -1
|
2022-06-18 18:09:49 +05:30
|
|
|
isOpaque = true
|
|
|
|
enableSetNeedsDisplay = false
|
2022-02-17 01:53:11 +05:30
|
|
|
|
|
|
|
fillBlack()
|
2024-09-08 01:52:09 +05:30
|
|
|
setupDisplayLink()
|
|
|
|
setupNotifications()
|
|
|
|
}
|
|
|
|
|
|
|
|
required init?(coder aDecoder: NSCoder) {
|
|
|
|
super.init(coder: aDecoder)
|
|
|
|
setupDisplayLink()
|
|
|
|
setupNotifications()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func setupDisplayLink() {
|
|
|
|
displayLink = CADisplayLink(target: self, selector: #selector(updateFrame))
|
|
|
|
displayLink?.add(to: .main, forMode: .common)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set up observers to detect display changes and custom refresh rate updates.
|
|
|
|
private func setupNotifications() {
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(updateDisplayLinkFromNotification(_:)), name: .updateDisplayLinkFrameRate, object: nil)
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(screenDidChange), name: UIScreen.didConnectNotification, object: nil)
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(screenDidChange), name: UIScreen.didDisconnectNotification, object: nil)
|
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(screenDidChange), name: UIScreen.modeDidChangeNotification, object: nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc private func screenDidChange(_: Notification) {
|
|
|
|
// Update the display link refresh rate when the screen configuration changes
|
|
|
|
updateDisplayLinkFrameRate()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the display link frame rate from the notification.
|
|
|
|
@objc private func updateDisplayLinkFromNotification(_ notification: Notification) {
|
|
|
|
guard let userInfo = notification.userInfo,
|
|
|
|
let refreshRate = userInfo["refreshRate"] as? Int else { return }
|
|
|
|
displayLink?.preferredFramesPerSecond = refreshRate
|
|
|
|
logger.info("Updated CADisplayLink frame rate to: \(refreshRate) from backend notification.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the display link's preferred frame rate based on the current screen refresh rate.
|
|
|
|
private func updateDisplayLinkFrameRate() {
|
|
|
|
guard let displayLink else { return }
|
|
|
|
let refreshRate = getScreenRefreshRate()
|
|
|
|
displayLink.preferredFramesPerSecond = refreshRate
|
|
|
|
logger.info("Updated CADisplayLink preferred frames per second to: \(refreshRate)")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Retrieve the screen's current refresh rate dynamically.
|
|
|
|
private func getScreenRefreshRate() -> Int {
|
|
|
|
// Use the main screen's maximumFramesPerSecond property
|
|
|
|
let refreshRate = UIScreen.main.maximumFramesPerSecond
|
|
|
|
logger.info("Screen refresh rate: \(refreshRate) Hz")
|
|
|
|
return refreshRate
|
|
|
|
}
|
|
|
|
|
|
|
|
@objc private func updateFrame() {
|
|
|
|
// Trigger the drawing process if needed
|
|
|
|
if needsDrawing {
|
2024-09-14 16:00:33 +05:30
|
|
|
markRegionAsDirty(bounds)
|
2024-09-08 01:52:09 +05:30
|
|
|
setNeedsDisplay()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
deinit {
|
|
|
|
// Invalidate the display link and remove observers to avoid memory leaks
|
|
|
|
displayLink?.invalidate()
|
|
|
|
NotificationCenter.default.removeObserver(self)
|
2022-02-17 01:53:11 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
func fillBlack() {
|
|
|
|
glClearColor(0, 0, 0, 0)
|
|
|
|
glClear(UInt32(GL_COLOR_BUFFER_BIT))
|
|
|
|
}
|
|
|
|
|
2024-09-14 16:00:33 +05:30
|
|
|
// Function to set a dirty region when a part of the screen changes
|
|
|
|
func markRegionAsDirty(_ region: CGRect) {
|
|
|
|
if dirtyRegion == nil {
|
|
|
|
dirtyRegion = region
|
|
|
|
} else {
|
|
|
|
// Expand the dirty region to include the new region
|
|
|
|
dirtyRegion = dirtyRegion!.union(region)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Logic to decide if only part of the screen needs updating
|
|
|
|
private func needsPartialUpdate() -> Bool {
|
|
|
|
// Check if there is a defined dirty region that needs updating
|
|
|
|
if let dirtyRegion, !dirtyRegion.isEmpty {
|
|
|
|
// Set up glScissor based on dirtyRegion coordinates
|
|
|
|
glScissor(GLint(dirtyRegion.origin.x), GLint(dirtyRegion.origin.y), GLsizei(dirtyRegion.width), GLsizei(dirtyRegion.height))
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Call this function when you know the entire screen needs updating
|
|
|
|
private func clearDirtyRegion() {
|
|
|
|
dirtyRegion = nil
|
|
|
|
}
|
|
|
|
|
2022-03-27 17:12:20 +05:30
|
|
|
override func draw(_: CGRect) {
|
2024-09-08 01:52:09 +05:30
|
|
|
guard needsDrawing, let mpvGL else { return }
|
2022-06-16 05:33:15 +05:30
|
|
|
|
2024-09-14 16:00:33 +05:30
|
|
|
// Ensure the correct context is set
|
|
|
|
guard EAGLContext.setCurrent(context) else {
|
|
|
|
logger.error("Failed to set current OpenGL context.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-09-08 01:52:09 +05:30
|
|
|
// Bind the default framebuffer
|
2022-02-17 01:53:11 +05:30
|
|
|
glGetIntegerv(UInt32(GL_FRAMEBUFFER_BINDING), &defaultFBO!)
|
|
|
|
|
2024-09-14 16:00:33 +05:30
|
|
|
// Ensure the framebuffer is valid
|
|
|
|
guard defaultFBO != nil && defaultFBO! != 0 else {
|
|
|
|
logger.error("Invalid framebuffer ID.")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-09-08 01:52:09 +05:30
|
|
|
// Get the current viewport dimensions
|
2022-03-27 17:12:20 +05:30
|
|
|
var dims: [GLint] = [0, 0, 0, 0]
|
|
|
|
glGetIntegerv(GLenum(GL_VIEWPORT), &dims)
|
|
|
|
|
2024-09-14 16:00:33 +05:30
|
|
|
// Check if we need partial updates
|
|
|
|
if needsPartialUpdate() {
|
|
|
|
logger.info("Performing partial update with scissor test.")
|
|
|
|
glEnable(GLenum(GL_SCISSOR_TEST))
|
|
|
|
}
|
|
|
|
|
2024-09-08 01:52:09 +05:30
|
|
|
// Set up the OpenGL FBO data
|
2022-06-16 05:33:15 +05:30
|
|
|
var data = mpv_opengl_fbo(
|
|
|
|
fbo: Int32(defaultFBO!),
|
|
|
|
w: Int32(dims[2]),
|
|
|
|
h: Int32(dims[3]),
|
|
|
|
internal_format: 0
|
|
|
|
)
|
2024-09-08 01:52:09 +05:30
|
|
|
|
|
|
|
// Flip Y coordinate for proper rendering
|
2022-06-16 05:33:15 +05:30
|
|
|
var flip: CInt = 1
|
2024-09-08 01:52:09 +05:30
|
|
|
|
|
|
|
// Render with the provided OpenGL FBO parameters
|
|
|
|
withUnsafeMutablePointer(to: &flip) { flipPtr in
|
|
|
|
withUnsafeMutablePointer(to: &data) { dataPtr in
|
2022-06-16 05:33:15 +05:30
|
|
|
var params = [
|
2024-09-08 01:52:09 +05:30
|
|
|
mpv_render_param(type: MPV_RENDER_PARAM_OPENGL_FBO, data: dataPtr),
|
|
|
|
mpv_render_param(type: MPV_RENDER_PARAM_FLIP_Y, data: flipPtr),
|
2022-06-16 05:33:15 +05:30
|
|
|
mpv_render_param()
|
|
|
|
]
|
2024-09-14 16:00:33 +05:30
|
|
|
// Call the render function and check for errors
|
|
|
|
let result = mpv_render_context_render(OpaquePointer(mpvGL), ¶ms)
|
|
|
|
if result < 0 {
|
|
|
|
logger.error("mpv_render_context_render() failed with error code: \(result)")
|
|
|
|
} else {
|
|
|
|
logger.info("mpv_render_context_render() called successfully.")
|
|
|
|
}
|
2022-02-17 01:53:11 +05:30
|
|
|
}
|
|
|
|
}
|
2024-09-14 16:00:33 +05:30
|
|
|
|
|
|
|
// Disable the scissor test after rendering if it was enabled
|
|
|
|
if needsPartialUpdate() {
|
|
|
|
glDisable(GLenum(GL_SCISSOR_TEST))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear dirty region after drawing
|
|
|
|
clearDirtyRegion()
|
2022-02-17 01:53:11 +05:30
|
|
|
}
|
2024-09-08 01:52:09 +05:30
|
|
|
}
|
2022-02-17 01:53:11 +05:30
|
|
|
|
2024-09-08 01:52:09 +05:30
|
|
|
extension Notification.Name {
|
|
|
|
static let updateDisplayLinkFrameRate = Notification.Name("updateDisplayLinkFrameRate")
|
2022-02-17 01:53:11 +05:30
|
|
|
}
|