CameraMain.dart 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757
  1. import 'dart:io';
  2. import 'dart:math';
  3. import 'package:camera/camera.dart';
  4. import 'package:douyin_demo/providers/CameraProvider.dart';
  5. import 'package:firebase_ml_vision/firebase_ml_vision.dart';
  6. import 'package:flutter/material.dart';
  7. import 'package:image_gallery_saver/image_gallery_saver.dart';
  8. import 'package:provider/provider.dart';
  9. class CameraPage extends StatelessWidget {
  10. const CameraPage({Key key}) : super(key: key);
  11. @override
  12. Widget build(BuildContext context) {
  13. return Scaffold(
  14. backgroundColor: Theme.of(context).primaryColor,
  15. body: CameraMain(
  16. rpx: MediaQuery.of(context).size.width / 750,
  17. ),
  18. bottomNavigationBar: BottomAppBar(),
  19. );
  20. }
  21. }
  22. class CameraMain extends StatefulWidget {
  23. CameraMain({Key key, @required this.rpx}) : super(key: key);
  24. final double rpx;
  25. @override
  26. _CameraMainState createState() => _CameraMainState();
  27. }
  28. class _CameraMainState extends State<CameraMain> {
  29. CameraProvider provider;
  30. double rpx;
  31. double toTop;
  32. double outBox;
  33. double innerBox;
  34. CameraController _controller;
  35. bool findFace = false;
  36. var cameras;
  37. @override
  38. void initState() {
  39. super.initState();
  40. // provider = Provider.of<CameraProvider>(context);
  41. // getCameras();
  42. rpx = widget.rpx;
  43. toTop = 100 * rpx;
  44. outBox = 170 * rpx;
  45. innerBox = 130 * rpx;
  46. }
  47. getCameras() async {
  48. cameras = await availableCameras();
  49. _controller = CameraController(cameras[1], ResolutionPreset.medium);
  50. _controller.initialize().then((_) {
  51. if (!mounted) {
  52. return;
  53. }
  54. _controller.startImageStream((CameraImage availableImage) {
  55. _controller.stopImageStream();
  56. _scanFrame(availableImage);
  57. });
  58. setState(() {});
  59. });
  60. }
  61. void _scanFrame(CameraImage availableImage) async {
  62. final FirebaseVisionImageMetadata metadata = FirebaseVisionImageMetadata(
  63. rawFormat: availableImage.format.raw,
  64. size: Size(
  65. availableImage.width.toDouble(), availableImage.height.toDouble()),
  66. planeData: availableImage.planes
  67. .map((currentPlane) => FirebaseVisionImagePlaneMetadata(
  68. bytesPerRow: currentPlane.bytesPerRow,
  69. height: currentPlane.height,
  70. width: currentPlane.width))
  71. .toList(),
  72. rotation: ImageRotation.rotation90);
  73. final FirebaseVisionImage visionImage =
  74. FirebaseVisionImage.fromBytes(availableImage.planes[0].bytes, metadata);
  75. final FaceDetector detector = FirebaseVision.instance.faceDetector();
  76. final List<Face> faces = await detector.processImage(visionImage);
  77. if (faces.length > 0) {
  78. setState(() {
  79. findFace = true;
  80. _controller.startImageStream((CameraImage availableImage) {
  81. _controller.stopImageStream();
  82. _scanFrame(availableImage);
  83. });
  84. });
  85. } else {
  86. setState(() {
  87. findFace = false;
  88. _controller.startImageStream((CameraImage availableImage) {
  89. _controller.stopImageStream();
  90. _scanFrame(availableImage);
  91. });
  92. });
  93. }
  94. // for (TextBlock block in visionText.blocks) {
  95. // // final Rectangle<int> boundingBox = block.boundingBox;
  96. // // final List<Point<int>> cornerPoints = block.cornerPoints;
  97. // print(block.text);
  98. // final List<RecognizedLanguage> languages = block.recognizedLanguages;
  99. // for (TextLine line in block.lines) {
  100. // // Same getters as TextBlock
  101. // print(line.text);
  102. // for (TextElement element in line.elements) {
  103. // // Same getters as TextBlock
  104. // print(element.text);
  105. // }
  106. // }
  107. // }
  108. }
  109. @override
  110. void dispose() {
  111. _controller.dispose();
  112. super.dispose();
  113. }
  114. @override
  115. Widget build(BuildContext context) {
  116. provider = Provider.of<CameraProvider>(context);
  117. _controller = provider.cameraController;
  118. if (provider == null || _controller == null) {
  119. return Container(
  120. child: Center(child: CircularProgressIndicator()),
  121. );
  122. }
  123. bool ifMakeVideo = provider.ifMakeVideo;
  124. if (_controller == null || _controller?.value == null) {
  125. return Container(
  126. child: Center(child: CircularProgressIndicator()),
  127. );
  128. }
  129. final size = MediaQuery.of(context).size;
  130. return _controller.value.isInitialized
  131. ? Stack(children: <Widget>[
  132. // Camera.open(cameraId),
  133. findFace
  134. ? Positioned(
  135. top: 0,
  136. left: 0,
  137. child: Container(
  138. width: 100,
  139. height: 100,
  140. color: Colors.red,
  141. ))
  142. : Container(),
  143. ClipRect(
  144. child: Transform.scale(
  145. scale: _controller.value.aspectRatio / size.aspectRatio,
  146. child: Center(
  147. child: AspectRatio(
  148. aspectRatio: _controller.value.aspectRatio,
  149. child: CameraPreview(_controller),
  150. ),
  151. ),
  152. )),
  153. Positioned(
  154. //顶部关闭按钮
  155. top: toTop,
  156. left: 30 * rpx,
  157. child: IconButton(
  158. icon: Icon(
  159. Icons.close,
  160. color: Colors.white,
  161. size: 60 * rpx,
  162. ),
  163. onPressed: () {
  164. Navigator.pop(context);
  165. },
  166. ),
  167. ),
  168. Positioned(
  169. //选择音乐
  170. top: toTop,
  171. left: 250 * rpx,
  172. child: Container(
  173. width: 250 * rpx,
  174. child: FlatButton(
  175. onPressed: () {},
  176. child: Row(
  177. children: <Widget>[
  178. Icon(
  179. Icons.music_note,
  180. color: Colors.white,
  181. ),
  182. SizedBox(
  183. width: 10 * rpx,
  184. ),
  185. Text(
  186. "选择音乐",
  187. style: TextStyle(color: Colors.white),
  188. ),
  189. ],
  190. ),
  191. ),
  192. ),
  193. ),
  194. Positioned(
  195. //拍照按钮
  196. bottom: 140 * rpx,
  197. // left: (750*rpx-outBox)/2,
  198. child: Container(
  199. width: 750 * rpx,
  200. child: Row(
  201. mainAxisAlignment: MainAxisAlignment.spaceAround,
  202. children: [
  203. ifMakeVideo
  204. ? Container(
  205. width: 80 * rpx,
  206. )
  207. : IconWithText(
  208. icon: Icon(
  209. Icons.search,
  210. color: Colors.white,
  211. ),
  212. text: "道具"),
  213. ifMakeVideo
  214. ? AnimVideoButton(
  215. rpx: rpx,
  216. outWidth: outBox,
  217. innerWidth: innerBox - 30 * rpx,
  218. provider: provider,
  219. )
  220. : CircleTakePhoto(
  221. outBox: outBox,
  222. innerBox: innerBox,
  223. ),
  224. ifMakeVideo
  225. ? IconButton(
  226. padding: EdgeInsets.all(0),
  227. icon: Icon(
  228. Icons.check_circle,
  229. color: Color.fromARGB(255, 219, 48, 85),
  230. size: 80 * rpx,
  231. ),
  232. onPressed: () async {
  233. provider.cameraController
  234. .stopVideoRecording();
  235. await ImageGallerySaver.saveFile(
  236. provider.fileName);
  237. File(provider.fileName).delete();
  238. },
  239. )
  240. : IconWithText(
  241. icon: Icon(
  242. Icons.search,
  243. color: Colors.white,
  244. ),
  245. text: "道具"),
  246. ])),
  247. ),
  248. Positioned(
  249. bottom: 40 * rpx,
  250. child: ScrollBottomBar(
  251. rpx: rpx,
  252. ),
  253. ),
  254. Positioned(
  255. right: 30 * rpx,
  256. top: 80 * rpx,
  257. child: IconButton(
  258. icon: Icon(Icons.camera_front),
  259. onPressed: () {
  260. provider.changeCamera();
  261. }),
  262. )
  263. ])
  264. : Container();
  265. }
  266. }
  267. // class CameraMain extends StatelessWidget {
  268. // const CameraMain({Key key}) : super(key: key);
  269. // @override
  270. // Widget build(BuildContext context) {
  271. // CameraProvider provider = Provider.of<CameraProvider>(context);
  272. // if (provider == null || provider.cameraController == null) {
  273. // return Container(
  274. // child: Center(child: CircularProgressIndicator()),
  275. // );
  276. // }
  277. // double rpx = MediaQuery.of(context).size.width / 750;
  278. // double toTop = 100 * rpx;
  279. // double outBox = 170 * rpx;
  280. // double innerBox = 130 * rpx;
  281. // CameraController _controller = provider.cameraController;
  282. // var cameras = provider.cameras;
  283. // bool ifMakeVideo = provider.ifMakeVideo;
  284. // if (_controller == null || _controller?.value == null) {
  285. // return Container(
  286. // child: Center(child: CircularProgressIndicator()),
  287. // );
  288. // }
  289. // final size = MediaQuery.of(context).size;
  290. // return _controller.value.isInitialized
  291. // ? Stack(children: <Widget>[
  292. // // Camera.open(cameraId),
  293. // ClipRect(
  294. // child: Transform.scale(
  295. // scale: _controller.value.aspectRatio / size.aspectRatio,
  296. // child: Center(
  297. // child: AspectRatio(
  298. // aspectRatio: _controller.value.aspectRatio,
  299. // child: CameraPreview(_controller),
  300. // ),
  301. // ),
  302. // )),
  303. // Positioned(
  304. // //顶部关闭按钮
  305. // top: toTop,
  306. // left: 30 * rpx,
  307. // child: IconButton(
  308. // icon: Icon(
  309. // Icons.close,
  310. // color: Colors.white,
  311. // size: 60 * rpx,
  312. // ),
  313. // onPressed: () {
  314. // Navigator.pop(context);
  315. // },
  316. // ),
  317. // ),
  318. // Positioned(
  319. // //选择音乐
  320. // top: toTop,
  321. // left: 250 * rpx,
  322. // child: Container(
  323. // width: 250 * rpx,
  324. // child: FlatButton(
  325. // onPressed: () {},
  326. // child: Row(
  327. // children: <Widget>[
  328. // Icon(
  329. // Icons.music_note,
  330. // color: Colors.white,
  331. // ),
  332. // SizedBox(
  333. // width: 10 * rpx,
  334. // ),
  335. // Text(
  336. // "选择音乐",
  337. // style: TextStyle(color: Colors.white),
  338. // ),
  339. // ],
  340. // ),
  341. // ),
  342. // ),
  343. // ),
  344. // Positioned(
  345. // //拍照按钮
  346. // bottom: 140 * rpx,
  347. // // left: (750*rpx-outBox)/2,
  348. // child: Container(
  349. // width: 750 * rpx,
  350. // child: Row(
  351. // mainAxisAlignment: MainAxisAlignment.spaceAround,
  352. // children: [
  353. // ifMakeVideo
  354. // ? Container(
  355. // width: 80 * rpx,
  356. // )
  357. // : IconWithText(
  358. // icon: Icon(
  359. // Icons.search,
  360. // color: Colors.white,
  361. // ),
  362. // text: "道具"),
  363. // ifMakeVideo
  364. // ? AnimVideoButton(
  365. // rpx: rpx,
  366. // outWidth: outBox,
  367. // innerWidth: innerBox - 30 * rpx,
  368. // provider: provider,
  369. // )
  370. // : CircleTakePhoto(
  371. // outBox: outBox,
  372. // innerBox: innerBox,
  373. // ),
  374. // ifMakeVideo
  375. // ? IconButton(
  376. // padding: EdgeInsets.all(0),
  377. // icon: Icon(
  378. // Icons.check_circle,
  379. // color: Color.fromARGB(255, 219, 48, 85),
  380. // size: 80 * rpx,
  381. // ),
  382. // onPressed: () async {
  383. // provider.cameraController
  384. // .stopVideoRecording();
  385. // await ImageGallerySaver.saveFile(
  386. // provider.fileName);
  387. // File(provider.fileName).delete();
  388. // },
  389. // )
  390. // : IconWithText(
  391. // icon: Icon(
  392. // Icons.search,
  393. // color: Colors.white,
  394. // ),
  395. // text: "道具"),
  396. // ])),
  397. // ),
  398. // Positioned(
  399. // bottom: 40 * rpx,
  400. // child: ScrollBottomBar(
  401. // rpx: rpx,
  402. // ),
  403. // ),
  404. // Positioned(
  405. // right: 30 * rpx,
  406. // top: 80 * rpx,
  407. // child: IconButton(
  408. // icon: Icon(Icons.camera_front),
  409. // onPressed: () {
  410. // provider.changeCamera();
  411. // }),
  412. // )
  413. // ])
  414. // : Container();
  415. // }
  416. // }
  417. class AnimVideoButton extends StatefulWidget {
  418. AnimVideoButton(
  419. {Key key,
  420. @required this.outWidth,
  421. @required this.innerWidth,
  422. @required this.rpx,
  423. @required this.provider})
  424. : super(key: key);
  425. final double outWidth;
  426. final double innerWidth;
  427. final double rpx;
  428. final CameraProvider provider;
  429. _AnimVideoButtonState createState() => _AnimVideoButtonState();
  430. }
  431. class _AnimVideoButtonState extends State<AnimVideoButton>
  432. with TickerProviderStateMixin {
  433. Animation<double> animation;
  434. AnimationController controller;
  435. double outWidth;
  436. double innerWidth;
  437. double outBorder;
  438. double rpx;
  439. double maxBorder;
  440. bool ifRecording;
  441. CameraProvider provider;
  442. double curBorder;
  443. @override
  444. void 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.cameraController.startVideoRecording();
  684. provider.changePhotoWidget();
  685. },
  686. child: Container(
  687. width: innerBox,
  688. height: innerBox,
  689. alignment: Alignment.center,
  690. decoration: BoxDecoration(
  691. color: Color.fromARGB(255, 219, 48, 85),
  692. borderRadius: BorderRadius.circular(75 * rpx)),
  693. )),
  694. );
  695. }
  696. }
  697. class IconWithText extends StatelessWidget {
  698. const IconWithText({Key key, @required this.icon, @required this.text})
  699. : super(key: key);
  700. final Icon icon;
  701. final String text;
  702. @override
  703. Widget build(BuildContext context) {
  704. return Column(
  705. children: <Widget>[
  706. icon,
  707. Text(
  708. text,
  709. style: TextStyle(color: Colors.white),
  710. )
  711. ],
  712. );
  713. }
  714. }