aves/lib/widgets/viewer/overlay/top.dart
2023-08-13 20:04:08 +02:00

145 lines
5.8 KiB
Dart

import 'package:aves/model/entry/entry.dart';
import 'package:aves/model/entry/extensions/multipage.dart';
import 'package:aves/model/settings/settings.dart';
import 'package:aves/theme/themes.dart';
import 'package:aves/widgets/common/fx/blurred.dart';
import 'package:aves/widgets/viewer/multipage/conductor.dart';
import 'package:aves/widgets/viewer/overlay/details/details.dart';
import 'package:aves/widgets/viewer/overlay/histogram.dart';
import 'package:aves/widgets/viewer/overlay/minimap.dart';
import 'package:aves/widgets/viewer/page_entry_builder.dart';
import 'package:aves/widgets/viewer/view/conductor.dart';
import 'package:aves/widgets/viewer/view/controller.dart';
import 'package:aves_model/aves_model.dart';
import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
class ViewerTopOverlay extends StatelessWidget {
final List<AvesEntry> entries;
final int index;
final AvesEntry mainEntry;
final Animation<double> scale;
final bool hasCollection;
final ValueNotifier<bool> expandedNotifier;
final Size availableSize;
final EdgeInsets? viewInsets, viewPadding;
static const Color componentBorderColor = Colors.white30;
static const double componentDimension = 96;
const ViewerTopOverlay({
super.key,
required this.entries,
required this.index,
required this.mainEntry,
required this.scale,
required this.hasCollection,
required this.expandedNotifier,
required this.availableSize,
required this.viewInsets,
required this.viewPadding,
});
@override
Widget build(BuildContext context) {
final multiPageController = mainEntry.isMultiPage ? context.read<MultiPageConductor>().getController(mainEntry) : null;
return PageEntryBuilder(
multiPageController: multiPageController,
builder: (pageEntry) {
pageEntry ??= mainEntry;
final showInfo = settings.showOverlayInfo;
final viewStateConductor = context.read<ViewStateConductor>();
final viewStateNotifier = viewStateConductor.getOrCreateController(pageEntry).viewStateNotifier;
final blurred = settings.enableBlurEffect;
final viewInsetsPadding = (viewInsets ?? EdgeInsets.zero) + (viewPadding ?? EdgeInsets.zero);
return Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (showInfo)
GestureDetector(
onTap: () => expandedNotifier.value = !expandedNotifier.value,
child: BlurredRect(
enabled: blurred,
child: Container(
color: Themes.overlayBackgroundColor(brightness: Theme.of(context).brightness, blurred: blurred),
child: SafeArea(
bottom: false,
minimum: EdgeInsets.only(
left: viewInsetsPadding.left,
top: viewInsetsPadding.top,
right: viewInsetsPadding.right,
),
child: ViewerDetailOverlay(
index: index,
entries: entries,
hasCollection: hasCollection,
multiPageController: multiPageController,
expandedNotifier: expandedNotifier,
availableSize: availableSize,
),
),
),
),
),
Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
if (settings.showOverlayMinimap)
SafeArea(
top: !showInfo,
minimum: EdgeInsets.only(
left: viewInsetsPadding.left,
right: viewInsetsPadding.right,
),
child: Padding(
padding: const EdgeInsets.all(8),
child: FadeTransition(
opacity: scale,
child: Minimap(
viewStateNotifier: viewStateNotifier,
),
),
),
),
const Spacer(),
if (settings.overlayHistogramStyle != OverlayHistogramStyle.none)
SafeArea(
top: !showInfo,
minimum: EdgeInsets.only(
left: viewInsetsPadding.left,
right: viewInsetsPadding.right,
),
child: Padding(
padding: const EdgeInsets.all(8),
child: FadeTransition(
opacity: scale,
child: Selector<ViewStateConductor, ViewStateController>(
selector: (context, vsc) => vsc.getOrCreateController(pageEntry!),
builder: (context, viewStateController, child) {
return ValueListenableBuilder<ImageProvider?>(
valueListenable: viewStateController.fullImageNotifier,
builder: (context, fullImage, child) {
if (fullImage == null || pageEntry == null) return const SizedBox();
return ImageHistogram(
viewStateController: viewStateController,
image: fullImage,
);
},
);
},
),
),
),
),
],
),
],
);
},
);
}
}