mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-04 03:39:37 -05:00 
			
		
		
		
	* refactor: autoroutex pushroute * refactor: autoroutex popRoute * refactor: autoroutex navigate and replace * chore: add doc comments for extension methods * refactor: Add LoggerMixin and refactor Album activities to use mixin * refactor: Activity page * chore: activity user from user constructor * fix: update current asset after build method * refactor: tests with similar structure as lib * chore: remove avoid-declaring-call-method rule from dcm analysis * test: fix proper expect order * test: activity_statistics_provider_test * test: activity_provider_test * test: use proper matchers * test: activity_text_field_test & dismissible_activity_test added * test: add http mock to return transparent image * test: download isar core libs during test * test: add widget tags to widget test cases * test: activity_tile_test * build: currentAlbumProvider to generator * movie add / remove like to activity input tile * test: activities_page_test.dart * chore: better error logs * chore: dismissibleactivity as statelesswidget --------- Co-authored-by: shalong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
		
			
				
	
	
		
			117 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			117 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
import 'package:flutter/material.dart';
 | 
						|
import 'package:hooks_riverpod/hooks_riverpod.dart';
 | 
						|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
 | 
						|
import 'package:immich_mobile/extensions/datetime_extensions.dart';
 | 
						|
import 'package:immich_mobile/modules/activities/models/activity.model.dart';
 | 
						|
import 'package:immich_mobile/modules/asset_viewer/providers/current_asset.provider.dart';
 | 
						|
import 'package:immich_mobile/shared/ui/immich_image.dart';
 | 
						|
import 'package:immich_mobile/shared/ui/user_circle_avatar.dart';
 | 
						|
 | 
						|
class ActivityTile extends HookConsumerWidget {
 | 
						|
  final Activity activity;
 | 
						|
 | 
						|
  const ActivityTile(this.activity, {super.key});
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context, WidgetRef ref) {
 | 
						|
    final asset = ref.watch(currentAssetProvider);
 | 
						|
    final isLike = activity.type == ActivityType.like;
 | 
						|
    // Asset thumbnail is displayed when we are accessing activities from the album page
 | 
						|
    // currentAssetProvider will not be set until we open the gallery viewer
 | 
						|
    final showAssetThumbnail = asset == null && activity.assetId != null;
 | 
						|
 | 
						|
    return ListTile(
 | 
						|
      minVerticalPadding: 15,
 | 
						|
      leading: isLike
 | 
						|
          ? Container(
 | 
						|
              width: 44,
 | 
						|
              alignment: Alignment.center,
 | 
						|
              child: Icon(
 | 
						|
                Icons.favorite_rounded,
 | 
						|
                color: Colors.red[700],
 | 
						|
              ),
 | 
						|
            )
 | 
						|
          : UserCircleAvatar(user: activity.user),
 | 
						|
      title: _ActivityTitle(
 | 
						|
        userName: activity.user.name,
 | 
						|
        createdAt: activity.createdAt.timeAgo(),
 | 
						|
        leftAlign: isLike || showAssetThumbnail,
 | 
						|
      ),
 | 
						|
      // No subtitle for like, so center title
 | 
						|
      titleAlignment:
 | 
						|
          !isLike ? ListTileTitleAlignment.top : ListTileTitleAlignment.center,
 | 
						|
      trailing: showAssetThumbnail
 | 
						|
          ? _ActivityAssetThumbnail(activity.assetId!)
 | 
						|
          : null,
 | 
						|
      subtitle: !isLike ? Text(activity.comment!) : null,
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class _ActivityTitle extends StatelessWidget {
 | 
						|
  final String userName;
 | 
						|
  final String createdAt;
 | 
						|
  final bool leftAlign;
 | 
						|
 | 
						|
  const _ActivityTitle({
 | 
						|
    required this.userName,
 | 
						|
    required this.createdAt,
 | 
						|
    required this.leftAlign,
 | 
						|
  });
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    final textColor = context.isDarkTheme ? Colors.white : Colors.black;
 | 
						|
    final textStyle = context.textTheme.bodyMedium
 | 
						|
        ?.copyWith(color: textColor.withOpacity(0.6));
 | 
						|
 | 
						|
    return Row(
 | 
						|
      mainAxisAlignment:
 | 
						|
          leftAlign ? MainAxisAlignment.start : MainAxisAlignment.spaceBetween,
 | 
						|
      mainAxisSize: leftAlign ? MainAxisSize.min : MainAxisSize.max,
 | 
						|
      children: [
 | 
						|
        Text(
 | 
						|
          userName,
 | 
						|
          style: textStyle,
 | 
						|
          overflow: TextOverflow.ellipsis,
 | 
						|
        ),
 | 
						|
        if (leftAlign)
 | 
						|
          Text(
 | 
						|
            " • ",
 | 
						|
            style: textStyle,
 | 
						|
          ),
 | 
						|
        Expanded(
 | 
						|
          child: Text(
 | 
						|
            createdAt,
 | 
						|
            style: textStyle,
 | 
						|
            overflow: TextOverflow.ellipsis,
 | 
						|
            textAlign: leftAlign ? TextAlign.left : TextAlign.right,
 | 
						|
          ),
 | 
						|
        ),
 | 
						|
      ],
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
class _ActivityAssetThumbnail extends StatelessWidget {
 | 
						|
  final String assetId;
 | 
						|
 | 
						|
  const _ActivityAssetThumbnail(this.assetId);
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    return Container(
 | 
						|
      width: 40,
 | 
						|
      height: 30,
 | 
						|
      decoration: BoxDecoration(
 | 
						|
        borderRadius: const BorderRadius.all(Radius.circular(4)),
 | 
						|
        image: DecorationImage(
 | 
						|
          image: ImmichImage.remoteThumbnailProviderForId(assetId),
 | 
						|
          fit: BoxFit.cover,
 | 
						|
        ),
 | 
						|
      ),
 | 
						|
      child: const SizedBox.shrink(),
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 |