我有一个字符串,由我的应用程序从网络浏览器中获取。字符串包含PNG图像,并编码为base64。
这不是我的愚蠢想法,将图像转换为base64字符串))这是做web-broser。 HTML5 canvas对象在做这件事时,需要将图像发送到某个地方。
那么,如何将此字符串解码为图像?
在Delphi中,我编写了我的Web服务器,它可以传输数据。
在PHP中,这可以像that一样解决:
function base64_to_jpeg( $base64_string, $output_file ) {
$ifp = fopen( $output_file, "wb" );
fwrite( $ifp, base64_decode( $base64_string) );
fclose( $ifp );
return( $output_file );
}
$image = base64_to_jpeg( $my_base64_string, 'tmp.jpg' );
但在delphi中,解码后:
if (sImg <> EmptyStr) then
begin
Coder := TIdDecoderMIME.Create(nil);
MS := TMemoryStream.Create;
try
Coder.DecodeStream(sImg, MS);
MS.SaveToFile(myDir + '1.png');
finally
FreeAndNil(MS);
FreeAndNil(Coder);
end;
end;
我有一个黑色的马列维奇广场。
我还尝试了RTL的EncdDesd模块中的函数,
if (sImg <> EmptyStr) then
begin
MS := TMemoryStream.Create;
SS := TStringStream.Create(sImg);
try
DecodeStream(SS, MS);
MS.SaveToFile(myDir + '1.png');
finally
FreeAndNil(MS);
FreeAndNil(SS);
end;
end;
我得到了相同的结果。
我编写了web扩展,用于从用户Web浏览器获取一些信息,使用TidHTTPServer获取Web服务器。 从JS我捕获web截图,并从exe中删除一些数据,之后,我将此字符串发送到服务器:
var Img = capture().toDataURL();
var Img = Img.replace('data:image/png;base64,','');
if (FLA.AddNewURL(sN,sD,sU,sI,Img)) {
window.close();
} else {
alert('can not establish connection!');
};
...
AddNewURL: function (aName, aDesc,sURL, aOwnerId,aImg) {
var http = new XMLHttpRequest();
var params = 'Owner=' + aOwnerId + '&Name=' + aName + '&Desc=' + aDesc+ '&URL=' + sURL;
if (aImg) {
params = params +'&Img='+aImg;
};
http.open("POST", "http://" + this.host + ":" + this.port + "/addurl", false);
http.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset=UTF-8');
http.send(params);
return (http.status === 200);
},
我的网络服务器像这样重复这些数据:
if (ARequestInfo.Document = '/addurl') then
begin
Id := ARequestInfo.Params.Values['Owner'];
sName := ARequestInfo.Params.Values['Name'];
sDesc := ARequestInfo.Params.Values['Desc'];
sURL := ARequestInfo.Params.Values['URL'];
sImg := ARequestInfo.Params.Values['Img'];
RootLI := nil;
LinksManager.FindByID(Id, RootLI);
if Assigned(RootLI) then
begin
LI := TLinkItem.Create(nil);
LI.name := sName;
LI.Description := sDesc;
LI.URL := sURL;
if (sImg <> EmptyStr) then
begin
MS := TMemoryStream.Create;
SS := TStringStream.Create(sImg);
try
DecodeStream(SS, MS);
MS.SaveToFile(myDir + '1.png');
finally
FreeAndNil(MS);
FreeAndNil(SS);
end;
end;
RootLI.Add(LI, True);
AResponseInfo.ResponseText := 'OK';
end
else
AResponseInfo.ResponseText := 'FAIL';
end;
毕竟我有图像,只包含黑色。
答案 0 :(得分:0)
我通过其他方式解决了我的问题。 在JS中,我通过头文件(Ajax.setRequestHeader)和我在POST流中发送的文件发送数据。 在DecodeStream之后,从EncdDecd正常工作,我从字符串中获取了图像。
Id := ARequestInfo.RawHeaders.Values['OwnerId'];
sName := ARequestInfo.RawHeaders.Values['Name'];
sDesc := ARequestInfo.RawHeaders.Values['Desc'];
sURL := ARequestInfo.RawHeaders.Values['URL'];
...
if (ARequestInfo.PostStream <> nil) then //We have the image!
begin
MS := TMemoryStream.Create;
MS2 := TMemoryStream.Create;
try
MS2.LoadFromStream(ARequestInfo.PostStream);
DecodeStream(MS2, MS);
MS.SaveToFile(myDir + '1.png');
finally
FreeAndNil(MS);
FreeAndNil(MS2);
end;
end;
您也可以尝试混合使用其他多部分/表单数据编码。但Indy没有实现以这种格式解析回收数据的类。