self.segmentedControl = [[UISegmentedControl alloc] initWithItems:[NSArray arrayWithObjects:@"Uno",@"Dos",nil]];self.segmentedControl.selectedSegmentIndex = 0;[self.segmentedControl addTarget:self action:@selector(segmentedControlChanged:) forControlEvents:UIControlEventValueChanged];self.segmentedControl.height = 32.0;self.segmentedControl.wIDth = 310.0;self.segmentedControl.segmentedControlStyle = UISegmentedControlStylebar;self.segmentedControl.tintcolor = [UIcolor colorWithWhite:0.9 Alpha:1.0];self.segmentedControl.autoresizingMask = UIVIEwautoresizingFlexibleleftmargin | UIVIEwautoresizingFlexibleRightmargin;UIVIEw* toolbar = [[UIVIEw alloc] initWithFrame:CGRectMake(0,self.vIEw.wIDth,header_HEIGHT)];toolbar.autoresizingMask = UIVIEwautoresizingFlexibleWIDth;CAGradIEntLayer *gradIEnt = [CAGradIEntLayer layer]; gradIEnt.frame = CGRectMake( toolbar.bounds.origin.x,toolbar.bounds.origin.y,// * 2 for enough slack when iPad rotates toolbar.bounds.size.wIDth * 2,toolbar.bounds.size.height ); gradIEnt.colors = [NSArray arrayWithObjects: (ID)[[UIcolor whitecolor] CGcolor],(ID)[[UIcolor colorWithWhite:0.8 Alpha:1.0 ] CGcolor ],nil];[toolbar.layer insertSublayer:gradIEnt atIndex:0];toolbar.backgroundcolor = [UIcolor navigationbarShadowcolor];[toolbar addSubvIEw:self.segmentedControl];UIVIEw* border = [[UIVIEw alloc] initWithFrame:CGRectMake(0,header_HEIGHT - 1,toolbar.wIDth,1)];border.autoresizingMask = UIVIEwautoresizingFlexibleWIDth | UIVIEwautoresizingFlexibletopmargin;border.backgroundcolor = [UIcolor colorWithWhite:0.7 Alpha:1.0];border.autoresizingMask = UIVIEwautoresizingFlexibleWIDth;[toolbar addSubvIEw:border];[self.segmentedControl centerInParent];self.tableVIEw.tableheaderVIEw = toolbar;
http://scs.veetle.com/soget/session-thumbnails/5363e222d2e10/86a8dd984fcaddee339dd881544ecac7/5363e222d2e10_86a8dd984fcaddee339dd881544ecac7_20140509171623_536d6fd78f503_68_896x672.jpg
解决方法 正如已经在其他答案中写过的那样,UINavigationbar抓住导航栏本身附近的触摸,但不是因为它有一些子视图扩展到边缘:这不是原因.如果您记录整个视图层次结构,您将看到UINavigationbar不会超出定义的边.
接收触摸的原因是另一个原因:
在UIKit中有很多“特殊情况”,这就是其中之一.
当您点击屏幕时,将启动一个名为“命中测试”的过程.从第一个UIWindow开始,所有视图都被要求回答两个“问题”:点击你的边界点?接触事件的子视图是什么?
这两个方法可以回答这个问题:
- (BOol)pointInsIDe:(CGPoint)point withEvent:(UIEvent *)event;- (UIVIEw *)hitTest:(CGPoint)point withEvent:(UIEvent *)event;
好的,现在我们可以继续
在水龙头之后,UIApplicationMain开始命中测试过程.命中测试从主UIWindow开始(例如在状态栏窗口和警报视图窗口中执行),并且遍历所有子视图.
此过程执行3次:
>从UIWindow开始两次
>从_UIApplicationHandleEvent开始一次
如果您点击导航栏,您将看到UIWindow上的hitTest将返回UINavigationbar(全部三次)
然而,如果您点击导航栏下方的区域,您将会感到奇怪的是:
>前两个hitTest将返回您的UISegmentedControl
>最后一个hitTest将返回UINavigationbar
为什么这个?
如果你打开UIVIEw子类,覆盖hitTest,你会看到前两次敲击点是正确的.第三次,有些事情改变了点 – 15点(或类似数字)
经过大量的搜索,我发现这里发生了什么:
UIWindow有一个(私有)方法调用
-(CGPoint)warpPoint:(CGPoint)point;
调试它,我看到如果这个方法立即在状态栏的下方,这个方法会改变点击点.
调试更多,我看到堆栈调用使这成为可能,只有3:
[UINavigationbar,_isChargeEnabled][UINavigationbar,isEnabled][UINavigationbar,_isAlphaHittableAndHasAlphaHittableAncestors]
所以,最后,这个warpPoint方法会检查是否启用了UINavigationbar并启用了hittable,如果是的话,它会“扭曲”.该点在0到15之间的像素扭曲,当您更靠近导航栏时,此“翘曲”会增加.
现在你知道幕后会发生什么,你必须知道如何避免它(如果你愿意的话).
你不能简单地覆盖warpPoint:如果应用程序必须去AppStore:它是一种私有的方法,你的应用程序将被拒绝.
你必须找到另一个系统(像建议,覆盖sendEvent,但我不知道它是否会工作)
因为这个问题是有趣的,我会在明天考虑一个合法的解决方案并更新这个答案(一个好的起点可以是子类化UINavigationbar,覆盖hitTest和pointInsIDe,返回nil / false,如果给同一个事件多次调用,点变化但是我明天必须测试它是否工作)
编辑
好的,我尝试了许多解决方案,但找到合法和稳定的解决方案并不简单.
我已经描述了系统的实际行为,可能因不同版本而异(hitTest调用多于或少于3次,warpPoint扭曲约15px可改变ecc ecc的点).
最稳定的显然是warpPoint的非法覆盖:在UIWindow子类中:
-(CGPoint)warpPoint:(CGPoint)point;{ return point;}
然而,我发现一个这样的方法(在UIWindow子类中)它足够稳定,并且做的诀窍:
- (UIVIEw *)hitTest:(CGPoint)point withEvent:(UIEvent *)event{ // this method is not safe if you tap the screen two times at the same x position and y position different for 16px,because it moves the point if (self.lastPoint.x == point.x) { // the points are on the same vertical line if ((0 < (self.lastPoint.y - point.y)) && ((self.lastPoint.y - point.y) < 16) ) { // there is a differenc of ~15px in the y position? // if so,the point has been changed point.y = self.lastPoint.y; } } self.lastPoint = point; return [super hitTest:point withEvent:event];}
该方法记录最后一个点,并且如果后续点击位于相同的x,并且y最大为16px,则使用上一个点.我测试了很多,看起来很稳定.如果需要,您可以添加更多的控件来仅在特定控制器中启用此行为,或仅在窗口的已定义部分ecc ecc上.如果我找到另一个解决方案,我会更新这个帖子
总结以上是内存溢出为你收集整理的ios – 为什么我的UISegmentedControl的顶部不可以插拔?全部内容,希望文章能够帮你解决ios – 为什么我的UISegmentedControl的顶部不可以插拔?所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)