UITableView демпфирование анимации и ограничения макета

Я пытаюсь анимировать UITableView, чтобы он действовал как dropdownMenu, используя ограничение по высоте и блок UIView.animateWithDamping(..). У меня возникают странные проблемы с белым фоном под tableView.

Симулятор iPhone, показывающий проблему

Я очистил каждый цвет фона, и это не очень помогает.
Вот код, устанавливающий все подпредставления dropDownView, который является UIView:

required public init?(coder aDecoder: NSCoder) {
    super.init(coder: aDecoder)
    self.elements = []
    defaultSetup()
}

private func defaultSetup() {
    configureActionButton()
    configureTableView()
}
private func configureActionButton() {
    actionButton = UIButton(frame: CGRectZero)
    actionButton.translatesAutoresizingMaskIntoConstraints = false
    addSubview(actionButton)
    guard let superview                                 = actionButton.superview else {
        assert(false, "ActionButton adding to superview failed.")
        return
    }
    // Constraints
    actionButton.constrain(.Leading, .Equal, superview, .Leading, constant: 0, multiplier: 1)?.constrain(.Trailing, .Equal, superview, .Trailing, constant: 0, multiplier: 1)?.constrain(.Top, .Equal, superview, .Top, constant: 0, multiplier: 1)?.constrain(.Bottom, .Equal, superview, .Bottom, constant: 0, multiplier: 1)
    // Appearance
    actionButton.backgroundColor                        = UIColor.clearColor()
    actionButton.opaque                                 = false
    actionButton.contentHorizontalAlignment             = .Left
    actionButton.contentEdgeInsets = UIEdgeInsets(top: 0, left: 20, bottom: 0, right: 0)
    if borderVisible {
        actionButton.layer.cornerRadius                 = 5
        actionButton.layer.borderColor                  = UIColor.blackColor().CGColor
        actionButton.layer.borderWidth                  = 1
        actionButton.clipsToBounds                      = true
    }
    // Actions
    actionButton.addTarget(self, action: "menuAction:", forControlEvents: .TouchUpInside)
}

private func configureTableView() {
    tableView                                           = BOTableView(frame: CGRectZero, items: elements, configuration: configuration)
    tableView.translatesAutoresizingMaskIntoConstraints = false
    tableView.delegate                                  = self
    tableView.dataSource                                = self
    addSubview(tableView)
    guard let tableViewSuperview = tableView.superview else {
        assert(false, "TableView adding to superview failed.")
        return
    }
    // Constraints
    tableView.constrain(.Trailing, .Equal, tableViewSuperview, .Trailing, constant: 0, multiplier: 1)?.constrain(.Top, .Equal, tableViewSuperview, .Bottom, constant: 0, multiplier: 1)?.constrain(.Leading, .Equal, tableViewSuperview, .Leading, constant: 0, multiplier: 1)
    tvHeightConstraint                                  = NSLayoutConstraint(item: tableView, attribute: .Height, relatedBy: .Equal, toItem: nil, attribute: .NotAnAttribute, multiplier: 1, constant: 0)
    tableView.addConstraint(tvHeightConstraint)

}

Инициализатор класса BOTableView:

init(frame: CGRect, items: [String], configuration: BOConfiguration) {
    super.init(frame: frame, style: UITableViewStyle.Plain)

    self.items                              = items
    self.selectedIndexPath                  = NSIndexPath(forRow: 0, inSection: 0)
    self.configuration                      = configuration

    // Setup table view
    self.opaque                             = false
    self.backgroundView?.backgroundColor    = UIColor.clearColor()
    self.backgroundColor                    = UIColor.clearColor()
    self.separatorColor                     = UIColor.blackColor()
    self.scrollEnabled                      = false
    self.separatorStyle                     = .SingleLine

    self.layer.cornerRadius                 = 5
    self.layer.borderColor                  = UIColor.blackColor().CGColor
    self.layer.borderWidth                  = 1
    self.clipsToBounds                      = true
}

UIView анимации:

private func showMenuWithCompletionBlock(completion: (succeeded: Bool) -> Void) {
    delegate?.menuWillShow(self)
    let tvHeight                            = frame.size.height * CGFloat(elements.count)
    tvHeightConstraint.constant             = tvHeight

    UIView.animateWithDuration(0.5, delay: 0, usingSpringWithDamping: 0.4, initialSpringVelocity: 0.5, options: .CurveEaseInOut, animations: { [weak self] () -> Void in
            guard let strongSelf = self else {
                completion(succeeded: false)
                return
            }
            strongSelf.layoutIfNeeded()
            }, completion: { (finished) -> Void in
                if finished {
                    completion(succeeded: true)
                }
        })
}

Вот код для расширения UIView + Constraints, используемый в коде:

extension UIView {
/**
 :returns: true if v is in this view's super view chain
 */
public func isSuper(v : UIView) -> Bool
{
    for var s : UIView? = self; s != nil; s = s?.superview {
        if(v == s) {
            return true;
        }
    }
    return false
}

public func constrain(attribute: NSLayoutAttribute, _ relation: NSLayoutRelation, _ otherView: UIView, _ otherAttribute: NSLayoutAttribute, constant: CGFloat = 0.0, multiplier : CGFloat = 1.0) -> UIView?
{
    let c = NSLayoutConstraint(item: self, attribute: attribute, relatedBy: relation, toItem: otherView, attribute: otherAttribute, multiplier: multiplier, constant: constant)

    if isSuper(otherView) {
        otherView.addConstraint(c)
        return self
    }
    else if(otherView.isSuper(self) || otherView == self)
    {
        self.addConstraint(c)
        return self
    }
    assert(false)
    return nil
}

public func constrain(attribute: NSLayoutAttribute, _ relation: NSLayoutRelation, constant: CGFloat, multiplier : CGFloat = 1.0) -> UIView?
{
    let c = NSLayoutConstraint(item: self, attribute: attribute, relatedBy: relation, toItem: nil, attribute: .NotAnAttribute, multiplier: multiplier, constant: constant)
    self.addConstraint(c)
    return self
}

}

Когда я попытался отладить иерархию представлений в отладчике, единственным представлением, имеющим белый фон, был tableView, но я очистил фон в коде. Я также пытался установить backgroundView для tableView равным nil, а backgroundView.backgroundColor для clearColor(). Ничего не изменилось.

1 ответ

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

    [_tableView setTableFooterView:[[UIView alloc] init]];
Другие вопросы по тегам