Join the social network of Tech Nerds, increase skill rank, get work, manage projects...
 
  • How to use Core Data in iOS Part 2?

    • 0
    • 0
    • 0
    • 0
    • 0
    • 0
    • 0
    • 0
    • 598
    Comment on it

    Hi Readers!


    This blog includes the advance topic of Core Data which is relationship between two entities. If you are not familiar with the concept of Core Data you can view the link to understand the Core Data (Part 1).  Link include the information about how to create entity and to perform insert,delete,update operations in the same.

     

    In 2nd part of this blog, we will learn how to implement relationship with the help of Core Data. Please download the sample project from the link as we are going to use the same project to create relationship.

    Firstly, we will create another entity named as Company, once it is created we will create a relationship between the Employee entity and new created entity Company. Below are the screenshots of creation of new entity and simultaneously creating a relationship.

     

    Relationship which we are going to include in this application is many to one. As a single company can have any number of employees so the relationship between Employee entity and Company entity is many to one. So while creating relationship always explain the relationship in terms of suitable names. As in below snapshot it is clearly visible that relationship column include belongsToCompany name, destination means the entity with whom relationship is applicable and inverse include the property names as hasEmployees. In this way we can create relationship between two entities. 

     

     

     

    Here in another entity which is destination will automatically get the filled information of relationship as it is already connected from previous entity.

     


     

     

     

     

     

    Now the changes done in storyboard will be shown with the help of screenshots. Changes in previous controllers are as follows:-

     

    In EmployeeListViewController an another button is added on the top to add new company.

     

     

     

    In AddCompanyViewController new company can be added which will be available at the time of adding employee as which employee belongs to which company.

     

     

     

    In AddEmployeeDetailViewController an another textfield  is added which provides a list of company with the help of picker.

     

     

     

     

    Now the whole view of storyboard will be visible in this way as shown in below screenshot.

     

     

     

     

    Now coding part will be started from here. As we are creating a new entity so we have to create new subclass also as created
    in previous part. As we are doing changes in Employee entity also so we need to create subclasses again so delete the previous subclasses of Employee entity and create new subclasses of both entity. So after creation of subclasses all properties will be shown in the following way:-

    Subclasses of Employee entity is as follows:-
     

     

    1. Employee.h
    2.  
    3. #import <Foundation/Foundation.h>
    4. #import <CoreData/CoreData.h>
    5.  
    6. @class Company;
    7.  
    8. NS_ASSUME_NONNULL_BEGIN
    9.  
    10. @interface Employee : NSManagedObject
    11.  
    12. // Insert code here to declare functionality of your managed object subclass
    13.  
    14. @end
    15.  
    16. NS_ASSUME_NONNULL_END
    17.  
    18. #import "Employee+CoreDataProperties.h"
    19.  
    20.  
    21. Employee.m
    22.  
    23. #import "Employee.h"
    24. #import "Company.h"
    25.  
    26. @implementation Employee
    27.  
    28. // Insert code here to add functionality to your managed object subclass
    29.  
    30. @end
    31.  
    32. Employee+CoreDataProperties.h
    33.  
    34. #import "Employee.h"
    35.  
    36. NS_ASSUME_NONNULL_BEGIN
    37.  
    38. @interface Employee (CoreDataProperties)
    39.  
    40. @property (nullable, nonatomic, retain) NSString *employee_Designation;
    41. @property (nullable, nonatomic, retain) NSString *employee_Name;
    42. @property (nullable, nonatomic, retain) NSString *employee_Salary;
    43. @property (nullable, nonatomic, retain) Company *belongsToCompany;
    44.  
    45. @end
    46.  
    47. NS_ASSUME_NONNULL_END
    48.  
    49. Employee+CoreDataProperties.m
    50.  
    51. #import "Employee+CoreDataProperties.h"
    52.  
    53. @implementation Employee (CoreDataProperties)
    54.  
    55. @dynamic employee_Designation;
    56. @dynamic employee_Name;
    57. @dynamic employee_Salary;
    58. @dynamic belongsToCompany;
    59.  
    60. @end

     

    Subclasses of Company entity is as follows:-

     

    1. Company.h
    2.  
    3. #import <Foundation/Foundation.h>
    4. #import <CoreData/CoreData.h>
    5.  
    6. @class Employee;
    7.  
    8. NS_ASSUME_NONNULL_BEGIN
    9.  
    10. @interface Company : NSManagedObject
    11.  
    12. // Insert code here to declare functionality of your managed object subclass
    13.  
    14. @end
    15.  
    16. NS_ASSUME_NONNULL_END
    17.  
    18. #import "Company+CoreDataProperties.h"
    19.  
    20. Company.m
    21.  
    22. #import "Company.h"
    23. #import "Employee.h"
    24.  
    25. @implementation Company
    26.  
    27. // Insert code here to add functionality to your managed object subclass
    28.  
    29. @end
    30.  
    31. Company+CoreDataProperties.h
    32.  
    33. #import "Company.h"
    34.  
    35. NS_ASSUME_NONNULL_BEGIN
    36.  
    37. @interface Company (CoreDataProperties)
    38.  
    39. @property (nullable, nonatomic, retain) NSString *company_Name;
    40. @property (nullable, nonatomic, retain) NSSet<Employee *> *hasEmployees;
    41.  
    42. @end
    43.  
    44. @interface Company (CoreDataGeneratedAccessors)
    45.  
    46. - (void)addHasEmployeesObject:(Employee *)value;
    47. - (void)removeHasEmployeesObject:(Employee *)value;
    48. - (void)addHasEmployees:(NSSet<Employee *> *)values;
    49. - (void)removeHasEmployees:(NSSet<Employee *> *)values;
    50.  
    51. @end
    52.  
    53. NS_ASSUME_NONNULL_END
    54.  
    55. Company+CoreDataProperties.m
    56.  
    57. #import "Company+CoreDataProperties.h"
    58.  
    59. @implementation Company (CoreDataProperties)
    60.  
    61. @dynamic company_Name;
    62. @dynamic hasEmployees;
    63.  
    64. @end

     

     

    Modification in StringDeclaration class according to addition of new strings.

     

    1. StringDeclaration.h
    2.  
    3. #import <Foundation/Foundation.h>
    4.  
    5. @interface StringDeclaration : NSObject
    6. extern NSString *const cellIdentifier,*const name,*const designation,*const salary,*const companyName,*const empBelongsToCompany;
    7. @end
    8.  
    9.  
    10. StringDeclaration.m
    11.  
    12. #import "StringDeclaration.h"
    13.  
    14. @implementation StringDeclaration
    15. NSString * const cellIdentifier=@"cell",*const name=@"employee_Name",*const designation=@"employee_Designation",*const salary=@"employee_Salary",*const companyName=@"company_Name",*const empBelongsToCompany=@"belongsToCompany";
    16. @end
    17.  
    18.  

     

     

    New file for addition of new company will be done with the file named as AddCompanyViewController.


     

    1. AddCompanyViewController.h
    2.  
    3.  
    4. #import <UIKit/UIKit.h>
    5. #import "Company.h"
    6. #import <CoreData/CoreData.h>
    7. #import "StringDeclaration.h"
    8. @interface AddCompanyViewController : UIViewController
    9. - (IBAction)btnSaveCompany:(id)sender;
    10. @property (strong) NSManagedObject *device;
    11. @property (weak, nonatomic) IBOutlet UITextField *txtCompanyName;
    12. - (IBAction)btnBack:(id)sender;
    13.  
    14. @end
    15.  
    16. AddCompanyViewController.m
    17.  
    18. #import "AddCompanyViewController.h"
    19.  
    20. @interface AddCompanyViewController ()
    21. @property (strong) NSMutableArray *devices;
    22.  
    23. @end
    24.  
    25. @implementation AddCompanyViewController
    26. - (NSManagedObjectContext *)managedObjectContext {
    27. NSManagedObjectContext *context = nil;
    28. id delegate = [[UIApplication sharedApplication] delegate];
    29. if ([delegate performSelector:@selector(managedObjectContext)]) {
    30. context = [delegate managedObjectContext];
    31. }
    32. return context;
    33. }
    34. - (void)viewDidLoad {
    35. [super viewDidLoad];
    36. // Do any additional setup after loading the view.
    37. }
    38.  
    39. - (void)didReceiveMemoryWarning {
    40. [super didReceiveMemoryWarning];
    41. // Dispose of any resources that can be recreated.
    42. }
    43.  
    44.  
    45. - (IBAction)btnSaveCompany:(id)sender {
    46. NSManagedObjectContext *context = [self managedObjectContext];
    47.  
    48. Company *newDevice = [NSEntityDescription insertNewObjectForEntityForName:NSStringFromClass([Company class]) inManagedObjectContext:context];
    49. newDevice.company_Name=self.txtCompanyName.text;
    50.  
    51. NSError *error = nil;
    52. // Save the object to persistent store
    53. if (![context save:&error]) {
    54. NSLog(@"Can't Save! %@ %@", error, [error localizedDescription]);
    55. }
    56.  
    57. [self dismissViewControllerAnimated:YES completion:nil];
    58. }
    59. - (IBAction)btnBack:(id)sender {
    60. [self dismissViewControllerAnimated:YES completion:nil];
    61. }
    62. @end

     

     

    AddEmployeeDetailViewController file includes one more textfield which provides picker on click. Picker includes the list of company where employee can be added.

     

    1. AddEmployeeDetailViewController.h
    2.  
    3. #import <UIKit/UIKit.h>
    4. #import <CoreData/CoreData.h>
    5. #import "Employee.h"
    6.  
    7. @interface AddEmployeeDetailViewController : UIViewController<UIPickerViewDataSource,UIPickerViewDelegate>
    8. {
    9. UIPickerView *picker;
    10. UIButton *done;
    11. }
    12. @property (weak, nonatomic) IBOutlet UITextField *txtEmployeeName;
    13. @property (weak, nonatomic) IBOutlet UITextField *txtEmployeeDesignation;
    14. @property (weak, nonatomic) IBOutlet UITextField *txtEmployeeSalary;
    15. - (IBAction)btnSave:(id)sender;
    16. - (IBAction)btnCancel:(id)sender;
    17. @property (weak, nonatomic) IBOutlet UITextField *txtCompany;
    18.  
    19. @property (strong) NSManagedObject *employeeObject;
    20. @property (strong, nonatomic)NSArray *dataSourceArray;
    21. @end
    22.  
    23.  
    24. AddEmployeeDetailViewController.m
    25.  
    26. #import "AddEmployeeDetailViewController.h"
    27. #import "StringDeclaration.h"
    28. #import "Company.h"
    29. @interface AddEmployeeDetailViewController (){
    30. Company *selectedCompany;
    31. }
    32. @property (strong) NSMutableArray *companyArray;
    33. @end
    34.  
    35. @implementation AddEmployeeDetailViewController
    36. - (NSManagedObjectContext *)managedObjectContext {
    37. NSManagedObjectContext *context = nil;
    38. id delegate = [[UIApplication sharedApplication] delegate];
    39. if ([delegate performSelector:@selector(managedObjectContext)]) {
    40. context = [delegate managedObjectContext];
    41. }
    42. return context;
    43. }
    44.  
    45.  
    46. - (void)viewDidLoad {
    47. [super viewDidLoad];
    48. if (self.employeeObject) {
    49. [self.txtEmployeeName setText:[self.employeeObject valueForKey:name]];
    50. [self.txtEmployeeDesignation setText:[self.employeeObject valueForKey:designation]];
    51. [self.txtEmployeeSalary setText:[self.employeeObject valueForKey:salary]];
    52. }
    53.  
    54. }
    55.  
    56.  
    57. - (void)viewDidAppear:(BOOL)animated
    58. {
    59. [super viewDidAppear:animated];
    60. [self data];
    61. // Fetch the devices from persistent data store
    62. }
    63. -(void)data
    64. {
    65. NSManagedObjectContext *managedObjectContext = [self managedObjectContext];
    66. NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] initWithEntityName:@"Company"];
    67. self.companyArray = [[managedObjectContext executeFetchRequest:fetchRequest error:nil] mutableCopy];
    68. NSLog(@"%@",self.companyArray);
    69. picker = [[UIPickerView alloc] init];
    70. [done setTitle:@"Done" forState:UIControlStateNormal];
    71. picker.dataSource = self;
    72. picker.delegate = self;
    73. [_txtCompany setInputView:picker];
    74. _dataSourceArray = [self.companyArray valueForKey:companyName];
    75. UIToolbar *toolBar = [[UIToolbar alloc] initWithFrame:CGRectMake(0, 0, 320, 44)];
    76. toolBar.barStyle = UIBarStyleBlackOpaque;
    77. UIBarButtonItem *doneButton = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemDone target:self action:@selector(doneTouched:)];
    78. [toolBar setItems:[NSArray arrayWithObjects:doneButton, nil]];
    79. _txtCompany.inputAccessoryView = toolBar; // include list of company
    80. }
    81. - (void)doneTouched:(UIBarButtonItem *)sender
    82. {
    83. [_txtCompany resignFirstResponder]; // hide the picker view
    84. }
    85. // Do any additional setup after loading the view.
    86. - (NSInteger)numberOfComponentsInPickerView: (UIPickerView *)pickerView
    87. {
    88. return 1;
    89. }
    90. - (NSInteger)pickerView:(UIPickerView *)pickerView numberOfRowsInComponent:(NSInteger)component
    91.  
    92. {
    93. return self.companyArray.count; //returns no of rows
    94. }
    95. - (NSString *)pickerView:(UIPickerView *)pickerView titleForRow:(NSInteger)row forComponent:(NSInteger)component
    96. {
    97. Company *company = self.companyArray[row];
    98. return company.company_Name;
    99.  
    100. }
    101. -(void)pickerView:(UIPickerView *)pickerView didSelectRow:(NSInteger)row inComponent:(NSInteger)component
    102. {
    103. Company *company = self.companyArray[row];
    104. selectedCompany = company;
    105. _txtCompany.text = company.company_Name;
    106. if(_txtCompany.text>0)
    107. {
    108. }
    109. else
    110. {
    111. [_txtCompany setInputView:picker];
    112. }
    113. }
    114. - (BOOL)textFieldShouldBeginEditing:(UITextField *)textField
    115. {
    116. return YES;
    117. }
    118. - (void)didReceiveMemoryWarning {
    119. [super didReceiveMemoryWarning];
    120. // Dispose of any resources that can be recreated.
    121. }
    122.  
    123. - (IBAction)btnSave:(id)sender {
    124.  
    125. NSManagedObjectContext *context = [self managedObjectContext];
    126. if (self.employeeObject) {
    127. // Update existing device
    128. [self.employeeObject setValue:self.txtEmployeeName.text forKey:name];
    129. [self.employeeObject setValue:self.txtEmployeeDesignation.text forKey:designation];
    130. [self.employeeObject setValue:self.txtEmployeeSalary.text forKey:salary];
    131. [self.employeeObject setValue:selectedCompany forKey:empBelongsToCompany];
    132. } else {
    133. // Create a new device
    134. Employee *newDevice = [NSEntityDescription insertNewObjectForEntityForName:NSStringFromClass([Employee class]) inManagedObjectContext:context];
    135. newDevice.employee_Name=self.txtEmployeeName.text;
    136. newDevice.employee_Designation=self.txtEmployeeDesignation.text;
    137. newDevice.employee_Salary=self.txtEmployeeSalary.text;
    138. newDevice.belongsToCompany=selectedCompany;
    139. NSError *error = nil;
    140. // Save the object to persistent store
    141. if (![context save:&error]) {
    142. NSLog(@"Can't Save! %@ %@", error, [error localizedDescription]);
    143. }
    144. }
    145. [self dismissViewControllerAnimated:YES completion:nil];
    146. }
    147.  
    148. - (IBAction)btnCancel:(id)sender {
    149. [self dismissViewControllerAnimated:YES completion:nil];
    150. }
    151.  
    152.  
    153. @end


     


    Now in the controller which display the list of employees can fetch the company name also which is saved in another entity named as Company. We are able to access the value from another entity because of relationship between the two entities otherwise it is not possible. So below is the code which will help in accessing the data store in another entity.

     

     

    1. EmployeeListViewController.h
    2.  
    3. #import <UIKit/UIKit.h>
    4.  
    5. @interface EmployeeViewController : UIViewController
    6. - (IBAction)btnAddEmployee:(id)sender;
    7.  
    8. @property (weak, nonatomic) IBOutlet UITableView *tblView;
    9.  
    10. - (IBAction)addCompany:(id)sender;
    11.  
    12. @end
    13.  
    14. EmployeeListViewController.m
    15.  
    16. #import "EmployeeViewController.h"
    17. #import <CoreData/CoreData.h>
    18. #import "AddEmployeeDetailViewController.h"
    19. #import "StringDeclaration.h"
    20. #import "AddCompanyViewController.h"
    21. @interface EmployeeViewController ()
    22. @property (strong) NSMutableArray *employeeArray;
    23. @end
    24.  
    25. @implementation EmployeeViewController
    26.  
    27. - (NSManagedObjectContext *)managedObjectContext
    28. {
    29. NSManagedObjectContext *context = nil;
    30. id delegate = [[UIApplication sharedApplication] delegate];
    31. if ([delegate performSelector:@selector(managedObjectContext)]) {
    32. context = [delegate managedObjectContext];
    33. }
    34. return context;
    35. }
    36.  
    37. - (void)viewDidLoad {
    38. [super viewDidLoad];
    39. // Do any additional setup after loading the view.
    40. }
    41.  
    42. - (void)didReceiveMemoryWarning {
    43. [super didReceiveMemoryWarning];
    44. // Dispose of any resources that can be recreated.
    45. }
    46.  
    47. - (void)viewDidAppear:(BOOL)animated
    48. {
    49. [super viewDidAppear:animated];
    50. [self data];
    51. [self.tblView reloadData];
    52. // Fetch the devices from persistent data store
    53. }
    54. -(void)data
    55. {
    56. NSManagedObjectContext *managedObjectContext = [self managedObjectContext];
    57. NSFetchRequest *fetchRequest = [[NSFetchRequest alloc] initWithEntityName:@"Employee"];
    58. self.employeeArray = [[managedObjectContext executeFetchRequest:fetchRequest error:nil] mutableCopy];
    59. }
    60.  
    61. -(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
    62. {
    63. // Return the number of sections.
    64. return 1;
    65. }
    66.  
    67. - (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
    68. {
    69. // Return the number of rows in the section.
    70. return self.employeeArray.count;
    71. }
    72.  
    73. - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
    74. {
    75. UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:cellIdentifier forIndexPath:indexPath];
    76. // Configure the cell...
    77. NSManagedObject *employee = [self.employeeArray objectAtIndex:indexPath.row];
    78. Company *company=[employee valueForKey:empBelongsToCompany];
    79. [cell.textLabel setText:[NSString stringWithFormat:@"%@ %@ %@", [employee valueForKey:name], [employee valueForKey:designation],company.company_Name]];//fetch the data from other entity(table)
    80. [cell.detailTextLabel setText:[employee valueForKey:salary]];
    81. return cell;
    82. }
    83.  
    84. - (BOOL)tableView:(UITableView *)tableView canEditRowAtIndexPath:(NSIndexPath *)indexPath
    85. {
    86. // Return NO if you do not want the specified item to be editable.
    87. return YES;
    88. }
    89.  
    90. - (void)tableView:(UITableView *)tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath *)indexPath
    91. {
    92. NSManagedObjectContext *context = [self managedObjectContext];
    93. if (editingStyle == UITableViewCellEditingStyleDelete) {
    94. // Delete object from database
    95. [context deleteObject:[self.employeeArray objectAtIndex:indexPath.row]];
    96. NSError *error = nil;
    97. if (![context save:&error]) {
    98. NSLog(@"Can't Delete! %@ %@", error, [error localizedDescription]);
    99. return;
    100. }
    101. // Remove device from table view
    102. [self.employeeArray removeObjectAtIndex:indexPath.row];
    103. [self.tblView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
    104. }
    105. }
    106.  
    107.  
    108. - (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender
    109. {
    110. if ([[segue identifier] isEqualToString:@"Update"]) {
    111. NSManagedObject *selectedDevice = [self.employeeArray objectAtIndex:[[self.tblView indexPathForSelectedRow] row]];
    112. AddEmployeeDetailViewController *destViewController = segue.destinationViewController;
    113. destViewController.employeeObject = selectedDevice;
    114. }
    115. }
    116.  
    117. - (IBAction)btnAddEmployee:(id)sender {
    118. UIStoryboard* sb = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
    119. AddEmployeeDetailViewController* vc1 = [sb instantiateViewControllerWithIdentifier:NSStringFromClass([AddEmployeeDetailViewController class])];
    120.  
    121. [vc1 setModalTransitionStyle:UIModalTransitionStyleFlipHorizontal];
    122. [self presentViewController:vc1 animated:YES completion:nil];
    123. }
    124. - (IBAction)addCompany:(id)sender {
    125. UIStoryboard* sb = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
    126. AddCompanyViewController* vc1 = [sb instantiateViewControllerWithIdentifier:NSStringFromClass([AddCompanyViewController class])];
    127. [vc1 setModalTransitionStyle:UIModalTransitionStyleFlipHorizontal];
    128. [self presentViewController:vc1 animated:YES completion:nil];
    129. }
    130. @end

     


    Output:-

     

     

     

     

     

     

     

    You can download the project from the given link. You need Xcode 7.2 to build the project.

     

 0 Comment(s)

Sign In
                           OR                           
                           OR                           
Register

Sign up using

                           OR                           
Forgot Password
Reset Password
Fill out the form below and reset your password: