# 焦点控制 自定义组件的走焦效果,可设置组件是否走焦和具体的走焦顺序,使用TAB键或方向键切换焦点。 > **说明:** > > - 从API version 8开始支持。后续版本如有新增内容,则采用上角标单独标记该内容的起始版本。 > > - 自定义组件无获焦能力,当设置[focusable](#focusable)、[enabled](ts-universal-attributes-enable.md#enabled)等属性为false,或者设置[visibility](ts-universal-attributes-visibility.md#visibility)属性为Hidden、None时,也不影响其子组件的获焦。 > > - 组件主动获取焦点不受窗口焦点的控制。 > > - 焦点开发参考[焦点开发指南](../../../ui/arkts-common-events-focus-event.md)。 ## focusable focusable(value: boolean): T 设置当前组件是否可以获焦。 **原子化服务API:** 从API version 11开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | value | boolean | 是 | 设置当前组件是否可以获焦,true表示组件可以获焦,false表示组件不可获焦。
**说明:**
存在默认交互逻辑的组件例如[Button](ts-basic-components-button.md)、[TextInput](ts-basic-components-textinput.md)等,默认即为可获焦,[Text](ts-basic-components-text.md)、[Image](ts-basic-components-image.md)等组件则默认状态为不可获焦。不可获焦状态下,无法触发[焦点事件](ts-universal-focus-event.md)。 | **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## tabIndex9+ tabIndex(index: number): T 自定义组件tab键走焦能力。当组件未设置tabIndex时,默认按照预设的焦点移动规则进行焦点移动。 > **说明:** > > - tabIndex只能够自定义Tab键走焦,若想同时自定义方向键等走焦能力,建议使用[nextFocus](#nextfocus18)。 **原子化服务API:** 从API version 11开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------ | ---- | ------------------------------------------------------------ | | index | number | 是 | 自定义组件tab键走焦能力。若有配置了tabIndex大于0的组件,则tab键走焦只会在tabIndex大于0的组件内按照tabIndex的值从小到大并循环依次走焦。若没有配置tabIndex大于0的组件,则tabIndex等于0的组件按照组件预设的走焦规则走焦。
[UiExtension](../js-apis-arkui-uiExtension.md)组件未适配tabIndex,在含有[UiExtension](../js-apis-arkui-uiExtension.md)组件的[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)使用tabIndex会导致走焦错乱。
- tabIndex >= 0:表示元素是可聚焦的,并且可以通过tab键走焦来访问到该元素。
- tabIndex < 0(通常是tabIndex = -1):表示元素是可聚焦的,但是不能通过tab键走焦来访问到该元素。
**说明:**
tabIndex与focusScopeId不能混用。 | **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## defaultFocus9+ defaultFocus(value: boolean): T 设置当前组件是否为当前[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)上的默认焦点。当未设置defaultFocus时,组件默认不为当前层级页面的默认焦点。 > **说明:** > > 可以设置默认焦点的页面指的是支持页面路由或是弹窗类的容器组件,例如Page、NaviDestination、NavBar、PopUp、Dialog等。 **原子化服务API:** 从API version 11开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | value | boolean | 是 | 设置当前组件是否为当前[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)上的默认焦点,仅在初次创建的[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)第一次进入时生效。
**说明:**
值为true则表示为默认焦点,值为false时无效。
若[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)内无任何组件设置defaultFocus(true),API version 11及之前,[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)的默认焦点是当前[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)上首个可获焦的非容器组件,API version 11之后,[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)的默认焦点就是[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)的根容器。
若某[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)内有多个组件设置了defaultFocus(true),则以组件树深度遍历找到的第一个组件为默认焦点。 | **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## groupDefaultFocus9+ groupDefaultFocus(value: boolean): T 设置当前组件是否为当前组件所在容器获焦时的默认焦点。当组件未设置groupDefaultFocus时,组件默认不为当前组件所在容器获焦时的默认焦点。 **原子化服务API:** 从API version 11开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | value | boolean | 是 | 设置当前组件是否为当前组件所在容器获焦时的默认焦点,仅在初次创建容器节点第一次获焦时生效。true表示当前组件为所在容器获焦时的默认焦点,false表示当前组件不是所在容器获焦时的默认焦点。
**说明:**
必须与[tabIndex](#tabindex9)联合使用,当某个容器设置了tabIndex,且容器内某子组件或容器自身设置了groupDefaultFocus(true),当该容器首次TAB键获焦时,会自动将焦点转移至该指定的组件上。若容器内(包含容器本身)有多个组件设置了groupDefaultFocus(true),则以组件树深度遍历找到的第一个组件为最终结果。 | **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## focusOnTouch9+ focusOnTouch(value: boolean): T 设置当前组件是否支持点击获焦能力。当组件未设置focusOnTouch时,组件默认不支持点击获焦能力。 **原子化服务API:** 从API version 11开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | value | boolean | 是 | 设置当前组件是否支持点击获焦能力。true表示组件支持点击获焦,false表示不支持点击获焦。
**说明:**
仅在组件可点击时才能正常获取焦点。 | **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## focusBox12+ focusBox(style: FocusBoxStyle): T 设置当前组件系统焦点框样式。 **原子化服务API:** 从API version 12开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ---- | ---- | ---- | ---- | | style | [FocusBoxStyle](#focusboxstyle12对象说明) | 是 | 设置当前组件系统焦点框样式。
**说明:**
仅影响走焦状态下展示了系统焦点框的组件。 | **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## focusControl9+ 焦点控制模块。 **原子化服务API:** 从API version 11开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full ### requestFocus9+ requestFocus(value: string): boolean 方法语句中可使用的全局接口,调用此接口可以主动让焦点在下一帧渲染时转移至参数指定的组件上。 如果需要指定组件立刻获焦,推荐使用FocusController中的焦点同步转移接口[requestFocus](../arkts-apis-uicontext-focuscontroller.md#requestfocus12)。 **原子化服务API:** 从API version 11开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ----- | ------ | ---- | ---- | | value | string | 是 | 目标组件使用接口key(value: string)或id(value: string)绑定的字符串。 | **返回值:** | 类型 | 说明 | | ------- | ---- | | boolean | 返回值表示是否成功给目标组件申请到焦点。若参数指向的目标组件存在且可获焦,则返回true,否则返回false。 | > **说明:** > > 支持焦点控制的组件:[TextInput](ts-basic-components-textinput.md)、[TextArea](ts-basic-components-textarea.md)、[Search](ts-basic-components-search.md)、[Button](ts-basic-components-button.md)、[Text](ts-basic-components-text.md)、[Image](ts-basic-components-image.md)、[List](ts-container-list.md)、[Grid](ts-container-grid.md)。焦点事件当前仅支持在真机上显示运行效果。 ## FocusBoxStyle12+对象说明 **原子化服务API:** 从API version 12开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full | 名称 | 参数类型 | 必填 | 描述 | | ---- | ---- | ---- | ---- | | margin | [LengthMetrics](../js-apis-arkui-graphics.md#lengthmetrics12) | 否 | 焦点框相对组件边缘的距离。
正数代表外侧,负数代表内侧。不支持百分比。 | | strokeColor | [ColorMetrics](../js-apis-arkui-graphics.md#colormetrics12) | 否 | 焦点框颜色。 | | strokeWidth | [LengthMetrics](../js-apis-arkui-graphics.md#lengthmetrics12) | 否 | 焦点框宽度。
不支持负数与百分比。| ## focusScopePriority12+ focusScopePriority(scopeId: string, priority?: FocusPriority): T 设置当前组件在指定容器内获焦的优先级。需要配合focusScopeId一起使用。 **原子化服务API:** 从API version 12开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | scopeId | string | 是 | 当前组件设置的获焦优先级生效的容器组件的id标识。
**说明:**
1.当前组件必须在scopeId所标识的容器内,或其所属容器在scopeId所标识的容器内。
2.组件不可重复设置多个优先级。
3.设置了focusScopeId的容器组件不可设置优先级。 | | priority | [FocusPriority](#focuspriority12) | 否 | 获焦优先级。
**说明:**
未设置priority时,默认为AUTO优先级。
优先级对走焦以及获焦组件的影响:
1.容器整体获焦([层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)切换/焦点切换到焦点组/容器组件使用requestFocus申请焦点)时,若容器内存在优先级为PREVIOUS的组件,则优先级为PREVIOUS的组件获焦,否则,由容器内上次获焦的组件获焦。
2.容器非整体获焦(非焦点组场景下使用tab键/方向键走焦)时,若容器为首次获焦,则容器内优先级最高的组件获焦,若容器非首次获焦,不考虑优先级按照位置顺序走焦。 | **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ### FocusPriority12+ **原子化服务API:** 从API version 12开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full | 名称 | 值 | 说明 | | ----------- | ----- |-------- | | AUTO | 0|默认的优先级,缺省时组件的获焦优先级。 | | PRIOR | 2000|容器内优先获焦的优先级。优先级高于AUTO。 | | PREVIOUS | 3000|上一次容器整体失焦时获焦节点的优先级。优先级高于PRIOR。 | ### KeyProcessingMode15+ 设置按键事件处理的优先级。 **原子化服务API:** 从API version 15开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full | 名称 | 值 | 说明 | | -----------| ----------- | --------- | | FOCUS_NAVIGATION | 0 | 默认值,当前组件不消费按键时,tab/方向键优先在当前容器内走焦。 | | ANCESTOR_EVENT | 1 | 当前组件不消费按键时,tab/方向键优先冒泡给父组件。 | ## focusScopeId12+ focusScopeId(id: string, isGroup?: boolean): T 设置当前容器组件的id标识,以及是否为焦点组。 **原子化服务API:** 从API version 12开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | id | string | 是 | 设置当前容器组件的id标识。
**说明:**
单个[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)下,id标识全局唯一,不可重复。 | | isGroup | boolean | 否 | 设置当前容器组件是否为焦点组。true表示容器组件为焦点组,false表示容器组件不是焦点组。默认值为false。
**说明:**
焦点组不可嵌套,不可重复配置。
焦点组不能和tabIndex混用。
配置焦点组的目的是使得容器及容器内的元素可以按照焦点组规则走焦。焦点组走焦规则:
1.焦点组容器内只能通过方向键走焦,tab键会使焦点跳出焦点组容器。
2.通过方向键使焦点从焦点组容器外切换到焦点组容器内时,若焦点组容器内存在优先级为PREVIOUS的组件,则优先级为PREVIOUS的组件获焦,否则,由焦点组容器内上次获焦的组件获焦。| **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## focusScopeId14+ focusScopeId(id: string, isGroup?: boolean, arrowStepOut?: boolean): T 设置当前容器组件的id标识,以及是否为焦点组。 **原子化服务API:** 从API version 14开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | id | string | 是 | 设置当前容器组件的id标识。
**说明:**
单个[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)下,id标识全局唯一,不可重复。| | isGroup | boolean | 否 | 设置当前容器组件是否为焦点组。true表示容器组件为焦点组,false表示容器组件不是焦点组。默认值为false。
**说明:**
焦点组不可嵌套,不可重复配置。
焦点组不能和tabIndex混用。
配置焦点组的目的是使得容器及容器内的元素可以按照焦点组规则走焦。焦点组走焦规则:
1.焦点组容器内只能通过方向键走焦,tab键会使焦点跳出焦点组容器。
2.通过方向键使焦点从焦点组容器外切换到焦点组容器内时,若焦点组容器内存在优先级为PREVIOUS的组件,则优先级为PREVIOUS的组件获焦,否则,由焦点组容器内上次获焦的组件获焦。| | arrowStepOut | boolean | 否 | 设置能否使用方向键走焦出当前焦点组。true表示可以使用方向键走焦出当前焦点组,false表示不能使用方向键走焦出当前焦点组。默认值为true。| **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## tabStop14+ tabStop(isTabStop: boolean): T 设置当前容器组件的tabStop,可决定焦点在走焦时是否会停留在当前容器。 **原子化服务API:** 从API version 14开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | isTabStop | boolean | 是 | 设置当前容器组件是否为走焦可停留容器。true表示当前容器组件为走焦可停留容器,false表示当前容器组件不是走焦可停留容器。
**说明:**
1.配置tabStop需要确保是容器组件且有可获焦的孩子组件,默认容器组件不能直接获焦。
2.通过[requestFocus](../arkts-apis-uicontext-focuscontroller.md#requestfocus12)请求焦点,如果是容器组件且配置tabStop,焦点能够停留在容器组件,如果未配置tabStop,即使整条焦点链上有配置了tabStop的组件,该组件依然能获取到焦点。
3.配置tabStop的容器不允许嵌套超过2层。
tabStop走焦规则:
1.通过tab键和方向键走焦,焦点会停留在配置了tabStop的组件上,如果焦点停留在配置了tabStop的容器内部时,可以走焦到容器内部的下一个可获焦组件,如果焦点停留在配置了tabStop的容器外部时,可以走焦到容器外的下一个可获焦组件。
2.当焦点停留在tabStop上时,按Enter键可以走焦到内部第一个可获焦组件,按ESC能够将焦点退回到不超过当前[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)根容器的上一个配置了tabStop的组件,按空格键可以响应该容器的onClick事件。
3.不建议根容器配置tabStop。如果根容器配置了tabStop,通过[clearFocus](../arkts-apis-uicontext-focuscontroller.md#clearfocus12)将焦点清理到根容器,再按Enter键会重新走回内部上一次获焦组件,通过ESC键将焦点清理到根容器,再按Enter键会走焦到内部第一个可获焦组件。| **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | **描述走焦的时候的按键以及获焦组件** ![tabStop](figures/tabStop.png) 如果当前焦点停留在button2上,按下tab键将会走焦到Column3,再按下tab键会循环走焦到button1。 ## nextFocus18+ nextFocus(nextStep: Optional\): T 设置组件的自定义焦点走焦逻辑。 **原子化服务API:** 从API version 18开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full **参数:** | 参数名 | 类型 | 必填 | 说明 | | ------ | ------- | ---- | ------------------------------------------------------------ | | nextStep | Optional\<[FocusMovement](#focusmovement18对象说明)> | 是 | 设置当前容器组件的自定义走焦规则。
**说明:**
默认值为重置nextStep为空。
没设置自定义走焦或者设置自定义组件容器不存在,仍进行默认走焦规则。| **返回值:** | 类型 | 说明 | | -------- | -------- | | T | 返回当前组件。 | ## FocusMovement18+对象说明 设置对应的按键对应的走焦目的组件,缺省则遵循默认走焦规则。 **原子化服务API:** 从API version 18开始,该接口支持在原子化服务中使用。 **系统能力:** SystemCapability.ArkUI.ArkUI.Full | 名称 | 类型 | 只读 | 可选 | 说明 | | ---- | ---- | ---- | ---- | ---- | | forward | string | 否 | 是 | 通过tab键走焦到组件的id。
默认值为重置forward为空。 | | backward | string | 否 | 是 | 通过shift+tab键走焦到组件的id。
默认值为重置backward为空。 | | up | string | 否 | 是 | 通过方向键上键走焦到组件的id。
默认值为重置up为空。 | | down | string | 否 | 是 | 通过方向键下键走焦到组件的id。
默认值为重置down为空。 | | left | string | 否 | 是 | 通过方向键左键走焦到组件的id。
默认值为重置left为空。 | | right | string | 否 | 是 | 通过方向键右键走焦到组件的id。
默认值为重置right为空。 | ## 示例 ### 示例1(设置组件获焦和走焦的效果) 该示例通过配置defaultFocus可以使绑定的组件成为[层级页面](../../../ui/arkts-common-events-focus-event.md#基础概念)创建后首次获焦的焦点,配置groupDefaultFocus可以使绑定的组件成为tabIndex容器创建后首次获焦的焦点,配置focusOnTouch可以使绑定的组件点击后立即获焦。 ```ts // focusTest.ets @Entry @Component struct FocusableExample { @State inputValue: string = ''; build() { Scroll() { Row({ space: 20 }) { Column({ space: 20 }) { Column({ space: 5 }) { Button('Group1') .width(165) .height(40) .fontColor(Color.White) .focusOnTouch(true) // 该Button组件点击后可获焦 Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) .focusOnTouch(true) // 该Button组件点击后可获焦 } Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } }.borderWidth(2).borderColor(Color.Red).borderStyle(BorderStyle.Dashed) .tabIndex(1) // 该Column组件为按TAB键走焦的第一个获焦的组件 Column({ space: 5 }) { Button('Group2') .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) .groupDefaultFocus(true) // 该Button组件上级Column组件获焦时获焦 } Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } }.borderWidth(2).borderColor(Color.Green).borderStyle(BorderStyle.Dashed) .tabIndex(2) // 该Column组件为按TAB键走焦的第二个获焦的组件 } Column({ space: 5 }) { TextInput({placeholder: 'input', text: this.inputValue}) .onChange((value: string) => { this.inputValue = value }) .width(156) .defaultFocus(true) // 该TextInput组件为层级页面的初始默认焦点 Button('Group3') .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } Button() .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } Button() .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } }.borderWidth(2).borderColor(Color.Orange).borderStyle(BorderStyle.Dashed) .tabIndex(3) // 该Column组件为按TAB键走焦的第三个获焦的组件 }.alignItems(VerticalAlign.Top) } } } ``` 示意图: 首次进入时,焦点默认在defaultFocus绑定的TextInput组件上: ![defaultFocus](figures/defaultFocus.png) 首次按TAB键,焦点切换到tabIndex(1)的容器上,且自动走焦到内部第一个可获焦组件上: ![groupDefaultFocus1](figures/groupDefaultFocus1.png) 第二次按TAB键,焦点切换到tabIndex(2)的容器上,且自动走到其内部的groupDefaultFocus绑定的组件上: ![groupDefaultFocus2](figures/groupDefaultFocus2.png) 第三次按TAB键,焦点切换到tabIndex(3)的容器上,且自动走焦到内部配置了DefaultFocus的组件上: ![groupDefaultFocus3](figures/groupDefaultFocus3.png) 点击绑定了focusOnTouch的组件,组件自身获焦,焦点框被清除,再按下TAB键后,显示焦点框: ![focusOnTouch](figures/focusOnTouch.png) ### 示例2(设置指定组件获焦) 该示例通过配置focusControl.requestFocus使指定组件获取焦点。 > **说明:** > > 直接使用focusControl可能导致[UI上下文不明确](../../../ui/arkts-global-interface.md)的问题,建议使用getUIContext()获取[UIContext](../arkts-apis-uicontext-uicontext.md)实例,并使用[getFocusController](../arkts-apis-uicontext-uicontext.md#getfocuscontroller12)获取绑定实例的focusControl。 ```ts // requestFocus.ets @Entry @Component struct RequestFocusExample { @State idList: string[] = ['A', 'B', 'C', 'D', 'E', 'F', 'LastPageId']; @State selectId: string = 'LastPageId'; build() { Column({ space:20 }){ Row({space: 5}) { Button("id: " + this.idList[0] + " focusable(false)") .width(200).height(70).fontColor(Color.White) .id(this.idList[0]) .focusable(false) Button("id: " + this.idList[1]) .width(200).height(70).fontColor(Color.White) .id(this.idList[1]) } Row({space: 5}) { Button("id: " + this.idList[2]) .width(200).height(70).fontColor(Color.White) .id(this.idList[2]) Button("id: " + this.idList[3]) .width(200).height(70).fontColor(Color.White) .id(this.idList[3]) } Row({space: 5}) { Button("id: " + this.idList[4]) .width(200).height(70).fontColor(Color.White) .id(this.idList[4]) Button("id: " + this.idList[5]) .width(200).height(70).fontColor(Color.White) .id(this.idList[5]) } Row({space: 5}) { Select([{value: this.idList[0]}, {value: this.idList[1]}, {value: this.idList[2]}, {value: this.idList[3]}, {value: this.idList[4]}, {value: this.idList[5]}, {value: this.idList[6]}]) .value(this.selectId) .onSelect((index: number) => { this.selectId = this.idList[index] }) Button("RequestFocus") .width(200).height(70).fontColor(Color.White) .onClick(() => { // 建议使用this.getUIContext().getFocusController().requestFocus() let res = focusControl.requestFocus(this.selectId) // 使选中的this.selectId的组件获焦 if (res) { this.getUIContext().getPromptAction().showToast({message: 'Request success'}) } else { this.getUIContext().getPromptAction().showToast({message: 'Request failed'}) } }) } }.width('100%').margin({ top:20 }) } } ``` 示意图: 按下TAB键,激活焦点态显示。 申请不存在的组件获焦: ![requestFocus1](figures/requestFocus1.png) 申请不可获焦的组件获焦: ![requestFocus2](figures/requestFocus2.png) 申请存在且可获焦的组件获焦: ![requestFocus3](figures/requestFocus3.png) ### 示例3(设置焦点框样式) 该示例通过配置focusBox修改组件的焦点框样式。 ```ts import { ColorMetrics, LengthMetrics } from '@kit.ArkUI'; @Entry @Component struct RequestFocusExample { build() { Column({ space: 30 }) { Button("small black focus box") .focusBox({ margin: new LengthMetrics(0), strokeColor: ColorMetrics.rgba(0, 0, 0), }) Button("large red focus box") .focusBox({ margin: LengthMetrics.px(20), strokeColor: ColorMetrics.rgba(255, 0, 0), strokeWidth: LengthMetrics.px(10) }) } .alignItems(HorizontalAlign.Center) .width('100%') } } ``` ![focusBox](figures/focusBox.gif) ### 示例4(设置焦点组走焦) 该示例通过配置focusScopePriority,可以使绑定的组件在所属容器首次获焦时成为焦点,配置focusScopeId,可以使绑定的容器组件成为焦点组。 ```ts // focusTest.ets @Entry @Component struct FocusableExample { @State inputValue: string = ''; build() { Scroll() { Row({ space: 20 }) { Column({ space: 20 }) { // 标记为Column1 Column({ space: 5 }) { Button('Group1') .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } }.borderWidth(2).borderColor(Color.Red).borderStyle(BorderStyle.Dashed) Column({ space: 5 }) { Button('Group2') .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) .focusScopePriority('ColumnScope1', FocusPriority.PRIOR) // Column1首次获焦时获焦 } Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } }.borderWidth(2).borderColor(Color.Green).borderStyle(BorderStyle.Dashed) } .focusScopeId('ColumnScope1') Column({ space: 5 }) { // 标记为Column2 TextInput({placeholder: 'input', text: this.inputValue}) .onChange((value: string) => { this.inputValue = value }) .width(156) Button('Group3') .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } Button() .width(165) .height(40) .fontColor(Color.White) .focusScopePriority('ColumnScope2', FocusPriority.PREVIOUS) // Column2获焦时获焦 Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } Button() .width(165) .height(40) .fontColor(Color.White) Row({ space: 5 }) { Button() .width(80) .height(40) .fontColor(Color.White) Button() .width(80) .height(40) .fontColor(Color.White) } }.borderWidth(2).borderColor(Color.Orange).borderStyle(BorderStyle.Dashed) .focusScopeId('ColumnScope2', true) // Column2为焦点组 }.alignItems(VerticalAlign.Top) } } } ``` 示意图: 首次按下TAB键时,焦点转移到容器1中绑定focusScopePriority的组件上。 ![focusScopePriority1](figures/focusScopePriority1.png) 继续按下TAB键,焦点转移到容器1下一个组件上。 ![focusScopePriority2](figures/focusScopePriority2.png) 再次按下TAB键,焦点转移到容器1下一个组件上。 ![focusScopePriority3](figures/focusScopePriority3.png) 继续按下TAB键,焦点转移到容器2中配置了focusScopePriority的组件上。 ![focusScopePriority4](figures/focusScopePriority4.png) 继续按下TAB键,焦点转移到容器1中名为Group1的组件上。 ![focusScopePriority5](figures/focusScopePriority5.png) ### 示例5(设置tab走焦停留) 该示例通过配置tabStop实现使用tab走焦停留在组件上。 ```ts import { ColorMetrics, LengthMetrics } from '@kit.ArkUI'; @Entry @Component struct TabStop { build() { Column({ space: 20 }) { Column({ space: 20 }) { Column({ space: 20 }) { Row({ space: 5 }) { Button("button 1") .width(200).height(70).fontColor(Color.White) .focusBox({ margin: LengthMetrics.px(20), strokeColor: ColorMetrics.rgba(23, 169, 141), strokeWidth: LengthMetrics.px(10) }) } Row({ space: 5 }) { Button("button 2") .width(200).height(70).fontColor(Color.White) .focusBox({ margin: LengthMetrics.px(20), strokeColor: ColorMetrics.rgba(23, 169, 141), strokeWidth: LengthMetrics.px(10) }) } }.width('80%').margin({ top: 30 }).borderColor(Color.Black) }.width('95%').margin({ top: 60 }).borderColor(Color.Black) Column({ space: 20 }) { Column({ space: 20 }) { Row({ space: 5 }) { Button("button 3") .width(200) .height('70%') .fontColor(Color.White) .focusBox({ margin: LengthMetrics.px(20), strokeColor: ColorMetrics.rgba(23, 169, 141), strokeWidth: LengthMetrics.px(10) }) .margin({ top: 15 }) } } .width('80%') .height('120') .borderColor(Color.Black) .margin({ top: 10 }) .tabStop(true) .focusBox({ margin: LengthMetrics.px(20), strokeColor: ColorMetrics.rgba(23, 169, 141), strokeWidth: LengthMetrics.px(10) }) .borderWidth(1) }.width('95%').margin({ top: 50 }).borderColor(Color.Black) } } } ``` 示意图: 连续按下两次TAB键,焦点转移到button2上。 ![tabStop2](figures/tabStop2.png) 接着按下TAB键,焦点转移到配置了tabStop的组件。 ![tabStop3](figures/tabStop3.png) 再按下Enter键,焦点转移至内部button3上。 ![tabStop4](figures/tabStop4.png) 再按下ESC键,焦点转移到配置了tabStop的组件上。 ![tabStop3](figures/tabStop3.png) 再按下TAB键,焦点循环走焦到button1上。 ![tabStop1](figures/tabStop1.png) ### 示例6(设置自定义走焦) 该示例通过配置nextFocus实现自定义走焦规则。 如果不配置nextFocus,默认的点击tab的走焦顺序为:M->A->B->C;配置了nextFocus以后,走焦顺序变更为:M->D->F->B。 ```ts class MyButtonModifier implements AttributeModifier { applyNormalAttribute(instance: ButtonAttribute): void { instance.id('M') instance.nextFocus({forward: 'D', up: 'C', down: 'D'}) } } @Entry @Component struct Index { @State modifier: MyButtonModifier = new MyButtonModifier(); @State idList: string[] = ['A', 'B', 'C', 'D', 'E', 'F']; build() { Column({space: 10}) { Row({space: 10}) { Button("id: M") .attributeModifier(this.modifier) Button("id: " + this.idList[0]) .id(this.idList[0]) .nextFocus({forward: 'C', backward: 'M', up: 'E', right: 'F', down: 'B', left: 'D'}); Button("id: " + this.idList[1]) .id(this.idList[1]) } Column({space: 10}) { Button("id: " + this.idList[2]) .id(this.idList[2]); Button("id: " + this.idList[3]) .id(this.idList[3]) .nextFocus({forward: 'F'}); } Row({space: 10}) { Button("id: " + this.idList[4]) .id(this.idList[4]); Button("id: " + this.idList[5]) .id(this.idList[5]) .nextFocus({forward: 'B'}); } } } } ``` ![focusBox](figures/nextStep.gif)