Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Swift: ScrollRectToVisible not working

After trying out most of the solutions posted here, I'm still having trouble moving the textfield to show up above keyboard in a scrollview.

These are links I followed from Stackoverflow solutions: Link 1 Link 2 Link 3

I'm working on a signup screen that has 1 field behind the keyboard when it shows up.

Here's my code:

class SignUpViewController: UIViewController, UITextFieldDelegate, UIScrollViewDelegate, UIPopoverPresentationControllerDelegate {

@IBOutlet var firstNameTextField: UITextField!
@IBOutlet var lastNameTextField: UITextField!
@IBOutlet var phoneNumberTextField: UITextField!
@IBOutlet var emailTextField: UITextField!
@IBOutlet var submitButton: UIButton!
@IBOutlet var professionButton: UIButton!

var scrollView: UIScrollView?
var activeTextField:UITextField? = UITextField()

override func viewDidLoad() {
    super.viewDidLoad()

    let notificationCenter = NSNotificationCenter.defaultCenter()

    notificationCenter.addObserver(self, selector: #selector(SignUpViewController.keyboardWasShown(_:)), name: UIKeyboardWillShowNotification, object: nil)
    notificationCenter.addObserver(self, selector: #selector(SignUpViewController.keyboardWillBeHidden(_:)), name: UIKeyboardWillHideNotification, object: nil)
    scrollView = UIScrollView(frame: CGRect(x: 0, y: 0, width: self.view.frame.size.width, height: self.view.frame.size.height))
    scrollView!.contentSize = CGSizeMake(self.view.frame.width, self.view.frame.height)

    defaultSettings()
}

func defaultSettings() {
    self.firstNameTextField.delegate = self
    self.lastNameTextField.delegate = self
    self.emailTextField.delegate = self
    self.phoneNumberTextField.delegate = self
}

func deregisterFromKeyboardNotifications()
{
    //Removing notifies on keyboard appearing
    NSNotificationCenter.defaultCenter().removeObserver(self, name: UIKeyboardWillShowNotification, object: nil)
    NSNotificationCenter.defaultCenter().removeObserver(self, name: UIKeyboardWillHideNotification, object: nil)
}

func keyboardWasShown(notification: NSNotification)
{
    //Need to calculate keyboard exact size due to Apple suggestions
    //self.scrollView!.scrollEnabled = true
    var info : NSDictionary = notification.userInfo!
    var keyboardSize = (info[UIKeyboardFrameBeginUserInfoKey] as? NSValue)?.CGRectValue().size
    var contentInsets : UIEdgeInsets = UIEdgeInsetsMake(0.0, 0.0, keyboardSize!.height, 0.0)

    self.scrollView!.contentInset = contentInsets
    self.scrollView!.scrollIndicatorInsets = contentInsets

    var aRect : CGRect = self.view.frame
    aRect.size.height -= keyboardSize!.height
    if (!CGRectContainsPoint(aRect, activeTextField!.frame.origin))
    {
//            print(activeTextField?.frame)
//            var scrollPoint = CGPointMake(0.0, activeTextField!.frame.origin.y - (keyboardSize!.height-15))
        self.scrollView!.scrollRectToVisible((activeTextField?.frame)!, animated: true)
        //self.scrollView?.setContentOffset(scrollPoint, animated: true)
    }
}


func keyboardWillBeHidden(notification: NSNotification)
{
    //Once keyboard disappears, restore original positions
    //var info : NSDictionary = notification.userInfo!
    //var keyboardSize = (info[UIKeyboardFrameBeginUserInfoKey] as? NSValue)?.CGRectValue().size
    var contentInsets : UIEdgeInsets = UIEdgeInsetsZero
    self.scrollView!.contentInset = contentInsets
    self.scrollView!.scrollIndicatorInsets = contentInsets
//        self.view.endEditing(true)
//        self.scrollView!.scrollEnabled = false

}

func textFieldDidBeginEditing(textField: UITextField)
{
    activeTextField = textField
}

func textFieldDidEndEditing(textField: UITextField)
{
    activeTextField = nil
}

As you can see, I've tried scrollRectToVisible with frame and setContentOffset with Point. Both didn't work. Whereas the code picks on the emailTextField right as the hidden textfield.

like image 543
Viginesh Avatar asked Jun 29 '16 12:06

Viginesh


2 Answers

I was also struggling with the same issue as you did, I don't know if you were successful and found solution, but I finally used setContentOffset function instead of scrollRectToVisible and it worked.

Swift 3.x example:

if (!aRect.contains(activeTextView!.frame.origin)) {
                self.scrollView.setContentOffset(CGPoint(x:0, y:self.activeTextView!.frame.origin.y), animated: true)
}
like image 68
ivo Avatar answered Oct 29 '22 15:10

ivo


SWIFT 4 Try this

scrollView.scrollRectToVisible(myElementView.frame, animated: true)

where myElementView can be any element

like image 1
Álvaro Agüero Avatar answered Oct 29 '22 17:10

Álvaro Agüero