using System; using System.Collections.Generic; using System.Linq; using Unity.UIWidgets.async2; using Unity.UIWidgets.DevTools; using Unity.UIWidgets.DevTools.analytics; using Unity.UIWidgets.DevTools.config_specific.ide_theme; using Unity.UIWidgets.DevTools.inspector; using Unity.UIWidgets.foundation; using Unity.UIWidgets.material; using Unity.UIWidgets.painting; using Unity.UIWidgets.rendering; using Unity.UIWidgets.ui; using Unity.UIWidgets.widgets; using UnityEngine; using Screen = Unity.UIWidgets.DevTools.Screen; namespace Unity.UIWidgets.DevTools { public class appUtils { public static readonly bool showVmDeveloperMode = false; /// Whether this DevTools build is external. public bool isExternalBuild = true; public static List> defaultScreens { get { return new List> { new DevToolsScreen( new InspectorScreen(), createController: () => new InspectorSettingsController() ) }; } } } public class DevToolsApp : StatefulWidget { public DevToolsApp( List> screens, IdeTheme ideTheme, AnalyticsProvider analyticsProvider ) { this.screens = screens; this.ideTheme = ideTheme; this.analyticsProvider = analyticsProvider; } public readonly List> screens; public readonly IdeTheme ideTheme; public readonly AnalyticsProvider analyticsProvider; public override State createState() { return new DevToolsAppState(); } } // TODO(https://github.com/flutter/devtools/issues/1146): Introduce tests that // navigate the full app. public class DevToolsAppState : State { List _screens { get { List screensList = new List(); foreach (var s in widget.screens) { screensList.Add(s.screen); } return screensList; } } IdeTheme ideTheme { get { return widget.ideTheme; } } bool isDarkThemeEnabled { get { return _isDarkThemeEnabled; } } bool _isDarkThemeEnabled; bool vmDeveloperModeEnabled { get { return _vmDeveloperModeEnabled; } } bool _vmDeveloperModeEnabled; public override void initState() { base.initState(); // ga.setupDimensions(); // // serviceManager.isolateManager.onSelectedIsolateChanged.listen((_) => { // setState(() => { // _clearCachedRoutes(); // }); // }); // // _isDarkThemeEnabled = preferences.darkModeTheme.value; // preferences.darkModeTheme.addListener(() => { // setState(() => { // _isDarkThemeEnabled = preferences.darkModeTheme.value; // }); // }); // // _vmDeveloperModeEnabled = preferences.vmDeveloperModeEnabled.value; // preferences.vmDeveloperModeEnabled.addListener(() => { // setState(() => { // _vmDeveloperModeEnabled = preferences.vmDeveloperModeEnabled.value; // }); // }); } public void didUpdateWidget(DevToolsApp oldWidget) { base.didUpdateWidget(oldWidget); //_clearCachedRoutes(); } // Gets the page for a given page/path and args. Page _getPage(BuildContext context2, string page, Dictionary args) { // Provide the appropriate page route. if (pages.ContainsKey(page)) { Widget widget = pages[page]( context2, page, args ); D.assert(() => { widget = new _AlternateCheckedModeBanner( builder: (context) => pages[page]( context, page, args ) ); return true; }); return MaterialPage(child: widget); } // Return a page not found. return MaterialPage( child: DevToolsScaffold.withChild( key: Key.key("not-found"), child: CenteredMessage("'$page' not found."), ideTheme: ideTheme, analyticsProvider: widget.analyticsProvider ) ); } Widget _buildTabbedPage( BuildContext context, string page, Dictionary _params ) { var vmServiceUri = _params["uri"]; // Always return the landing screen if there's no VM service URI. if (vmServiceUri?.isEmpty() ?? true) { return DevToolsScaffold.withChild( key: Key.key("landing"), child: LandingScreenBody(), ideTheme: ideTheme, analyticsProvider: widget.analyticsProvider, actions: new List{ new OpenSettingsAction(), new OpenAboutAction() } ); } // TODO(dantup): We should be able simplify this a little, removing params['page'] // and only supporting /inspector (etc.) instead of also &page=inspector if // all IDEs switch over to those URLs. if (page?.isEmpty() ?? true) { page = _params["page"]; } var embed = _params["embed"] == "true"; var hide = _params["hide"]?.Split(','); return Initializer( url: vmServiceUri, allowConnectionScreenOnDisconnect: !embed, builder: (_) => { return new ValueListenableBuilder( valueListenable: preferences.vmDeveloperModeEnabled, builder: (_, __, ___) => { var tabs = _visibleScreens() .where((p) => embed && page != null ? p.screenId == page : true) .where((p) => !hide.Contains(p.screenId)) .toList(); if (tabs.isEmpty) { return DevToolsScaffold.withChild( child: CenteredMessage( $"The \"{page}\" screen is not available for this application."), ideTheme: ideTheme, analyticsProvider: widget.analyticsProvider ); } List widgets = new List(); if (serviceManager.connectedApp.isFlutterAppNow) { widgets.Add(HotReloadButton()); widgets.Add(HotRestartButton()); } widgets.Add(new OpenSettingsAction()); widgets.Add(new OpenAboutAction()); return _providedControllers( child: DevToolsScaffold( embed: embed, ideTheme: ideTheme, page: page, tabs: tabs, analyticsProvider: widget.analyticsProvider, actions: widgets ) ); } ); } ); } // The pages that the app exposes. Dictionary pages { get { if (_routes == null) { return null; } return _routes ?? { homePageId: _buildTabbedPage, foreach (Screen screen in widget.screens) screen.screen.screenId: _buildTabbedPage, snapshotPageId: (_, __, args) => { final snapshotArgs = SnapshotArguments.fromArgs(args); return DevToolsScaffold.withChild( key: new UniqueKey(), analyticsProvider: widget.analyticsProvider, child: _providedControllers( offline: true, child: SnapshotScreenBody(snapshotArgs, _screens), ), ideTheme: ideTheme, ); }, appSizePageId: (_, __, ___) =>{ return DevToolsScaffold.withChild( key: Key.key("appsize"), analyticsProvider: widget.analyticsProvider, child: _providedControllers( child: AppSizeBody() ), ideTheme: ideTheme, actions: [ OpenSettingsAction(), OpenAboutAction(), ], ); }, }; } } Dictionary _routes; void _clearCachedRoutes() { _routes = null; } List _visibleScreens() => _screens.where(shouldShowScreen).toList(); Widget _providedControllers({@required Widget child, bool offline = false}) { var _providers = widget.screens .where((s) => s.createController != null && (offline ? s.supportsOffline : true)) .map((s) => s.controllerProvider) .toList(); return MultiProvider( providers: _providers, child: child, ); } public override Widget build(BuildContext context) { return new MaterialApp( debugShowCheckedModeBanner: false, theme: ThemeData.light(),//themeFor(isDarkTheme: isDarkThemeEnabled, ideTheme: ideTheme), builder: (context2, child) => null//Notifications(child: child), // routerDelegate: DevToolsRouterDelegate(_getPage), // routeInformationParser: DevToolsRouteInformationParser() ); } } public class DevToolsScreen { public delegate C CreateController(); public DevToolsScreen( Screen screen, CreateController createController = null, bool supportsOffline = false ) { this.screen = screen; this.createController = createController; this.supportsOffline = supportsOffline; } public readonly Screen screen; public readonly CreateController createController; public readonly bool supportsOffline; Provider controllerProvider { get { D.assert(createController != null); return Provider(create: (_) => createController()); } } } public delegate Widget UrlParametersBuilder( BuildContext buildContext, string s, Dictionary dictionary ); public class _AlternateCheckedModeBanner : StatelessWidget { public _AlternateCheckedModeBanner(Key key = null, WidgetBuilder builder = null) : base(key: key) { this.builder = builder; } public readonly WidgetBuilder builder; public override Widget build(BuildContext context) { return new Banner( message: "DEBUG", textDirection: TextDirection.ltr, location: BannerLocation.topStart, child: new Builder( builder: builder ) ); } } /*public class OpenAboutAction : StatelessWidget { public override Widget build(BuildContext context) { return DevToolsTooltip( tooltip: "About DevTools", child: new InkWell( onTap: () => { unawaited(showDialog( context: context, builder: (context2) => new DevToolsAboutDialog() )); }, child: new Container( width: DevToolsScaffold.actionWidgetSize, height: DevToolsScaffold.actionWidgetSize, alignment: Alignment.center, child: new Icon( Icons.help_outline, size: actionsIconSize ) ) ) ); } } public class OpenSettingsAction : StatelessWidget { public override Widget build(BuildContext context) { return DevToolsTooltip( tooltip: "Settings", child: new InkWell( onTap: () => { unawaited(showDialog( context: context, builder: (context2) => new SettingsDialog() )); }, child: new Container( width: DevToolsScaffold.actionWidgetSize, height: DevToolsScaffold.actionWidgetSize, alignment: Alignment.center, child: new Icon( Icons.settings, size: actionsIconSize ) ) ) ); } } public class DevToolsAboutDialog : StatelessWidget { public override Widget build(BuildContext context) { var theme = Theme.of(context); List widgets = new List(); widgets.Add(new SizedBox(height: defaultSpacing)); List temp = dialogSubHeader(theme, "Feedback"); foreach (var widget in temp) { widgets.Add(widget); } widgets.Add(new Wrap( children: new List{ new Text("Encountered an issue? Let us know at "), _createFeedbackLink(context), new Text(".") } )); return DevToolsDialog( title: dialogTitleText(theme, "About DevTools"), content: new Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: widgets ), actions: [ DialogCloseButton(), ], ); } Widget _aboutDevTools(BuildContext context) { return const SelectableText('DevTools version ${devtools.version}'); } Widget _createFeedbackLink(BuildContext context) { const urlPath = 'github.com/flutter/devtools/issues'; final colorScheme = Theme.of(context).colorScheme; return InkWell( onTap: () async { ga.select(devToolsMain, feedback); const reportIssuesUrl = 'https://$urlPath'; await launchUrl(reportIssuesUrl, context); }, child: Text(urlPath, style: linkTextStyle(colorScheme)), ); } } TODO(kenz): merge the checkbox functionality here with [NotifierCheckbox] class SettingsDialog extends StatelessWidget { @override Widget build(BuildContext context) { return DevToolsDialog( title: dialogTitleText(Theme.of(context), 'Settings'), content: Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: [ _buildOption( label: const Text('Use a dark theme'), listenable: preferences.darkModeTheme, toggle: preferences.toggleDarkModeTheme, ), if (isExternalBuild && isDevToolsServerAvailable) _buildOption( label: const Text('Enable analytics'), listenable: ga.gaEnabledNotifier, toggle: ga.setAnalyticsEnabled, ), _buildOption( label: const Text('Enable VM developer mode'), listenable: preferences.vmDeveloperModeEnabled, toggle: preferences.toggleVmDeveloperMode, ), ], ), actions: [ DialogCloseButton(), ], ); } Widget _buildOption({ Text label, ValueListenable listenable, Function(bool) toggle, }) { return InkWell( onTap: () => toggle(!listenable.value), child: Row( children: [ ValueListenableBuilder( valueListenable: listenable, builder: (context, value, _) { return Checkbox( value: value, onChanged: toggle, ); }, ), label, ], ), ); }*/ }