Compare commits

...

8 Commits

7 changed files with 212 additions and 79 deletions

View File

@@ -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.",

View File

@@ -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);

View File

@@ -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),

View File

@@ -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,10 +47,17 @@ 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(
builder: (context, constraints) {
final metrics = _OverlayMetrics.fromConstraints(constraints);
return AnimatedContainer(
duration: Durations.short4, duration: Durations.short4,
curve: Curves.decelerate, curve: Curves.decelerate,
padding: EdgeInsets.all(isSelected || lockSelection ? 6 : 0), padding: EdgeInsets.all(isSelected || lockSelection ? 6 : 0),
@@ -70,43 +79,58 @@ class ThumbnailTile extends ConsumerWidget {
if (asset != null) if (asset != null)
Align( Align(
alignment: Alignment.topRight, alignment: Alignment.topRight,
child: _AssetTypeIcons(asset: asset), 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) if (storageIndicator && asset != null)
switch (asset.storage) { Padding(
AssetState.local => const Align( padding: EdgeInsets.only(right: metrics.iconSpacing),
alignment: Alignment.bottomRight, child: _TileOverlayIcon(switch (asset.storage) {
child: Padding( AssetState.local => Icons.cloud_off_outlined,
padding: EdgeInsets.only(right: 10.0, bottom: 6.0), AssetState.remote => Icons.cloud_outlined,
child: _TileOverlayIcon(Icons.cloud_off_outlined), AssetState.merged => Icons.cloud_done_outlined,
}, metrics: metrics),
), ),
],
), ),
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),
);
}
}

View File

@@ -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,
), ),
), ),
); );

View File

@@ -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);

View File

@@ -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(),
]; ];