Unverified Commit ee08c227 authored by Per Classon's avatar Per Classon Committed by GitHub

Add decoration parameter to DataTable and DataTableTheme (#66640)

* Add decoration parameter to DataTable and DataTableTheme
parent 0d8696f6
...@@ -10,6 +10,7 @@ import 'package:flutter/widgets.dart'; ...@@ -10,6 +10,7 @@ import 'package:flutter/widgets.dart';
import 'checkbox.dart'; import 'checkbox.dart';
import 'constants.dart'; import 'constants.dart';
import 'data_table_theme.dart';
import 'debug.dart'; import 'debug.dart';
import 'divider.dart'; import 'divider.dart';
import 'dropdown.dart'; import 'dropdown.dart';
...@@ -411,6 +412,7 @@ class DataTable extends StatelessWidget { ...@@ -411,6 +412,7 @@ class DataTable extends StatelessWidget {
this.sortColumnIndex, this.sortColumnIndex,
this.sortAscending = true, this.sortAscending = true,
this.onSelectAll, this.onSelectAll,
this.decoration,
this.dataRowColor, this.dataRowColor,
this.dataRowHeight, this.dataRowHeight,
this.dataTextStyle, this.dataTextStyle,
...@@ -473,6 +475,14 @@ class DataTable extends StatelessWidget { ...@@ -473,6 +475,14 @@ class DataTable extends StatelessWidget {
/// row is selectable. /// row is selectable.
final ValueSetter<bool?>? onSelectAll; final ValueSetter<bool?>? onSelectAll;
/// {@template flutter.material.dataTable.decoration}
/// The background and border decoration for the table.
/// {@endtemplate}
///
/// If null, [DataTableThemeData.decoration] is used. By default there is no
/// decoration.
final Decoration? decoration;
/// {@template flutter.material.dataTable.dataRowColor} /// {@template flutter.material.dataTable.dataRowColor}
/// The background color for the data rows. /// The background color for the data rows.
/// ///
...@@ -484,9 +494,10 @@ class DataTable extends StatelessWidget { ...@@ -484,9 +494,10 @@ class DataTable extends StatelessWidget {
/// color. /// color.
/// {@endtemplate} /// {@endtemplate}
/// ///
/// By default, the background color is transparent unless selected. Selected /// If null, [DataTableThemeData.dataRowColor] is used. By default, the
/// rows have a grey translucent color. To set a different color for /// background color is transparent unless selected. Selected rows have a grey
/// individual rows, see [DataRow.color]. /// translucent color. To set a different color for individual rows, see
/// [DataRow.color].
/// ///
/// {@template flutter.material.dataTable.dataRowColorCode} /// {@template flutter.material.dataTable.dataRowColorCode}
/// ```dart /// ```dart
...@@ -511,15 +522,17 @@ class DataTable extends StatelessWidget { ...@@ -511,15 +522,17 @@ class DataTable extends StatelessWidget {
/// The height of each row (excluding the row that contains column headings). /// The height of each row (excluding the row that contains column headings).
/// {@endtemplate} /// {@endtemplate}
/// ///
/// This value defaults to [kMinInteractiveDimension] to adhere to the Material /// If null, [DataTableThemeData.dataRowHeight] is used. This value defaults
/// Design specifications. /// to [kMinInteractiveDimension] to adhere to the Material Design
/// specifications.
final double? dataRowHeight; final double? dataRowHeight;
/// {@template flutter.material.dataTable.dataTextStyle} /// {@template flutter.material.dataTable.dataTextStyle}
/// The text style for data rows. /// The text style for data rows.
/// {@endtemplate} /// {@endtemplate}
/// ///
/// By default, the text style is [TextTheme.bodyText2]. /// If null, [DataTableThemeData.dataTextStyle] is used. By default, the text
/// style is [TextTheme.bodyText2].
final TextStyle? dataTextStyle; final TextStyle? dataTextStyle;
/// {@template flutter.material.dataTable.headingRowColor} /// {@template flutter.material.dataTable.headingRowColor}
...@@ -530,7 +543,11 @@ class DataTable extends StatelessWidget { ...@@ -530,7 +543,11 @@ class DataTable extends StatelessWidget {
/// sorted. The color is painted as an overlay to the row. To make sure that /// sorted. The color is painted as an overlay to the row. To make sure that
/// the row's [InkWell] is visible (when pressed, hovered and focused), it is /// the row's [InkWell] is visible (when pressed, hovered and focused), it is
/// recommended to use a translucent color. /// recommended to use a translucent color.
/// {@endtemplate}
/// ///
/// If null, [DataTableThemeData.headingRowColor] is used.
///
/// {@template flutter.material.dataTable.headingRowColorCode}
/// ```dart /// ```dart
/// DataTable( /// DataTable(
/// headingRowColor: MaterialStateProperty.resolveWith<Color>((Set<MaterialState> states) { /// headingRowColor: MaterialStateProperty.resolveWith<Color>((Set<MaterialState> states) {
...@@ -553,14 +570,16 @@ class DataTable extends StatelessWidget { ...@@ -553,14 +570,16 @@ class DataTable extends StatelessWidget {
/// The height of the heading row. /// The height of the heading row.
/// {@endtemplate} /// {@endtemplate}
/// ///
/// This value defaults to 56.0 to adhere to the Material Design specifications. /// If null, [DataTableThemeData.headingRowHeight] is used. This value
/// defaults to 56.0 to adhere to the Material Design specifications.
final double? headingRowHeight; final double? headingRowHeight;
/// {@template flutter.material.dataTable.headingTextStyle} /// {@template flutter.material.dataTable.headingTextStyle}
/// The text style for the heading row. /// The text style for the heading row.
/// {@endtemplate} /// {@endtemplate}
/// ///
/// By default, the text style is [TextTheme.subtitle2]. /// If null, [DataTableThemeData.headingTextStyle] is used. By default, the
/// text style is [TextTheme.subtitle2].
final TextStyle? headingTextStyle; final TextStyle? headingTextStyle;
/// {@template flutter.material.dataTable.horizontalMargin} /// {@template flutter.material.dataTable.horizontalMargin}
...@@ -571,14 +590,16 @@ class DataTable extends StatelessWidget { ...@@ -571,14 +590,16 @@ class DataTable extends StatelessWidget {
/// the content in the first data column. /// the content in the first data column.
/// {@endtemplate} /// {@endtemplate}
/// ///
/// This value defaults to 24.0 to adhere to the Material Design specifications. /// If null, [DataTableThemeData.horizontalMargin] is used. This value
/// defaults to 24.0 to adhere to the Material Design specifications.
final double? horizontalMargin; final double? horizontalMargin;
/// {@template flutter.material.dataTable.columnSpacing} /// {@template flutter.material.dataTable.columnSpacing}
/// The horizontal margin between the contents of each data column. /// The horizontal margin between the contents of each data column.
/// {@endtemplate} /// {@endtemplate}
/// ///
/// This value defaults to 56.0 to adhere to the Material Design specifications. /// If null, [DataTableThemeData.columnSpacing] is used. This value defaults
/// to 56.0 to adhere to the Material Design specifications.
final double? columnSpacing; final double? columnSpacing;
/// {@template flutter.material.dataTable.showCheckboxColumn} /// {@template flutter.material.dataTable.showCheckboxColumn}
...@@ -603,13 +624,15 @@ class DataTable extends StatelessWidget { ...@@ -603,13 +624,15 @@ class DataTable extends StatelessWidget {
/// ///
/// Must be greater than or equal to zero. /// Must be greater than or equal to zero.
/// {@endtemplate} /// {@endtemplate}
/// This value defaults to 1.0. ///
/// If null, [DataTableThemeData.dividerThickness] is used. This value
/// defaults to 1.0.
final double? dividerThickness; final double? dividerThickness;
/// Whether a border at the bottom of the table is displayed. /// Whether a border at the bottom of the table is displayed.
/// ///
/// By default, a border is not shown at the bottom to allow for a border /// By default, a border is not shown at the bottom to allow for a border
/// around the table. /// around the table defined by [decoration].
final bool showBottomBorder; final bool showBottomBorder;
// Set by the constructor to the index of the only Column that is // Set by the constructor to the index of the only Column that is
...@@ -975,9 +998,15 @@ class DataTable extends StatelessWidget { ...@@ -975,9 +998,15 @@ class DataTable extends StatelessWidget {
displayColumnIndex += 1; displayColumnIndex += 1;
} }
return Table( return Container(
columnWidths: tableColumns.asMap(), decoration: decoration ?? theme.dataTableTheme.decoration,
children: tableRows, child: Material(
type: MaterialType.transparency,
child: Table(
columnWidths: tableColumns.asMap(),
children: tableRows,
),
),
); );
} }
} }
......
...@@ -35,6 +35,7 @@ import 'theme.dart'; ...@@ -35,6 +35,7 @@ import 'theme.dart';
class DataTableThemeData with Diagnosticable { class DataTableThemeData with Diagnosticable {
/// Creates a theme that can be used for [ThemeData.dataTableTheme]. /// Creates a theme that can be used for [ThemeData.dataTableTheme].
const DataTableThemeData({ const DataTableThemeData({
this.decoration,
this.dataRowColor, this.dataRowColor,
this.dataRowHeight, this.dataRowHeight,
this.dataTextStyle, this.dataTextStyle,
...@@ -46,6 +47,9 @@ class DataTableThemeData with Diagnosticable { ...@@ -46,6 +47,9 @@ class DataTableThemeData with Diagnosticable {
this.dividerThickness, this.dividerThickness,
}); });
/// {@macro flutter.material.dataTable.decoration}
final Decoration? decoration;
/// {@macro flutter.material.dataTable.dataRowColor} /// {@macro flutter.material.dataTable.dataRowColor}
/// {@macro flutter.material.dataTable.dataRowColorCode} /// {@macro flutter.material.dataTable.dataRowColorCode}
final MaterialStateProperty<Color?>? dataRowColor; final MaterialStateProperty<Color?>? dataRowColor;
...@@ -57,6 +61,7 @@ class DataTableThemeData with Diagnosticable { ...@@ -57,6 +61,7 @@ class DataTableThemeData with Diagnosticable {
final TextStyle? dataTextStyle; final TextStyle? dataTextStyle;
/// {@macro flutter.material.dataTable.headingRowColor} /// {@macro flutter.material.dataTable.headingRowColor}
/// {@macro flutter.material.dataTable.headingRowColorCode}
final MaterialStateProperty<Color?>? headingRowColor; final MaterialStateProperty<Color?>? headingRowColor;
/// {@macro flutter.material.dataTable.headingRowHeight} /// {@macro flutter.material.dataTable.headingRowHeight}
...@@ -77,6 +82,7 @@ class DataTableThemeData with Diagnosticable { ...@@ -77,6 +82,7 @@ class DataTableThemeData with Diagnosticable {
/// Creates a copy of this object but with the given fields replaced with the /// Creates a copy of this object but with the given fields replaced with the
/// new values. /// new values.
DataTableThemeData copyWith({ DataTableThemeData copyWith({
Decoration? decoration,
MaterialStateProperty<Color?>? dataRowColor, MaterialStateProperty<Color?>? dataRowColor,
double? dataRowHeight, double? dataRowHeight,
TextStyle? dataTextStyle, TextStyle? dataTextStyle,
...@@ -88,6 +94,7 @@ class DataTableThemeData with Diagnosticable { ...@@ -88,6 +94,7 @@ class DataTableThemeData with Diagnosticable {
double? dividerThickness, double? dividerThickness,
}) { }) {
return DataTableThemeData( return DataTableThemeData(
decoration: decoration ?? this.decoration,
dataRowColor: dataRowColor ?? this.dataRowColor, dataRowColor: dataRowColor ?? this.dataRowColor,
dataRowHeight: dataRowHeight ?? this.dataRowHeight, dataRowHeight: dataRowHeight ?? this.dataRowHeight,
dataTextStyle: dataTextStyle ?? this.dataTextStyle, dataTextStyle: dataTextStyle ?? this.dataTextStyle,
...@@ -108,6 +115,7 @@ class DataTableThemeData with Diagnosticable { ...@@ -108,6 +115,7 @@ class DataTableThemeData with Diagnosticable {
static DataTableThemeData lerp(DataTableThemeData a, DataTableThemeData b, double t) { static DataTableThemeData lerp(DataTableThemeData a, DataTableThemeData b, double t) {
assert(t != null); assert(t != null);
return DataTableThemeData( return DataTableThemeData(
decoration: Decoration.lerp(a.decoration, b.decoration, t),
dataRowColor: _lerpProperties<Color?>(a.dataRowColor, b.dataRowColor, t, Color.lerp), dataRowColor: _lerpProperties<Color?>(a.dataRowColor, b.dataRowColor, t, Color.lerp),
dataRowHeight: lerpDouble(a.dataRowHeight, b.dataRowHeight, t), dataRowHeight: lerpDouble(a.dataRowHeight, b.dataRowHeight, t),
dataTextStyle: TextStyle.lerp(a.dataTextStyle, b.dataTextStyle, t), dataTextStyle: TextStyle.lerp(a.dataTextStyle, b.dataTextStyle, t),
...@@ -123,6 +131,7 @@ class DataTableThemeData with Diagnosticable { ...@@ -123,6 +131,7 @@ class DataTableThemeData with Diagnosticable {
@override @override
int get hashCode { int get hashCode {
return hashValues( return hashValues(
decoration,
dataRowColor, dataRowColor,
dataRowHeight, dataRowHeight,
dataTextStyle, dataTextStyle,
...@@ -142,6 +151,7 @@ class DataTableThemeData with Diagnosticable { ...@@ -142,6 +151,7 @@ class DataTableThemeData with Diagnosticable {
if (other.runtimeType != runtimeType) if (other.runtimeType != runtimeType)
return false; return false;
return other is DataTableThemeData return other is DataTableThemeData
&& other.decoration == decoration
&& other.dataRowColor == dataRowColor && other.dataRowColor == dataRowColor
&& other.dataRowHeight == dataRowHeight && other.dataRowHeight == dataRowHeight
&& other.dataTextStyle == dataTextStyle && other.dataTextStyle == dataTextStyle
...@@ -156,6 +166,7 @@ class DataTableThemeData with Diagnosticable { ...@@ -156,6 +166,7 @@ class DataTableThemeData with Diagnosticable {
@override @override
void debugFillProperties(DiagnosticPropertiesBuilder properties) { void debugFillProperties(DiagnosticPropertiesBuilder properties) {
super.debugFillProperties(properties); super.debugFillProperties(properties);
properties.add(DiagnosticsProperty<Decoration>('decoration', decoration, defaultValue: null));
properties.add(DiagnosticsProperty<MaterialStateProperty<Color?>>('dataRowColor', dataRowColor, defaultValue: null)); properties.add(DiagnosticsProperty<MaterialStateProperty<Color?>>('dataRowColor', dataRowColor, defaultValue: null));
properties.add(DoubleProperty('dataRowHeight', dataRowHeight, defaultValue: null)); properties.add(DoubleProperty('dataRowHeight', dataRowHeight, defaultValue: null));
properties.add(DiagnosticsProperty<TextStyle>('dataTextStyle', dataTextStyle, defaultValue: null)); properties.add(DiagnosticsProperty<TextStyle>('dataTextStyle', dataTextStyle, defaultValue: null));
......
...@@ -471,41 +471,35 @@ void main() { ...@@ -471,41 +471,35 @@ void main() {
), ),
), ),
)); ));
expect(tester.renderObject<RenderBox>(
find.widgetWithText(Container, 'Name') // The finder matches with the Container of the cell content, as well as the
).size.height, 56.0); // This is the header row height // Container wrapping the whole table. The first one is used to test row
expect(tester.renderObject<RenderBox>( // heights.
find.widgetWithText(Container, 'Frozen yogurt') Finder findFirstContainerFor(String text) => find.widgetWithText(Container, text).first;
).size.height, 48.0); // This is the data row height
expect(tester.getSize(findFirstContainerFor('Name')).height, 56.0);
expect(tester.getSize(findFirstContainerFor('Frozen yogurt')).height, 48.0);
// CUSTOM VALUES // CUSTOM VALUES
await tester.pumpWidget(MaterialApp( await tester.pumpWidget(MaterialApp(
home: Material(child: buildCustomTable(headingRowHeight: 48.0)), home: Material(child: buildCustomTable(headingRowHeight: 48.0)),
)); ));
expect(tester.renderObject<RenderBox>( expect(tester.getSize(findFirstContainerFor('Name')).height, 48.0);
find.widgetWithText(Container, 'Name')
).size.height, 48.0);
await tester.pumpWidget(MaterialApp( await tester.pumpWidget(MaterialApp(
home: Material(child: buildCustomTable(headingRowHeight: 64.0)), home: Material(child: buildCustomTable(headingRowHeight: 64.0)),
)); ));
expect(tester.renderObject<RenderBox>( expect(tester.getSize(findFirstContainerFor('Name')).height, 64.0);
find.widgetWithText(Container, 'Name')
).size.height, 64.0);
await tester.pumpWidget(MaterialApp( await tester.pumpWidget(MaterialApp(
home: Material(child: buildCustomTable(dataRowHeight: 30.0)), home: Material(child: buildCustomTable(dataRowHeight: 30.0)),
)); ));
expect(tester.renderObject<RenderBox>( expect(tester.getSize(findFirstContainerFor('Frozen yogurt')).height, 30.0);
find.widgetWithText(Container, 'Frozen yogurt')
).size.height, 30.0);
await tester.pumpWidget(MaterialApp( await tester.pumpWidget(MaterialApp(
home: Material(child: buildCustomTable(dataRowHeight: 56.0)), home: Material(child: buildCustomTable(dataRowHeight: 56.0)),
)); ));
expect(tester.renderObject<RenderBox>( expect(tester.getSize(findFirstContainerFor('Frozen yogurt')).height, 56.0);
find.widgetWithText(Container, 'Frozen yogurt')
).size.height, 56.0);
}); });
testWidgets('DataTable custom horizontal padding - checkbox', (WidgetTester tester) async { testWidgets('DataTable custom horizontal padding - checkbox', (WidgetTester tester) async {
...@@ -1303,4 +1297,56 @@ void main() { ...@@ -1303,4 +1297,56 @@ void main() {
expect(tester.takeException(), isNull); expect(tester.takeException(), isNull);
}); });
testWidgets('DataTable renders with border and background decoration', (WidgetTester tester) async {
const double width = 800;
const double height = 600;
const double borderHorizontal = 5.0;
const double borderVertical = 10.0;
const Color borderColor = Color(0xff2196f3);
const Color backgroundColor = Color(0xfff5f5f5);
await tester.pumpWidget(
MaterialApp(
home: DataTable(
decoration: const BoxDecoration(
color: backgroundColor,
border: Border.symmetric(
vertical: BorderSide(width: borderVertical, color: borderColor),
horizontal: BorderSide(width: borderHorizontal, color: borderColor),
),
),
columns: const <DataColumn>[
DataColumn(label: Text('Col1')),
],
rows: const <DataRow>[
DataRow(cells: <DataCell>[DataCell(Text('1'))]),
],
),
),
);
expect(
find.ancestor(of: find.byType(Table), matching: find.byType(Container)),
paints..rect(
rect: const Rect.fromLTRB(0.0, 0.0, width, height),
color: backgroundColor,
),
);
expect(
find.ancestor(of: find.byType(Table), matching: find.byType(Container)),
paints
..path(color: borderColor)
..path(color: borderColor)
..path(color: borderColor)
..path(color: borderColor),
);
expect(
tester.getTopLeft(find.byType(Table)),
const Offset(borderVertical, borderHorizontal),
);
expect(
tester.getBottomRight(find.byType(Table)),
const Offset(width - borderVertical, height - borderHorizontal),
);
});
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment