diff --git a/accessibility-checker-engine/help-v4/en-US/element_tabbable_unobscured.html b/accessibility-checker-engine/help-v4/en-US/element_tabbable_unobscured.html
new file mode 100644
index 000000000..d4d749dc6
--- /dev/null
+++ b/accessibility-checker-engine/help-v4/en-US/element_tabbable_unobscured.html
@@ -0,0 +1,117 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/accessibility-checker-engine/src/v4/rules/element_tabbable_unobscured.ts b/accessibility-checker-engine/src/v4/rules/element_tabbable_unobscured.ts
new file mode 100644
index 000000000..f12fed433
--- /dev/null
+++ b/accessibility-checker-engine/src/v4/rules/element_tabbable_unobscured.ts
@@ -0,0 +1,100 @@
+/******************************************************************************
+ Copyright:: 2022- IBM, Inc
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+ http://www.apache.org/licenses/LICENSE-2.0
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
+ *****************************************************************************/
+
+import { RPTUtil } from "../../v2/checker/accessibility/util/legacy";
+import { getDefinedStyles, getComputedStyle } from "../util/CSSUtil";
+import { Rule, RuleResult, RuleFail, RuleContext, RulePass, RuleContextHierarchy, RulePotential } from "../api/IRule";
+import { eRulePolicy, eToolkitLevel } from "../api/IRule";
+
+export let element_tabbable_unobscured: Rule = {
+ id: "element_tabbable_unobscured",
+ context: "dom:*",
+ dependencies: [],
+ help: {
+ "en-US": {
+ "group": "element_tabbable_unobscured.html",
+ "pass": "element_tabbable_unobscured.html",
+ "potential_visible": "element_tabbable_unobscured.html"
+ }
+ },
+ messages: {
+ "en-US": {
+ "group": "A tabbable element should be visible on the screen when it has keyboard focus",
+ "pass": "The tabbable element is visible on the screen",
+ "potential_visible": "Confirm the element should be tabbable, and is visible on the screen when it has keyboard focus"
+ }
+ },
+ rulesets: [{
+ id: ["WCAG_2_2"],
+ num: ["2.4.11"],
+ level: eRulePolicy.RECOMMENDATION,
+ toolkitLevel: eToolkitLevel.LEVEL_THREE
+ }],
+ act: [],
+ run: (context: RuleContext, options?: {}, contextHierarchies?: RuleContextHierarchy): RuleResult | RuleResult[] => {
+ const ruleContext = context["dom"].node as HTMLElement;
+ if (!RPTUtil.isTabbable(ruleContext))
+ return null;
+
+ const nodeName = ruleContext.nodeName.toLocaleLowerCase();
+ const bounds = context["dom"].bounds;
+ //in case the bounds not available
+ if (!bounds) return null;
+
+ // defined styles only give the styles that changed
+ const defined_styles = getDefinedStyles(ruleContext);
+ const onfocus_styles = getDefinedStyles(ruleContext, ":focus");
+
+ if (bounds['height'] === 0 || bounds['width'] === 0
+ || (defined_styles['position']==='absolute' && defined_styles['clip'] && defined_styles['clip'].replaceAll(' ', '')==='rect(0px,0px,0px,0px)'
+ && !onfocus_styles['clip']))
+ return RulePotential("potential_visible", []);
+
+ if (bounds['top'] >= 0 && bounds['left'] >= 0)
+ return RulePass("pass");
+
+ const default_styles = getComputedStyle(ruleContext);
+
+ let top = bounds['top'];
+ let left = bounds['left'];
+
+ if (Object.keys(onfocus_styles).length === 0 ) {
+ // no onfocus position change, but could be changed from js
+ return RulePotential("potential_visible", []);
+ } else {
+ // with onfocus position change
+ var positions = ['absolute', 'fixed'];
+ if (typeof onfocus_styles['top'] !== 'undefined') {
+ if (positions.includes(onfocus_styles['position']) || (typeof onfocus_styles['position'] === 'undefined' && positions.includes(default_styles['position']))) {
+ top = onfocus_styles['top'].replace(/\D/g,'');
+ } else {
+ // the position is undefined and the parent's position is 'relative'
+ top = Number.MIN_VALUE;
+ }
+ }
+ if (typeof onfocus_styles['left'] !== 'undefined') {
+ if (positions.includes(onfocus_styles['position']) || (typeof onfocus_styles['position'] === 'undefined' && positions.includes(default_styles['position']))) {
+ left = onfocus_styles['left'].replace(/\D/g,'');
+ } else {
+ // the position is undefined and the parent's position is 'relative'
+ left = Number.MIN_VALUE;
+ }
+ }
+ }
+
+ if (top >= 0 && left >= 0)
+ return RulePass("pass");
+ else
+ return RulePotential("potential_visible", []);
+ }
+}