How can I debug NSObject observers? - objective-c

Let's say I've got a Player instance, and I want to add an observer to their score so I can update the score display in my iOS application. I believe I should be calling a method that looks like this to add the observer:
[[self.game player] addObserver:self
forKeyPath:#"score"
options:NSKeyValueObservingOptionNew
context:nil];
In the same class, I then have a method defined that should receive this message:
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
However, the breakpoint that I have placed in this method never gets hit.
I would like to know how to diagnose this issue. I have verified that player instance is filled, and when the score is updated, the same instance is updated.
If it helps, I have just converted my project to ARC, but this code above is new and has never worked.

Either your key path "score" is not KVO compliant (are you not defining a #property for *score?) or the adding of the observer method is never being called. You can easily diagnose the latter by placing a breakpoint onto the line of code in your first snippet, and verifying that an observer indeed is being registered.

Related

When using NSSearchField , secondary NSTableView looses focus and do not update simultaneously

I am using core data binding in cocoa app. My app has three NSTableViews bound to there NSArrayControllers in relationship such that if first table is selected, relevant data using Core data relationship is shown in the other two tables.
I have hooked NSSearchField to first NSArrayController. The problem is when search begins the other two tables do not remain in sync with the first one.They show nothing at all.
The other two tables only shows content when first one if clicked by user.
How can I make all three NSTables update simultaneously when search occurs.
So, I have this solution which works, I would like to know if this can be improved further.
Since when search starts the first NSTableView's selection is lost and hence the connected tables too, although the table is updated correctly showing filtered value. I am setting self as observer for change is NSArrayController selectedObject.
-(void)awakeFromNib{
[super awakeFromNib];
[_firstArrayController addObserver:self forKeyPath:#"selectedObjects" options:NSKeyValueObservingOptionNew|NSKeyValueObservingOptionOld context:NULL];
}
Now I am observing for changes and if selected object do change and if NSSearchField is the one changing it, programatically select NSTableView first row
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)
object change:(NSDictionary *)change context:(void *)context
{
NSResponder *firstResponder = [[NSApp keyWindow] firstResponder];
if ([firstResponder isKindOfClass:[NSText class]] && [(id)firstResponder delegate] == _searchField) {
NSIndexSet *indexSet = [NSIndexSet indexSetWithIndex:0];
[_authorTable selectRowIndexes:indexSet byExtendingSelection:NO];
}
}
Since when filtering using NSSearchField, the first object is what the most relevant, hence I am setting selection to row 0.

OS X Yosemite window style mask NSFullSizeContentViewWindowMask cause only clicked button of collection view can call drawRect

My environment is OS X Yosemite 10.10.2.
Xcode are 6.1.1 and 6.2 beta4.
I wrote an calendar app for OS X. I used a cocoa bindings framework that I wrote. And I got
a problem. There should be only one focus day button in the collection view. But something
wrong. The app has multiple focuses.
I found a statement cause this problem.
[[[self view] window] setStyleMask:([[self view] window].styleMask | NSFullSizeContentViewWindowMask)];
When app sets the NSFullSizeContentViewWindowMask, only the clicked button of collection
view will call drawRect because of performance. This caused my problem.
- (void)drawRect:(NSRect)dirtyRect
The related post in Apple developer forum:https://devforums.apple.com/message/1097305#1097305
I found a solution. In the "- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context;" of button add "[self setNeedsDisplay:YES
];"
The original tutorial of cocoa bindings, observeValueForKeyPath action has [self
setNeedsDisplay :YES]; But I don't remember that why I commented it. When
NSFullSizeContentViewWindowMask not set, the app just got only one focus.

Changing UITextField in UITableViewCell value does not trigger KVO in iOS 7

I have a UITableView that has cells containing text fields. As the user edits each text field, I need to keep track of the cell's text field value (even when the cell is no longer displayed, so simply tracking the text field does not work). So I used Key-Value Observing and set an observer for each text field in my UITableViewController subclass:
[cell addObserver:self forKeyPath:#"textField.text" options:NSKeyValueObservingOptionNew context:nil];
And save new changes to the text field in the class's observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context method.
This works great in iOS 6, but in iOS 7 KVO isn't called when the user changes the text field value. Apparently the setter method called by pressing "done" on the keyboard doesn't call KVO anymore.
Is there a workaround to this? Or a better way of listening for this change? I need to know what cell the text field belongs to, so implementing the text field's method editingDidEnd doesn't work for me.
Thanks in advance.
I'm not sure about whether there is a workaround for this or not but there surely is a better way of listening for the UITextField text value. If you are creating instances of UITextField programmatically, all you need to do is add an event handler for them to notify your class about the text changing events. For example:
[myTextField addTarget:self action:#selector(editingChanged:) forControlEvents:UIControlEventEditingChanged];
- (void)editingChanged:(UITextField *)sender {
NSString *targetText = sender.text;
}

Getting the UITextField that's associated with my observer

I've got an observer on my UItextFields. It basically looks to see if the "enabled" property has changed.
If the enabled status changes, I want a method called fade to run. Fade requires a TextField to be passed to it.
How do I use the textField associated with the Observer that sends the message?
- (void) observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change: (NSDictionary *)change context:(void *) context;
{
int new = [change objectForKey:NSKeyValueChangeNewKey];
int old = [change objectForKey:NSKeyValueChangeOldKey];
if (new != old)
{
[self fadeEnable:"requires txtField"];
}
}
Thanks
As shown in the Docs,
https://developer.apple.com/library/mac/#documentation/Cocoa/Reference/Foundation/Protocols/NSKeyValueObserving_Protocol/Reference/Reference.html, the parameter 'Object' in the method call encapsulates your textfield.
you just have to cast it and youre done.
To cast it, use:
UITextField *tf = (UITextField *)object;
Edit to address your follow-up Question:
In The case of a Gesture recognizer, you want to get the view that you added the recognizer to. The 'sender' being the recognizer itself, sender.view will get you the associated view.
None of this will ever change, because recognizers only work with VIEWS.
Now, with an observer, you do not necessarily want a view. You want an OBJECT. Thats why 'object' is of type 'id', so you can add observers to objects of arbitrary types. The downside is, of course, that you need a cast. If you dont know what type to cast to, the
-isKindOfClass:
method might be useful.
Oh, and none of this is about interface-builder versus doing stuff in code.
If you find this confusing, i suggest you go study the docs on gestureRecognizers and Key-Value-Observing. If you have specific questions, let me know, or just ask an new question ;)

How to observe changes to an NSTextStorage

I have an NSTextStorage delegate imlementing textStorageDidProcessEditing:. Unfortunately, the userInfo doesn't contain any details about the changes to the text.
How can I find out what actually changed?
Ideally I'd like to know everything sent to [NSLayoutManager textStorage:edited:range:changeInLength:invalidatedRange:], but just the edited range would be good enough.
You could just write your own subclass of NSLayoutManager and add it to your NSTextSTorage using addLayoutManager:.
You could also use KVO to observe the characters property of your NSTextStorage object (or any other property of it) and you get exactly what is changed in
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
Use NSTextStorage's getter methods editedRange and changeInLength.