CameraMain.dart 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761
  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: CameraMain(
  19. rpx: MediaQuery.of(context).size.width / 750,
  20. ),
  21. bottomNavigationBar: BottomAppBar(),
  22. );
  23. }
  24. }
  25. class CameraMain extends StatefulWidget {
  26. CameraMain({Key key, @required this.rpx}) : super(key: key);
  27. final double rpx;
  28. @override
  29. _CameraMainState createState() => _CameraMainState();
  30. }
  31. class _CameraMainState extends State<CameraMain> {
  32. CameraProvider provider;
  33. double rpx;
  34. double toTop;
  35. double outBox;
  36. double innerBox;
  37. CameraController _controller;
  38. bool findFace = false;
  39. var cameras;
  40. @override
  41. void initState() {
  42. super.initState();
  43. // provider = Provider.of<CameraProvider>(context);
  44. // getCameras();
  45. rpx = widget.rpx;
  46. toTop = 100 * rpx;
  47. outBox = 170 * rpx;
  48. innerBox = 130 * rpx;
  49. }
  50. getCameras() async {
  51. cameras = await availableCameras();
  52. _controller = CameraController(cameras[1], ResolutionPreset.medium);
  53. _controller.initialize().then((_) {
  54. if (!mounted) {
  55. return;
  56. }
  57. _controller.startImageStream((CameraImage availableImage) {
  58. _controller.stopImageStream();
  59. _scanFrame(availableImage);
  60. });
  61. setState(() {});
  62. });
  63. }
  64. void _scanFrame(CameraImage availableImage) async {
  65. final FirebaseVisionImageMetadata metadata = FirebaseVisionImageMetadata(
  66. rawFormat: availableImage.format.raw,
  67. size: Size(
  68. availableImage.width.toDouble(), availableImage.height.toDouble()),
  69. planeData: availableImage.planes
  70. .map((currentPlane) => FirebaseVisionImagePlaneMetadata(
  71. bytesPerRow: currentPlane.bytesPerRow,
  72. height: currentPlane.height,
  73. width: currentPlane.width))
  74. .toList(),
  75. rotation: ImageRotation.rotation90);
  76. final FirebaseVisionImage visionImage =
  77. FirebaseVisionImage.fromBytes(availableImage.planes[0].bytes, metadata);
  78. final FaceDetector detector = FirebaseVision.instance.faceDetector();
  79. final List<Face> faces = await detector.processImage(visionImage);
  80. if (faces.length > 0) {
  81. setState(() {
  82. findFace = true;
  83. _controller.startImageStream((CameraImage availableImage) {
  84. _controller.stopImageStream();
  85. _scanFrame(availableImage);
  86. });
  87. });
  88. } else {
  89. setState(() {
  90. findFace = false;
  91. _controller.startImageStream((CameraImage availableImage) {
  92. _controller.stopImageStream();
  93. _scanFrame(availableImage);
  94. });
  95. });
  96. }
  97. // for (TextBlock block in visionText.blocks) {
  98. // // final Rectangle<int> boundingBox = block.boundingBox;
  99. // // final List<Point<int>> cornerPoints = block.cornerPoints;
  100. // print(block.text);
  101. // final List<RecognizedLanguage> languages = block.recognizedLanguages;
  102. // for (TextLine line in block.lines) {
  103. // // Same getters as TextBlock
  104. // print(line.text);
  105. // for (TextElement element in line.elements) {
  106. // // Same getters as TextBlock
  107. // print(element.text);
  108. // }
  109. // }
  110. // }
  111. }
  112. @override
  113. void dispose() {
  114. // TODO: implement dispose
  115. _controller.dispose();
  116. super.dispose();
  117. }
  118. @override
  119. Widget build(BuildContext context) {
  120. provider = Provider.of<CameraProvider>(context);
  121. _controller = provider.cameraController;
  122. if (provider == null || _controller == null) {
  123. return Container(
  124. child: Center(child: CircularProgressIndicator()),
  125. );
  126. }
  127. bool ifMakeVideo = provider.ifMakeVideo;
  128. if (_controller == null || _controller?.value == null) {
  129. return Container(
  130. child: Center(child: CircularProgressIndicator()),
  131. );
  132. }
  133. final size = MediaQuery.of(context).size;
  134. return _controller.value.isInitialized
  135. ? Stack(children: <Widget>[
  136. // Camera.open(cameraId),
  137. findFace
  138. ? Positioned(
  139. top: 0,
  140. left: 0,
  141. child: Container(
  142. width: 100,
  143. height: 100,
  144. color: Colors.red,
  145. ))
  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. }