UISwitch setThumbTintColor causing crash (iOS 6 only)?
Asked Answered
D

3

18

UPDATE: Got a mail from Apple saying that the bug/issue has been fixed now and the next SDK release won't have this issue. Peace!

I have this in the code for my AppDelegate:

- (void) customizeAppearance {
    [[UISwitch appearance] setOnTintColor:[UIColor colorWithRed:0 green:175.0/255.0 blue:176.0/255.0 alpha:1.0]];
    [[UISwitch appearance] setTintColor:[UIColor colorWithRed:255.0f/255.0f green:255.0f/255.0f blue:255.0f/255.0f alpha:1.000f]];
    [[UISwitch appearance] setThumbTintColor:[UIColor colorWithRed:0.9 green:0.9 blue:0.9 alpha:1.0]];
 }

Which I then call from - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions

I also use ARC. In iOS 6 my app keeps crashing. I enabled NSZombie and it keeps saying: *** -[UIDeviceRGBColor release]: message sent to deallocated instance 0x9658eb0

And now I've realized one perfectly reproducible flow for the above. When I comment out the setThumbTintColor line alone inside customizeAppearance, then everything works fine as it should. When I use the setThumbTintColor line instead, the app crashes the exact same way every time.

Is this a known issue to anyone with UISwitch/setThumbTintColor/UIColor? What else could be the cause if not the switch color?

Demoralize answered 19/10, 2012 at 10:37 Comment(10)
I don't know why this happens, but if you can reproduce this in a basic example app, you might want to post a bug at bugreport.apple.comAdscititious
See raywenderlich.com/21703/user-interface-customization-in-ios-6 . Doesn't crash in his sample app. I think this is something else. What I'm messing up in my app, I've no idea.Demoralize
When does it crash? On calling this lines from your post? If you call this in -application:didFinishLaunchingWithOptions: it should be easy to find as there isn't much that has been executed by now. If it crashes somewhere else, it might be interesting to see some of the surrounding code.Adscititious
The code posted here works fine. The problem must be somewhere else. Did you try to set an exception breakpoint?Puebla
This still exists. No activity on the filed radar yet.Demoralize
Any update on the radar? I'm able to reproduce it consistently.Autoicous
Most likely, Apple's code includes a weak reference that should be strong, because there's no error when thumbTintColor is set to [UIColor redColor];Autoicous
Here's the radar that I had filed long ago. Dupe it if you're also facing this issue. rdar://12562867Demoralize
Bug fixed in the next release according to Apple.Demoralize
Thank you very much. This linked me to the UISwitch Bug, otherwise I had to search for ages.Arvell
M
19

I was also doing this tutorial and had the same problem. (Not sure why you don't experience this, as both my hand typed in code and the solution code have the same problem for me?)

The first segue would happen ok, but after going back the next segue would fail.

After setting a global exception breakpoint I could see thumbColorTint in the call stack when the exception was generated. I made a guess that the object was being released too early. To fix I created a property in my app delegate..(you don't need to do it in the app delegate just the object that you are setting the UISwitch appearance, which in my case was the appdelegate)

@interface SurfsUpAppDelegate()
@property (strong, nonatomic) UIColor *thumbTintColor;
@end

Then I set it up as so

[self setThumbTintColor:[UIColor colorWithRed:0.211 green:0.550 blue:1.000 alpha:1.000]];
[[UISwitch appearance] setThumbTintColor:[self thumbTintColor]];

And now everything works as expected as the object is not released early. This is probably a defect and the object is released even though it is still needed. UISwitch seems to have a defect for the API :(

Munmro answered 22/10, 2012 at 18:48 Comment(4)
This solves the issue. Hence accepted. I also found one more workaround. Setting the switch's parent view to strong prevents the crash as well. I went ahead and filed a bug with a sample project anyway, as this never seems to happen for any other UI element other than a switch and that too only while setting the thumbTintColor property.Demoralize
I also have the same issue. Both Bill and Bourne's solutions work for me. However, I have another project which contains a navigation controller. 5 view controllers are in the stack and sometimes the view could be pushed/popped. When either scene contains the switch. Neither the above solutions works. I think we just wait Apple to fix this.Monica
This workaround doesn't seem to have any effect for me. My app crashes when switch instances are dealloc'd no matter how I set the thumbTintColor.Functional
Sorry it doesn't work in all cases. I suspect that in some cases it released correctly and others not (As Wayne indicates). A bug fix will resolve it once and for all. NONT makes a good suggestion, but this may fail when it is fixed. Memory issues like these are difficult to retro fix.Munmro
I
3

I also ran into this bug with Apple's UISwitch over-releasing. I have a similar solution, but I think its just a little bit nicer because it doesn't require the addition of a extraneous property:

UIColor *thumbTintColor =  [[UIColor alloc] initWithRed:red green:green blue:blue alpha:alpha]];

//we're calling retain even though we're in ARC,
// but the compiler doesn't know that

[thumbTintColor performSelector:NSSelectorFromString(@"retain")]; //generates warning, but OK
[[UISwitch appearance] setThumbTintColor:[self thumbTintColor]];

On the downside, it does create a compiler warning, but then - there really is a bug here, just not ours!

Inductive answered 19/4, 2013 at 1:53 Comment(0)
A
1

For now, I'm going with this per Bill's answer:

// SomeClass.m

@interface SomeClass ()

// ...

@property (weak,   nonatomic) IBOutlet UISwitch *thumbControl;
@property (strong, nonatomic)           UIColor *thumbControlThumbTintColor;

// ...

@end

@implementation SomeClass

// ...

- (void)viewDidLoad
{
    // ...

    self.thumbControl.thumbTintColor = self.thumbControlThumbTintColor = [UIColor colorWithRed:0.2 green:0.0 blue:0.0 alpha:1.0];

    // ...
}

// ...

@end
Autoicous answered 18/6, 2013 at 4:29 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.