我想做的是:
使用我自己的PictureActivity*拍摄照片并添加EXIF (地理标签)数据
*:实现SurfaceHolder.Callback和使用Camera
不起作用的是:
添加EXIF GPS数据
我尝试过的:
使用ExifInterface和手动设置Camera.Parameters (使用设置GPS元数据的特定方法和使用params.set(String, Value))。
我正在使用FlickrJ上传图片到Flickr (是的,我已经将Flickr设置为导入GPS数据--其他图片可以正常工作),但是这个工具也显示EXIF:http://regex.info/exif.cgi文件中没有GPS数据。
我遗漏了什么?
(Android 2.2,HTC Desire)
编辑:
使用硬编码的虚拟
将相机设置为已尝试
下面是手动设置参数的代码(先尝试删除GPS数据和不删除GPS数据,在set(String, Value)中也提到过):
@Override
public void surfaceCreated(SurfaceHolder holder) {
mCamera = Camera.open();
Camera.Parameters p = mCamera.getParameters();
p.setPreviewSize(p.getPreviewSize().width, p.getPreviewSize().height);
Log.e("PictureActivity", "EXIF: "+AGlanceLocationListener.getLatitude());
p.removeGpsData();
p.setGpsLatitude( AGlanceLocationListener.getLatitude() );
p.setGpsLongitude( AGlanceLocationListener.getLongitude() );
p.setGpsAltitude( AGlanceLocationListener.getAltitude() );
p.setGpsTimestamp( AGlanceLocationListener.getTime() );
mCamera.setParameters(p);
}下面是使用ExifInterface的代码
//Save EXIF location data to JPEG
ExifInterface exif;
try {
exif = new ExifInterface("/sdcard/DCIM/"+filename+".jpeg");
exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE,
String.valueOf(AGlanceLocationListener.getLatitude()));
exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE,
String.valueOf(AGlanceLocationListener.getLongitude()));
exif.saveAttributes();
} catch (IOException e) {
Log.e("PictureActivity", e.getLocalizedMessage());
}以下是将JPEG文件写入SD卡的代码:
Camera.PictureCallback jpegCallback = new Camera.PictureCallback() {
public void onPictureTaken(byte[] imageData, Camera c)
{
// Bitmap pic = BitmapFactory.decodeByteArray(imageData, 0, imageData.length);
String day = String.valueOf(Calendar.getInstance().getTime().getDay());
String hour = String.valueOf(Calendar.getInstance().getTime().getHours());
String minute = String.valueOf(Calendar.getInstance().getTime().getMinutes());
String second = String.valueOf(Calendar.getInstance().getTime().getSeconds());
filename = "Billede"+day+hour+minute+second;
try {
FileOutputStream fos = new FileOutputStream(new File("/sdcard/DCIM/"+filename+".jpeg"));
fos.write(imageData);
fos.flush();
fos.close();
} catch (Exception e) {
e.printStackTrace();
}
if(imageData != null){
Intent mIntent = new Intent();
setResult(0,mIntent);
PictureActivity.this.showDialog(0);
}
}
};我也尝试过从Bitmap中写入图像(不起作用),另外还有另一个问题,在这里使用FileOutputStream写入报告有效
发布于 2012-05-21 20:57:36
找到问题了:
查看SD卡的原始图像显示:
如果使用EXIFInterface,则
用于EXIFInterface的Android API有this documentation
公共静态最终字符串TAG_GPS_LONGITUDE
自: API级别5
字符串。格式为"num1/denom1,num2/denom2,num3/denom3“。
常量值:"GPSLongitude“
我的原始代码的问题是,我传递的GPS坐标是以十进制度为单位的--在Location对象上调用getLatitude/getLogitude得到的坐标是以十进制度为单位的。EXIFInterface期望坐标以度、分、秒为单位,然后写成有理数(这是EXIF规范的一部分)。更多关于全球定位系统坐标格式和转换here。
Here是另一个问题/答案,它解释了如何将十进制度转换为度分秒。
使用此代码,可以在EXIF中正确写入GPS坐标,并且Flickr在导入数据时没有任何问题:
ExifInterface exif;
double latitude = AGlanceLocationListener.getLatitude();
double longitude = AGlanceLocationListener.getLongitude();
try {
exif = new ExifInterface("/sdcard/DCIM/"+filename+".jpeg");
int num1Lat = (int)Math.floor(latitude);
int num2Lat = (int)Math.floor((latitude - num1Lat) * 60);
double num3Lat = (latitude - ((double)num1Lat+((double)num2Lat/60))) * 3600000;
int num1Lon = (int)Math.floor(longitude);
int num2Lon = (int)Math.floor((longitude - num1Lon) * 60);
double num3Lon = (longitude - ((double)num1Lon+((double)num2Lon/60))) * 3600000;
exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE, num1Lat+"/1,"+num2Lat+"/1,"+num3Lat+"/1000");
exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE, num1Lon+"/1,"+num2Lon+"/1,"+num3Lon+"/1000");
if (latitude > 0) {
exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE_REF, "N");
} else {
exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE_REF, "S");
}
if (longitude > 0) {
exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE_REF, "E");
} else {
exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE_REF, "W");
}
exif.saveAttributes();
} catch (IOException e) {
Log.e("PictureActivity", e.getLocalizedMessage());
} 注意:使用度、分、秒时,还需要设置GPS参考属性(N、S、E、W)。
发布于 2013-03-27 03:40:57
不幸的是,这只适用于半球的四分之一。格林威治以东赤道以北。我就是这么猜到你一定住在那里的:)。你的'Math.floor‘会使所有的负值都是错误的(比如-105变成-106)。这是同样的事情,即使在美国也应该有效。
public void loc2Exif(String flNm, Location loc) {
try {
ExifInterface ef = new ExifInterface(flNm);
ef.setAttribute(ExifInterface.TAG_GPS_LATITUDE, dec2DMS(loc.getLatitude()));
ef.setAttribute(ExifInterface.TAG_GPS_LONGITUDE,dec2DMS(loc.getLongitude()));
if (loc.getLatitude() > 0)
ef.setAttribute(ExifInterface.TAG_GPS_LATITUDE_REF, "N");
else
ef.setAttribute(ExifInterface.TAG_GPS_LATITUDE_REF, "S");
if (loc.getLongitude()>0)
ef.setAttribute(ExifInterface.TAG_GPS_LONGITUDE_REF, "E");
else
ef.setAttribute(ExifInterface.TAG_GPS_LONGITUDE_REF, "W");
ef.saveAttributes();
} catch (IOException e) {}
}
//-----------------------------------------------------------------------------------
String dec2DMS(double coord) {
coord = coord > 0 ? coord : -coord; // -105.9876543 -> 105.9876543
String sOut = Integer.toString((int)coord) + "/1,"; // 105/1,
coord = (coord % 1) * 60; // .987654321 * 60 = 59.259258
sOut = sOut + Integer.toString((int)coord) + "/1,"; // 105/1,59/1,
coord = (coord % 1) * 60000; // .259258 * 60000 = 15555
sOut = sOut + Integer.toString((int)coord) + "/1000"; // 105/1,59/1,15555/1000
return sOut;
}..。一旦你让我开始,这是相反的
public Location exif2Loc(String flNm) {
String sLat = "", sLatR = "", sLon = "", sLonR = "";
try {
ExifInterface ef = new ExifInterface(flNm);
sLat = ef.getAttribute(ExifInterface.TAG_GPS_LATITUDE);
sLon = ef.getAttribute(ExifInterface.TAG_GPS_LONGITUDE);
sLatR = ef.getAttribute(ExifInterface.TAG_GPS_LATITUDE_REF);
sLonR = ef.getAttribute(ExifInterface.TAG_GPS_LONGITUDE_REF);
} catch (IOException e) {return null;}
double lat = dms2Dbl(sLat);
if (lat > 180.0) return null;
double lon = dms2Dbl(sLon);
if (lon > 180.0) return null;
lat = sLatR.contains("S") ? -lat : lat;
lon = sLonR.contains("W") ? -lon : lon;
Location loc = new Location("exif");
loc.setLatitude(lat);
loc.setLongitude(lon);
return loc;
}
//-------------------------------------------------------------------------
double dms2Dbl(String sDMS){
double dRV = 999.0;
try {
String[] DMSs = sDMS.split(",", 3);
String s[] = DMSs[0].split("/", 2);
dRV = (new Double(s[0])/new Double(s[1]));
s = DMSs[1].split("/", 2);
dRV += ((new Double(s[0])/new Double(s[1]))/60);
s = DMSs[2].split("/", 2);
dRV += ((new Double(s[0])/new Double(s[1]))/3600);
} catch (Exception e) {}
return dRV;
}..。总有一天,我会开始写漂亮的代码。地理标记快乐,肖恩
发布于 2015-02-06 17:13:00
此解决方案将满足lat/lng负值和正值的需求:
static public boolean setGeoTag(File image, LatLng geoTag) {
if (geoTag != null) {
try {
ExifInterface exif = new ExifInterface(
image.getAbsolutePath());
double latitude = Math.abs(geoTag.latitude);
double longitude = Math.abs(geoTag.longitude);
int num1Lat = (int) Math.floor(latitude);
int num2Lat = (int) Math.floor((latitude - num1Lat) * 60);
double num3Lat = (latitude - ((double) num1Lat + ((double) num2Lat / 60))) * 3600000;
int num1Lon = (int) Math.floor(longitude);
int num2Lon = (int) Math.floor((longitude - num1Lon) * 60);
double num3Lon = (longitude - ((double) num1Lon + ((double) num2Lon / 60))) * 3600000;
String lat = num1Lat + "/1," + num2Lat + "/1," + num3Lat + "/1000";
String lon = num1Lon + "/1," + num2Lon + "/1," + num3Lon + "/1000";
if (geoTag.latitude > 0) {
exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE_REF, "N");
} else {
exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE_REF, "S");
}
if (geoTag.longitude > 0) {
exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE_REF, "E");
} else {
exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE_REF, "W");
}
exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE, lat);
exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE, lon);
exif.saveAttributes();
} catch (IOException e) {
e.printStackTrace();
return false;
}
} else {
return false;
}
return true;
}https://stackoverflow.com/questions/10531544
复制相似问题