を使用してiOSアプリを作成していUIPageViewController
ます。私のルートビューコントローラーはでViewController
、そのUIPageViewController
下に子VCとして追加しました。
最初に、私は自分ViewController
をページビューVCデリゲートとして作成しましたUIPageViewControllerDelegate
::
extension ViewController: UIPageViewControllerDelegate {
func pageViewController(_ pageViewController: UIPageViewController, didFinishAnimating finished: Bool, previousViewControllers: [UIViewController], transitionCompleted completed: Bool) {
// set the pageControl.currentPage to the index of the current viewController in pages
if let viewControllers = pageViewController.viewControllers as? [UIViewController] {
if let viewControllerIndex = self.detailPagedVC.pages.index(of: viewControllers[0]) {
self.detailPagedVC.pageControl.currentPage = viewControllerIndex
// if current page is a single person view controller, zoom to that person's face
if let singlePersonViewController = self.detailPagedVC.pages[viewControllerIndex] as? SinglePersonPageViewController {
// print("didFinishAnimating: \(viewControllerIndex)")
self.zoomableImageVC.zoomableImageView.zoom(to: self.identificationResults[viewControllerIndex].face.rect, with: Constants.contentSpanRatio, animated: true)
} else if let summaryPageViewController = self.detailPagedVC.pages[viewControllerIndex] as? SummaryPageViewController {
self.zoomableImageVC.zoomableImageView.zoom(to: self.zoomableImageVC.zoomableImageView.imageView.bounds, with: Constants.contentSpanRatio, animated: true)
} else {
print("gw: err: unkown type of page controller in paged view ")
}
}
}
}
}
この方法は問題なく機能します。委任関数を専用クラスに移動することを決定するまで:
class PeoplePageViewDelegate: NSObject, UIPageViewControllerDelegate{
private struct Constants {
// the ratio of the content (e..g face) taken inside the entire view
static let contentSpanRatio: CGFloat = 0.8
}
// store a reference to the object which will take the actual action
// action 1: zooming
weak var zoomingActionTaker: ZoomableImageView?
// action 2: paging
weak var pagingActionTaker: PeoplePageViewController?
// the delegator who relies on this object
unowned let delegator: PeoplePageViewController
init(delegator: PeoplePageViewController) {
self.delegator = delegator
// wire back to delegator
// self.pagingActionTaker = delegator // gw: mind the nuance difference, you chain a weak ref on a unowned ref, what can go wrong?
super.init()
}
func pageViewController(_ pageViewController: UIPageViewController, didFinishAnimating finished: Bool, previousViewControllers: [UIViewController], transitionCompleted completed: Bool) {
print("gw:0")
// set the pageControl.currentPage to the index of the current viewController in pages
if let viewControllers = pageViewController.viewControllers as? [UIViewController] {
print("gw:1")
if let viewControllerIndex = self.delegator.pages.index(of: viewControllers[0]) {
print("gw:2")
self.pagingActionTaker?.pageControl.currentPage = viewControllerIndex
// if current page is a single person view controller, zoom to that person's face
if let singlePersonViewController = self.delegator.pages[viewControllerIndex] as? SinglePersonPageViewController {
print("gw:3")
self.zoomingActionTaker?.zoom(to: singlePersonViewController.identification.face.rect, with: Constants.contentSpanRatio, animated: true)
} else if let summaryPageViewController = self.delegator.pages[viewControllerIndex] as? SummaryPageViewController,
let entireImageBounds = self.zoomingActionTaker?.imageView.bounds {
print("gw:4")
self.zoomingActionTaker?.zoom(to: entireImageBounds, with: Constants.contentSpanRatio, animated: true)
} else {
print("gw: err: unkown type of page controller in paged view ")
}
}
}
}
}
これにより問題が発生します。pageViewController(_: didFinishAnimating:previousViewControllers:transitionCompleted)
デリゲートをこの新しいクラスのインスタンスに設定したにもかかわらず、関数がまったく呼び出されません。
私が考えることができる唯一のことは、新しいデリゲートオブジェクトはもうVCではないということです(以前は私のメインVCでした)。だから私はこの行動の変化がVC階層に関連していると思いますか?
私は何か見落としてますか?
(この回答は、OOPerのコメントに触発されています)。
原因は、最初weak
はデリゲートオブジェクトへの参照のみを保持しているためであることが判明しました。このような:
前:
class ViewController: UIViewController {
init() {
self.pageViewController.delegate = PeoplePageViewDelegate(….)
}
}
ここでの問題は、実行ポイントがinit
メソッドスコープを通過するPeoplePageViewDelegate
と、参照カウントがゼロであるためにガベージコレクションが行われることです。
修正方法:
のフィールドとして強力な参照を追加したViewController
ので、デリゲートオブジェクトが保持されます。
後:
class ViewController: UIViewController {
var myStrongReference: PeoplePageViewDelegate?
init() {
self.myStrongReference = PeoplePageViewDelegate(….)
self.pageViewController.delegate = self.myStrongReference
}
}
この記事はインターネットから収集されたものであり、転載の際にはソースを示してください。
侵害の場合は、連絡してください[email protected]
コメントを追加