DEV Community

Vladimir Simić
Vladimir Simić

Posted on

When a Single URL Stops Being Enough: Multi-Table Pages in Inertia.js

Two Tables, One URL: Solving Inertia.js State Collisions

If you've worked with Inertia.js + TanStack Table for server-side filtering and pagination, you know how satisfying the pattern feels.

A single router.get() call gives you a lot for free:

  • Filter state lives in the URL
  • Links are shareable
  • Browser back/forward works out of the box
  • Sorting, pagination, and filtering are all server-driven

It's a great default — until you need two independent tables on the same page.


The Problem

Picture a typical admin dashboard with a Users table and an Orders table. Both support filtering, sorting, and pagination. Naturally, you reach for the same Inertia pattern you've been using:

router.get('/dashboard', {
  page: 3,
  sortby: 'name',
  status: 'active'
})
Enter fullscreen mode Exit fullscreen mode

That works beautifully for one table. With two, the URL becomes a shared state container — and things start to collide.

When the Users table navigates to page 3:

/dashboard?users_page=3
Enter fullscreen mode Exit fullscreen mode

What happens to the Orders filters? Without extra care, they get reset or overwritten.


Option 1: Namespace Your Parameters

The most obvious fix is prefixing every parameter by table:

users_page=3&users_sort=name
orders_page=2&orders_status=paid
Enter fullscreen mode Exit fullscreen mode

It works. But every filter hook, every query builder, and every table abstraction now needs to understand namespaced parameters. In my experience, that complexity compounds quickly and adds friction faster than it adds value.


Option 2: Give Each Table Its Own Request

Instead of routing all table interactions through router.get(), I switched to axios.get() per table.

Each table gets:

  • Its own dedicated endpoint
  • Its own local state
  • Its own independent requests

No shared URL, no collisions.


The Hook

The core of the pattern is a reusable useDataTable hook:

export function useDataTable<TData>({
  initialData,
  endpoint,
  columns,
}) {
  const [tableData, setTableData] = useState(initialData)
  const [sorting, setSorting] = useState([])
  const [columnFilters, setColumnFilters] = useState([])
  const [pagination, setPagination] = useState({
    pageIndex: 0,
    pageSize: 50,
  })

  const fetchData = useDebounce(async () => {
    const { data } = await axios.get(endpoint, {
      params: {
        filters: Object.fromEntries(
          columnFilters.map(({ id, value }) => [id, value])
        ),
        page: pagination.pageIndex + 1,
        sortby: sorting[0]?.id,
        sort: sorting[0]?.desc ? 'desc' : 'asc',
      },
    })

    setTableData(data)
  }, 300)

  useEffect(() => {
    fetchData()
  }, [sorting, pagination, columnFilters])

  return useReactTable({
    data: tableData.data,
    rowCount: tableData.total,
    manualFiltering: true,
    manualPagination: true,
    manualSorting: true,
  })
}
Enter fullscreen mode Exit fullscreen mode

Why This Works Well

Tables are truly independent. Paginate one, and the other doesn't flinch. That's exactly the behavior you want from an admin dashboard.

The first render still comes from Inertia. Initial data is hydrated from Inertia props:

const [tableData, setTableData] = useState(initialData)
Enter fullscreen mode Exit fullscreen mode

No loading spinner on mount, no client-side waterfall. The page feels fast from the start — the best of both worlds.

CSRF is handled automatically. Because axios reads Laravel's XSRF cookie by default, there's no extra ceremony. It just works.


The Tradeoff

This pattern makes a deliberate trade.

What you gain: independent tables, simpler code than full query namespacing, and cleaner admin pages.

What you give up: filter state is no longer in the URL. Filters aren't bookmarkable. Shared links won't preserve table state.

Whether that matters depends entirely on your use case.


When to Use Each Approach

Stick with router.get() + URL state when:

  • You're building a public-facing search or filter page
  • Users need to share or bookmark filtered views
  • Deep-linking is part of the product (product catalogs, reporting screens, search results)

Switch to axios per table when:

  • It's internal admin tooling
  • The page has multiple independent data views
  • URL shareability simply isn't a requirement

This Isn't a Criticism of Inertia

I almost called this post "When Inertia isn't enough" — but that framing misses the point.

Inertia is excellent. This is just a case where a single URL stops being the right state container for the job. Recognizing that distinction is the whole insight.


How Are You Handling This?

If you're managing multiple server-side tables with Inertia, I'd love to know your approach. Are you namespacing query params? Using local state and axios? Something smarter?

Drop a comment — if this is a common enough pain point, I'm considering packaging the hook or writing a more detailed follow-up.

Top comments (0)