mirror of
https://github.com/immich-app/immich.git
synced 2025-12-20 01:11:46 +03:00
Compare commits
8 Commits
workflow-u
...
feature/mo
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c90c7a56ce | ||
|
|
501617ba93 | ||
|
|
5050f8fbb4 | ||
|
|
ea37b12b39 | ||
|
|
cc3307d335 | ||
|
|
85086cdb02 | ||
|
|
9262f69519 | ||
|
|
f13a5ba418 |
@@ -2062,6 +2062,7 @@
|
|||||||
"theme": "Theme",
|
"theme": "Theme",
|
||||||
"theme_selection": "Theme selection",
|
"theme_selection": "Theme selection",
|
||||||
"theme_selection_description": "Automatically set the theme to light or dark based on your browser's system preference",
|
"theme_selection_description": "Automatically set the theme to light or dark based on your browser's system preference",
|
||||||
|
"theme_setting_asset_list_show_owner_name_title": "Show asset owner name",
|
||||||
"theme_setting_asset_list_storage_indicator_title": "Show storage indicator on asset tiles",
|
"theme_setting_asset_list_storage_indicator_title": "Show storage indicator on asset tiles",
|
||||||
"theme_setting_asset_list_tiles_per_row_title": "Number of assets per row ({count})",
|
"theme_setting_asset_list_tiles_per_row_title": "Number of assets per row ({count})",
|
||||||
"theme_setting_colorful_interface_subtitle": "Apply primary color to background surfaces.",
|
"theme_setting_colorful_interface_subtitle": "Apply primary color to background surfaces.",
|
||||||
|
|||||||
@@ -9,7 +9,8 @@ enum Setting<T> {
|
|||||||
autoPlayVideo<bool>(StoreKey.autoPlayVideo, true),
|
autoPlayVideo<bool>(StoreKey.autoPlayVideo, true),
|
||||||
preferRemoteImage<bool>(StoreKey.preferRemoteImage, false),
|
preferRemoteImage<bool>(StoreKey.preferRemoteImage, false),
|
||||||
advancedTroubleshooting<bool>(StoreKey.advancedTroubleshooting, false),
|
advancedTroubleshooting<bool>(StoreKey.advancedTroubleshooting, false),
|
||||||
enableBackup<bool>(StoreKey.enableBackup, false);
|
enableBackup<bool>(StoreKey.enableBackup, false),
|
||||||
|
showOwnerName<bool>(StoreKey.showOwnerName, false);
|
||||||
|
|
||||||
const Setting(this.storeKey, this.defaultValue);
|
const Setting(this.storeKey, this.defaultValue);
|
||||||
|
|
||||||
|
|||||||
@@ -72,6 +72,7 @@ enum StoreKey<T> {
|
|||||||
|
|
||||||
autoPlayVideo<bool>._(139),
|
autoPlayVideo<bool>._(139),
|
||||||
albumGridView<bool>._(140),
|
albumGridView<bool>._(140),
|
||||||
|
showOwnerName<bool>._(141),
|
||||||
|
|
||||||
// Experimental stuff
|
// Experimental stuff
|
||||||
photoManagerCustomFilter<bool>._(1000),
|
photoManagerCustomFilter<bool>._(1000),
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ class ThumbnailTile extends ConsumerWidget {
|
|||||||
this.showStorageIndicator = false,
|
this.showStorageIndicator = false,
|
||||||
this.lockSelection = false,
|
this.lockSelection = false,
|
||||||
this.heroOffset,
|
this.heroOffset,
|
||||||
|
this.ownerName,
|
||||||
super.key,
|
super.key,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -28,6 +29,7 @@ class ThumbnailTile extends ConsumerWidget {
|
|||||||
final bool showStorageIndicator;
|
final bool showStorageIndicator;
|
||||||
final bool lockSelection;
|
final bool lockSelection;
|
||||||
final int? heroOffset;
|
final int? heroOffset;
|
||||||
|
final String? ownerName;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
@@ -45,68 +47,90 @@ class ThumbnailTile extends ConsumerWidget {
|
|||||||
final bool storageIndicator =
|
final bool storageIndicator =
|
||||||
ref.watch(settingsProvider.select((s) => s.get(Setting.showStorageIndicator))) && showStorageIndicator;
|
ref.watch(settingsProvider.select((s) => s.get(Setting.showStorageIndicator))) && showStorageIndicator;
|
||||||
|
|
||||||
|
final bool showOwnerNameSetting = ref.watch(settingsProvider.select((s) => s.get(Setting.showOwnerName)));
|
||||||
|
final shouldShowOwnerName = showOwnerNameSetting && ownerName != null;
|
||||||
|
|
||||||
return Stack(
|
return Stack(
|
||||||
children: [
|
children: [
|
||||||
Container(color: lockSelection ? context.colorScheme.surfaceContainerHighest : assetContainerColor),
|
Container(color: lockSelection ? context.colorScheme.surfaceContainerHighest : assetContainerColor),
|
||||||
AnimatedContainer(
|
LayoutBuilder(
|
||||||
duration: Durations.short4,
|
builder: (context, constraints) {
|
||||||
curve: Curves.decelerate,
|
final metrics = _OverlayMetrics.fromConstraints(constraints);
|
||||||
padding: EdgeInsets.all(isSelected || lockSelection ? 6 : 0),
|
|
||||||
child: TweenAnimationBuilder<double>(
|
return AnimatedContainer(
|
||||||
tween: Tween<double>(begin: 0.0, end: (isSelected || lockSelection) ? 15.0 : 0.0),
|
duration: Durations.short4,
|
||||||
duration: Durations.short4,
|
curve: Curves.decelerate,
|
||||||
curve: Curves.decelerate,
|
padding: EdgeInsets.all(isSelected || lockSelection ? 6 : 0),
|
||||||
builder: (context, value, child) {
|
child: TweenAnimationBuilder<double>(
|
||||||
return ClipRRect(borderRadius: BorderRadius.all(Radius.circular(value)), child: child);
|
tween: Tween<double>(begin: 0.0, end: (isSelected || lockSelection) ? 15.0 : 0.0),
|
||||||
},
|
duration: Durations.short4,
|
||||||
child: Stack(
|
curve: Curves.decelerate,
|
||||||
children: [
|
builder: (context, value, child) {
|
||||||
Positioned.fill(
|
return ClipRRect(borderRadius: BorderRadius.all(Radius.circular(value)), child: child);
|
||||||
child: Hero(
|
},
|
||||||
tag: '${asset?.heroTag ?? ''}_$heroIndex',
|
child: Stack(
|
||||||
child: Thumbnail.fromAsset(asset: asset, size: size),
|
children: [
|
||||||
),
|
Positioned.fill(
|
||||||
|
child: Hero(
|
||||||
|
tag: '${asset?.heroTag ?? ''}_$heroIndex',
|
||||||
|
child: Thumbnail.fromAsset(asset: asset, size: size),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (asset != null)
|
||||||
|
Align(
|
||||||
|
alignment: Alignment.topRight,
|
||||||
|
child: Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: metrics.padding, vertical: metrics.padding),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.end,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.center,
|
||||||
|
children: [_AssetTypeIcons(asset: asset, metrics: metrics)],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (shouldShowOwnerName ||
|
||||||
|
(storageIndicator && asset != null) ||
|
||||||
|
(asset != null && asset.isFavorite))
|
||||||
|
Align(
|
||||||
|
alignment: Alignment.bottomCenter,
|
||||||
|
child: Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: metrics.padding, vertical: metrics.padding),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
if (asset != null && asset.isFavorite)
|
||||||
|
Padding(
|
||||||
|
padding: EdgeInsets.only(right: metrics.iconSpacing),
|
||||||
|
child: _TileOverlayIcon(Icons.favorite_rounded, metrics: metrics),
|
||||||
|
)
|
||||||
|
else
|
||||||
|
const SizedBox.shrink(),
|
||||||
|
if (shouldShowOwnerName)
|
||||||
|
Flexible(
|
||||||
|
child: Padding(
|
||||||
|
padding: EdgeInsets.symmetric(horizontal: metrics.iconSpacing),
|
||||||
|
child: _OwnerNameLabel(ownerName: ownerName!, metrics: metrics),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (storageIndicator && asset != null)
|
||||||
|
Padding(
|
||||||
|
padding: EdgeInsets.only(right: metrics.iconSpacing),
|
||||||
|
child: _TileOverlayIcon(switch (asset.storage) {
|
||||||
|
AssetState.local => Icons.cloud_off_outlined,
|
||||||
|
AssetState.remote => Icons.cloud_outlined,
|
||||||
|
AssetState.merged => Icons.cloud_done_outlined,
|
||||||
|
}, metrics: metrics),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
if (asset != null)
|
),
|
||||||
Align(
|
);
|
||||||
alignment: Alignment.topRight,
|
},
|
||||||
child: _AssetTypeIcons(asset: asset),
|
|
||||||
),
|
|
||||||
if (storageIndicator && asset != null)
|
|
||||||
switch (asset.storage) {
|
|
||||||
AssetState.local => const Align(
|
|
||||||
alignment: Alignment.bottomRight,
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsets.only(right: 10.0, bottom: 6.0),
|
|
||||||
child: _TileOverlayIcon(Icons.cloud_off_outlined),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
AssetState.remote => const Align(
|
|
||||||
alignment: Alignment.bottomRight,
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsets.only(right: 10.0, bottom: 6.0),
|
|
||||||
child: _TileOverlayIcon(Icons.cloud_outlined),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
AssetState.merged => const Align(
|
|
||||||
alignment: Alignment.bottomRight,
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsets.only(right: 10.0, bottom: 6.0),
|
|
||||||
child: _TileOverlayIcon(Icons.cloud_done_outlined),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
},
|
|
||||||
if (asset != null && asset.isFavorite)
|
|
||||||
const Align(
|
|
||||||
alignment: Alignment.bottomLeft,
|
|
||||||
child: Padding(
|
|
||||||
padding: EdgeInsets.only(left: 10.0, bottom: 6.0),
|
|
||||||
child: _TileOverlayIcon(Icons.favorite_rounded),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
TweenAnimationBuilder<double>(
|
TweenAnimationBuilder<double>(
|
||||||
tween: Tween<double>(begin: 0.0, end: (isSelected || lockSelection) ? 1.0 : 0.0),
|
tween: Tween<double>(begin: 0.0, end: (isSelected || lockSelection) ? 1.0 : 0.0),
|
||||||
@@ -157,12 +181,12 @@ class _SelectionIndicator extends StatelessWidget {
|
|||||||
|
|
||||||
class _VideoIndicator extends StatelessWidget {
|
class _VideoIndicator extends StatelessWidget {
|
||||||
final Duration duration;
|
final Duration duration;
|
||||||
const _VideoIndicator(this.duration);
|
final _OverlayMetrics metrics;
|
||||||
|
const _VideoIndicator(this.duration, {required this.metrics});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Row(
|
return Row(
|
||||||
spacing: 3,
|
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
mainAxisAlignment: MainAxisAlignment.end,
|
mainAxisAlignment: MainAxisAlignment.end,
|
||||||
// CrossAxisAlignment.start looks more centered vertically than CrossAxisAlignment.center
|
// CrossAxisAlignment.start looks more centered vertically than CrossAxisAlignment.center
|
||||||
@@ -170,14 +194,15 @@ class _VideoIndicator extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
duration.format(),
|
duration.format(),
|
||||||
style: const TextStyle(
|
style: TextStyle(
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
fontSize: 12,
|
fontSize: metrics.fontSize,
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
shadows: [Shadow(blurRadius: 5.0, color: Color.fromRGBO(0, 0, 0, 0.6))],
|
shadows: [Shadow(blurRadius: metrics.blurRadius, color: const Color.fromRGBO(0, 0, 0, 0.6))],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const _TileOverlayIcon(Icons.play_circle_outline_rounded),
|
SizedBox(width: metrics.iconSpacing),
|
||||||
|
_TileOverlayIcon(Icons.play_circle_outline_rounded, metrics: metrics),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -185,47 +210,115 @@ class _VideoIndicator extends StatelessWidget {
|
|||||||
|
|
||||||
class _TileOverlayIcon extends StatelessWidget {
|
class _TileOverlayIcon extends StatelessWidget {
|
||||||
final IconData icon;
|
final IconData icon;
|
||||||
|
final _OverlayMetrics metrics;
|
||||||
|
|
||||||
const _TileOverlayIcon(this.icon);
|
const _TileOverlayIcon(this.icon, {required this.metrics});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Icon(
|
return Icon(
|
||||||
icon,
|
icon,
|
||||||
color: Colors.white,
|
color: Colors.white,
|
||||||
size: 16,
|
size: metrics.iconSize,
|
||||||
shadows: [const Shadow(blurRadius: 5.0, color: Color.fromRGBO(0, 0, 0, 0.6), offset: Offset(0.0, 0.0))],
|
shadows: [
|
||||||
|
Shadow(
|
||||||
|
blurRadius: metrics.blurRadius,
|
||||||
|
color: const Color.fromRGBO(0, 0, 0, 0.6),
|
||||||
|
offset: const Offset(0.0, 0.0),
|
||||||
|
),
|
||||||
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class _AssetTypeIcons extends StatelessWidget {
|
class _AssetTypeIcons extends StatelessWidget {
|
||||||
final BaseAsset asset;
|
final BaseAsset asset;
|
||||||
|
final _OverlayMetrics metrics;
|
||||||
|
|
||||||
const _AssetTypeIcons({required this.asset});
|
const _AssetTypeIcons({required this.asset, required this.metrics});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final hasStack = asset is RemoteAsset && (asset as RemoteAsset).stackId != null;
|
final hasStack = asset is RemoteAsset && (asset as RemoteAsset).stackId != null;
|
||||||
final isLivePhoto = asset is RemoteAsset && asset.livePhotoVideoId != null;
|
final isLivePhoto = asset is RemoteAsset && asset.livePhotoVideoId != null;
|
||||||
|
|
||||||
return Column(
|
return Row(
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
crossAxisAlignment: CrossAxisAlignment.end,
|
|
||||||
children: [
|
children: [
|
||||||
if (asset.isVideo)
|
if (asset.isVideo)
|
||||||
Padding(padding: const EdgeInsets.only(right: 10.0, top: 6.0), child: _VideoIndicator(asset.duration)),
|
Padding(
|
||||||
|
padding: EdgeInsets.only(left: metrics.iconSpacing),
|
||||||
|
child: _VideoIndicator(asset.duration, metrics: metrics),
|
||||||
|
),
|
||||||
if (hasStack)
|
if (hasStack)
|
||||||
const Padding(
|
Padding(
|
||||||
padding: EdgeInsets.only(right: 10.0, top: 6.0),
|
padding: EdgeInsets.only(left: metrics.iconSpacing),
|
||||||
child: _TileOverlayIcon(Icons.burst_mode_rounded),
|
child: _TileOverlayIcon(Icons.burst_mode_rounded, metrics: metrics),
|
||||||
),
|
),
|
||||||
if (isLivePhoto)
|
if (isLivePhoto)
|
||||||
const Padding(
|
Padding(
|
||||||
padding: EdgeInsets.only(right: 10.0, top: 6.0),
|
padding: EdgeInsets.only(left: metrics.iconSpacing),
|
||||||
child: _TileOverlayIcon(Icons.motion_photos_on_rounded),
|
child: _TileOverlayIcon(Icons.motion_photos_on_rounded, metrics: metrics),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class _OwnerNameLabel extends StatelessWidget {
|
||||||
|
final String ownerName;
|
||||||
|
final _OverlayMetrics metrics;
|
||||||
|
|
||||||
|
const _OwnerNameLabel({required this.ownerName, required this.metrics});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Text(
|
||||||
|
ownerName,
|
||||||
|
style: TextStyle(
|
||||||
|
color: Colors.white,
|
||||||
|
fontSize: metrics.fontSize,
|
||||||
|
fontWeight: FontWeight.w500,
|
||||||
|
shadows: [
|
||||||
|
Shadow(
|
||||||
|
blurRadius: metrics.blurRadius,
|
||||||
|
color: const Color.fromRGBO(0, 0, 0, 0.6),
|
||||||
|
offset: const Offset(0.0, 0.0),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
overflow: TextOverflow.fade,
|
||||||
|
softWrap: false,
|
||||||
|
maxLines: 1,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class _OverlayMetrics {
|
||||||
|
final double padding;
|
||||||
|
final double iconSize;
|
||||||
|
final double fontSize;
|
||||||
|
final double iconSpacing;
|
||||||
|
final double blurRadius;
|
||||||
|
|
||||||
|
const _OverlayMetrics({
|
||||||
|
required this.padding,
|
||||||
|
required this.iconSize,
|
||||||
|
required this.fontSize,
|
||||||
|
required this.iconSpacing,
|
||||||
|
required this.blurRadius,
|
||||||
|
});
|
||||||
|
|
||||||
|
factory _OverlayMetrics.fromConstraints(BoxConstraints constraints) {
|
||||||
|
const baseSize = 120.0;
|
||||||
|
final scale = (constraints.maxWidth / baseSize).clamp(0.5, 2.0);
|
||||||
|
|
||||||
|
return _OverlayMetrics(
|
||||||
|
padding: (2.0 * scale).clamp(1.0, 4.0),
|
||||||
|
iconSize: (16.0 * scale).clamp(14.0, 20.0),
|
||||||
|
fontSize: (12.0 * scale).clamp(11.0, 14.0),
|
||||||
|
iconSpacing: (2.0 * scale).clamp(1.0, 4.0),
|
||||||
|
blurRadius: (5.0 * scale).clamp(3.0, 7.0),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import 'dart:async';
|
|||||||
import 'dart:math' as math;
|
import 'dart:math' as math;
|
||||||
|
|
||||||
import 'package:auto_route/auto_route.dart';
|
import 'package:auto_route/auto_route.dart';
|
||||||
|
import 'package:collection/collection.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
||||||
@@ -18,6 +19,7 @@ import 'package:immich_mobile/providers/asset_viewer/is_motion_video_playing.pro
|
|||||||
import 'package:immich_mobile/providers/haptic_feedback.provider.dart';
|
import 'package:immich_mobile/providers/haptic_feedback.provider.dart';
|
||||||
import 'package:immich_mobile/providers/infrastructure/current_album.provider.dart';
|
import 'package:immich_mobile/providers/infrastructure/current_album.provider.dart';
|
||||||
import 'package:immich_mobile/providers/infrastructure/readonly_mode.provider.dart';
|
import 'package:immich_mobile/providers/infrastructure/readonly_mode.provider.dart';
|
||||||
|
import 'package:immich_mobile/providers/infrastructure/remote_album.provider.dart';
|
||||||
import 'package:immich_mobile/providers/infrastructure/timeline.provider.dart';
|
import 'package:immich_mobile/providers/infrastructure/timeline.provider.dart';
|
||||||
import 'package:immich_mobile/providers/timeline/multiselect.provider.dart';
|
import 'package:immich_mobile/providers/timeline/multiselect.provider.dart';
|
||||||
import 'package:immich_mobile/routing/router.dart';
|
import 'package:immich_mobile/routing/router.dart';
|
||||||
@@ -191,6 +193,29 @@ class _AssetTileWidget extends ConsumerWidget {
|
|||||||
return lockSelectionAssets.contains(asset);
|
return lockSelectionAssets.contains(asset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String? _getOwnerName(WidgetRef ref) {
|
||||||
|
final album = ref.watch(currentRemoteAlbumProvider);
|
||||||
|
if (album == null || !album.isShared) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (asset case RemoteAsset remoteAsset) {
|
||||||
|
final ownerId = remoteAsset.ownerId;
|
||||||
|
|
||||||
|
if (album.ownerId == ownerId) {
|
||||||
|
return album.ownerName;
|
||||||
|
}
|
||||||
|
|
||||||
|
final sharedUsersAsync = ref.watch(remoteAlbumSharedUsersProvider(album.id));
|
||||||
|
return sharedUsersAsync.maybeWhen(
|
||||||
|
data: (sharedUsers) => sharedUsers.firstWhereOrNull((user) => user.id == ownerId)?.name,
|
||||||
|
orElse: () => null,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
final heroOffset = TabsRouterScope.of(context)?.controller.activeIndex ?? 0;
|
final heroOffset = TabsRouterScope.of(context)?.controller.activeIndex ?? 0;
|
||||||
@@ -198,6 +223,7 @@ class _AssetTileWidget extends ConsumerWidget {
|
|||||||
final lockSelection = _getLockSelectionStatus(ref);
|
final lockSelection = _getLockSelectionStatus(ref);
|
||||||
final showStorageIndicator = ref.watch(timelineArgsProvider.select((args) => args.showStorageIndicator));
|
final showStorageIndicator = ref.watch(timelineArgsProvider.select((args) => args.showStorageIndicator));
|
||||||
final isReadonlyModeEnabled = ref.watch(readonlyModeProvider);
|
final isReadonlyModeEnabled = ref.watch(readonlyModeProvider);
|
||||||
|
final ownerName = _getOwnerName(ref);
|
||||||
|
|
||||||
return RepaintBoundary(
|
return RepaintBoundary(
|
||||||
child: GestureDetector(
|
child: GestureDetector(
|
||||||
@@ -208,6 +234,7 @@ class _AssetTileWidget extends ConsumerWidget {
|
|||||||
lockSelection: lockSelection,
|
lockSelection: lockSelection,
|
||||||
showStorageIndicator: showStorageIndicator,
|
showStorageIndicator: showStorageIndicator,
|
||||||
heroOffset: heroOffset,
|
heroOffset: heroOffset,
|
||||||
|
ownerName: ownerName,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -54,7 +54,8 @@ enum AppSettingsEnum<T> {
|
|||||||
readonlyModeEnabled<bool>(StoreKey.readonlyModeEnabled, "readonlyModeEnabled", false),
|
readonlyModeEnabled<bool>(StoreKey.readonlyModeEnabled, "readonlyModeEnabled", false),
|
||||||
albumGridView<bool>(StoreKey.albumGridView, "albumGridView", false),
|
albumGridView<bool>(StoreKey.albumGridView, "albumGridView", false),
|
||||||
backupRequireCharging<bool>(StoreKey.backupRequireCharging, null, false),
|
backupRequireCharging<bool>(StoreKey.backupRequireCharging, null, false),
|
||||||
backupTriggerDelay<int>(StoreKey.backupTriggerDelay, null, 30);
|
backupTriggerDelay<int>(StoreKey.backupTriggerDelay, null, 30),
|
||||||
|
showOwnerName<bool>(StoreKey.showOwnerName, "showOwnerName", false);
|
||||||
|
|
||||||
const AppSettingsEnum(this.storeKey, this.hiveKey, this.defaultValue);
|
const AppSettingsEnum(this.storeKey, this.hiveKey, this.defaultValue);
|
||||||
|
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ class AssetListSettings extends HookConsumerWidget {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
final showStorageIndicator = useAppSettingsState(AppSettingsEnum.storageIndicator);
|
final showStorageIndicator = useAppSettingsState(AppSettingsEnum.storageIndicator);
|
||||||
|
final showOwnerName = useAppSettingsState(AppSettingsEnum.showOwnerName);
|
||||||
|
|
||||||
final assetListSetting = [
|
final assetListSetting = [
|
||||||
SettingsSwitchListTile(
|
SettingsSwitchListTile(
|
||||||
@@ -27,6 +28,14 @@ class AssetListSettings extends HookConsumerWidget {
|
|||||||
ref.invalidate(settingsProvider);
|
ref.invalidate(settingsProvider);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
SettingsSwitchListTile(
|
||||||
|
valueNotifier: showOwnerName,
|
||||||
|
title: 'theme_setting_asset_list_show_owner_name_title'.tr(),
|
||||||
|
onChanged: (_) {
|
||||||
|
ref.invalidate(appSettingsServiceProvider);
|
||||||
|
ref.invalidate(settingsProvider);
|
||||||
|
},
|
||||||
|
),
|
||||||
const LayoutSettings(),
|
const LayoutSettings(),
|
||||||
const GroupSettings(),
|
const GroupSettings(),
|
||||||
];
|
];
|
||||||
|
|||||||
Reference in New Issue
Block a user