Определить тип изменения записи при получении измененных записей CloudKit - PullRequest
0 голосов
/ 04 июня 2019

Я пытаюсь завершить приложение, используя синхронизацию CloudKit и локальные CoreData.Большинство операций работают должным образом, но я не могу найти методологию для определения типа изменений, о которых сообщает CloudKit.Я получаю измененные записи, но мне нужно знать, было ли изменение редактированием, новой записью или удалением.Любое руководство будет оценено.

Вот фрагмент кода, который, как я думал, может быть настроен для определения типа редактирования, который мне нужно будет сделать в CoreData.Xcode 10.2.1 iOS 12.2 Swift (последняя версия)

func fetchZoneChangesInZones( _ zones : [CKRecordZone.ID], _ completionHandler: @escaping (Error?) -> Void) {

    var fetchConfigurations = [CKRecordZone.ID : CKFetchRecordZoneChangesOperation.ZoneConfiguration]()

    for zone in zones {
        if let changeToken = UserDefaults.standard.zoneChangeToken(forZone: zone) {
            let configuration = CKFetchRecordZoneChangesOperation.ZoneConfiguration(previousServerChangeToken: changeToken, resultsLimit: nil, desiredKeys: nil)
            fetchConfigurations[zone] = configuration
        }//if let changeToken

    }//for in

    let operation = CKFetchRecordZoneChangesOperation(recordZoneIDs: zones, configurationsByRecordZoneID: fetchConfigurations)

    operation.fetchAllChanges = true

    var changedPatients = [CKRecord]()
    var changedCategory1s = [CKRecord]()

    //I thought that I should be able to query for the change type here and make separate arrays for each change type
    operation.recordChangedBlock = { record in
        if record.recordType == "Patient" {
            changedPatients.append(record)
        }
    }//recordChangedBlock

    operation.fetchRecordZoneChangesCompletionBlock = { [weak self] error in
        for record in changedPatients {
            //my actions here - need to choose new, changed or delete
            self!.saveCKRecordToCoreData(record: record)
        }//for record in

        completionHandler(error)

    }//fetchRecordZoneChangesCompletionBlock

    operation.recordZoneFetchCompletionBlock = { recordZone, changeToken, data, moreComing, error in
        UserDefaults.standard.set(changeToken, forZone: recordZone)
    }//recordZoneFetchCompletionBlock

    privateDatabase.add(operation)

}//fetchZoneChangesInZones

Ответы [ 2 ]

1 голос
/ 04 июня 2019

Я не так хорош в swift, но я опубликую в цели c, чтобы вы могли преобразовать это в swift

  1. Перво-наперво, если вы хотите уведомить, если запись имеетОтредактировано, удалено или создано, вам нужно зарегистрироваться для push-уведомлений.

  2. Затем Подписаться, чтобы обновить, добавить этот блок в didFinishLaunchingWithOptions


    - (void)subscribeToEventChanges
    {
        BOOL isSubscribed = [[NSUserDefaults standardUserDefaults] boolForKey:@"subscribedToUpdates"];
        if (isSubscribed == NO) {
        NSPredicate *predicate = [NSPredicate predicateWithFormat:@"TRUEPREDICATE"];

        CKQuerySubscription *subscription = [[CKQuerySubscription alloc] initWithRecordType:@"Patient" predicate:predicate options:CKQuerySubscriptionOptionsFiresOnRecordCreation | CKQueryNotificationReasonRecordDeleted | CKQueryNotificationReasonRecordUpdated];

        CKNotificationInfo *CKNotification=[[CKNotificationInfo alloc]init];
        CKNotification.shouldSendContentAvailable=YES;
        CKNotification.soundName=@"";
        subscription.notificationInfo=CKNotification;
        CKDatabase *publicDatabase = [[CKContainer containerWithIdentifier:@"your container identifir"] privateCloudDatabase];

        [publicDatabase saveSubscription:subscription completionHandler:^(CKSubscription * _Nullable subscription, NSError * _Nullable error) {
            if (error) {
                // Handle here the error
            } else {
                // Save that we have subscribed successfully to keep track and avoid trying to subscribe again
                [[NSUserDefaults standardUserDefaults] setBool:YES forKey:@"subscribedToUpdates"];
                [[NSUserDefaults standardUserDefaults] synchronize];
            }
        }];
        }
    }

Вы получите уведомление в didReceiveRemoteNotification

Вот фрагмент кода


    - (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {

      CKNotification *cloudKitNotification = [CKNotification notificationFromRemoteNotificationDictionary:userInfo];
        if (cloudKitNotification.notificationType == CKNotificationTypeQuery) {
            CKQueryNotification *queryNotification = (CKQueryNotification *)cloudKitNotification;

            if (queryNotification.queryNotificationReason == CKQueryNotificationReasonRecordDeleted) {
                // If the record has been deleted in CloudKit then delete the local copy here

            } else {
                // If the record has been created or changed, we fetch the data from CloudKit
                CKDatabase *database;
                if (queryNotification.databaseScope) {
                    database = [[CKContainer containerWithIdentifier:@"your container identifier"] privateCloudDatabase];
                }

    [database fetchRecordWithID:queryNotification.recordID completionHandler:^(CKRecord * _Nullable record, NSError * _Nullable error) {
                    if (error) {
                        // Handle the error here
                    } else {

                  if (queryNotification.queryNotificationReason == CKQueryNotificationReasonRecordUpdated) {
                            // Use the information in the record object to modify your local data
    }else{
                            // Use the information in the record object to create a new local object

    }



    }
            }];
     }
    }

}

0 голосов
/ 06 июня 2019

Решение - это отдельный метод для версии используемой операции.Я уже получал уведомления, я просто не мог сказать, были ли они обновлены, созданы или удалены.Обновление и создание могут быть обработаны простым поиском в основных данных для recordName (которое является UUID).Если найдено, то отредактируйте, если не создайте.Проблема заключается в удалении - использование fetchRecordZoneChangesCompletionBlock не может идентифицировать удаления.Однако у семейства операций есть метод только для сообщения об удалениях - operation.recordWithIDWasDeletedBlock.Я изменил прежний код и добавил код удаления, как показано ниже.

Моя подписка на одну базу данных охватывает всю частную базу данных, поэтому нет необходимости подписываться на каждый тип записи.

operation.fetchRecordZoneChangesCompletionBlock = { error in
        for record in changedPatients {

            //search for the record in coredata
            if self.isSingleCoreDataRecord(ckRecord: record) {
                //if found - then modify
                self.saveUpdatedCloudKitRecordToCoreData(record: record)
            } else {
                //else add new
                self.saveCKRecordToCoreData(record: record)
            }

        }//for record in

        completionHandler(error)

}//fetchRecordZoneChangesCompletionBlock


operation.recordWithIDWasDeletedBlock = { (recordID, recordType) in

        //delete the core data record here
        let ckRecordToDelete = CKRecord(recordType: recordType, recordID: recordID)

        self.removeOnePatientRecordFromCoreData(ckRecord: ckRecordToDelete)

}//recordWithIDWasDeletedBlock
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...