// // NCViewerVideoToolBar.swift // Nextcloud // // Created by Marino Faggiana on 01/07/21. // Copyright © 2021 Marino Faggiana. All rights reserved. // // Author Marino Faggiana // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // import Foundation import NCCommunication class NCViewerVideoToolBar: UIView { @IBOutlet weak var playButton: UIButton! @IBOutlet weak var muteButton: UIButton! @IBOutlet weak var forwardButton: UIButton! @IBOutlet weak var backButton: UIButton! @IBOutlet weak var playbackSlider: UISlider! @IBOutlet weak var labelOverallDuration: UILabel! @IBOutlet weak var labelCurrentTime: UILabel! enum sliderEventType { case began case ended case moved } var player: AVPlayer? var metadata: tableMetadata? private var wasInPlay: Bool = false private var playbackSliderEvent: sliderEventType = .ended private let seekDuration: Float64 = 15 // MARK: - View Life Cycle override func awakeFromNib() { super.awakeFromNib() // for disable gesture of UIPageViewController let panRecognizer = UIPanGestureRecognizer(target: self, action: nil) addGestureRecognizer(panRecognizer) let singleTapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(didSingleTapWith(gestureRecognizer:))) addGestureRecognizer(singleTapGestureRecognizer) let blurEffect = UIBlurEffect(style: .dark) let blurEffectView = UIVisualEffectView(effect: blurEffect) self.layer.cornerRadius = 15 self.layer.masksToBounds = true blurEffectView.frame = self.bounds blurEffectView.autoresizingMask = [.flexibleWidth, .flexibleHeight] self.insertSubview(blurEffectView, at:0) playbackSlider.value = 0 playbackSlider.minimumValue = 0 playbackSlider.maximumValue = 0 playbackSlider.isContinuous = true playbackSlider.tintColor = .lightGray labelCurrentTime.text = NCUtility.shared.stringFromTimeInterval(interval: 0) labelCurrentTime.textColor = .lightGray labelOverallDuration.text = NCUtility.shared.stringFromTimeInterval(interval: 0) labelOverallDuration.textColor = .lightGray setToolBarImage() } func setBarPlayer(player: AVPlayer?, metadata: tableMetadata) { self.player = player self.metadata = metadata let duration: CMTime = (player?.currentItem?.asset.duration)! let durationSeconds: Float64 = CMTimeGetSeconds(duration) playbackSlider.value = 0 playbackSlider.minimumValue = 0 playbackSlider.maximumValue = Float(durationSeconds) playbackSlider.addTarget(self, action: #selector(onSliderValChanged(slider:event:)), for: .valueChanged) labelCurrentTime.text = NCUtility.shared.stringFromTimeInterval(interval: 0) labelOverallDuration.text = "-" + NCUtility.shared.stringFromTimeInterval(interval: durationSeconds) NCManageDatabase.shared.addVideoTime(metadata: metadata, time: nil, durationSeconds: durationSeconds) player?.addPeriodicTimeObserver(forInterval: CMTimeMakeWithSeconds(1, preferredTimescale: 1), queue: .main, using: { (CMTime) in if self.player?.currentItem?.status == .readyToPlay { if self.isHidden == false { self.updateOutlet() } } }) setToolBarImage() showToolBar() } @objc public func hideToolBar() { updateOutlet() UIView.animate(withDuration: 0.3, animations: { self.alpha = 0 }, completion: { (value: Bool) in self.isHidden = true }) } @objc public func showToolBar() { updateOutlet() UIView.animate(withDuration: 0.3, animations: { self.alpha = 1 }, completion: { (value: Bool) in self.isHidden = false }) } public func setToolBarImage() { var namedPlay = "play.fill" if player?.rate == 1 { namedPlay = "pause.fill"} backButton.setImage(NCUtility.shared.loadImage(named: "gobackward.15", color: .white), for: .normal) if #available(iOS 13.0, *) { playButton.setImage(NCUtility.shared.loadImage(named: namedPlay, color: .white, symbolConfiguration: UIImage.SymbolConfiguration(pointSize: 30)), for: .normal) } else { playButton.setImage(NCUtility.shared.loadImage(named: namedPlay, color: .white), for: .normal) } forwardButton.setImage(NCUtility.shared.loadImage(named: "goforward.15", color: .white), for: .normal) if CCUtility.getAudioMute() { muteButton.setImage(NCUtility.shared.loadImage(named: "audioOff", color: .white), for: .normal) } else { muteButton.setImage(NCUtility.shared.loadImage(named: "audioOn", color: .white), for: .normal) } } private func updateOutlet() { if let duration = player?.currentItem?.asset.duration, let currentTime = player?.currentTime() { let durationSeconds: Float64 = CMTimeGetSeconds(duration) let currentSeconds: Float64 = CMTimeGetSeconds(currentTime) self.playbackSlider.value = Float(currentSeconds) self.labelCurrentTime.text = NCUtility.shared.stringFromTimeInterval(interval: currentSeconds) self.labelOverallDuration.text = "-" + NCUtility.shared.stringFromTimeInterval(interval: durationSeconds - currentSeconds) } } //MARK: - Event / Gesture @objc func onSliderValChanged(slider: UISlider, event: UIEvent) { if let touchEvent = event.allTouches?.first { let seconds: Int64 = Int64(self.playbackSlider.value) let targetTime: CMTime = CMTimeMake(value: seconds, timescale: 1) switch touchEvent.phase { case .began: wasInPlay = player?.rate == 1 ? true : false self.player?.pause() playbackSliderEvent = .began case .moved: self.player?.seek(to: targetTime) playbackSliderEvent = .moved case .ended: self.player?.seek(to: targetTime) if let metadata = self.metadata { NCManageDatabase.shared.addVideoTime(metadata: metadata, time: targetTime, durationSeconds: nil) } if wasInPlay { self.player?.play() } playbackSliderEvent = .ended default: break } } } @objc func didSingleTapWith(gestureRecognizer: UITapGestureRecognizer) { hideToolBar() } //MARK: - Action @IBAction func buttonTouchInside(_ sender: UIButton) { hideToolBar() } @IBAction func playerPause(_ sender: Any) { if player?.timeControlStatus == .playing { player?.pause() } else if player?.timeControlStatus == .paused { player?.play() } } @IBAction func setMute(_ sender: Any) { let mute = CCUtility.getAudioMute() CCUtility.setAudioMute(!mute) player?.isMuted = !mute setToolBarImage() } @IBAction func forwardButtonSec(_ sender: Any) { guard let player = self.player else { return } guard let duration = player.currentItem?.duration else { return } guard let metadata = self.metadata else { return } let playerCurrentTime = CMTimeGetSeconds(player.currentTime()) let newTime = playerCurrentTime + seekDuration if newTime < CMTimeGetSeconds(duration) { let targetTime: CMTime = CMTimeMake(value: Int64(newTime * 1000 as Float64), timescale: 1000) self.player?.seek(to: targetTime) NCManageDatabase.shared.addVideoTime(metadata: metadata, time: targetTime, durationSeconds: nil) } } @IBAction func backButtonSec(_ sender: Any) { guard let player = self.player else { return } guard let metadata = self.metadata else { return } let playerCurrenTime = CMTimeGetSeconds(player.currentTime()) var newTime = playerCurrenTime - seekDuration if newTime < 0 { newTime = 0 } let targetTime: CMTime = CMTimeMake(value: Int64(newTime * 1000 as Float64), timescale: 1000) self.player?.seek(to: targetTime) NCManageDatabase.shared.addVideoTime(metadata: metadata, time: targetTime, durationSeconds: nil) } }