How to make Android talkback focus inside tabcontent of TabActivity?


Rahul Rahul Verma Asked on Jan 6, 2025
Summary:-

I am not good at English so I am using Google Translate to ask a question.

My Android project is implemented in Java and is implemented using tabhost and TabActivity.

I know that this method is deprecated when implementing tabs, but I do not have enough time to change it, so I have to use them as they are.

In this situation, I need to add "Android talkback" to this project.

My code is similar to the following.

// TabActivity.java
public class TabActivity extends android.app.TabActivity {

    private TabHost tabHost;

    @Override
    protected void onCreate(Bundle saveInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_tab);

        tabHost = getTabHost();
        tabHost.setup();

        TabHost.TabSpec spec;
        Intent intent;

        spec = tabHost.newTabSpec("Tab 00");
        spec.setIndicator("Tab1");
        intent = new Intent(this, FirstTabActivity.class).addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
        spec.setContent(intent);
        tabHost.addTab(spec);

        spec = tabHost.newTabSpec("Tab 01");
        spec.setIndicator("Tab2");
        intent = new Intent(this, TabItem.class).addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
        spec.setContent(intent);
        tabhost.addTab(spec);

        spec = tabHost.newTabSpec("Tab 02");
        spec.setIndicator("Tab3");
        intent = new Intent(this, TabItem.class).addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
        spec.setContent(intent);
        tabHost.addTab(spec);

        tabHost.setCurrentTab(0);
    }

}

Urgency of Question:-

High Urgency

Skill Level:-

Intermediate

Sajid Sajid Ansari Commented on Jan 6, 2025

Below is a step-by-step approach to help Android TalkBack (screen reader) focus on the contents of each tab in an older TabActivity setup. Note that TabActivity is deprecated, and the recommended approach is to use Fragments and the TabLayout (from Material Components) or ViewPager. However, if you need to continue with TabActivity, you can try the following.

1. Set Focus Programmatically After Tab Switch

One way to ensure TalkBack focuses on the contents of the new tab is to request focus for a view inside that tab once the tab is switched. You can do this by:

  1. Listen for tab changes using TabHost.OnTabChangeListener.
  2. Once the new tab is selected, find a suitable view inside that tab’s layout and call requestFocus() or send an accessibility event to move the TalkBack focus onto it.

Example:

tabHost.setOnTabChangedListener(new TabHost.OnTabChangeListener() {
    @Override
    public void onTabChanged(String tabId) {
        // Use the tabId to check which tab is selected
        // For instance, if tabId.equals("Tab 01"), you know you're on Tab 2

        // Find a view in the newly selected tab's layout
        View newTabContent = tabHost.getCurrentView();

        // Make sure the view you want is focusable
        newTabContent.setFocusable(true);
        newTabContent.requestFocus();

        // Optionally, send an accessibility event so TalkBack jumps there
        newTabContent.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED);
    }
});

Points to Remember

  • Ensure the view you are calling requestFocus() on is visible and focusable.
  • If the root view is not the best candidate, find a specific child view (like a heading TextView or the first interactive control in the layout).

2. Ensure Views in the Tab Are Focusable for Accessibility

In your layouts (e.g., activity_first_tab.xml, activity_tab_item.xml), check that interactive or important elements have:

  • android:focusable="true"
  • android:clickable="true" (if it’s a clickable control)
  • Appropriate content descriptions for non-text elements using android:contentDescription="@string/..." for images or icons.

For text views that you want TalkBack to read, just ensure the text is present. TalkBack will read it automatically if it’s in focus.

3. Use Accessibility Labels Where Appropriate

For better clarity, label your UI elements. Some tips:

  • For important text views that serve as headings, you can add:
<TextView
    android:id="@+id/tab_heading"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="@string/tab_title"
    android:focusable="true"
    android:contentDescription="@string/accessibility_tab_heading" />

For images or icons, add a meaningful content description:

 <ImageView
    android:id="@+id/tab_icon"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:src="@drawable/ic_tab_icon"
    android:contentDescription="@string/accessibility_tab_icon" />

4. Consider Sending Accessibility Events

If simply calling requestFocus() is not enough, you can also trigger an AccessibilityEvent from the newly focused view:

newTabContent.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED);

This event should alert TalkBack that focus has shifted to the new view. Additionally, you can provide custom accessibility text by overriding the view’s dispatchPopulateAccessibilityEvent() if needed.

5. Verify Behavior on Different OS Versions and Devices

TabActivity is quite old, and behavior can vary across devices:

  1. Test your solution on multiple Android versions to ensure consistent TalkBack behavior.
  2. Consider migrating to a more modern approach (TabLayout + ViewPager or ViewPager2 + Fragment).

(Root Cause)

  • TalkBack relies on focusable views and accessibility events to know where to read or navigate.
  • In a traditional TabActivity, views for each tab can be swapped out of view (hence no focus).
  • By explicitly focusing on a visible and focusable view after each tab change, you direct the screen reader to the correct location in the new tab content.

Do you know the Answer?


Got a question? Ask our extensive community for help. Submit Your Question