文本输入 (TextInput/TextArea/Search)

TextInput、TextArea是输入框组件,用于响应用户输入,比如评论区的输入、聊天框的输入、表格的输入等,也可以结合其它组件构建功能页面,例如登录注册页面。具体用法请参考TextInputTextArea组件的API文档。Search是特殊的输入框组件,称为搜索框,默认样式包含搜索图标。具体用法请参考Search组件的API文档。

说明:

仅支持单文本样式,若需实现富文本样式,建议使用RichEditor组件。

创建输入框

TextInput是单行输入框,TextArea是多行输入框,Search是搜索框。通过以下接口创建这些组件。

TextInput(value?:{placeholder?: ResourceStr, text?: ResourceStr, controller?: TextInputController})
TextArea(value?:{placeholder?: ResourceStr, text?: ResourceStr, controller?: TextAreaController})
Search(options?:{placeholder?: ResourceStr, value?: ResourceStr, controller?: SearchController, icon?: string})
  • 单行输入框。

    TextInput()
    

    zh-cn_image_0000001511580844

  • 多行输入框。

    TextArea()
    

    zh-cn_image_0000001562940481

  • 多行输入框文字超出一行时会自动折行。

    /* 请将$r('app.string.CreatTextInput_textContent')替换为实际资源文件,在本示例中该资源文件的value值为
     * "我是TextArea我是TextArea我是TextArea我是TextArea"
     */
    TextArea({ text: $r('app.string.CreatTextInput_textContent') })
      .width(300)
    

    zh-cn_image_0000001511580836

  • 搜索框。

    Search()
      // 请将$r('app.string.Creat_TextInput_Content')替换为实际资源文件,在本示例中该资源文件的value值为"搜索"
      .searchButton($r('app.string.Creat_TextInput_Content'))
    

    zh-ch_image_ui_arkts-common-components-text-input_search_default

设置输入框类型

TextInput、TextArea和Search都支持设置输入框类型,通过type属性进行设置,但是各组件的枚举值略有不同。下面以单行输入框为例进行说明。

TextInput有以下类型可选择:Normal基本输入模式、Password密码输入模式、Email邮箱地址输入模式、Number纯数字输入模式、PhoneNumber电话号码输入模式、USER_NAME用户名输入模式、NEW_PASSWORD新密码输入模式、NUMBER_PASSWORD纯数字密码输入模式、SCREEN_LOCK_PASSWORD锁屏应用密码输入模式、NUMBER_DECIMAL带小数点的数字输入模式、带URL的输入模式。通过type属性进行设置:

基本输入模式(默认类型)

TextInput()
  .type(InputType.Normal)

zh-cn_image_0000001562820765

密码模式

包括Password密码输入模式、NUMBER_PASSWORD纯数字密码模式、NEW_PASSWORD新密码输入模式。

以下示例是Password密码输入模式的输入框。

TextInput()
  .type(InputType.Password)

zh-cn_image_0000001511580840

邮箱地址输入模式

邮箱地址输入模式的输入框,只能存在一个@符号。

TextInput()
  .type(InputType.Email)

text_input_type_email

纯数字输入模式

纯数字输入模式的输入框,只能输入数字[0-9]。

TextInput()
  .type(InputType.Number)

text_input_type_number

电话号码输入模式

电话号码输入模式的输入框,支持输入数字、空格、+ 、-、*、#、(、),长度不限。

TextInput()
  .type(InputType.PhoneNumber)

text_input_type_phone_number

带小数点的数字输入模式

带小数点的数字输入模式的输入框,只能输入数字[0-9]和小数点,只能存在一个小数点。

TextInput()
  .type(InputType.NUMBER_DECIMAL)

text_input_type_number_decimal

带URL的输入模式

带URL的输入模式,无特殊限制。

TextInput()
  .type(InputType.URL)

text_input_type_url

设置输入框多态样式

TextInput、TextArea支持设置输入框多态样式,通过style属性进行设置。下面以多行输入框TextArea为例进行说明。

TextArea有以下2种类型可选择:默认风格,入参是TextContentStyle.DEFAULT;内联模式,也称内联输入风格,入参是TextContentStyle.INLINE。

默认风格

默认风格的输入框,在编辑态和非编辑态,样式没有区别。

TextArea()
  .style(TextContentStyle.DEFAULT)

textArea_style_default

内联模式

内联模式,也称内联输入风格。内联模式的输入框在编辑态和非编辑态样式有明显区分。

TextArea()
  .style(TextContentStyle.INLINE)

textArea_style_inline

自定义样式

  • 设置无输入时的提示文本。

    // 请将$r('app.string.i_am_placeholder')替换为实际资源文件,在本示例中该资源文件的value值为"我是提示文本"
    TextInput({ placeholder: $r('app.string.i_am_placeholder') })
    

    zh-cn_image_0000001511900400

  • 设置输入框当前的文本内容。

    TextInput({
      // 请将$r('app.string.i_am_placeholder')替换为实际资源文件,在本示例中该资源文件的value值为"我是提示文本"
      placeholder: $r('app.string.i_am_placeholder'),
      // 请将$r('app.string.i_am_current_text_content')替换为实际资源文件,在本示例中该资源文件的value值为"我是当前文本内容"
      text: $r('app.string.i_am_current_text_content')
    })
    

    zh-cn_image_0000001562820761

  • 添加backgroundColor改变输入框的背景颜色。

    TextInput({
      // 请将$r('app.string.i_am_placeholder')替换为实际资源文件,在本示例中该资源文件的value值为"我是提示文本"
      placeholder: $r('app.string.i_am_placeholder'),
      // 请将$r('app.string.i_am_current_text_content')替换为实际资源文件,在本示例中该资源文件的value值为"我是当前文本内容"
      text: $r('app.string.i_am_current_text_content')
    })
      .backgroundColor(Color.Pink)
    

    zh-cn_image_0000001511740444

    更丰富的样式可以结合通用属性实现。

添加事件

文本框主要用于获取用户输入的信息,并将信息处理成数据进行上传,绑定onChange事件可以获取输入框内改变的文本内容,绑定onSubmit事件可以获取回车提交的文本信息,绑定onTextSelectionChange事件可以获取文本选中时手柄的位置信息或者编辑时光标的位置信息等等。用户也可以使用通用事件进行相应的交互操作。

说明:

在密码模式下,设置showPassword属性时,在onSecurityStateChange回调中,建议增加状态同步,具体详见如下示例。

onWillInsertonDidInsertonWillDeleteonDidDelete回调仅支持系统输入法的场景。

onWillChange的回调时序晚于onWillInsertonWillDelete,早于onDidInsertonDidDelete

import { hilog } from '@kit.PerformanceAnalysisKit';

const TAG = '[Sample_Textcomponent]';
const DOMAIN = 0xF811;
const BUNDLE = 'Textcomponent_';

@Entry
@Component
struct TextInputEventAdd {
  @State text: string = '';
  @State textStr1: string = '';
  @State textStr2: string = '';
  @State textStr3: string = '';
  @State textStr4: string = '';
  @State textStr5: string = '';
  @State textStr6: string = '';
  @State textStr7: string = '';
  @State textStr8: string = '';
  @State textStr9: string = '';
  @State passwordState: boolean = false;
  controller: TextInputController = new TextInputController();

  build() {
    Row() {
      Column() {
        Text(`${this.textStr1}\n${this.textStr2}\n${this.textStr3}
          \n${this.textStr4}\n${this.textStr5}\n${this.textStr6}
          \n${this.textStr7}\n${this.textStr8}\n${this.textStr9}`)
          .fontSize(20)
          .width('70%')
        TextInput({ text: this.text, placeholder: 'input your word...', controller: this.controller })
          .type(InputType.Password)
          .showPassword(this.passwordState)
          .onChange((value: string) => {
            // 文本内容发生变化时触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onChange is triggering: ' + value);
            this.textStr1 = `onChange is triggering: ${value}`;
          })
          .onSubmit((enterKey: EnterKeyType, event: SubmitEvent) => {
            // 按下输入法回车键时触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onSubmit is triggering: ' + enterKey + event.text);
            this.textStr2 = `onSubmit is triggering: ${enterKey} ${event.text}`;
          })
          .onTextSelectionChange((selectionStart: number, selectionEnd: number) => {
            // 文本选择的位置发生变化或编辑状态下光标位置发生变化时,触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onTextSelectionChange is triggering: ' + selectionStart + selectionEnd);
            this.textStr3 = `onTextSelectionChange is triggering: ${selectionStart} ${selectionEnd}`;
          })
          .onSecurityStateChange((isShowPassword: boolean) => {
            // 密码显隐状态切换时,触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onSecurityStateChange is triggering: ' + isShowPassword);
            this.passwordState = isShowPassword;
            this.textStr4 = `onSecurityStateChange is triggering: ${isShowPassword}`;
          })
          .onWillInsert((info: InsertValue) => {
            // 在将要输入时,触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onWillInsert is triggering: ' + info.insertValue + info.insertOffset);
            this.textStr5 = `onWillInsert is triggering: ${info.insertValue} ${info.insertOffset}`;
            return true;
          })
          .onDidInsert((info: InsertValue) => {
            // 在输入完成时,触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onDidInsert is triggering: ' + info.insertValue + info.insertOffset);
            this.textStr6 = `onDidInsert is triggering: ${info.insertValue} ${info.insertOffset}`;
          })
          .onWillDelete((info: DeleteValue) => {
            // 在将要删除时,触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onWillDelete is triggering: ' + info.deleteValue + info.deleteOffset);
            this.textStr7 = `onWillDelete is triggering: ${info.deleteValue} ${info.deleteOffset}`;
            return true;
          })
          .onDidDelete((info: DeleteValue) => {
            // 在删除完成时,触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onDidDelete is triggering: ' + info.deleteValue + info.deleteOffset);
            this.textStr8 = `onDidDelete is triggering: ${info.deleteValue} ${info.deleteOffset}`;
          })
          .onFocus(() => {
            // 绑定通用事件,输入框获焦时触发该回调
            hilog.info(DOMAIN, TAG, BUNDLE + 'onFocus is triggering');
            this.textStr9 = `onFocus is triggering`;
          })
      }.width('100%')
    }
    .height('100%')
  }
}

text_input_event

选中菜单

输入框中的文字被选中时会弹出包含剪切、复制、翻译、分享的菜单。

TextInput:

// 请将$r('app.string.show_selected_menu')替换为实际资源文件,在本示例中该资源文件的value值为"这是一段文本,用来展示选中菜单"
TextInput({ text: $r('app.string.show_selected_menu') })

TextInput_select_menu

TextArea:

// 请将$r('app.string.show_selected_menu')替换为实际资源文件,在本示例中该资源文件的value值为"这是一段文本,用来展示选中菜单"
TextArea({ text: $r('app.string.show_selected_menu') })

TextArea_select_menu

禁用系统服务类菜单

从API version 20开始,支持使用disableSystemServiceMenuItems方法屏蔽文本选择菜单中的所有系统服务菜单项。更多详见disableSystemServiceMenuItems的API文档接口说明。以下示例只是完整示例工程中的一个示例,为了不影响工程其他页面示例效果,仅在页面的出现和消失生命周期中进行系统服务菜单的禁用和恢复,实际场景可自行选择其他时机,比如UIAbilityonCreateonDestroy

import { TextMenuController } from '@kit.ArkUI';

@Entry
@Component
struct DisableSystemServiceMenuItem {
  aboutToAppear(): void {
    // 禁用所有系统服务菜单项
    TextMenuController.disableSystemServiceMenuItems(true)
  }

  aboutToDisappear(): void {
    // 页面消失时恢复系统服务菜单项
    TextMenuController.disableSystemServiceMenuItems(false)
  }

  build() {
    Row() {
      Column() {
        // 请将$r('app.string.ProhibitSelectMenu_content')替换为实际资源文件,在本示例中该资源文件的value值为"这是一个TextInput,长按弹出文本选择菜单"
        TextInput({ text: $r('app.string.ProhibitSelectMenu_content') })
          .height(60)
          .fontStyle(FontStyle.Italic)
          .fontWeight(FontWeight.Bold)
          .textAlign(TextAlign.Center)
          .caretStyle({ width: '4vp' })
          .editMenuOptions({
            onCreateMenu: (menuItems: Array<TextMenuItem>) => {
              // menuItems不包含被屏蔽的系统菜单项
              return menuItems
            },
            onMenuItemClick: (menuItem: TextMenuItem, textRange: TextRange) => {
              return false
            }
          })
      }.width('100%')
    }
    .height('100%')
  }
}

TextInput_disable_system_service_menu_items

从API version 20开始,支持使用disableMenuItems方法屏蔽文本选择菜单中指定的系统服务菜单项。更多详见disableMenuItems的API文档接口说明。以下示例只是完整示例工程中的一个示例,为了不影响工程其他页面示例效果,仅在页面的出现和消失生命周期中进行系统服务菜单的禁用和恢复,实际场景可自行选择其他时机,比如UIAbilityonCreateonDestroy

import { TextMenuController } from '@kit.ArkUI';

@Entry
@Component
struct DisableMenuItem {
  aboutToAppear(): void {
    // 禁用搜索,翻译和AI帮写
    TextMenuController.disableMenuItems([TextMenuItemId.SEARCH, TextMenuItemId.TRANSLATE, TextMenuItemId.AI_WRITER])
  }

  aboutToDisappear(): void {
    // 页面消失时恢复系统服务菜单项
    TextMenuController.disableMenuItems([])
  }

  build() {
    Row() {
      Column() {
        // 请将$r('app.string.ProhibitSelectMenu_content')替换为实际资源文件,在本示例中该资源文件的value值为"这是一个TextInput,长按弹出文本选择菜单"
        TextInput({ text: $r('app.string.ProhibitSelectMenu_content') })
          .height(60)
          .fontStyle(FontStyle.Italic)
          .fontWeight(FontWeight.Bold)
          .textAlign(TextAlign.Center)
          .caretStyle({ width: '4vp' })
          .editMenuOptions({
            onCreateMenu: (menuItems: Array<TextMenuItem>) => {
              // menuItems不包含搜索和翻译
              return menuItems;
            },
            onMenuItemClick: (menuItem: TextMenuItem, textRange: TextRange) => {
              return false
            }
          })
      }.width('100%')
    }
    .height('100%')
  }
}

Text_input_disable_menu_items

自动填充

输入框可以通过contentType属性设置自动填充类型。

支持的类型请参考ContentType

// 请将$r('app.string.Auto_Fill_PlaceHolder')替换为实际资源文件,在本示例中该资源文件的value值为"输入你的邮箱..."
TextInput({ placeholder: $r('app.string.Auto_Fill_PlaceHolder') })
  .width('95%')
  .height(40)
  .margin(20)
  .contentType(ContentType.EMAIL_ADDRESS)

设置属性

  • 设置省略属性。

    输入框可以通过ellipsisMode属性设置省略位置。

    ellipsisMode属性需要配合textOverflow属性设置为TextOverflow.Ellipsis使用,单独设置ellipsisMode属性不生效。

    // 请将$r('app.string.Set_Omission_Property_textContent')替换为实际资源文件,在本示例中该资源文件的value值为"这是一段文本,用来展示省略模式"
    TextInput({ text: $r('app.string.Set_Omission_Property_textContent') })
      .textOverflow(TextOverflow.Ellipsis)
      .ellipsisMode(EllipsisMode.END)
      .style(TextInputStyle.Inline)
      .fontSize(30)
      .margin(30)
    
    ![TextInput_ellipsismode](figures/TextInput_ellipsismode.jpg)
  • 设置文本描边属性。

    从API version 20开始,输入框可以通过strokeWidthstrokeColor属性设置文本的描边宽度及颜色。

    TextInput({ text: 'Text with stroke' })
      .width('100%')
      .height(60)
      .borderWidth(1)
      .fontSize(40)
      .strokeWidth(LengthMetrics.px(3.0))
      .strokeColor(Color.Red)
    
    ![TextInput_stroke](figures/TextInput_stroke.jpg)

设置文本行间距

从API version 20开始,支持通过lineSpacing设置文本的行间距。如果不配置LineSpacingOptions时,首行上方和尾行下方默认会有行间距。如果onlyBetweenLines设置为true时,行间距仅适用于行与行之间,首行上方无额外行间距。

TextArea({
  text: 'The line spacing of this TextArea is set to 20_px, and the spacing is effective only between the lines.'
})
  .fontSize(22)
  .lineSpacing(LengthMetrics.px(20), { onlyBetweenLines: true })

TextInput_line_spacing

键盘避让

键盘抬起后,具有滚动能力的容器组件在横竖屏切换时,才会生效键盘避让,若希望无滚动能力的容器组件也生效键盘避让,建议在组件外嵌套一层具有滚动能力的容器组件,比如ScrollListGrid

@Entry
@Component
struct KeyboardAvoid {
  placeHolderArr: string[] = ['1', '2', '3', '4', '5', '6', '7'];

  build() {
    Scroll() {
      Column() {
        ForEach(this.placeHolderArr, (placeholder: string) => {
          TextInput({ placeholder: 'TextInput ' + placeholder })
            .margin(30)
            // ···
        })
      }
    }
    .height('100%')
    .width('100%')
  }
}

textinputkeyboardavoid

光标避让

keyBoardAvoidMode枚举中的OFFSET和RESIZE在键盘抬起后,不支持二次避让。如果想要支持光标位置在点击或者通过接口设置变化后发生二次避让,可以考虑使用OFFSET_WITH_CARET和RESIZE_CARET替换原有的OFFSET和RESIZE模式。

对于滚动容器更推荐使用RESIZE_WITH_CARET,非滚动容器应该使用OFFSET_WITH_CARET。

import { hilog } from '@kit.PerformanceAnalysisKit';
import { window } from '@kit.ArkUI';
import { KeyboardAvoidMode } from '@kit.ArkUI';
// Used in UIAbility
onWindowStageCreate(windowStage: window.WindowStage): void {
  // Main window is created, set main page for this ability
  hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onWindowStageCreate');

  windowStage.loadContent('pages/Index', (err, data) => {
    let keyboardAvoidMode = windowStage.getMainWindowSync().getUIContext().getKeyboardAvoidMode();
    windowStage.getMainWindowSync().getUIContext().setKeyboardAvoidMode(KeyboardAvoidMode.OFFSET_WITH_CARET);
    if (err.code) {
      hilog.error(0x0000, 'testTag', 'Failed to load the content. Cause: %{public}s', JSON.stringify(err) ?? '');
      return;
    }
    hilog.info(0x0000, 'testTag', 'Succeeded in loading the content. Data: %{public}s', JSON.stringify(data) ?? '');
  });
}
@Entry
@Component
struct CursorAvoid {
  @State caretPosition: number = 600;
  areaController: TextAreaController = new TextAreaController();
  text = 'Most of us compare ourselves with anyone we think is happier — a relative, someone we know a lot,' +
    ' or someone we hardly know. As a result, what we do remember is anything that makes others happy, ' +
    'anything that makes ourselves unhappy,' +
    ' totally forgetting that there is something happy in our own life.\
    So the best way to destroy happiness is to look at something and focus on even the smallest flaw. ' +
    'It is the smallest flaw that would make us complain. And it is the complaint that leads to us becoming unhappy.\
    If one chooses to be happy, he will be blessed; if he chooses to be unhappy, he will be cursed. ' +
    'Happiness is just what you think will make you happy.' +
    'Most of us compare ourselves with anyone we think is happier — a relative, someone we know a lot, ' +
    'or someone we hardly know. As a result, what we do remember is anything that makes others happy, ' +
    'anything that makes ourselves unhappy, totally forgetting that there is something happy in our own life.\
  ';

  build() {
    Scroll() {
      Column() {
        Row() {
          Button('CaretPosition++: ' + this.caretPosition).onClick(() => {
            this.caretPosition += 1;
          }).fontSize(10)
          Button('CaretPosition--: ' + this.caretPosition).onClick(() => {
            this.caretPosition -= 1;
          }).fontSize(10)
          Button('SetCaretPosition: ').onClick(() => {
            this.areaController.caretPosition(this.caretPosition);
          }).fontSize(10)
        }

        TextArea({ text: this.text, controller: this.areaController })
          .width('100%')
          .fontSize('20fp')
      }
    }.width('100%').height('100%')
  }
}

textinputkeyboardavoid

相关实例

针对文本输入开发,有以下相关实例可供参考:

常见问题

如何设置TextArea的文本最少展示行数并自适应高度

问题现象

设置TextArea的初始高度来控制最少文本展示行数,当输入文本超过初始高度时,TextArea的高度自适应。

解决措施

设置minLines(从API version 20开始),或者设置height为"auto",并使用constraintSize自行计算高度。

import { MeasureUtils } from '@kit.ArkUI';

@Entry
@Component
struct TextExample {
  private textAreaPadding = 12;
  private setMaxLines = 3;
  private resourceManager = this.getUIContext().getHostContext()?.resourceManager;
  // 请在resources\base\element\string.json文件中配置name为'NormalQuestion_change',value为非空字符串的资源
  private changeText = this.resourceManager?.getStringByNameSync('NormalQuestion_change') as string;
  @State fullText: string = this.changeText;
  @State originText: string = this.changeText;
  @State uiContext: UIContext = this.getUIContext();
  @State uiContextMeasure: MeasureUtils = this.uiContext.getMeasureUtils();
  textSize: SizeOptions = this.uiContextMeasure.measureTextSize({
    textContent: this.originText,
    fontSize: 18
  });

  build() {
    Column() {
      TextArea({ text: 'minLines: ' + this.fullText })
        .fontSize(18)
        .width(300)
        .minLines(3)

      Blank(50)

      TextArea({ text: 'constraintSize: ' + this.fullText })
        .fontSize(18)
        .padding({ top: this.textAreaPadding, bottom: this.textAreaPadding })
        .width(300)
        .height('auto')
        .constraintSize({
          // 结合padding计算,设置至少显示this.setMaxLines行文本
          // 若涉及适老化字号缩放,需要监听并调整高度
          minHeight: this.textAreaPadding * 2 +
            this.setMaxLines * this.getUIContext().px2vp(Number(this.textSize.height))
        })

      Blank(50)
      // 请将$r('app.string.NormalQuestion_AddInput')替换为实际资源文件,在本示例中该资源文件的value值为"增加输入"
      Button($r('app.string.NormalQuestion_AddInput'))
        .onClick(() => {
          this.fullText += this.changeText;
        })
    }
    .justifyContent(FlexAlign.Center)
    .width('100%')
    .padding({ top: 30 })
  }
}

textinputkeyboardavoid