Изменение размера многострочного UILabel в UIScrollView прерывает прокрутку. Зачем? - PullRequest
0 голосов
/ 07 марта 2019

Я использую следующую настройку:

  • Представление ViewControllers содержит UIScrollView с ограничениями Top, Leading, Trailing и Bottom, чтобы соответствовать размеру VC
  • ScrollView содержит дваподпредставления:
    • A UIView для определения размера содержимого представления прокрутки.Он имеет ту же высоту, что и ScrollView, но в два раза больше своей ширины.Таким образом, возможна только горизонтальная прокрутка.
    • A UILabel с некоторым длинным текстом с ограничением высоты и ширины, чтобы установить фиксированный размер, и ограничением верхних и ведущих линий в ScrollView, чтобы установить фиксированную позицию.
  • Ширина метки изменяется при прокрутке ScrollView.

Проблема: Если для метки задано использование более одной строки Исвойство ScrollViews contenOffset устанавливается вручную, ScrollView прекращает прокрутку.

ViewController View
+---------------------+
|+-------------------+| 
||ScrollView         ||
||+------------------||--------------------+
|||UIView to define  || content size       |
|||                  ||                    |
|||                  ||                    |
|||  [MultiLine]     ||                    |
|||  [  Label  ]     ||                    |
|||                  ||                    |
||+------------------||--------------------+
|+-------------------+|
+---------------------+


- (void)viewDidLayoutSubviews {
    [super viewDidLayoutSubviews];

    // Setting the ContentOffset will stop scrolling 
    //[self.scrollView setContentOffset:CGPointMake(0, 0)];
}

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    // Resize Label when scrolling
    self.labelWidthConstraint.constant = MAX (50, 50 + self.scrollView.contentOffset.x);
}

Изменение размера метки с помощью этого кода работает нормально, если

  • метка установлена, использовать oneлиния .В этом случае установка параметра смещение контента НЕ приносит никакого вреда. ИЛИ
  • смещение содержимого не изменяется (даже не устанавливается (0, 0)).В этом случае установка метки на многострочное сообщение НЕ приносит никакого вреда

Установка смещения содержимого И одновременное использование многострочного НЕ работает .Свиток больше нельзя прокручивать.

Почему это так? Есть идеи, что может вызвать это и как его решить?

Ответы [ 2 ]

1 голос
/ 07 марта 2019

Проблема заключается в том, что при изменении ограничений метки вызывается viewDidLayoutSubviews, который затем устанавливает UIScrollView на отсутствие прокрутки, так как set contentOffset вызывается снова и снова. Вы можете преодолеть это, если вы только хотите установить для UIScrollView значение CGPoint.zero в исходном макете, используя bool в качестве флага. Очевидно, поскольку UILabel нуждается в перерисовке при изменении размера, он вызывает viewDidLayoutSubviews. Вот пример в Swift.

import UIKit

class ViewController: UIViewController {

    lazy var scrollView : UIScrollView = {
        let sv = UIScrollView(frame: self.view.bounds)
        sv.translatesAutoresizingMaskIntoConstraints = false
        sv.delegate = self
        return sv
    }()

    lazy var contentView : UIView = {
        let v = UIView(frame: CGRect(x: 0, y: 0, width: self.view.bounds.width * 4, height: self.view.bounds.height))
        v.translatesAutoresizingMaskIntoConstraints = false
        return v
    }()

    lazy var label : UILabel = {
        let lbl = UILabel(frame: CGRect(x: 0, y: 0, width: self.view.bounds.width, height: 50))
        lbl.numberOfLines = 0
        lbl.text = "Lorem Ipsum is simply dummy text of the printing and typesetting industry. Lorem Ipsum has been the industry's standard dummy text ever since the 1500s"
        lbl.minimumScaleFactor = 0.5
        lbl.adjustsFontSizeToFitWidth = true
        lbl.font = UIFont.systemFont(ofSize: 22)
        lbl.translatesAutoresizingMaskIntoConstraints = false
        return lbl
    }()

    var widthConstraint : NSLayoutConstraint?
    var heightConstraint : NSLayoutConstraint?
    var startingHeight : CGFloat = 0
    var startingWidth : CGFloat = 0
    override func viewDidLoad() {
        super.viewDidLoad()
        //first scrollview
        self.view.addSubview(scrollView)
        pinToAllSides(target: scrollView)

        //now content view
        self.scrollView.addSubview(contentView)
        contentView.widthAnchor.constraint(equalTo: self.view.widthAnchor, multiplier: 2).isActive = true
        contentView.heightAnchor.constraint(equalTo: self.scrollView.heightAnchor, multiplier: 1).isActive = true
        contentView.backgroundColor = .green
        pinToAllSides(target: contentView)
        scrollView.layoutIfNeeded()

        //now the label
        self.scrollView.addSubview(label)
        label.leadingAnchor.constraint(equalTo: self.scrollView.leadingAnchor, constant: 20).isActive = true
        label.topAnchor.constraint(equalTo: self.scrollView.topAnchor, constant: 60).isActive = true
        label.backgroundColor = .red
        widthConstraint = NSLayoutConstraint(item: label, attribute: .width, relatedBy: .equal, toItem: nil, attribute: .notAnAttribute, multiplier: 1, constant: self.view.bounds.width/2)

        heightConstraint = NSLayoutConstraint(item: label, attribute: .height, relatedBy: .equal, toItem: nil, attribute: .notAnAttribute, multiplier: 1, constant: 300)
        if let wc = widthConstraint,
            let hc = heightConstraint{
            startingHeight = hc.constant
            startingWidth = wc.constant
            label.addConstraint(wc)
            label.addConstraint(hc)
        }

    }

    func pinToAllSides(target:UIView){
        guard let superview = target.superview else{
            return
        }
        target.leadingAnchor.constraint(equalTo: superview.leadingAnchor).isActive = true
        target.trailingAnchor.constraint(equalTo: superview.trailingAnchor).isActive = true
        target.topAnchor.constraint(equalTo: superview.topAnchor).isActive = true
        target.bottomAnchor.constraint(equalTo: superview.bottomAnchor).isActive = true
    }

    var hasHappenedOnce : Bool = false
    override func viewDidLayoutSubviews() {
        super.viewDidLayoutSubviews()
        if hasHappenedOnce == false{
            hasHappenedOnce = true
            self.scrollView.contentOffset = .zero
        }
    }
}

extension ViewController : UIScrollViewDelegate{

    func scrollViewDidScroll(_ scrollView: UIScrollView) {
        //hopefully it is laggy due to simulator but for the label i would ditch constraints myself
        self.widthConstraint?.constant = max(startingWidth, self.scrollView.contentOffset.x * 1.1 + startingWidth)

        let height = startingHeight - self.scrollView.contentOffset.x
        self.heightConstraint?.constant = height
        label.updateConstraints()
    }
}
1 голос
/ 07 марта 2019

Проблема здесь:

- (void)viewDidLayoutSubviews {
    [super viewDidLayoutSubviews];

    // Setting the ContentOffset will stop scrolling 
    [self.scrollView setContentOffset:CGPointMake(0, 0)];
}

Изменение self.labelWidthConstraint.constant при прокрутке вида прокрутки вызывает viewDidLayoutSubviews!Так что, как только вы начнете прокручивать, ваш код немедленно сбрасывается с .contentOffset на 0,0.

Я не знаю, почему вы все равно хотите звонить на setContentOffset, конечно, не на viewDidLayoutSubviews.

Выполнение быстрого теста, после удаления кода viewDidLayoutSubviews ...

Я позвонил [self.scrollView setContentOffset:CGPointMake(0, 0)]; в конце viewDidload (а также попытался в viewDidAppear).... прокрутка (и обновление константы ограничения) продолжает работать нормально.

Я также добавил кнопку, которая вызывает [self.scrollView setContentOffset:CGPointMake(0, 0)]; при нажатии ... прокрутка (и обновление константы ограничения) продолжает работать нормально.

...