Accéder au contenu principal

UISwitch dans un UITableViewCell





Je construis des formulaire dans des UITable, chaque UITableViewCell peut donc contenir des UILabel, UISwitch, UIButton, etc. Je partage le code entre plusieurs application, mais j'ai eu un bug dans l'une d'elles que je n'avais pas dans les autres. Les gadgets interactifs ne fonctionnaient pas...

En fait, j'avais construit mes UITableViewCell avec du code comme celui-ci :

-(instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier {

    self = [super initWithStyle:style reuseIdentifier:reuseIdentifier];

    self.backgroundColor = [UIColor clearColor];


    _uiLabel = [UILabel new];

    _uiLabel.frame = CGRectMake(8, 4, self.frame.size.width-51-24, self.frame.size.height-8);

    _uiLabel.translatesAutoresizingMaskIntoConstraints = NO;

    _uiLabel.textAlignment = NSTextAlignmentLeft;

    _uiLabel.lineBreakMode = NSLineBreakByWordWrapping;

    _uiLabel.adjustsFontSizeToFitWidth = YES;

    _uiLabel.minimumScaleFactor = 0.5;

    _uiLabel.numberOfLines = 0  ;

    _uiLabel.text = NSLocalizedString(@"none", nil);

    [_uiLabel  removeConstraints:_uiLabel.constraints];

    [self addSubview:_uiLabel];

    

    _uiSwitch = [UISwitch new];

    _uiSwitch.frame = CGRectMake(self.frame.size.width-_uiSwitch.frame.size.width-8,

                                 self.frame.size.height/2-16,

                                 _uiSwitch.frame.size.width,

                                 _uiSwitch.frame.size.height);

    _uiSwitch.translatesAutoresizingMaskIntoConstraints = NO;

    

    [_uiSwitch addTarget:self action:@selector(changeOn:) forControlEvents:UIControlEventValueChanged];


    [_uiSwitch removeConstraints:_uiSwitch.constraints];

    [self addSubview:_uiSwitch];



    NSDictionary *children =@{ @"label":_uiLabel,@"sw":_uiSwitch };

    [self addConstraints:[NSLayoutConstraint

                          constraintsWithVisualFormat:@"H:|-8-[label(>=160)]-8-[sw]-16-|"

                          options:0 metrics:nil views:children]];

    [self addConstraints:[NSLayoutConstraint

                          constraintsWithVisualFormat:@"V:|-8-[label]-8-|"

                          options:0 metrics:nil views:children]];

    [self addConstraints:[NSLayoutConstraint

                          constraintsWithVisualFormat:@"V:|-8-[sw]-8-|"

                          options:0 metrics:nil views:children]];


    [self setNeedsLayout];

    return self;

}


Mon erreur était dans l'appel de addSubView, il ne faut pas mettre les éléments directement dans le UITableViewCell, car la contentView peut alors se retrouver au-dessus et empêcher les interactions. 
Il faut donc mettre :


    [self.contentView addSubview:_uiLabel];

    [self.contentView addSubview:_uiSwitch];


Remplacer self par self.contentView règle le problème. On doit aussi s'occuper des addConstraints qui devraient aussi être mis à self.contentView.

    NSDictionary *children =@{ @"label":_uiLabel,@"sw":_uiSwitch };

    [self.contentView addConstraints:[NSLayoutConstraint 

     constraintsWithVisualFormat:@"H:|-8-[label(>=160)]-8-[sw]-16-|"

                         options:0 metrics:nil views:children]];

    [self.contentView addConstraints:[NSLayoutConstraint

     constraintsWithVisualFormat:@"V:|-8-[label]-8-|"

                         options:0 metrics:nil views:children]];

    [self.contentView addConstraints:[NSLayoutConstraint

     constraintsWithVisualFormat:@"V:|-8-[sw]-8-|"

                         options:0 metrics:nil views:children]];




Commentaires

Posts les plus consultés de ce blog

UISceneSession is only available on iOS 13

Lorsque vous faites une nouvelle application, celle-ci est par défaut pour le dernier IOS.  Mais si vos clients, comme les miens, ont de vieux appareils, vous pourriez vouloir que votre application fonctionne avec IOS9 (pour le moment on peut encore faire des apps IOS9, mais je crois que la limite sera IOS12 dans pas longtemps). Dans les infos de déploiement, vous changez donc le IOS 14  en IOS 9. Mais, le lancement de l'app vous donne un écran noir ! Pour régler le problème, il faut ajouter une propriété "window" dans l'interface du appDelegate. @interface AppDelegate : UIResponder < UIApplicationDelegate > @property ( strong , nonatomic ) UIWindow * window ; @end Reste les warnings... Le plus simple est de faire confiance à Xcode, tapez sur les warnings et un menu vous proposera un "fix" qui ajoutera un API_AVAILABLE à la définition de la fonction. - ( void ) scene :( UIScene *)scene willConnectToSession :( UISceneSession *)session options :(

_UISheetInteractionBackgroundDismissRecognizer

Avec IOS13 les FormSheet et les PageSheet peuvent être quittés par un geste de glissement vers le haut. Mais si votre UIViewController contient un élément qui utilise les déplacements de doigts (UITouch) alors cela ne fonctionne plus à cause de la gestion de ce geste. Pour désactiver le geste, utilisez le code suivant : -( void ) viewDidAppear :( BOOL )animated {     [ super viewDidAppear :animated];     for ( UIGestureRecognizer *gr in self . presentationController . presentedView . gestureRecognizers ) {         if ( @available (iOS 11.0 , *)) {             if ([gr. name isEqualToString : @"_UISheetInteractionBackgroundDismissRecognizer" ]) {                 // disable la possibilité de quitter la feuille FormSheet ou PageSheet avec un glissement vers le haut                 gr. enabled = false ;             }         }     } }

IOS14 et Bonjour qui ne fonctionne pas.

 J'utilise Bonjour dans certaines de mes application pour échanger des données entre appareils se trouvant sur le même réseau local.  Depuis IOS14, vous devez déclarer dans info.plist : <key> NSLocalNetworkUsageDescription </key> <string> L'accès au réseau local est obligatoire pour.... </string> <key> NSBonjourServices </key> <array> <string> _http._tcp </string> </array>  La clef NSLocalNetworkUsageDescription est obligatoire pour toute utilisation du réseau local (en dehors des produits Apple comme AirDrop, AirPrint, etc.). C'est la même chose que l'accès à la caméra ou au micro, vous devez dire pourquoi vous désirez cet accès dans votre application. La clef NSBonjourServices est spécifique pour le protocole Bonjour, elle contient la liste des services que vous partagez avec Bonjour. Dans l'exemple, il s'agit d'un serveur web (http).