我有一个应用程序(支持界面方向 - 仅限纵向)具有下一个层次结构的模态呈现 View Controller :
A -> B -> AVP
A 是位于标签栏 Controller 中的 View Controller ,标签栏 Controller 又是窗口的根。
B 是一个相当简单的 View Controller ,带有按钮、图像和标签,但显示为弹出窗口:
// ... presentation method in A
let B = // create B
B.modalPresentationStyle = .popover
B.preferredContentSize = CGSize(width: 300, height: 400)
B.isModalInPopover = true
if let BPopover = B.popoverPresentationController {
BPopover.delegate = self
BPopover.permittedArrowDirections = []
let window = // grab current window
BPopover.sourceView = window
BPopover.sourceRect = window.bounds
BPopover.passthroughViews = nil
}
self.tabBarController?.present(B, animated: true, completion: nil)
AVP 是 B 提供的 AVPlayerViewController:
// This method is in B.
@IBAction func playVideoButtonPressed(_ sender: Any) {
if let videoURL = self.videoURL {
let videoPlayer = AVPlayer(url: videoURL)
let videoVC = AVPlayerViewController()
videoVC.player = videoPlayer
self.present(videoVC, animated: true, completion: nil)
}
}
在 iOS 10.0 上,如果我执行后续步骤,我会遇到问题:
当我回来时,我的 View Controller B 被弄乱了——移到了窗口的顶部并且它的尺寸变小了(内部也被弄乱了,但我猜内部由于我的自动布局约束而被弄乱了)。
这在 iOS 11 上似乎没有发生。
有什么办法可以解决吗?
编辑:按要求截图(出于隐私原因隐藏了标签栏):
附加信息:
我还拦截了一个委托(delegate)回调以获取更多信息:
func popoverPresentationController(_ popoverPresentationController: UIPopoverPresentationController,
willRepositionPopoverTo rect: UnsafeMutablePointer<CGRect>,
in view: AutoreleasingUnsafeMutablePointer<UIView>) {
print("willRepositionPopoverTo")
print(popoverPresentationController)
print(rect.pointee)
print(view.pointee)
}
打印 view 大小为 (w: 568; h: 320) 所以当我旋转 AVP Controller 时它似乎改变了我的应用程序的窗口方向,这导致调整大小我的弹出窗口。虽然它不会尝试重新调整大小 :( 在我关闭 AVP 之后。
最佳答案
我已经成功重现了您的问题,所以您可以高枕无忧,因为您知道这不仅仅是您的问题。我还花了相当多的时间尝试使用各种“黑客”来修复 iOS 10 的行为。我最初并不成功。此外,似乎尽管 iOS 11 解决了弹出窗口的定位问题,但它也引入了标签栏大小的错误(图片)。所以我的解决方案也需要解决这个问题。
在重新审视这个问题后,我重新考虑了一个我最初排除的解决方案。事实证明,AVPlayerViewController 只会影响其父级 UIWindow 实例的方向。使用额外的 UIWindow 并不是 iOS 中经常使用的解决方案,但在这里效果很好。
解决方案是创建一个清晰的 UIWindow 并带有填充 rootViewController,其唯一目的是在第二次出现时在没有动画的情况下关闭自身.此解决方案在 iOS 10 和 iOS 11 上同样适用。它不会改变用户体验(除了修复错误)。
步骤:
UIWindow 并使其背景清晰ShimVC 的一个实例>AVPlayerViewController(这样你就可以得到动画,就像你不这样做一样)这是一个 gif 动图,展示了它的效果:
class FirstViewController: UIViewController, UIViewControllerTransitioningDelegate {
@IBAction func popover(_ sender: UIButton) {
let b = UIStoryboard(name: "Main", bundle: nil).instantiateViewController(withIdentifier: "B") as! B
b.modalPresentationStyle = .popover
b.preferredContentSize = CGSize(width: 300, height: 400)
b.isModalInPopover = true
if let ppc = b.popoverPresentationController {
ppc.delegate = self
ppc.permittedArrowDirections = []
let window = view.window!
ppc.sourceView = window
ppc.sourceRect = window.frame
ppc.passthroughViews = nil
}
present(b, animated: true, completion: nil)
}
}
extension FirstViewController: UIPopoverPresentationControllerDelegate {
func adaptivePresentationStyle(for controller: UIPresentationController) -> UIModalPresentationStyle {
return .none
}
}
class ShimVC: UIViewController {
var appearances: Int = 0
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
if appearances > 0 {
// When the rootViewController of a window dismisses, that window
// gets removed from the view hiearchy and discarded, making the
// the previous window key automatically
dismiss(animated: true)
}
appearances += 1
}
}
class B: UIViewController {
let videoURL: URL? = Bundle.main.url(forResource: "ImAfraidWeNeedToUseMath", withExtension: "m4v")
@IBAction func playVideo(_ sender: UIButton) {
if let videoURL = self.videoURL {
let videoPlayer = AVPlayer(url: videoURL)
let videoVC = AVPlayerViewController()
videoVC.player = videoPlayer
let vc = ShimVC(nibName: nil, bundle: nil)
let videoWindow = UIWindow()
videoWindow.backgroundColor = .clear
videoWindow.rootViewController = vc
videoWindow.makeKeyAndVisible()
// Present the `AVPlayerViewController` from the root
// of the window
vc.present(videoVC, animated: true)
}
}
@IBAction func done(_ sender: UIButton) {
dismiss(animated: true)
}
}
我将提出一个解决方法。这是我之前在某些应用程序中看到的行为。基本上,不是从弹出窗口中显示视频播放器,而是使用 AVC 切换出 B。可能还有其他解决方案需要更多偏离现有 UIKit 功能。 (例如可能实现您自己的呈现 Controller 来实现弹出窗口或将弹出窗口实现为 subview Controller ,以便您可以直接从 A 呈现 AVC。)
在下面的解决方案中,A -> B 然后当用户按下播放器 B 时通知 A 它需要呈现 AVC。然后你得到 A -> AVC。当 AVC 被关闭时,我重新呈现 B,因此您返回到 A -> B。总体而言,在 iOS 10 或 iOS 11 上都没有 UI 问题,但您的用户将需要多等几分之一秒。如果时间紧迫,您可以禁用(或尝试缩短?)动画,但总的来说它感觉非常流畅和自然。
此外,我建议提交有关标签栏大小问题的雷达。
这是我的解决方案:
class A: UIViewController, UIViewControllerTransitioningDelegate {
var popover: B?
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
if let popover = popover {
if let ppc = popover.popoverPresentationController {
ppc.delegate = self
ppc.permittedArrowDirections = []
let window = view.window!
ppc.sourceView = window
ppc.sourceRect = window.frame
ppc.passthroughViews = nil
}
present(popover, animated: true, completion: nil)
}
}
@IBAction func popover(_ sender: UIButton) {
let b = UIStoryboard(name: "Main", bundle: nil).instantiateViewController(withIdentifier: "B") as! B
b.modalPresentationStyle = .popover
b.preferredContentSize = CGSize(width: 300, height: 400)
b.isModalInPopover = true
b.delegate = self
if let ppc = b.popoverPresentationController {
ppc.delegate = self
ppc.permittedArrowDirections = []
let window = view.window!
ppc.sourceView = window
ppc.sourceRect = window.frame
ppc.passthroughViews = nil
}
self.popover = b
present(b, animated: true, completion: nil)
}
}
extension A: UIPopoverPresentationControllerDelegate {
func adaptivePresentationStyle(for controller: UIPresentationController) -> UIModalPresentationStyle {
return .none
}
}
extension A: BDelegate {
func dismissB() {
popover?.dismiss(animated: true)
popover = nil
}
func showAVPlayerViewController(_ vc: AVPlayerViewController) {
popover?.dismiss(animated: true) {
// Dispatch async allows it to come up in landscape if the phone is already rotated
DispatchQueue.main.async {
self.present(vc, animated: true)
}
}
}
}
protocol BDelegate: class {
func showAVPlayerViewController(_ vc: AVPlayerViewController)
func dismissB()
}
class B: UIViewController {
weak var delegate: BDelegate?
let videoURL: URL? = Bundle.main.url(forResource: "ImAfraidWeNeedToUseMath", withExtension: "m4v")
@IBAction func playVideo(_ sender: UIButton) {
if let videoURL = self.videoURL {
let videoPlayer = AVPlayer(url: videoURL)
let videoVC = AVPlayerViewController()
videoVC.player = videoPlayer
delegate?.showAVPlayerViewController(videoVC)
}
}
@IBAction func done(_ sender: UIButton) {
delegate?.dismissB()
}
}
关于ios - AVPlayerViewController 在 iOS10 的弹出窗口中弄乱了底层模态视图 Controller ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48667855/
我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>
刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr
我知道您通常应该在Rails中使用新建/创建和编辑/更新之间的链接,但我有一个情况需要其他东西。无论如何我可以实现同样的连接吗?我有一个模型表单,我希望它发布数据(类似于新View如何发布到创建操作)。这是我的表格prohibitedthisjobfrombeingsaved: 最佳答案 使用:url选项。=form_for@job,:url=>company_path,:html=>{:method=>:post/:put} 关于ruby-on-rails-rails:Howtomak
这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下
我有一个rubyonrails应用程序。我按照facebook的说明添加了一个像素。但是,要跟踪转化,Facebook要求您将页面置于达到预期结果时出现的转化中。即,如果我想显示客户已注册,我会将您注册后转到的页面作为成功对象进行跟踪。我的问题是,当客户注册时,在我的应用程序中没有登陆页面。该应用程序将用户带回主页。它在主页上显示了一条消息,所以我想看看是否有一种方法可以跟踪来自Controller操作而不是实际页面的转化。我需要计数的Action没有页面,它们是ControllerAction。是否有任何人都知道的关于如何执行此操作的gem、文档或最佳实践?这是进入布局文件的像素
我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的
1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里
我有一个Controller,我想为这个Controller创建一个助手,我可以在不包含它的情况下使用它。我尝试像这样创建一个与Controller同名的助手classCars::EnginesController我创建的助手是moduleCars::EnginesHelperdefcheck_fuellogger.debug("chekingfuel")endend我得到的错误是undefinedlocalvariableormethod`check_fuel'for#有没有我遗漏的约定? 最佳答案 如果你真的想在Controll
我有一个名为Post的类,我需要能够适应以下场景:如果用户选择了一个类别,则只显示该类别的帖子如果用户选择了一种类型,则只显示该类型的帖子如果用户选择了一个类别和类型,则只显示该类别中该类型的帖子如果用户没有选择任何内容,则显示所有帖子我想知道我的Controller是否不可避免地会因大量条件语句而显得粗糙...这是我解决此问题的错误方法-有谁知道我如何才能做到这一点?classPostsController 最佳答案 您最好遵循“胖模型,瘦Controller”的惯例,这意味着您应该将这种逻辑放在模型本身中。Post类应该能够报告
让多条路线去同一条路的最优雅的方式是什么ControllerAction?我有:get'dashboard',to:'dashboard#index'get'dashboard/pending',to:'dashboard#index'get'dashboard/live',to:'dashboard#index'get'dashboard/sold',to:'dashboard#index'这很丑陋。有什么“更优雅”的建议吗?一个类轮的奖励积分。 最佳答案 为什么不只有一个路由和一个Controller操作,并根据传递给它的参数来