キーボードが表示されたら、ビューを上に移動したいと思います。キーボード(高さ:216)は、その高さでビューを押し上げる必要があります。これは簡単なコードで可能ですか?
ビューup
を移動するには、そのcenter
を変更します。最初に、元のプロパティをCGPoint
プロパティに保持します。
- (void)viewDidLoad
{
...
self.originalCenter = self.view.center;
...
}
次に、キーボードが表示されたら必要に応じて変更します。
self.view.center = CGPointMake(self.originalCenter.x, /* new calculated y */);
最後に、キーボードが非表示になったら復元します。
self.view.center = self.originalCenter;
必要に応じてアニメーションシュガーを追加します
キーボードがいつ表示されるかを知る方法は複数あります。
監視IKeyboardDidShowNotification通知。
/* register notification in any of your initWithNibName:bundle:, viewDidLoad, awakeFromNib, etc. */
{
...
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardDidShow:) name:UIKeyboardDidShowNotification object:nil];
...
}
- (void)keyboardDidShow:(NSNotification *)note
{
/* move your views here */
}
UIKeyboardDidHideNotification
を使用して反対の操作を行います。
-または-
実装ITextFieldDelegate
ビューを移動するために開始/終了を編集するときに検出します。
- (void)textFieldDidBeginEditing:(UITextField *)textField
{
/* keyboard is visible, move views */
}
- (void)textFieldDidEndEditing:(UITextField *)textField
{
/* resign first responder, hide keyboard, move views */
}
実際のテキストフィールドに応じて、ユーザーが編集しているフィールドを追跡する必要がある場合があります。ビューを移動しすぎないようにタイマーを追加します。
このようにします。キーボードが表示された後、このコードを使用します
- (void)textFieldDidBeginEditing:(UITextField *)textField
{
[UIView beginAnimations:nil context:NULL];
[UIView setAnimationDuration:0.25];
self.view.frame = CGRectMake(0,-10,320,480);
[UIView commitAnimations];
}
これは、これを達成する最も簡単で効率的な方法です。
次の定数を追加します。
static const CGFloat KEYBOARD_ANIMATION_DURATION = 0.3;
static const CGFloat MINIMUM_SCROLL_FRACTION = 0.2;
static const CGFloat MAXIMUM_SCROLL_FRACTION = 0.8;
static const CGFloat PORTRAIT_KEYBOARD_HEIGHT = 216;
static const CGFloat LANDSCAPE_KEYBOARD_HEIGHT = 162;
これをView Controllerに追加します:
CGFloat animatedDistance;
そして、これらのメソッドをコードに追加します。
- (void)textFieldDidBeginEditing:(UITextField *)textField{
CGRect textFieldRect =
[self.view.window convertRect:textField.bounds fromView:textField];
CGRect viewRect =
[self.view.window convertRect:self.view.bounds fromView:self.view];
CGFloat midline = textFieldRect.Origin.y + 0.5 * textFieldRect.size.height;
CGFloat numerator =
midline - viewRect.Origin.y
- MINIMUM_SCROLL_FRACTION * viewRect.size.height;
CGFloat denominator =
(MAXIMUM_SCROLL_FRACTION - MINIMUM_SCROLL_FRACTION)
* viewRect.size.height;
CGFloat heightFraction = numerator / denominator;
if (heightFraction < 0.0)
{
heightFraction = 0.0;
}
else if (heightFraction > 1.0)
{
heightFraction = 1.0;
}
UIInterfaceOrientation orientation =
[[UIApplication sharedApplication] statusBarOrientation];
if (orientation == UIInterfaceOrientationPortrait ||
orientation == UIInterfaceOrientationPortraitUpsideDown)
{
animatedDistance = floor(PORTRAIT_KEYBOARD_HEIGHT * heightFraction);
}
else
{
animatedDistance = floor(LANDSCAPE_KEYBOARD_HEIGHT * heightFraction);
}
CGRect viewFrame = self.view.frame;
viewFrame.Origin.y -= animatedDistance;
[UIView beginAnimations:nil context:NULL];
[UIView setAnimationBeginsFromCurrentState:YES];
[UIView setAnimationDuration:KEYBOARD_ANIMATION_DURATION];
[self.view setFrame:viewFrame];
[UIView commitAnimations];
}
- (void)textFieldDidEndEditing:(UITextField *)textfield{
CGRect viewFrame = self.view.frame;
viewFrame.Origin.y += animatedDistance;
[UIView beginAnimations:nil context:NULL];
[UIView setAnimationBeginsFromCurrentState:YES];
[UIView setAnimationDuration:KEYBOARD_ANIMATION_DURATION];
[self.view setFrame:viewFrame];
[UIView commitAnimations];
}
ビューの中心ではなく、フレームの原点を調整したことを除いて、djromeroと同様の方法でこれを行いました。
私が移動しているビューはUIScrollViewであり、テキストフィールドが常に表示されるように、UITextField要素に対して相対的に移動したいです。このテキストフィールドの位置は、スクロールビューのオフセットによって異なる場合があります。
したがって、私のコードは次のようになります。
- (void)viewDidLoad
{
[super viewDidLoad];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardDidShow:) name:UIKeyboardDidShowNotification object:nil];
}
- (BOOL)textFieldShouldReturn:(UITextField *)textField
{
[textField resignFirstResponder];
dispatch_async(dispatch_get_main_queue(), ^{
[UIView beginAnimations:nil context:nil];
[UIView setAnimationDuration:0.2];
[UIView setAnimationCurve:UIViewAnimationCurveLinear];
self.scrollView.frame = CGRectMake(0,0,self.scrollView.frame.size.width,self.scrollView.frame.size.height);
[UIView commitAnimations];
});
return YES;
}
- (NSInteger)getKeyBoardHeight:(NSNotification *)notification
{
NSDictionary* keyboardInfo = [notification userInfo];
NSValue* keyboardFrameBegin = [keyboardInfo valueForKey:UIKeyboardFrameBeginUserInfoKey];
CGRect keyboardFrameBeginRect = [keyboardFrameBegin CGRectValue];
NSInteger keyboardHeight = keyboardFrameBeginRect.size.height;
return keyboardHeight;
}
-(void) keyboardDidShow:(NSNotification*) notification
{
NSInteger keyboardHeight;
keyboardHeight = [self getKeyBoardHeight:notification];
NSInteger scrollViewFrameHeight = self.scrollView.frame.size.height;
NSInteger textFieldRelativePosition = self.tableView.frame.Origin.y - self.scrollView.contentOffset.y;
NSInteger textFieldFrameOffset = scrollViewFrameHeight - textFieldRelativePosition;
NSInteger movement = MAX(0,keyboardHeight-textFieldFrameOffset); // Offset from where the keyboard will appear.
dispatch_async(dispatch_get_main_queue(), ^{
[UIView beginAnimations:nil context:nil];
[UIView setAnimationDuration:0.2];
[UIView setAnimationCurve:UIViewAnimationCurveLinear];
self.scrollView.frame = CGRectMake(0,-movement,
self.scrollView.frame.size.width,
self.scrollView.frame.size.height);
[UIView commitAnimations];
});
}
View ControllerはUITextFieldDelegateであり、UIKeyboardDidShowNotificationもサブスクライブしているため、キーボードのサイズにアクセスできます。
キーボードが表示されたら、UITextField(スクロールオフセット用に調整)とキーボードの相対オフセットを計算し、UIScrollViewのOriginを変更して、UITextFieldが表示されるのに十分なだけ移動するようにします。
キーボードが表示されてもUITextFieldが表示される場合、Originは変わりません。
mark、Apple document: キーボードの管理-キーボードの下にあるコンテンツの移動
-(void)textFieldDidBeginEditing:(UITextField *)textField
{
CGFloat y = textField.frame.Origin.y;
if (y >= 350) //not 380
{
CGRect frame = self.view.frame;
frame.Origin.y = 320 - textField.frame.Origin.y;
[UIView animateWithDuration:0.3 animations:^{self.view.frame = frame;}];
}
}
-(void)textFieldDidEndEditing:(UITextField *)textField
{
CGRect returnframe =self.view.frame;
returnframe.Origin.y = 0;
[UIView animateWithDuration:0.3 animations:^{self.view.frame = frame;}];
}
これら2つのメソッドを編集するだけです。
すべてのdコードに対する単純な答え。 ifステートメントでは、iphoneに従って値を変更します。つまり、iphone 4Sが265に変更し、didbegineditingで320を240に変更し、iphone 5でそれを350に変更し、didbegineditingメソッドでは320のままにします。もし理解したら
おそらく[myTextField becomeFirstResponder];
を呼び出すコードがあります。この呼び出しの直後にビューを移動する必要があります。
これはTendulkarのソリューションですが、元のフレームサイズに留意し、キーボードを削除します。このソリューションはすべてのデバイスで機能します。
-(BOOL) textFieldShouldReturn:(UITextField *)textField{
[textField resignFirstResponder];
[UIView beginAnimations:nil context:NULL];
[UIView setAnimationDuration:0.25];
self.view.frame = CGRectMake(0,0,self.view.frame.size.width,self.view.frame.size.height);
[UIView commitAnimations];
return YES;
}
- (void)textFieldDidBeginEditing:(UITextField *)textField
{
[UIView beginAnimations:nil context:NULL];
[UIView setAnimationDuration:0.25];
self.view.frame = CGRectMake(0,-50,self.view.frame.size.width,self.view.frame.size.height);
[UIView commitAnimations];
}
UITextFieldデリゲートの設定を忘れないでください!
サンプルプロジェクトに基づく Appleリファレンス キーボードドキュメント
Hファイル:(UITextFieldDelegate)を忘れないでください
@property (weak, nonatomic) IBOutlet UIScrollView *scrollview;
@property (weak, nonatomic) IBOutlet UIView *myView;
//items on view
@property (weak, nonatomic) IBOutlet UISwitch *partySwitch;
@property (weak, nonatomic) IBOutlet UILabel *partyLabel;
@property (weak, nonatomic) IBOutlet UITextField *partyNameTextfield;
Mファイル:
//MARK: View Loading
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view.
}//eom
-(void)viewDidAppear:(BOOL)animated
{
[self registerForKeyboardNotifications];
}//eom
//MARK: textfield delegates
-(bool)textFieldShouldReturn:(UITextField *)textField
{
[textField resignFirstResponder];
return true;
}//eom
//MARK: - Keyboard Observers
// Call this method somewhere in your view controller setup code.
- (void)registerForKeyboardNotifications
{
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(keyboardWasShown:)
name:UIKeyboardDidShowNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(keyboardWillBeHidden:)
name:UIKeyboardWillHideNotification object:nil];
}
- (void)keyboardWasShown:(NSNotification*)aNotification
{
NSDictionary* info = [aNotification userInfo];
CGSize kbSize = [[info objectForKey:UIKeyboardFrameBeginUserInfoKey] CGRectValue].size;
CGRect bkgndRect = myView.superview.frame;
bkgndRect.size.height += kbSize.height;
[myView.superview setFrame:bkgndRect];
[scrollview setContentOffset:CGPointMake(0.0, myView.frame.Origin.y-kbSize.height) animated:YES];
}//eom
// Called when the UIKeyboardWillHideNotification is sent
- (void)keyboardWillBeHidden:(NSNotification*)aNotification
{
UIEdgeInsets contentInsets = UIEdgeInsetsZero;
scrollview.contentInset = contentInsets;
scrollview.scrollIndicatorInsets = contentInsets;
}//eom