How to resize UITextView while typing inside it?
Asked Answered
A

5

15

I'm creating a comment section much like the one Facebook uses for it's messaging section in it's iOS app. I want the UITextView to resize the height so the text I'm typing fits inside it rather than you having to scroll to see the text that overflows. Any ideas how I might go about doing this? I've looked into maybe using a CGRect that is assigned to the size and height of the text view which then matches the content size:

CGRect textFrame = textView.frame;
textFrame.size.height = textView.contentSize.height;
textView.frame = textFrame;

I assume I need some sort of function that detects when the text reaches the bounds of the UITextView and then resizes the height of the view? Has anyone struggled with this same concept?

Aversion answered 21/11, 2012 at 12:51 Comment(1)
You can find detailed working solution hereRanitta
B
23

You can adjust frame in this delegate method, do not forget to set textView's delegate to self.

-(BOOL)textView:(UITextView *)_textView shouldChangeTextInRange:(NSRange)range replacementText:(NSString *)text {
      [self adjustFrames];  
      return YES;
}


-(void) adjustFrames
{
   CGRect textFrame = textView.frame;
   textFrame.size.height = textView.contentSize.height;
   textView.frame = textFrame;
}

this solution is for iOS6 and prior... for iOS7 refer to this

StackOverflow Answer

Belloc answered 21/11, 2012 at 13:0 Comment(2)
Oh really? I didn't think it'd be that simple.. ok thanks I'll give that a go!Aversion
to make working in iOS 7+, add adjustFrames in - (void)textViewDidChange:(UITextView *)textViewCufic
S
8

This is my solution, using autolayout and textView.contentSize.height. Tested on iOS8 Xcode6.3 beta4.

There's one catch about the setContentOffset at the end. I put it to avoid "wrong contentOffset" artefact when line count changes. It adds an extra unwanted blank space below the last line and it doesn't look very nice unless you set it back right after changing the constraint. Took me hours to figure this out!

// set this up somewhere
let minTextViewHeight: CGFloat = 32
let maxTextViewHeight: CGFloat = 64

func textViewDidChange(textView: UITextView) {

    var height = ceil(textView.contentSize.height) // ceil to avoid decimal

    if (height < minTextViewHeight + 5) { // min cap, + 5 to avoid tiny height difference at min height
        height = minTextViewHeight
    }
    if (height > maxTextViewHeight) { // max cap
        height = maxTextViewHeight
    }

    if height != textViewHeight.constant { // set when height changed
        textViewHeight.constant = height // change the value of NSLayoutConstraint
        textView.setContentOffset(CGPointZero, animated: false) // scroll to top to avoid "wrong contentOffset" artefact when line count changes
    }
}
Starstudded answered 6/4, 2015 at 9:26 Comment(6)
where is textViewHeight declared?Mccown
@Mccown it's a NSLayoutConstraint object for the height of the UITextView. I forgot to add sorry.Starstudded
I'm pretty new to this. Can you please elaborate/add more code to show how to get textViewHeight?Mccown
@Mccown If you are using storyboard, then it will be an IBOutlet variable like @IBOutlet weak var textViewHeight: NSLayoutConstraint! which you connect to the height constraint of your UITextView in interface builder.Starstudded
this works pretty well for me and i'm happy but i wonder if there's something small i could change to make it even better. the cursor seems to always be vertically centered. this looks good with one line: dropbox.com/s/dwjbjhcm80a6co5/Screenshot%202016-09-28%2022.54.10.png but not so good with two lines: dropbox.com/s/7216694bohuqfin/… ideally the second line would stay flat on the bottom. I assume this is because of the cursor vertically centered.Mccown
This almost worked for me, but I had to add this line between setting the constraint and the content offset: textView.bounds.size.height = textView.contentSize.height. Also I noticed that using textView.contentOffset = CGPointZero doesn't work.Agonizing
P
5

On the TableViewController that holds the UITextView, update the data from the tableViewDataSource that gets put in the cell and then simply call this:

tableView.beginUpdates()
tableView.endUpdates()

Unlike tableView.reloadData(), this does not call resignFirstResponder

Peery answered 1/6, 2017 at 13:2 Comment(1)
This is perfect! Thanks. Here the same in ObjC: - (void)textViewDidChange:(UITextView *)textView { [self.tableView beginUpdates]; [self.tableView endUpdates]; } You even get the animation for free ;)Topic
L
5

First set the minimum height constraints to your TextView:

textView.heightAnchor.constraint(greaterThanOrEqualTo: view.heightAnchor, multiplier: 0.20)

(Make sure you are setting greaterThanOrEqualTo Constraint so that if intrinsic content height is more than this height, it takes intrinsic content height )

OR simple constant

textView.heightAnchor.constraint(greaterThanOrEqualToConstant: someConstant)

While configuring your textView, set isScrollEnabled to false

textView.isScrollEnabled = false

Now when you type on the textView, its intrinsic content size height will increase and it will push views below it automatically.

Lobbyism answered 13/9, 2017 at 7:39 Comment(1)
one of the best answers to adjust text view height.Phenformin
S
1

contentsize will not work in ios 7. Try this:

CGFloat textViewContentHeight = textView.contentSize.height;

 textViewContentHeight = ceilf([textView sizeThatFits:textView.frame.size].height + 9);
Sulphurate answered 27/1, 2014 at 7:55 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.