<div class="steps" role="group" aria-label="Group 1">

  <button aria-pressed="false">Shop</button>
  <button aria-pressed="false">Cart</button>
  <button aria-pressed="true" aria-current="step">Shipping</button>
  <button aria-pressed="false">Checkout</button>
  <button aria-pressed="false">Thank You</button>

  <div class="message"></div>

</div>

<div class="steps steps-2" aria-label="Group 2">

  <button aria-pressed="true" aria-current="step">Shop</button>
  <button aria-pressed="false">Cart</button>
  <button aria-pressed="false">Shipping</button>
  <button aria-pressed="false">Checkout</button>
  <button aria-pressed="false">Thank You</button>

  <div class="message"></div>

</div>

<dl>
  <dt>Using:</dt>
  <dd><a href="https://www.w3.org/TR/wai-aria-1.1/#aria-current"><code>aria-current</code></a></dd>
  <dt>Specifically the <code>step</code> value:</dt>
  <dd><q>Represents the current step within a process.</q></dd>
  <dt>Using:</dt>
  <dd><a href="https://www.w3.org/TR/wai-aria-1.1/#aria-pressed"><code>aria-pressed</code></a></dd>
  <dt>Replaced:</dt>
  <dd><code>button.active</code></dd>
  <dt>With:</dt>
  <dd><code>button[aria-current="step"][aria-pressed="true"]</code></dd>
  <dt>Replaced:</dt>
  <dd><code>button.classList.remove("active");</code></dd>
  <dt>With:</dt>
  <dd><code>button.removeAttribute("aria-current");</code><br>
    <code>button.setAttribute("aria-pressed", "false");</code></dd>
  <dt>Replaced:</dt>
  <dd><code>button.classList.add("active");</code></dd>
  <dt>With:</dt>
  <dd><code>button.setAttribute("aria-current", "step");</code><br>
    <code>button.setAttribute("aria-pressed", "true");</code></dd>
</dl>
.steps {
  counter-reset: currentStep 0 remainder 0 totalStep 0;
}

button {
  counter-increment: totalStep;
}
button::before {
  content: "";
  counter-increment: currentStep;
}
button[aria-current="step"][aria-pressed="true"] ~ button::before {
  // prevents currentStep from being incremented!
  counter-increment: remainder;
}

.message::before {
  content: "Step: " counter(currentStep) " / " counter(totalStep);
}

.steps-2 .message::before {
  content: "Step: " counter(currentStep) " / " counter(totalStep) " ("
    counter(remainder) " to go!)";
}

// JUST FOR DEMO
button {
  border: 0;
  background: #a5d6a7;
  color: black;
  padding: 0.5rem 1rem;
  border-radius: 5px;
  font-family: inherit;
}
button[aria-current="step"][aria-pressed="true"] {
  background: #43a047;
}
body {
  font: 110% system-ui;
  background: #eceff1;
  text-align: center;
}
.message {
  padding: 1rem;
  font-size: 2rem;
  font-weight: 900;
}
.steps {
  margin: 0 0 3rem 0;
}

dl > * {
  text-align: left;
}
View Compiled
const buttons = document.querySelectorAll("button");

buttons.forEach((button) => {
  button.addEventListener("click", () => {
    let siblings = button.parentNode.querySelectorAll("button");
    siblings.forEach((button) => {
      button.removeAttribute("aria-current");
      button.setAttribute("aria-pressed", "false");
    });
    button.setAttribute("aria-current", "step");
    button.setAttribute("aria-pressed", "true");
  });
});
Run Pen

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

This Pen doesn't use any external JavaScript resources.