4 import java.nio.channels.FileChannel;
\r
5 import java.text.DateFormat;
\r
6 import java.text.DecimalFormat;
\r
7 import java.text.ParseException;
\r
8 import java.text.SimpleDateFormat;
\r
9 import java.util.ArrayList;
\r
10 import java.util.Arrays;
\r
11 import java.util.Calendar;
\r
12 import java.util.Comparator;
\r
13 import java.util.Date;
\r
14 import java.util.Map;
\r
15 import java.util.TimeZone;
\r
16 import java.util.logging.LogManager;
\r
17 import java.util.logging.Logger;
\r
19 import javax.xml.parsers.*;
\r
20 import javax.xml.transform.OutputKeys;
\r
21 import javax.xml.transform.Transformer;
\r
22 import javax.xml.transform.TransformerException;
\r
23 import javax.xml.transform.TransformerFactory;
\r
24 import javax.xml.transform.dom.DOMSource;
\r
25 import javax.xml.transform.stream.StreamResult;
\r
27 import org.apache.commons.imaging.ImageReadException;
\r
28 import org.apache.commons.imaging.ImageWriteException;
\r
29 import org.apache.commons.imaging.Imaging;
\r
30 import org.apache.commons.imaging.common.ImageMetadata;
\r
31 import org.apache.commons.imaging.common.RationalNumber;
\r
32 import org.apache.commons.imaging.formats.jpeg.JpegImageMetadata;
\r
33 import org.apache.commons.imaging.formats.jpeg.exif.ExifRewriter;
\r
34 import org.apache.commons.imaging.formats.tiff.TiffImageMetadata;
\r
35 import org.apache.commons.imaging.formats.tiff.constants.ExifTagConstants;
\r
36 import org.apache.commons.imaging.formats.tiff.constants.GpsTagConstants;
\r
37 import org.apache.commons.imaging.formats.tiff.write.TiffOutputDirectory;
\r
38 import org.apache.commons.imaging.formats.tiff.write.TiffOutputSet;
\r
39 import org.w3c.dom.*;
\r
40 import org.xml.sax.SAXException;
\r
42 public class ImportPicture extends Thread {
\r
44 * 実行中に発生したExceptionを保持する場所
\r
46 public Exception ex = null;
\r
49 * ログ設定プロパティファイルのファイル内容
\r
51 protected static final String LOGGING_PROPERTIES_DATA
\r
52 = "handlers=java.util.logging.ConsoleHandler\n"
\r
54 + "java.util.logging.ConsoleHandler.level=INFO\n"
\r
55 + "java.util.logging.ConsoleHandler.formatter=osm.jp.gpx.YuuLogFormatter";
\r
58 * static initializer によるログ設定の初期化
\r
60 public static final Logger logger = Logger.getLogger("CommandLogging");
\r
62 InputStream inStream = null;
\r
64 inStream = new ByteArrayInputStream(LOGGING_PROPERTIES_DATA.getBytes("UTF-8"));
\r
66 LogManager.getLogManager().readConfiguration(inStream);
\r
67 logger.config("ログ設定: LogManagerを設定しました。");
\r
69 catch (IOException e) {
\r
70 logger.warning("ログ設定: LogManager設定の際に例外が発生しました。:" + e.toString());
\r
73 catch (UnsupportedEncodingException e) {
\r
74 logger.severe("ログ設定: UTF-8エンコーディングがサポートされていません。:" + e.toString());
\r
78 if (inStream != null) {
\r
81 } catch (IOException e) {
\r
82 logger.warning("ログ設定: ログ設定プロパティファイルのストリームクローズ時に例外が発生しました。:"+ e.toString());
\r
90 * 画像ファイルをGPXファイルに取り込みます。
\r
92 * ・画像ファイルの更新日付をその画像の撮影日時とします。(Exi情報は無視します)
\r
93 * ※ 対象とするファイルは'*.jpg'のみ
\r
94 * ・精確な時刻との時差を入力することで、撮影日時を補正します。
\r
95 * ・画像ファイルの更新日付リストをCSV形式のファイルとして出力する。
\r
96 * ・・結果は、取り込み元のGPXファイルとは別に、元ファイル名にアンダーバー「_」を付加した.ファイルに出力します。
\r
98 * exp) $ java -cp .:AdjustTime.jar:commons-imaging-1.0-SNAPSHOT.jar [AdjustTime.ini]
\r
99 * exp) > java -cp .;AdjustTime.jar;commons-imaging-1.0-SNAPSHOT.jar [AdjustTime.ini]
\r
102 * argv[0] = INIファイルのパス名
\r
105 * argv[0] = 画像ファイルが格納されているディレクトリ --> imgDir
\r
106 * argv[1] = 時刻補正の基準とする画像ファイル --> baseFile
\r
107 * argv[2] = 基準画像ファイルの精確な撮影日時 "yyyy-MM-dd'T'HH:mm:ss" --> timeStr
\r
108 * argv[3] = 出力先フォルダ --> outDir
\r
109 * argv[4] = 撮影位置をロギングしたGPXファイル --> gpxDir
\r
111 * @throws IOException
\r
112 * @throws ImageReadException
\r
114 public static void main(String[] argv) throws Exception
\r
118 String paramFilePath = ((argv.length < 1) ? AppParameters.FILE_PATH : argv[0]);
\r
119 System.out.println("Param File = '"+ paramFilePath +"'");
\r
120 ImportPicture obj = new ImportPicture();
\r
121 obj.params = new AppParameters(paramFilePath);
\r
123 System.out.println(" - param: "+ AppParameters.IMG_TIME +"="+ obj.params.getProperty(AppParameters.IMG_TIME) );
\r
124 System.out.println(" - param: "+ AppParameters.IMG_BASE_FILE +"="+ obj.params.getProperty(AppParameters.IMG_BASE_FILE) );
\r
125 System.out.println(" - param: "+ AppParameters.GPX_BASETIME +"="+ obj.params.getProperty(AppParameters.GPX_BASETIME) );
\r
126 System.out.println(" - param: "+ AppParameters.IMG_SOURCE_FOLDER +"="+ obj.params.getProperty(AppParameters.IMG_SOURCE_FOLDER) );
\r
127 System.out.println(" - param: "+ AppParameters.IMG_OUTPUT_FOLDER +"="+ obj.params.getProperty(AppParameters.IMG_OUTPUT_FOLDER) );
\r
128 System.out.println(" - param: "+ AppParameters.IMG_OUTPUT +"="+ obj.params.getProperty(AppParameters.IMG_OUTPUT));
\r
129 System.out.println(" - param: "+ AppParameters.IMG_OUTPUT_ALL +"="+ obj.param_ImgOutputAll);
\r
130 System.out.println(" - param: "+ AppParameters.IMG_OUTPUT_EXIF +"= "+ String.valueOf(obj.exif));
\r
131 System.out.println(" - param: "+ AppParameters.GPX_SOURCE_FOLDER +"="+ obj.param_GpxSourceFolder);
\r
132 System.out.println(" - param: "+ AppParameters.GPX_OUTPUT_WPT +"="+ obj.param_GpxOutputWpt);
\r
133 System.out.println(" - param: "+ AppParameters.GPX_OVERWRITE_MAGVAR +"="+ Complementation.param_GpxOverwriteMagvar);
\r
134 System.out.println(" - param: "+ AppParameters.GPX_OUTPUT_SPEED +"="+ Complementation.param_GpxOutputSpeed);
\r
135 System.out.println(" - param: "+ AppParameters.GPX_GPXSPLIT +"="+ obj.param_GpxSplit);
\r
136 System.out.println(" - param: "+ AppParameters.GPX_NO_FIRST_NODE +"="+ ImportPicture.param_GpxNoFirstNode);
\r
137 System.out.println(" - param: "+ AppParameters.GPX_REUSE +"="+ obj.param_GpxReuse);
\r
140 // argv[0] --> AppParameters.IMG_SOURCE_FOLDER に置き換え
\r
141 obj.imgDir = new File(obj.params.getProperty(AppParameters.IMG_SOURCE_FOLDER));
\r
143 // 基準時刻(ファイル更新日時 | EXIF撮影日時)
\r
144 obj.exifBase = (obj.params.getProperty(AppParameters.GPX_BASETIME).equals("EXIF_TIME") ? true : false);
\r
146 // 基準時刻ファイルの「更新日時」を使って時刻合わせを行う。
\r
147 // argv[1] --> AppParameters.IMG_BASE_FILE に置き換え
\r
148 File baseFile = new File(obj.imgDir, obj.params.getProperty(AppParameters.IMG_BASE_FILE));
\r
149 if (obj.exifBase) {
\r
150 ImageMetadata meta = Imaging.getMetadata(baseFile);
\r
151 JpegImageMetadata jpegMetadata = (JpegImageMetadata)meta;
\r
152 if (jpegMetadata == null) {
\r
153 System.out.println("'"+ baseFile.getAbsolutePath() +"' にEXIF情報がありません");
\r
156 TiffImageMetadata exif = jpegMetadata.getExif();
\r
157 if (exif == null) {
\r
158 System.out.println("'"+ baseFile.getAbsolutePath() +"' にEXIF情報がありません");
\r
161 String dateTimeOriginal = exif.getFieldValue(ExifTagConstants.EXIF_TAG_DATE_TIME_ORIGINAL)[0];
\r
162 long lastModifyTime = ImportPicture.toEXIFDate(dateTimeOriginal).getTime();
\r
163 imgtime = new Date(lastModifyTime);
\r
166 imgtime = new Date(baseFile.lastModified());
\r
170 // argv[3] --> AppParameters.IMG_OUTPUT に置き換え
\r
171 obj.outDir = new File(obj.params.getProperty(AppParameters.IMG_OUTPUT_FOLDER));
\r
174 String paramStr = obj.params.getProperty(AppParameters.GPX_GPXSPLIT);
\r
175 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
176 obj.param_GpxSplit = true;
\r
179 paramStr = obj.params.getProperty(AppParameters.GPX_NO_FIRST_NODE);
\r
180 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
181 ImportPicture.param_GpxNoFirstNode = true;
\r
184 paramStr = obj.params.getProperty(AppParameters.GPX_REUSE);
\r
185 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
186 obj.param_GpxReuse = true;
\r
189 paramStr = obj.params.getProperty(AppParameters.IMG_OUTPUT_ALL);
\r
190 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
191 obj.param_ImgOutputAll = true;
\r
194 paramStr = obj.params.getProperty(AppParameters.GPX_OUTPUT_WPT);
\r
195 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
196 obj.param_GpxOutputWpt = true;
\r
199 paramStr = obj.params.getProperty(AppParameters.GPX_OVERWRITE_MAGVAR);
\r
200 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
201 Complementation.param_GpxOverwriteMagvar = true;
\r
204 paramStr = obj.params.getProperty(AppParameters.GPX_OUTPUT_SPEED);
\r
205 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
206 Complementation.param_GpxOutputSpeed = true;
\r
209 paramStr = obj.params.getProperty(AppParameters.GPX_SOURCE_FOLDER);
\r
210 if (paramStr != null) {
\r
211 obj.param_GpxSourceFolder = new String(paramStr);
\r
212 obj.gpxDir = new File(obj.param_GpxSourceFolder);
\r
213 if (!obj.gpxDir.exists()) {
\r
214 // GPXファイルまたはディレクトリが存在しません。
\r
215 System.out.println("GPXファイルまたはディレクトリが存在しません。('"+ paramStr +"')");
\r
220 obj.gpxDir = obj.imgDir;
\r
223 // 指定されたディレクトリ内のGPXファイルすべてを対象とする
\r
224 if (obj.gpxDir.isDirectory()) {
\r
225 File[] files = obj.gpxDir.listFiles();
\r
226 if (files == null) {
\r
227 // 対象となるGPXファイルがありませんでした。
\r
228 System.out.println("対象となるGPXファイルがありませんでした。('"+ obj.gpxDir.getAbsolutePath() +"')");
\r
231 if (obj.param_ImgOutputAll && (files.length > 1)) {
\r
232 System.out.println("複数のGPXファイルがあるときには、'IMG.OUTPUT_ALL'オプションは指定できません。");
\r
236 java.util.Arrays.sort(files, new java.util.Comparator<File>() {
\r
238 public int compare(File file1, File file2){
\r
239 return file1.getName().compareTo(file2.getName());
\r
242 for (File file : files) {
\r
243 if (file.isFile()) {
\r
244 String filename = file.getName().toUpperCase();
\r
245 if (filename.toUpperCase().endsWith(".GPX")) {
\r
246 if (!filename.toUpperCase().endsWith("_.GPX") || obj.param_GpxReuse) {
\r
247 obj.gpxFiles.add(file);
\r
254 obj.gpxFiles.add(obj.gpxDir);
\r
257 paramStr = obj.params.getProperty(AppParameters.IMG_OUTPUT_EXIF);
\r
258 if ((paramStr != null) && (paramStr.equals(Boolean.toString(true)))) {
\r
262 String timeStr = obj.params.getProperty(AppParameters.IMG_TIME);
\r
264 Date t = toUTCDate(timeStr);
\r
265 obj.delta = t.getTime() - imgtime.getTime();
\r
267 catch (ParseException e) {
\r
268 System.out.println("'"+ timeStr +"' の書式が違います("+ TIME_FORMAT_STRING +")");
\r
275 } catch(InterruptedException end) {}
\r
276 if (obj.ex != null) {
\r
281 public File gpxDir;
\r
282 public File imgDir;
\r
283 public File outDir;
\r
284 public long delta = 0;
\r
285 public boolean exif = false;
\r
286 public boolean exifBase = false;
\r
287 public ArrayList<File> gpxFiles = new ArrayList<>();
\r
288 public AppParameters params;
\r
289 public boolean param_GpxSplit = false;
\r
290 public static boolean param_GpxNoFirstNode = false;
\r
291 public boolean param_GpxReuse = false;
\r
292 public boolean param_GpxOutputWpt = true;
\r
293 public boolean param_ImgOutputAll = false;
\r
294 public String param_GpxSourceFolder = ".";
\r
297 public static final String TIME_FORMAT_STRING = "yyyy-MM-dd'T'HH:mm:ss'Z'";
\r
298 private static final String EXIF_DATE_TIME_FORMAT_STRING = "yyyy:MM:dd HH:mm:ss";
\r
301 public void run() {
\r
304 <wpt lat="35.25714922" lon="139.15490497">
\r
305 <ele>62.099998474121094</ele>
\r
306 <time>2012-06-11T00:44:38Z</time>
\r
308 <name><![CDATA[写真]]></name>
\r
309 <cmt><![CDATA[精度: 3.0m]]></cmt>
\r
310 <link href="2012-06-11_09-44-38.jpg">
\r
311 <text>2012-06-11_09-44-38.jpg</text>
\r
317 if (params.getProperty(AppParameters.IMG_OUTPUT).equals(Boolean.toString(true))) {
\r
318 outDir = new File(outDir, imgDir.getName());
\r
324 for (File gpxFile : this.gpxFiles) {
\r
325 procGPXfile(gpxFile);
\r
328 catch(ParserConfigurationException | DOMException | SAXException | IOException | ParseException | ImageReadException | ImageWriteException | IllegalArgumentException | TransformerException e) {
\r
329 e.printStackTrace();
\r
330 this.ex = new Exception(e);
\r
337 * @throws ParserConfigurationException
\r
338 * @throws IOException
\r
339 * @throws SAXException
\r
340 * @throws ParseException
\r
341 * @throws ImageWriteException
\r
342 * @throws ImageReadException
\r
343 * @throws TransformerException
\r
345 void procGPXfile(File gpxFile) throws ParserConfigurationException, SAXException, IOException, ParseException, ImageReadException, ImageWriteException, TransformerException {
\r
346 DocumentBuilderFactory factory = null;
\r
347 DocumentBuilder builder = null;
\r
348 ElementMapTRKSEG mapTRKSEG = null;
\r
353 String fileName = gpxFile.getName();
\r
354 String iStr = fileName.substring(0, fileName.length() - 4);
\r
356 File outputFile = new File(outDir, iStr +"_.gpx");
\r
357 System.out.println(gpxFile.getAbsolutePath() + " => "+ outputFile.getAbsolutePath());
\r
358 System.out.println(" 時差: "+ (delta / 1000) +"(sec)");
\r
359 System.out.println(" Target GPX: ["+ gpxFile.getAbsolutePath() +"]");
\r
360 System.out.println(" EXIF: "+ (exif ? ("convert to '" + outDir.getAbsolutePath() +"'") : "off"));
\r
361 System.out.println();
\r
363 factory = DocumentBuilderFactory.newInstance();
\r
364 builder = factory.newDocumentBuilder();
\r
365 factory.setIgnoringElementContentWhitespace(true);
\r
366 factory.setIgnoringComments(true);
\r
367 factory.setValidating(true);
\r
370 mapTRKSEG = new ElementMapTRKSEG();
\r
371 document = mapTRKSEG.parse(gpxFile);
\r
373 // パースされた mapTRKSEG の中身を出力する
\r
374 mapTRKSEG.printinfo();
\r
376 // GPX file --> Node root
\r
377 gpx = builder.parse(gpxFile).getFirstChild();
\r
379 // imgDir内の画像ファイルを処理する
\r
380 System.out.println("|--------------------------------|--------------------|--------------------|--------------|--------------|--------|------|------|");
\r
381 System.out.println("| name | Camera Time | GPStime | Latitude | Longitude | ele |magvar| km/h |");
\r
382 System.out.println("|--------------------------------|--------------------|--------------------|--------------|--------------|--------|------|------|");
\r
383 proc(imgDir, delta, mapTRKSEG, exif, gpx);
\r
384 System.out.println("|--------------------------------|--------------------|--------------------|--------------|--------------|--------|------|------|");
\r
387 outputFile.getParentFile().mkdirs();
\r
388 DOMSource source = new DOMSource(gpx);
\r
389 FileOutputStream os = new FileOutputStream(outputFile);
\r
390 StreamResult result = new StreamResult(os);
\r
391 TransformerFactory transFactory = TransformerFactory.newInstance();
\r
392 Transformer transformer = transFactory.newTransformer();
\r
393 transformer.setOutputProperty(OutputKeys.INDENT, "yes");
\r
394 transformer.setOutputProperty(OutputKeys.METHOD, "xml");
\r
395 transformer.transform(source, result);
\r
397 os = new FileOutputStream(outputFile);
\r
398 result = new StreamResult(os);
\r
399 transformer.transform(source, result);
\r
405 * @throws ParseException
\r
406 * @throws IOException
\r
407 * @throws ImageReadException
\r
408 * @throws ImageWriteException
\r
410 boolean proc(File dir, long delta, ElementMapTRKSEG mapTRKSEG, boolean exifWrite, Node gpx) throws ParseException, ImageReadException, IOException, ImageWriteException {
\r
411 boolean ret = false;
\r
412 File[] files = dir.listFiles(new JpegFileFilter());
\r
413 Arrays.sort(files, new FileSort());
\r
414 for (File image : files) {
\r
415 System.out.print(String.format("|%-32s|", image.getName()));
\r
416 if (image.isDirectory()) {
\r
417 ret = proc(image, delta, mapTRKSEG, exifWrite, gpx);
\r
421 String imageName = image.getName();
\r
422 if (!checkFile(imageName)) {
\r
423 System.out.println(String.format("%20s ", "it is not image file."));
\r
427 Discripter result = procImageFile(image, delta, mapTRKSEG, exifWrite, gpx);
\r
429 switch (result.control) {
\r
430 case Discripter.CONTINUE:
\r
432 case Discripter.BREAK:
\r
440 static final int NEXT = 0;
\r
441 static final int CONTINUE = -1;
\r
442 static final int BREAK = 1;
\r
444 public boolean ret;
\r
445 public int control;
\r
446 public Discripter(boolean ret) {
\r
448 this.control = Discripter.NEXT;
\r
452 Discripter procImageFile(File imageFile, long delta, ElementMapTRKSEG mapTRKSEG, boolean exifWrite, Node gpx) throws ParseException, ImageReadException, IOException, ImageWriteException {
\r
453 Discripter result = new Discripter(false);
\r
455 // itime <-- 画像ファイルの撮影時刻
\r
456 // ファイルの更新日時/EXIFの撮影日時
\r
457 Date itime = new Date(imageFile.lastModified());
\r
458 if (this.exifBase) {
\r
459 ImageMetadata meta = Imaging.getMetadata(imageFile);
\r
460 JpegImageMetadata jpegMetadata = (JpegImageMetadata)meta;
\r
461 if (jpegMetadata == null) {
\r
462 System.out.println("'"+ imageFile.getAbsolutePath() +"' にEXIF情報がありません");
\r
463 result.control = Discripter.CONTINUE;
\r
466 TiffImageMetadata exif = jpegMetadata.getExif();
\r
467 if (exif == null) {
\r
468 System.out.println("'"+ imageFile.getAbsolutePath() +"' にEXIF情報がありません");
\r
469 result.control = Discripter.CONTINUE;
\r
472 String dateTimeOriginal = exif.getFieldValue(ExifTagConstants.EXIF_TAG_DATE_TIME_ORIGINAL)[0];
\r
473 itime = ImportPicture.toEXIFDate(dateTimeOriginal);
\r
475 System.out.print(String.format("%20s|", toUTCString(itime)));
\r
477 // uktime <-- 画像撮影時刻に対応するGPX時刻(補正日時)
\r
478 Date correctedtime = new Date(itime.getTime() + delta);
\r
479 System.out.print(String.format("%20s|", toUTCString(correctedtime)));
\r
481 // 時刻uktimeにおける<magver>をtrkptに追加する
\r
482 String eleStr = "-";
\r
483 String magvarStr = "-";
\r
484 String speedStr = "-";
\r
485 double latitude = 90.5D;
\r
486 double longitude = 180.5D;
\r
487 TagTrkpt trkptT = null;
\r
489 for (Map.Entry<Date,ElementMapTRKPT> map : mapTRKSEG.entrySet()) {
\r
490 ElementMapTRKPT mapTRKPT = map.getValue();
\r
491 trkptT = mapTRKPT.getValue(correctedtime);
\r
492 if (trkptT != null) {
\r
497 if (trkptT == null) {
\r
498 System.out.print(String.format("%-14s|%-14s|", "", ""));
\r
499 System.out.println(String.format("%8s|%6s|%6s|", "", "", ""));
\r
500 if (!this.param_ImgOutputAll) {
\r
501 result.control = Discripter.CONTINUE;
\r
506 latitude = trkptT.lat;
\r
507 longitude = trkptT.lon;
\r
509 if (trkptT.eleStr != null) {
\r
510 eleStr = new String(trkptT.eleStr);
\r
513 if (trkptT.magvarStr != null) {
\r
514 magvarStr = new String(trkptT.magvarStr);
\r
517 if (trkptT.speedStr != null) {
\r
518 speedStr = new String(trkptT.speedStr);
\r
520 //System.out.print(String.format("%-14s|%-14s|", (new Double(latitude)).toString(), (new Double(longitude)).toString()));
\r
521 System.out.print(String.format("%14.10f|%14.10f|", latitude, longitude));
\r
522 System.out.println(String.format("%8s|%6s|%6s|", eleStr, magvarStr, speedStr));
\r
529 exifWrite(imageFile, correctedtime, trkptT);
\r
531 if (Boolean.parseBoolean(params.getProperty(AppParameters.GPX_OUTPUT_WPT))) {
\r
532 if (trkptT != null) {
\r
533 Element temp = createWptTag(imageFile, itime.getTime(), trkptT.trkpt);
\r
534 gpx.appendChild(temp);
\r
539 if (this.param_ImgOutputAll) {
\r
540 // EXIFの変換を伴わない単純なファイルコピー
\r
541 FileInputStream sStream = new FileInputStream(imageFile);
\r
542 FileInputStream dStream = new FileInputStream(new File(outDir, imageFile.getName()));
\r
543 FileChannel srcChannel = sStream.getChannel();
\r
544 FileChannel destChannel = dStream.getChannel();
\r
546 srcChannel.transferTo(0, srcChannel.size(), destChannel);
\r
549 srcChannel.close();
\r
550 destChannel.close();
\r
556 result.control = Discripter.NEXT;
\r
560 void exifWrite(File imageFile, Date correctedtime, TagTrkpt trkptT) throws ImageReadException, IOException, ImageWriteException {
\r
561 DecimalFormat yearFormatter = new DecimalFormat("0000");
\r
562 DecimalFormat monthFormatter = new DecimalFormat("00");
\r
563 DecimalFormat dayFormatter = new DecimalFormat("00");
\r
565 TiffOutputSet outputSet = null;
\r
567 ImageMetadata meta = Imaging.getMetadata(imageFile);
\r
568 JpegImageMetadata jpegMetadata = (JpegImageMetadata)meta;
\r
569 if (jpegMetadata != null) {
\r
570 TiffImageMetadata exif = jpegMetadata.getExif();
\r
571 if (exif != null) {
\r
572 outputSet = exif.getOutputSet();
\r
576 if (outputSet == null) {
\r
577 System.out.println("added : new tiff output set");
\r
578 outputSet = new TiffOutputSet();
\r
581 //---- EXIF_TAG_DATE_TIME_ORIGINAL / 「撮影日時/オリジナル画像の生成日時」----
\r
582 TiffOutputDirectory exifDir = outputSet.getOrCreateExifDirectory();
\r
584 Calendar cal = Calendar.getInstance();
\r
585 cal.setTimeZone(TimeZone.getTimeZone("UTC"));
\r
586 cal.setTime(correctedtime);
\r
587 exifDir.removeField(ExifTagConstants.EXIF_TAG_DATE_TIME_ORIGINAL);
\r
588 exifDir.add(ExifTagConstants.EXIF_TAG_DATE_TIME_ORIGINAL, ImportPicture.toEXIFString(cal.getTime()));
\r
591 //---- EXIF GPS_TIME_STAMP ----
\r
592 TiffOutputDirectory gpsDir = outputSet.getOrCreateGPSDirectory();
\r
594 Calendar cal = Calendar.getInstance(TimeZone.getTimeZone("UTC"));
\r
595 cal.setTimeZone(TimeZone.getTimeZone("GMT+00"));
\r
596 cal.setTime(correctedtime);
\r
597 final String yearStr = yearFormatter.format(cal.get(Calendar.YEAR));
\r
598 final String monthStr = monthFormatter.format(cal.get(Calendar.MONTH) + 1);
\r
599 final String dayStr = dayFormatter.format(cal.get(Calendar.DAY_OF_MONTH));
\r
600 final String dateStamp = yearStr +":"+ monthStr +":"+ dayStr;
\r
602 gpsDir.removeField(GpsTagConstants.GPS_TAG_GPS_TIME_STAMP);
\r
603 gpsDir.add(GpsTagConstants.GPS_TAG_GPS_TIME_STAMP,
\r
604 RationalNumber.valueOf(cal.get(Calendar.HOUR_OF_DAY)),
\r
605 RationalNumber.valueOf(cal.get(Calendar.MINUTE)),
\r
606 RationalNumber.valueOf(cal.get(Calendar.SECOND)));
\r
607 gpsDir.removeField(GpsTagConstants.GPS_TAG_GPS_DATE_STAMP);
\r
608 gpsDir.add(GpsTagConstants.GPS_TAG_GPS_DATE_STAMP, dateStamp);
\r
611 if (trkptT != null) {
\r
612 //---- EXIF GPS elevation/ALTITUDE ----
\r
613 if (trkptT.eleStr != null) {
\r
614 final double altitude = Double.parseDouble(trkptT.eleStr);
\r
615 gpsDir.removeField(GpsTagConstants.GPS_TAG_GPS_ALTITUDE);
\r
616 gpsDir.add(GpsTagConstants.GPS_TAG_GPS_ALTITUDE, RationalNumber.valueOf(altitude));
\r
619 //---- EXIF GPS magvar/IMG_DIRECTION ----
\r
620 if (trkptT.magvarStr != null) {
\r
621 final double magvar = Double.parseDouble(trkptT.magvarStr);
\r
622 gpsDir.removeField(GpsTagConstants.GPS_TAG_GPS_IMG_DIRECTION);
\r
623 gpsDir.add(GpsTagConstants.GPS_TAG_GPS_IMG_DIRECTION, RationalNumber.valueOf(magvar));
\r
626 //---- EXIF GPS_ ----
\r
627 outputSet.setGPSInDegrees(trkptT.lon.doubleValue(), trkptT.lat.doubleValue());
\r
630 ExifRewriter rewriter = new ExifRewriter();
\r
631 try (FileOutputStream fos = new FileOutputStream(new File(outDir, imageFile.getName()))) {
\r
632 rewriter.updateExifMetadataLossy(imageFile, fos, outputSet);
\r
638 * 対象は '*.JPG' のみ対象とする
\r
642 public static boolean checkFile(String name) {
\r
643 return ((name != null) && name.toUpperCase().endsWith(".JPG"));
\r
647 * <wpt lat="35.25714922" lon="139.15490497">
\r
648 * <ele>62.099998474121094</ele>
\r
649 * <time>2012-06-11T00:44:38Z</time>
\r
650 * <name><![CDATA[写真]]></name>
\r
651 * <link href="2012-06-11_09-44-38.jpg">
\r
652 * <text>2012-06-11_09-44-38.jpg</text>
\r
656 * <trkpt lat="35.32123832" lon="139.56965631">
\r
657 * <ele>47.20000076293945</ele>
\r
658 * <time>2012-06-15T03:00:29Z</time>
\r
666 public Element createWptTag(File iFile, long timestamp, Element trkpt) {
\r
667 Element wpt = document.createElement("wpt");
\r
669 NamedNodeMap nodeMap = trkpt.getAttributes();
\r
670 if (null != nodeMap) {
\r
671 for (int j=0; j < nodeMap.getLength(); j++ ) {
\r
672 switch (nodeMap.item(j).getNodeName()) {
\r
674 String lat = nodeMap.item(j).getNodeValue();
\r
675 wpt.setAttribute("lat", lat);
\r
678 String lon = nodeMap.item(j).getNodeValue();
\r
679 wpt.setAttribute("lon", lon);
\r
685 NodeList nodes1 = trkpt.getChildNodes();
\r
686 for (int i1=0; i1 < nodes1.getLength(); i1++) {
\r
687 Node node1 = nodes1.item(i1);
\r
688 NodeList nodes2 = node1.getChildNodes();
\r
689 switch (node1.getNodeName()) {
\r
691 for (int i2=0; i2 < nodes2.getLength(); i2++) {
\r
692 Node node2 = nodes2.item(i2);
\r
693 if (node2 != null) {
\r
694 if (node2.getNodeType() == Node.TEXT_NODE) {
\r
695 String eleStr = node2.getNodeValue();
\r
696 Element eleE = document.createElement("ele");
\r
697 eleE.setTextContent(eleStr);
\r
698 wpt.appendChild(eleE);
\r
704 for (int i2=0; i2 < nodes2.getLength(); i2++) {
\r
705 Node node2 = nodes2.item(i2);
\r
706 if (node2 != null) {
\r
707 if (node2.getNodeType() == Node.TEXT_NODE) {
\r
708 String timeStr = node2.getNodeValue();
\r
709 Element timeE = document.createElement("time");
\r
710 timeE.setTextContent(timeStr);
\r
711 wpt.appendChild(timeE);
\r
717 for (int i2=0; i2 < nodes2.getLength(); i2++) {
\r
718 Node node2 = nodes2.item(i2);
\r
719 if (node2 != null) {
\r
720 if (node2.getNodeType() == Node.TEXT_NODE) {
\r
721 String magvarStr = node2.getNodeValue();
\r
722 Element magvarE = document.createElement("magvar");
\r
723 magvarE.setTextContent(magvarStr);
\r
724 wpt.appendChild(magvarE);
\r
730 for (int i2=0; i2 < nodes2.getLength(); i2++) {
\r
731 Node node2 = nodes2.item(i2);
\r
732 if (node2 != null) {
\r
733 if (node2.getNodeType() == Node.TEXT_NODE) {
\r
734 String speedStr = node2.getNodeValue();
\r
735 Element speedE = document.createElement("speed");
\r
736 speedE.setTextContent(speedStr);
\r
737 wpt.appendChild(speedE);
\r
745 Element name = document.createElement("name");
\r
746 name.appendChild(document.createCDATASection("写真"));
\r
747 wpt.appendChild(name);
\r
749 Element link = document.createElement("link");
\r
750 link.setAttribute("href", ImportPicture.getShortPathName(imgDir, iFile));
\r
751 Element text = document.createElement("text");
\r
752 text.setTextContent(iFile.getName());
\r
753 link.appendChild(text);
\r
754 wpt.appendChild(link);
\r
760 * DateをEXIFの文字列に変換する。
\r
761 * 注意:EXiFの撮影時刻はUTC時間ではない
\r
765 public static String toEXIFString(Date localdate) {
\r
766 DateFormat dfUTC = new SimpleDateFormat(EXIF_DATE_TIME_FORMAT_STRING);
\r
767 return dfUTC.format(localdate);
\r
771 * EXIFの文字列をDateに変換する。
\r
772 * 注意:EXiFの撮影時刻はUTC時間ではない
\r
775 * @throws ParseException
\r
777 public static Date toEXIFDate(String timeStr) throws ParseException {
\r
778 DateFormat dfUTC = new SimpleDateFormat(EXIF_DATE_TIME_FORMAT_STRING);
\r
779 //dfUTC.setTimeZone(TimeZone.getTimeZone("UTC"));
\r
780 return dfUTC.parse(timeStr);
\r
783 public static String toUTCString(Date localdate) {
\r
784 DateFormat dfUTC = new SimpleDateFormat(TIME_FORMAT_STRING);
\r
785 dfUTC.setTimeZone(TimeZone.getTimeZone("UTC"));
\r
786 return dfUTC.format(localdate);
\r
789 public static Date toUTCDate(String timeStr) throws ParseException {
\r
790 DateFormat dfUTC = new SimpleDateFormat(TIME_FORMAT_STRING);
\r
791 dfUTC.setTimeZone(TimeZone.getTimeZone("UTC"));
\r
792 return dfUTC.parse(timeStr);
\r
795 static String getShortPathName(File dir, File iFile) {
\r
796 String dirPath = dir.getAbsolutePath();
\r
797 String filePath = iFile.getAbsolutePath();
\r
798 if (filePath.startsWith(dirPath)) {
\r
799 return filePath.substring(dirPath.length()+1);
\r
807 * ファイル名の順序に並び替えるためのソートクラス
\r
811 static class FileSort implements Comparator<File> {
\r
813 public int compare(File src, File target){
\r
814 int diff = src.getName().compareTo(target.getName());
\r
823 class JpegFileFilter implements FilenameFilter {
\r
825 public boolean accept(File dir, String name) {
\r
826 if (name.toUpperCase().matches(".*\\.JPG$")) {
\r