Flutter - 實現(xiàn)多圖選擇,相機拍照功能

demo 地址: https://github.com/iotjin/jh_flutter_demo

使用版本

  photo: ^0.4.8       #相冊多圖選擇  
  image_picker: ^0.6.3+1  #相機拍照,單圖選擇

包地址

photo packages地址

image_picker packages地址

效果圖

PhotoSelectTest1.png
PhotoSelectTest2.png
PhotoSelectTest3.png
選擇界面.png

主界面 代碼

import 'package:flutter/material.dart';
import 'jhPhotoPickerTool.dart';

class PhotoSelectTest extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
        return
          Scaffold(
            appBar:AppBar(
                title:Text('PhotoSelectTest')
            ),
            body:

              Container(
              padding: EdgeInsets.fromLTRB(80, 10, 30, 10),
              color: Colors.red,
              child:
              JhPhotoPickerTool(
                lfPaddingSpace: 110,
                callBack: (var img){
                  print("img-------------");
                  print(img.length);
                  print(img);
                  print("img-------------");

                },
              )
              )

          );

  }
}


jhPhotoPickerTool 代碼

import 'package:flutter/material.dart';
import 'jhBottomSheet.dart';
import 'jhImageTool.dart';
import 'package:photo/photo.dart';
import 'package:photo_manager/photo_manager.dart';
import 'dart:io';
import 'package:image_picker/image_picker.dart';

const double itemSpace = 10.0;
const double space = 5.0; //上下左右間距
const double deleBtnWH = 20.0;
const Color bgColor = Colors.yellow;

typedef CallBack = void Function(List imgData);

class JhPhotoPickerTool extends StatefulWidget {

    final double lfPaddingSpace; //外部設(shè)置的左右間距
    final CallBack callBack;

    JhPhotoPickerTool({
      this.lfPaddingSpace,
      this.callBack,
    });

  @override
  _JhPhotoPickerToolState createState() => _JhPhotoPickerToolState();
}

class _JhPhotoPickerToolState extends State<JhPhotoPickerTool> {

      List imgData = List();  //圖片list
      List<AssetEntity> imgPicked = [];

    @override
    void initState() {
      // TODO: implement initState
      super.initState();
      imgData.add("selectPhoto_add"); //先添加 加號按鈕 的圖片
    }

    @override
  void setState(fn) {
    // TODO: implement setState
    super.setState(fn);
    List data = List();
    data.addAll(imgData);
    data.removeAt(imgData.length-1);
    widget.callBack(data);
  }

  @override
  Widget build(BuildContext context) {

    var kScreenWidth = MediaQuery.of(context).size.width;

    var lfPadding  = widget.lfPaddingSpace==null ?0.0: widget.lfPaddingSpace;
    var ninePictureW =(kScreenWidth-space*2-2*itemSpace-lfPadding);
    var itemWH = ninePictureW/3;
    int columnCount = imgData.length >6 ? 3:imgData.length <= 3 ? 1:2;

    return Container(
        color: bgColor,
        width: kScreenWidth-lfPadding,
        height: columnCount *itemWH +space*2+(columnCount -1)*itemSpace,
        child:
        GridView.builder(
            gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(//可以直接指定每行(列)顯示多少個Item
              //一行的Widget數(shù)量
              crossAxisCount:3,
              crossAxisSpacing: itemSpace, //水平間距
              mainAxisSpacing: itemSpace, //垂直間距
              childAspectRatio: 1.0,//子Widget寬高比例
            ),
            physics:NeverScrollableScrollPhysics(),
            padding: EdgeInsets.all(space),//GridView內(nèi)邊距
            itemCount: imgData.length ,
            itemBuilder: (context, index) {

              if(index == imgData.length-1){
                return addBtn(context,setState,imgData,imgPicked);
              }else{
                return
                imgItem(index,setState,imgData,imgPicked);
              }

            }
        )

    );
  }
}


/** 添加按鈕 */
Widget addBtn(context,setState,imgData,imgPicked){
  return GestureDetector(
    child: Image(image: JhImageTool.getAssetImage("selectPhoto_add")),
    onTap: (){


      //點擊添加按鈕
        JhBottomSheet.showText(context, dataArr: ["拍照","相冊"],title: "請選擇",
        clickCallback: (index,str) async{
          if(index==0){

            var image = await ImagePicker.pickImage(source: ImageSource.camera);
             print(image);
            imgData.insert(imgData.length-1, image.absolute.path);
//            imgPicked.add(image);
            setState(() {
            });

          }
          if(index==1){
            pickAsset(context,setState,imgData,imgPicked);
          }

        }
        );


    },
  );
}



/** 圖片和刪除按鈕 */
Widget imgItem (index,setState,imgData,imgPicked){
  return
    GestureDetector(
      child: Container(
        color: Colors.transparent,
        child: Stack(
            alignment: Alignment.topRight,
            children: <Widget>[
              ConstrainedBox(
//                child:Image.file(imgData[index], fit: BoxFit.cover),
                child:Image.file(File(imgData[index]), fit: BoxFit.cover),
                constraints: BoxConstraints.expand(),
              ),
              GestureDetector(
                child: Image(image: JhImageTool.getAssetImage("selectPhoto_close"),width: deleBtnWH,height: deleBtnWH,),
                onTap: (){
                  //點擊刪除按鈕
                  setState(() {
                    imgData.removeAt(index);
//                    imgPicked.removeAt(index);
                  });
                },
              )
            ]
        ),
      ),
    onTap: (){
        print("點擊第${index}張圖片");
    },
    );

}


/** 多圖選擇 */
void pickAsset(context,setState,imgData,imgPicked) async {

  final result = await PhotoPicker.pickAsset(
    context: context,
//    pickedAssetList: imgPicked,
    maxSelected: 10 -imgData.length,
    pickType:PickType.onlyImage
  );

  if (result != null && result.isNotEmpty) {
    for (var e in result) {
      var file = await e.file;
//      print(file.absolute.path)
      if (!imgData.contains(file.absolute.path)) {
          imgData.insert(imgData.length-1, file.absolute.path);
        }

//      imgData.insert(imgData.length-1, file);
//      if (!imgData.contains(file)) {
//        imgData.insert(imgData.length-1, file);
//      }

    }

  }
  setState(() {});

}


最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容