CameraMain.dart 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755
  1. import 'dart:io';
  2. import 'dart:math';
  3. import 'package:camera/camera.dart';
  4. import 'package:camera/new/src/support_android/camera.dart';
  5. import 'package:douyin_demo/providers/CameraProvider.dart';
  6. import 'package:firebase_ml_vision/firebase_ml_vision.dart';
  7. import 'package:flutter/material.dart';
  8. import 'package:image_gallery_saver/image_gallery_saver.dart';
  9. import 'package:provider/provider.dart';
  10. import 'package:path/path.dart' as p;
  11. import 'package:image_picker_saver/image_picker_saver.dart';
  12. class CameraPage extends StatelessWidget {
  13. const CameraPage({Key key}) : super(key: key);
  14. @override
  15. Widget build(BuildContext context) {
  16. return Scaffold(
  17. backgroundColor: Theme.of(context).primaryColor,
  18. body: MultiProvider(
  19. providers: [
  20. ChangeNotifierProvider(
  21. builder: (_) => CameraProvider(),
  22. )
  23. ],
  24. child: CameraMain(
  25. rpx: MediaQuery.of(context).size.width / 750,
  26. )),
  27. bottomNavigationBar: BottomAppBar(),
  28. );
  29. }
  30. }
  31. class CameraMain extends StatefulWidget {
  32. CameraMain({Key key, @required this.rpx}) : super(key: key);
  33. final double rpx;
  34. @override
  35. _CameraMainState createState() => _CameraMainState();
  36. }
  37. class _CameraMainState extends State<CameraMain> {
  38. CameraProvider provider;
  39. double rpx;
  40. double toTop;
  41. double outBox;
  42. double innerBox;
  43. CameraController _controller;
  44. bool findFace = false;
  45. var cameras;
  46. @override
  47. void initState() {
  48. super.initState();
  49. // provider = Provider.of<CameraProvider>(context);
  50. getCameras();
  51. rpx = widget.rpx;
  52. toTop = 100 * rpx;
  53. outBox = 170 * rpx;
  54. innerBox = 130 * rpx;
  55. }
  56. getCameras() async {
  57. cameras = await availableCameras();
  58. _controller = CameraController(cameras[1], ResolutionPreset.medium);
  59. _controller.initialize().then((_) {
  60. if (!mounted) {
  61. return;
  62. }
  63. _controller.startImageStream((CameraImage availableImage) {
  64. _controller.stopImageStream();
  65. _scanFrame(availableImage);
  66. });
  67. setState(() {});
  68. });
  69. }
  70. void _scanFrame(CameraImage availableImage) async {
  71. final FirebaseVisionImageMetadata metadata = FirebaseVisionImageMetadata(
  72. rawFormat: availableImage.format.raw,
  73. size: Size(
  74. availableImage.width.toDouble(), availableImage.height.toDouble()),
  75. planeData: availableImage.planes
  76. .map((currentPlane) => FirebaseVisionImagePlaneMetadata(
  77. bytesPerRow: currentPlane.bytesPerRow,
  78. height: currentPlane.height,
  79. width: currentPlane.width))
  80. .toList(),
  81. rotation: ImageRotation.rotation90);
  82. final FirebaseVisionImage visionImage =
  83. FirebaseVisionImage.fromBytes(availableImage.planes[0].bytes, metadata);
  84. final FaceDetector detector = FirebaseVision.instance.faceDetector();
  85. final List<Face> faces = await detector.processImage(visionImage);
  86. if (faces.length > 0) {
  87. setState(() {
  88. findFace = true;
  89. _controller.startImageStream((CameraImage availableImage) {
  90. _controller.stopImageStream();
  91. _scanFrame(availableImage);
  92. });
  93. });
  94. } else {
  95. setState(() {
  96. findFace = false;
  97. _controller.startImageStream((CameraImage availableImage) {
  98. _controller.stopImageStream();
  99. _scanFrame(availableImage);
  100. });
  101. });
  102. }
  103. // for (TextBlock block in visionText.blocks) {
  104. // // final Rectangle<int> boundingBox = block.boundingBox;
  105. // // final List<Point<int>> cornerPoints = block.cornerPoints;
  106. // print(block.text);
  107. // final List<RecognizedLanguage> languages = block.recognizedLanguages;
  108. // for (TextLine line in block.lines) {
  109. // // Same getters as TextBlock
  110. // print(line.text);
  111. // for (TextElement element in line.elements) {
  112. // // Same getters as TextBlock
  113. // print(element.text);
  114. // }
  115. // }
  116. // }
  117. }
  118. @override
  119. void dispose() {
  120. // TODO: implement dispose
  121. _controller.dispose();
  122. super.dispose();
  123. }
  124. @override
  125. Widget build(BuildContext context) {
  126. provider = Provider.of<CameraProvider>(context);
  127. if (provider == null || _controller == null) {
  128. return Container(
  129. child: Center(child: CircularProgressIndicator()),
  130. );
  131. }
  132. bool ifMakeVideo = provider.ifMakeVideo;
  133. if (_controller == null || _controller?.value == null) {
  134. return Container(
  135. child: Center(child: CircularProgressIndicator()),
  136. );
  137. }
  138. final size = MediaQuery.of(context).size;
  139. return _controller.value.isInitialized
  140. ? Stack(children: <Widget>[
  141. // Camera.open(cameraId),
  142. findFace? Positioned(
  143. top: 0,
  144. left: 0,
  145. child: Container(width: 100,height: 100,color: Colors.red,)
  146. ):Container(),
  147. ClipRect(
  148. child: Transform.scale(
  149. scale: _controller.value.aspectRatio / size.aspectRatio,
  150. child: Center(
  151. child: AspectRatio(
  152. aspectRatio: _controller.value.aspectRatio,
  153. child: CameraPreview(_controller),
  154. ),
  155. ),
  156. )),
  157. Positioned(
  158. //顶部关闭按钮
  159. top: toTop,
  160. left: 30 * rpx,
  161. child: IconButton(
  162. icon: Icon(
  163. Icons.close,
  164. color: Colors.white,
  165. size: 60 * rpx,
  166. ),
  167. onPressed: () {
  168. Navigator.pop(context);
  169. },
  170. ),
  171. ),
  172. Positioned(
  173. //选择音乐
  174. top: toTop,
  175. left: 250 * rpx,
  176. child: Container(
  177. width: 250 * rpx,
  178. child: FlatButton(
  179. onPressed: () {},
  180. child: Row(
  181. children: <Widget>[
  182. Icon(
  183. Icons.music_note,
  184. color: Colors.white,
  185. ),
  186. SizedBox(
  187. width: 10 * rpx,
  188. ),
  189. Text(
  190. "选择音乐",
  191. style: TextStyle(color: Colors.white),
  192. ),
  193. ],
  194. ),
  195. ),
  196. ),
  197. ),
  198. Positioned(
  199. //拍照按钮
  200. bottom: 140 * rpx,
  201. // left: (750*rpx-outBox)/2,
  202. child: Container(
  203. width: 750 * rpx,
  204. child: Row(
  205. mainAxisAlignment: MainAxisAlignment.spaceAround,
  206. children: [
  207. ifMakeVideo
  208. ? Container(
  209. width: 80 * rpx,
  210. )
  211. : IconWithText(
  212. icon: Icon(
  213. Icons.search,
  214. color: Colors.white,
  215. ),
  216. text: "道具"),
  217. ifMakeVideo
  218. ? AnimVideoButton(
  219. rpx: rpx,
  220. outWidth: outBox,
  221. innerWidth: innerBox - 30 * rpx,
  222. provider: provider,
  223. )
  224. : CircleTakePhoto(
  225. outBox: outBox,
  226. innerBox: innerBox,
  227. ),
  228. ifMakeVideo
  229. ? IconButton(
  230. padding: EdgeInsets.all(0),
  231. icon: Icon(
  232. Icons.check_circle,
  233. color: Color.fromARGB(255, 219, 48, 85),
  234. size: 80 * rpx,
  235. ),
  236. onPressed: () async {
  237. provider.cameraController
  238. .stopVideoRecording();
  239. await ImageGallerySaver.saveFile(
  240. provider.fileName);
  241. File(provider.fileName).delete();
  242. },
  243. )
  244. : IconWithText(
  245. icon: Icon(
  246. Icons.search,
  247. color: Colors.white,
  248. ),
  249. text: "道具"),
  250. ])),
  251. ),
  252. Positioned(
  253. bottom: 40 * rpx,
  254. child: ScrollBottomBar(
  255. rpx: rpx,
  256. ),
  257. ),
  258. Positioned(
  259. right: 30 * rpx,
  260. top: 80 * rpx,
  261. child: IconButton(
  262. icon: Icon(Icons.camera_front),
  263. onPressed: () {
  264. provider.changeCamera();
  265. }),
  266. )
  267. ])
  268. : Container();
  269. }
  270. }
  271. // class CameraMain extends StatelessWidget {
  272. // const CameraMain({Key key}) : super(key: key);
  273. // @override
  274. // Widget build(BuildContext context) {
  275. // CameraProvider provider = Provider.of<CameraProvider>(context);
  276. // if (provider == null || provider.cameraController == null) {
  277. // return Container(
  278. // child: Center(child: CircularProgressIndicator()),
  279. // );
  280. // }
  281. // double rpx = MediaQuery.of(context).size.width / 750;
  282. // double toTop = 100 * rpx;
  283. // double outBox = 170 * rpx;
  284. // double innerBox = 130 * rpx;
  285. // CameraController _controller = provider.cameraController;
  286. // var cameras = provider.cameras;
  287. // bool ifMakeVideo = provider.ifMakeVideo;
  288. // if (_controller == null || _controller?.value == null) {
  289. // return Container(
  290. // child: Center(child: CircularProgressIndicator()),
  291. // );
  292. // }
  293. // final size = MediaQuery.of(context).size;
  294. // return _controller.value.isInitialized
  295. // ? Stack(children: <Widget>[
  296. // // Camera.open(cameraId),
  297. // ClipRect(
  298. // child: Transform.scale(
  299. // scale: _controller.value.aspectRatio / size.aspectRatio,
  300. // child: Center(
  301. // child: AspectRatio(
  302. // aspectRatio: _controller.value.aspectRatio,
  303. // child: CameraPreview(_controller),
  304. // ),
  305. // ),
  306. // )),
  307. // Positioned(
  308. // //顶部关闭按钮
  309. // top: toTop,
  310. // left: 30 * rpx,
  311. // child: IconButton(
  312. // icon: Icon(
  313. // Icons.close,
  314. // color: Colors.white,
  315. // size: 60 * rpx,
  316. // ),
  317. // onPressed: () {
  318. // Navigator.pop(context);
  319. // },
  320. // ),
  321. // ),
  322. // Positioned(
  323. // //选择音乐
  324. // top: toTop,
  325. // left: 250 * rpx,
  326. // child: Container(
  327. // width: 250 * rpx,
  328. // child: FlatButton(
  329. // onPressed: () {},
  330. // child: Row(
  331. // children: <Widget>[
  332. // Icon(
  333. // Icons.music_note,
  334. // color: Colors.white,
  335. // ),
  336. // SizedBox(
  337. // width: 10 * rpx,
  338. // ),
  339. // Text(
  340. // "选择音乐",
  341. // style: TextStyle(color: Colors.white),
  342. // ),
  343. // ],
  344. // ),
  345. // ),
  346. // ),
  347. // ),
  348. // Positioned(
  349. // //拍照按钮
  350. // bottom: 140 * rpx,
  351. // // left: (750*rpx-outBox)/2,
  352. // child: Container(
  353. // width: 750 * rpx,
  354. // child: Row(
  355. // mainAxisAlignment: MainAxisAlignment.spaceAround,
  356. // children: [
  357. // ifMakeVideo
  358. // ? Container(
  359. // width: 80 * rpx,
  360. // )
  361. // : IconWithText(
  362. // icon: Icon(
  363. // Icons.search,
  364. // color: Colors.white,
  365. // ),
  366. // text: "道具"),
  367. // ifMakeVideo
  368. // ? AnimVideoButton(
  369. // rpx: rpx,
  370. // outWidth: outBox,
  371. // innerWidth: innerBox - 30 * rpx,
  372. // provider: provider,
  373. // )
  374. // : CircleTakePhoto(
  375. // outBox: outBox,
  376. // innerBox: innerBox,
  377. // ),
  378. // ifMakeVideo
  379. // ? IconButton(
  380. // padding: EdgeInsets.all(0),
  381. // icon: Icon(
  382. // Icons.check_circle,
  383. // color: Color.fromARGB(255, 219, 48, 85),
  384. // size: 80 * rpx,
  385. // ),
  386. // onPressed: () async {
  387. // provider.cameraController
  388. // .stopVideoRecording();
  389. // await ImageGallerySaver.saveFile(
  390. // provider.fileName);
  391. // File(provider.fileName).delete();
  392. // },
  393. // )
  394. // : IconWithText(
  395. // icon: Icon(
  396. // Icons.search,
  397. // color: Colors.white,
  398. // ),
  399. // text: "道具"),
  400. // ])),
  401. // ),
  402. // Positioned(
  403. // bottom: 40 * rpx,
  404. // child: ScrollBottomBar(
  405. // rpx: rpx,
  406. // ),
  407. // ),
  408. // Positioned(
  409. // right: 30 * rpx,
  410. // top: 80 * rpx,
  411. // child: IconButton(
  412. // icon: Icon(Icons.camera_front),
  413. // onPressed: () {
  414. // provider.changeCamera();
  415. // }),
  416. // )
  417. // ])
  418. // : Container();
  419. // }
  420. // }
  421. class AnimVideoButton extends StatefulWidget {
  422. AnimVideoButton(
  423. {Key key,
  424. @required this.outWidth,
  425. @required this.innerWidth,
  426. @required this.rpx,
  427. @required this.provider})
  428. : super(key: key);
  429. final double outWidth;
  430. final double innerWidth;
  431. final double rpx;
  432. final CameraProvider provider;
  433. _AnimVideoButtonState createState() => _AnimVideoButtonState();
  434. }
  435. class _AnimVideoButtonState extends State<AnimVideoButton>
  436. with TickerProviderStateMixin {
  437. Animation<double> animation;
  438. AnimationController controller;
  439. double outWidth;
  440. double innerWidth;
  441. double outBorder;
  442. double rpx;
  443. double maxBorder;
  444. bool ifRecording;
  445. CameraProvider provider;
  446. double curBorder;
  447. @override
  448. void dispose() {
  449. // TODO: implement dispose
  450. controller.dispose();
  451. super.dispose();
  452. }
  453. @override
  454. void initState() {
  455. super.initState();
  456. ifRecording = true;
  457. provider = widget.provider;
  458. outWidth = widget.outWidth;
  459. innerWidth = widget.innerWidth;
  460. rpx = widget.rpx;
  461. outBorder = 5 * rpx;
  462. maxBorder = (outWidth - innerWidth) / 2 - 10 * rpx;
  463. curBorder = outBorder;
  464. controller =
  465. AnimationController(duration: Duration(milliseconds: 500), vsync: this);
  466. animation =
  467. Tween<double>(begin: outBorder, end: maxBorder).animate(controller)
  468. ..addListener(() {
  469. setState(() {
  470. curBorder = animation.value;
  471. });
  472. });
  473. controller.repeat(reverse: true);
  474. }
  475. pauseRecording() {
  476. // provider.cameraController.pauseVideoRecording();
  477. controller.stop();
  478. provider.cameraController.pauseVideoRecording();
  479. setState(() {
  480. ifRecording = false;
  481. });
  482. }
  483. resumeRecording() {
  484. // provider.cameraController.resumeVideoRecording();
  485. controller.repeat(reverse: true);
  486. provider.cameraController.resumeVideoRecording();
  487. setState(() {
  488. ifRecording = true;
  489. });
  490. }
  491. @override
  492. Widget build(BuildContext context) {
  493. return Container(
  494. width: outWidth,
  495. height: outWidth,
  496. decoration: BoxDecoration(
  497. shape: BoxShape.circle,
  498. color: Colors.transparent,
  499. border: Border.all(
  500. width: curBorder, color: Color.fromARGB(128, 219, 48, 85))),
  501. child: Container(
  502. child: !ifRecording
  503. ? IconButton(
  504. padding: EdgeInsets.all(0),
  505. icon: Icon(
  506. Icons.play_arrow,
  507. size: innerWidth,
  508. color: Color.fromARGB(255, 219, 48, 85),
  509. ),
  510. onPressed: () {
  511. resumeRecording();
  512. },
  513. )
  514. : IconButton(
  515. padding: EdgeInsets.all(0),
  516. icon: Icon(
  517. Icons.pause,
  518. size: innerWidth,
  519. color: Color.fromARGB(255, 219, 48, 85),
  520. ),
  521. onPressed: () {
  522. pauseRecording();
  523. },
  524. ),
  525. ),
  526. );
  527. }
  528. }
  529. class ScrollBottomBar extends StatefulWidget {
  530. ScrollBottomBar({Key key, @required this.rpx}) : super(key: key);
  531. final double rpx;
  532. _ScrollBottomBarState createState() => _ScrollBottomBarState();
  533. }
  534. class _ScrollBottomBarState extends State<ScrollBottomBar> {
  535. double rpx;
  536. double eachWidth;
  537. double eachSide;
  538. List<String> items;
  539. ScrollController controller;
  540. double startX = 0;
  541. double finalX = 0;
  542. double minValue;
  543. double maxValue;
  544. double curX;
  545. int curIndex;
  546. @override
  547. void initState() {
  548. super.initState();
  549. rpx = widget.rpx;
  550. eachWidth = 130 * rpx;
  551. eachSide = (750 - eachWidth / rpx) / 2 * rpx;
  552. curIndex = 2;
  553. minValue = 0;
  554. items = [
  555. '拍照',
  556. '拍15秒',
  557. '拍60秒',
  558. '影集',
  559. '开直播',
  560. ];
  561. maxValue = (items.length - 1) * eachWidth;
  562. curX = curIndex * eachWidth;
  563. controller = ScrollController(initialScrollOffset: curX);
  564. }
  565. moveToItem(index) {
  566. curX = index * eachWidth;
  567. controller.animateTo(curX,
  568. duration: Duration(milliseconds: 200), curve: Curves.linear);
  569. setState(() {
  570. curX = curX;
  571. curIndex = index;
  572. });
  573. }
  574. @override
  575. Widget build(BuildContext context) {
  576. return Column(children: [
  577. Listener(
  578. onPointerDown: (result) {
  579. setState(() {
  580. startX = result.position.dx;
  581. });
  582. },
  583. onPointerMove: (result) {
  584. double moveValue = result.position.dx;
  585. double moved = startX - moveValue;
  586. // curX+moved
  587. double afterMoved = min(max(curX + moved, minValue), maxValue);
  588. setState(() {
  589. curX = afterMoved;
  590. startX = result.position.dx;
  591. });
  592. },
  593. onPointerUp: (result) {
  594. int index = 0;
  595. double finalPosition = curX - eachWidth / 2;
  596. index = (finalPosition / eachWidth).ceil();
  597. moveToItem(index);
  598. },
  599. child: Container(
  600. width: 750 * rpx,
  601. height: 100 * rpx,
  602. child: SingleChildScrollView(
  603. scrollDirection: Axis.horizontal,
  604. controller: controller,
  605. child: Container(
  606. child: Row(
  607. mainAxisSize: MainAxisSize.min,
  608. children: <Widget>[
  609. SizedBox(
  610. width: eachSide,
  611. ),
  612. Row(
  613. children: List.generate(items.length, (index) {
  614. return Container(
  615. width: eachWidth,
  616. child: FlatButton(
  617. child: Text(
  618. items[index],
  619. style: TextStyle(
  620. color: curIndex == index
  621. ? Colors.white
  622. : Colors.white.withOpacity(0.5)),
  623. ),
  624. padding: EdgeInsets.all(0),
  625. onPressed: () {
  626. moveToItem(index);
  627. },
  628. ),
  629. );
  630. })),
  631. SizedBox(
  632. width: eachSide,
  633. ),
  634. ],
  635. ),
  636. ),
  637. )),
  638. ),
  639. Center(
  640. child: Container(
  641. decoration:
  642. BoxDecoration(shape: BoxShape.circle, color: Colors.white),
  643. width: 8 * rpx,
  644. height: 8 * rpx,
  645. ),
  646. )
  647. ]);
  648. }
  649. }
  650. class CircleTakePhoto extends StatelessWidget {
  651. const CircleTakePhoto(
  652. {Key key, @required this.outBox, @required this.innerBox})
  653. : super(key: key);
  654. final double outBox;
  655. final double innerBox;
  656. @override
  657. Widget build(BuildContext context) {
  658. double rpx = MediaQuery.of(context).size.width / 750;
  659. CameraProvider provider = Provider.of<CameraProvider>(context);
  660. // double outBox=160*rpx;
  661. // double innerBox=130*rpx;
  662. return Container(
  663. width: outBox,
  664. height: outBox,
  665. padding: EdgeInsets.all(10 * rpx),
  666. decoration: BoxDecoration(
  667. color: Colors.transparent,
  668. borderRadius: BorderRadius.circular(90 * rpx),
  669. border: Border.all(
  670. width: 10 * rpx, color: Color.fromARGB(128, 219, 48, 85)),
  671. ),
  672. child: FlatButton(
  673. padding: EdgeInsets.all(0),
  674. onPressed: () async {
  675. // provider.changeFileName('png');
  676. // print(provider.fileName);
  677. // await provider.cameraController
  678. // .takePicture(provider.fileName)
  679. // .then((_) {
  680. // // Navigator.push(context, MaterialPageRoute(fullscreenDialog: true,builder: (_){
  681. // // return Image.file(File(provider.fileName) );
  682. // // }));
  683. // ImagePickerSaver.saveFile(
  684. // fileData: File(provider.fileName).readAsBytesSync());
  685. // });
  686. provider.changeFileName('mp4');
  687. provider.cameraController.startVideoRecording(provider.fileName);
  688. provider.changePhotoWidget();
  689. },
  690. child: Container(
  691. width: innerBox,
  692. height: innerBox,
  693. alignment: Alignment.center,
  694. decoration: BoxDecoration(
  695. color: Color.fromARGB(255, 219, 48, 85),
  696. borderRadius: BorderRadius.circular(75 * rpx)),
  697. )),
  698. );
  699. }
  700. }
  701. class IconWithText extends StatelessWidget {
  702. const IconWithText({Key key, @required this.icon, @required this.text})
  703. : super(key: key);
  704. final Icon icon;
  705. final String text;
  706. @override
  707. Widget build(BuildContext context) {
  708. return Column(
  709. children: <Widget>[
  710. icon,
  711. Text(
  712. text,
  713. style: TextStyle(color: Colors.white),
  714. )
  715. ],
  716. );
  717. }
  718. }