Skip to content

Commit

Permalink
feat(neon): add AdaptiveListTile
Browse files Browse the repository at this point in the history
Signed-off-by: Nikolas Rimikis <[email protected]>
  • Loading branch information
Leptopoda committed Nov 2, 2023
1 parent b5c1ef2 commit e888216
Show file tree
Hide file tree
Showing 10 changed files with 185 additions and 31 deletions.
1 change: 1 addition & 0 deletions .cspell/dart_flutter.txt
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
autofocus
Cupertino
endtemplate
expando
gapless
Expand Down
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import 'dart:async';

import 'package:flutter/material.dart';
import 'package:meta/meta.dart';
import 'package:neon/src/models/account.dart';
Expand All @@ -18,11 +20,11 @@ class AccountSettingsTile extends SettingsTile {
/// {@macro neon.AccountTile.account}
final Account account;

/// {@macro neon.AccountTile.trailing}
/// {@macro neon.AdaptiveListTile.trailing}
final Widget? trailing;

/// {@macro neon.AccountTile.onTap}
final GestureTapCallback? onTap;
/// {@macro neon.AdaptiveListTile.onTap}
final FutureOr<void> Function()? onTap;

@override
Widget build(final BuildContext context) => NeonAccountTile(
Expand Down
Original file line number Diff line number Diff line change
@@ -1,26 +1,29 @@
import 'dart:async';

import 'package:flutter/material.dart';
import 'package:meta/meta.dart';
import 'package:neon/src/settings/widgets/settings_tile.dart';
import 'package:neon/src/widgets/adaptive_widgets/list_tile.dart';

@internal
class CustomSettingsTile extends SettingsTile {
const CustomSettingsTile({
this.title,
required this.title,
this.subtitle,
this.leading,
this.trailing,
this.onTap,
super.key,
});

final Widget? title;
final Widget title;
final Widget? subtitle;
final Widget? leading;
final Widget? trailing;
final GestureTapCallback? onTap;
final FutureOr<void> Function()? onTap;

@override
Widget build(final BuildContext context) => ListTile(
Widget build(final BuildContext context) => AdaptiveListTile(
title: title,
subtitle: subtitle,
leading: leading,
Expand Down
19 changes: 19 additions & 0 deletions packages/neon/neon/lib/src/utils/adaptive.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import 'package:flutter/material.dart';

/// Returns whether the current platform is a Cupertino one.
///
/// This is true for both `TargetPlatform.iOS` and `TargetPlatform.macOS`.
bool isCupertino(final BuildContext context) {
final theme = Theme.of(context);

switch (theme.platform) {
case TargetPlatform.android:
case TargetPlatform.fuchsia:
case TargetPlatform.linux:
case TargetPlatform.windows:
return false;
case TargetPlatform.iOS:
case TargetPlatform.macOS:
return true;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import 'package:neon/src/pages/settings.dart';
import 'package:neon/src/router.dart';
import 'package:neon/src/utils/provider.dart';
import 'package:neon/src/widgets/account_selection_dialog.dart';
import 'package:neon/src/widgets/adaptive_widgets/list_tile.dart';
import 'package:neon/src/widgets/user_avatar.dart';

@internal
Expand All @@ -23,7 +24,7 @@ class AccountSwitcherButton extends StatelessWidget {
builder: (final context) => NeonAccountSelectionDialog(
highlightActiveAccount: true,
children: [
ListTile(
AdaptiveListTile(
leading: const Icon(Icons.settings),
title: Text(NeonLocalizations.of(context).settingsAccountManage),
onTap: () {
Expand Down
23 changes: 7 additions & 16 deletions packages/neon/neon/lib/src/widgets/account_tile.dart
Original file line number Diff line number Diff line change
@@ -1,10 +1,13 @@
import 'dart:async';

import 'package:flutter/material.dart';
import 'package:intersperse/intersperse.dart';
import 'package:meta/meta.dart';
import 'package:neon/src/bloc/result.dart';
import 'package:neon/src/blocs/accounts.dart';
import 'package:neon/src/models/account.dart';
import 'package:neon/src/utils/provider.dart';
import 'package:neon/src/widgets/adaptive_widgets/list_tile.dart';
import 'package:neon/src/widgets/error.dart';
import 'package:neon/src/widgets/linear_progress_indicator.dart';
import 'package:neon/src/widgets/user_avatar.dart';
Expand All @@ -27,23 +30,11 @@ class NeonAccountTile extends StatelessWidget {
/// {@endtemplate}
final Account account;

/// {@template neon.AccountTile.trailing}
/// A widget to display after the title.
///
/// Typically an [Icon] widget.
///
/// To show right-aligned metadata (assuming left-to-right reading order;
/// left-aligned for right-to-left reading order), consider using a [Row] with
/// [CrossAxisAlignment.baseline] alignment whose first item is [Expanded] and
/// whose second child is the metadata text, instead of using the [trailing]
/// property.
/// {@endtemplate}
/// {@macro neon.AdaptiveListTile.trailing}
final Widget? trailing;

/// {@template neon.AccountTile.onTap}
/// Called when the user taps this list tile.
/// {@endtemplate}
final GestureTapCallback? onTap;
/// {@macro neon.AdaptiveListTile.onTap}
final FutureOr<void> Function()? onTap;

/// Whether to also show the status on the avatar.
///
Expand All @@ -55,7 +46,7 @@ class NeonAccountTile extends StatelessWidget {
Widget build(final BuildContext context) {
final userDetailsBloc = NeonProvider.of<AccountsBloc>(context).getUserDetailsBlocFor(account);

return ListTile(
return AdaptiveListTile(
onTap: onTap,
leading: NeonUserAvatar(
account: account,
Expand Down
135 changes: 135 additions & 0 deletions packages/neon/neon/lib/src/widgets/adaptive_widgets/list_tile.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,135 @@
import 'dart:async';

import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';

/// A wrapper widget that adaptively displays a [ListTile] on Material platforms
/// and a [CupertinoListTile] on Cupertino ones.
class AdaptiveListTile extends StatelessWidget {
/// Creates a new adaptive list tile.
///
/// If supplied the [subtitle] will be displayed below the title.
const AdaptiveListTile({
required this.title,
this.enabled = true,
this.subtitle,
this.leading,
this.trailing,
this.onTap,
super.key,
}) : additionalInfo = null;

/// Creates a new adaptive list tile.
///
/// If supplied the [additionalInfo] will be displayed below the title on
/// Material platforms and as a trailing widget on Cupertino ones.
const AdaptiveListTile.additionalInfo({
required this.title,
this.enabled = true,
this.additionalInfo,
this.leading,
this.trailing,
this.onTap,
super.key,
}) : subtitle = null;

/// {@template neon.AdaptiveListTile.title}
/// A [title] is used to convey the central information. Usually a [Text].
/// {@endtemplate}
final Widget title;

/// {@template neon.AdaptiveListTile.subtitle}
/// A [subtitle] is used to display additional information. It is located
/// below [title]. Usually a [Text] widget.
final Widget? subtitle;

/// {@template neon.AdaptiveListTile.additionalInfo}
/// Similar to [subtitle], an [additionalInfo] is used to display additional
/// information. However, instead of being displayed below [title], it is
/// displayed on the right, before [trailing]. Usually a [Text] widget.
///
/// This is only available on Cupertino platforms.
/// {@endtemplate}
final Widget? additionalInfo;

/// {@template neon.AdaptiveListTile.leading}
/// A widget displayed at the start of the [AdaptiveListTile]. This is
/// typically an `Icon` or an `Image`.
/// {@endtemplate}
final Widget? leading;

/// {@template neon.AdaptiveListTile.trailing}
/// A widget displayed at the end of the [AdaptiveListTile].
/// {@endtemplate}
final Widget? trailing;

/// {@template neon.AdaptiveListTile.onTap}
/// The [onTap] function is called when a user taps on the[AdaptiveListTile].
/// If left `null`, the [AdaptiveListTile] will not react to taps.
///
/// If the platform is a Cupertino one and this is a `Future<void> Function()`,
/// then the [AdaptiveListTile] remains activated until the returned future is
/// awaited. This is according to iOS behavior.
/// However, if this function is a `void Function()`, then the tile is active
/// only for the duration of invocation.
/// {@endtemplate}
final FutureOr<void> Function()? onTap;

/// {@template neon.AdaptiveListTile.enabled}
/// Whether this list tile is interactive.
///
/// If false, this list tile is styled with the disabled color from the
/// current [Theme] and the [onTap] callback is inoperative.
/// {@endtemplate}
final bool enabled;

@override
Widget build(final BuildContext context) {
final theme = Theme.of(context);

switch (theme.platform) {
case TargetPlatform.android:
case TargetPlatform.fuchsia:
case TargetPlatform.linux:
case TargetPlatform.windows:
return ListTile(
title: title,
subtitle: subtitle,
leading: leading,
trailing: trailing,
onTap: onTap,
enabled: enabled,
);
case TargetPlatform.iOS:
case TargetPlatform.macOS:
final tile = CupertinoListTile(
title: title,
subtitle: additionalInfo == null ? subtitle : null,
additionalInfo: additionalInfo,
leading: leading,
trailing: trailing,
onTap: enabled ? onTap : null,
);

if (!enabled) {
var data = CupertinoTheme.of(context);
data = data.copyWith(
textTheme: data.resolveFrom(context).textTheme.copyWith(
textStyle: data.textTheme.textStyle.merge(
TextStyle(
color: theme.disabledColor,
),
),
),
);

return CupertinoTheme(
data: data,
child: tile,
);
}

return tile;
}
}
}
8 changes: 4 additions & 4 deletions packages/neon/neon/lib/src/widgets/error.dart
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import 'package:neon/src/blocs/accounts.dart';
import 'package:neon/src/router.dart';
import 'package:neon/src/utils/exceptions.dart';
import 'package:neon/src/utils/provider.dart';
import 'package:neon/src/widgets/adaptive_widgets/list_tile.dart';
import 'package:nextcloud/nextcloud.dart';
import 'package:universal_io/io.dart';

Expand All @@ -22,7 +23,7 @@ enum NeonErrorType {
/// Shows a column with the error message and a retry button.
column,

/// Shows a [ListTile] with the error.
/// Shows a [AdaptiveListTile] with the error.
listTile,
}

Expand Down Expand Up @@ -147,10 +148,9 @@ class NeonError extends StatelessWidget {
),
);
case NeonErrorType.listTile:
return ListTile(
return AdaptiveListTile(
leading: errorIcon,
title: Text(message),
titleTextStyle: textStyle,
title: Text(message, style: textStyle),
onTap: onPressed,
);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import 'package:neon/src/blocs/unified_search.dart';
import 'package:neon/src/models/account.dart';
import 'package:neon/src/theme/sizes.dart';
import 'package:neon/src/utils/provider.dart';
import 'package:neon/src/widgets/adaptive_widgets/list_tile.dart';
import 'package:neon/src/widgets/error.dart';
import 'package:neon/src/widgets/image.dart';
import 'package:neon/src/widgets/linear_progress_indicator.dart';
Expand Down Expand Up @@ -81,7 +82,7 @@ class NeonUnifiedSearchResults extends StatelessWidget {
visible: result.isLoading,
),
if (entries.isEmpty) ...[
ListTile(
AdaptiveListTile(
leading: const Icon(
Icons.close,
size: largeIconSize,
Expand All @@ -90,7 +91,7 @@ class NeonUnifiedSearchResults extends StatelessWidget {
),
],
for (final entry in entries) ...[
ListTile(
AdaptiveListTile(
leading: NeonImageWrapper(
size: const Size.square(largeIconSize),
child: _buildThumbnail(context, accountsBloc.activeAccount.value!, entry),
Expand Down
3 changes: 2 additions & 1 deletion packages/neon/neon/lib/src/widgets/validation_tile.dart
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import 'package:flutter/material.dart';
import 'package:neon/src/widgets/adaptive_widgets/list_tile.dart';

/// Validation list tile.
///
Expand Down Expand Up @@ -48,7 +49,7 @@ class NeonValidationTile extends StatelessWidget {
size: size,
),
};
return ListTile(
return AdaptiveListTile(
leading: leading,
title: Text(
title,
Expand Down

0 comments on commit e888216

Please sign in to comment.