>@end


>@implementation ViewController


Теперь создадим текстовое поле с подписью и нужные нам методы делегата текстового поля. Обойдемся без реализации многих методов UITextFieldDelegate, так как в этом примере они нам не требуются:


>– (void) calculateAndDisplayTextFieldLengthWithText:(NSString *)paramText{


>NSString *characterOrCharacters = @"Characters";

>if ([paramText length] == 1){

>characterOrCharacters = @"Character";

>}


>self.labelCounter.text = [NSString stringWithFormat:@"%lu %@",

>(unsigned long)[paramText length],

>characterOrCharacters];

>}


>– (BOOL) textField:(UITextField *)textField

>shouldChangeCharactersInRange:(NSRange)range

>replacementString:(NSString *)string{


>if ([textField isEqual: self.myTextField]){

>NSString *wholeText =

>[textField.text stringByReplacingCharactersInRange: range

>withString: string];

>[self calculateAndDisplayTextFieldLengthWithText: wholeText];

>}


>return YES;


>}


>– (BOOL)textFieldShouldReturn:(UITextField *)textField{

>[textField resignFirstResponder];

>return YES;

>}


>– (void)viewDidLoad{

>[super viewDidLoad];


>CGRect textFieldFrame = CGRectMake(38.0f,

>30.0f,

>220.0f,

>31.0f);


>self.myTextField = [[UITextField alloc]

>initWithFrame: textFieldFrame];


>self.myTextField.delegate = self;


>self.myTextField.borderStyle = UITextBorderStyleRoundedRect;


>self.myTextField.contentVerticalAlignment =

>UIControlContentVerticalAlignmentCenter;


>self.myTextField.textAlignment = NSTextAlignmentCenter;


>self.myTextField.text = @"Sir Richard Branson";


>[self.view addSubview: self.myTextField];


>CGRect labelCounterFrame = self.myTextField.frame;

>labelCounterFrame.origin.y += textFieldFrame.size.height + 10;

>self.labelCounter = [[UILabel alloc] initWithFrame: labelCounterFrame];

>[self.view addSubview: self.labelCounter];


>[self calculateAndDisplayTextFieldLengthWithText: self.myTextField.text];


>}


Мы делаем важное вычисление в методе textField: shouldChangeCharactersInRange: replacementString:. Здесь мы объявляем и используем переменную wholeText. Когда вызывается этот метод, параметр replacementString указывает строку, которую пользователь ввел в текстовое поле. Вы, возможно, полагаете, что пользователь может вводить по одному символу в каждый момент времени, поэтому почему бы не присвоить данному полю значение char? Но не забывайте, что пользователь может вставить в текстовое поле целый фрагмент текста, по этой причине данный параметр должен быть строковым. Параметр shouldChangeCharactersInRange указывает место в текстовом поле, с которого пользователь начинает вводить текст. Итак, с помощью двух этих параметров мы создаем строку, которая сначала считывает весь текст из текстового поля, а потом использует заданный диапазон, чтобы разместить новый текст рядом со старым. Итак, получается, что вводимый нами текст будет появляться в поле после того, как метод textField: shouldChangeCharactersInRange: replacementString: возвратит YES. На рис. 1.51 показано, как приложение будет выглядеть в эмуляторе.


Рис. 1.51. Реагирование на сообщения-делегаты текстового поля


В текстовом поле может отображаться не только текст, но и подстановочные (джокерные) символы. Подстановочный текст отображается до того, как пользователь введет в это поле какой-нибудь собственный текст, пока свойство text текстового поля является пустым. В качестве подстановочного текста вы можете использовать любую строку, какую хотите, но лучше этим текстом подсказать пользователю, для ввода какой именно информации предназначено данное поле. Многие программисты указывают в подстановочном тексте, значения какого типа может принимать данное поле. Например, на рис. 1.49 в двух текстовых полях (для ввода имени пользователя и пароля) стоит подстановочный текст Required (Обязательно). Можно использовать свойство placeholder текстового поля для установки или получения актуального подстановочного текста: