it-roy-ru.com

iOS9 popover всегда указывает на левый верхний угол привязки

Я использую сценарий segue, который представляет контроллер представления как popover. Seque имеет свой собственный UIView в качестве своего якоря. На предыдущей iOS9 всплывающее окно правильно указывало бы на центральную часть настраиваемой UIView (представленной под UIView). На iOS9 это указывает на верхний левый угол UIView.

Я попытался отследить все вызовы селектора для пользовательской UIView, чтобы выяснить, есть ли что-то, что мне может понадобиться реализовать в моем пользовательском UIView, чтобы обеспечить «горячую точку» для всплывающего окна, но не смог ничего найти

Есть идеи..? Спасибо

Спасибо @Igor Camilo за его ответ - в случае, если это полезно для некоторых, я исправил это в своем коде:

- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {

     UIPopoverPresentationController* possiblePopOver = segue.destinationViewController.popoverPresentationController;
     if (possiblePopOver != nil) {
         //
         // iOS9 -- ensure correct sourceRect
         //
         possiblePopOver.sourceRect = possiblePopOver.sourceView.bounds;
     }
    ...
 }

Пример: Короткая кнопка запускает всплывающее окно, оно указывает на верхний левый угол элемента управления «Сортировка»

 Resulting popover

 Segue settings

31
Nikos

У меня была точно такая же проблема. Я только что решил, установив sourceRect в prepareForSegue:

override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
    switch segue.identifier {
    case "Popover Identifier"?:
        if #available(iOS 9.0, *) {
            segue.destinationViewController?.popoverPresentationController?.sourceRect = anchorView.bounds
        }
    default:
        break
    }
}
32
Igor Camilo

У меня была та же проблема, но у моего приложения было множество всплывающих окон, поэтому я создал централизованную функцию для исправления (но все равно приходилось использовать ее на каждом контроллере представления, в котором были всплывающие окна). 

// Fix for IOS 9 pop-over arrow anchor bug
// ---------------------------------------
// - IOS9 points pop-over arrows on the top left corner of the anchor view
// - It seems that the popover controller's sourceRect is not being set
//   so, if it is empty  CGRect(0,0,0,0), we simply set it to the source view's bounds
//   which produces the same result as the IOS8 behaviour.
// - This method is to be called in the prepareForSegue method override of all
//   view controllers that use a PopOver segue
//
//   example use:
//
//          override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) 
//          {
//             fixIOS9PopOverAnchor(segue)   
//          }
//      
extension UIViewController
{
   func fixIOS9PopOverAnchor(segue:UIStoryboardSegue?)
   {
      guard #available(iOS 9.0, *) else { return }          
      if let popOver = segue?.destinationViewController.popoverPresentationController,
         let anchor  = popOver.sourceView
         where popOver.sourceRect == CGRect()      
            && segue!.sourceViewController === self 
      { popOver.sourceRect = anchor.bounds }   
   }       
}
18
Alain T.

Вот пример фрагмента кода Игоря Камило в Objective-C.

- (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender
{
    // If the sender is a UIView, we might have to correct the sourceRect of
    // a potential popover being presented due to an iOS 9 bug. See:
    // https://openradar.appspot.com/22095792 and http://stackoverflow.com/a/32698841/368674
    if ([sender isKindOfClass:UIView.class]) {
        // Fetch the destination view controller
        UIViewController *destinationViewController = [segue destinationViewController];

        // If there is indeed a UIPopoverPresentationController involved
        if ([destinationViewController respondsToSelector:@selector(popoverPresentationController)]) {
            // Fetch the popover presentation controller
            UIPopoverPresentationController *popoverPresentationController =
            destinationViewController.popoverPresentationController;

            // Set the correct sourceRect given the sender's bounds
            popoverPresentationController.sourceRect = ((UIView *)sender).bounds;
        }
    }
}
8
zath

Вот мое решение, внутри prepareForSegue:

segue.destinationViewController.popoverPresentationController?.sourceRect = CGRectMake(anchorView.frame.size.width/2, anchorView.frame.size.height, 0, 0)

Это переместит указатель в нижнюю середину вида привязки

7
s0m3chill

Я также столкнулся с этой проблемой, но теперь она работает, когда я добавил это в свою функцию PrepareForSegue . Учитывая, что мой Segue ID содержит строку Popover

if ([[segue identifier] containsString:@"Popover"]) {
    [segue destinationViewController].popoverPresentationController.sourceRect = self.navigationItem.titleView.bounds;
}
1
bravehart_sf

Если вы ссылаетесь на свой popover UIViewController в своем основном UIViewController, вы можете настроить свойство sourceRect для смещения всплывающего окна. Например, учитывая popover popoverVC, вы можете сделать что-то вроде этого:

float xOffset = 10.0;
float yOffset = 5.0;
popoverVC.popoverPresentationController.sourceRect = CGMakeRect(xOffset, yOffset, 0.0, 0.0);
0
Chris Stillwell

Немного более обновленного ответа для Swift 3! Простите за все кастинги 

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        if segue.identifier == "popSegue" {
            let popoverViewController = segue.destination
            popoverViewController.popoverPresentationController?.delegate = self
            segue.destination.popoverPresentationController?.sourceRect = ((sender as? UIButton)?.bounds)!
        }
    }
0
Matt Bart

просто обновить до реального примера с рабочим кодом для любого UIView

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    switch segue.identifier {
    case "PopoverSegueId"?:
        if #available(iOS 9.0, *) {
            segue.destination.popoverPresentationController?.sourceRect = (segue.destination.popoverPresentationController?.sourceView?.bounds)!
        }
    default:
        break
    }

}
0
J Mustermann

Попробуйте установить ширину и высоту привязки исходного прямоугольника (UIView или UIBarButtonItem) и установить его активным. Установите его при инициализации UIView или UIBarButtonItem.

UIBarButtonItem

 [[youruibarbuttonitem.customView.widthAnchor constraintEqualToConstant:youruibarbuttonitem.customView.bounds.size.width] setActive:YES];
 [[youruibarbuttonitem.customView.heightAnchor constraintEqualToConstant:youruibarbuttonitem.customView.bounds.size.height] setActive:YES];

UIView

[[uiview.widthAnchor constraintEqualToConstant:uiview.bounds.size.width] setActive:YES];
 [[uiview.heightAnchor constraintEqualToConstant:uiview.bounds.size.height] setActive:YES];
0
Tanvir Singh