it-swarm.com.ru

iOS7 UIS переключите его Event ValueChanged: непрерывный вызов - это ошибка или как? ..

Правка

Теперь это исправлено на ios7.1
Не делайте никаких твиков, чтобы исправить это.

Edit2

Видимо та же проблема повторяется в iOS 8.0 и 8.1

Edit3

Это теперь исправлено на ios9.2
Не делайте никаких твиков, чтобы исправить это.


Привет, сегодня я видел в UISwitch's Событии ValueChanged: Вызов continuously, пока я меняю On на Off или Off на On, и мой палец все еще двигался как на правой, так и на левой стороне. Я посетил GIF изображение для более четкого просмотра с NSLog.

enter image description here

Мой метод изменения стоимости:

- (IBAction)changeSwitch:(id)sender{

    if([sender isOn]){
        NSLog(@"Switch is ON");
    } else{
        NSLog(@"Switch is OFF");
    }

}

iOS6 тот же код Switch работает отлично, как мы ожидали:

enter image description here

так может кто-нибудь предложить мне, чтобы звонить только один раз его состояние вкл или выкл. или это баг или что ..?

ОБНОВЛЕНИЕ

Вот мое демо этого:

программный Добавить UISwitch

из XIB с добавлением UISwitch

91
Nitin Gohel

У меня есть много пользователей, которые сталкиваются с той же проблемой, так что, возможно, это ошибка UISwitch, поэтому я нашел только для временного решения этой проблемы. Я нашел один gitHub custom KLSwitch, используйте это на данный момент, надеюсь, Apple исправит это в следующем обновлении xCode: -

https://github.com/KieranLafferty/KLSwitch

7
Nitin Gohel

Пожалуйста, смотрите следующий код:

-(void)viewDidLoad
{
    [super viewDidLoad];    
    UISwitch *mySwitch = [[UISwitch alloc] initWithFrame:CGRectMake(130, 235, 0, 0)];    
    [mySwitch addTarget:self action:@selector(changeSwitch:) forControlEvents:UIControlEventValueChanged];
    [self.view addSubview:mySwitch];
}

- (void)changeSwitch:(id)sender{
    if([sender isOn]){
        NSLog(@"Switch is ON");
    } else{
        NSLog(@"Switch is OFF");
    }
}
44
AeroStar

Та же ошибка здесь. Я думаю, что нашел простой обходной путь. Нам просто нужно использовать новую BOOL, которая хранит предыдущее состояние UISwitch, и оператор if в нашей IBAction (Value Changed fired), чтобы проверить, что значение параметра действительно изменилось.

previousValue = FALSE;

[...]

-(IBAction)mySwitchIBAction {
    if(mySwitch.on == previousValue)
        return;
    // resetting the new switch value to the flag
    previousValue = mySwitch.on;
 }

Нет больше странного поведения. Надеюсь, поможет.

12
nnarayann

Вы можете использовать свойство .selectedUISwitch, чтобы убедиться, что ваш код выполняется только один раз, когда фактическое значение изменяется. Я думаю, что это отличное решение, потому что оно избегает необходимости создавать подклассы или добавлять новые свойства.

 //Add action for `ValueChanged`
 [toggleSwitch addTarget:self action:@selector(switchTwisted:) forControlEvents:UIControlEventValueChanged];

 //Handle action
- (void)switchTwisted:(UISwitch *)twistedSwitch
{
    if ([twistedSwitch isOn] && (![twistedSwitch isSelected]))
    {
        [twistedSwitch setSelected:YES];

        //Write code for SwitchON Action
    }
    else if ((![twistedSwitch isOn]) && [twistedSwitch isSelected])
    {
        [twistedSwitch setSelected:NO];

        //Write code for SwitchOFF Action
    }
}

И вот оно в Swift: 

func doToggle(switch: UISwitch) {
    if switch.on && !switch.selected {
        switch.selected = true
        // SWITCH ACTUALLY CHANGED -- DO SOMETHING HERE
    } else {
        switch.selected = false
    }
}
12
itsji10dra

Если вы используете так много переключателей в своем приложении, тогда возникает проблема с изменением кода во всех местах, где определен метод действия t UISwitch. Вы можете сделать пользовательское переключение и обрабатывать события только в случае изменения значения.

CustomSwitch.h

#import <UIKit/UIKit.h>

@interface Care4TodayCustomSwitch : UISwitch
@end

CustomSwitch.m

@interface CustomSwitch(){
    BOOL previousValue;
}
@end

@implementation CustomSwitch



- (id)initWithFrame:(CGRect)frame
{
    self = [super initWithFrame:frame];
    if (self) {
        // Initialization code
        previousValue = self.isOn;
    }
    return self;
}


-(void)awakeFromNib{
    [super awakeFromNib];
    previousValue = self.isOn;
    self.exclusiveTouch = YES;
}


- (void)setOn:(BOOL)on animated:(BOOL)animated{

    [super setOn:on animated:animated];
    previousValue = on;
}


-(void)sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event{

    if(previousValue != self.isOn){
        for (id targetForEvent in [self allTargets]) {
            for (id actionForEvent in [self actionsForTarget:targetForEvent forControlEvent:UIControlEventValueChanged]) {
                [super sendAction:NSSelectorFromString(actionForEvent) to:targetForEvent forEvent:event];
            }
        }
        previousValue = self.isOn;
    }
}

@end

Мы игнорируем события, если значение совпадает с измененным значением. Поместите CustomSwitch во все классы UISwitch в раскадровке. Это решит проблему и вызовет target только один раз при изменении значения. 

9
codester

Если вам не нужно мгновенно реагировать на изменение значения коммутатора, решение может быть следующим:

- (IBAction)switchChanged:(id)sender {
  [NSObject cancelPreviousPerformRequestsWithTarget:self];

  if ([switch isOn]) {
      [self performSelector:@selector(enable) withObject:nil afterDelay:2];
  } else {
      [self performSelector:@selector(disable) withObject:nil afterDelay:2];
  }
}
6
pre

Эта проблема все еще здесь с iOS 9.3 бета. Если вы не возражаете против того, что пользователь не может перетаскивать за пределы коммутатора, я считаю, что использование .TouchUpInside вместо .ValueChanged работает надежно.

3
Nick Kohrn

Эта проблема мучает меня, когда я привязываю переключатель к другому поведению. Обычно вещи не любят переходить от on к on. Вот мое простое решение:

@interface MyView : UIView
@parameter (assign) BOOL lastSwitchState;
@parameter (strong) IBOutlet UISwitch *mySwitch;
@end

@implementation MyView

// Standard stuff goes here

- (void)mySetupMethodThatsCalledWhenever
{
    [self.mySwitch addTarget:self action:@selector(switchToggled:) forControlEvents:UIControlEventValueChanged];
}

- (void)switchToggled:(UISwitch *)someSwitch
{
    BOOL newSwitchState = self.mySwitch.on;
    if (newSwitchState == self.lastSwitchState)
    {
        return;
    }
    self.lastSwitchState = newSwitchState;

    // Do your thing
}

Обязательно установите self.lastSwitchState каждый раз, когда вы вручную меняете mySwitch.on! :)

2
Ben Leggiero

Я все еще сталкиваюсь с той же проблемой в iOS 9.2 

Я получил решение и позировал, как это могло бы помочь другим 

  1. Создайте переменную count, чтобы отследить, сколько раз метод получил вызов

    int switchMethodCallCount = 0;
    
  2. Сохранить значение bool для значения переключателя

    bool isSwitchOn = No;
    
  3. В методе изменения значения Switch выполните желаемое действие только для первого вызова метода. Когда значение переключателя снова изменяется, установите значение счетчика и значение переменной bool на значение по умолчанию

    - (IBAction)frontCameraCaptureSwitchToggle:(id)sender {
    
    
    
    //This method will be called multiple times if user drags on Switch,
    //But desire action should be perform only on first call of this method
    
    
    //1. 'switchMethodCallCount' variable is maintain to check number of calles to method,
    //2. Action is peform for 'switchMethodCallCount = 1' i.e first call
    //3. When switch value change to another state, 'switchMethodCallCount' is reset and desire action perform
    
    switchMethodCallCount++ ;
    
    //NSLog(@"Count --> %d", switchMethodCallCount);
    
    if (switchMethodCallCount == 1) {
    
    //NSLog(@"**************Perform Acction******************");
    
    isSwitchOn = frontCameraCaptureSwitch.on
    
    [self doStuff];
    
    }
    else
    {
    //NSLog(@"Do not perform");
    
    
    if (frontCameraCaptureSwitch.on != isSwitchOn) {
    
        switchMethodCallCount = 0;
    
        isSwitchOn = frontCameraCaptureSwitch.on
    
        //NSLog(@"Count again start");
    
        //call value change method again 
        [self frontCameraCaptureSwitchToggle:frontCameraCaptureSwitch];
    
    
        }
    }
    
    
    }
    
2
Chetan Koli

Этот тип проблемы часто вызывается ValueChanged. Вам не нужно нажимать кнопку, чтобы запустить функцию. Это не сенсорное событие. Каждый раз, когда вы программно переключаете переключатель вкл/выкл, значение меняется, и оно снова вызывает функцию IBAction. 

У @RoNiT был правильный ответ:

Стриж 

func doToggle(switch: UISwitch) {
    if switch.on && !switch.selected {
        switch.selected = true
        // SWITCH ACTUALLY CHANGED -- DO SOMETHING HERE
    } else {
        switch.selected = false
    }
}
1
Dave G

Это сработало для меня.

DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()){
    self.switch.isOn = true
}
0
user3305074
DispatchQueue.main.async {
        self.mySwitch.setOn(false, animated: true)
    }

Это работает нормально и больше не вызывает функцию выбора.

0
anuraagdjain