DEV Community

Cover image for πŸ”ŽDo You ACTUALLY Need NgRx? (Or Are You Solving the Wrong Problem?)
abdelaaziz ouakala
abdelaaziz ouakala

Posted on

πŸ”ŽDo You ACTUALLY Need NgRx? (Or Are You Solving the Wrong Problem?)

Most Angular apps don't have a state-management problem. They have a state-ownership problem.

In enterprise Angular projects, the pattern is almost always the same:

A team starts a project. Someone says, "we'll need state management eventually."

NgRx gets added on day one.

Six months later, they're maintaining 400+ lines of boilerplate β€” actions, reducers, effects, selectors β€” just to manage a loading spinner and a modal toggle.

This isn't an NgRx problem. It's an ownership problem.

🚩Ownership defines architecture. Without it, even the best tools become unnecessary complexity.


πŸ“š Table of Contents


The Real Question Isn't "Which Library?"

The Real Question Isn't "Which Library?"

It's "Who owns this state?"

Most teams reach for a global store before they understand their state boundaries. They assume "reactive" means "global." It doesn't.

Angular Signals fundamentally changes this conversation.


What Signals Actually Changed

Before Signals, even local state was awkward. You'd reach for a BehaviorSubject, expose an observable, subscribe somewhere, handle takeUntil cleanup. It worked β€” but it was ceremonial.

Now:

// That's it. Reactive. Zero ceremony.
const count = signal(0);
const doubled = computed(() => count() * 2);

// Update
count.update(n => n + 1);
Enter fullscreen mode Exit fullscreen mode

Two lines. No subscription management. No boilerplate.

Your modal state, filter toggles, tab selection, loading indicators β€” all handled. Locally. Elegantly.

"Signals gave us the ability to start simple and add complexity only when boundaries prove insufficient."


The State Spectrum (Tool-Agnostic)
Not all states are created equal. Before choosing a tool, define the scope:

Scope Ownership Angular Solution
Local Component-owned. Lives and dies with the component. signal() + computed()
Shared Service-managed. Multiple components in the same feature. Injectable service + Signals
Global Cross-feature. Event-sourced. Auditable. NgRx (SignalStore or full)

The mistake is treating everything as global by default.


When You DON'T Need a Global Store
βœ… Modal visibility
βœ… Filter selections
βœ… Tab active state
βœ… Loading indicators
βœ… Form field state
βœ… Pagination cursor
βœ… Local UI preferences

None of these need NgRx. None of them ever did. Signals just made that obvious.


When NgRx Is Actually Justified
Let me be clear: NgRx still matters. Just not for everything.

You should consider NgRx when:

  • πŸ”„ Complex multi-step workflows β€” checkout flows, multi-stage forms, wizard-style processes.
  • πŸ“‹ Auditability requirements β€” compliance needs every state change logged and replayable.
  • πŸ‘₯ Distributed team boundaries β€” multiple teams writing to the same domain with clear contracts.
  • ⚑ Event-heavy orchestration β€” actions as the single source of truth across features.
  • πŸ› Time-travel debugging β€” when you genuinely need to replay state changes.

What NgRx gives you at scale:

-➑️ Actions as documented contracts.
-➑️ Reducers as pure, predictable transformations.
-➑️ Effects for side-effect isolation.
-➑️ DevTools for distributed debugging.
-➑️ Feature state isolation via modules.


The Blast Radius Framework
When deciding on state architecture, ask one question:

"What's the blast radius of this state change?"

Blast Radius Solution
1 component affected signal() locally
1 feature (3–5 components) Service + Signals
Multiple features / teams NgRx SignalStore
Cross-app events + compliance Full NgRx

This removes opinion from the decision and replaces it with architecture logic.


The Senior Developer's Rule

State complexity should justify architecture complexity. Never the reverse.

If your state-management setup is harder to explain than the business problem it solves, you've already shipped the wrong answer.

Don't scale your tooling faster than your app scales.


The Modern Angular Answer (Hybrid Model)
It's not "NgRx vs. Signals."

It's Signals locally, services for shared scope, NgRx for organizational scale.

◼️ signal() β€” Local Component State (Simplest)


// LOCAL: Component state with signals
@Component({...})
export class DashboardComponent {
  activeTab = signal(0);
  filtersOpen = signal(false);
}

// modal.component.ts β€” No NgRx needed here
@Component({
  selector: 'app-modal',
  standalone: true
})
export class ModalComponent {
  // βœ… Local state β€” stays local
  protected isOpen = signal(false);
  protected title = signal('');

  // βœ… Derived state β€” automatic reactivity
  protected headerClass = computed(() =>
    `modal-header ${this.isOpen() ? 'active' : 'hidden'}`
  );

  open(title: string) {
    this.title.set(title);
    this.isOpen.set(true);
  }

  close() {
    this.isOpen.set(false);
  }
}

Enter fullscreen mode Exit fullscreen mode

◼️ Service-based Shared State (Mid-tier)

// SHARED: Service-scoped signals
@Injectable({
  providedIn: 'root'
})
export class UserPreferencesService {
  // βœ… Private write, public read
  private _theme = signal<Theme>('light');
  private _language = signal<string>('en');

  // βœ… Public signals (read-only surface)
  theme = this._theme.asReadonly();
  language = this._language.asReadonly();

  // βœ… Derived computed state
  isDark = computed(() => this._theme() === 'dark');

  setTheme(t: Theme) {
    this._theme.set(t);
  }

  setLanguage(l: string) {
    this._language.set(l);
  }
}

Enter fullscreen mode Exit fullscreen mode

◼️ NgRx SignalStore β€” Scalable Domain State (Enterprise)

// GLOBAL: NgRx SignalStore for enterprise scale
// order.store.ts β€” When NgRx is justified
import { signalStore, withState, withMethods, withComputed } from '@ngrx/signals';

type OrderState = {
  orders: Order[];
  selectedId: string | null;
  loading: boolean;
};

export const OrderStore = signalStore(
  withState<OrderState>({
    orders: [],
    selectedId: null,
    loading: false
  }),
  withComputed(({ orders, selectedId }) => ({
    selectedOrder: computed(() =>
      orders().find(o => o.id === selectedId()) ?? null
    ),
    pendingCount: computed(() =>
      orders().filter(o => o.status === 'pending').length
    ),
  })),
  withMethods((store, orderService = inject(OrderService)) => ({
    async loadOrders() {
      patchState(store, { loading: true });
      const orders = await orderService.getAll();
      patchState(store, { orders, loading: false });
    },
  }))
);

Enter fullscreen mode Exit fullscreen mode

◼️ computed() β€” Derived State Pattern (Reactive)


// cart.component.ts β€” Derived state without manual subscriptions
@Component({
  standalone: true
})
export class CartComponent {
  private items = signal<CartItem[]>([]);
  private discount = signal(0);

  // βœ… All derived from signals β€” always in sync
  subtotal = computed(() =>
    this.items().reduce((sum, i) => sum + i.price * i.qty, 0)
  );
  discountAmt = computed(() => this.subtotal() * this.discount());
  total = computed(() => this.subtotal() - this.discountAmt());
  isEmpty = computed(() => this.items().length === 0);
  itemCount = computed(() =>
    this.items().reduce((n, i) => n + i.qty, 0)
  );
}

Enter fullscreen mode Exit fullscreen mode

◼️ Hybrid β€” Signals Local + NgRx Global (Architecture)


// checkout.component.ts β€” Hybrid architecture pattern
@Component({
  standalone: true
})
export class CheckoutComponent {
  // βœ… Global: complex order domain β†’ NgRx
  private orderStore = inject(OrderStore);
  selectedOrder = this.orderStore.selectedOrder; // Signal from store

  // βœ… Local: UI-only state β†’ Signals
  protected activeStep = signal(1);
  protected isReviewing = signal(false);

  // βœ… Bridge: derived from both worlds
  protected canConfirm = computed(() =>
    this.activeStep() === 3 && !!this.selectedOrder() && this.isReviewing()
  );
}

Enter fullscreen mode Exit fullscreen mode

Signals vs. Store: A Balanced Discussion
This isn't about picking a winner. It's about picking the right tool for the job.

Aspect Signals + Services NgRx Store
Learning curve Minimal Steep
Boilerplate Near zero High
DevTools Limited Excellent
Audit trails Manual Built-in
Team boundaries Convention Enforced
Cross-domain events Complex Native
Performance Granular Predictable

Use Signals when:

  • State is a component/feature local
  • Team understands reactive boundaries
  • No audit requirements
  • Simple to moderate complexity

Use NgRx when:

  • Multiple teams write to the same state
  • Compliance needs action logging.
  • Complex cross-domain workflows.
  • Time-travel debugging provides value.

Enterprise Reality Check
Large Angular systems have real needs that Signals alone cannot address at team-scale:

  • Predictable workflows across features
  • Ownership boundaries between teams
  • Debugging visibility across deployment environments
  • Scalable orchestration for complex event flows

NgRx addresses these organizational problems β€” not just technical ones.

The mistake is importing this complexity before the organization needs it.


What I Apply as an Architect
Start simple. Escalate when complexity demands it. Never reverse this order.

Default to signal() + computed() for component-local state

Use injectable services with Signals for feature boundaries

Add ComponentStore or SignalStore when patterns repeat

Reach for full NgRx only when organizational scale justifies it

The best Angular state management is the one you don't notice. If new developers ask about your store setup before understanding the business domain, you probably overengineered it.

Signals gave us a gift: the ability to start simple and add complexity only when boundaries prove insufficient.

Use that gift wisely.


Let's Discuss
What's the FIRST sign your Angular app actually needs a global state library?

Drop your answer below. Let's build an architecture checklist together.

Possible answers:

πŸ”„ Multiple teams writing to the same state
πŸ“Š Audit and compliance requirements
πŸ› Time-travel debugging needs
πŸ‘₯ Team coordination overhead

Further Reading
Angular Signals Guide
NgRx SignalStore Documentation

Found this useful? Follow for more Angular architecture insights.


πŸ“Œ More From Me
I share daily insights on web development, architecture, and frontend ecosystems.
Follow me here on Dev.to, and connect on LinkedIn for professional discussions.

🌐 Connect With Me
If you enjoyed this post and want more insights on scalable frontend systems, follow my work across platforms:

πŸ”— LinkedIn β€” Professional discussions, architecture breakdowns, and engineering insights.
πŸ“Έ Instagram β€” Visuals, carousels, and design‑driven posts under the Terminal Elite aesthetic.
🧠 Website β€” Articles, tutorials, and project showcases.
πŸŽ₯ YouTube β€” Deep‑dive videos and live coding sessions.

Top comments (0)