tags:開發(fā)隨筆
緣起
燈下鼠同學(xué)在使用了MarkNotes后,建議增加筆記分享的功能。
其實這個問題我已經(jīng)思考了很久,除了自己做服務(wù)器端,沒有一個太好的方案。所以在目前的版本中,我暫時只提供了通過郵件發(fā)送筆記的功能。話說,iOS版本的mail應(yīng)用真是弱的可以,想寫一篇富文本的郵件都很難。MarkNote for iOS簡直是一個完美的iOS郵件編輯器。
當(dāng)然了,使用 MarkNotes/Marknote你也可以將筆記很方便的導(dǎo)出為HTML或者PDF。甚至你可以借助其強大的過濾功能,導(dǎo)出為靜態(tài)網(wǎng)站,扔到web服務(wù)器上,所有的人都可以看。
然而,這還不是理想的分享。
需求
我覺得一個理想的分享,應(yīng)該可以滿足以下幾個場景:
A: 可以設(shè)置為public,所有人都可以讀,甚至不需要擁有賬號;
B: 邀請者只讀;
C: 邀請者可寫,從而實現(xiàn)協(xié)同工作;
簡單分析一下, 場景A其實是一種publish,導(dǎo)出為HTML通過web發(fā)布基本上可以滿足,只是目前門檻稍微有點高,需要一種方式讓整個過程更簡單;
場景B和C需要:
- 一種機制,可以識別并邀請其他用戶;
- 一種機制將邀請發(fā)出;
- 被邀請者可能有各種古怪的場合,比如還沒有安裝你的應(yīng)用;
- 在此基礎(chǔ)上,對讀寫權(quán)限進(jìn)行控制;
如果可以用服務(wù)器端,上面的需求還是有望從技術(shù)上完美實現(xiàn)。但是運營的成本會極大的增加:你需要將大量的用戶吸引到你的平臺上來,注冊,留存,還需要對數(shù)據(jù)安全等方方面面考慮周全... 可能還要考慮天朝的監(jiān)管。
因此我還是希望MarkNotes保持Serverless的架構(gòu),保持簡單。
曙光
蘋果在WWDC2016上宣布了Cloudkit的新特性,其中CKShare的特性尤其是吸引了我的目光。
我將下面的視頻看了2遍:
https://developer.apple.com/videos/play/wwdc2016/226/ 初步覺得可能可以解決大部分的問題。
查了一些資料后,簡單實驗了一下。順便吐槽一下,CKShare方面的文檔和代碼很少。尤其是代碼,能找到的少之又少,不少還有問題。
CloudKit有2個數(shù)據(jù)庫privateDatabase和publicDatabase。
要實現(xiàn)場景A,將數(shù)據(jù)放在public database即可。簡單。
所以我們還是集中精力在場景2和3。簡單的說,即讓受邀用戶來參與。
CKShare需要一個root record,它是通過樹的方式來控制分享的數(shù)據(jù)的。Root record可以有自己的子節(jié)點,子節(jié)點還可以有子節(jié)點。將root record傳給CKShare對象,CKShare來控制誰被邀請,是只讀還是讀寫。
有一個坑,被CKShare分享的數(shù)據(jù)只能存在于custom zone中,如果存放于default zone則會報錯。
所以在進(jìn)行分享前,先要創(chuàng)建custom zone:
let container: CKContainer = CKContainer.default()
let privateDatabase = container.privateCloudDatabase
let customZone = CKRecordZone(zoneName: customZoneName)
privateDatabase.save(customZone, completionHandler: ({returnRecord, error in
if error != nil {
// Zone creation failed
OperationQueue.main.addOperation {
print("Cloud Error:\(error?.localizedDescription)")
}
} else {
// Zone creation succeeded
OperationQueue.main.addOperation {
print( "The \(self.customZoneName) was successfully created in the private database.")
}
}
}))
之后,我們只需要以root record作為參數(shù)創(chuàng)建CKShare對象,然后二者保存到private database中就可以分享了:
let share = CKShare(rootRecord: newRecord)
share[CKShareTitleKey] = "hello" as CKRecordValue?
let modifyRecordsOperation = CKModifyRecordsOperation( recordsToSave: [newRecord, share], recordIDsToDelete: nil)
modifyRecordsOperation.modifyRecordsCompletionBlock = { records, recordIDs, error in
if let error = error {
print(error.localizedDescription)
}
if records != nil {
print("Share and Root records saved successfully")
}
preparationCompletionHandler(share, container , error)
}
privateDatabase.add(modifyRecordsOperation)
創(chuàng)建完CKShare對象后,需要將邀請發(fā)送給被邀請人。新的API提供了一個UICloudSharingController來簡化被邀請人查找等操作UI的工作量。使用方式如下:
let cloudSharingController: UICloudSharingController = UICloudSharingController{ controller,
preparationCompletionHandler in
...
}
cloudSharingController.delegate = self
cloudSharingController.popoverPresentationController?.sourceView = self.view
// Set sharing permissions
cloudSharingController.availablePermissions = [.allowPublic, .allowReadOnly]
// Show cloud sharing dialog
self.present(cloudSharingController, animated: true, completion: nil)
運行效果如下:
總結(jié)
CloudKit中最新的Share API提供了一種分享的好機制,雖然如果要考慮用戶沒裝app時的降級措施,可能還需要服務(wù)器端,但已經(jīng)大大的降低了分享的難度。
說點局限吧:
- 顧名思義,用戶的身份是和iCloud賬號綁定的;
- 邀請者需要知道被邀請者iCloud賬號綁定的的郵件或者手機;
- 基于apple一貫的借助系統(tǒng)升級來促進(jìn)硬件銷售的策略,CloudKit Share API只支持iOS10以上的設(shè)備。
代碼
代碼放在github上供參考 https://github.com/marknote/CloudKitSharing。請注意,目前的代碼只是 最簡單的探索,功能并不完善,細(xì)節(jié)也未考慮 。