|
@@ -1,93 +1,153 @@
|
|
import 'dart:io';
|
|
import 'dart:io';
|
|
|
|
+import 'dart:math';
|
|
|
|
|
|
import 'package:camera/camera.dart';
|
|
import 'package:camera/camera.dart';
|
|
|
|
+import 'package:camera/new/src/support_android/camera.dart';
|
|
import 'package:douyin_demo/providers/CameraProvider.dart';
|
|
import 'package:douyin_demo/providers/CameraProvider.dart';
|
|
|
|
+import 'package:firebase_ml_vision/firebase_ml_vision.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:image_gallery_saver/image_gallery_saver.dart';
|
|
import 'package:image_gallery_saver/image_gallery_saver.dart';
|
|
import 'package:provider/provider.dart';
|
|
import 'package:provider/provider.dart';
|
|
import 'package:path/path.dart' as p;
|
|
import 'package:path/path.dart' as p;
|
|
import 'package:image_picker_saver/image_picker_saver.dart';
|
|
import 'package:image_picker_saver/image_picker_saver.dart';
|
|
-import 'dart:core';
|
|
|
|
|
|
|
|
class CameraPage extends StatelessWidget {
|
|
class CameraPage extends StatelessWidget {
|
|
const CameraPage({Key key}) : super(key: key);
|
|
const CameraPage({Key key}) : super(key: key);
|
|
|
|
|
|
@override
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
Widget build(BuildContext context) {
|
|
- double rpx = MediaQuery.of(context).size.width / 750;
|
|
|
|
- return MultiProvider(
|
|
|
|
- providers: [ChangeNotifierProvider(builder: (_) => CameraProvider())],
|
|
|
|
- child: Scaffold(
|
|
|
|
- backgroundColor: Theme.of(context).primaryColor,
|
|
|
|
- body: CameraMain(),
|
|
|
|
- // bottomNavigationBar: SafeArea(
|
|
|
|
- // child: Container(
|
|
|
|
- // height: 100*rpx,
|
|
|
|
- // child: BottomTab()
|
|
|
|
- // )
|
|
|
|
- // ),
|
|
|
|
- ));
|
|
|
|
|
|
+ return Scaffold(
|
|
|
|
+ backgroundColor: Theme.of(context).primaryColor,
|
|
|
|
+ body: MultiProvider(
|
|
|
|
+ providers: [
|
|
|
|
+ ChangeNotifierProvider(
|
|
|
|
+ builder: (_) => CameraProvider(),
|
|
|
|
+ )
|
|
|
|
+ ],
|
|
|
|
+ child: CameraMain(
|
|
|
|
+ rpx: MediaQuery.of(context).size.width / 750,
|
|
|
|
+ )),
|
|
|
|
+ bottomNavigationBar: BottomAppBar(),
|
|
|
|
+ );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-// class CameraMain extends StatefulWidget {
|
|
|
|
-// CameraMain({Key key}) : super(key: key);
|
|
|
|
|
|
+class CameraMain extends StatefulWidget {
|
|
|
|
+ CameraMain({Key key, @required this.rpx}) : super(key: key);
|
|
|
|
+ final double rpx;
|
|
|
|
+ @override
|
|
|
|
+ _CameraMainState createState() => _CameraMainState();
|
|
|
|
+}
|
|
|
|
|
|
-// _CameraMainState createState() => _CameraMainState();
|
|
|
|
-// }
|
|
|
|
|
|
+class _CameraMainState extends State<CameraMain> {
|
|
|
|
+ CameraProvider provider;
|
|
|
|
+ double rpx;
|
|
|
|
+ double toTop;
|
|
|
|
+ double outBox;
|
|
|
|
+ double innerBox;
|
|
|
|
+ CameraController _controller;
|
|
|
|
+ bool findFace = false;
|
|
|
|
+ var cameras;
|
|
|
|
+ @override
|
|
|
|
+ void initState() {
|
|
|
|
+ super.initState();
|
|
|
|
|
|
-// class _CameraMainState extends State<CameraMain> {
|
|
|
|
-// var cameras;
|
|
|
|
-// CameraController _controller;
|
|
|
|
-// @override
|
|
|
|
-// initState() {
|
|
|
|
-// // TODO: implement initState
|
|
|
|
-// super.initState();
|
|
|
|
-// initCamera();
|
|
|
|
-// }
|
|
|
|
|
|
+ // provider = Provider.of<CameraProvider>(context);
|
|
|
|
+ getCameras();
|
|
|
|
+ rpx = widget.rpx;
|
|
|
|
+ toTop = 100 * rpx;
|
|
|
|
+ outBox = 170 * rpx;
|
|
|
|
+ innerBox = 130 * rpx;
|
|
|
|
+ }
|
|
|
|
|
|
-// Future initCamera() async {
|
|
|
|
-// cameras = await availableCameras();
|
|
|
|
-// _controller = CameraController(cameras[0], ResolutionPreset.medium);
|
|
|
|
-// _controller.initialize().then((_) {
|
|
|
|
-// if (!mounted) {
|
|
|
|
-// return;
|
|
|
|
-// }
|
|
|
|
-// setState(() {});
|
|
|
|
-// });
|
|
|
|
|
|
+ getCameras() async {
|
|
|
|
+ cameras = await availableCameras();
|
|
|
|
+ _controller = CameraController(cameras[1], ResolutionPreset.medium);
|
|
|
|
|
|
-// }
|
|
|
|
|
|
+ _controller.initialize().then((_) {
|
|
|
|
+ if (!mounted) {
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+ _controller.startImageStream((CameraImage availableImage) {
|
|
|
|
+ _controller.stopImageStream();
|
|
|
|
+ _scanFrame(availableImage);
|
|
|
|
+ });
|
|
|
|
|
|
-// @override
|
|
|
|
-// void dispose() {
|
|
|
|
-// // TODO: implement dispose
|
|
|
|
-// _controller?.dispose();
|
|
|
|
-// super.dispose();
|
|
|
|
-// }
|
|
|
|
|
|
+ setState(() {});
|
|
|
|
+ });
|
|
|
|
+ }
|
|
|
|
|
|
-// @override
|
|
|
|
-// Widget build(BuildContext context) {
|
|
|
|
|
|
+ void _scanFrame(CameraImage availableImage) async {
|
|
|
|
+
|
|
|
|
+ final FirebaseVisionImageMetadata metadata = FirebaseVisionImageMetadata(
|
|
|
|
+ rawFormat: availableImage.format.raw,
|
|
|
|
+ size: Size(
|
|
|
|
+ availableImage.width.toDouble(), availableImage.height.toDouble()),
|
|
|
|
+ planeData: availableImage.planes
|
|
|
|
+ .map((currentPlane) => FirebaseVisionImagePlaneMetadata(
|
|
|
|
+ bytesPerRow: currentPlane.bytesPerRow,
|
|
|
|
+ height: currentPlane.height,
|
|
|
|
+ width: currentPlane.width))
|
|
|
|
+ .toList(),
|
|
|
|
+ rotation: ImageRotation.rotation90);
|
|
|
|
+ final FirebaseVisionImage visionImage =
|
|
|
|
+ FirebaseVisionImage.fromBytes(availableImage.planes[0].bytes, metadata);
|
|
|
|
+ final FaceDetector detector = FirebaseVision.instance.faceDetector();
|
|
|
|
+ final List<Face> faces = await detector.processImage(visionImage);
|
|
|
|
|
|
-// }
|
|
|
|
-// }
|
|
|
|
|
|
+ if (faces.length > 0) {
|
|
|
|
+ setState(() {
|
|
|
|
+ findFace = true;
|
|
|
|
+ _controller.startImageStream((CameraImage availableImage) {
|
|
|
|
+ _controller.stopImageStream();
|
|
|
|
+ _scanFrame(availableImage);
|
|
|
|
+ });
|
|
|
|
+ });
|
|
|
|
+ } else {
|
|
|
|
+ setState(() {
|
|
|
|
+ findFace = false;
|
|
|
|
+ _controller.startImageStream((CameraImage availableImage) {
|
|
|
|
+ _controller.stopImageStream();
|
|
|
|
+ _scanFrame(availableImage);
|
|
|
|
+ });
|
|
|
|
+ });
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // for (TextBlock block in visionText.blocks) {
|
|
|
|
+
|
|
|
|
+ // // final Rectangle<int> boundingBox = block.boundingBox;
|
|
|
|
+ // // final List<Point<int>> cornerPoints = block.cornerPoints;
|
|
|
|
+ // print(block.text);
|
|
|
|
+ // final List<RecognizedLanguage> languages = block.recognizedLanguages;
|
|
|
|
|
|
-class CameraMain extends StatelessWidget {
|
|
|
|
- const CameraMain({Key key}) : super(key: key);
|
|
|
|
|
|
+ // for (TextLine line in block.lines) {
|
|
|
|
+ // // Same getters as TextBlock
|
|
|
|
+ // print(line.text);
|
|
|
|
+ // for (TextElement element in line.elements) {
|
|
|
|
+ // // Same getters as TextBlock
|
|
|
|
+ // print(element.text);
|
|
|
|
+ // }
|
|
|
|
+ // }
|
|
|
|
+ // }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @override
|
|
|
|
+ void dispose() {
|
|
|
|
+ // TODO: implement dispose
|
|
|
|
+ _controller.dispose();
|
|
|
|
+ super.dispose();
|
|
|
|
+ }
|
|
|
|
|
|
@override
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
Widget build(BuildContext context) {
|
|
- CameraProvider provider = Provider.of<CameraProvider>(context);
|
|
|
|
- if (provider == null || provider.cameraController == null) {
|
|
|
|
|
|
+ provider = Provider.of<CameraProvider>(context);
|
|
|
|
+ if (provider == null || _controller == null) {
|
|
return Container(
|
|
return Container(
|
|
child: Center(child: CircularProgressIndicator()),
|
|
child: Center(child: CircularProgressIndicator()),
|
|
);
|
|
);
|
|
}
|
|
}
|
|
- double rpx = MediaQuery.of(context).size.width / 750;
|
|
|
|
- double toTop = 100 * rpx;
|
|
|
|
- double outBox = 170 * rpx;
|
|
|
|
- double innerBox = 130 * rpx;
|
|
|
|
- CameraController _controller = provider.cameraController;
|
|
|
|
- var cameras = provider.cameras;
|
|
|
|
|
|
+
|
|
|
|
+ bool ifMakeVideo = provider.ifMakeVideo;
|
|
if (_controller == null || _controller?.value == null) {
|
|
if (_controller == null || _controller?.value == null) {
|
|
return Container(
|
|
return Container(
|
|
child: Center(child: CircularProgressIndicator()),
|
|
child: Center(child: CircularProgressIndicator()),
|
|
@@ -96,6 +156,12 @@ class CameraMain extends StatelessWidget {
|
|
final size = MediaQuery.of(context).size;
|
|
final size = MediaQuery.of(context).size;
|
|
return _controller.value.isInitialized
|
|
return _controller.value.isInitialized
|
|
? Stack(children: <Widget>[
|
|
? Stack(children: <Widget>[
|
|
|
|
+ // Camera.open(cameraId),
|
|
|
|
+ findFace? Positioned(
|
|
|
|
+ top: 0,
|
|
|
|
+ left: 0,
|
|
|
|
+ child: Container(width: 100,height: 100,color: Colors.red,)
|
|
|
|
+ ):Container(),
|
|
ClipRect(
|
|
ClipRect(
|
|
child: Transform.scale(
|
|
child: Transform.scale(
|
|
scale: _controller.value.aspectRatio / size.aspectRatio,
|
|
scale: _controller.value.aspectRatio / size.aspectRatio,
|
|
@@ -149,56 +215,49 @@ class CameraMain extends StatelessWidget {
|
|
),
|
|
),
|
|
Positioned(
|
|
Positioned(
|
|
//拍照按钮
|
|
//拍照按钮
|
|
- bottom: 100 * rpx,
|
|
|
|
|
|
+ bottom: 140 * rpx,
|
|
// left: (750*rpx-outBox)/2,
|
|
// left: (750*rpx-outBox)/2,
|
|
child: Container(
|
|
child: Container(
|
|
width: 750 * rpx,
|
|
width: 750 * rpx,
|
|
child: Row(
|
|
child: Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceAround,
|
|
mainAxisAlignment: MainAxisAlignment.spaceAround,
|
|
children: [
|
|
children: [
|
|
- provider.ifMakeVideo
|
|
|
|
- ? Container(width: 80*rpx,)
|
|
|
|
|
|
+ ifMakeVideo
|
|
|
|
+ ? Container(
|
|
|
|
+ width: 80 * rpx,
|
|
|
|
+ )
|
|
: IconWithText(
|
|
: IconWithText(
|
|
icon: Icon(
|
|
icon: Icon(
|
|
Icons.search,
|
|
Icons.search,
|
|
color: Colors.white,
|
|
color: Colors.white,
|
|
),
|
|
),
|
|
text: "道具"),
|
|
text: "道具"),
|
|
- AnimatedSwitcher(
|
|
|
|
- duration: Duration(milliseconds: 300),
|
|
|
|
- child: provider.ifMakeVideo
|
|
|
|
- ? VideoButtonAnim(
|
|
|
|
- provider: provider,
|
|
|
|
- rpx: rpx,
|
|
|
|
- outWidth: outBox,
|
|
|
|
- innerWidth: innerBox - 40 * rpx,
|
|
|
|
- )
|
|
|
|
- : CircleTakePhoto(
|
|
|
|
- outBox: outBox,
|
|
|
|
- innerBox: innerBox,
|
|
|
|
- ),
|
|
|
|
- ),
|
|
|
|
- provider.ifMakeVideo
|
|
|
|
- ? Container(
|
|
|
|
- child: IconButton(
|
|
|
|
- padding: EdgeInsets.all(0),
|
|
|
|
- icon: Icon(
|
|
|
|
- Icons.check_circle,
|
|
|
|
- color: Color.fromARGB(128, 219, 48, 85),
|
|
|
|
- size: 80*rpx,
|
|
|
|
- ),
|
|
|
|
- onPressed: () async {
|
|
|
|
- provider.cameraController
|
|
|
|
- .stopVideoRecording();
|
|
|
|
- // await ImagePickerSaver.saveFile(
|
|
|
|
- // fileData: File(provider.fileName)
|
|
|
|
- // .readAsBytesSync());
|
|
|
|
-
|
|
|
|
- await ImageGallerySaver.saveFile(provider.fileName);
|
|
|
|
- File(provider.fileName).deleteSync();
|
|
|
|
- provider.changePhotoWidget();
|
|
|
|
- },
|
|
|
|
|
|
+ ifMakeVideo
|
|
|
|
+ ? AnimVideoButton(
|
|
|
|
+ rpx: rpx,
|
|
|
|
+ outWidth: outBox,
|
|
|
|
+ innerWidth: innerBox - 30 * rpx,
|
|
|
|
+ provider: provider,
|
|
|
|
+ )
|
|
|
|
+ : CircleTakePhoto(
|
|
|
|
+ outBox: outBox,
|
|
|
|
+ innerBox: innerBox,
|
|
|
|
+ ),
|
|
|
|
+ ifMakeVideo
|
|
|
|
+ ? IconButton(
|
|
|
|
+ padding: EdgeInsets.all(0),
|
|
|
|
+ icon: Icon(
|
|
|
|
+ Icons.check_circle,
|
|
|
|
+ color: Color.fromARGB(255, 219, 48, 85),
|
|
|
|
+ size: 80 * rpx,
|
|
),
|
|
),
|
|
|
|
+ onPressed: () async {
|
|
|
|
+ provider.cameraController
|
|
|
|
+ .stopVideoRecording();
|
|
|
|
+ await ImageGallerySaver.saveFile(
|
|
|
|
+ provider.fileName);
|
|
|
|
+ File(provider.fileName).delete();
|
|
|
|
+ },
|
|
)
|
|
)
|
|
: IconWithText(
|
|
: IconWithText(
|
|
icon: Icon(
|
|
icon: Icon(
|
|
@@ -208,6 +267,12 @@ class CameraMain extends StatelessWidget {
|
|
text: "道具"),
|
|
text: "道具"),
|
|
])),
|
|
])),
|
|
),
|
|
),
|
|
|
|
+ Positioned(
|
|
|
|
+ bottom: 40 * rpx,
|
|
|
|
+ child: ScrollBottomBar(
|
|
|
|
+ rpx: rpx,
|
|
|
|
+ ),
|
|
|
|
+ ),
|
|
Positioned(
|
|
Positioned(
|
|
right: 30 * rpx,
|
|
right: 30 * rpx,
|
|
top: 80 * rpx,
|
|
top: 80 * rpx,
|
|
@@ -216,167 +281,193 @@ class CameraMain extends StatelessWidget {
|
|
onPressed: () {
|
|
onPressed: () {
|
|
provider.changeCamera();
|
|
provider.changeCamera();
|
|
}),
|
|
}),
|
|
- ),
|
|
|
|
- provider.ifMakeVideo//底部导航栏
|
|
|
|
- ? Container()
|
|
|
|
- : Positioned(
|
|
|
|
- bottom: 0,
|
|
|
|
- left: 0,
|
|
|
|
- child: ScrollTabBar(
|
|
|
|
- rpx: rpx,
|
|
|
|
- ),
|
|
|
|
- )
|
|
|
|
|
|
+ )
|
|
])
|
|
])
|
|
: Container();
|
|
: Container();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-class ScrollTabBar extends StatefulWidget {
|
|
|
|
- ScrollTabBar({Key key, @required this.rpx}) : super(key: key);
|
|
|
|
- final double rpx;
|
|
|
|
- _ScrollTabBarState createState() => _ScrollTabBarState();
|
|
|
|
-}
|
|
|
|
|
|
+// class CameraMain extends StatelessWidget {
|
|
|
|
+// const CameraMain({Key key}) : super(key: key);
|
|
|
|
|
|
-class _ScrollTabBarState extends State<ScrollTabBar>
|
|
|
|
- with AutomaticKeepAliveClientMixin {
|
|
|
|
- double curCenter = 0;
|
|
|
|
- ScrollController controller;
|
|
|
|
- List<String> items = ['拍照', '拍15秒', '拍60秒', '影集', '直播'];
|
|
|
|
- double eachWidth;
|
|
|
|
- double eachSide;
|
|
|
|
- double rpx;
|
|
|
|
- double maxPos;
|
|
|
|
- double minPos;
|
|
|
|
- double startDx = 0;
|
|
|
|
- double finalDx = 0;
|
|
|
|
- int curIndex = 2;
|
|
|
|
- @override
|
|
|
|
- void initState() {
|
|
|
|
- super.initState();
|
|
|
|
- rpx = widget.rpx;
|
|
|
|
- eachWidth = 130 * rpx;
|
|
|
|
- eachSide = (750 - eachWidth / rpx) / 2 * rpx;
|
|
|
|
- curCenter = curIndex * eachWidth;
|
|
|
|
- maxPos = eachWidth * items.length;
|
|
|
|
- minPos = 0;
|
|
|
|
- controller = ScrollController(initialScrollOffset: curCenter);
|
|
|
|
- }
|
|
|
|
|
|
+// @override
|
|
|
|
+// Widget build(BuildContext context) {
|
|
|
|
+// CameraProvider provider = Provider.of<CameraProvider>(context);
|
|
|
|
+// if (provider == null || provider.cameraController == null) {
|
|
|
|
+// return Container(
|
|
|
|
+// child: Center(child: CircularProgressIndicator()),
|
|
|
|
+// );
|
|
|
|
+// }
|
|
|
|
+// double rpx = MediaQuery.of(context).size.width / 750;
|
|
|
|
+// double toTop = 100 * rpx;
|
|
|
|
+// double outBox = 170 * rpx;
|
|
|
|
+// double innerBox = 130 * rpx;
|
|
|
|
+// CameraController _controller = provider.cameraController;
|
|
|
|
+// var cameras = provider.cameras;
|
|
|
|
|
|
- moveToItem(index) {
|
|
|
|
- controller.animateTo(index * eachWidth,
|
|
|
|
- duration: Duration(milliseconds: 200), curve: Curves.linearToEaseOut);
|
|
|
|
- setState(() {
|
|
|
|
- curCenter = index * eachWidth;
|
|
|
|
- curIndex = index;
|
|
|
|
- });
|
|
|
|
- }
|
|
|
|
|
|
+// bool ifMakeVideo = provider.ifMakeVideo;
|
|
|
|
+// if (_controller == null || _controller?.value == null) {
|
|
|
|
+// return Container(
|
|
|
|
+// child: Center(child: CircularProgressIndicator()),
|
|
|
|
+// );
|
|
|
|
+// }
|
|
|
|
+// final size = MediaQuery.of(context).size;
|
|
|
|
+// return _controller.value.isInitialized
|
|
|
|
+// ? Stack(children: <Widget>[
|
|
|
|
+// // Camera.open(cameraId),
|
|
|
|
|
|
- @override
|
|
|
|
- Widget build(BuildContext context) {
|
|
|
|
- return Listener(
|
|
|
|
- onPointerDown: (result) {
|
|
|
|
- setState(() {
|
|
|
|
- startDx = result.position.dx;
|
|
|
|
- });
|
|
|
|
- },
|
|
|
|
- onPointerUp: (result) {
|
|
|
|
- finalDx = result.position.dx;
|
|
|
|
- double finalPosition = curCenter + startDx - finalDx;
|
|
|
|
- int index = (finalPosition / eachWidth).floor();
|
|
|
|
- // print('curCenter=$index,moveTo:$index');
|
|
|
|
- moveToItem(index);
|
|
|
|
- },
|
|
|
|
- child: Container(
|
|
|
|
- width: 750 * rpx,
|
|
|
|
- child: Column(children: [
|
|
|
|
- SingleChildScrollView(
|
|
|
|
- scrollDirection: Axis.horizontal,
|
|
|
|
- controller: controller,
|
|
|
|
- child: Column(
|
|
|
|
- mainAxisAlignment: MainAxisAlignment.start,
|
|
|
|
- mainAxisSize: MainAxisSize.min,
|
|
|
|
- children: [
|
|
|
|
- Row(
|
|
|
|
- children: <Widget>[
|
|
|
|
- Container(
|
|
|
|
- width: eachSide,
|
|
|
|
- ),
|
|
|
|
- Row(
|
|
|
|
- children: List.generate(
|
|
|
|
- items.length,
|
|
|
|
- (index) => Container(
|
|
|
|
- width: eachWidth,
|
|
|
|
- child: FlatButton(
|
|
|
|
- padding: EdgeInsets.all(0),
|
|
|
|
- child: Text(
|
|
|
|
- items[index],
|
|
|
|
- style: TextStyle(
|
|
|
|
- fontSize: 28 * rpx,
|
|
|
|
- color: curIndex == index
|
|
|
|
- ? Colors.white
|
|
|
|
- : Colors.white
|
|
|
|
- .withOpacity(0.3)),
|
|
|
|
- ),
|
|
|
|
- onPressed: () {
|
|
|
|
- moveToItem(index);
|
|
|
|
- },
|
|
|
|
- ),
|
|
|
|
- )),
|
|
|
|
- ),
|
|
|
|
- Container(
|
|
|
|
- width: eachSide,
|
|
|
|
- ),
|
|
|
|
- ],
|
|
|
|
- ),
|
|
|
|
- ]),
|
|
|
|
- ),
|
|
|
|
- Container(
|
|
|
|
- height: 8 * rpx,
|
|
|
|
- width: 750 * rpx,
|
|
|
|
- child: Center(
|
|
|
|
- child: Container(
|
|
|
|
- decoration: BoxDecoration(
|
|
|
|
- shape: BoxShape.circle, color: Colors.white),
|
|
|
|
- )),
|
|
|
|
- ),
|
|
|
|
- SizedBox(
|
|
|
|
- height: 10 * rpx,
|
|
|
|
- )
|
|
|
|
- ])));
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- @override
|
|
|
|
- // TODO: implement wantKeepAlive
|
|
|
|
- bool get wantKeepAlive => true;
|
|
|
|
-}
|
|
|
|
|
|
+// ClipRect(
|
|
|
|
+// child: Transform.scale(
|
|
|
|
+// scale: _controller.value.aspectRatio / size.aspectRatio,
|
|
|
|
+// child: Center(
|
|
|
|
+// child: AspectRatio(
|
|
|
|
+// aspectRatio: _controller.value.aspectRatio,
|
|
|
|
+// child: CameraPreview(_controller),
|
|
|
|
+// ),
|
|
|
|
+// ),
|
|
|
|
+// )),
|
|
|
|
+// Positioned(
|
|
|
|
+// //顶部关闭按钮
|
|
|
|
+// top: toTop,
|
|
|
|
+// left: 30 * rpx,
|
|
|
|
+// child: IconButton(
|
|
|
|
+// icon: Icon(
|
|
|
|
+// Icons.close,
|
|
|
|
+// color: Colors.white,
|
|
|
|
+// size: 60 * rpx,
|
|
|
|
+// ),
|
|
|
|
+// onPressed: () {
|
|
|
|
+// Navigator.pop(context);
|
|
|
|
+// },
|
|
|
|
+// ),
|
|
|
|
+// ),
|
|
|
|
+// Positioned(
|
|
|
|
+// //选择音乐
|
|
|
|
+// top: toTop,
|
|
|
|
+// left: 250 * rpx,
|
|
|
|
+// child: Container(
|
|
|
|
+// width: 250 * rpx,
|
|
|
|
+// child: FlatButton(
|
|
|
|
+// onPressed: () {},
|
|
|
|
+// child: Row(
|
|
|
|
+// children: <Widget>[
|
|
|
|
+// Icon(
|
|
|
|
+// Icons.music_note,
|
|
|
|
+// color: Colors.white,
|
|
|
|
+// ),
|
|
|
|
+// SizedBox(
|
|
|
|
+// width: 10 * rpx,
|
|
|
|
+// ),
|
|
|
|
+// Text(
|
|
|
|
+// "选择音乐",
|
|
|
|
+// style: TextStyle(color: Colors.white),
|
|
|
|
+// ),
|
|
|
|
+// ],
|
|
|
|
+// ),
|
|
|
|
+// ),
|
|
|
|
+// ),
|
|
|
|
+// ),
|
|
|
|
+// Positioned(
|
|
|
|
+// //拍照按钮
|
|
|
|
+// bottom: 140 * rpx,
|
|
|
|
+// // left: (750*rpx-outBox)/2,
|
|
|
|
+// child: Container(
|
|
|
|
+// width: 750 * rpx,
|
|
|
|
+// child: Row(
|
|
|
|
+// mainAxisAlignment: MainAxisAlignment.spaceAround,
|
|
|
|
+// children: [
|
|
|
|
+// ifMakeVideo
|
|
|
|
+// ? Container(
|
|
|
|
+// width: 80 * rpx,
|
|
|
|
+// )
|
|
|
|
+// : IconWithText(
|
|
|
|
+// icon: Icon(
|
|
|
|
+// Icons.search,
|
|
|
|
+// color: Colors.white,
|
|
|
|
+// ),
|
|
|
|
+// text: "道具"),
|
|
|
|
+// ifMakeVideo
|
|
|
|
+// ? AnimVideoButton(
|
|
|
|
+// rpx: rpx,
|
|
|
|
+// outWidth: outBox,
|
|
|
|
+// innerWidth: innerBox - 30 * rpx,
|
|
|
|
+// provider: provider,
|
|
|
|
+// )
|
|
|
|
+// : CircleTakePhoto(
|
|
|
|
+// outBox: outBox,
|
|
|
|
+// innerBox: innerBox,
|
|
|
|
+// ),
|
|
|
|
+// ifMakeVideo
|
|
|
|
+// ? IconButton(
|
|
|
|
+// padding: EdgeInsets.all(0),
|
|
|
|
+// icon: Icon(
|
|
|
|
+// Icons.check_circle,
|
|
|
|
+// color: Color.fromARGB(255, 219, 48, 85),
|
|
|
|
+// size: 80 * rpx,
|
|
|
|
+// ),
|
|
|
|
+// onPressed: () async {
|
|
|
|
+// provider.cameraController
|
|
|
|
+// .stopVideoRecording();
|
|
|
|
+// await ImageGallerySaver.saveFile(
|
|
|
|
+// provider.fileName);
|
|
|
|
+// File(provider.fileName).delete();
|
|
|
|
+// },
|
|
|
|
+// )
|
|
|
|
+// : IconWithText(
|
|
|
|
+// icon: Icon(
|
|
|
|
+// Icons.search,
|
|
|
|
+// color: Colors.white,
|
|
|
|
+// ),
|
|
|
|
+// text: "道具"),
|
|
|
|
+// ])),
|
|
|
|
+// ),
|
|
|
|
+// Positioned(
|
|
|
|
+// bottom: 40 * rpx,
|
|
|
|
+// child: ScrollBottomBar(
|
|
|
|
+// rpx: rpx,
|
|
|
|
+// ),
|
|
|
|
+// ),
|
|
|
|
+// Positioned(
|
|
|
|
+// right: 30 * rpx,
|
|
|
|
+// top: 80 * rpx,
|
|
|
|
+// child: IconButton(
|
|
|
|
+// icon: Icon(Icons.camera_front),
|
|
|
|
+// onPressed: () {
|
|
|
|
+// provider.changeCamera();
|
|
|
|
+// }),
|
|
|
|
+// )
|
|
|
|
+// ])
|
|
|
|
+// : Container();
|
|
|
|
+// }
|
|
|
|
+// }
|
|
|
|
|
|
-class VideoButtonAnim extends StatefulWidget {
|
|
|
|
- VideoButtonAnim(
|
|
|
|
|
|
+class AnimVideoButton extends StatefulWidget {
|
|
|
|
+ AnimVideoButton(
|
|
{Key key,
|
|
{Key key,
|
|
- @required this.rpx,
|
|
|
|
@required this.outWidth,
|
|
@required this.outWidth,
|
|
@required this.innerWidth,
|
|
@required this.innerWidth,
|
|
|
|
+ @required this.rpx,
|
|
@required this.provider})
|
|
@required this.provider})
|
|
: super(key: key);
|
|
: super(key: key);
|
|
- final double rpx;
|
|
|
|
final double outWidth;
|
|
final double outWidth;
|
|
final double innerWidth;
|
|
final double innerWidth;
|
|
|
|
+ final double rpx;
|
|
final CameraProvider provider;
|
|
final CameraProvider provider;
|
|
- _VideoButtonAnimState createState() => _VideoButtonAnimState();
|
|
|
|
|
|
+ _AnimVideoButtonState createState() => _AnimVideoButtonState();
|
|
}
|
|
}
|
|
|
|
|
|
-class _VideoButtonAnimState extends State<VideoButtonAnim>
|
|
|
|
|
|
+class _AnimVideoButtonState extends State<AnimVideoButton>
|
|
with TickerProviderStateMixin {
|
|
with TickerProviderStateMixin {
|
|
- double extraPadding = 0; //内外圆之间距离
|
|
|
|
- double borderWidth = 0;
|
|
|
|
- double outWidth = 0;
|
|
|
|
- double innerWidth = 0;
|
|
|
|
- double minDist = 0;
|
|
|
|
- double rpx = 0;
|
|
|
|
Animation<double> animation;
|
|
Animation<double> animation;
|
|
AnimationController controller;
|
|
AnimationController controller;
|
|
|
|
+ double outWidth;
|
|
|
|
+ double innerWidth;
|
|
|
|
+ double outBorder;
|
|
|
|
+ double rpx;
|
|
|
|
+ double maxBorder;
|
|
|
|
+ bool ifRecording;
|
|
CameraProvider provider;
|
|
CameraProvider provider;
|
|
- bool ifPauseVideo = false;
|
|
|
|
|
|
+ double curBorder;
|
|
@override
|
|
@override
|
|
void dispose() {
|
|
void dispose() {
|
|
// TODO: implement dispose
|
|
// TODO: implement dispose
|
|
@@ -387,43 +478,41 @@ class _VideoButtonAnimState extends State<VideoButtonAnim>
|
|
@override
|
|
@override
|
|
void initState() {
|
|
void initState() {
|
|
super.initState();
|
|
super.initState();
|
|
- rpx = widget.rpx;
|
|
|
|
|
|
+ ifRecording = true;
|
|
|
|
+ provider = widget.provider;
|
|
outWidth = widget.outWidth;
|
|
outWidth = widget.outWidth;
|
|
innerWidth = widget.innerWidth;
|
|
innerWidth = widget.innerWidth;
|
|
- provider = widget.provider;
|
|
|
|
- minDist = 10 * rpx;
|
|
|
|
- extraPadding = (outWidth - innerWidth) / 2 - borderWidth;
|
|
|
|
- borderWidth = 15 * rpx;
|
|
|
|
|
|
+ rpx = widget.rpx;
|
|
|
|
+ outBorder = 5 * rpx;
|
|
|
|
+ maxBorder = (outWidth - innerWidth) / 2 - 10 * rpx;
|
|
|
|
+ curBorder = outBorder;
|
|
controller =
|
|
controller =
|
|
- AnimationController(duration: Duration(milliseconds: 800), vsync: this);
|
|
|
|
|
|
+ AnimationController(duration: Duration(milliseconds: 500), vsync: this);
|
|
animation =
|
|
animation =
|
|
- Tween(begin: (outWidth - innerWidth) / 2 - borderWidth, end: minDist)
|
|
|
|
- .animate(controller)
|
|
|
|
- ..addListener(() {
|
|
|
|
- setState(() {
|
|
|
|
- borderWidth = animation.value;
|
|
|
|
- extraPadding = (outWidth - innerWidth) / 2 - animation.value;
|
|
|
|
- });
|
|
|
|
- });
|
|
|
|
|
|
+ Tween<double>(begin: outBorder, end: maxBorder).animate(controller)
|
|
|
|
+ ..addListener(() {
|
|
|
|
+ setState(() {
|
|
|
|
+ curBorder = animation.value;
|
|
|
|
+ });
|
|
|
|
+ });
|
|
controller.repeat(reverse: true);
|
|
controller.repeat(reverse: true);
|
|
- // controller.forward(from: 0.0).then((f){
|
|
|
|
- // controller.reverse(from: 1.0);
|
|
|
|
- // });
|
|
|
|
}
|
|
}
|
|
|
|
|
|
- pauseAnimation() {
|
|
|
|
- controller.reset();
|
|
|
|
|
|
+ pauseRecording() {
|
|
|
|
+ // provider.cameraController.pauseVideoRecording();
|
|
|
|
+ controller.stop();
|
|
provider.cameraController.pauseVideoRecording();
|
|
provider.cameraController.pauseVideoRecording();
|
|
setState(() {
|
|
setState(() {
|
|
- ifPauseVideo = true;
|
|
|
|
|
|
+ ifRecording = false;
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
|
|
- playAnimation() {
|
|
|
|
|
|
+ resumeRecording() {
|
|
|
|
+ // provider.cameraController.resumeVideoRecording();
|
|
controller.repeat(reverse: true);
|
|
controller.repeat(reverse: true);
|
|
provider.cameraController.resumeVideoRecording();
|
|
provider.cameraController.resumeVideoRecording();
|
|
setState(() {
|
|
setState(() {
|
|
- ifPauseVideo = false;
|
|
|
|
|
|
+ ifRecording = true;
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
|
|
@@ -434,45 +523,165 @@ class _VideoButtonAnimState extends State<VideoButtonAnim>
|
|
height: outWidth,
|
|
height: outWidth,
|
|
decoration: BoxDecoration(
|
|
decoration: BoxDecoration(
|
|
shape: BoxShape.circle,
|
|
shape: BoxShape.circle,
|
|
|
|
+ color: Colors.transparent,
|
|
border: Border.all(
|
|
border: Border.all(
|
|
- width: borderWidth, color: Color.fromARGB(128, 219, 48, 85))),
|
|
|
|
- padding: EdgeInsets.all(extraPadding),
|
|
|
|
- child: Center(
|
|
|
|
- child: Container(
|
|
|
|
- width: innerWidth,
|
|
|
|
- height: innerWidth,
|
|
|
|
- // decoration: BoxDecoration(
|
|
|
|
- // color: Color.fromARGB(255, 219, 48, 85),
|
|
|
|
- // borderRadius: BorderRadius.circular(20 * rpx)),
|
|
|
|
- child: ifPauseVideo
|
|
|
|
- ? IconButton(
|
|
|
|
- padding: EdgeInsets.all(0),
|
|
|
|
- icon: Icon(
|
|
|
|
- Icons.pause_circle_filled,
|
|
|
|
- size: innerWidth,
|
|
|
|
- color: Color.fromARGB(255, 219, 48, 85),
|
|
|
|
- ),
|
|
|
|
- onPressed: () {
|
|
|
|
- playAnimation();
|
|
|
|
- },
|
|
|
|
- )
|
|
|
|
- : IconButton(
|
|
|
|
- padding: EdgeInsets.all(0),
|
|
|
|
- icon: Icon(
|
|
|
|
- Icons.play_circle_filled,
|
|
|
|
- size: innerWidth,
|
|
|
|
- color: Color.fromARGB(255, 219, 48, 85),
|
|
|
|
- ),
|
|
|
|
- onPressed: () {
|
|
|
|
- pauseAnimation();
|
|
|
|
- },
|
|
|
|
|
|
+ width: curBorder, color: Color.fromARGB(128, 219, 48, 85))),
|
|
|
|
+ child: Container(
|
|
|
|
+ child: !ifRecording
|
|
|
|
+ ? IconButton(
|
|
|
|
+ padding: EdgeInsets.all(0),
|
|
|
|
+ icon: Icon(
|
|
|
|
+ Icons.play_arrow,
|
|
|
|
+ size: innerWidth,
|
|
|
|
+ color: Color.fromARGB(255, 219, 48, 85),
|
|
),
|
|
),
|
|
- ),
|
|
|
|
|
|
+ onPressed: () {
|
|
|
|
+ resumeRecording();
|
|
|
|
+ },
|
|
|
|
+ )
|
|
|
|
+ : IconButton(
|
|
|
|
+ padding: EdgeInsets.all(0),
|
|
|
|
+ icon: Icon(
|
|
|
|
+ Icons.pause,
|
|
|
|
+ size: innerWidth,
|
|
|
|
+ color: Color.fromARGB(255, 219, 48, 85),
|
|
|
|
+ ),
|
|
|
|
+ onPressed: () {
|
|
|
|
+ pauseRecording();
|
|
|
|
+ },
|
|
|
|
+ ),
|
|
),
|
|
),
|
|
);
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+class ScrollBottomBar extends StatefulWidget {
|
|
|
|
+ ScrollBottomBar({Key key, @required this.rpx}) : super(key: key);
|
|
|
|
+ final double rpx;
|
|
|
|
+ _ScrollBottomBarState createState() => _ScrollBottomBarState();
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+class _ScrollBottomBarState extends State<ScrollBottomBar> {
|
|
|
|
+ double rpx;
|
|
|
|
+ double eachWidth;
|
|
|
|
+ double eachSide;
|
|
|
|
+ List<String> items;
|
|
|
|
+ ScrollController controller;
|
|
|
|
+ double startX = 0;
|
|
|
|
+ double finalX = 0;
|
|
|
|
+ double minValue;
|
|
|
|
+ double maxValue;
|
|
|
|
+ double curX;
|
|
|
|
+ int curIndex;
|
|
|
|
+
|
|
|
|
+ @override
|
|
|
|
+ void initState() {
|
|
|
|
+ super.initState();
|
|
|
|
+ rpx = widget.rpx;
|
|
|
|
+ eachWidth = 130 * rpx;
|
|
|
|
+ eachSide = (750 - eachWidth / rpx) / 2 * rpx;
|
|
|
|
+ curIndex = 2;
|
|
|
|
+ minValue = 0;
|
|
|
|
+
|
|
|
|
+ items = [
|
|
|
|
+ '拍照',
|
|
|
|
+ '拍15秒',
|
|
|
|
+ '拍60秒',
|
|
|
|
+ '影集',
|
|
|
|
+ '开直播',
|
|
|
|
+ ];
|
|
|
|
+ maxValue = (items.length - 1) * eachWidth;
|
|
|
|
+ curX = curIndex * eachWidth;
|
|
|
|
+ controller = ScrollController(initialScrollOffset: curX);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ moveToItem(index) {
|
|
|
|
+ curX = index * eachWidth;
|
|
|
|
+ controller.animateTo(curX,
|
|
|
|
+ duration: Duration(milliseconds: 200), curve: Curves.linear);
|
|
|
|
+ setState(() {
|
|
|
|
+ curX = curX;
|
|
|
|
+ curIndex = index;
|
|
|
|
+ });
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @override
|
|
|
|
+ Widget build(BuildContext context) {
|
|
|
|
+ return Column(children: [
|
|
|
|
+ Listener(
|
|
|
|
+ onPointerDown: (result) {
|
|
|
|
+ setState(() {
|
|
|
|
+ startX = result.position.dx;
|
|
|
|
+ });
|
|
|
|
+ },
|
|
|
|
+ onPointerMove: (result) {
|
|
|
|
+ double moveValue = result.position.dx;
|
|
|
|
+ double moved = startX - moveValue;
|
|
|
|
+ // curX+moved
|
|
|
|
+ double afterMoved = min(max(curX + moved, minValue), maxValue);
|
|
|
|
+ setState(() {
|
|
|
|
+ curX = afterMoved;
|
|
|
|
+ startX = result.position.dx;
|
|
|
|
+ });
|
|
|
|
+ },
|
|
|
|
+ onPointerUp: (result) {
|
|
|
|
+ int index = 0;
|
|
|
|
+ double finalPosition = curX - eachWidth / 2;
|
|
|
|
+ index = (finalPosition / eachWidth).ceil();
|
|
|
|
+ moveToItem(index);
|
|
|
|
+ },
|
|
|
|
+ child: Container(
|
|
|
|
+ width: 750 * rpx,
|
|
|
|
+ height: 100 * rpx,
|
|
|
|
+ child: SingleChildScrollView(
|
|
|
|
+ scrollDirection: Axis.horizontal,
|
|
|
|
+ controller: controller,
|
|
|
|
+ child: Container(
|
|
|
|
+ child: Row(
|
|
|
|
+ mainAxisSize: MainAxisSize.min,
|
|
|
|
+ children: <Widget>[
|
|
|
|
+ SizedBox(
|
|
|
|
+ width: eachSide,
|
|
|
|
+ ),
|
|
|
|
+ Row(
|
|
|
|
+ children: List.generate(items.length, (index) {
|
|
|
|
+ return Container(
|
|
|
|
+ width: eachWidth,
|
|
|
|
+ child: FlatButton(
|
|
|
|
+ child: Text(
|
|
|
|
+ items[index],
|
|
|
|
+ style: TextStyle(
|
|
|
|
+ color: curIndex == index
|
|
|
|
+ ? Colors.white
|
|
|
|
+ : Colors.white.withOpacity(0.5)),
|
|
|
|
+ ),
|
|
|
|
+ padding: EdgeInsets.all(0),
|
|
|
|
+ onPressed: () {
|
|
|
|
+ moveToItem(index);
|
|
|
|
+ },
|
|
|
|
+ ),
|
|
|
|
+ );
|
|
|
|
+ })),
|
|
|
|
+ SizedBox(
|
|
|
|
+ width: eachSide,
|
|
|
|
+ ),
|
|
|
|
+ ],
|
|
|
|
+ ),
|
|
|
|
+ ),
|
|
|
|
+ )),
|
|
|
|
+ ),
|
|
|
|
+ Center(
|
|
|
|
+ child: Container(
|
|
|
|
+ decoration:
|
|
|
|
+ BoxDecoration(shape: BoxShape.circle, color: Colors.white),
|
|
|
|
+ width: 8 * rpx,
|
|
|
|
+ height: 8 * rpx,
|
|
|
|
+ ),
|
|
|
|
+ )
|
|
|
|
+ ]);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
class CircleTakePhoto extends StatelessWidget {
|
|
class CircleTakePhoto extends StatelessWidget {
|
|
const CircleTakePhoto(
|
|
const CircleTakePhoto(
|
|
{Key key, @required this.outBox, @required this.innerBox})
|
|
{Key key, @required this.outBox, @required this.innerBox})
|
|
@@ -499,7 +708,7 @@ class CircleTakePhoto extends StatelessWidget {
|
|
child: FlatButton(
|
|
child: FlatButton(
|
|
padding: EdgeInsets.all(0),
|
|
padding: EdgeInsets.all(0),
|
|
onPressed: () async {
|
|
onPressed: () async {
|
|
- // provider.changeFileName();
|
|
|
|
|
|
+ // provider.changeFileName('png');
|
|
// print(provider.fileName);
|
|
// print(provider.fileName);
|
|
// await provider.cameraController
|
|
// await provider.cameraController
|
|
// .takePicture(provider.fileName)
|
|
// .takePicture(provider.fileName)
|
|
@@ -507,14 +716,12 @@ class CircleTakePhoto extends StatelessWidget {
|
|
// // Navigator.push(context, MaterialPageRoute(fullscreenDialog: true,builder: (_){
|
|
// // Navigator.push(context, MaterialPageRoute(fullscreenDialog: true,builder: (_){
|
|
// // return Image.file(File(provider.fileName) );
|
|
// // return Image.file(File(provider.fileName) );
|
|
// // }));
|
|
// // }));
|
|
-
|
|
|
|
// ImagePickerSaver.saveFile(
|
|
// ImagePickerSaver.saveFile(
|
|
// fileData: File(provider.fileName).readAsBytesSync());
|
|
// fileData: File(provider.fileName).readAsBytesSync());
|
|
- // File(provider.fileName).deleteSync();
|
|
|
|
// });
|
|
// });
|
|
- provider.changePhotoWidget();
|
|
|
|
provider.changeFileName('mp4');
|
|
provider.changeFileName('mp4');
|
|
provider.cameraController.startVideoRecording(provider.fileName);
|
|
provider.cameraController.startVideoRecording(provider.fileName);
|
|
|
|
+ provider.changePhotoWidget();
|
|
},
|
|
},
|
|
child: Container(
|
|
child: Container(
|
|
width: innerBox,
|
|
width: innerBox,
|