I stand up for children in need. Please join me in helping this family.

Skip to content
Steven Roland

Building a Dynamic Tabs Component with Alpine.js

Tabs are a popular UI element that allow users to navigate between different sections of content without leaving the page. With Alpine.js, we can create an interactive and accessible tabs component with minimal JavaScript. Let's dive into building a tabs component that's both functional and user-friendly.

Setting Up Alpine.js

First, ensure you have Alpine.js included in your project. You can add it via CDN by including this script tag in your HTML file:

<script src="https://cdn.jsdelivr.net/gh/alpinejs/[email protected]/dist/alpine.min.js" defer></script>

Basic Tabs Structure

Let's start with a basic tabs structure using Alpine.js:

<div x-data="{ activeTab: 'tab1' }">
  <div role="tablist">
    <button @click="activeTab = 'tab1'" :class="{ 'active': activeTab === 'tab1' }" role="tab">Tab 1</button>
    <button @click="activeTab = 'tab2'" :class="{ 'active': activeTab === 'tab2' }" role="tab">Tab 2</button>
    <button @click="activeTab = 'tab3'" :class="{ 'active': activeTab === 'tab3' }" role="tab">Tab 3</button>
  </div>
  <div x-show="activeTab === 'tab1'" role="tabpanel">Content for Tab 1</div>
  <div x-show="activeTab === 'tab2'" role="tabpanel">Content for Tab 2</div>
  <div x-show="activeTab === 'tab3'" role="tabpanel">Content for Tab 3</div>
</div>

This basic structure uses Alpine.js directives to control the tabs:

  • x-data="{ activeTab: 'tab1' }" initializes the component's state with the first tab active.

  • @click="activeTab = 'tab1'" changes the active tab when a tab button is clicked.

  • :class="{ 'active': activeTab === 'tab1' }" applies an 'active' class to the current tab.

  • x-show="activeTab === 'tab1'" displays the content for the active tab.

Enhancing the Tabs Component

Now, let's enhance our tabs with styling, transitions, and improved accessibility:

<div x-data="tabs()" class="max-w-3xl mx-auto mt-8">
  <div role="tablist" aria-label="Tabs" class="flex border-b">
    <template x-for="tab in tabItems" :key="tab.id">
      <button 
        :id="tab.id" 
        @click="activeTab = tab.id"
        :class="{ 'border-b-2 border-blue-500': activeTab === tab.id }"
        class="px-4 py-2 text-gray-600 hover:text-blue-500 focus:outline-none"
        role="tab"
        :aria-selected="activeTab === tab.id"
        :aria-controls="tab.id + '-panel'"
      >
        <span x-text="tab.label"></span>
      </button>
    </template>
  </div>
  <div class="mt-4">
    <template x-for="tab in tabItems" :key="tab.id">
      <div 
        :id="tab.id + '-panel'"
        x-show="activeTab === tab.id"
        role="tabpanel"
        :aria-labelledby="tab.id"
        x-transition:enter="transition ease-out duration-300"
        x-transition:enter-start="opacity-0 transform scale-90"
        x-transition:enter-end="opacity-100 transform scale-100"
      >
        <div x-html="tab.content"></div>
      </div>
    </template>
  </div>
</div>
<script>
  function tabs() {
    return {
      activeTab: 'tab1',
      tabItems: [
        { id: 'tab1', label: 'Tab 1', content: '<p>Content for Tab 1</p>' },
        { id: 'tab2', label: 'Tab 2', content: '<p>Content for Tab 2</p>' },
        { id: 'tab3', label: 'Tab 3', content: '<p>Content for Tab 3</p>' }
      ]
    }
  }
</script>

Let's break down the enhancements:

  1. We've moved the tab data into a JavaScript function, making it easier to manage and extend.

  2. We're using x-for to iterate over the tab items, reducing repetition in our HTML.

  3. We've added proper ARIA attributes for improved accessibility.

  4. Tailwind CSS classes are used for styling (you'll need to include Tailwind in your project).

  5. Alpine.js transition directives are used for smooth content transitions.

Adding Keyboard Navigation

To make our tabs fully accessible, let's add keyboard navigation:

<div x-data="tabs()" @keydown.right.prevent="nextTab()" @keydown.left.prevent="prevTab()" class="max-w-3xl mx-auto mt-8">
  <div role="tablist" aria-label="Tabs" class="flex border-b">
    <template x-for="(tab, index) in tabItems" :key="tab.id">
      <button 
        :id="tab.id" 
        @click="activeTab = tab.id"
        @focus="activeTab = tab.id"
        :class="{ 'border-b-2 border-blue-500': activeTab === tab.id }"
        class="px-4 py-2 text-gray-600 hover:text-blue-500 focus:outline-none"
        role="tab"
        :aria-selected="activeTab === tab.id"
        :aria-controls="tab.id + '-panel'"
        :tabindex="activeTab === tab.id ? 0 : -1"
      >
        <span x-text="tab.label"></span>
      </button>
    </template>
  </div>
  <!-- Tab content remains the same -->
</div>
<script>
  function tabs() {
    return {
      activeTab: 'tab1',
      tabItems: [
        { id: 'tab1', label: 'Tab 1', content: '<p>Content for Tab 1</p>' },
        { id: 'tab2', label: 'Tab 2', content: '<p>Content for Tab 2</p>' },
        { id: 'tab3', label: 'Tab 3', content: '<p>Content for Tab 3</p>' }
      ],
      nextTab() {
        let index = this.tabItems.findIndex(tab => tab.id === this.activeTab);
        index = (index + 1) % this.tabItems.length;
        this.activeTab = this.tabItems[index].id;
        this.$nextTick(() => {
          document.getElementById(this.activeTab).focus();
        });
      },
      prevTab() {
        let index = this.tabItems.findIndex(tab => tab.id === this.activeTab);
        index = (index - 1 + this.tabItems.length) % this.tabItems.length;
        this.activeTab = this.tabItems[index].id;
        this.$nextTick(() => {
          document.getElementById(this.activeTab).focus();
        });
      }
    }
  }
</script>

In this final version:

  1. We've added @keydown.right.prevent="nextTab()" and @keydown.left.prevent="prevTab()" to enable keyboard navigation.

  2. The nextTab() and prevTab() functions handle circular navigation through the tabs.

  3. We've added @focus="activeTab = tab.id" to change the active tab when it receives focus.

  4. The tabindex attribute is dynamically set to ensure only the active tab is in the tab order.

Conclusion

With Alpine.js, we've created a tabs component that's interactive, visually appealing, and fully accessible. This component can be easily integrated into any project, providing a smooth user experience with minimal JavaScript.

Alpine.js's declarative syntax allows us to create complex UI components with ease, right in our HTML. By leveraging its directives and combining them with proper HTML structure and CSS, we can build powerful, responsive UI elements that enhance our web applications.

Remember, when implementing tabs, always consider accessibility and user experience. Properly implemented tabs can significantly improve navigation and content organization in your web applications.

More posts

Streamlining Laravel Development with Homestead

Laravel Homestead simplifies development environment setup using a pre-configured Vagrant box. Install Homestead, customize the Homestead.yaml file, and start developing. It's ideal for team development, managing multiple projects, and testing different PHP versions. Use it to streamline your Laravel development workflow.

Supercharging Your Laravel Application with Octane

Laravel Octane boosts application performance using high-powered servers like Swoole and RoadRunner. It offers features like pre-initialized workers, concurrent tasks, and a high-performance cache. Ideal for high-traffic websites, APIs, and real-time applications. Implement best practices like profiling your app and optimizing database queries for maximum benefit.

Supercharge Your Laravel Development with Jetstream

Laravel Jetstream is a powerful starter kit for modern web applications. It offers authentication, team management, API support, and profile management out of the box. Install Jetstream, customize features like team creation, and use it for SaaS apps, API-driven projects, and membership sites. Follow best practices to extend functionality responsibly.

"The only source of knowledge is experience."

Albert Einstein BrainyQuote