androidbitmapandroid-photos

Upload a picture taken by the camera to a server with limited size


the title sounds maybe a bit like a "noob question" but I know quite well how to program for Android, I just to figure out what it is the best way to achieve what I want.

My use case is: the user takes a photo and sends it to our server which have a file-size limit (which could mean that we have to resize the photo directly on the device).

Seems easy, right? My problem are the following:

1) Better use intents which could crash because some camera apps are coded with the ass or build a basic view "take photo and confirm" with cawc camera libs ? (I did the two, I prefer intents but I'd like to have an opinion on that).

2) How do you handle the file size limit? I mean getting the size of the photo is quite easy with the File.length() (even if the returned value is not perfectly right) but if you goes over the limit, how can you say how big will be the resized picture? (you need to convert in bitmap to resize it and it's then a lot of problems with OOMException and you cannot calculate final size of a bitmap on the disk, you need to compress and write it to the disk and analyse the newly created file after).

Thanks for help :D


Solution

  • I did the same thing before.

    1.I use intent to call the other camera app, and inside onActivityResult, I get back the URI and process it as I need.

    1. We do resize the pic, but I also keep the original ratio, and rotate it based on exif data. Hopefully this resizing code block can give you some hints.

      public static Bitmap DecodeImage(String path, int resolution) {
      BitmapFactory.Options opts = new BitmapFactory.Options();
      
      opts.inJustDecodeBounds = true;
      BitmapFactory.decodeFile(path, opts);
      opts.inSampleSize = computeSampleSize(opts, -1, resolution);
      opts.inJustDecodeBounds = false;
      return BitmapFactory.decodeFile(path, opts);
          }
      
      public static int computeSampleSize(BitmapFactory.Options options,
                                      int minSideLength, int maxNumOfPixels) {
      int initialSize = computeInitialSampleSize(options, minSideLength,
              maxNumOfPixels);
      
      int roundedSize;
      
      if (initialSize <= 8) {
          roundedSize = 1;
          while (roundedSize < initialSize) {
              roundedSize <<= 1;
          }
      } else {
          roundedSize = (initialSize + 7) / 8 * 8;
      }
      
      return roundedSize;
          }
      
      private static int computeInitialSampleSize(BitmapFactory.Options options, int minSideLength, int maxNumOfPixels) {
      double w = options.outWidth;
      double h = options.outHeight;
      
      int lowerBound = (maxNumOfPixels == -1) ? 1 :
              (int) Math.ceil(Math.sqrt(w * h / maxNumOfPixels));
      
      int upperBound = (minSideLength == -1) ? 128 :
              (int) Math.min(Math.floor(w / minSideLength),
                      Math.floor(h / minSideLength));
      
      if (upperBound < lowerBound) {
          // return the larger one when there is no overlapping zone.
          return lowerBound;
      }
      
      if ((maxNumOfPixels == -1) &&
      
              (minSideLength == -1)) {
          return 1;
      } else if (minSideLength == -1) {
          return lowerBound;
      } else {
          return upperBound;
      }
      
          }
      

      The solution is not fancy but it is what I did it in the project, and we so far have no problems with it after release.