demo.dart 4.29 KB
Newer Older
1 2 3 4 5 6
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

import 'package:flutter/material.dart';

7
import 'example_code_parser.dart';
8
import 'syntax_highlighter.dart';
9

10 11
class ComponentDemoTabData {
  ComponentDemoTabData({
12
    this.widget,
13
    this.exampleCodeTag,
14
    this.description,
15
    this.tabName
16 17 18
  });

  final Widget widget;
19
  final String exampleCodeTag;
20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43
  final String description;
  final String tabName;

  @override
  bool operator==(Object other) {
    if (other.runtimeType != runtimeType)
      return false;
    ComponentDemoTabData typedOther = other;
    return typedOther.tabName == tabName && typedOther.description == description;
  }

  @override
  int get hashCode => hashValues(tabName.hashCode, description.hashCode);
}

class TabbedComponentDemoScaffold extends StatelessWidget {
  TabbedComponentDemoScaffold({
    this.title,
    this.demos
  });

  final List<ComponentDemoTabData> demos;
  final String title;

44
  void _showExampleCode(BuildContext context) {
Hans Muller's avatar
Hans Muller committed
45
    String tag = demos[DefaultTabController.of(context).index].exampleCodeTag;
46 47 48 49 50 51 52
    if (tag != null) {
      Navigator.push(context, new MaterialPageRoute<FullScreenCodeDialog>(
        builder: (BuildContext context) => new FullScreenCodeDialog(exampleCodeTag: tag)
      ));
    }
  }

53 54
  @override
  Widget build(BuildContext context) {
Hans Muller's avatar
Hans Muller committed
55 56
    return new DefaultTabController(
      length: demos.length,
57 58 59
      child: new Scaffold(
        appBar: new AppBar(
          title: new Text(title),
60 61 62 63 64 65
          actions: <Widget>[
            new Builder(
              builder: (BuildContext context) {
                return new IconButton(
                  icon: new Icon(Icons.description),
                  tooltip: 'Show example code',
Hans Muller's avatar
Hans Muller committed
66 67 68
                  onPressed: () {
                    _showExampleCode(context);
                  },
69
                );
Hans Muller's avatar
Hans Muller committed
70 71
              },
            ),
72
          ],
Hans Muller's avatar
Hans Muller committed
73
          bottom: new TabBar(
74
            isScrollable: true,
Hans Muller's avatar
Hans Muller committed
75 76
            tabs: demos.map((ComponentDemoTabData data) => new Tab(text: data.tabName)).toList(),
          ),
77
        ),
Hans Muller's avatar
Hans Muller committed
78
        body: new TabBarView(
79 80 81 82 83
          children: demos.map((ComponentDemoTabData demo) {
            return new Column(
              children: <Widget>[
                new Padding(
                  padding: const EdgeInsets.all(16.0),
84 85 86
                  child: new Text(demo.description,
                    style: Theme.of(context).textTheme.subhead
                  )
87
                ),
88
                new Expanded(child: demo.widget)
Hans Muller's avatar
Hans Muller committed
89
              ],
90
            );
Hans Muller's avatar
Hans Muller committed
91 92 93
          }).toList(),
        ),
      ),
94 95 96 97
    );
  }
}

98 99 100 101 102 103 104 105 106 107
class FullScreenCodeDialog extends StatefulWidget {
  FullScreenCodeDialog({ this.exampleCodeTag });

  final String exampleCodeTag;

  @override
  FullScreenCodeDialogState createState() => new FullScreenCodeDialogState();
}

class FullScreenCodeDialogState extends State<FullScreenCodeDialog> {
108

109 110 111
  String _exampleCode;

  @override
112
  void dependenciesChanged() {
113 114 115 116 117 118
    getExampleCode(config.exampleCodeTag, DefaultAssetBundle.of(context)).then<Null>((String code) {
      if (mounted) {
        setState(() {
          _exampleCode = code;
        });
      }
119
    });
120
    super.dependenciesChanged();
121
  }
122 123 124

  @override
  Widget build(BuildContext context) {
125 126 127 128
    final SyntaxHighlighterStyle style = Theme.of(context).brightness == Brightness.dark
      ? SyntaxHighlighterStyle.darkThemeStyle()
      : SyntaxHighlighterStyle.lightThemeStyle();

129 130 131 132 133 134
    Widget body;
    if (_exampleCode == null) {
      body = new Center(
        child: new CircularProgressIndicator()
      );
    } else {
135
      body = new SingleChildScrollView(
136 137
        child: new Padding(
          padding: new EdgeInsets.all(16.0),
138 139 140 141 142 143 144 145
          child: new RichText(
            text: new TextSpan(
              style: new TextStyle(fontFamily: 'monospace', fontSize: 10.0),
              children: <TextSpan>[
                new DartSyntaxHighlighter(style).format(_exampleCode)
              ]
            )
          )
146 147 148 149
        )
      );
    }

150 151 152
    return new Scaffold(
      appBar: new AppBar(
        leading: new IconButton(
Ian Hickson's avatar
Ian Hickson committed
153
          icon: new Icon(Icons.clear),
154 155
          onPressed: () { Navigator.pop(context); }
        ),
156
        title: new Text('Example code')
157
      ),
158
      body: body
159 160 161
    );
  }
}