Cobub無碼埋點(diǎn)關(guān)鍵技術(shù)的實(shí)現(xiàn)
隨著大數(shù)據(jù)時(shí)代的到來,數(shù)據(jù)采集也已經(jīng)變的越來越重要。前端埋點(diǎn)作為一個(gè)比較成熟的數(shù)據(jù)接入手段被廣泛應(yīng)用著。目前埋點(diǎn)分為兩種方式,有碼與無碼埋點(diǎn)。有碼埋點(diǎn)比較容易理解,即調(diào)用SDK的API,在代碼中插入埋點(diǎn)的相關(guān)代碼,實(shí)現(xiàn)用戶行為采集。由于我們?cè)陂_發(fā)項(xiàng)目的時(shí)候,埋點(diǎn)都是手動(dòng)的,每次業(yè)務(wù)需求的改變都要到處埋點(diǎn),而無碼埋點(diǎn),即不需要手動(dòng)插入代碼,只需要前期進(jìn)行相關(guān)配置,SDK自動(dòng)采集用戶行為,極大程度避免了因需求變更、埋點(diǎn)錯(cuò)誤等原因?qū)е碌闹匦侣顸c(diǎn)繁復(fù)工作。本文主要介紹無碼埋點(diǎn)的技術(shù)實(shí)現(xiàn)。
無碼埋點(diǎn)的實(shí)現(xiàn)流程
1.可視化視圖圈選,在頁面上會(huì)出現(xiàn)浮動(dòng)的圓圈,拖動(dòng)圓圈至想配置事件的控件上,將會(huì)彈出輸入事件的彈框。
2.在上一步的彈框中輸入自定義的事件名稱,名稱將會(huì)和視圖的viewPath綁定起來。viewPath是視圖的唯一標(biāo)識(shí),在下文中將詳細(xì)講解。
3.用戶點(diǎn)擊了控件,判斷控件是否綁定過事件,如綁定則進(jìn)行事件上傳。
實(shí)現(xiàn)流程中的技術(shù)點(diǎn)
可視化視圖圈選實(shí)現(xiàn)
自定義UIWindow的子類,當(dāng)做懸浮小圓圈,添加UIPanGestureRecognizer手勢(shì),根據(jù)手勢(shì)的位移,設(shè)置懸浮框的位移。手勢(shì)停止時(shí)獲取懸浮窗中心點(diǎn)的坐標(biāo)。
遍歷主window上的子視圖,找到包含上述懸浮窗中心點(diǎn)且能響應(yīng)用戶交互的最里層視圖,即為用戶可以圈選的視圖。
參考iOS控件的消息傳遞鏈,有個(gè)核心方法。UIView hitTest:(CGPoint)point withEvent:(UIEvent *)event。此API自動(dòng)遍歷子視圖,找到包含point的視圖,event傳nil。由于event參數(shù)是nil,最終找到的視圖并不一定是能響應(yīng)用戶手勢(shì)的視圖,如果不能響應(yīng)則遍歷其父視圖,直到找到能響應(yīng)用戶行為的視圖。
圈選視圖綁定事件
視圖唯一標(biāo)識(shí)viewPath生成,上述步驟已經(jīng)拿到了圈選的視圖。如何確定視圖的viewPath也是重點(diǎn)。viewPath需要整個(gè)應(yīng)用唯一,才可以區(qū)別不同的事件。由于是無碼,所以只能從視圖本身的屬性去分析。我們可以把App的視圖結(jié)構(gòu)理解成樹的概念,樹的根節(jié)點(diǎn)是UIWindow,樹的枝干由UIViewController和UIView組成,葉子節(jié)點(diǎn)都是UIView。那么從根節(jié)點(diǎn)到葉子節(jié)點(diǎn)的路徑可以看做是唯一的。也就是視圖的viewPath。下面介紹下實(shí)現(xiàn)的邏輯,viewPath由兩部分組成,第一部分是節(jié)點(diǎn)路徑,另一部分是與之對(duì)應(yīng)的節(jié)點(diǎn)index。節(jié)點(diǎn)路徑是由每個(gè)節(jié)點(diǎn)的Class拼接而成,節(jié)點(diǎn)index,就是節(jié)點(diǎn)在父節(jié)點(diǎn)中的下標(biāo),比如子視圖在父視圖的subviews數(shù)組中的下標(biāo)。下圖是遍歷節(jié)點(diǎn)的邏輯圖。
計(jì)算節(jié)點(diǎn)的index,這個(gè)步驟,有種特殊的視圖需要注意,可復(fù)用視圖的index是跟數(shù)據(jù)源相關(guān)的,比如UITableViewCell,此類視圖的index不能使用父視圖的subviews的下標(biāo)代替,應(yīng)該使用數(shù)據(jù)源的下標(biāo)代表,比如cell的indexPath.section:indexPath.row。下面給出一個(gè)簡(jiǎn)單視圖和可復(fù)用視圖的viewPath的例子。TestViewController-UIView-UIButton&0-0-0和TableViewController-UITableView-UITableViewCell&0-0-1:0。
如何檢測(cè)用戶觸發(fā)了綁定了事件ID的視圖也是重點(diǎn),此處運(yùn)用的核心技術(shù)是runtime中Method Swizzle。下面介紹一下針對(duì)不同類型的控件,如何hook相應(yīng)的方法。
1. UIControl類型的控件hook – (void)sendAction:(SEL) to:(id)target forEvent:(UIEvent *)event
2. UIScrollView,UITextView,UITableView,UICollectionView 類型的控件,先hook -(void)setDelegate:(id
3. 帶手勢(shì)事件的視圖 hook -(void)addGestureRecognizer方法,并在方法實(shí)現(xiàn)中給手勢(shì)對(duì)象添加新的target和action ,- (void)addTarget:(id)target action:(SEL)action。
總結(jié)
無碼埋點(diǎn)的關(guān)鍵技術(shù),就是以上分析的幾點(diǎn),首先通過可視化圈選拿到需要綁定事件視圖,并生成唯一標(biāo)識(shí)viewPath,通過hook系統(tǒng)控件的方法,拿到用戶觸發(fā)的視圖,生成視圖的viewPath與本地的事件列表比對(duì),比對(duì)成功則上傳viewPath對(duì)應(yīng)的事件。