ป้ายกำกับภายใต้รูปภาพใน UIButton


158

ฉันกำลังพยายามสร้างปุ่มที่มีข้อความอยู่ใต้ไอคอน (sorta เหมือนกับปุ่มแอป) แต่ดูเหมือนว่าจะทำได้ยาก แนวคิดใดบ้างที่ฉันสามารถใช้เพื่อให้ข้อความปรากฏใต้ภาพด้วยUIButton?


มันค่อนข้างง่ายและสามารถทำได้เพื่อสร้าง subclass แบบกำหนดเองของ UIbutton ที่มี UIImage และ UILabel อยู่ในตำแหน่งที่คุณต้องการ ...
NP Compete

7
หรือเพียงแค่ใช้ UIButton และ UILabel
raidfive

เพื่อควบคุมขนาดและเลย์เอาต์อัตโนมัติอย่างแม่นยำคุณสามารถลองทำสิ่งนี้: https://github.com/albert-zhang/AZCenterLabelButton( ลิงก์ )
อัลเบิร์ตจาง

ใช้งานได้ดีกับโซลูชันนี้stackoverflow.com/a/59666154/1576134
Shreyank

คำตอบ:


111

หรือคุณสามารถใช้หมวดหมู่นี้:

ObjC

@interface UIButton (VerticalLayout)

- (void)centerVerticallyWithPadding:(float)padding;
- (void)centerVertically;

@end

@implementation UIButton (VerticalLayout)

- (void)centerVerticallyWithPadding:(float)padding {
    CGSize imageSize = self.imageView.frame.size;
    CGSize titleSize = self.titleLabel.frame.size;
    CGFloat totalHeight = (imageSize.height + titleSize.height + padding);
    
    self.imageEdgeInsets = UIEdgeInsetsMake(- (totalHeight - imageSize.height),
                                            0.0f,
                                            0.0f,
                                            - titleSize.width);
    
    self.titleEdgeInsets = UIEdgeInsetsMake(0.0f,
                                            - imageSize.width,
                                            - (totalHeight - titleSize.height),
                                            0.0f);
    
    self.contentEdgeInsets = UIEdgeInsetsMake(0.0f,
                                              0.0f,
                                              titleSize.height,
                                              0.0f);
}

- (void)centerVertically {
    const CGFloat kDefaultPadding = 6.0f;
    [self centerVerticallyWithPadding:kDefaultPadding];
}

@end

ส่วนขยายที่รวดเร็ว

extension UIButton {
    
    func centerVertically(padding: CGFloat = 6.0) {
        guard
            let imageViewSize = self.imageView?.frame.size,
            let titleLabelSize = self.titleLabel?.frame.size else {
            return
        }
        
        let totalHeight = imageViewSize.height + titleLabelSize.height + padding
        
        self.imageEdgeInsets = UIEdgeInsets(
            top: -(totalHeight - imageViewSize.height),
            left: 0.0,
            bottom: 0.0,
            right: -titleLabelSize.width
        )
        
        self.titleEdgeInsets = UIEdgeInsets(
            top: 0.0,
            left: -imageViewSize.width,
            bottom: -(totalHeight - titleLabelSize.height),
            right: 0.0
        )
        
        self.contentEdgeInsets = UIEdgeInsets(
            top: 0.0,
            left: 0.0,
            bottom: titleLabelSize.height,
            right: 0.0
        )
    }
    
}

คำแนะนำ: หากความสูงของปุ่มน้อยกว่าtotalHeightภาพก็จะวาดเส้นขอบนอก

imageEdgeInset.top ควรจะเป็น:

max(0, -(totalHeight - imageViewSize.height))

5
ฉันคิดว่านี่เป็นคำตอบที่ดีที่สุดเพราะใช้ edgeInsets แทนการปรับเฟรมด้วยตนเอง มันทำงานได้ดีกับเลย์เอาต์อัตโนมัติเช่นกันเมื่อเรียกจาก layoutSubviews ในปุ่มควบคุมของปุ่ม ข้อเสนอแนะเท่านั้นที่จะใช้CGRectGetHeight()และCGRectGetWidth()เมื่อได้รับ imageView และ titleLabel ความสูงและความกว้าง
Jesse

1
เมื่อฉันใช้รูปนี้จะปรากฏขึ้นเหนือมุมมองปุ่มเพื่อจัดให้อยู่กึ่งกลางฉันควร CGFloat inset = (self.frame.size.height - totalHeight)/2; self.contentEdgeInsets = UIEdgeInsetsMake(inset, 0.0f, inset, 0.0f);
Alex Hedley

12
ส่วนขยาย Swift ไม่ได้จัดวางอย่างถูกต้องสำหรับฉัน
แพทริค

มันทำงานได้ถ้ารูปภาพถูกตั้งเป็น setImage ไม่ใช่ setBackgroundImage
Argus

1
ใช้งานได้ดีกับโซลูชันนี้stackoverflow.com/a/59666154/1576134
Shreyank

87

ใน Xcode คุณสามารถตั้งค่า Edge Title Left Inset ให้เป็นลบความกว้างของภาพ สิ่งนี้จะแสดงฉลากที่อยู่ตรงกลางของภาพ

หากต้องการให้ป้ายกำกับแสดงใต้ภาพ (sorta เหมือนปุ่มแอป) คุณอาจต้องตั้งค่า Edge Title Top Inset ให้เป็นจำนวนบวก


1
นี่เป็นวิธีที่จะทำ ... เว้นแต่คุณจะทำซ้ำหลายครั้งด้วยปุ่ม (ขนาดต่าง ๆ ) ... ซึ่งในกรณีนี้ฉันได้ผลลัพธ์ที่ดีกับโซลูชัน Erik W รุ่น tweaked
Kenny Winker

5
เพียงเพื่อให้แน่ใจว่าผู้คนตระหนักถึงสิ่งนี้ ค่าควรเป็นความกว้างติดลบของรูปภาพแม้ว่าปุ่มจะกว้างกว่าความกว้างของรูปภาพ
Liron

1
สิ่งนี้ไม่ได้ผลสำหรับฉัน ข้อความของฉันยังปรากฏที่ด้านขวาของภาพกล่าวคือไม่ได้อยู่ด้านล่าง
Cindeselia

1
@Cindeselia นั่นน่าแปลกใจ คุณค่าที่คุณใช้กับ Top Inset มีค่าเท่าใด บางทีลองเพิ่มเป็นค่าที่มากกว่านี้
Chris

3
ใน iOS7 ดูเหมือนว่าจะไม่ทำงาน ป้ายกำกับจะเลื่อนไปที่ด้านล่างของภาพและซ่อนอยู่จะไม่แสดงอีกต่อไป
Brave

50

นี้เป็นเรื่องง่ายปุ่มชื่อเป็นศูนย์กลางการดำเนินการในสวิฟท์โดยเอาชนะและtitleRect(forContentRect:) imageRect(forContentRect:)นอกจากนี้ยังใช้intrinsicContentSizeสำหรับใช้กับ AutoLayout

import UIKit

class CenteredButton: UIButton
{
    override func titleRect(forContentRect contentRect: CGRect) -> CGRect {
        let rect = super.titleRect(forContentRect: contentRect)

        return CGRect(x: 0, y: contentRect.height - rect.height + 5,
            width: contentRect.width, height: rect.height)
    }

    override func imageRect(forContentRect contentRect: CGRect) -> CGRect {
        let rect = super.imageRect(forContentRect: contentRect)
        let titleRect = self.titleRect(forContentRect: contentRect)

        return CGRect(x: contentRect.width/2.0 - rect.width/2.0,
            y: (contentRect.height - titleRect.height)/2.0 - rect.height/2.0,
            width: rect.width, height: rect.height)
    }

    override var intrinsicContentSize: CGSize {
        let size = super.intrinsicContentSize

        if let image = imageView?.image {
            var labelHeight: CGFloat = 0.0

            if let size = titleLabel?.sizeThatFits(CGSize(width: self.contentRect(forBounds: self.bounds).width, height: CGFloat.greatestFiniteMagnitude)) {
                labelHeight = size.height
            }

            return CGSize(width: size.width, height: image.size.height + labelHeight + 5)
        }

        return size
    }

    override init(frame: CGRect) {
        super.init(frame: frame)
        centerTitleLabel()
    }

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        centerTitleLabel()
    }

    private func centerTitleLabel() {
        self.titleLabel?.textAlignment = .center
    }
}

5
นั่นคือทางออกที่ถูกต้องที่สุด แต่การแก้ไขบางอย่างจำเป็นสำหรับขนาดเนื้อหาที่แท้จริง ควรคืนค่าความกว้างสูงสุดระหว่างรูปภาพและป้ายกำกับ: return CGSizeMake (MAX (labelSize.width, self.imageView.image.size.width), self.imageView.image.size.height + labelHeight)
kirander

39

ดูคำตอบที่ยอดเยี่ยมนี้ใน Swift

extension UIButton {

    func alignImageAndTitleVertically(padding: CGFloat = 6.0) {
        let imageSize = self.imageView!.frame.size
        let titleSize = self.titleLabel!.frame.size
        let totalHeight = imageSize.height + titleSize.height + padding

        self.imageEdgeInsets = UIEdgeInsets(
            top: -(totalHeight - imageSize.height),
            left: 0,
            bottom: 0,
            right: -titleSize.width
        )

        self.titleEdgeInsets = UIEdgeInsets(
            top: 0,
            left: -imageSize.width,
            bottom: -(totalHeight - titleSize.height),
            right: 0
        )
    }

}

4
หากคุณยังต้องการภาพที่เป็นศูนย์กลางในแนวตั้งแทนleftในimageEdgeInsetsด้วย(self.frame.size.width - imageSize.width) / 2
elsurudo

หากคุณใช้การเรียกอัตโนมัติอัตโนมัติวิธีนี้ในlayoutSubviews()superview ของคุณ
AlexVogel

33

UIButtonประเภทรอง Override - layoutSubviewsเพื่อย้าย built-in subviewsไปยังตำแหน่งใหม่:

- (void)layoutSubviews
{
    [super layoutSubviews];

    CGRect frame = self.imageView.frame;
    frame = CGRectMake(truncf((self.bounds.size.width - frame.size.width) / 2), 0.0f, frame.size.width, frame.size.height);
    self.imageView.frame = frame;

    frame = self.titleLabel.frame;
    frame = CGRectMake(truncf((self.bounds.size.width - frame.size.width) / 2), self.bounds.size.height - frame.size.height, frame.size.width, frame.size.height);
    self.titleLabel.frame = frame;
}

โดยส่วนตัวฉันต้องตั้งค่า titleLabel y ให้เป็น 0 และความสูงเป็นความสูงของเฟรมเพื่อให้แสดงข้อความด้วยภาพ มันไม่สมเหตุสมผลสำหรับฉัน แต่มันใช้งานได้ ... แม้ว่าฉันจะยังเรียนรู้วิธี 'Apple' ในการตั้งค่าการควบคุม
Russ

6
ที่จริงแล้ววิธีที่ดีกว่าคือการแทนที่titleRectForContentRectและimageRectForContentRect
Mazyod

28

ปรับคำตอบของ icecrystal23

Swift 3 สามารถทำงานร่วมกับ autolayouts, xib, storyboards ได้

ปุ่มในคำตอบของ icecrystal23 เดิมมีกรอบที่คำนวณได้ไม่ดี ฉันคิดว่าฉันคงที่

แก้ไข:อัปเดตเป็น Swift 5 และทำงานภายใน Interface Builder / Storyboards

import UIKit

@IBDesignable
class VerticalButton: UIButton {

    @IBInspectable public var padding: CGFloat = 20.0 {
        didSet {
            setNeedsLayout()
        }
    }

    override var intrinsicContentSize: CGSize {
        let maxSize = CGSize(width: CGFloat.greatestFiniteMagnitude, height: CGFloat.greatestFiniteMagnitude)

        if let titleSize = titleLabel?.sizeThatFits(maxSize), let imageSize = imageView?.sizeThatFits(maxSize) {
            let width = ceil(max(imageSize.width, titleSize.width))
            let height = ceil(imageSize.height + titleSize.height + padding)

            return CGSize(width: width, height: height)
        }

        return super.intrinsicContentSize
    }

    override func layoutSubviews() {
        if let image = imageView?.image, let title = titleLabel?.attributedText {
            let imageSize = image.size
            let titleSize = title.size()

            titleEdgeInsets = UIEdgeInsets(top: 0.0, left: -imageSize.width, bottom: -(imageSize.height + padding), right: 0.0)
            imageEdgeInsets = UIEdgeInsets(top: -(titleSize.height + padding), left: 0.0, bottom: 0.0, right: -titleSize.width)
        }

        super.layoutSubviews()
    }

}

2
มีปัญหาเกี่ยวกับสิ่งนี้เมื่อรูปภาพถูกลบ ฉันใช้รูปภาพสำหรับสถานะที่เลือกและไม่มีรูปภาพสำหรับสถานะเริ่มต้น เมื่อสถานะเปลี่ยนจากที่เลือกเป็นค่าเริ่มต้นฉลากจะเลอะ ดังนั้นจึงจำเป็นต้องมีการแก้ไขเล็กน้อย: อย่าตรวจสอบมุมมองภาพ แต่ใช้ 'image (for: state)' ตั้งค่าการแทรกขอบเป็นศูนย์เมื่อไม่มีภาพในคำสั่งอื่นของเค้าโครงภาพรวม
Matic Oblak

ทางออกเดียวที่นี่ใช้งานได้ คำตอบอื่น ๆ ดูเหมือนจะใช้งานได้ แต่จริงๆแล้วขอบเขตของปุ่มนั้นไม่ได้ปรับขนาดตามขนาดของฉลากและรูปภาพ ตั้งค่าสีพื้นหลังเพื่อดูสิ่งนี้
มานูเอล

วิธีแก้ปัญหานี้ใช้งานไม่ได้ฉันคิดว่ามันทำให้ลูปไม่สิ้นสุดและในที่สุด Xcode ก็เกิดข้อผิดพลาด ฉันลบส่วน intrinsicContentSize และทำงานได้ดี (Xcode 11.5)
mojuba

23

แก้ไขหนึ่งในคำตอบที่นี่:

สวิฟท์ 3:

class CenteredButton: UIButton
{
    override func titleRect(forContentRect contentRect: CGRect) -> CGRect {
        let rect = super.titleRect(forContentRect: contentRect)
        let imageRect = super.imageRect(forContentRect: contentRect)

        return CGRect(x: 0, y: imageRect.maxY + 10,
                      width: contentRect.width, height: rect.height)
    }

    override func imageRect(forContentRect contentRect: CGRect) -> CGRect {
        let rect = super.imageRect(forContentRect: contentRect)
        let titleRect = self.titleRect(forContentRect: contentRect)

        return CGRect(x: contentRect.width/2.0 - rect.width/2.0,
                      y: (contentRect.height - titleRect.height)/2.0 - rect.height/2.0,
                      width: rect.width, height: rect.height)
    }

    override init(frame: CGRect) {
        super.init(frame: frame)
        centerTitleLabel()
    }

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        centerTitleLabel()
    }

    private func centerTitleLabel() {
        self.titleLabel?.textAlignment = .center
    }
}

16

นี่เป็นคำตอบที่ยอดเยี่ยมของ Erik W รุ่นดัดแปลง แต่แทนที่จะวางภาพไว้กึ่งกลางที่ด้านบนของมุมมองภาพนั้นจะวางภาพและป้ายกำกับที่อยู่กึ่งกลางในมุมมองเป็นกลุ่ม

ความแตกต่างคือ:

+-----------+
|    ( )    |
|   Hello   |     // Erik W's code
|           |
|           |
+-----------+

VS

+-----------+
|           |
|    ( )    |     // My modified version
|   Hello   |
|           |
+-----------+

ที่มาด้านล่าง:

-(void)layoutSubviews {
    [super layoutSubviews];

    CGRect titleLabelFrame = self.titleLabel.frame;
    CGSize labelSize = [self.titleLabel.text sizeWithFont:self.titleLabel.font constrainedToSize:CGSizeMake(self.frame.size.width, CGFLOAT_MAX) lineBreakMode:NSLineBreakByWordWrapping];

    CGRect imageFrame = self.imageView.frame;

    CGSize fitBoxSize = (CGSize){.height = labelSize.height + kTextTopPadding +  imageFrame.size.height, .width = MAX(imageFrame.size.width, labelSize.width)};

    CGRect fitBoxRect = CGRectInset(self.bounds, (self.bounds.size.width - fitBoxSize.width)/2, (self.bounds.size.height - fitBoxSize.height)/2);

    imageFrame.origin.y = fitBoxRect.origin.y;
    imageFrame.origin.x = CGRectGetMidX(fitBoxRect) - (imageFrame.size.width/2);
    self.imageView.frame = imageFrame;

    // Adjust the label size to fit the text, and move it below the image

    titleLabelFrame.size.width = labelSize.width;
    titleLabelFrame.size.height = labelSize.height;
    titleLabelFrame.origin.x = (self.frame.size.width / 2) - (labelSize.width / 2);
    titleLabelFrame.origin.y = fitBoxRect.origin.y + imageFrame.size.height + kTextTopPadding;
    self.titleLabel.frame = titleLabelFrame;
}

FYI: สิ่งนี้สามารถแตกหักได้เมื่อรวมเข้ากับภาพเคลื่อนไหว UIView เนื่องจากมีการเรียกใช้เค้าโครงภาพตัวอย่างในระหว่างนั้น


บรรทัดควรคำนวณ labelSize หรือไม่ใช้ self.bounds.size.width แทน self.frame.size.width
Jeremy Wiebe

16

โซลูชันของ Dave ใน Swift:

override func layoutSubviews() {
    super.layoutSubviews()
    if let imageView = self.imageView {
        imageView.frame.origin.x = (self.bounds.size.width - imageView.frame.size.width) / 2.0
        imageView.frame.origin.y = 0.0
    }
    if let titleLabel = self.titleLabel {
        titleLabel.frame.origin.x = (self.bounds.size.width - titleLabel.frame.size.width) / 2.0
        titleLabel.frame.origin.y = self.bounds.size.height - titleLabel.frame.size.height
    }
}

คำตอบที่ดี. เพิ่ม @IBDesignable ลงในคลาสย่อยของคุณและดูในกระดานเรื่องราว
Joel Teply

16

หากคุณคลาสย่อยUIButtonและoverride layoutSubviewsคุณสามารถใช้ด้านล่างเพื่อจัดภาพให้อยู่กึ่งกลางและวางหัวเรื่องที่อยู่กึ่งกลางด้านล่าง

kTextTopPadding เป็นค่าคงที่คุณจะต้องแนะนำที่กำหนดช่องว่างระหว่างภาพและข้อความด้านล่าง

-(void)layoutSubviews {
    [super layoutSubviews];

    // Move the image to the top and center it horizontally
    CGRect imageFrame = self.imageView.frame;
    imageFrame.origin.y = 0;
    imageFrame.origin.x = (self.frame.size.width / 2) - (imageFrame.size.width / 2);
    self.imageView.frame = imageFrame;

    // Adjust the label size to fit the text, and move it below the image
    CGRect titleLabelFrame = self.titleLabel.frame;
    CGSize labelSize = [self.titleLabel.text sizeWithFont:self.titleLabel.font
                                        constrainedToSize:CGSizeMake(self.frame.size.width, CGFLOAT_MAX)
                                        lineBreakMode:NSLineBreakByWordWrapping];
    titleLabelFrame.size.width = labelSize.width;
    titleLabelFrame.size.height = labelSize.height;
    titleLabelFrame.origin.x = (self.frame.size.width / 2) - (labelSize.width / 2);
    titleLabelFrame.origin.y = self.imageView.frame.origin.y + self.imageView.frame.size.height + kTextTopPadding;
    self.titleLabel.frame = titleLabelFrame;

}

8

อัปเดตคำตอบของ Kenny Winker ตั้งแต่ขนาดโดยไม่เลิกใช้ใน iOS 7

-(void)layoutSubviews {
[super layoutSubviews];

int kTextTopPadding = 3;

CGRect titleLabelFrame = self.titleLabel.frame;

CGRect labelSize = [self.titleLabel.text boundingRectWithSize:CGSizeMake(CGFLOAT_MAX, CGRectGetHeight(self.bounds)) options:NSStringDrawingUsesLineFragmentOrigin attributes:@{NSFontAttributeName:self.titleLabel.font} context:nil];

CGRect imageFrame = self.imageView.frame;

CGSize fitBoxSize = (CGSize){.height = labelSize.size.height + kTextTopPadding +  imageFrame.size.height, .width = MAX(imageFrame.size.width, labelSize.size.width)};

CGRect fitBoxRect = CGRectInset(self.bounds, (self.bounds.size.width - fitBoxSize.width)/2, (self.bounds.size.height - fitBoxSize.height)/2);

imageFrame.origin.y = fitBoxRect.origin.y;
imageFrame.origin.x = CGRectGetMidX(fitBoxRect) - (imageFrame.size.width/2);
self.imageView.frame = imageFrame;

// Adjust the label size to fit the text, and move it below the image

titleLabelFrame.size.width = labelSize.size.width;
titleLabelFrame.size.height = labelSize.size.height;
titleLabelFrame.origin.x = (self.frame.size.width / 2) - (labelSize.size.width / 2);
titleLabelFrame.origin.y = fitBoxRect.origin.y + imageFrame.size.height + kTextTopPadding;
self.titleLabel.frame = titleLabelFrame;
}

เนื่องจาก iOS 7 ยุคก่อนเริ่มมีจำนวนเพิ่มขึ้นเรื่อย ๆ นี่จึงเป็นคำตอบที่ได้รับการยอมรับใหม่
Mehlyfication

7

บน iOS 11 / Swift 4 ไม่มีคำตอบใด ๆ ที่ได้ผลจริงสำหรับฉัน ฉันพบตัวอย่างและใส่สปินของฉัน:

extension UIButton {

    func centerImageAndButton(_ gap: CGFloat, imageOnTop: Bool) {

      guard let imageView = self.currentImage,
      let titleLabel = self.titleLabel?.text else { return }

      let sign: CGFloat = imageOnTop ? 1 : -1
      self.titleEdgeInsets = UIEdgeInsetsMake((imageView.size.height + gap) * sign, -imageView.size.width, 0, 0);

      let titleSize = titleLabel.size(withAttributes:[NSAttributedStringKey.font: self.titleLabel!.font!])
      self.imageEdgeInsets = UIEdgeInsetsMake(-(titleSize.height + gap) * sign, 0, 0, -titleSize.width)
    }
}

หวังว่านี่จะช่วยใครซักคน


ขอขอบคุณที่แนะนำ Roman ว่ามีปัญหาที่ contentEdgeInsets ไม่รวมชื่อและรูปภาพทั้งหมด
แพทริค

6

การใช้รหัสของ Kenny Winker และ simeon ฉันทำรหัส swift ที่เหมาะกับฉัน

import UIKit

@IBDesignable
class TopIconButton: UIButton {
    override func layoutSubviews() {
        super.layoutSubviews()

        let kTextTopPadding:CGFloat = 3.0;

        var titleLabelFrame = self.titleLabel!.frame;


        let labelSize = titleLabel!.sizeThatFits(CGSizeMake(CGRectGetWidth(self.contentRectForBounds(self.bounds)), CGFloat.max))

        var imageFrame = self.imageView!.frame;

        let fitBoxSize = CGSizeMake(max(imageFrame.size.width, labelSize.width), labelSize.height + kTextTopPadding + imageFrame.size.    height)

        let fitBoxRect = CGRectInset(self.bounds, (self.bounds.size.width - fitBoxSize.width)/2, (self.bounds.size.height - fitBoxSize.    height)/2);

        imageFrame.origin.y = fitBoxRect.origin.y;
        imageFrame.origin.x = CGRectGetMidX(fitBoxRect) - (imageFrame.size.width/2);
        self.imageView!.frame = imageFrame;

        // Adjust the label size to fit the text, and move it below the image

        titleLabelFrame.size.width = labelSize.width;
        titleLabelFrame.size.height = labelSize.height;
        titleLabelFrame.origin.x = (self.frame.size.width / 2) - (labelSize.width / 2);
        titleLabelFrame.origin.y = fitBoxRect.origin.y + imageFrame.size.height + kTextTopPadding;
        self.titleLabel!.frame = titleLabelFrame;
        self.titleLabel!.textAlignment = .Center
    }

}

3

คุณเพียงแค่ต้องปรับการแทรกขอบทั้งสามตามขนาดของรูปภาพและป้ายชื่อของคุณ:

button.contentEdgeInsets = UIEdgeInsetsMake(0, 0, titleLabelBounds.height + 4, 0)
button.titleEdgeInsets = UIEdgeInsetsMake(image.size.height + 8, -image.size.width, 0, 0)
button.imageEdgeInsets = UIEdgeInsetsMake(0, 0, 0, -titleLabelBounds.width)

คุณสามารถรับขอบเขตหัวเรื่องของป้ายชื่อโดยเรียก sizeToFit หลังจากตั้งค่าข้อความ ระยะห่างแนวนอนควรใช้งานได้โดยไม่คำนึงถึงขนาดของข้อความตัวอักษรและรูปภาพ แต่ฉันไม่รู้วิธีแก้ปัญหาเดียวเพื่อให้ได้ระยะห่างแนวตั้งและขอบเนื้อหาด้านล่างที่สอดคล้องกัน


3

นี่คือ "หมีกับฉัน" Swift 2.0คำตอบเป็นคลาสย่อยใน หากต้องการใช้เพียงเปลี่ยนคลาสปุ่มของคุณInterface Builderเป็นVerticalButtonและจะอัปเดตตัวอย่างอย่างน่าอัศจรรย์

ฉันยังอัปเดตเพื่อคำนวณขนาดเนื้อหาภายในที่ถูกต้องสำหรับการชำระอัตโนมัติ

import UIKit

@IBDesignable

class VerticalButton: UIButton {
    @IBInspectable var padding: CGFloat = 8

    override func prepareForInterfaceBuilder() {
        super.prepareForInterfaceBuilder()

        update()
    }

    override func layoutSubviews() {
        super.layoutSubviews()

        update()
    }

    func update() {
        let imageBounds = self.imageView!.bounds
        let titleBounds = self.titleLabel!.bounds
        let totalHeight = CGRectGetHeight(imageBounds) + padding + CGRectGetHeight(titleBounds)

        self.imageEdgeInsets = UIEdgeInsets(
            top: -(totalHeight - CGRectGetHeight(imageBounds)),
            left: 0,
            bottom: 0,
            right: -CGRectGetWidth(titleBounds)
        )

        self.titleEdgeInsets = UIEdgeInsets(
            top: 0,
            left: -CGRectGetWidth(imageBounds),
            bottom: -(totalHeight - CGRectGetHeight(titleBounds)),
            right: 0
        )
    }

    override func intrinsicContentSize() -> CGSize {
        let imageBounds = self.imageView!.bounds
        let titleBounds = self.titleLabel!.bounds

        let width = CGRectGetWidth(imageBounds) > CGRectGetWidth(titleBounds) ? CGRectGetWidth(imageBounds) : CGRectGetWidth(titleBounds)
        let height = CGRectGetHeight(imageBounds) + padding + CGRectGetHeight(titleBounds)

        return CGSizeMake(width, height)
    }
}

2
จบลงด้วยการวนซ้ำไม่สิ้นสุดที่layoutSubviews()ถูกเรียกซ้ำ ๆ กันในกรณีของฉัน: การintrinsicContentSizeเข้าถึงimageViewที่ทำให้ layoutSubviewsถูกเรียกซึ่งเข้าถึงได้imageViewฯลฯ
ctietze

3

@ Tiago ฉันเปลี่ยนคำตอบของคุณเช่นนี้ มันใช้งานได้ดีกับทุกขนาด

func alignImageAndTitleVertically(padding: CGFloat = 5.0) {
        self.sizeToFit()
        let imageSize = self.imageView!.frame.size
        let titleSize = self.titleLabel!.frame.size
        let totalHeight = imageSize.height + titleSize.height + padding

        self.imageEdgeInsets = UIEdgeInsets(
            top: -(totalHeight - imageSize.height),
            left: 0,
            bottom: 0,
            right: -titleSize.width
        )

        self.titleEdgeInsets = UIEdgeInsets(
            top: 0,
            left: 0,
            bottom: -(totalHeight - titleSize.height),
            right: titleSize.height
        )
    }

3

ฉันรวบรวมคำตอบที่นี่และหาคำตอบที่เหมาะกับฉันใน Swift ฉันไม่ชอบว่าฉันจะเอาชนะสิ่งที่ใส่เข้าไป แต่มันได้ผล ฉันจะเปิดให้มีการปรับปรุงข้อเสนอแนะในความคิดเห็น ดูเหมือนว่าจะทำงานอย่างถูกต้องกับsizeToFit()และด้วยรูปแบบอัตโนมัติ

import UIKit

/// A button that displays an image centered above the title.  This implementation 
/// only works when both an image and title are set, and ignores
/// any changes you make to edge insets.
class CenteredButton: UIButton
{
    let padding: CGFloat = 0.0

    override func layoutSubviews() {
        if imageView?.image != nil && titleLabel?.text != nil {
            let imageSize: CGSize = imageView!.image!.size
            titleEdgeInsets = UIEdgeInsetsMake(0.0, -imageSize.width, -(imageSize.height + padding), 0.0)
            let labelString = NSString(string: titleLabel!.text!)
            let titleSize = labelString.sizeWithAttributes([NSFontAttributeName: titleLabel!.font])
            imageEdgeInsets = UIEdgeInsetsMake(-(titleSize.height + padding), 0.0, 0.0, -titleSize.width)
            let edgeOffset = abs(titleSize.height - imageSize.height) / 2.0;
            contentEdgeInsets = UIEdgeInsetsMake(edgeOffset, 0.0, edgeOffset, 0.0)
        }
        super.layoutSubviews()
    }

    override func sizeThatFits(size: CGSize) -> CGSize {
        let defaultSize = super.sizeThatFits(size)
        if let titleSize = titleLabel?.sizeThatFits(size),
        let imageSize = imageView?.sizeThatFits(size) {
            return CGSize(width: ceil(max(imageSize.width, titleSize.width)), height: ceil(imageSize.height + titleSize.height + padding))
        }
        return defaultSize
    }

    override func intrinsicContentSize() -> CGSize {
        let size = sizeThatFits(CGSize(width: CGFloat.max, height: CGFloat.max))
        return size
    }
}

3

ใช้สองวิธีนี้:

func titleRect(forContentRect contentRect: CGRect) -> CGRect
func imageRect(forContentRect contentRect: CGRect) -> CGRect

ตัวอย่าง:

class VerticalButton: UIButton {

  override func titleRect(forContentRect contentRect: CGRect) -> CGRect {
    let titleRect = super.titleRect(forContentRect: contentRect)
    let imageRect = super.imageRect(forContentRect: contentRect)

    return CGRect(x: 0,
                  y: contentRect.height - (contentRect.height - padding - imageRect.size.height - titleRect.size.height) / 2 - titleRect.size.height,
                  width: contentRect.width,
                  height: titleRect.height)
  }

  override func imageRect(forContentRect contentRect: CGRect) -> CGRect {
    let imageRect = super.imageRect(forContentRect: contentRect)
    let titleRect = self.titleRect(forContentRect: contentRect)

    return CGRect(x: contentRect.width/2.0 - imageRect.width/2.0,
                  y: (contentRect.height - padding - imageRect.size.height - titleRect.size.height) / 2,
                  width: imageRect.width,
                  height: imageRect.height)
  }

  private let padding: CGFloat
  init(padding: CGFloat) {
    self.padding = padding

    super.init(frame: .zero)
    self.titleLabel?.textAlignment = .center
  }

  required init?(coder aDecoder: NSCoder) { fatalError() }
}

extension UIButton {

  static func vertical(padding: CGFloat) -> UIButton {
    return VerticalButton(padding: padding)
  }
}

และคุณสามารถใช้:

let myButton = UIButton.vertical(padding: 6)

3

Swift 5 - วิธีด้านล่างใช้ได้สำหรับฉัน

func centerVerticallyWithPadding(padding : CGFloat) {
        guard
            let imageViewSize = self.imageView?.frame.size,
            let titleLabelSize = self.titleLabel?.frame.size else {
            return
        }

        let totalHeight = imageViewSize.height + titleLabelSize.height + padding

        self.imageEdgeInsets = UIEdgeInsets(
            top: max(0, -(totalHeight - imageViewSize.height)),
            left: 0.0,
            bottom: 0.0,
            right: -titleLabelSize.width
        )

        self.titleEdgeInsets = UIEdgeInsets(
            top: (totalHeight - imageViewSize.height),
            left: -imageViewSize.width,
            bottom: -(totalHeight - titleLabelSize.height),
            right: 0.0
        )

        self.contentEdgeInsets = UIEdgeInsets(
            top: 0.0,
            left: 0.0,
            bottom: titleLabelSize.height,
            right: 0.0
        )
    }

ตรวจสอบให้แน่ใจว่าชื่อปุ่มของคุณไม่ถูกตัดทอนลงในกระดานเรื่องราว / xib อย่างอื่นให้ไปที่
Solution 2

class SVVerticalButton: UIButton {

    override func layoutSubviews() {
        super.layoutSubviews()
        let padding : CGFloat = 2.0
        if let imageView = self.imageView {
            imageView.frame.origin.x = (self.bounds.size.width - imageView.frame.size.width) / 2.0
            imageView.frame.origin.y = max(0,(self.bounds.size.height - (imageView.frame.size.height + (titleLabel?.frame.size.height ?? 0.0) + padding)) / 2.0)
        }
        if let titleLabel = self.titleLabel {
            titleLabel.frame.origin.x = 0
            titleLabel.frame.origin.y = self.bounds.size.height - titleLabel.frame.size.height
            titleLabel.frame.size.width = self.bounds.size.width
            titleLabel.textAlignment = .center
        }
    }

}

2

ฉันพบว่าคำตอบของไซเมียนน่าจะดีที่สุด แต่มันก็ให้ผลลัพธ์ที่แปลกแก่ฉันในบางปุ่มและฉันก็ไม่สามารถหาสาเหตุได้ ดังนั้นการใช้คำตอบของเขาเป็นฐานฉันใช้ปุ่มของฉันตามด้านล่าง:

#define PADDING 2.0f

@implementation OOButtonVerticalImageText

-(CGSize) intrinsicContentSize {
  CGSize size = [super intrinsicContentSize];
  CGFloat labelHeight = 0.0f;
  CGSize titleSize = [self.titleLabel sizeThatFits:CGSizeMake([self contentRectForBounds:self.bounds].size.width, CGFLOAT_MAX)];
  labelHeight = titleSize.height;
  return CGSizeMake(MAX(titleSize.width, self.imageView.image.size.width), self.imageView.image.size.height + labelHeight + PADDING);
}

-(void) layoutSubviews {
  [super layoutSubviews];

  CGSize titleSize = [self.titleLabel sizeThatFits:CGSizeMake([self contentRectForBounds:self.bounds].size.width, CGFLOAT_MAX)];
  self.titleLabel.frame = CGRectMake((self.bounds.size.width - titleSize.width)/2.0f,
                                     self.bounds.size.height - titleSize.height - PADDING,
                                     titleSize.width,
                                     titleSize.height);

  CGSize ivSize = self.imageView.frame.size;
  self.imageView.frame = CGRectMake((self.bounds.size.width - ivSize.width)/2.0f,
                                    self.titleLabel.frame.origin.y - ivSize.height - PADDING,
                                    ivSize.width,
                                    ivSize.height);
}

@end

1

นี่คือ subclass ของฉันUIButtonที่แก้ปัญหานี้:

@implementation MyVerticalButton

@synthesize titleAtBottom; // BOOL property

- (id)initWithFrame:(CGRect)frame
{
  self = [super initWithFrame:frame];
  if (self) {
    self.titleAtBottom = YES;
  }
  return self;
}

- (CGSize)sizeThatFits:(CGSize)size {
  self.titleLabel.text = [self titleForState: self.state];

  UIEdgeInsets imageInsets = self.imageEdgeInsets;
  UIEdgeInsets titleInsets = self.titleEdgeInsets;

  CGSize imageSize = [self imageForState: self.state].size;
  if (!CGSizeEqualToSize(imageSize, CGSizeZero)) {
    imageSize.width += imageInsets.left + imageInsets.right;
    imageSize.height += imageInsets.top + imageInsets.bottom;

  }

  CGSize textSize = [self.titleLabel sizeThatFits: CGSizeMake(size.width - titleInsets.left - titleInsets.right,
                                                              size.height -(imageSize.width +
                                                                            titleInsets.top+titleInsets.bottom))];
  if (!CGSizeEqualToSize(textSize, CGSizeZero)) {
    textSize.width += titleInsets.left + titleInsets.right;
    textSize.height += titleInsets.top + titleInsets.bottom;
  }

  CGSize result = CGSizeMake(MAX(textSize.width, imageSize.width),
                             textSize.height + imageSize.height);
  return result;
}

- (void)layoutSubviews {
  // needed to update all properities of child views:
  [super layoutSubviews];

  CGRect bounds = self.bounds;

  CGRect titleFrame = UIEdgeInsetsInsetRect(bounds, self.titleEdgeInsets);
  CGRect imageFrame = UIEdgeInsetsInsetRect(bounds, self.imageEdgeInsets);
  if (self.titleAtBottom) {
    CGFloat titleHeight = [self.titleLabel sizeThatFits: titleFrame.size].height;
    titleFrame.origin.y = CGRectGetMaxY(titleFrame)-titleHeight;
    titleFrame.size.height = titleHeight;
    titleFrame = CGRectStandardize(titleFrame);
    self.titleLabel.frame = titleFrame;

    CGFloat imageBottom = CGRectGetMinY(titleFrame)-(self.titleEdgeInsets.top+self.imageEdgeInsets.bottom);
    imageFrame.size.height = imageBottom - CGRectGetMinY(imageFrame);
    self.imageView.frame = CGRectStandardize(imageFrame);
  } else {
    CGFloat titleHeight = [self.titleLabel sizeThatFits: titleFrame.size].height;
    titleFrame.size.height = titleHeight;
    titleFrame = CGRectStandardize(titleFrame);
    self.titleLabel.frame = titleFrame;

    CGFloat imageTop = CGRectGetMaxY(titleFrame)+(self.titleEdgeInsets.bottom+self.imageEdgeInsets.top);
    imageFrame.size.height = CGRectGetMaxY(imageFrame) - imageTop;
    self.imageView.frame = CGRectStandardize(imageFrame);
  }
}

- (void)setTitleAtBottom:(BOOL)newTitleAtBottom {
  if (titleAtBottom!=newTitleAtBottom) {
    titleAtBottom=newTitleAtBottom;
    [self setNeedsLayout];
  }
}

@end

แค่นั้นแหละ. ทำงานเหมือนจับใจ ปัญหาอาจปรากฏขึ้นหากปุ่มเล็กไปจนถึงชื่อและข้อความ


มันใช้งานได้อย่างมีเสน่ห์อย่างแน่นอน! และด้วยเลย์เอาต์อัตโนมัติด้วย ขอบคุณมากที่แบ่งปันโซลูชันนี้ ฉันกำลังบ้ากับสิ่งนี้และหันไปใช้สร้างคลาสย่อย UIControl ของฉันเอง
valeCocoa

1

ฉันคิดว่าหนึ่งในวิธีที่ดีที่สุดที่จะทำคือโดย subclassing UIButton และแทนที่วิธีการแสดงผลบาง:

- (void)awakeFromNib
{
    [super awakeFromNib];
    [self setupSubViews];
}

- (instancetype)init
{
    if (self = [super init])
    {
        [self setupSubViews];
    }
    return self;
}

- (void)setupSubViews
{
    [self.imageView setTranslatesAutoresizingMaskIntoConstraints:NO];
    [self addConstraint:[NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeCenterX relatedBy:NSLayoutRelationEqual toItem:self.imageView attribute:NSLayoutAttributeCenterX multiplier:1 constant:0]];
    [self.titleLabel setTranslatesAutoresizingMaskIntoConstraints:NO];
    [self addConstraints:[NSLayoutConstraint constraintsWithVisualFormat:@"V:|[imageView][titleLabel]|" options:NSLayoutFormatAlignAllCenterX metrics:nil views:@{@"imageView": self.imageView, @"titleLabel": self.titleLabel}]];
}

- (CGSize)intrinsicContentSize
{
    CGSize imageSize = self.imageView.image.size;
    CGSize titleSize = [self.titleLabel.text sizeWithAttributes:@{NSFontAttributeName: self.titleLabel.font}];
    return CGSizeMake(MAX(imageSize.width, titleSize.width), imageSize.height + titleSize.height);
}

1

วิธีการแก้ปัญหา @ simeon ใน Objective-C

#import "CenteredButton.h"

@implementation CenteredButton

- (CGRect)titleRectForContentRect:(CGRect)contentRect
{
    CGRect rect = [super titleRectForContentRect: contentRect];
    return CGRectMake(0,
                      contentRect.size.height - rect.size.height - 5,
                      contentRect.size.width,
                      rect.size.height);
}

- (CGRect)imageRectForContentRect:(CGRect)contentRect
{
    CGRect rect = [super imageRectForContentRect: contentRect];
    CGRect titleRect = [self titleRectForContentRect: contentRect];

    return CGRectMake(contentRect.size.width / 2.0 - rect.size.width / 2.0,
                      (contentRect.size.height - titleRect.size.height)/2.0 - rect.size.height/2.0,
                      rect.size.width,
                      rect.size.height);
}

- (CGSize)intrinsicContentSize {
    CGSize imageSize = [super intrinsicContentSize];

    if (self.imageView.image) {
        UIImage* image = self.imageView.image;
        CGFloat labelHeight = 0.0;

        CGSize labelSize = [self.titleLabel sizeThatFits: CGSizeMake([self contentRectForBounds: self.bounds].size.width, CGFLOAT_MAX)];
        if (CGSizeEqualToSize(imageSize, labelSize)) {
            labelHeight = imageSize.height;
        }

        return CGSizeMake(MAX(labelSize.width, imageSize.width), image.size.height + labelHeight + 5);
    }

    return imageSize;
}

- (id) initWithFrame:(CGRect)frame {
    self = [super initWithFrame:frame];
     if (self) {
         [self centerTitleLabel];
     }
    return self;

}

- (id)initWithCoder:(NSCoder *)aDecoder {
    self = [super initWithCoder:aDecoder];
    if (self) {
        [self centerTitleLabel];
    }
    return self;
}

- (void)centerTitleLabel {
    self.titleLabel.textAlignment = NSTextAlignmentCenter;
}

@end

ฉันคิดว่า intrinsicContentSize ไม่ถูกต้องที่นี่ ฉันไม่เข้าใจว่าส่วนใดของ CGSizeEqualToSize มีไว้สำหรับคุณ แต่จะมีขนาดป้ายกำกับ> 0 ถ้าขนาดของป้ายตรงกับ intrinsicContentSize ของ UILabel มันควรจะเพียงพอที่จะเพียงแค่กลับมาCGSizeMake(MAX(labelSize.width, image.size.width), image.size.height + labelSize.height + 5.0)ในกรณีที่
โอลิเวอร์

1

หากคุณกำลังใช้แบบอักษรที่กำหนดเองการคำนวณขนาด titleLabel จะไม่ทำงานอย่างถูกต้องคุณควรแทนที่ด้วย

let titleLabelSize = self.titleLabel?.text?.size(withAttributes: [NSAttributedStringKey.font: self.titleLabel!.font!])


1

แทนที่จะต้องผ่านนรกที่พยายามวางตำแหน่งไอคอนและข้อความด้วยการแทรกขอบคุณสามารถสร้าง NSAttributedString ด้วยภาพของคุณเป็นสิ่งที่แนบมาและตั้งเป็นชื่อเรื่องที่เกี่ยวข้องกับปุ่มของคุณแทน:

let titleText = NSAttributedString(string: yourTitle, attributes: attributes)
let imageAttachment = NSTextAttachment()
imageAttachment.image = yourImage

let title = NSMutableAttributedString(string: "")
title.append(NSAttributedString(attachment: imageAttachment))
title.append(titleText)

button.setAttributedTitle(title, for: .normal)

ไม่ทำงานสำหรับคำถาม OPs ที่ข้อความควรอยู่กึ่งกลางด้านล่างของภาพ UIButtonรูปแบบฟิลด์ข้อความของA เพื่อแสดงเพียง 1 บรรทัดดังนั้นจึงไม่ทำงานแม้ว่าจะใช้ตัวแบ่งบรรทัดในสตริงที่ประกอบ จะเป็นทางออกที่ดีเป็นอย่างอื่น
มานูเอล

นอกจากนี้ยังเป็นสิ่งสำคัญที่จะต้องกำหนดbutton.titleLabel?.numberOfLinesเพื่อให้ได้จำนวนบรรทัดที่ต้องการ
swearwolf

1

โซลูชันรองรับหลายภาษา:

พวกโซลูชั่นที่ยอดเยี่ยมมากมาย แต่ฉันต้องการเพิ่มบันทึกย่อที่นี่สำหรับผู้ที่ใช้การโลคัลไลเซชัน

คุณต้องย้อนกลับค่า EdgeInstets ทางซ้ายและขวาเพื่อให้ปุ่มวางอย่างถูกต้องในกรณีที่เปลี่ยนทิศทางภาษาจาก LtR เป็น RtL

ใช้โซลูชันที่คล้ายกันฉันจะใช้มันดังนี้

extension UIButton {

    func alignVertical(spacing: CGFloat, lang: String) {
        guard let imageSize = self.imageView?.image?.size,
            let text = self.titleLabel?.text,
            let font = self.titleLabel?.font
        else { return }

        let labelString = NSString(string: text)
        let titleSize = labelString.size(
            withAttributes: [NSAttributedString.Key.font: font]
        )

        var titleLeftInset: CGFloat = -imageSize.width
        var titleRigtInset: CGFloat = 0.0

        var imageLeftInset: CGFloat = 0.0
        var imageRightInset: CGFloat = -titleSize.width

        if Locale.current.languageCode! != "en" { // If not Left to Right language
            titleLeftInset = 0.0
            titleRigtInset = -imageSize.width

            imageLeftInset = -titleSize.width
            imageRightInset = 0.0
        }

        self.titleEdgeInsets = UIEdgeInsets(
            top: 0.0,
            left: titleLeftInset,
            bottom: -(imageSize.height + spacing),
            right: titleRigtInset
        )
        self.imageEdgeInsets = UIEdgeInsets(
            top: -(titleSize.height + spacing),
            left: imageLeftInset,
            bottom: 0.0,
            right: imageRightInset
        )
        let edgeOffset = abs(titleSize.height - imageSize.height) / 2.0;
        self.contentEdgeInsets = UIEdgeInsets(
            top: edgeOffset,
            left: 0.0,
            bottom: edgeOffset,
            right: 0.0
        )
    }
}

1
มีภาษา LTR ที่ไม่ใช่ภาษาอังกฤษมากมาย คุณดีกว่าการตรวจสอบผู้ใช้ที่มีประสิทธิภาพ InterfaceLayoutDirection ที่ปุ่ม
Alexsander Akers

1

รูปภาพด้านบนและปุ่มชื่อด้านล่างพร้อมด้วยการแบ่งคลาส UIButton

class VerticalButton: UIButton {
  override func layoutSubviews() {
    super.layoutSubviews()
    let padding: CGFloat = 8
    let iH = imageView?.frame.height ?? 0
    let tH = titleLabel?.frame.height ?? 0
    let v: CGFloat = (frame.height - iH - tH - padding) / 2
    if let iv = imageView {
      let x = (frame.width - iv.frame.width) / 2
      iv.frame.origin.y = v
      iv.frame.origin.x = x
    }

    if let tl = titleLabel {
      let x = (frame.width - tl.frame.width) / 2
      tl.frame.origin.y = frame.height - tl.frame.height - v
      tl.frame.origin.x = x
    }
  }
}

1

นั่นเป็น overkill แน่นอนสำหรับคำถามนี้ แต่ ... ในหนึ่งในโครงการของฉันก่อนอื่นฉันต้องใช้ปุ่มที่มีไอคอนชิดซ้ายสุด จากนั้นเรามีปุ่มอื่นที่มีชื่ออยู่ใต้ภาพ ฉันค้นหาวิธีแก้ปัญหาที่มีอยู่ แต่ไม่มีโชคดังนั้นปุ่มปรับตำแหน่งได้:

@IBDesignable
class AlignableButton: UIButton {

override class var requiresConstraintBasedLayout: Bool {
    return true
}

@objc enum IconAlignment: Int {
    case top, left, right, bottom
}

// MARK: - Designables
@IBInspectable var iconAlignmentValue: Int {
    set {
        iconAlignment = IconAlignment(rawValue: newValue) ?? .left
    }
    get {
        return iconAlignment.rawValue
    }
}

var iconAlignment: IconAlignment = .left

@IBInspectable var titleAlignmentValue: Int {
    set {
        titleAlignment = NSTextAlignment(rawValue: newValue) ?? .left
    }
    get {
        return titleAlignment.rawValue
    }
}

var titleAlignment: NSTextAlignment = .left

// MARK: - Corner Radius
@IBInspectable
var cornerRadius: CGFloat {
    get {
        return layer.cornerRadius
    }
    set {
        layer.masksToBounds = (newValue != 0)
        layer.cornerRadius = newValue
    }
}

// MARK: - Content size
override var intrinsicContentSize: CGSize {
    
    switch iconAlignment {
    case .top, .bottom:
        return verticalAlignedIntrinsicContentSize
    
    default:
        return super.intrinsicContentSize
    }
}

private var verticalAlignedIntrinsicContentSize: CGSize {
    
    if let imageSize = imageView?.intrinsicContentSize,
        let labelSize = titleLabel?.intrinsicContentSize {
        
        let width = max(imageSize.width, labelSize.width) + contentEdgeInsets.left + contentEdgeInsets.right
        let height = imageSize.height + labelSize.height + contentEdgeInsets.top + contentEdgeInsets.bottom
        
        return CGSize(
            width: ceil(width),
            height: ceil(height)
        )
    }
    
    return super.intrinsicContentSize
}

// MARK: - Image Rect
override func imageRect(forContentRect contentRect: CGRect) -> CGRect {
    
    switch iconAlignment {
    case .top:
        return topAlignedImageRect(forContentRect: contentRect)
    case .bottom:
        return bottomAlignedImageRect(forContentRect: contentRect)
    case .left:
        return leftAlignedImageRect(forContentRect: contentRect)
    case .right:
        return rightAlignedImageRect(forContentRect: contentRect)
    }
}

func topAlignedImageRect(forContentRect contentRect: CGRect) -> CGRect {
    let rect = super.imageRect(forContentRect: contentRect)
    
    let x = (contentRect.width - rect.width) / 2.0 + contentRect.minX
    let y = contentRect.minY
    let w = rect.width
    let h = rect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: imageEdgeInsets)
}

func bottomAlignedImageRect(forContentRect contentRect: CGRect) -> CGRect {
    let rect = super.imageRect(forContentRect: contentRect)
    
    let x = (contentRect.width - rect.width) / 2.0 + contentRect.minX
    let y = contentRect.height - rect.height + contentRect.minY
    let w = rect.width
    let h = rect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: imageEdgeInsets)
}

func leftAlignedImageRect(forContentRect contentRect: CGRect) -> CGRect {
    let rect = super.imageRect(forContentRect: contentRect)
    
    let x = contentRect.minX
    let y = (contentRect.height - rect.height) / 2 + contentRect.minY
    let w = rect.width
    let h = rect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: imageEdgeInsets)
}

func rightAlignedImageRect(forContentRect contentRect: CGRect) -> CGRect {
    let rect = super.imageRect(forContentRect: contentRect)
    
    let x = (contentRect.width - rect.width) + contentRect.minX
    let y = (contentRect.height - rect.height) / 2 + contentRect.minY
    let w = rect.width
    let h = rect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: imageEdgeInsets)
}

// MARK: - Title Rect
override func titleRect(forContentRect contentRect: CGRect) -> CGRect {
    
    switch iconAlignment {
    case .top:
        return topAlignedTitleRect(forContentRect: contentRect)
    case .bottom:
        return bottomAlignedTitleRect(forContentRect: contentRect)
    case .left:
        return leftAlignedTitleRect(forContentRect: contentRect)
    case .right:
        return rightAlignedTitleRect(forContentRect: contentRect)
    }
}

func topAlignedTitleRect(forContentRect contentRect: CGRect) -> CGRect {
    
    let rect = super.titleRect(forContentRect: contentRect)

    let x = contentRect.minX
    let y = contentRect.height - rect.height + contentRect.minY
    let w = contentRect.width
    let h = rect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: titleEdgeInsets)
}

func bottomAlignedTitleRect(forContentRect contentRect: CGRect) -> CGRect {
    
    let rect = super.titleRect(forContentRect: contentRect)
    
    let x = contentRect.minX
    let y = contentRect.minY
    let w = contentRect.width
    let h = rect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: titleEdgeInsets)
}

func leftAlignedTitleRect(forContentRect contentRect: CGRect) -> CGRect {
    
    let titleRect = super.titleRect(forContentRect: contentRect)
    let imageRect = self.imageRect(forContentRect: contentRect)
    
    let x = imageRect.width + imageRect.minX
    let y = (contentRect.height - titleRect.height) / 2.0 + contentRect.minY
    let w = contentRect.width - imageRect.width * 2.0
    let h = titleRect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: titleEdgeInsets)
}

func rightAlignedTitleRect(forContentRect contentRect: CGRect) -> CGRect {
    
    let titleRect = super.titleRect(forContentRect: contentRect)
    let imageRect = self.imageRect(forContentRect: contentRect)

    let x = contentRect.minX + imageRect.width
    let y = (contentRect.height - titleRect.height) / 2.0 + contentRect.minY
    let w = contentRect.width - imageRect.width * 2.0
    let h = titleRect.height
    
    return CGRect(
        x: x,
        y: y,
        width: w,
        height: h
    ).inset(by: titleEdgeInsets)
}

// MARK: - Lifecycle
override func awakeFromNib() {
    super.awakeFromNib()
    
    titleLabel?.textAlignment = titleAlignment
}

override func prepareForInterfaceBuilder() {
    super.prepareForInterfaceBuilder()
    
    titleLabel?.textAlignment = titleAlignment
}
}

หวังว่าคุณจะพบว่ามีประโยชน์


0

บางอย่างเช่นนี้ภายในUIButtonsubclass

public override func layoutSubviews() {
    super.layoutSubviews()

    imageEdgeInsets = UIEdgeInsetsMake(-10, 0, 0, 0)
    titleEdgeInsets = UIEdgeInsetsMake(0, -bounds.size.width/2 - 10, -30, 0)
}

0

มันค่อนข้างง่าย

แทนสิ่งนี้:

   button.setImage(UIImage(named: "image"), forState: .Normal)

ใช้สิ่งนี้:

   button.setBackgroundImage(UIImage(named: "image", forState: .Normal)

จากนั้นคุณสามารถเพิ่มข้อความบนปุ่มได้อย่างง่ายดายโดยใช้:

// button.titleLabel! .font = UIFont (ชื่อ: "FontName", ขนาด: 30)

 button.setTitle("TitleText", forState: UIControlState.Normal)
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.