谢谢原作提供的修复方式//www.greatytc.com/p/bdc23111b2bb
但是我项目中无效的。我通过打印数据,发现其实是tabbar.frame设置错误导致的。这估计就是12.1系统的问题,也是带刘海屏的机型才有的问题。
以下就是我的修复方法,给自己一个记录;
#pragma mark - -----------------以下两个方法解决ios12.1tabbar图标位移问题,如以后IOS12.1解决则可移除--------------
/**
* 用 block 重写某个 class 的指定方法
* @paramtargetClass 要重写的 class
* @paramtargetSelector 要重写的 class 里的实例方法,注意如果该方法不存在于 targetClass 里,则什么都不做
* @paramimplementationBlock 该 block 必须返回一个 block,返回的 block 将被当成 targetSelector 的新实现,所以要在内部自己处理对 super 的调用,以及对当前调用方法的 self 的 class 的保护判断(因为如果 targetClass 的 targetSelector 是继承自父类的,targetClass 内部并没有重写这个方法,则我们这个函数最终重写的其实是父类的 targetSelector,所以会产生预期之外的 class 的影响,例如 targetClass 传进来 UIButton.class,则最终可能会影响到 UIView.class),implementationBlock 的参数里第一个为你要修改的 class,也即等同于 targetClass,第二个参数为你要修改的 selector,也即等同于 targetSelector,第三个参数是 targetSelector 原本的实现,由于 IMP 可以直接当成 C 函数调用,所以可利用它来实现“调用 super”的效果,但由于 targetSelector 的参数个数、参数类型、返回值类型,都会影响 IMP 的调用写法,所以这个调用只能由业务自己写。
*/
CG_INLINE BOOL
OverrideImplementation(ClasstargetClass,SELtargetSelector,id(^implementationBlock)(ClassoriginClass,SELoriginCMD,IMPoriginIMP)) {
MethodoriginMethod =class_getInstanceMethod(targetClass, targetSelector);
if(!originMethod) {
returnNO;
}
IMPoriginIMP =method_getImplementation(originMethod);
method_setImplementation(originMethod,imp_implementationWithBlock(implementationBlock(targetClass, targetSelector, originIMP)));
return YES;
}
+ (void)load {
if(@available(iOS12.1, *)) {
OverrideImplementation(NSClassFromString(@"UITabBar"),@selector(setFrame:), ^id(__unsafe_unretainedClassoriginClass,SELoriginCMD,IMPoriginIMP) {
return^(UIView*selfObject,CGRectfirstArgv) {
CGFloatheight = [[UIApplicationsharedApplication].delegatewindow].safeAreaInsets.bottom;
if([selfObjectisKindOfClass:originClass] && height) {
if(firstArgv.size.height==49) {
firstArgv.size.height+= height;
firstArgv.origin.y-= height;
}
}
// call super
void(*originSelectorIMP)(id,SEL,CGRect);
originSelectorIMP = (void(*)(id,SEL,CGRect))originIMP;
originSelectorIMP(selfObject, originCMD, firstArgv);
};
});
}
}