-
Notifications
You must be signed in to change notification settings - Fork 1.4k
/
Copy pathUINavigationBar+QMUI.m
377 lines (312 loc) · 24.4 KB
/
UINavigationBar+QMUI.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
/**
* Tencent is pleased to support the open source community by making QMUI_iOS available.
* Copyright (C) 2016-2021 THL A29 Limited, a Tencent company. All rights reserved.
* Licensed under the MIT License (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
* http://opensource.org/licenses/MIT
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
*/
//
// UINavigationBar+QMUI.m
// QMUIKit
//
// Created by QMUI Team on 2018/O/8.
//
#import "UINavigationBar+QMUI.h"
#import "QMUICore.h"
#import "NSObject+QMUI.h"
#import "UIView+QMUI.h"
#import "NSArray+QMUI.h"
#import "UINavigationItem+QMUI.h"
NSString *const kShouldFixTitleViewBugKey = @"kShouldFixTitleViewBugKey";
@implementation UINavigationBar (QMUI)
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// [UIKit Bug] Xcode 14 编译的 App 在 iOS 16.0 上可能存在顶部标题布局错乱
// https://github.com/Tencent/QMUI_iOS/issues/1457
//#ifdef IOS16_SDK_ALLOWED 有些机子在 Xcode 13 编译的包上也有问题,所以先不做 Xcode 版本判断
if (@available(iOS 16.0, *)) {
if (@available(iOS 16.1, *)) {
// iOS 16.1 系统已修复
} else {
OverrideImplementation([UINavigationItem class], @selector(setTitleView:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationItem *selfObject, UIView *firstArgv) {
// call super
void (*originSelectorIMP)(id, SEL, UIView *);
originSelectorIMP = (void (*)(id, SEL, UIView *))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, firstArgv);
if (!firstArgv) return;
UINavigationBar *navigationBar = selfObject.qmui_navigationBar;
[navigationBar qmuinb_fixTitleViewLayoutInIOS16];
};
});
OverrideImplementation([UINavigationBar class], @selector(pushNavigationItem:animated:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, UINavigationItem *navigationItem, BOOL animated) {
if (!animated && !selfObject.topItem.titleView && navigationItem.titleView) {
[selfObject qmuinb_fixTitleViewLayoutInIOS16];
}
// call super
void (*originSelectorIMP)(id, SEL, UINavigationItem *, BOOL);
originSelectorIMP = (void (*)(id, SEL, UINavigationItem *, BOOL))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, navigationItem, animated);
};
});
OverrideImplementation([UINavigationBar class], @selector(setItems:animated:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, NSArray<UINavigationItem *> *items, BOOL animated) {
if (!animated && !selfObject.topItem.titleView && items.lastObject.titleView) {
[selfObject qmuinb_fixTitleViewLayoutInIOS16];
}
// call super
void (*originSelectorIMP)(id, SEL, NSArray<UINavigationItem *> *, BOOL);
originSelectorIMP = (void (*)(id, SEL, NSArray<UINavigationItem *> *, BOOL))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, items, animated);
};
});
}
}
//#endif
// [UIKit Bug] iOS 12 及以上的系统,如果设置了自己的 leftBarButtonItem,且 title 很长时,则当 pop 的时候,title 会瞬间跳到左边,与 leftBarButtonItem 重叠
// https://github.com/Tencent/QMUI_iOS/issues/1217
// _UITAMICAdaptorView
Class adaptorClass = NSClassFromString([NSString qmui_stringByConcat:@"_", @"UITAMIC", @"Adaptor", @"View", nil]);
// -[_UINavigationBarContentView didAddSubview:]
OverrideImplementation(NSClassFromString([NSString qmui_stringByConcat:@"_", @"UINavigationBar", @"ContentView", nil]), @selector(didAddSubview:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UIView *selfObject, UIView *firstArgv) {
// call super
void (*originSelectorIMP)(id, SEL, UIView *);
originSelectorIMP = (void (*)(id, SEL, UIView *))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, firstArgv);
if ([firstArgv isKindOfClass:adaptorClass] || [firstArgv isKindOfClass:UILabel.class]) {
firstArgv.qmui_frameWillChangeBlock = ^CGRect(__kindof UIView * _Nonnull view, CGRect followingFrame) {
if ([view qmui_getBoundObjectForKey:kShouldFixTitleViewBugKey]) {
followingFrame = [[view qmui_getBoundObjectForKey:kShouldFixTitleViewBugKey] CGRectValue];
}
return followingFrame;
};
}
};
});
void (^boundTitleViewMinXBlock)(UINavigationBar *, BOOL) = ^void(UINavigationBar *navigationBar, BOOL cleanup) {
if (!navigationBar.topItem.leftBarButtonItem) return;
UIView *titleView = nil;
UIView *adapterView = navigationBar.topItem.titleView.superview;
if ([adapterView isKindOfClass:adaptorClass]) {
titleView = adapterView;
} else {
titleView = [navigationBar.qmui_contentView.subviews qmui_filterWithBlock:^BOOL(__kindof UIView * _Nonnull item) {
return [item isKindOfClass:UILabel.class];
}].firstObject;
}
if (!titleView) return;
if (cleanup) {
[titleView qmui_bindObject:nil forKey:kShouldFixTitleViewBugKey];
} else if (CGRectGetWidth(titleView.frame) > CGRectGetWidth(navigationBar.bounds) / 2) {
[titleView qmui_bindObject:[NSValue valueWithCGRect:titleView.frame] forKey:kShouldFixTitleViewBugKey];
}
};
// // - [UINavigationBar _popNavigationItemWithTransition:]
// - (id) _popNavigationItemWithTransition:(int)arg1; (0x1a15513a0)
OverrideImplementation([UINavigationBar class], NSSelectorFromString([NSString qmui_stringByConcat:@"_", @"popNavigationItem", @"With", @"Transition:", nil]), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^id(UINavigationBar *selfObject, NSInteger firstArgv) {
boundTitleViewMinXBlock(selfObject, NO);
// call super
id (*originSelectorIMP)(id, SEL, NSInteger);
originSelectorIMP = (id (*)(id, SEL, NSInteger))originalIMPProvider();
id result = originSelectorIMP(selfObject, originCMD, firstArgv);
return result;
};
});
// - (void) _completePopOperationAnimated:(BOOL)arg1 transitionAssistant:(id)arg2; (0x1a1551668)
OverrideImplementation([UINavigationBar class], NSSelectorFromString([NSString qmui_stringByConcat:@"_", @"complete", @"PopOperationAnimated:", @"transitionAssistant:", nil]), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, BOOL firstArgv, id secondArgv) {
// call super
void (*originSelectorIMP)(id, SEL, BOOL, id);
originSelectorIMP = (void (*)(id, SEL, BOOL, id))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, firstArgv, secondArgv);
boundTitleViewMinXBlock(selfObject, YES);
};
});
// 以下是将 iOS 12 修改 UINavigationBar 样式的接口转换成用 iOS 13 的新接口去设置(因为新旧方法是互斥的,所以统一在新系统都用新方法)
// 虽然系统的新接口是 iOS 13 就已经存在,但由于 iOS 13、14 都没必要用新接口,所以 QMUI 里在 iOS 15 才开始使用新接口,所以下方的 @available 填的是 iOS 15 而非 iOS 13(与 QMUIConfiguration.m 对应)。
// 但这样有个风险,因为 QMUIConfiguration 配置表里都是用 appearance 的方式去设置 standardAppearance,所以如果在 UINavigationBar 实例被添加到 window 之前修改过旧版任意一个样式接口,就会导致一个新的 UINavigationBarAppearance 对象被设置给 standardAppearance 属性,这样系统就会认为你这个 UINavigationBar 实例自定义了 standardAppearance,那么当它被 moveToWindow 时就不会自动应用 appearance 的值了,因此需要保证在添加到 window 前不要自行修改属性
#ifdef IOS15_SDK_ALLOWED
if (@available(iOS 15.0, *)) {
void (^syncAppearance)(UINavigationBar *, void(^barActionBlock)(UINavigationBarAppearance *appearance)) = ^void(UINavigationBar *navigationBar, void(^barActionBlock)(UINavigationBarAppearance *appearance)) {
if (!barActionBlock) return;
// 需要确保这里获取到的 navigationBar.standardAppearance 是已经被应用了 UIAppearance 全局样式后的值,否则会出现下方 issue 描述的问题
// https://github.com/Tencent/QMUI_iOS/issues/1437
UINavigationBarAppearance *appearance = navigationBar.standardAppearance;
barActionBlock(appearance);
navigationBar.standardAppearance = appearance;
if (QMUICMIActivated && NavBarUsesStandardAppearanceOnly) {
navigationBar.scrollEdgeAppearance = appearance;
}
};
OverrideImplementation([UINavigationBar class], @selector(setBarTintColor:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, UIColor *barTintColor) {
// call super
void (*originSelectorIMP)(id, SEL, UIColor *);
originSelectorIMP = (void (*)(id, SEL, UIColor *))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, barTintColor);
syncAppearance(selfObject, ^void(UINavigationBarAppearance *appearance) {
appearance.backgroundColor = barTintColor;
});
};
});
OverrideImplementation([UINavigationBar class], @selector(barTintColor), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^UIColor *(UINavigationBar *selfObject) {
return selfObject.standardAppearance.backgroundColor;
};
});
OverrideImplementation([UINavigationBar class], @selector(setBackgroundImage:forBarPosition:barMetrics:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, UIImage *image, UIBarPosition barPosition, UIBarMetrics barMetrics) {
// call super
void (*originSelectorIMP)(id, SEL, UIImage *, UIBarPosition, UIBarMetrics);
originSelectorIMP = (void (*)(id, SEL, UIImage *, UIBarPosition, UIBarMetrics))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, image, barPosition, barMetrics);
syncAppearance(selfObject, ^void(UINavigationBarAppearance *appearance) {
appearance.backgroundImage = image;
});
};
});
OverrideImplementation([UINavigationBar class], @selector(backgroundImageForBarPosition:barMetrics:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^UIImage *(UINavigationBar *selfObject, UIBarPosition firstArgv, UIBarMetrics secondArgv) {
return selfObject.standardAppearance.backgroundImage;
};
});
OverrideImplementation([UINavigationBar class], @selector(setShadowImage:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, UIImage *shadowImage) {
// call super
void (*originSelectorIMP)(id, SEL, UIImage *);
originSelectorIMP = (void (*)(id, SEL, UIImage *))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, shadowImage);
syncAppearance(selfObject, ^void(UINavigationBarAppearance *appearance) {
appearance.shadowImage = shadowImage;
});
};
});
OverrideImplementation([UINavigationBar class], @selector(shadowImage), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^UIImage *(UINavigationBar *selfObject) {
return selfObject.standardAppearance.shadowImage;
};
});
OverrideImplementation([UINavigationBar class], @selector(setBarStyle:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, UIBarStyle barStyle) {
// call super
void (*originSelectorIMP)(id, SEL, UIBarStyle);
originSelectorIMP = (void (*)(id, SEL, UIBarStyle))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, barStyle);
syncAppearance(selfObject, ^void(UINavigationBarAppearance *appearance) {
appearance.backgroundEffect = [UIBlurEffect effectWithStyle:barStyle == UIBarStyleDefault ? UIBlurEffectStyleSystemChromeMaterialLight : UIBlurEffectStyleSystemChromeMaterialDark];
});
};
});
// iOS 15 没有对应的属性
// OverrideImplementation([UINavigationBar class], @selector(barStyle), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
// return ^UIBarStyle(UINavigationBar *selfObject) {
//
// if (@available(iOS 15.0, *)) {
// return ???;
// }
//
//
// // call super
// UIBarStyle (*originSelectorIMP)(id, SEL);
// originSelectorIMP = (UIBarStyle (*)(id, SEL))originalIMPProvider();
// UIBarStyle result = originSelectorIMP(selfObject, originCMD);
//
// return result;
// };
// });
OverrideImplementation([UINavigationBar class], @selector(setTitleTextAttributes:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, NSDictionary<NSAttributedStringKey, id> *titleTextAttributes) {
// call super
void (*originSelectorIMP)(id, SEL, NSDictionary<NSAttributedStringKey, id> *);
originSelectorIMP = (void (*)(id, SEL, NSDictionary<NSAttributedStringKey, id> *))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, titleTextAttributes);
syncAppearance(selfObject, ^void(UINavigationBarAppearance *appearance) {
appearance.titleTextAttributes = titleTextAttributes;
});
};
});
}
if (@available(iOS 15.0, *)) {
if (!QMUICMIActivated) return;
if (!(NavBarRemoveBackgroundEffectAutomatically || TabBarRemoveBackgroundEffectAutomatically || ToolBarRemoveBackgroundEffectAutomatically)
&& !(NavBarUsesStandardAppearanceOnly || TabBarUsesStandardAppearanceOnly || ToolBarUsesStandardAppearanceOnly)) return;
// - [_UIBarBackground updateBackground]
OverrideImplementation(NSClassFromString(@"_UIBarBackground"), NSSelectorFromString(@"updateBackground"), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UIView *selfObject) {
// call super
void (*originSelectorIMP)(id, SEL);
originSelectorIMP = (void (*)(id, SEL))originalIMPProvider();
originSelectorIMP(selfObject, originCMD);
if (!selfObject.superview) return;
if (!NavBarRemoveBackgroundEffectAutomatically && !NavBarUsesStandardAppearanceOnly && [selfObject.superview isKindOfClass:UINavigationBar.class]) return;
if (!TabBarRemoveBackgroundEffectAutomatically && !TabBarUsesStandardAppearanceOnly && [selfObject.superview isKindOfClass:UITabBar.class]) return;
if (!ToolBarRemoveBackgroundEffectAutomatically && !ToolBarUsesStandardAppearanceOnly && [selfObject.superview isKindOfClass:UIToolbar.class]) return;
UIImageView *backgroundImageView1 = [selfObject valueForKey:@"_colorAndImageView1"];
UIImageView *backgroundImageView2 = [selfObject valueForKey:@"_colorAndImageView2"];
UIVisualEffectView *backgroundEffectView1 = [selfObject valueForKey:@"_effectView1"];
UIVisualEffectView *backgroundEffectView2 = [selfObject valueForKey:@"_effectView2"];
// iOS 14 系统默认特性是存在 backgroundImage 则不存在其他任何背景,但如果存在 barTintColor 则磨砂 view 也可以共存。
// iOS 15 系统默认特性是 backgroundImage、backgroundColor、backgroundEffect 三者都可以共存,其中前两者共用 _colorAndImageView,而我们这个开关为了符合 iOS 14 的特性,仅针对 _colorAndImageView 是因为 backgroundImage 存在而出现的情况做处理。
if (NavBarRemoveBackgroundEffectAutomatically || TabBarRemoveBackgroundEffectAutomatically || ToolBarRemoveBackgroundEffectAutomatically) {
BOOL hasBackgroundImage1 = backgroundImageView1 && backgroundImageView1.superview && !backgroundImageView1.hidden && backgroundImageView1.image;
BOOL hasBackgroundImage2 = backgroundImageView2 && backgroundImageView2.superview && !backgroundImageView2.hidden && backgroundImageView2.image;
BOOL shouldHideEffectView = hasBackgroundImage1 || hasBackgroundImage2;
if (shouldHideEffectView) {
backgroundEffectView1.hidden = YES;
backgroundEffectView2.hidden = YES;
} else {
// 把 backgroundImage 置为 nil,理应要恢复 effectView 的显示,但由于 iOS 15 里 effectView 有2个,什么时候显示哪个取决于 contentScrollView 的滚动位置,而这个位置在当前上下文里我们是无法得知的,所以先不处理了,交给系统在下一次 updateBackground 时刷新吧...
}
}
// 虽然 4.4.0 增加的这些开关会保证 scrollEdgeAppearance 也被设置,但系统始终都会同时显示两份 view(一份 standard 的一份 scrollEdge 的),当你的样式是不透明时没问题,但如果存在半透明,同时显示两份 view 就会导致两个半透明的效果重叠在一起,最终肉眼看到的样式和预期是不符合的,所以 4.4.4 开始,我们会强制让其中一份 view 隐藏掉。
if (NavBarUsesStandardAppearanceOnly || TabBarUsesStandardAppearanceOnly || ToolBarUsesStandardAppearanceOnly) {
backgroundImageView2.hidden = YES;
backgroundEffectView2.hidden = YES;
}
};
});
// 尚未应用 UIAppearance 就已经修改 bar 的样式的场景,可能导致 bar 样式无法与全局保持一致,所以这里做个提醒
// https://github.com/Tencent/QMUI_iOS/issues/1451
// - [UINavigationBar setStandardAppearance:]
OverrideImplementation([UINavigationBar class], @selector(setStandardAppearance:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(UINavigationBar *selfObject, UINavigationBarAppearance * firstArgv) {
// call super
void (*originSelectorIMP)(id, SEL, UINavigationBarAppearance *);
originSelectorIMP = (void (*)(id, SEL, UINavigationBarAppearance *))originalIMPProvider();
originSelectorIMP(selfObject, originCMD, firstArgv);
// 这里只希望识别 UINavigationController 自带的 navigationBar,不希望处理业务自己 new 的 bar,所以用 superview 是否为 UILayoutContainerView 来作为判断条件。
BOOL isSystemBar = [NSStringFromClass(selfObject.superview.class) hasPrefix:@"UILayoutContainer"];
BOOL alreadyMoveToWindow = !!selfObject.window;
BOOL isPresenting = NO;
if (!alreadyMoveToWindow) {
UINavigationController *nav = [selfObject.qmui_viewController isKindOfClass:UINavigationController.class] ? selfObject.qmui_viewController : nil;
isPresenting = nav && nav.presentedViewController;
}
if (isSystemBar && !alreadyMoveToWindow && !isPresenting) {
QMUIAssert(NO, @"UINavigationBar (QMUI)", @"试图在 UINavigationBar 尚未添加到 window 上时就修改它的样式,可能导致 UINavigationBar 的样式无法与全局保持一致。");
}
};
});
}
#endif
});
}
- (UIView *)qmui_contentView {
return [self valueForKeyPath:@"visualProvider.contentView"];
}
- (void)qmuinb_fixTitleViewLayoutInIOS16 {
// _UINavigationBarTitleControl,在每次转场动画时都会被重建,但无动画则一直都是这个实例(横竖屏切换也是同一个实例)
Class titleControlClass = NSClassFromString([NSString qmui_stringByConcat:@"_", @"UINavigationBar", @"TitleControl", nil]);
UIView *titleControl = [self.qmui_contentView.subviews qmui_filterWithBlock:^BOOL(__kindof UIView * _Nonnull item) {
return [item isKindOfClass:titleControlClass];
}].firstObject;
titleControl.qmui_frameWillChangeBlock = ^CGRect(__kindof UIView * _Nonnull view, CGRect followingFrame) {
followingFrame = CGRectSetY(followingFrame, CGRectGetMinYVerticallyCenterInParentRect(view.superview.bounds, followingFrame));
return followingFrame;
};
}
@end