CameraMain.dart 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750
  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? Positioned(
  138. top: 0,
  139. left: 0,
  140. child: Container(width: 100,height: 100,color: Colors.red,)
  141. ):Container(),
  142. ClipRect(
  143. child: Transform.scale(
  144. scale: _controller.value.aspectRatio / size.aspectRatio,
  145. child: Center(
  146. child: AspectRatio(
  147. aspectRatio: _controller.value.aspectRatio,
  148. child: CameraPreview(_controller),
  149. ),
  150. ),
  151. )),
  152. Positioned(
  153. //顶部关闭按钮
  154. top: toTop,
  155. left: 30 * rpx,
  156. child: IconButton(
  157. icon: Icon(
  158. Icons.close,
  159. color: Colors.white,
  160. size: 60 * rpx,
  161. ),
  162. onPressed: () {
  163. Navigator.pop(context);
  164. },
  165. ),
  166. ),
  167. Positioned(
  168. //选择音乐
  169. top: toTop,
  170. left: 250 * rpx,
  171. child: Container(
  172. width: 250 * rpx,
  173. child: FlatButton(
  174. onPressed: () {},
  175. child: Row(
  176. children: <Widget>[
  177. Icon(
  178. Icons.music_note,
  179. color: Colors.white,
  180. ),
  181. SizedBox(
  182. width: 10 * rpx,
  183. ),
  184. Text(
  185. "选择音乐",
  186. style: TextStyle(color: Colors.white),
  187. ),
  188. ],
  189. ),
  190. ),
  191. ),
  192. ),
  193. Positioned(
  194. //拍照按钮
  195. bottom: 140 * rpx,
  196. // left: (750*rpx-outBox)/2,
  197. child: Container(
  198. width: 750 * rpx,
  199. child: Row(
  200. mainAxisAlignment: MainAxisAlignment.spaceAround,
  201. children: [
  202. ifMakeVideo
  203. ? Container(
  204. width: 80 * rpx,
  205. )
  206. : IconWithText(
  207. icon: Icon(
  208. Icons.search,
  209. color: Colors.white,
  210. ),
  211. text: "道具"),
  212. ifMakeVideo
  213. ? AnimVideoButton(
  214. rpx: rpx,
  215. outWidth: outBox,
  216. innerWidth: innerBox - 30 * rpx,
  217. provider: provider,
  218. )
  219. : CircleTakePhoto(
  220. outBox: outBox,
  221. innerBox: innerBox,
  222. ),
  223. ifMakeVideo
  224. ? IconButton(
  225. padding: EdgeInsets.all(0),
  226. icon: Icon(
  227. Icons.check_circle,
  228. color: Color.fromARGB(255, 219, 48, 85),
  229. size: 80 * rpx,
  230. ),
  231. onPressed: () async {
  232. provider.cameraController
  233. .stopVideoRecording();
  234. await ImageGallerySaver.saveFile(
  235. provider.fileName);
  236. File(provider.fileName).delete();
  237. },
  238. )
  239. : IconWithText(
  240. icon: Icon(
  241. Icons.search,
  242. color: Colors.white,
  243. ),
  244. text: "道具"),
  245. ])),
  246. ),
  247. Positioned(
  248. bottom: 40 * rpx,
  249. child: ScrollBottomBar(
  250. rpx: rpx,
  251. ),
  252. ),
  253. Positioned(
  254. right: 30 * rpx,
  255. top: 80 * rpx,
  256. child: IconButton(
  257. icon: Icon(Icons.camera_front),
  258. onPressed: () {
  259. provider.changeCamera();
  260. }),
  261. )
  262. ])
  263. : Container();
  264. }
  265. }
  266. // class CameraMain extends StatelessWidget {
  267. // const CameraMain({Key key}) : super(key: key);
  268. // @override
  269. // Widget build(BuildContext context) {
  270. // CameraProvider provider = Provider.of<CameraProvider>(context);
  271. // if (provider == null || provider.cameraController == null) {
  272. // return Container(
  273. // child: Center(child: CircularProgressIndicator()),
  274. // );
  275. // }
  276. // double rpx = MediaQuery.of(context).size.width / 750;
  277. // double toTop = 100 * rpx;
  278. // double outBox = 170 * rpx;
  279. // double innerBox = 130 * rpx;
  280. // CameraController _controller = provider.cameraController;
  281. // var cameras = provider.cameras;
  282. // bool ifMakeVideo = provider.ifMakeVideo;
  283. // if (_controller == null || _controller?.value == null) {
  284. // return Container(
  285. // child: Center(child: CircularProgressIndicator()),
  286. // );
  287. // }
  288. // final size = MediaQuery.of(context).size;
  289. // return _controller.value.isInitialized
  290. // ? Stack(children: <Widget>[
  291. // // Camera.open(cameraId),
  292. // ClipRect(
  293. // child: Transform.scale(
  294. // scale: _controller.value.aspectRatio / size.aspectRatio,
  295. // child: Center(
  296. // child: AspectRatio(
  297. // aspectRatio: _controller.value.aspectRatio,
  298. // child: CameraPreview(_controller),
  299. // ),
  300. // ),
  301. // )),
  302. // Positioned(
  303. // //顶部关闭按钮
  304. // top: toTop,
  305. // left: 30 * rpx,
  306. // child: IconButton(
  307. // icon: Icon(
  308. // Icons.close,
  309. // color: Colors.white,
  310. // size: 60 * rpx,
  311. // ),
  312. // onPressed: () {
  313. // Navigator.pop(context);
  314. // },
  315. // ),
  316. // ),
  317. // Positioned(
  318. // //选择音乐
  319. // top: toTop,
  320. // left: 250 * rpx,
  321. // child: Container(
  322. // width: 250 * rpx,
  323. // child: FlatButton(
  324. // onPressed: () {},
  325. // child: Row(
  326. // children: <Widget>[
  327. // Icon(
  328. // Icons.music_note,
  329. // color: Colors.white,
  330. // ),
  331. // SizedBox(
  332. // width: 10 * rpx,
  333. // ),
  334. // Text(
  335. // "选择音乐",
  336. // style: TextStyle(color: Colors.white),
  337. // ),
  338. // ],
  339. // ),
  340. // ),
  341. // ),
  342. // ),
  343. // Positioned(
  344. // //拍照按钮
  345. // bottom: 140 * rpx,
  346. // // left: (750*rpx-outBox)/2,
  347. // child: Container(
  348. // width: 750 * rpx,
  349. // child: Row(
  350. // mainAxisAlignment: MainAxisAlignment.spaceAround,
  351. // children: [
  352. // ifMakeVideo
  353. // ? Container(
  354. // width: 80 * rpx,
  355. // )
  356. // : IconWithText(
  357. // icon: Icon(
  358. // Icons.search,
  359. // color: Colors.white,
  360. // ),
  361. // text: "道具"),
  362. // ifMakeVideo
  363. // ? AnimVideoButton(
  364. // rpx: rpx,
  365. // outWidth: outBox,
  366. // innerWidth: innerBox - 30 * rpx,
  367. // provider: provider,
  368. // )
  369. // : CircleTakePhoto(
  370. // outBox: outBox,
  371. // innerBox: innerBox,
  372. // ),
  373. // ifMakeVideo
  374. // ? IconButton(
  375. // padding: EdgeInsets.all(0),
  376. // icon: Icon(
  377. // Icons.check_circle,
  378. // color: Color.fromARGB(255, 219, 48, 85),
  379. // size: 80 * rpx,
  380. // ),
  381. // onPressed: () async {
  382. // provider.cameraController
  383. // .stopVideoRecording();
  384. // await ImageGallerySaver.saveFile(
  385. // provider.fileName);
  386. // File(provider.fileName).delete();
  387. // },
  388. // )
  389. // : IconWithText(
  390. // icon: Icon(
  391. // Icons.search,
  392. // color: Colors.white,
  393. // ),
  394. // text: "道具"),
  395. // ])),
  396. // ),
  397. // Positioned(
  398. // bottom: 40 * rpx,
  399. // child: ScrollBottomBar(
  400. // rpx: rpx,
  401. // ),
  402. // ),
  403. // Positioned(
  404. // right: 30 * rpx,
  405. // top: 80 * rpx,
  406. // child: IconButton(
  407. // icon: Icon(Icons.camera_front),
  408. // onPressed: () {
  409. // provider.changeCamera();
  410. // }),
  411. // )
  412. // ])
  413. // : Container();
  414. // }
  415. // }
  416. class AnimVideoButton extends StatefulWidget {
  417. AnimVideoButton(
  418. {Key key,
  419. @required this.outWidth,
  420. @required this.innerWidth,
  421. @required this.rpx,
  422. @required this.provider})
  423. : super(key: key);
  424. final double outWidth;
  425. final double innerWidth;
  426. final double rpx;
  427. final CameraProvider provider;
  428. _AnimVideoButtonState createState() => _AnimVideoButtonState();
  429. }
  430. class _AnimVideoButtonState extends State<AnimVideoButton>
  431. with TickerProviderStateMixin {
  432. Animation<double> animation;
  433. AnimationController controller;
  434. double outWidth;
  435. double innerWidth;
  436. double outBorder;
  437. double rpx;
  438. double maxBorder;
  439. bool ifRecording;
  440. CameraProvider provider;
  441. double curBorder;
  442. @override
  443. void dispose() {
  444. // TODO: implement dispose
  445. controller.dispose();
  446. super.dispose();
  447. }
  448. @override
  449. void initState() {
  450. super.initState();
  451. ifRecording = true;
  452. provider = widget.provider;
  453. outWidth = widget.outWidth;
  454. innerWidth = widget.innerWidth;
  455. rpx = widget.rpx;
  456. outBorder = 5 * rpx;
  457. maxBorder = (outWidth - innerWidth) / 2 - 10 * rpx;
  458. curBorder = outBorder;
  459. controller =
  460. AnimationController(duration: Duration(milliseconds: 500), vsync: this);
  461. animation =
  462. Tween<double>(begin: outBorder, end: maxBorder).animate(controller)
  463. ..addListener(() {
  464. setState(() {
  465. curBorder = animation.value;
  466. });
  467. });
  468. controller.repeat(reverse: true);
  469. }
  470. pauseRecording() {
  471. // provider.cameraController.pauseVideoRecording();
  472. controller.stop();
  473. provider.cameraController.pauseVideoRecording();
  474. setState(() {
  475. ifRecording = false;
  476. });
  477. }
  478. resumeRecording() {
  479. // provider.cameraController.resumeVideoRecording();
  480. controller.repeat(reverse: true);
  481. provider.cameraController.resumeVideoRecording();
  482. setState(() {
  483. ifRecording = true;
  484. });
  485. }
  486. @override
  487. Widget build(BuildContext context) {
  488. return Container(
  489. width: outWidth,
  490. height: outWidth,
  491. decoration: BoxDecoration(
  492. shape: BoxShape.circle,
  493. color: Colors.transparent,
  494. border: Border.all(
  495. width: curBorder, color: Color.fromARGB(128, 219, 48, 85))),
  496. child: Container(
  497. child: !ifRecording
  498. ? IconButton(
  499. padding: EdgeInsets.all(0),
  500. icon: Icon(
  501. Icons.play_arrow,
  502. size: innerWidth,
  503. color: Color.fromARGB(255, 219, 48, 85),
  504. ),
  505. onPressed: () {
  506. resumeRecording();
  507. },
  508. )
  509. : IconButton(
  510. padding: EdgeInsets.all(0),
  511. icon: Icon(
  512. Icons.pause,
  513. size: innerWidth,
  514. color: Color.fromARGB(255, 219, 48, 85),
  515. ),
  516. onPressed: () {
  517. pauseRecording();
  518. },
  519. ),
  520. ),
  521. );
  522. }
  523. }
  524. class ScrollBottomBar extends StatefulWidget {
  525. ScrollBottomBar({Key key, @required this.rpx}) : super(key: key);
  526. final double rpx;
  527. _ScrollBottomBarState createState() => _ScrollBottomBarState();
  528. }
  529. class _ScrollBottomBarState extends State<ScrollBottomBar> {
  530. double rpx;
  531. double eachWidth;
  532. double eachSide;
  533. List<String> items;
  534. ScrollController controller;
  535. double startX = 0;
  536. double finalX = 0;
  537. double minValue;
  538. double maxValue;
  539. double curX;
  540. int curIndex;
  541. @override
  542. void initState() {
  543. super.initState();
  544. rpx = widget.rpx;
  545. eachWidth = 130 * rpx;
  546. eachSide = (750 - eachWidth / rpx) / 2 * rpx;
  547. curIndex = 2;
  548. minValue = 0;
  549. items = [
  550. '拍照',
  551. '拍15秒',
  552. '拍60秒',
  553. '影集',
  554. '开直播',
  555. ];
  556. maxValue = (items.length - 1) * eachWidth;
  557. curX = curIndex * eachWidth;
  558. controller = ScrollController(initialScrollOffset: curX);
  559. }
  560. moveToItem(index) {
  561. curX = index * eachWidth;
  562. controller.animateTo(curX,
  563. duration: Duration(milliseconds: 200), curve: Curves.linear);
  564. setState(() {
  565. curX = curX;
  566. curIndex = index;
  567. });
  568. }
  569. @override
  570. Widget build(BuildContext context) {
  571. return Column(children: [
  572. Listener(
  573. onPointerDown: (result) {
  574. setState(() {
  575. startX = result.position.dx;
  576. });
  577. },
  578. onPointerMove: (result) {
  579. double moveValue = result.position.dx;
  580. double moved = startX - moveValue;
  581. // curX+moved
  582. double afterMoved = min(max(curX + moved, minValue), maxValue);
  583. setState(() {
  584. curX = afterMoved;
  585. startX = result.position.dx;
  586. });
  587. },
  588. onPointerUp: (result) {
  589. int index = 0;
  590. double finalPosition = curX - eachWidth / 2;
  591. index = (finalPosition / eachWidth).ceil();
  592. moveToItem(index);
  593. },
  594. child: Container(
  595. width: 750 * rpx,
  596. height: 100 * rpx,
  597. child: SingleChildScrollView(
  598. scrollDirection: Axis.horizontal,
  599. controller: controller,
  600. child: Container(
  601. child: Row(
  602. mainAxisSize: MainAxisSize.min,
  603. children: <Widget>[
  604. SizedBox(
  605. width: eachSide,
  606. ),
  607. Row(
  608. children: List.generate(items.length, (index) {
  609. return Container(
  610. width: eachWidth,
  611. child: FlatButton(
  612. child: Text(
  613. items[index],
  614. style: TextStyle(
  615. color: curIndex == index
  616. ? Colors.white
  617. : Colors.white.withOpacity(0.5)),
  618. ),
  619. padding: EdgeInsets.all(0),
  620. onPressed: () {
  621. moveToItem(index);
  622. },
  623. ),
  624. );
  625. })),
  626. SizedBox(
  627. width: eachSide,
  628. ),
  629. ],
  630. ),
  631. ),
  632. )),
  633. ),
  634. Center(
  635. child: Container(
  636. decoration:
  637. BoxDecoration(shape: BoxShape.circle, color: Colors.white),
  638. width: 8 * rpx,
  639. height: 8 * rpx,
  640. ),
  641. )
  642. ]);
  643. }
  644. }
  645. class CircleTakePhoto extends StatelessWidget {
  646. const CircleTakePhoto(
  647. {Key key, @required this.outBox, @required this.innerBox})
  648. : super(key: key);
  649. final double outBox;
  650. final double innerBox;
  651. @override
  652. Widget build(BuildContext context) {
  653. double rpx = MediaQuery.of(context).size.width / 750;
  654. CameraProvider provider = Provider.of<CameraProvider>(context);
  655. // double outBox=160*rpx;
  656. // double innerBox=130*rpx;
  657. return Container(
  658. width: outBox,
  659. height: outBox,
  660. padding: EdgeInsets.all(10 * rpx),
  661. decoration: BoxDecoration(
  662. color: Colors.transparent,
  663. borderRadius: BorderRadius.circular(90 * rpx),
  664. border: Border.all(
  665. width: 10 * rpx, color: Color.fromARGB(128, 219, 48, 85)),
  666. ),
  667. child: FlatButton(
  668. padding: EdgeInsets.all(0),
  669. onPressed: () async {
  670. // provider.changeFileName('png');
  671. // print(provider.fileName);
  672. // await provider.cameraController
  673. // .takePicture(provider.fileName)
  674. // .then((_) {
  675. // // Navigator.push(context, MaterialPageRoute(fullscreenDialog: true,builder: (_){
  676. // // return Image.file(File(provider.fileName) );
  677. // // }));
  678. // ImagePickerSaver.saveFile(
  679. // fileData: File(provider.fileName).readAsBytesSync());
  680. // });
  681. provider.changeFileName('mp4');
  682. provider.cameraController.startVideoRecording(provider.fileName);
  683. provider.changePhotoWidget();
  684. },
  685. child: Container(
  686. width: innerBox,
  687. height: innerBox,
  688. alignment: Alignment.center,
  689. decoration: BoxDecoration(
  690. color: Color.fromARGB(255, 219, 48, 85),
  691. borderRadius: BorderRadius.circular(75 * rpx)),
  692. )),
  693. );
  694. }
  695. }
  696. class IconWithText extends StatelessWidget {
  697. const IconWithText({Key key, @required this.icon, @required this.text})
  698. : super(key: key);
  699. final Icon icon;
  700. final String text;
  701. @override
  702. Widget build(BuildContext context) {
  703. return Column(
  704. children: <Widget>[
  705. icon,
  706. Text(
  707. text,
  708. style: TextStyle(color: Colors.white),
  709. )
  710. ],
  711. );
  712. }
  713. }