您最多选择25个主题 主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
 
 
 
 
 
 

548 行
14 KiB

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<DevToolsScreen<object>> defaultScreens {
get
{
return new List<DevToolsScreen<object>>
{
new DevToolsScreen<object>(
new InspectorScreen(),
createController: () => new InspectorSettingsController()
)
};
}
}
}
public class DevToolsApp : StatefulWidget {
public DevToolsApp(
List<DevToolsScreen<object>> screens,
IdeTheme ideTheme,
AnalyticsProvider analyticsProvider
)
{
this.screens = screens;
this.ideTheme = ideTheme;
this.analyticsProvider = analyticsProvider;
}
public readonly List<DevToolsScreen<object>> 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<DevToolsApp> {
List<Screen> _screens
{
get
{
List<Screen> screensList = new List<Screen>();
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<string, string> 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<string, string> _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<Widget>{
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<bool>(
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<Widget> widgets = new List<Widget>();
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<string, UrlParametersBuilder> 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<string, UrlParametersBuilder> _routes;
void _clearCachedRoutes() {
_routes = null;
}
List<Screen> _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<C> {
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<C> controllerProvider {
get
{
D.assert(createController != null);
return Provider<C>(create: (_) => createController());
}
}
}
public delegate Widget UrlParametersBuilder(
BuildContext buildContext,
string s,
Dictionary<string, string> 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<Widget> widgets = new List<Widget>();
widgets.Add(new SizedBox(height: defaultSpacing));
List<Widget> temp = dialogSubHeader(theme, "Feedback");
foreach (var widget in temp)
{
widgets.Add(widget);
}
widgets.Add(new Wrap(
children: new List<Widget>{
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<bool> listenable,
Function(bool) toggle,
}) {
return InkWell(
onTap: () => toggle(!listenable.value),
child: Row(
children: [
ValueListenableBuilder<bool>(
valueListenable: listenable,
builder: (context, value, _) {
return Checkbox(
value: value,
onChanged: toggle,
);
},
),
label,
],
),
);
}*/
}