Управление положением курсора в UITextField
У меня есть UITextField
что я заставляю форматирование, изменяя текст внутри обработчика уведомлений об изменениях. Это прекрасно работает (как только я решил проблемы с повторным входом), но оставляет мне еще одну неприятную проблему. Если пользователь перемещает курсор в другое место, кроме конца строки, то мое изменение форматирования перемещает его в конец строки. Это означает, что пользователи не могут вставлять более одного символа за раз в середину текстового поля. Есть ли способ запомнить и затем сбросить положение курсора в UITextField
?
7 ответов
Управление позицией курсора в UITextField является сложным, потому что очень много абстракций связано с полями ввода и вычислением позиций. Тем не менее, это, безусловно, возможно. Вы можете использовать функцию-член setSelectedTextRange
:
[input setSelectedTextRange:[input textRangeFromPosition:start toPosition:end]];
Вот функция, которая берет диапазон и выбирает тексты в этом диапазоне. Если вы просто хотите поместить курсор на определенный индекс, просто используйте диапазон длиной 0:
+ (void)selectTextForInput:(UITextField *)input atRange:(NSRange)range {
UITextPosition *start = [input positionFromPosition:[input beginningOfDocument]
offset:range.location];
UITextPosition *end = [input positionFromPosition:start
offset:range.length];
[input setSelectedTextRange:[input textRangeFromPosition:start toPosition:end]];
}
Например, чтобы поместить курсор в idx
в UITextField input
:
[Helpers selectTextForInput:input
atRange:NSMakeRange(idx, 0)];
Полезно для позиционирования по индексу (Swift 3)
private func setCursorPosition(input: UITextField, position: Int) {
let position = input.position(from: input.beginningOfDocument, offset: position)!
input.selectedTextRange = input.textRange(from: position, to: position)
}
Я наконец нашел решение этой проблемы! Вы можете поместить нужный текст в системный монтажный щит, а затем вставить его в текущую позицию курсора:
[myTextField paste:self]
Я нашел решение в блоге этого человека:
http://dev.ragfield.com/2009/09/insert-text-at-current-cursor-location.html
Функциональность вставки зависит от ОС V3.0, но я проверил ее, и она отлично работает для меня с пользовательской клавиатурой.
Если вы выберете это решение, то вам, вероятно, следует сохранить существующее содержимое буфера обмена пользователя и сразу же восстановить его.
Вот Swift-версия @Chris R. - обновлена для Swift3
private func selectTextForInput(input: UITextField, range: NSRange) {
let start: UITextPosition = input.position(from: input.beginningOfDocument, offset: range.location)!
let end: UITextPosition = input.position(from: start, offset: range.length)!
input.selectedTextRange = input.textRange(from: start, to: end)
}
Не стесняйтесь использовать это UITextField
Категория для получения и установки позиции курсора:
@interface UITextField (CursorPosition)
@property (nonatomic) NSInteger cursorPosition;
@end
-
@implementation UITextField (CursorPosition)
- (NSInteger)cursorPosition
{
UITextRange *selectedRange = self.selectedTextRange;
UITextPosition *textPosition = selectedRange.start;
return [self offsetFromPosition:self.beginningOfDocument toPosition:textPosition];
}
- (void)setCursorPosition:(NSInteger)position
{
UITextPosition *textPosition = [self positionFromPosition:self.beginningOfDocument offset:position];
[self setSelectedTextRange:[self textRangeFromPosition:textPosition toPosition:textPosition]];
}
@end
Я не думаю, что есть способ поместить курсор в определенное место в вашем UITextField
(если только вы не стали очень хитрыми и не смоделировали сенсорное событие). Вместо этого я бы обработал форматирование, когда пользователь закончил редактировать свой текст (в textFieldShouldEndEditing:
) и если они введены неверно, не позволяйте текстовому полю завершать редактирование.
Вот фрагмент кода, который отлично работает для этой проблемы:
- (void)textFieldDidBeginEditing:(UITextField *)textField{
UITextPosition *positionBeginning = [textField beginningOfDocument];
UITextRange *textRange =[textField textRangeFromPosition:positionBeginning
toPosition:positionBeginning];
[textField setSelectedTextRange:textRange];
}