3

I tried to get the image from clipboard when I press ctrl+v on a page, with the code from a gist, which will console.log the image content it gets.

I found the image content is a long base64 string with a special prefix:



How to decode it to binary? I tried to remove the prefix data:image/png;base64, and use a base64 library to decode the rest into binary, and save it to a image.png file. But the file can't be displayed, with invalid format error.

But if I paste the whole string to http://base64online.org/decode/, it can play the image as well.

The base64 library I used is https://github.com/wstrange/base64, and my code is:

import 'package:base64_codec/base64_codec.dart';

var body = /* the long str above */;

var prefix = "data:image/png;base64,";
var bStr = body.substring(prefix.length);
var bs = Base64Codec.codec.decodeList(bStr.codeUnits);
var file = new File("image.png");
file.writeAsBytesSync(bs);

I don't know where is wrong :(

Shailen Tuli
  • 13,815
  • 5
  • 40
  • 51
Freewind
  • 193,756
  • 157
  • 432
  • 708

3 Answers3

4

Try using the crypto package from the Dart SDK. This creates an image.png that opens fine (I believe it's this image... enter image description here)

library foo;

import 'package:crypto/crypto.dart';
import 'dart:io';

void main() {

  var body = "... snip"; 
  var prefix = "data:image/png;base64,";
  var bStr = body.substring(prefix.length);
  var bytes = CryptoUtils.base64StringToBytes(bStr); // using CryptoUtils from Dart SDK
  var file = new File("image.png");
  file.writeAsBytesSync(bytes);
}  

You'll need to add crypto to your pubspec.yaml :

dependencies:
  crypto: any

The library linked at https://github.com/wstrange/base64, also notes in the README:

Deprecated

Note: The Dart SDK now includes a Base64 codec with url safe options. Unless you need streaming support, you should use the SDK methods as they are slightly faster.

See CryptoUtils

See also How to native convert string -> base64 and base64 -> string

Community
  • 1
  • 1
Chris Buckett
  • 13,738
  • 5
  • 39
  • 46
4

Decoding as a string and then writing the bytes works:

var prefix = "data:image/png;base64,";
var bStr = body.substring(prefix.length);
var bs = Base64Codec.codec.decodeString(bStr);
var file = new File("image.png");
file.writeAsBytesSync(bs.codeUnits);
Pixel Elephant
  • 20,649
  • 9
  • 66
  • 83
  • Thank you, it works and I know where is my mistake now. But since Chris's answer is a better solution, so I accept his. Thanks again ~ – Freewind Jul 17 '13 at 16:07
  • just a small correction `body.substring(prefix.length)` would be `body.substring(prefix.length+1)` – sh0umik Sep 23 '19 at 16:07
1

The easiest approach is:

import 'dart:convert';

final result = base64Decode(stringValue);
Andrey Gordeev
  • 30,606
  • 13
  • 135
  • 162