iOS开发的那些坑

最近重新拿起了iOS的开发,使用OC和Swift混编,碰到了一些比较棘手的问题,在这里记录下来,方便自己以后或他人不再入坑。这篇文章的内容包含:

  1. UITableViewCell的真实结构
  2. 在iOS的环境下使用正则表达式
  3. 如何优雅的隐藏tabbar
  4. 如何修改navigationbar的颜色
  5. 运行时修改使用autolayout的view
  6. navigation controller子view偏移问题
  7. CoreData的使用总结
  8. 解决git的Xcode冲突

1.UITableViewCell的真实结构

目标是想把UITableview在editing模式下的drag按钮去掉,换成自己的样式,但又想保留原生拖动排序的行为,所以研究了很多方法,最后还是找到了UITableViewCell的结构,并将拖动按钮替换。不多说,上代码。

//打印出来的自定义的cell在editing模式下的结构    
(lldb) po self
>

(lldb) po self.subviews
 5 elements
  - [0] : ; layer = >
  - [1] : <_uitableviewcellseparatorview: frame="(15" layer="<CALayer:">>
  - [2] : <_uitableviewcellseparatorview: frame="(15" layer="<CALayer:">>
  - [3] : >
  - [4] : >

(lldb) po self.subviews.last
   Optional
    - Some : >

(lldb) po self.subviews.last?.subviews
   Optional>
     Some : 1 elements
      - [0] : >

(lldb) 复制代码

可以看到此时contentView左右都向内缩进了一定的距离,最后有一个view叫UITableViewCellReorderControl,我想就是它了,然后看它的subviews,竟然包含了一个UIImageView,果断替换之,代码如下。

override func layoutSubviews() {
    super.layoutSubviews()
    setupReorderControl()
}

func setupReorderControl() {
    if (self.reorderControl != nil) {
        return;
    }

    for view in self.subviews {

        if view.description.containsString("UITableViewCellReorderControl") {
            self.reorderControl = view
        }
    }


    if ((self.reorderControl) != nil)
    {
        let imageOfReorder = self.reorderControl?.subviews[0] as? UIImageView
        imageOfReorder?.removeFromSuperview()
    }
}复制代码

此时就完成了将拖动按钮隐藏的功能,但是注意,在这里如果想通过设置reorderControl的frame去改变它的位置是不成功的,我想可能它的布局使用autolayout,并没有深入的再去研究。

2. 在iOS的环境下使用正则表达式

在iOS下使用正则似乎挺麻烦的,如果只是想用正则判断格式,那么可以使用NSPredicate:

let string = "abcd"
let predicate = NSPredicate.init(format: "self matches %@", "^a(b|c)d$")
predicate.evaluateWithObject(string)复制代码

如果要使用正则判断格式,然后获取group的子字符串,就比较麻烦了,并没有Swift的原生方法来做,必须使用NSRegularExpression,话不多说,上代码:

let reg = try! NSRegularExpression.init(pattern: "^a(b|c)d$", options: NSRegularExpressionOptions.CaseInsensitive)
let matches = reg.matchesInString(string, options: NSMatchingOptions.init(rawValue: 0), range: NSMakeRange(0, string.characters.count))
if matches?.count > 0 {
   let returnString = (string as NSString).substringWithRange(matches![0].rangeAtIndex(1))
}复制代码

3. 如何优雅的隐藏tabbar

很多APP都使用TabBarController套NavigationController的方法来作为应用的框架,那么隐藏TabBar就成了一个必要的功能,目前最简单的方法还是使用hidesBottomBarWhenPushed来实现,最简单的方法就是在要隐藏tab bar的Controller里写入下面的方法,来覆默认值。

- (BOOL) hidesBottomBarWhenPushed {
    return (self.navigationController.topViewController == self);
}复制代码

4. 如何修改navigationbar的颜色

[self.navigationController.navigationBar setTranslucent:NO];
self.navigationController.navigationBar.barTintColor = [UIColor redColor];

//在此方法中还原颜色
-(void)viewWillDisappear:(BOOL)animated{
self.navigationController.navigationBar.barTintColor = [UIColor whiteColor];
[super viewWillDisappear:animated];复制代码

}

5. 运行时修改使用autolayout的view

方案是运行时给view添加constraint。

NSLayoutConstraint *c = [NSLayoutConstraint constraintWithItem:view1
                                                     attribute:NSLayoutAttributeTop
                                                     relatedBy:NSLayoutRelationEqual
                                                        toItem: view2
                                                     attribute: NSLayoutAttributeTop
                                                    multiplier:1
                                                      constant:2];复制代码

view1(的顶部) = view2(的顶部) * 1 + 2 这个代码的意思是把view1的顶部放在view2的顶部向下两个像素的地方(因为此时view1是view2 的subview,所以达到的效果就是view1在view2顶部向下2个像素的地方) 如下是所有属性:

NSLayoutAttributeLeft    视图的左边
NSLayoutAttributeRight    视图的右边
NSLayoutAttributeTop    视图的上边
NSLayoutAttributeBottom    视图的下边
NSLayoutAttributeLeading    视图的前边
NSLayoutAttributeTrailing    视图的后边
NSLayoutAttributeWidth    视图的宽度
NSLayoutAttributeHeight    视图的高度
NSLayoutAttributeCenterX    视图的中点的X值
NSLayoutAttributeCenterY    视图中点的Y值
NSLayoutAttributeBaseline    视图的基准线
NSLayoutAttributeNotAnAttribute    无属性复制代码

6. navigation controller子view偏移问题

场景是这样的:页面使用UItableview布局,由于页面比较复杂,在push进去再出来时第一个cell会向下偏移。解决方案:

 //在viewDidLoad中加入:
if ([self respondsToSelector:@selector(edgesForExtendedLayout)]){

    self.edgesForExtendedLayout = UIRectEdgeNone;

}复制代码

7. CoreData的使用总结

//
//  CoreDataHelper.swift
//  TestCode
//
//  Created by lk on 16/7/6.
//  Copyright © 2016年 lk. All rights reserved.
//

import Foundation
import CoreData

@objc class CoreDataHelper: NSObject {

static let KEY_OF_MAX_ORDER_NUMBER_OF_I = "MIK"
static let KEY_OF_MAX_ORDER_NUMBER_OF_O = "MOK"

var context : NSManagedObjectContext!
static let sharedInstance = CoreDataHelper()

private func fetchQuery(request request : NSFetchRequest) -> [NSManagedObject] {
    do{
        let fetchResults = try context?.executeFetchRequest(request) as! [NSManagedObject]!
        return fetchResults
    }catch let error as NSError {
        print("查询失败: \\(error)")
    }
    return []
}

private func save() -> Void {
    //保存
    do{
        try context.save()
    } catch let error as NSError {
        print("保存失败: \\(error)")
    }
}

//TODO: - 这里可能存在溢出漏洞,以后再说
private func addIncrement(key _key : String, newNumber : Int) -> Void {
    if _key == CoreDataHelper. KEY_OF_MAX_ORDER_NUMBER_OF_O || _key == CoreDataHelper.KEY_OF_MAX_ORDER_NUMBER_OF_I {
        NSUserDefaults.standardUserDefaults().setInteger(newNumber, forKey: _key)
    }
}

//Mark: - 重新计算最大值
func recaculateOptionalMaxOrder() -> Void {
    let topOptinal = getOptionals(count: 1)
    if topOptinal.count > 0 {
        NSUserDefaults.standardUserDefaults().setInteger(topOptinal[0].order!.integerValue, forKey: CoreDataHelper.KEY_OF_MAX_ORDER_NUMBER_OF_O)
    }
}

//MARK: - 添加
func addXXX(code code: String, name: String) -> Void {
    if (context == nil) {
        print("添加失败,context为nil")
        return
    }
    let s = NSEntityDescription.insertNewObjectForEntityForName("S", inManagedObjectContext: context) as! S

    //对象赋值
    s.code = code
    s.name = name
    s.numericalCode = info.code
    s.type = info.type

    save()
}

//MARK: - 删除
func removeAll() -> Void {
    let fetchRequest = NSFetchRequest(entityName: "S")

    let fetchResults = fetchQuery(request: fetchRequest)
    for info:NSManagedObject in fetchResults {
        //删除对象
        context.deleteObject(info)
    }
    save()

}

//MARK: - 查询
func getObject(code code : String) -> S? {
    if (context == nil) {
        print("查询失败,context为nil")
        return nil;
    }

    let fetchRequest = NSFetchRequest(entityName: "S")
    let predicate = NSPredicate(format: "code == %@ || numericalCode == %@", code, code)
    fetchRequest.predicate = predicate
    let fetchResults = fetchQuery(request: fetchRequest)
    if let ses = fetchResults as? [S] {
        if ses.count > 0{
            return ses[0]
        }else{
            return nil
        }
    }
    return nil
}

//MARK: - 搜索
func searchStocks(filterString : String) -> [Stock] {
    var attribute = "code";
    let num = Int(filterString)

    if filterString.lengthOfBytesUsingEncoding(NSUTF8StringEncoding) <= 0="" {="" return="" []="" }="" else="" if="" num="=" nil="" todo="" attribute="name" let="" fetchrequest="NSFetchRequest(entityName:" "s")="" fetchrequest.predicate="NSPredicate.init(format:" "%k="" contains="" %@",="" attribute,="" filterstring)="" fetchresults="fetchQuery(request:" fetchrequest)="" print("结果:")="" print(fetchresults)="" var="" ses="[S]()" for="" res="" in="" s="res" as!="" ses.append(s)="" }<="" code="">=>复制代码

8. 解决git的Xcode冲突

待续。。。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值