aves/lib/widgets/viewer/panorama_page.dart
2022-05-12 17:32:51 +09:00

198 lines
7.1 KiB
Dart

import 'dart:math';
import 'package:aves/model/device.dart';
import 'package:aves/model/entry.dart';
import 'package:aves/model/entry_images.dart';
import 'package:aves/model/panorama.dart';
import 'package:aves/theme/icons.dart';
import 'package:aves/widgets/common/basic/insets.dart';
import 'package:aves/widgets/common/extensions/build_context.dart';
import 'package:aves/widgets/common/extensions/media_query.dart';
import 'package:aves/widgets/common/providers/media_query_data_provider.dart';
import 'package:aves/widgets/viewer/overlay/common.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/scheduler.dart';
import 'package:flutter/services.dart';
import 'package:latlong2/latlong.dart';
import 'package:panorama/panorama.dart';
import 'package:provider/provider.dart';
class PanoramaPage extends StatefulWidget {
static const routeName = '/viewer/panorama';
final AvesEntry entry;
final PanoramaInfo info;
const PanoramaPage({
Key? key,
required this.entry,
required this.info,
}) : super(key: key);
@override
State<PanoramaPage> createState() => _PanoramaPageState();
}
class _PanoramaPageState extends State<PanoramaPage> {
final ValueNotifier<bool> _overlayVisible = ValueNotifier(true);
final ValueNotifier<SensorControl> _sensorControl = ValueNotifier(SensorControl.None);
AvesEntry get entry => widget.entry;
PanoramaInfo get info => widget.info;
@override
void initState() {
super.initState();
_overlayVisible.addListener(_onOverlayVisibleChange);
WidgetsBinding.instance.addPostFrameCallback((_) => _initOverlay());
}
@override
void dispose() {
_overlayVisible.removeListener(_onOverlayVisibleChange);
super.dispose();
}
@override
Widget build(BuildContext context) {
return WillPopScope(
onWillPop: () {
_onLeave();
return SynchronousFuture(true);
},
child: MediaQueryDataProvider(
child: Scaffold(
body: Stack(
children: [
ValueListenableBuilder<SensorControl>(
valueListenable: _sensorControl,
builder: (context, sensorControl, child) {
void onTap(longitude, latitude, tilt) => _overlayVisible.value = !_overlayVisible.value;
final imageChild = child as Image;
if (info.hasCroppedArea) {
final croppedArea = info.croppedAreaRect!;
final fullSize = info.fullPanoSize!;
final longitude = ((croppedArea.left + croppedArea.width / 2) / fullSize.width - 1 / 2) * 360;
return Panorama(
// TODO TLAD [panorama] fork and fix
// as of panorama v0.4.0, doc says `latitude` and `longitude` parameters are in degrees,
// but they are actually converted from radians in state initialization
// as of panorama v0.4.0, state uses longitude in degrees as radians with `Quaternion.axisAngle`
longitude: degToRadian(longitude),
sensorControl: sensorControl,
croppedArea: croppedArea,
croppedFullWidth: fullSize.width,
croppedFullHeight: fullSize.height,
onTap: onTap,
child: imageChild,
);
} else {
return Panorama(
sensorControl: sensorControl,
onTap: onTap,
child: imageChild,
);
}
},
child: Image(
image: entry.uriImage,
),
),
Positioned(
right: 0,
bottom: 0,
child: TooltipTheme(
data: TooltipTheme.of(context).copyWith(
preferBelow: false,
),
child: ValueListenableBuilder<bool>(
valueListenable: _overlayVisible,
builder: (context, overlayVisible, child) {
return Visibility(
visible: overlayVisible,
child: Selector<MediaQueryData, double>(
selector: (context, mq) => max(mq.effectiveBottomPadding, mq.systemGestureInsets.bottom),
builder: (context, mqPaddingBottom, child) {
return SafeArea(
bottom: false,
child: Padding(
padding: const EdgeInsets.all(8) + EdgeInsets.only(bottom: mqPaddingBottom),
child: child,
),
);
},
child: OverlayButton(
child: ValueListenableBuilder<SensorControl>(
valueListenable: _sensorControl,
builder: (context, sensorControl, child) {
return IconButton(
icon: Icon(sensorControl == SensorControl.None ? AIcons.sensorControlEnabled : AIcons.sensorControlDisabled),
onPressed: _toggleSensor,
tooltip: sensorControl == SensorControl.None ? context.l10n.panoramaEnableSensorControl : context.l10n.panoramaDisableSensorControl,
);
},
),
),
),
);
},
),
),
),
const BottomGestureAreaProtector(),
],
),
resizeToAvoidBottomInset: false,
),
),
);
}
void _toggleSensor() {
switch (_sensorControl.value) {
case SensorControl.None:
_sensorControl.value = SensorControl.AbsoluteOrientation;
break;
case SensorControl.AbsoluteOrientation:
case SensorControl.Orientation:
_sensorControl.value = SensorControl.None;
break;
}
}
void _onLeave() {
_showSystemUI();
}
// system UI
static void _showSystemUI() {
if (device.supportEdgeToEdgeUIMode) {
SystemChrome.setEnabledSystemUIMode(SystemUiMode.edgeToEdge);
} else {
SystemChrome.setEnabledSystemUIMode(SystemUiMode.manual, overlays: SystemUiOverlay.values);
}
}
static void _hideSystemUI() => SystemChrome.setEnabledSystemUIMode(SystemUiMode.immersive);
// overlay
Future<void> _initOverlay() async {
// wait for MaterialPageRoute.transitionDuration
// to show overlay after page animation is complete
await Future.delayed(ModalRoute.of(context)!.transitionDuration * timeDilation);
await _onOverlayVisibleChange();
}
Future<void> _onOverlayVisibleChange() async {
if (_overlayVisible.value) {
_showSystemUI();
} else {
_hideSystemUI();
}
}
}