• Swift


    约束

    1. Storyboard中通过拖拽设置constraints;
    2. VFL和原生语法使用代码设置constraints;
    3. oc - Masonry, swift - SnapKit;对应框架设置约束

    iOS布局机制 auto layout

      某个View需要使用auto layout布局,需要设置起translatesAutoresizingMaskIntoConstraints属性设置为NO

    重要的API

    1. intrinsinContentSize(固有内容尺寸,这些内容就是可以放内容的)
     就UILable而言下面这个方法比intrinsinContentSize先调用,且修改后的tmpRect就是修改前的contentSize
     override func textRectForBounds(bounds: CGRect, limitedToNumberOfLines numberOfLines: Int) -> CGRect {
            var tmpRect = super.textRectForBounds(bounds, limitedToNumberOfLines: numberOfLines)
            print("textRectForBoundsA:(tmpRect)")
            tmpRect.size.width  += 15
            tmpRect.size.height += 15
            print("textRectForBoundsB:(tmpRect)")
            return tmpRect
        }
    //实现这些效果,只是应为我们知道现有内容所需的尺寸,并且在其内容下扩展,就造成了这个临界效果
    override func intrinsicContentSize() -> CGSize { var contentSize = super.intrinsicContentSize() //这个方法就是通过约束来返回一个渲染时候需要的Size,会多次回调 print(contentSize) contentSize.width += 20 contentSize.height += 20 print(contentSize) return contentSize }

    intrinsicContentSize可以通过重写这个方法,返回一个通过约束计算出的frame,也可以如上,改变这个frame来达到渲染后的frame。(

    UILableUIButtonUIImage这种可以通过内容通过约束得出size

    其他的类似View重写的这个方法统一都返回(-1,-1),包括UITextView都返回的是(-1,-1)

    总结:上面这个方法的理解,有一种效果没法简单实现,就是多行内容的contentInset,宽貌似没法实现(已经校验过),在这2个方法里面对其

       返回Size的修改,都必须在其固有约束里面,比如本来这个View宽小于等于30,不能说修改后返回的size宽大于30了。这样就没没法实现

            上面说的多行文本的contentInset(换句话说就是完整约束下的宽高和通过这个方法改变的宽高取小者)

    2.preferredMaxLayoutWidth

    就我目前的手法而言这个方法是没有用的,这个方法可以用一个上限约束 "<=" 代替

    3.sizeThatFits:方法和sizeToFit方法

          let tmpsize = testTextView.sizeThatFits(self.testTextView.bounds.size)
             self.testTextViewHeightConstraint.constant = tmpsize.height

    类似上面的方法可以实现那种内容和高度一致的效果,不产生滚动条。对于tableview实现这个还没尝试过,只测试了textView

    总结:调用sizeThatFits:并不改变View的size,它只是根据已有的content和给定的size计算出最合适的view的size。

       sizeToFit会改变View的size,对我的手法而言,直接弃用

    4.systemLayoutSizeFittingSize:方法

    对于自有内容的View在布局完成之前获得frame可以用intrinsinContentSize,

    对于非自有内容在布局完成之前要获得一个View的frame那就需要这个方法了。

    使用这个方法之前确保约束的完整性能足够撑大外层的view,否则约束不完整

    根据传入的参数UILayoutFittingCompressedSize对应size.width = 0

           UILayoutFittingExpandedSize对应size.width = 1000

    eg:动态计算cell的高度常用这个方法,此方法不能计算包含UITextView的,这种情况的解决方案就是,还是上面的计算然后加上textView的高,textView用上面的sizeThatFits:来计算

    这个方法有时候要和preferredMaxLayoutWidth搭载一起使用才有效果

        

          self.testLabelB.text = "这包含的另外一层意思,即在布局完成前,我们是不能通过view.frame.size准确获取view的size的。但有时候,我们需要在auto layout system对view完成布局前就知道它的这包      含的另外一层意思,即在布局完成前,我们是不能通过view.frame.size准确获取view的size的。但有时候,我们需要在auto layout system对view完成布局前就知道它的"
             self.testLabelB.preferredMaxLayoutWidth = 300
             let outViewFitSize = self.testUIViewA.systemLayoutSizeFittingSize(UILayoutFittingExpandedSize)
              print("outViewFitSize:(outViewFitSize)")

     总结:就我的理解为什么要在这里设置preferredMaxLayoutWidth才有效果的原因,因为以前不需要设置这个是因为由外到内的约束都是完整的,可以计算出preferredMaxLayoutWidth,但是现在的情况是

       计算外层的size,那么就无法使用外层约束来揣测内层约束,从而无法得到preferredMaxLayoutWidth,那么就需要显示声明了

    5.压缩阻力(Compression Resistance)和 内容吸附(Content Hugging)

    值越大,越不容易被压缩和吸附(拉伸),用过自动布局就知道这种场景吧,,场景:2个view并排,Width都是>=0 ,且2边都leading 0 ,training 为 0 ,你想要拉伸或者压缩那个view呢,就取决于这个条件约束

    参考资料

  • 相关阅读:
    CharacterEncodingFilter详解及源码解析
    SPI 串行Flash闪存W25Q128FV 的使用(STM32F407)_硬件篇
    STM32 TFT LCD
    什么叫状态机:按键消抖实例
    STM32 ADC单通道采集 (STM32F103C8T6 ADC1的0通道 )
    STM32 PWM输出 (STM32F103C8T6 TIM2_CH2 )
    STM32 多通道ADC连续采集之数据到内存 DMA传输
    STM32 DAM之串口通讯
    STM32定时器之PWM 4路输出 TIM3、TIM14
    STM32定时器 TIM14之PWM 可调脉宽输出 呼吸灯
  • 原文地址:https://www.cnblogs.com/liyang31tg/p/5131821.html
Copyright © 2020-2023  润新知