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

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

使用版本

  photo: ^0.4.8       #相冊多圖選擇  
  image_picker: ^0.6.3+1  #相機(jī)拍照棍现,單圖選擇

包地址

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"); //先添加 加號(hào)按鈕 的圖片
    }

    @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(//可以直接指定每行(列)顯示多少個(gè)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: (){


      //點(diǎn)擊添加按鈕
        JhBottomSheet.showText(context, dataArr: ["拍照","相冊"],title: "請(qǐng)選擇",
        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: (){
                  //點(diǎn)擊刪除按鈕
                  setState(() {
                    imgData.removeAt(index);
//                    imgPicked.removeAt(index);
                  });
                },
              )
            ]
        ),
      ),
    onTap: (){
        print("點(diǎn)擊第${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)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市镜遣,隨后出現(xiàn)的幾起案子己肮,更是在濱河造成了極大的恐慌,老刑警劉巖悲关,帶你破解...
    沈念sama閱讀 217,277評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件谎僻,死亡現(xiàn)場離奇詭異,居然都是意外死亡寓辱,警方通過查閱死者的電腦和手機(jī)艘绍,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來秫筏,“玉大人诱鞠,你說我怎么就攤上這事≌饩矗” “怎么了航夺?”我有些...
    開封第一講書人閱讀 163,624評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長崔涂。 經(jīng)常有香客問我阳掐,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評(píng)論 1 293
  • 正文 為了忘掉前任缭保,我火速辦了婚禮汛闸,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘涮俄。我一直安慰自己蛉拙,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評(píng)論 6 392
  • 文/花漫 我一把揭開白布彻亲。 她就那樣靜靜地躺著孕锄,像睡著了一般。 火紅的嫁衣襯著肌膚如雪苞尝。 梳的紋絲不亂的頭發(fā)上畸肆,一...
    開封第一講書人閱讀 51,292評(píng)論 1 301
  • 那天,我揣著相機(jī)與錄音宙址,去河邊找鬼轴脐。 笑死,一個(gè)胖子當(dāng)著我的面吹牛抡砂,可吹牛的內(nèi)容都是我干的大咱。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼注益,長吁一口氣:“原來是場噩夢啊……” “哼碴巾!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起丑搔,我...
    開封第一講書人閱讀 38,992評(píng)論 0 275
  • 序言:老撾萬榮一對(duì)情侶失蹤厦瓢,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后啤月,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體煮仇,經(jīng)...
    沈念sama閱讀 45,429評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評(píng)論 3 334
  • 正文 我和宋清朗相戀三年谎仲,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了浙垫。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡郑诺,死狀恐怖绞呈,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情间景,我是刑警寧澤,帶...
    沈念sama閱讀 35,492評(píng)論 5 345
  • 正文 年R本政府宣布艺智,位于F島的核電站倘要,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜封拧,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評(píng)論 3 328
  • 文/蒙蒙 一志鹃、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧泽西,春花似錦曹铃、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至味抖,卻和暖如春评甜,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背仔涩。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評(píng)論 1 269
  • 我被黑心中介騙來泰國打工忍坷, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人熔脂。 一個(gè)月前我還...
    沈念sama閱讀 47,891評(píng)論 2 370
  • 正文 我出身青樓佩研,卻偏偏與公主長得像,于是被迫代替她去往敵國和親霞揉。 傳聞我的和親對(duì)象是個(gè)殘疾皇子旬薯,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容