Xcode and code signing

As an iOS developer who is member of multiple teams and who is managing multiple clients Apple IDs, I totally approve the guidelines of Jared Sinclair in its post “Follow These Guidelines and Never Struggle with Xcode Code Signing Again“, more particularly the one about Xcode helpers:

Never use Xcode’s built-in code signing helpers. Especially avoid any button that says Fix Issue.

As Jared explains, it fills developer portal with worthless “iOS Team Provisioning Profile: XXX” provisioning profiles. Once sync in Xcode, there are so many profiles that you can’t find the one you are really looking for. The worst part of it is that this helper isn’t always working as you expect, especially if you don’t know what you are doing, and it arrives that you still can’t deploy on device after using it…

I will add that you shouldn’t be the one in your team touching the code signing settings if you don’t know how the whole code signing system works.

Understanding fonts and UIFont

For the first time in years, I had to dig inside one of the fundamental things we use everyday and everywhere: fonts.

Do you know how a fond is built? Applied to iOS, do you know how a UILabel will render a font?

First of all, a font is basically a container for vectors representing symbols from different categories such as letters, numbers, punctuation and other characters you can usually type with your keyboard. The more existing symbol a font contains, the more complete it is.

Within a font editor (Glyphs, Fontlab or whatever software allowing you to create and / or edit fonts) and for each symbol, the font “creator” draws vectorial paths according to some kind of grid. In iOS, UIFont’s grid looks like:

UIFont explained
UIFont as explained in Apple Developer documentation

The baseline if the base of all symbols. Then you have to distinguish the cases between minuscule letters with the x-height (X-height in the previous figure) since the minus “x” is supposed to be a perfect lower character; and the capital letters having their theoretical height represented in cap height.

Space on top of the baseline to the higher symbol Y offset is called the ascent or ascender. Space under the baseline and used for letter legs is called the descent or descender. There is more space under descent, it represents a gap to add between lines.

Adding the ascent to the descent and the line gap results in the line height which is basically the height of a single line of text.

That explains a lot! Did you ever notice that a label vertically centered in a view is not perfectly in the middle of the view?

It looks good...
It looks good…
... but if you zoom
… but if you zoom
...and it's even more visible if you add a background color to the label
…and it’s even more visible if you add a background color to the label

This is not a bug, just the way a font is rendered in a label. The extra space here is just the combination of both ascender and descender, and this space could be use (even if it is not in this case) to display accents, cedillas or other things that will be drew outside of the cap height bounding box.

Now, how can we respect a graphical chart dictating exact point size spacing between labels starting from the exact bottom of a label and ending to the exact top of another one?

chart
yeah, that’s always easier to say for designers when they are in Photoshop or InDesign…

So now, I have to work on a solution, the basic idea behind it is to subclass the UILabel class to have it read the metrics of UIFont’s and know its exact height. Even if it’s not working perfectly for every font (especially for some custom fonts that seems to be badly created) at the moment, the result is promising!

Promising results using the fitting label solution
Promising results using the fitting label solution

 

Core Data Stack

Recently, I’ve read this article Marcus Zarra about the base Core Data stack he uses in its projects. I couldn’t agree more about some topics discussed by Marcus:

The Cora Data stack doesn’t belong to the App Delegate

I do not create the Core Data stack in the App Delegate. That is not where it belongs. I create it in its own controller class and I store it in the cloud (check here) but dealing with data goes beyond that so if you start dealing with this type of data, I’d suggest checking this helpful resource first. Our persistence layer is far too important to be delegated to the Application Delegate. It deserves a proper controller of its own.

This is not only true about the Cora Data stack, but this is a very good example since Apple project templates using Cora Data are putting everything into the base AppDelegate class; most developers never question that fact and continue to think that the AppDelegate is some sort of garbage where they should write everything and anything.

Dependency Injection

The proper way to get access to the persistence controller is to inject it into each view controller as they are built and accessed.

If there is something I wish I could never see or write again, this is probably this ugly MANAGED_OBJECT_CONTEXT macro pointing to something like:

((AppDelegate *)[[UIApplication sharedApplication] delegate]) managedObjectContext;

The dependency injection is the solution, unfortunately, this technique is pretty annoying to use since the easier way is to do it is to inject our persistence controller during controllers and view controllers initialization…

This apart, I truly think dependency injection can help when it comes to testing. My approach for that will be very slightly different since I would probably prefer a different initializer for the PersistenceController class, with something more like:

/**
 Default initializer
 @param storeType Core data store type (eg: NSSQLiteStoreType or NSInMemoryStoreType)
 @param modelName Name of the managed object model file
 @param dataStoreName Name for the data store file that will be written on disk
 @param callback A block to be called once persistence stack is ready to be used
 @return An instance of PersistenceController class
*/
- (instancetype)initWithStoreType:(NSString *)storeType modelName:(NSString *)modelName dataStoreName:(NSString *)dataStoreName callback:(void (^)(void))callback
{
  if (self = [super init])
  {
     // Hold parameters
     _storeType = storeType;
     _modelName = modelName;
     _dataStoreName = dataStoreName;
     _initCallback = callback;

     // Initialize Core Data stack
     [self initializeCoreData];
   }
   return self;
}

This allows in-memory data stores that can be created for testing purpose, just make sure to fully protect it with a service like the one offered at https://www.couchbase.com/products/cloud.

However, I’m really looking for a way to enjoy the benefits of dependency injection without the struggle of passing my persistence controller again and again. This is where I go for now, any suggestion will of course be welcomed! For more information on data management, visit https://www.venyu.com/cloud/.

The original post discussed here can be found here: http://martiancraft.com/blog/2015/03/core-data-stack