debug_overlay 1.0.0 copy "debug_overlay: ^1.0.0" to clipboard
debug_overlay: ^1.0.0 copied to clipboard

View debug infos and change settings via a central overlay for your app.

example/lib/main.dart

// ignore_for_file: avoid_redundant_argument_values
// ignore_for_file: depend_on_referenced_packages

import 'dart:math';

import 'package:black_hole_flutter/black_hole_flutter.dart';
import 'package:collection/collection.dart';
import 'package:debug_overlay/debug_overlay.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_localizations/flutter_localizations.dart';

final logs = LogCollection(
  // Default values:
  maximumSize: 500,
  onlyStoreLogsInDebugMode: true,
);
final mediaOverrideState = ValueNotifier(MediaOverrideState());

final supportedLocales =
    kMaterialSupportedLanguages.sortedBy((it) => it).map(Locale.new).toList();

void main() {
  if (kDebugMode) {
    DebugOverlay.prependHelper(
      MediaOverrideDebugHelper(
        mediaOverrideState,
        // To support overriding locales, this value must be set and should
        // contain the same locales as passed to [MaterialApp.supportedLocales],
        // [CupertinoApp.supportedLocales] or [WidgetsApp.supportedLocales].
        supportedLocales: supportedLocales,
      ),
    );
    DebugOverlay.appendHelper(LogsDebugHelper(logs));
  }

  runApp(const MyApp());
}

class MyApp extends StatelessWidget {
  const MyApp({super.key});

  @override
  Widget build(BuildContext context) {
    // To use the [MediaOverrideDebugHelper], wrap your app in a
    // [ValueListenableBuilder] to access the overridden values:
    return ValueListenableBuilder(
      valueListenable: mediaOverrideState,
      builder: (context, overrideState, child) {
        return MaterialApp(
          title: '🐛 debug_overlay example',

          // You can access overridden values via [overrideState]:
          themeMode: overrideState.themeMode,
          locale: overrideState.locale,

          // This creates the actual [DebugOverlay] (only in debug mode; not in
          // profile oder release mode).
          builder: DebugOverlay.builder(),

          // And the usual customization:
          supportedLocales: supportedLocales,
          theme: ThemeData.light(),
          darkTheme: ThemeData.dark(),
          localizationsDelegates: GlobalMaterialLocalizations.delegates,
          home: const HomePage(),
        );
      },
    );
  }
}

class HomePage extends StatelessWidget {
  const HomePage({super.key});

  static final _random = Random();

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: const Text('🐛 debug_overlay example')),
      body: SizedBox.expand(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            Text('Brightness: ${context.theme.brightness}'),
            const SizedBox(height: 4),
            Text('Locale: ${context.locale}'),
            const SizedBox(height: 16),
            TextButton(
              onPressed: _createLog,
              child: const Text('Add log'),
            ),
            const SizedBox(height: 16),
            const ElevatedButton(
              onPressed: DebugOverlay.show,
              child: Text('Show Debug Overlay'),
            ),
          ],
        ),
      ),
    );
  }

  /// Creates a single log randomly and appends it to [logs].
  void _createLog() {
    // The last value is `off`, which should only be used for filtering.
    final level = DiagnosticLevel
        .values[_random.nextInt(DiagnosticLevel.values.length - 1)];
    final hasError = _random.nextDouble() >= 0.8;
    final log = Log(
      level: level,
      message: 'Log entry #${logs.logs.length + 1}',
      error: hasError ? ArgumentError.value('bar', 'foo') : null,
      stackTrace: hasError ? StackTrace.current : null,
    );

    // If you use a custom logging solution, you have to also append logs to
    // `debug_overlay`'s LogCollection.
    logs.add(log);
  }
}