iOS開發(fā)的那些坑

最近重新拿起了iOS的開發(fā),使用OC和Swift混編,碰到了一些比較棘手的問題,在這里記錄下來,方便自己以后或他人不再入坑。這篇文章的內(nèi)容包含:

  1. UITableViewCell的真實結(jié)構(gòu)
  2. 在iOS的環(huán)境下使用正則表達(dá)式
  3. 如何優(yōu)雅的隱藏tabbar
  4. 如何修改navigationbar的顏色
  5. 運行時修改使用autolayout的view
  6. navigation controller子view偏移問題
  7. CoreData的使用總結(jié)
  8. 解決git的Xcode沖突

1.UITableViewCell的真實結(jié)構(gòu)

目標(biāo)是想把UITableview在editing模式下的drag按鈕去掉,換成自己的樣式,但又想保留原生拖動排序的行為,所以研究了很多方法,最后還是找到了UITableViewCell的結(jié)構(gòu),并將拖動按鈕替換。不多說,上代碼。

//打印出來的自定義的cell在editing模式下的結(jié)構(gòu)   
(lldb) po self
<QKZhi_iOS.EditingCell: 0x7c148800; baseClass = UITableViewCell; frame = (0 0; 320 56); autoresize = W; layer = <CALayer: 0x7d086b40>>

(lldb) po self.subviews
 5 elements
- [0] : <UITableViewCellContentView: 0x7d086d10; frame = (38 0; 242 55.5); opaque = NO; gestureRecognizers = <NSArray: 0x7d0882b0>; layer = <CALayer: 0x7d086de0>>
- [1] : <_UITableViewCellSeparatorView: 0x7d087c40; frame = (15 55; 305 1); layer = <CALayer: 0x7d0852c0>>
- [2] : <_UITableViewCellSeparatorView: 0x7b163240; frame = (15 55.5; 305 0.5); layer = <CALayer: 0x7b162de0>>
- [3] : <UITableViewCellEditControl: 0x7b164700; frame = (0 0; 47 56); opaque = NO; layer = <CALayer: 0x7b164950>>
- [4] : <UITableViewCellReorderControl: 0x7b164d10; frame = (268 0; 52 56); opaque = NO; autoresize = LM; layer = <CALayer: 0x7b164f60>>

(lldb) po self.subviews.last
 Optional<UIView>
    - Some : <UITableViewCellReorderControl: 0x7b164d10; frame = (268 0; 52 56); opaque = NO; autoresize = LM; layer = <CALayer: 0x7b164f60>>

(lldb) po self.subviews.last?.subviews
 Optional<Array<UIView>>
  Some : 1 elements
    - [0] : <UIImageView: 0x7b165130; frame = (15 24; 22 8.5); opaque = NO; userInteractionEnabled = NO; tag = 1; layer = <CALayer: 0x7b165210>>

(lldb) 

可以看到此時contentView左右都向內(nèi)縮進(jìn)了一定的距離,最后有一個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()
    }
}

此時就完成了將拖動按鈕隱藏的功能,但是注意,在這里如果想通過設(shè)置reorderControl的frame去改變它的位置是不成功的,我想可能它的布局使用autolayout,并沒有深入的再去研究。

2. 在iOS的環(huán)境下使用正則表達(dá)式

在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. 如何優(yōu)雅的隱藏tabbar

很多APP都使用TabBarController套NavigationController的方法來作為應(yīng)用的框架,那么隱藏TabBar就成了一個必要的功能,目前最簡單的方法還是使用hidesBottomBarWhenPushed來實現(xiàn),最簡單的方法就是在要隱藏tab bar的Controller里寫入下面的方法,來覆默認(rèn)值。
- (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,所以達(dá)到的效果就是view1在view2頂部向下2個像素的地方)
如下是所有屬性:

NSLayoutAttributeLeft   視圖的左邊
NSLayoutAttributeRight  視圖的右邊
NSLayoutAttributeTop    視圖的上邊
NSLayoutAttributeBottom 視圖的下邊
NSLayoutAttributeLeading    視圖的前邊
NSLayoutAttributeTrailing   視圖的后邊
NSLayoutAttributeWidth  視圖的寬度
NSLayoutAttributeHeight 視圖的高度
NSLayoutAttributeCenterX    視圖的中點的X值
NSLayoutAttributeCenterY    視圖中點的Y值
NSLayoutAttributeBaseline   視圖的基準(zhǔn)線
NSLayoutAttributeNotAnAttribute 無屬性

6. navigation controller子view偏移問題

場景是這樣的:頁面使用UItableview布局,由于頁面比較復(fù)雜,在push進(jìn)去再出來時第一個cell會向下偏移。解決方案:

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

}

7. CoreData的使用總結(jié)

//
//  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)
    let fetchResults = fetchQuery(request: fetchRequest)
    print("結(jié)果:")
    print(fetchResults)
    var ses = [S]()
    for res in fetchResults {
        let s = res as! S
        ses.append(s)
    }
    return ses
}

8. 解決git的Xcode沖突

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 當(dāng)時光流逝,記憶也開始散去,猛然回頭卻發(fā)現(xiàn)自己還在原地。 目前做iOS平臺開發(fā)有兩種語言,這就導(dǎo)致了一個項目組可能...
    TheAlchemist閱讀 2,838評論 3 15
  • 這篇文章的內(nèi)容包含:UITableViewCell的真實結(jié)構(gòu)在iOS的環(huán)境下使用正則表達(dá)式如何優(yōu)雅的隱藏tabba...
    施憶閱讀 1,130評論 0 0
  • 2月8日 圖片發(fā)自簡書App 1, 一個人無拘無束,沒有固定的時間,睡覺都變得隨意起來,我還是個早起的好孩子嗎? ...
    魯莽書生閱讀 480評論 0 0
  • 只有真愛過, 才會懂的33句話 ↓↓↓ ● 先愛的人總是輸,主動的那個總是容易被打敗,委曲求全的那個死的最慘。 ●...
    小淘米_TTMIX閱讀 675評論 0 0
  • 晚上看電影,《東方快車謀殺案》。這兩天晚上都睡得特晚,早上起的特別早。中午又沒休息,所以看電影時特別困! 也許是因...
    transformer321閱讀 284評論 0 0

友情鏈接更多精彩內(nèi)容