gpt4 book ai didi

iphone - 如何在 UINavigationBar 下面添加一个 View

转载 作者:塔克拉玛干 更新时间:2023-11-02 09:30:39 25 4
gpt4 key购买 nike

好吧,在深入 iOS 私有(private)类的深渊和追求最终可重用代码之间只有一线之隔。

我想在 UINavigationBar 下方有一个大约 40 像素高的任意 UIView。现在,通常这可以扔到封闭的 View 中,但我希望此 UIView 留在 UINavigationBar 中。因此,当用户向下钻取到不同的 View 时,导航栏下方的 UIView 将保持存在。

我试图构建一个容器 UIViewController,它将作为 UINavigationController 中的 RootViewController 放置,但这没有帮助,因为当用户按下一个新 View 时,我的容器 View 滑出,并且我的持久性酒吧也随之而来。

如果我将 UINavigationController 放在 UIViewController 中,那么我可以覆盖我想要的 UIView 栏,但是内部 View 似乎仍然没有正确更改它们的框架,以及当 View 被插入或弹出时Navigation Stack, View 的动画效果很差(更改它们的 Y 位置以及移出屏幕,因为 UINavigationController 不知道它们在屏幕下方的新 Y 位置。)

因此,我研究了 UINavigationController 的子类化。我知道,它说“此类不用于子类化。”,但总有异常(exception)。这个子类将简单地在导航栏下方添加持久性 UIView,调整内容部分(正常 View 将出现的位置)的大小,一切都会正常进行。

但是在尝试了 ObjC 运行时类之后,我找不到神奇的变量来调整它以使其工作(UINavigationController 的_containerView 就是这样一个变量)。

执行此操作的最佳方法是什么?有没有比复制和粘贴代码更好的方法来将此 Bar 放入将显示它的所有 ViewController 中?

最佳答案

我猜您希望在导航栏下方放置一个上传进度条或类似的东西,无论您转到哪个 View ,它都会与您保持一致。几周前我刚刚实现了这个,但有一个额外的警告 - 我们正在使用 IIViewDeck,因此我们还必须能够处理整个 UINavigationController 的变化。

我所做的是将 View 添加为导航栏的 subview 。我的导航栏和导航 Controller 都是子类(并且应用程序已顺利提交到 App Store)。起初我这样做是因为我需要在没有 UIAppearance 帮助的情况下自定义导航栏,但现在它似乎是一个不断给予的礼物,因为它为我提供了相当多的任务灵 active ,例如这个。

  1. 创建自定义的 UINavigationController 和 UINavigationBar
  2. 如果您需要处理触摸,则必须实现 hitTest:withEvent - 我也在下面包含了该代码。
  3. 如果您有一个可以更改 UINavigationController View 的应用(例如标签栏应用,或者有一个侧边菜单可以让您更改位于中心的 UINavigationController) ,我实现了 KVO 来监视中心 View Controller ,以便进度 View 可以“跟随”它。代码也在下面。

这就是我创建自定义 UINavigationController 和 UINavigationBar 的方式:

+ (ZNavigationController *)customizedNavigationController
{
//This is just a regular UINavigationBar subclass - doesn't have to have any code but I personally override the pop/push methods to call my own flavor of viewWill/Did/Appear/Disappear methods - iOS has always been a bit finicky about what gets called when
ZNavigationController *navController = [[ZNavigationController alloc] initWithNibName:nil bundle:nil];

// Ensure the UINavigationBar is created so that it can be archived. If we do not access the
// navigation bar then it will not be allocated, and thus, it will not be archived by the
// NSKeyedArchvier.
[navController navigationBar];

// Archive the navigation controller.
NSMutableData *data = [NSMutableData data];
NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
[archiver encodeObject:navController forKey:@"root"];
[archiver finishEncoding];

// Unarchive the navigation controller and ensure that our UINavigationBar subclass is used.
NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
[unarchiver setClass:[ZNavigationBar class] forClassName:@"UINavigationBar"];
ZNavigationController *customizedNavController = [unarchiver decodeObjectForKey:@"root"];
[unarchiver finishDecoding];

// Modify the navigation bar to have a background image.
ZNavigationBar *navBar = (ZNavigationBar *)[customizedNavController navigationBar];

[navBar setTintColor:kZodioColorBlue];
[navBar setBackgroundImage:[UIImage imageNamed:@"Home_TopBar_RoundCorner_withLogo"] forBarMetrics:UIBarMetricsDefault];

return customizedNavController;
}

现在您有了自己的自定义 ZNavigationBar(这些是我的代码中的实际名称 - 经历并更改它们很痛苦)您实现触摸处理 - 变量名称非常自定解释性的 - 我想在此进度条的“左附件 View ”、“主要区域”或“右附件 View ”中检测触摸:

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event
{
DDLogVerbose(@"Got a touch at point X: %f Y: %f", point.x, point.y);

//First check if the progress view is visible and touch is within that frame
if ([[JGUploadManager sharedManager] progressViewVisible] &&
CGRectContainsPoint([[JGUploadManager sharedManager] uploadProgressView].frame, point))
{
//Modified frame - have to compensate for actual position of buttons in the view from the standpoint of the UINavigationBar. Can probably use another/smarter method to do this.
CGRect modifiedCancelButtonFrame = [[JGUploadManager sharedManager] uploadProgressView].leftAccessoryFrame;
modifiedCancelButtonFrame.origin.y += [[JGUploadManager sharedManager] uploadProgressView].leftAccessoryView.superview.frame.origin.y;

//Do the same thing for the right view
CGRect modifiedRetryButtonFrame = [[JGUploadManager sharedManager] uploadProgressView].rightAccessoryFrame;
modifiedRetryButtonFrame.origin.y += [[JGUploadManager sharedManager] uploadProgressView].rightAccessoryView.superview.frame.origin.y;


//Touch is on the left button
if ([[JGUploadManager sharedManager] progressViewVisible] &&
[[JGUploadManager sharedManager] uploadProgressView].leftAccessoryView &&
CGRectContainsPoint(modifiedCancelButtonFrame, point))
{
return [[JGUploadManager sharedManager] uploadProgressView].leftAccessoryView;
}


//Touch is on the right button
else if ([[JGUploadManager sharedManager] progressViewVisible] &&
[[JGUploadManager sharedManager] uploadProgressView].rightAccessoryView &&
CGRectContainsPoint(modifiedRetryButtonFrame, point))
{
return [[JGUploadManager sharedManager] uploadProgressView].rightAccessoryView;
}

//Touch is in the main/center area
else
{
return [[JGUploadManager sharedManager] uploadProgressView];
}

}

//Touch is not in the progress view, pass to super
else
{
return [super hitTest:point withEvent:event];
}
}

现在是“跟随”导航 Controller 的 KVO 部分。在某处实现 KVO - 我使用单例“上传管理器”类并将其放在该类的初始化中,但这取决于您的应用程序的设计:

ZRootViewController *rootViewController = ((JGAppDelegate*)[UIApplication sharedApplication].delegate).rootViewDeckController;
[rootViewController addObserver:sharedManager
forKeyPath:@"centerController"
options:NSKeyValueObservingOptionNew
context:nil];

然后当然要实现这个:

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
{
DDLogVerbose(@"KVO shows something changed: %@", keyPath);

if ([keyPath isEqualToString:@"centerController"])
{
DDLogVerbose(@"Center controller changed to: %@", object);
[self centerViewControllerChanged];
}

}

最后是 centerViewControllerChanged 函数:

- (void)centerViewControllerChanged
{
ZRootViewController *rootViewController = ((JGAppDelegate*)[UIApplication sharedApplication].delegate).rootViewDeckController;

ZNavigationController *centerController = (ZNavigationController *)rootViewController.centerController;

//Check if the upload progress view is visible and/or active
if (self.uploadProgressView.frame.origin.y > 0 && self.uploadProgressView.activityIndicatorView.isAnimating)
{
[centerController.navigationBar addSubview:self.uploadProgressView];
}

//Keep weak pointer to center controller for other stuff
DDLogVerbose(@"Setting center view controller: %@", centerController);
self.centerViewController = centerController;
}

如果我误解了你的问题,那么我只是徒劳地输入了世界上最长的 SO 答案,哈哈。我希望这对您有所帮助,如果有任何部分需要澄清,请告诉我!

关于iphone - 如何在 UINavigationBar 下面添加一个 View ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15649342/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com