iOS開發的那些坑

最近從新拿起了iOS的開發,使用OC和Swift混編,碰到了一些比較棘手的問題,在這裏記錄下來,方便本身之後或他人再也不入坑。這篇文章的內容包含:ios

  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的結構,並將拖動按鈕替換。很少說,上代碼。git

//打印出來的自定義的cell在editing模式下的結構    
(lldb) po self

  
  
  

 
  
  > (lldb) po self.subviews 5 elements - [0] : 
 
  
    ; layer = 
   
     > - [1] : <_uitableviewcellseparatorview: 305="" 0x7d087c40;="" frame="(15" 55;="" 1);="" layer=" 
    
      > - [2] : <_uitableviewcellseparatorview: 305="" 0x7b163240;="" frame="(15" 55.5;="" 0.5);="" layer=" 
     
       > - [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,並無深刻的再去研究。swift

2. 在iOS的環境下使用正則表達式

在iOS下使用正則彷佛挺麻煩的,若是隻是想用正則判斷格式,那麼可使用NSPredicate:xcode

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

若是要使用正則判斷格式,而後獲取group的子字符串,就比較麻煩了,並無Swift的原生方法來作,必須使用NSRegularExpression,話很少說,上代碼:app

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];複製代碼

}iview

5. 運行時修改使用autolayout的view

方案是運行時給view添加constraint。ide

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衝突

待續。。。

相關文章
相關標籤/搜索