| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
2
|
|
|
|
|
|
|
# File: Photoshop.pm |
|
3
|
|
|
|
|
|
|
# |
|
4
|
|
|
|
|
|
|
# Description: Read/write Photoshop IRB meta information |
|
5
|
|
|
|
|
|
|
# |
|
6
|
|
|
|
|
|
|
# Revisions: 02/06/2004 - P. Harvey Created |
|
7
|
|
|
|
|
|
|
# 02/25/2004 - P. Harvey Added hack for problem with old photoshops |
|
8
|
|
|
|
|
|
|
# 10/04/2004 - P. Harvey Added a bunch of tags (ref Image::MetaData::JPEG) |
|
9
|
|
|
|
|
|
|
# but left most of them commented out until I have enough |
|
10
|
|
|
|
|
|
|
# information to write PrintConv routines for them to |
|
11
|
|
|
|
|
|
|
# display something useful |
|
12
|
|
|
|
|
|
|
# 07/08/2005 - P. Harvey Added support for reading PSD files |
|
13
|
|
|
|
|
|
|
# 01/07/2006 - P. Harvey Added PSD write support |
|
14
|
|
|
|
|
|
|
# 11/04/2006 - P. Harvey Added handling of resource name |
|
15
|
|
|
|
|
|
|
# |
|
16
|
|
|
|
|
|
|
# References: 1) http://www.fine-view.com/jp/lab/doc/ps6ffspecsv2.pdf |
|
17
|
|
|
|
|
|
|
# 2) http://www.ozhiker.com/electronics/pjmt/jpeg_info/irb_jpeg_qual.html |
|
18
|
|
|
|
|
|
|
# 3) Matt Mueller private communication (tests with PS CS2) |
|
19
|
|
|
|
|
|
|
# 4) http://www.fileformat.info/format/psd/egff.htm |
|
20
|
|
|
|
|
|
|
# 5) http://www.telegraphics.com.au/svn/psdparse/trunk/resources.c |
|
21
|
|
|
|
|
|
|
# 6) http://libpsd.graphest.com/files/Photoshop%20File%20Formats.pdf |
|
22
|
|
|
|
|
|
|
# 7) http://www.adobe.com/devnet-apps/photoshop/fileformatashtml/ |
|
23
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
24
|
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
package Image::ExifTool::Photoshop; |
|
26
|
|
|
|
|
|
|
|
|
27
|
23
|
|
|
23
|
|
4605
|
use strict; |
|
|
23
|
|
|
|
|
46
|
|
|
|
23
|
|
|
|
|
1281
|
|
|
28
|
23
|
|
|
23
|
|
121
|
use vars qw($VERSION $AUTOLOAD $iptcDigestInfo); |
|
|
23
|
|
|
|
|
50
|
|
|
|
23
|
|
|
|
|
1264
|
|
|
29
|
23
|
|
|
23
|
|
128
|
use Image::ExifTool qw(:DataAccess :Utils); |
|
|
23
|
|
|
|
|
41
|
|
|
|
23
|
|
|
|
|
105872
|
|
|
30
|
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
$VERSION = '1.65'; |
|
32
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
sub ProcessPhotoshop($$$); |
|
34
|
|
|
|
|
|
|
sub WritePhotoshop($$$); |
|
35
|
|
|
|
|
|
|
sub ProcessLayers($$$); |
|
36
|
|
|
|
|
|
|
|
|
37
|
|
|
|
|
|
|
# map of where information is stored in PSD image |
|
38
|
|
|
|
|
|
|
my %psdMap = ( |
|
39
|
|
|
|
|
|
|
IPTC => 'Photoshop', |
|
40
|
|
|
|
|
|
|
XMP => 'Photoshop', |
|
41
|
|
|
|
|
|
|
EXIFInfo => 'Photoshop', |
|
42
|
|
|
|
|
|
|
IFD0 => 'EXIFInfo', |
|
43
|
|
|
|
|
|
|
IFD1 => 'IFD0', |
|
44
|
|
|
|
|
|
|
ICC_Profile => 'Photoshop', |
|
45
|
|
|
|
|
|
|
ExifIFD => 'IFD0', |
|
46
|
|
|
|
|
|
|
GPS => 'IFD0', |
|
47
|
|
|
|
|
|
|
SubIFD => 'IFD0', |
|
48
|
|
|
|
|
|
|
GlobParamIFD => 'IFD0', |
|
49
|
|
|
|
|
|
|
PrintIM => 'IFD0', |
|
50
|
|
|
|
|
|
|
InteropIFD => 'ExifIFD', |
|
51
|
|
|
|
|
|
|
MakerNotes => 'ExifIFD', |
|
52
|
|
|
|
|
|
|
); |
|
53
|
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
# tag information for PhotoshopThumbnail and PhotoshopBGRThumbnail |
|
55
|
|
|
|
|
|
|
my %thumbnailInfo = ( |
|
56
|
|
|
|
|
|
|
Writable => 'undef', |
|
57
|
|
|
|
|
|
|
Protected => 1, |
|
58
|
|
|
|
|
|
|
RawConv => 'my $img=substr($val,0x1c); $self->ValidateImage(\$img,$tag)', |
|
59
|
|
|
|
|
|
|
ValueConvInv => q{ |
|
60
|
|
|
|
|
|
|
my $et = new Image::ExifTool; |
|
61
|
|
|
|
|
|
|
my @tags = qw{ImageWidth ImageHeight FileType}; |
|
62
|
|
|
|
|
|
|
my $info = $et->ImageInfo(\$val, @tags); |
|
63
|
|
|
|
|
|
|
my ($w, $h, $type) = @$info{@tags}; |
|
64
|
|
|
|
|
|
|
$w and $h and $type eq 'JPEG' or warn("Not a valid JPEG image\n"), return undef; |
|
65
|
|
|
|
|
|
|
my $wbytes = int(($w * 24 + 31) / 32) * 4; |
|
66
|
|
|
|
|
|
|
return pack('N6n2', 1, $w, $h, $wbytes, $wbytes * $h, length($val), 24, 1) . $val; |
|
67
|
|
|
|
|
|
|
}, |
|
68
|
|
|
|
|
|
|
); |
|
69
|
|
|
|
|
|
|
|
|
70
|
|
|
|
|
|
|
# tag info to decode Photoshop Unicode string |
|
71
|
|
|
|
|
|
|
my %unicodeString = ( |
|
72
|
|
|
|
|
|
|
ValueConv => sub { |
|
73
|
|
|
|
|
|
|
my ($val, $et) = @_; |
|
74
|
|
|
|
|
|
|
return '' if length($val) < 4; |
|
75
|
|
|
|
|
|
|
my $len = unpack('N', $val) * 2; |
|
76
|
|
|
|
|
|
|
return '' if length($val) < 4 + $len; |
|
77
|
|
|
|
|
|
|
return $et->Decode(substr($val, 4, $len), 'UCS2', 'MM'); |
|
78
|
|
|
|
|
|
|
}, |
|
79
|
|
|
|
|
|
|
ValueConvInv => sub { |
|
80
|
|
|
|
|
|
|
my ($val, $et) = @_; |
|
81
|
|
|
|
|
|
|
return pack('N', length $val) . $et->Encode($val, 'UCS2', 'MM'); |
|
82
|
|
|
|
|
|
|
}, |
|
83
|
|
|
|
|
|
|
); |
|
84
|
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
# Photoshop APP13 tag table |
|
86
|
|
|
|
|
|
|
# (set Unknown flag for information we don't want to display normally) |
|
87
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::Main = ( |
|
88
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
89
|
|
|
|
|
|
|
PROCESS_PROC => \&ProcessPhotoshop, |
|
90
|
|
|
|
|
|
|
WRITE_PROC => \&WritePhotoshop, |
|
91
|
|
|
|
|
|
|
0x03e8 => { Unknown => 1, Name => 'Photoshop2Info' }, |
|
92
|
|
|
|
|
|
|
0x03e9 => { Unknown => 1, Name => 'MacintoshPrintInfo' }, |
|
93
|
|
|
|
|
|
|
0x03ea => { Unknown => 1, Name => 'XMLData', Binary => 1 }, #PH |
|
94
|
|
|
|
|
|
|
0x03eb => { Unknown => 1, Name => 'Photoshop2ColorTable' }, |
|
95
|
|
|
|
|
|
|
0x03ed => { |
|
96
|
|
|
|
|
|
|
Name => 'ResolutionInfo', |
|
97
|
|
|
|
|
|
|
SubDirectory => { |
|
98
|
|
|
|
|
|
|
TagTable => 'Image::ExifTool::Photoshop::Resolution', |
|
99
|
|
|
|
|
|
|
}, |
|
100
|
|
|
|
|
|
|
}, |
|
101
|
|
|
|
|
|
|
0x03ee => { |
|
102
|
|
|
|
|
|
|
Name => 'AlphaChannelsNames', |
|
103
|
|
|
|
|
|
|
ValueConv => 'Image::ExifTool::Photoshop::ConvertPascalString($self,$val)', |
|
104
|
|
|
|
|
|
|
}, |
|
105
|
|
|
|
|
|
|
0x03ef => { Unknown => 1, Name => 'DisplayInfo' }, |
|
106
|
|
|
|
|
|
|
0x03f0 => { Unknown => 1, Name => 'PStringCaption' }, |
|
107
|
|
|
|
|
|
|
0x03f1 => { Unknown => 1, Name => 'BorderInformation' }, |
|
108
|
|
|
|
|
|
|
0x03f2 => { Unknown => 1, Name => 'BackgroundColor' }, |
|
109
|
|
|
|
|
|
|
0x03f3 => { Unknown => 1, Name => 'PrintFlags', Format => 'int8u' }, |
|
110
|
|
|
|
|
|
|
0x03f4 => { Unknown => 1, Name => 'BW_HalftoningInfo' }, |
|
111
|
|
|
|
|
|
|
0x03f5 => { Unknown => 1, Name => 'ColorHalftoningInfo' }, |
|
112
|
|
|
|
|
|
|
0x03f6 => { Unknown => 1, Name => 'DuotoneHalftoningInfo' }, |
|
113
|
|
|
|
|
|
|
0x03f7 => { Unknown => 1, Name => 'BW_TransferFunc' }, |
|
114
|
|
|
|
|
|
|
0x03f8 => { Unknown => 1, Name => 'ColorTransferFuncs' }, |
|
115
|
|
|
|
|
|
|
0x03f9 => { Unknown => 1, Name => 'DuotoneTransferFuncs' }, |
|
116
|
|
|
|
|
|
|
0x03fa => { Unknown => 1, Name => 'DuotoneImageInfo' }, |
|
117
|
|
|
|
|
|
|
0x03fb => { Unknown => 1, Name => 'EffectiveBW', Format => 'int8u' }, |
|
118
|
|
|
|
|
|
|
0x03fc => { Unknown => 1, Name => 'ObsoletePhotoshopTag1' }, |
|
119
|
|
|
|
|
|
|
0x03fd => { Unknown => 1, Name => 'EPSOptions' }, |
|
120
|
|
|
|
|
|
|
0x03fe => { Unknown => 1, Name => 'QuickMaskInfo' }, |
|
121
|
|
|
|
|
|
|
0x03ff => { Unknown => 1, Name => 'ObsoletePhotoshopTag2' }, |
|
122
|
|
|
|
|
|
|
0x0400 => { Unknown => 1, Name => 'TargetLayerID', Format => 'int16u' }, # (LayerStateInfo) |
|
123
|
|
|
|
|
|
|
0x0401 => { Unknown => 1, Name => 'WorkingPath' }, |
|
124
|
|
|
|
|
|
|
0x0402 => { Unknown => 1, Name => 'LayersGroupInfo', Format => 'int16u' }, |
|
125
|
|
|
|
|
|
|
0x0403 => { Unknown => 1, Name => 'ObsoletePhotoshopTag3' }, |
|
126
|
|
|
|
|
|
|
0x0404 => { |
|
127
|
|
|
|
|
|
|
Name => 'IPTCData', |
|
128
|
|
|
|
|
|
|
SubDirectory => { |
|
129
|
|
|
|
|
|
|
DirName => 'IPTC', |
|
130
|
|
|
|
|
|
|
TagTable => 'Image::ExifTool::IPTC::Main', |
|
131
|
|
|
|
|
|
|
}, |
|
132
|
|
|
|
|
|
|
}, |
|
133
|
|
|
|
|
|
|
0x0405 => { Unknown => 1, Name => 'RawImageMode' }, |
|
134
|
|
|
|
|
|
|
0x0406 => { #2 |
|
135
|
|
|
|
|
|
|
Name => 'JPEG_Quality', |
|
136
|
|
|
|
|
|
|
SubDirectory => { |
|
137
|
|
|
|
|
|
|
TagTable => 'Image::ExifTool::Photoshop::JPEG_Quality', |
|
138
|
|
|
|
|
|
|
}, |
|
139
|
|
|
|
|
|
|
}, |
|
140
|
|
|
|
|
|
|
0x0408 => { Unknown => 1, Name => 'GridGuidesInfo' }, |
|
141
|
|
|
|
|
|
|
0x0409 => { |
|
142
|
|
|
|
|
|
|
Name => 'PhotoshopBGRThumbnail', |
|
143
|
|
|
|
|
|
|
Notes => 'this is a JPEG image, but in BGR format instead of RGB', |
|
144
|
|
|
|
|
|
|
%thumbnailInfo, |
|
145
|
|
|
|
|
|
|
Groups => { 2 => 'Preview' }, |
|
146
|
|
|
|
|
|
|
}, |
|
147
|
|
|
|
|
|
|
0x040a => { |
|
148
|
|
|
|
|
|
|
Name => 'CopyrightFlag', |
|
149
|
|
|
|
|
|
|
Writable => 'int8u', |
|
150
|
|
|
|
|
|
|
Groups => { 2 => 'Author' }, |
|
151
|
|
|
|
|
|
|
ValueConv => 'join(" ",unpack("C*", $val))', |
|
152
|
|
|
|
|
|
|
ValueConvInv => 'pack("C*",split(" ",$val))', |
|
153
|
|
|
|
|
|
|
PrintConv => { #3 |
|
154
|
|
|
|
|
|
|
0 => 'False', |
|
155
|
|
|
|
|
|
|
1 => 'True', |
|
156
|
|
|
|
|
|
|
}, |
|
157
|
|
|
|
|
|
|
}, |
|
158
|
|
|
|
|
|
|
0x040b => { |
|
159
|
|
|
|
|
|
|
Name => 'URL', |
|
160
|
|
|
|
|
|
|
Writable => 'string', |
|
161
|
|
|
|
|
|
|
Groups => { 2 => 'Author' }, |
|
162
|
|
|
|
|
|
|
}, |
|
163
|
|
|
|
|
|
|
0x040c => { |
|
164
|
|
|
|
|
|
|
Name => 'PhotoshopThumbnail', |
|
165
|
|
|
|
|
|
|
%thumbnailInfo, |
|
166
|
|
|
|
|
|
|
Groups => { 2 => 'Preview' }, |
|
167
|
|
|
|
|
|
|
}, |
|
168
|
|
|
|
|
|
|
0x040d => { |
|
169
|
|
|
|
|
|
|
Name => 'GlobalAngle', |
|
170
|
|
|
|
|
|
|
Writable => 'int32u', |
|
171
|
|
|
|
|
|
|
ValueConv => 'unpack("N",$val)', |
|
172
|
|
|
|
|
|
|
ValueConvInv => 'pack("N",$val)', |
|
173
|
|
|
|
|
|
|
}, |
|
174
|
|
|
|
|
|
|
0x040e => { Unknown => 1, Name => 'ColorSamplersResource' }, |
|
175
|
|
|
|
|
|
|
0x040f => { |
|
176
|
|
|
|
|
|
|
Name => 'ICC_Profile', |
|
177
|
|
|
|
|
|
|
SubDirectory => { |
|
178
|
|
|
|
|
|
|
TagTable => 'Image::ExifTool::ICC_Profile::Main', |
|
179
|
|
|
|
|
|
|
}, |
|
180
|
|
|
|
|
|
|
}, |
|
181
|
|
|
|
|
|
|
0x0410 => { Unknown => 1, Name => 'Watermark', Format => 'int8u' }, |
|
182
|
|
|
|
|
|
|
0x0411 => { Unknown => 1, Name => 'ICC_Untagged', Format => 'int8u' }, |
|
183
|
|
|
|
|
|
|
0x0412 => { Unknown => 1, Name => 'EffectsVisible', Format => 'int8u' }, |
|
184
|
|
|
|
|
|
|
0x0413 => { Unknown => 1, Name => 'SpotHalftone' }, |
|
185
|
|
|
|
|
|
|
0x0414 => { Unknown => 1, Name => 'IDsBaseValue', Description => 'IDs Base Value', Format => 'int32u' }, |
|
186
|
|
|
|
|
|
|
0x0415 => { Unknown => 1, Name => 'UnicodeAlphaNames' }, |
|
187
|
|
|
|
|
|
|
0x0416 => { Unknown => 1, Name => 'IndexedColorTableCount', Format => 'int16u' }, |
|
188
|
|
|
|
|
|
|
0x0417 => { Unknown => 1, Name => 'TransparentIndex', Format => 'int16u' }, |
|
189
|
|
|
|
|
|
|
0x0419 => { |
|
190
|
|
|
|
|
|
|
Name => 'GlobalAltitude', |
|
191
|
|
|
|
|
|
|
Writable => 'int32u', |
|
192
|
|
|
|
|
|
|
ValueConv => 'unpack("N",$val)', |
|
193
|
|
|
|
|
|
|
ValueConvInv => 'pack("N",$val)', |
|
194
|
|
|
|
|
|
|
}, |
|
195
|
|
|
|
|
|
|
0x041a => { |
|
196
|
|
|
|
|
|
|
Name => 'SliceInfo', |
|
197
|
|
|
|
|
|
|
SubDirectory => { TagTable => 'Image::ExifTool::Photoshop::SliceInfo' }, |
|
198
|
|
|
|
|
|
|
}, |
|
199
|
|
|
|
|
|
|
0x041b => { Name => 'WorkflowURL', %unicodeString }, |
|
200
|
|
|
|
|
|
|
0x041c => { Unknown => 1, Name => 'JumpToXPEP' }, |
|
201
|
|
|
|
|
|
|
0x041d => { Unknown => 1, Name => 'AlphaIdentifiers' }, |
|
202
|
|
|
|
|
|
|
0x041e => { |
|
203
|
|
|
|
|
|
|
Name => 'URL_List', |
|
204
|
|
|
|
|
|
|
List => 1, |
|
205
|
|
|
|
|
|
|
Writable => 1, |
|
206
|
|
|
|
|
|
|
ValueConv => sub { |
|
207
|
|
|
|
|
|
|
my ($val, $et) = @_; |
|
208
|
|
|
|
|
|
|
return '' if length($val) < 4; |
|
209
|
|
|
|
|
|
|
my $num = unpack('N', $val); |
|
210
|
|
|
|
|
|
|
my ($i, @vals); |
|
211
|
|
|
|
|
|
|
my $pos = 4; |
|
212
|
|
|
|
|
|
|
for ($i=0; $i<$num; ++$i) { |
|
213
|
|
|
|
|
|
|
$pos += 8; # (skip word and ID) |
|
214
|
|
|
|
|
|
|
last if length($val) < $pos + 4; |
|
215
|
|
|
|
|
|
|
my $len = unpack("x${pos}N", $val) * 2; |
|
216
|
|
|
|
|
|
|
last if length($val) < $pos + 4 + $len; |
|
217
|
|
|
|
|
|
|
push @vals, $et->Decode(substr($val,$pos+4,$len), 'UCS2', 'MM'); |
|
218
|
|
|
|
|
|
|
$pos += 4 + $len; |
|
219
|
|
|
|
|
|
|
} |
|
220
|
|
|
|
|
|
|
return \@vals; |
|
221
|
|
|
|
|
|
|
}, |
|
222
|
|
|
|
|
|
|
# (this is tricky to make writable) |
|
223
|
|
|
|
|
|
|
}, |
|
224
|
|
|
|
|
|
|
0x0421 => { |
|
225
|
|
|
|
|
|
|
Name => 'VersionInfo', |
|
226
|
|
|
|
|
|
|
SubDirectory => { |
|
227
|
|
|
|
|
|
|
TagTable => 'Image::ExifTool::Photoshop::VersionInfo', |
|
228
|
|
|
|
|
|
|
}, |
|
229
|
|
|
|
|
|
|
}, |
|
230
|
|
|
|
|
|
|
0x0422 => { |
|
231
|
|
|
|
|
|
|
Name => 'EXIFInfo', #PH (Found in EPS and PSD files) |
|
232
|
|
|
|
|
|
|
SubDirectory => { |
|
233
|
|
|
|
|
|
|
TagTable=> 'Image::ExifTool::Exif::Main', |
|
234
|
|
|
|
|
|
|
ProcessProc => \&Image::ExifTool::ProcessTIFF, |
|
235
|
|
|
|
|
|
|
WriteProc => \&Image::ExifTool::WriteTIFF, |
|
236
|
|
|
|
|
|
|
}, |
|
237
|
|
|
|
|
|
|
}, |
|
238
|
|
|
|
|
|
|
0x0423 => { Unknown => 1, Name => 'ExifInfo2', Binary => 1 }, #5 |
|
239
|
|
|
|
|
|
|
0x0424 => { |
|
240
|
|
|
|
|
|
|
Name => 'XMP', |
|
241
|
|
|
|
|
|
|
SubDirectory => { |
|
242
|
|
|
|
|
|
|
TagTable => 'Image::ExifTool::XMP::Main', |
|
243
|
|
|
|
|
|
|
}, |
|
244
|
|
|
|
|
|
|
}, |
|
245
|
|
|
|
|
|
|
0x0425 => { |
|
246
|
|
|
|
|
|
|
Name => 'IPTCDigest', |
|
247
|
|
|
|
|
|
|
Writable => 'string', |
|
248
|
|
|
|
|
|
|
Protected => 1, |
|
249
|
|
|
|
|
|
|
Notes => q{ |
|
250
|
|
|
|
|
|
|
this tag indicates provides a way for XMP-aware applications to indicate |
|
251
|
|
|
|
|
|
|
that the XMP is synchronized with the IPTC. The MWG recommendation is to |
|
252
|
|
|
|
|
|
|
ignore the XMP if IPTCDigest exists and doesn't match the CurrentIPTCDigest. |
|
253
|
|
|
|
|
|
|
When writing, special values of "new" and "old" represent the digests of the |
|
254
|
|
|
|
|
|
|
IPTC from the edited and original files respectively, and are undefined if |
|
255
|
|
|
|
|
|
|
the IPTC does not exist in the respective file. Set this to "new" as an |
|
256
|
|
|
|
|
|
|
indication that the XMP is synchronized with the IPTC |
|
257
|
|
|
|
|
|
|
}, |
|
258
|
|
|
|
|
|
|
# also note the 'new' feature requires that the IPTC comes before this tag is written |
|
259
|
|
|
|
|
|
|
ValueConv => 'unpack("H*", $val)', |
|
260
|
|
|
|
|
|
|
ValueConvInv => q{ |
|
261
|
|
|
|
|
|
|
if (lc($val) eq 'new' or lc($val) eq 'old') { |
|
262
|
|
|
|
|
|
|
{ |
|
263
|
|
|
|
|
|
|
local $SIG{'__WARN__'} = sub { }; |
|
264
|
|
|
|
|
|
|
return lc($val) if eval { require Digest::MD5 }; |
|
265
|
|
|
|
|
|
|
} |
|
266
|
|
|
|
|
|
|
warn "Digest::MD5 must be installed\n"; |
|
267
|
|
|
|
|
|
|
return undef; |
|
268
|
|
|
|
|
|
|
} |
|
269
|
|
|
|
|
|
|
return pack('H*', $val) if $val =~ /^[0-9a-f]{32}$/i; |
|
270
|
|
|
|
|
|
|
warn "Value must be 'new', 'old' or 32 hexadecimal digits\n"; |
|
271
|
|
|
|
|
|
|
return undef; |
|
272
|
|
|
|
|
|
|
} |
|
273
|
|
|
|
|
|
|
}, |
|
274
|
|
|
|
|
|
|
0x0426 => { |
|
275
|
|
|
|
|
|
|
Name => 'PrintScaleInfo', |
|
276
|
|
|
|
|
|
|
SubDirectory => { TagTable => 'Image::ExifTool::Photoshop::PrintScaleInfo' }, |
|
277
|
|
|
|
|
|
|
}, |
|
278
|
|
|
|
|
|
|
0x0428 => { |
|
279
|
|
|
|
|
|
|
Name => 'PixelInfo', |
|
280
|
|
|
|
|
|
|
SubDirectory => { TagTable => 'Image::ExifTool::Photoshop::PixelInfo' }, |
|
281
|
|
|
|
|
|
|
}, |
|
282
|
|
|
|
|
|
|
0x0429 => { Unknown => 1, Name => 'LayerComps' }, #5 |
|
283
|
|
|
|
|
|
|
0x042a => { Unknown => 1, Name => 'AlternateDuotoneColors' }, #5 |
|
284
|
|
|
|
|
|
|
0x042b => { Unknown => 1, Name => 'AlternateSpotColors' }, #5 |
|
285
|
|
|
|
|
|
|
0x042d => { #7 |
|
286
|
|
|
|
|
|
|
Name => 'LayerSelectionIDs', |
|
287
|
|
|
|
|
|
|
Description => 'Layer Selection IDs', |
|
288
|
|
|
|
|
|
|
Unknown => 1, |
|
289
|
|
|
|
|
|
|
ValueConv => q{ |
|
290
|
|
|
|
|
|
|
my ($n, @a) = unpack("nN*",$val); |
|
291
|
|
|
|
|
|
|
$#a = $n - 1 if $n > @a; |
|
292
|
|
|
|
|
|
|
return join(' ', @a); |
|
293
|
|
|
|
|
|
|
}, |
|
294
|
|
|
|
|
|
|
}, |
|
295
|
|
|
|
|
|
|
0x042e => { Unknown => 1, Name => 'HDRToningInfo' }, #7 |
|
296
|
|
|
|
|
|
|
0x042f => { Unknown => 1, Name => 'PrintInfo' }, #7 |
|
297
|
|
|
|
|
|
|
0x0430 => { Unknown => 1, Name => 'LayerGroupsEnabledID', Format => 'int8u' }, #7 |
|
298
|
|
|
|
|
|
|
0x0431 => { Unknown => 1, Name => 'ColorSamplersResource2' }, #7 |
|
299
|
|
|
|
|
|
|
0x0432 => { Unknown => 1, Name => 'MeasurementScale' }, #7 |
|
300
|
|
|
|
|
|
|
0x0433 => { Unknown => 1, Name => 'TimelineInfo' }, #7 |
|
301
|
|
|
|
|
|
|
0x0434 => { Unknown => 1, Name => 'SheetDisclosure' }, #7 |
|
302
|
|
|
|
|
|
|
0x0435 => { Unknown => 1, Name => 'DisplayInfo' }, #7 |
|
303
|
|
|
|
|
|
|
0x0436 => { Unknown => 1, Name => 'OnionSkins' }, #7 |
|
304
|
|
|
|
|
|
|
0x0438 => { Unknown => 1, Name => 'CountInfo' }, #7 |
|
305
|
|
|
|
|
|
|
0x043a => { Unknown => 1, Name => 'PrintInfo2' }, #7 |
|
306
|
|
|
|
|
|
|
0x043b => { Unknown => 1, Name => 'PrintStyle' }, #7 |
|
307
|
|
|
|
|
|
|
0x043c => { Unknown => 1, Name => 'MacintoshNSPrintInfo' }, #7 |
|
308
|
|
|
|
|
|
|
0x043d => { Unknown => 1, Name => 'WindowsDEVMODE' }, #7 |
|
309
|
|
|
|
|
|
|
0x043e => { Unknown => 1, Name => 'AutoSaveFilePath' }, #7 |
|
310
|
|
|
|
|
|
|
0x043f => { Unknown => 1, Name => 'AutoSaveFormat' }, #7 |
|
311
|
|
|
|
|
|
|
0x0440 => { Unknown => 1, Name => 'PathSelectionState' }, #7 |
|
312
|
|
|
|
|
|
|
# 0x07d0-0x0bb6 Path information |
|
313
|
|
|
|
|
|
|
0x0bb7 => { |
|
314
|
|
|
|
|
|
|
Name => 'ClippingPathName', |
|
315
|
|
|
|
|
|
|
# convert from a Pascal string (ignoring 6 bytes of unknown data after string) |
|
316
|
|
|
|
|
|
|
ValueConv => q{ |
|
317
|
|
|
|
|
|
|
my $len = ord($val); |
|
318
|
|
|
|
|
|
|
$val = substr($val, 0, $len+1) if $len < length($val); |
|
319
|
|
|
|
|
|
|
return Image::ExifTool::Photoshop::ConvertPascalString($self,$val); |
|
320
|
|
|
|
|
|
|
}, |
|
321
|
|
|
|
|
|
|
}, |
|
322
|
|
|
|
|
|
|
0x0bb8 => { Unknown => 1, Name => 'OriginPathInfo' }, #7 |
|
323
|
|
|
|
|
|
|
# 0x0fa0-0x1387 - plug-in resources (ref 7) |
|
324
|
|
|
|
|
|
|
0x1b58 => { Unknown => 1, Name => 'ImageReadyVariables' }, #7 |
|
325
|
|
|
|
|
|
|
0x1b59 => { Unknown => 1, Name => 'ImageReadyDataSets' }, #7 |
|
326
|
|
|
|
|
|
|
0x1f40 => { Unknown => 1, Name => 'LightroomWorkflow' }, #7 |
|
327
|
|
|
|
|
|
|
0x2710 => { Unknown => 1, Name => 'PrintFlagsInfo' }, |
|
328
|
|
|
|
|
|
|
); |
|
329
|
|
|
|
|
|
|
|
|
330
|
|
|
|
|
|
|
# Photoshop JPEG quality record (ref 2) |
|
331
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::JPEG_Quality = ( |
|
332
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
333
|
|
|
|
|
|
|
WRITE_PROC => \&Image::ExifTool::WriteBinaryData, |
|
334
|
|
|
|
|
|
|
CHECK_PROC => \&Image::ExifTool::CheckBinaryData, |
|
335
|
|
|
|
|
|
|
FORMAT => 'int16s', |
|
336
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
337
|
|
|
|
|
|
|
0 => { |
|
338
|
|
|
|
|
|
|
Name => 'PhotoshopQuality', |
|
339
|
|
|
|
|
|
|
Writable => 1, |
|
340
|
|
|
|
|
|
|
PrintConv => '$val + 4', |
|
341
|
|
|
|
|
|
|
PrintConvInv => '$val - 4', |
|
342
|
|
|
|
|
|
|
}, |
|
343
|
|
|
|
|
|
|
1 => { |
|
344
|
|
|
|
|
|
|
Name => 'PhotoshopFormat', |
|
345
|
|
|
|
|
|
|
PrintConv => { |
|
346
|
|
|
|
|
|
|
0x0000 => 'Standard', |
|
347
|
|
|
|
|
|
|
0x0001 => 'Optimized', |
|
348
|
|
|
|
|
|
|
0x0101 => 'Progressive', |
|
349
|
|
|
|
|
|
|
}, |
|
350
|
|
|
|
|
|
|
}, |
|
351
|
|
|
|
|
|
|
2 => { |
|
352
|
|
|
|
|
|
|
Name => 'ProgressiveScans', |
|
353
|
|
|
|
|
|
|
PrintConv => { |
|
354
|
|
|
|
|
|
|
1 => '3 Scans', |
|
355
|
|
|
|
|
|
|
2 => '4 Scans', |
|
356
|
|
|
|
|
|
|
3 => '5 Scans', |
|
357
|
|
|
|
|
|
|
}, |
|
358
|
|
|
|
|
|
|
}, |
|
359
|
|
|
|
|
|
|
); |
|
360
|
|
|
|
|
|
|
|
|
361
|
|
|
|
|
|
|
# Photoshop Slices |
|
362
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::SliceInfo = ( |
|
363
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
364
|
|
|
|
|
|
|
20 => { Name => 'SlicesGroupName', Format => 'var_ustr32' }, |
|
365
|
|
|
|
|
|
|
24 => { Name => 'NumSlices', Format => 'int32u' }, |
|
366
|
|
|
|
|
|
|
); |
|
367
|
|
|
|
|
|
|
|
|
368
|
|
|
|
|
|
|
# Photoshop resolution information #PH |
|
369
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::Resolution = ( |
|
370
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
371
|
|
|
|
|
|
|
WRITE_PROC => \&Image::ExifTool::WriteBinaryData, |
|
372
|
|
|
|
|
|
|
CHECK_PROC => \&Image::ExifTool::CheckBinaryData, |
|
373
|
|
|
|
|
|
|
FORMAT => 'int16u', |
|
374
|
|
|
|
|
|
|
FIRST_ENTRY => 0, |
|
375
|
|
|
|
|
|
|
WRITABLE => 1, |
|
376
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
377
|
|
|
|
|
|
|
0 => { |
|
378
|
|
|
|
|
|
|
Name => 'XResolution', |
|
379
|
|
|
|
|
|
|
Format => 'int32u', |
|
380
|
|
|
|
|
|
|
Priority => 0, |
|
381
|
|
|
|
|
|
|
ValueConv => '$val / 0x10000', |
|
382
|
|
|
|
|
|
|
ValueConvInv => 'int($val * 0x10000 + 0.5)', |
|
383
|
|
|
|
|
|
|
PrintConv => 'int($val * 100 + 0.5) / 100', |
|
384
|
|
|
|
|
|
|
PrintConvInv => '$val', |
|
385
|
|
|
|
|
|
|
}, |
|
386
|
|
|
|
|
|
|
2 => { |
|
387
|
|
|
|
|
|
|
Name => 'DisplayedUnitsX', |
|
388
|
|
|
|
|
|
|
PrintConv => { |
|
389
|
|
|
|
|
|
|
1 => 'inches', |
|
390
|
|
|
|
|
|
|
2 => 'cm', |
|
391
|
|
|
|
|
|
|
}, |
|
392
|
|
|
|
|
|
|
}, |
|
393
|
|
|
|
|
|
|
4 => { |
|
394
|
|
|
|
|
|
|
Name => 'YResolution', |
|
395
|
|
|
|
|
|
|
Format => 'int32u', |
|
396
|
|
|
|
|
|
|
Priority => 0, |
|
397
|
|
|
|
|
|
|
ValueConv => '$val / 0x10000', |
|
398
|
|
|
|
|
|
|
ValueConvInv => 'int($val * 0x10000 + 0.5)', |
|
399
|
|
|
|
|
|
|
PrintConv => 'int($val * 100 + 0.5) / 100', |
|
400
|
|
|
|
|
|
|
PrintConvInv => '$val', |
|
401
|
|
|
|
|
|
|
}, |
|
402
|
|
|
|
|
|
|
6 => { |
|
403
|
|
|
|
|
|
|
Name => 'DisplayedUnitsY', |
|
404
|
|
|
|
|
|
|
PrintConv => { |
|
405
|
|
|
|
|
|
|
1 => 'inches', |
|
406
|
|
|
|
|
|
|
2 => 'cm', |
|
407
|
|
|
|
|
|
|
}, |
|
408
|
|
|
|
|
|
|
}, |
|
409
|
|
|
|
|
|
|
); |
|
410
|
|
|
|
|
|
|
|
|
411
|
|
|
|
|
|
|
# Photoshop version information |
|
412
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::VersionInfo = ( |
|
413
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
414
|
|
|
|
|
|
|
WRITE_PROC => \&Image::ExifTool::WriteBinaryData, |
|
415
|
|
|
|
|
|
|
CHECK_PROC => \&Image::ExifTool::CheckBinaryData, |
|
416
|
|
|
|
|
|
|
FIRST_ENTRY => 0, |
|
417
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
418
|
|
|
|
|
|
|
# (always 1) 0 => { Name => 'PhotoshopVersion', Format => 'int32u' }, |
|
419
|
|
|
|
|
|
|
4 => { Name => 'HasRealMergedData', Format => 'int8u', PrintConv => { 0 => 'No', 1 => 'Yes' } }, |
|
420
|
|
|
|
|
|
|
5 => { Name => 'WriterName', Format => 'var_ustr32' }, |
|
421
|
|
|
|
|
|
|
9 => { Name => 'ReaderName', Format => 'var_ustr32' }, |
|
422
|
|
|
|
|
|
|
# (always 1) 13 => { Name => 'FileVersion', Format => 'int32u' }, |
|
423
|
|
|
|
|
|
|
); |
|
424
|
|
|
|
|
|
|
|
|
425
|
|
|
|
|
|
|
# Print Scale |
|
426
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::PrintScaleInfo = ( |
|
427
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
428
|
|
|
|
|
|
|
WRITE_PROC => \&Image::ExifTool::WriteBinaryData, |
|
429
|
|
|
|
|
|
|
CHECK_PROC => \&Image::ExifTool::CheckBinaryData, |
|
430
|
|
|
|
|
|
|
FIRST_ENTRY => 0, |
|
431
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
432
|
|
|
|
|
|
|
0 => { |
|
433
|
|
|
|
|
|
|
Name => 'PrintStyle', |
|
434
|
|
|
|
|
|
|
Format => 'int16u', |
|
435
|
|
|
|
|
|
|
PrintConv => { |
|
436
|
|
|
|
|
|
|
0 => 'Centered', |
|
437
|
|
|
|
|
|
|
1 => 'Size to Fit', |
|
438
|
|
|
|
|
|
|
2 => 'User Defined', |
|
439
|
|
|
|
|
|
|
}, |
|
440
|
|
|
|
|
|
|
}, |
|
441
|
|
|
|
|
|
|
2 => { Name => 'PrintPosition', Format => 'float[2]' }, |
|
442
|
|
|
|
|
|
|
10 => { Name => 'PrintScale', Format => 'float' }, |
|
443
|
|
|
|
|
|
|
); |
|
444
|
|
|
|
|
|
|
|
|
445
|
|
|
|
|
|
|
# Pixel Aspect Ratio |
|
446
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::PixelInfo = ( |
|
447
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
448
|
|
|
|
|
|
|
WRITE_PROC => \&Image::ExifTool::WriteBinaryData, |
|
449
|
|
|
|
|
|
|
CHECK_PROC => \&Image::ExifTool::CheckBinaryData, |
|
450
|
|
|
|
|
|
|
FIRST_ENTRY => 0, |
|
451
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
452
|
|
|
|
|
|
|
# 0 - version |
|
453
|
|
|
|
|
|
|
4 => { Name => 'PixelAspectRatio', Format => 'double' }, |
|
454
|
|
|
|
|
|
|
); |
|
455
|
|
|
|
|
|
|
|
|
456
|
|
|
|
|
|
|
# Photoshop PSD file header |
|
457
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::Header = ( |
|
458
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
459
|
|
|
|
|
|
|
FORMAT => 'int16u', |
|
460
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
461
|
|
|
|
|
|
|
NOTES => 'This information is found in the PSD file header.', |
|
462
|
|
|
|
|
|
|
6 => 'NumChannels', |
|
463
|
|
|
|
|
|
|
7 => { Name => 'ImageHeight', Format => 'int32u' }, |
|
464
|
|
|
|
|
|
|
9 => { Name => 'ImageWidth', Format => 'int32u' }, |
|
465
|
|
|
|
|
|
|
11 => 'BitDepth', |
|
466
|
|
|
|
|
|
|
12 => { |
|
467
|
|
|
|
|
|
|
Name => 'ColorMode', |
|
468
|
|
|
|
|
|
|
PrintConvColumns => 2, |
|
469
|
|
|
|
|
|
|
PrintConv => { |
|
470
|
|
|
|
|
|
|
0 => 'Bitmap', |
|
471
|
|
|
|
|
|
|
1 => 'Grayscale', |
|
472
|
|
|
|
|
|
|
2 => 'Indexed', |
|
473
|
|
|
|
|
|
|
3 => 'RGB', |
|
474
|
|
|
|
|
|
|
4 => 'CMYK', |
|
475
|
|
|
|
|
|
|
7 => 'Multichannel', |
|
476
|
|
|
|
|
|
|
8 => 'Duotone', |
|
477
|
|
|
|
|
|
|
9 => 'Lab', |
|
478
|
|
|
|
|
|
|
}, |
|
479
|
|
|
|
|
|
|
}, |
|
480
|
|
|
|
|
|
|
); |
|
481
|
|
|
|
|
|
|
|
|
482
|
|
|
|
|
|
|
# Layer information |
|
483
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::Layers = ( |
|
484
|
|
|
|
|
|
|
PROCESS_PROC => \&ProcessLayers, |
|
485
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
486
|
|
|
|
|
|
|
NOTES => 'Tags extracted from Photoshop layer information.', |
|
487
|
|
|
|
|
|
|
# tags extracted from layer information |
|
488
|
|
|
|
|
|
|
# (tag ID's are for convenience only) |
|
489
|
|
|
|
|
|
|
_xcnt => { Name => 'LayerCount', Format => 'int16u' }, |
|
490
|
|
|
|
|
|
|
_xrct => { |
|
491
|
|
|
|
|
|
|
Name => 'LayerRectangles', |
|
492
|
|
|
|
|
|
|
Format => 'int32u', |
|
493
|
|
|
|
|
|
|
Count => 4, |
|
494
|
|
|
|
|
|
|
List => 1, |
|
495
|
|
|
|
|
|
|
Notes => 'top left bottom right', |
|
496
|
|
|
|
|
|
|
}, |
|
497
|
|
|
|
|
|
|
_xnam => { Name => 'LayerNames', |
|
498
|
|
|
|
|
|
|
Format => 'string', |
|
499
|
|
|
|
|
|
|
List => 1, |
|
500
|
|
|
|
|
|
|
ValueConv => q{ |
|
501
|
|
|
|
|
|
|
my $charset = $self->Options('CharsetPhotoshop') || 'Latin'; |
|
502
|
|
|
|
|
|
|
return $self->Decode($val, $charset); |
|
503
|
|
|
|
|
|
|
}, |
|
504
|
|
|
|
|
|
|
}, |
|
505
|
|
|
|
|
|
|
_xbnd => { |
|
506
|
|
|
|
|
|
|
Name => 'LayerBlendModes', |
|
507
|
|
|
|
|
|
|
Format => 'undef', |
|
508
|
|
|
|
|
|
|
List => 1, |
|
509
|
|
|
|
|
|
|
RawConv => 'GetByteOrder() eq "II" ? pack "N*", unpack "V*", $val : $val', |
|
510
|
|
|
|
|
|
|
PrintConv => { |
|
511
|
|
|
|
|
|
|
pass => 'Pass Through', |
|
512
|
|
|
|
|
|
|
norm => 'Normal', |
|
513
|
|
|
|
|
|
|
diss => 'Dissolve', |
|
514
|
|
|
|
|
|
|
dark => 'Darken', |
|
515
|
|
|
|
|
|
|
'mul '=> 'Multiply', |
|
516
|
|
|
|
|
|
|
idiv => 'Color Burn', |
|
517
|
|
|
|
|
|
|
lbrn => 'Linear Burn', |
|
518
|
|
|
|
|
|
|
dkCl => 'Darker Color', |
|
519
|
|
|
|
|
|
|
lite => 'Lighten', |
|
520
|
|
|
|
|
|
|
scrn => 'Screen', |
|
521
|
|
|
|
|
|
|
'div '=> 'Color Dodge', |
|
522
|
|
|
|
|
|
|
lddg => 'Linear Dodge', |
|
523
|
|
|
|
|
|
|
lgCl => 'Lighter Color', |
|
524
|
|
|
|
|
|
|
over => 'Overlay', |
|
525
|
|
|
|
|
|
|
sLit => 'Soft Light', |
|
526
|
|
|
|
|
|
|
hLit => 'Hard Light', |
|
527
|
|
|
|
|
|
|
vLit => 'Vivid Light', |
|
528
|
|
|
|
|
|
|
lLit => 'Linear Light', |
|
529
|
|
|
|
|
|
|
pLit => 'Pin Light', |
|
530
|
|
|
|
|
|
|
hMix => 'Hard Mix', |
|
531
|
|
|
|
|
|
|
diff => 'Difference', |
|
532
|
|
|
|
|
|
|
smud => 'Exclusion', |
|
533
|
|
|
|
|
|
|
fsub => 'Subtract', |
|
534
|
|
|
|
|
|
|
fdiv => 'Divide', |
|
535
|
|
|
|
|
|
|
'hue '=> 'Hue', |
|
536
|
|
|
|
|
|
|
'sat '=> 'Saturation', |
|
537
|
|
|
|
|
|
|
colr => 'Color', |
|
538
|
|
|
|
|
|
|
'lum '=> 'Luminosity', |
|
539
|
|
|
|
|
|
|
}, |
|
540
|
|
|
|
|
|
|
}, |
|
541
|
|
|
|
|
|
|
_xopc => { |
|
542
|
|
|
|
|
|
|
Name => 'LayerOpacities', |
|
543
|
|
|
|
|
|
|
Format => 'int8u', |
|
544
|
|
|
|
|
|
|
List => 1, |
|
545
|
|
|
|
|
|
|
ValueConv => '100 * $val / 255', |
|
546
|
|
|
|
|
|
|
PrintConv => 'sprintf("%d%%",$val)', |
|
547
|
|
|
|
|
|
|
}, |
|
548
|
|
|
|
|
|
|
# tags extracted from additional layer information (tag ID's are real) |
|
549
|
|
|
|
|
|
|
# - must be able to accommodate a blank entry to preserve the list ordering |
|
550
|
|
|
|
|
|
|
luni => { |
|
551
|
|
|
|
|
|
|
Name => 'LayerUnicodeNames', |
|
552
|
|
|
|
|
|
|
List => 1, |
|
553
|
|
|
|
|
|
|
RawConv => q{ |
|
554
|
|
|
|
|
|
|
return '' if length($val) < 4; |
|
555
|
|
|
|
|
|
|
my $len = Get32u(\$val, 0); |
|
556
|
|
|
|
|
|
|
return $self->Decode(substr($val, 4, $len * 2), 'UCS2'); |
|
557
|
|
|
|
|
|
|
}, |
|
558
|
|
|
|
|
|
|
}, |
|
559
|
|
|
|
|
|
|
lyid => { |
|
560
|
|
|
|
|
|
|
Name => 'LayerIDs', |
|
561
|
|
|
|
|
|
|
Description => 'Layer IDs', |
|
562
|
|
|
|
|
|
|
Format => 'int32u', |
|
563
|
|
|
|
|
|
|
List => 1, |
|
564
|
|
|
|
|
|
|
Unknown => 1, |
|
565
|
|
|
|
|
|
|
}, |
|
566
|
|
|
|
|
|
|
shmd => { # layer metadata (undocumented structure) |
|
567
|
|
|
|
|
|
|
# (for now, only extract layerTime. May also contain "layerXMP" -- |
|
568
|
|
|
|
|
|
|
# it would be nice to decode this but I need a sample) |
|
569
|
|
|
|
|
|
|
Name => 'LayerModifyDates', |
|
570
|
|
|
|
|
|
|
Groups => { 2 => 'Time' }, |
|
571
|
|
|
|
|
|
|
List => 1, |
|
572
|
|
|
|
|
|
|
RawConv => q{ |
|
573
|
|
|
|
|
|
|
return '' unless $val =~ /layerTime(doub|buod)(.{8})/s; |
|
574
|
|
|
|
|
|
|
my $tmp = $2; |
|
575
|
|
|
|
|
|
|
return GetDouble(\$tmp, 0); |
|
576
|
|
|
|
|
|
|
}, |
|
577
|
|
|
|
|
|
|
ValueConv => 'length $val ? ConvertUnixTime($val,1) : ""', |
|
578
|
|
|
|
|
|
|
PrintConv => 'length $val ? $self->ConvertDateTime($val) : ""', |
|
579
|
|
|
|
|
|
|
}, |
|
580
|
|
|
|
|
|
|
); |
|
581
|
|
|
|
|
|
|
|
|
582
|
|
|
|
|
|
|
# tags extracted from ImageSourceData found in TIFF images (ref PH) |
|
583
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::DocumentData = ( |
|
584
|
|
|
|
|
|
|
PROCESS_PROC => \&ProcessDocumentData, |
|
585
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
586
|
|
|
|
|
|
|
Layr => { |
|
587
|
|
|
|
|
|
|
Name => 'Layers', |
|
588
|
|
|
|
|
|
|
SubDirectory => { TagTable => 'Image::ExifTool::Photoshop::Layers' }, |
|
589
|
|
|
|
|
|
|
}, |
|
590
|
|
|
|
|
|
|
Lr16 => { # (NC) |
|
591
|
|
|
|
|
|
|
Name => 'Layers', |
|
592
|
|
|
|
|
|
|
SubDirectory => { TagTable => 'Image::ExifTool::Photoshop::Layers' }, |
|
593
|
|
|
|
|
|
|
}, |
|
594
|
|
|
|
|
|
|
); |
|
595
|
|
|
|
|
|
|
|
|
596
|
|
|
|
|
|
|
# image data |
|
597
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::ImageData = ( |
|
598
|
|
|
|
|
|
|
PROCESS_PROC => \&Image::ExifTool::ProcessBinaryData, |
|
599
|
|
|
|
|
|
|
GROUPS => { 2 => 'Image' }, |
|
600
|
|
|
|
|
|
|
0 => { |
|
601
|
|
|
|
|
|
|
Name => 'Compression', |
|
602
|
|
|
|
|
|
|
Format => 'int16u', |
|
603
|
|
|
|
|
|
|
PrintConv => { |
|
604
|
|
|
|
|
|
|
0 => 'Uncompressed', |
|
605
|
|
|
|
|
|
|
1 => 'RLE', |
|
606
|
|
|
|
|
|
|
2 => 'ZIP without prediction', |
|
607
|
|
|
|
|
|
|
3 => 'ZIP with prediction', |
|
608
|
|
|
|
|
|
|
}, |
|
609
|
|
|
|
|
|
|
}, |
|
610
|
|
|
|
|
|
|
); |
|
611
|
|
|
|
|
|
|
|
|
612
|
|
|
|
|
|
|
# tags for unknown resource types |
|
613
|
|
|
|
|
|
|
%Image::ExifTool::Photoshop::Unknown = ( |
|
614
|
|
|
|
|
|
|
GROUPS => { 2 => 'Unknown' }, |
|
615
|
|
|
|
|
|
|
); |
|
616
|
|
|
|
|
|
|
|
|
617
|
|
|
|
|
|
|
# define reference to IPTCDigest tagInfo hash for convenience |
|
618
|
|
|
|
|
|
|
$iptcDigestInfo = $Image::ExifTool::Photoshop::Main{0x0425}; |
|
619
|
|
|
|
|
|
|
|
|
620
|
|
|
|
|
|
|
|
|
621
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
622
|
|
|
|
|
|
|
# AutoLoad our writer routines when necessary |
|
623
|
|
|
|
|
|
|
# |
|
624
|
|
|
|
|
|
|
sub AUTOLOAD |
|
625
|
|
|
|
|
|
|
{ |
|
626
|
15
|
|
|
15
|
|
86
|
return Image::ExifTool::DoAutoLoad($AUTOLOAD, @_); |
|
627
|
|
|
|
|
|
|
} |
|
628
|
|
|
|
|
|
|
|
|
629
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
630
|
|
|
|
|
|
|
# Convert pascal string(s) to something we can use |
|
631
|
|
|
|
|
|
|
# Inputs: 1) Pascal string data |
|
632
|
|
|
|
|
|
|
# Returns: Strings, concatenated with ', ' |
|
633
|
|
|
|
|
|
|
sub ConvertPascalString($$) |
|
634
|
|
|
|
|
|
|
{ |
|
635
|
0
|
|
|
0
|
0
|
0
|
my ($et, $inStr) = @_; |
|
636
|
0
|
|
|
|
|
0
|
my $outStr = ''; |
|
637
|
0
|
|
|
|
|
0
|
my $len = length($inStr); |
|
638
|
0
|
|
|
|
|
0
|
my $i=0; |
|
639
|
0
|
|
|
|
|
0
|
while ($i < $len) { |
|
640
|
0
|
|
|
|
|
0
|
my $n = ord(substr($inStr, $i, 1)); |
|
641
|
0
|
0
|
|
|
|
0
|
last if $i + $n >= $len; |
|
642
|
0
|
0
|
|
|
|
0
|
$i and $outStr .= ', '; |
|
643
|
0
|
|
|
|
|
0
|
$outStr .= substr($inStr, $i+1, $n); |
|
644
|
0
|
|
|
|
|
0
|
$i += $n + 1; |
|
645
|
|
|
|
|
|
|
} |
|
646
|
0
|
|
0
|
|
|
0
|
my $charset = $et->Options('CharsetPhotoshop') || 'Latin'; |
|
647
|
0
|
|
|
|
|
0
|
return $et->Decode($outStr, $charset); |
|
648
|
|
|
|
|
|
|
} |
|
649
|
|
|
|
|
|
|
|
|
650
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
651
|
|
|
|
|
|
|
# Process Photoshop layers and mask information section of PSD/PSB file |
|
652
|
|
|
|
|
|
|
# Inputs: 0) ExifTool ref, 1) DirInfo ref, 2) tag table ref |
|
653
|
|
|
|
|
|
|
# Returns: 1 on success (and seeks to the end of this section) |
|
654
|
|
|
|
|
|
|
sub ProcessLayersAndMask($$$) |
|
655
|
|
|
|
|
|
|
{ |
|
656
|
4
|
|
|
4
|
0
|
7
|
local $_; |
|
657
|
4
|
|
|
|
|
10
|
my ($et, $dirInfo, $tagTablePtr) = @_; |
|
658
|
4
|
|
|
|
|
11
|
my $raf = $$dirInfo{RAF}; |
|
659
|
4
|
|
|
|
|
11
|
my $fileType = $$et{VALUE}{FileType}; |
|
660
|
4
|
|
|
|
|
7
|
my $data; |
|
661
|
|
|
|
|
|
|
|
|
662
|
4
|
50
|
33
|
|
|
15
|
return 0 unless $fileType eq 'PSD' or $fileType eq 'PSB'; # (no layer section in CS1 files) |
|
663
|
|
|
|
|
|
|
|
|
664
|
|
|
|
|
|
|
# (some words are 4 bytes in PSD files and 8 bytes in PSB) |
|
665
|
4
|
50
|
|
|
|
16
|
my ($psb, $psiz) = $fileType eq 'PSB' ? (1, 8) : (undef, 4); |
|
666
|
|
|
|
|
|
|
|
|
667
|
|
|
|
|
|
|
# read the layer information header |
|
668
|
4
|
|
|
|
|
10
|
my $n = $psiz * 2 + 2; |
|
669
|
4
|
50
|
|
|
|
22
|
$raf->Read($data, $n) == $n or return 0; |
|
670
|
4
|
50
|
|
|
|
19
|
my $tot = $psb ? Get64u(\$data, 0) : Get32u(\$data, 0); # length of layer and mask info |
|
671
|
4
|
50
|
|
|
|
16
|
return 1 if $tot == 0; |
|
672
|
4
|
|
|
|
|
26
|
my $end = $raf->Tell() - $psiz - 2 + $tot; |
|
673
|
4
|
|
|
|
|
14
|
$data = substr $data, $psiz; |
|
674
|
4
|
50
|
|
|
|
15
|
my $len = $psb ? Get64u(\$data, 0) : Get32u(\$data, 0); # length of layer info section |
|
675
|
4
|
|
|
|
|
12
|
my $num = Get16s(\$data, $psiz); |
|
676
|
|
|
|
|
|
|
# check for Lr16 block if layers length is 0 (ref https://forums.adobe.com/thread/1540914) |
|
677
|
4
|
50
|
33
|
|
|
22
|
if ($len == 0 and $num == 0) { |
|
678
|
4
|
50
|
|
|
|
12
|
$raf->Read($data,10) == 10 or return 0; |
|
679
|
4
|
50
|
|
|
|
26
|
if ($data =~ /^..8BIMLr16/s) { |
|
|
|
50
|
|
|
|
|
|
|
680
|
0
|
0
|
|
|
|
0
|
$raf->Read($data, $psiz+2) == $psiz+2 or return 0; |
|
681
|
0
|
0
|
|
|
|
0
|
$len = $psb ? Get64u(\$data, 0) : Get32u(\$data, 0); |
|
682
|
|
|
|
|
|
|
} elsif ($data =~ /^..8BIMMt16/s) { # (have seen Mt16 before Lr16, ref PH) |
|
683
|
0
|
0
|
|
|
|
0
|
$raf->Read($data, $psiz) == $psiz or return 0; |
|
684
|
0
|
0
|
|
|
|
0
|
$raf->Read($data, 8) == 8 or return 0; |
|
685
|
0
|
0
|
|
|
|
0
|
if ($data eq '8BIMLr16') { |
|
686
|
0
|
0
|
|
|
|
0
|
$raf->Read($data, $psiz+2) == $psiz+2 or return 0; |
|
687
|
0
|
0
|
|
|
|
0
|
$len = $psb ? Get64u(\$data, 0) : Get32u(\$data, 0); |
|
688
|
|
|
|
|
|
|
} else { |
|
689
|
0
|
0
|
|
|
|
0
|
$raf->Seek(-18-$psiz, 1) or return 0; |
|
690
|
|
|
|
|
|
|
} |
|
691
|
|
|
|
|
|
|
} else { |
|
692
|
4
|
50
|
|
|
|
17
|
$raf->Seek(-10, 1) or return 0; |
|
693
|
|
|
|
|
|
|
} |
|
694
|
|
|
|
|
|
|
} |
|
695
|
4
|
|
|
|
|
12
|
$len += 2; # include layer count with layer info section |
|
696
|
4
|
50
|
|
|
|
14
|
$raf->Seek(-2, 1) or return 0; |
|
697
|
4
|
|
|
|
|
23
|
my %dinfo = ( |
|
698
|
|
|
|
|
|
|
RAF => $raf, |
|
699
|
|
|
|
|
|
|
DirLen => $len, |
|
700
|
|
|
|
|
|
|
); |
|
701
|
4
|
|
|
|
|
13
|
$$et{IsPSB} = $psb; # set PSB flag |
|
702
|
4
|
|
|
|
|
19
|
ProcessLayers($et, \%dinfo, $tagTablePtr); |
|
703
|
|
|
|
|
|
|
|
|
704
|
|
|
|
|
|
|
# seek to the end of this section and return success flag |
|
705
|
4
|
50
|
|
|
|
18
|
return $raf->Seek($end, 0) ? 1 : 0; |
|
706
|
|
|
|
|
|
|
} |
|
707
|
|
|
|
|
|
|
|
|
708
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
709
|
|
|
|
|
|
|
# Process Photoshop layers (beginning with layer count) |
|
710
|
|
|
|
|
|
|
# Inputs: 0) ExifTool ref, 1) DirInfo ref, 2) tag table ref |
|
711
|
|
|
|
|
|
|
# Returns: 1 on success |
|
712
|
|
|
|
|
|
|
# Notes: Uses ExifTool IsPSB member to determine whether file is PSB format |
|
713
|
|
|
|
|
|
|
sub ProcessLayers($$$) |
|
714
|
|
|
|
|
|
|
{ |
|
715
|
4
|
|
|
4
|
0
|
7
|
local $_; |
|
716
|
4
|
|
|
|
|
12
|
my ($et, $dirInfo, $tagTablePtr) = @_; |
|
717
|
4
|
|
|
|
|
9
|
my ($i, $n, %count, $buff, $buf2); |
|
718
|
4
|
|
|
|
|
8
|
my $raf = $$dirInfo{RAF}; |
|
719
|
4
|
|
|
|
|
10
|
my $dirLen = $$dirInfo{DirLen}; |
|
720
|
4
|
|
|
|
|
10
|
my $verbose = $$et{OPTIONS}{Verbose}; |
|
721
|
4
|
|
|
|
|
16
|
my %dinfo = ( DataPt => \$buff, Base => $raf->Tell() ); |
|
722
|
4
|
|
|
|
|
9
|
my $pos = 0; |
|
723
|
4
|
50
|
|
|
|
12
|
return 0 if $dirLen < 2; |
|
724
|
4
|
50
|
|
|
|
14
|
$raf->Read($buff, 2) == 2 or return 0; |
|
725
|
4
|
|
|
|
|
14
|
my $num = Get16s(\$buff, 0); |
|
726
|
4
|
50
|
|
|
|
17
|
$num = -$num if $num < 0; # (first channel is transparency data if negative) |
|
727
|
4
|
|
|
|
|
22
|
$et->VerboseDir('Layers', $num, $dirLen); |
|
728
|
4
|
|
|
|
|
22
|
$et->HandleTag($tagTablePtr, '_xcnt', $num, Start => $pos, Size => 2, %dinfo); # LayerCount |
|
729
|
4
|
|
|
|
|
10
|
my $oldIndent = $$et{INDENT}; |
|
730
|
4
|
|
|
|
|
12
|
$$et{INDENT} .= '| '; |
|
731
|
|
|
|
|
|
|
|
|
732
|
4
|
|
|
|
|
8
|
$pos += 2; |
|
733
|
4
|
|
|
|
|
7
|
my $psb = $$et{IsPSB}; # is PSB format? |
|
734
|
4
|
50
|
|
|
|
9
|
my $psiz = $psb ? 8 : 4; |
|
735
|
4
|
|
|
|
|
18
|
for ($i=0; $i<$num; ++$i) { |
|
736
|
0
|
|
|
|
|
0
|
$et->VPrint(0, $oldIndent.'+ [Layer '.($i+1)." of $num]\n"); |
|
737
|
0
|
0
|
|
|
|
0
|
last if $pos + 18 > $dirLen; |
|
738
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, 18) == 18 or last; |
|
739
|
0
|
|
|
|
|
0
|
$dinfo{DataPos} = $pos; |
|
740
|
|
|
|
|
|
|
# save the layer rectangle |
|
741
|
0
|
|
|
|
|
0
|
$et->HandleTag($tagTablePtr, '_xrct', undef, Start => 0, Size => 16, %dinfo); |
|
742
|
0
|
|
|
|
|
0
|
my $numChannels = Get16u(\$buff, 16); |
|
743
|
0
|
|
|
|
|
0
|
$n = (2 + $psiz) * $numChannels; # size of channel information |
|
744
|
0
|
0
|
|
|
|
0
|
$raf->Seek($n, 1) or last; |
|
745
|
0
|
|
|
|
|
0
|
$pos += 18 + $n; |
|
746
|
0
|
0
|
|
|
|
0
|
last if $pos + 20 > $dirLen; |
|
747
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, 20) == 20 or last; |
|
748
|
0
|
|
|
|
|
0
|
$dinfo{DataPos} = $pos; |
|
749
|
0
|
|
|
|
|
0
|
my $sig = substr($buff, 0, 4); |
|
750
|
0
|
0
|
|
|
|
0
|
$sig =~ /^(8BIM|MIB8)$/ or last; # verify signature |
|
751
|
0
|
|
|
|
|
0
|
$et->HandleTag($tagTablePtr, '_xbnd', undef, Start => 4, Size => 4, %dinfo); |
|
752
|
0
|
|
|
|
|
0
|
$et->HandleTag($tagTablePtr, '_xopc', undef, Start => 8, Size => 1, %dinfo); |
|
753
|
0
|
|
|
|
|
0
|
my $nxt = $pos + 16 + Get32u(\$buff, 12); |
|
754
|
0
|
|
|
|
|
0
|
$n = Get32u(\$buff, 16); # get size of layer mask data |
|
755
|
0
|
|
|
|
|
0
|
$pos += 20 + $n; # skip layer mask data |
|
756
|
0
|
0
|
|
|
|
0
|
last if $pos + 4 > $dirLen; |
|
757
|
0
|
0
|
0
|
|
|
0
|
$raf->Seek($n, 1) and $raf->Read($buff, 4) == 4 or last; |
|
758
|
0
|
|
|
|
|
0
|
$n = Get32u(\$buff, 0); # get size of layer blending ranges |
|
759
|
0
|
|
|
|
|
0
|
$pos += 4 + $n; # skip layer blending ranges data |
|
760
|
0
|
0
|
|
|
|
0
|
last if $pos + 1 > $dirLen; |
|
761
|
0
|
0
|
0
|
|
|
0
|
$raf->Seek($n, 1) and $raf->Read($buff, 1) == 1 or last; |
|
762
|
0
|
|
|
|
|
0
|
$n = Get8u(\$buff, 0); # get length of layer name |
|
763
|
0
|
0
|
|
|
|
0
|
last if $pos + 1 + $n > $dirLen; |
|
764
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, $n) == $n or last; |
|
765
|
0
|
|
|
|
|
0
|
$dinfo{DataPos} = $pos + 1; |
|
766
|
0
|
|
|
|
|
0
|
$et->HandleTag($tagTablePtr, '_xnam', undef, Start => 0, Size => $n, %dinfo); |
|
767
|
0
|
|
|
|
|
0
|
my $frag = ($n + 1) & 0x3; |
|
768
|
0
|
0
|
0
|
|
|
0
|
$raf->Seek(4 - $frag, 1) or last if $frag; |
|
769
|
0
|
|
|
|
|
0
|
$n = ($n + 4) & 0xfffffffc; # +1 for length byte then pad to multiple of 4 bytes |
|
770
|
0
|
|
|
|
|
0
|
$pos += $n; |
|
771
|
|
|
|
|
|
|
# process additional layer info |
|
772
|
0
|
|
|
|
|
0
|
while ($pos + 12 <= $nxt) { |
|
773
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, 12) == 12 or last; |
|
774
|
0
|
|
|
|
|
0
|
my $dat = substr($buff, 0, 8); |
|
775
|
0
|
0
|
|
|
|
0
|
$dat = pack 'N*', unpack 'V*', $dat if GetByteOrder() eq 'II'; |
|
776
|
0
|
|
|
|
|
0
|
my $sig = substr($dat, 0, 4); |
|
777
|
0
|
0
|
0
|
|
|
0
|
last unless $sig eq '8BIM' or $sig eq '8B64'; # verify signature |
|
778
|
0
|
|
|
|
|
0
|
my $tag = substr($dat, 4, 4); |
|
779
|
|
|
|
|
|
|
# (some structures have an 8-byte size word [augh!] |
|
780
|
|
|
|
|
|
|
# --> it would be great if '8B64' indicated a 64-bit version, and this may well |
|
781
|
|
|
|
|
|
|
# be the case, but it is not mentioned in the Photoshop file format specification) |
|
782
|
0
|
0
|
0
|
|
|
0
|
if ($psb and $tag =~ /^(LMsk|Lr16|Lr32|Layr|Mt16|Mt32|Mtrn|Alph|FMsk|lnk2|FEid|FXid|PxSD)$/) { |
|
783
|
0
|
0
|
|
|
|
0
|
last if $pos + 16 > $nxt; |
|
784
|
0
|
0
|
|
|
|
0
|
$raf->Read($buf2, 4) == 4 or last; |
|
785
|
0
|
|
|
|
|
0
|
$buff .= $buf2; |
|
786
|
0
|
|
|
|
|
0
|
$n = Get64u(\$buff, 8); |
|
787
|
0
|
|
|
|
|
0
|
$pos += 4; |
|
788
|
|
|
|
|
|
|
} else { |
|
789
|
0
|
|
|
|
|
0
|
$n = Get32u(\$buff, 8); |
|
790
|
|
|
|
|
|
|
} |
|
791
|
0
|
|
|
|
|
0
|
$pos += 12; |
|
792
|
0
|
0
|
|
|
|
0
|
last if $pos + $n > $nxt; |
|
793
|
0
|
|
|
|
|
0
|
$frag = $n & 0x3; |
|
794
|
0
|
0
|
0
|
|
|
0
|
if ($$tagTablePtr{$tag} or $verbose) { |
|
795
|
|
|
|
|
|
|
# pad with empty entries if necessary to keep the same index for each item in the layer |
|
796
|
0
|
0
|
|
|
|
0
|
$count{$tag} = 0 unless defined $count{$tag}; |
|
797
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, $n) == $n or last; |
|
798
|
0
|
|
|
|
|
0
|
$dinfo{DataPos} = $pos; |
|
799
|
0
|
|
|
|
|
0
|
while ($count{$tag} < $i) { |
|
800
|
0
|
|
|
|
|
0
|
$et->HandleTag($tagTablePtr, $tag, ''); |
|
801
|
0
|
|
|
|
|
0
|
++$count{$tag}; |
|
802
|
|
|
|
|
|
|
} |
|
803
|
0
|
|
|
|
|
0
|
$et->HandleTag($tagTablePtr, $tag, undef, Start => 0, Size => $n, %dinfo); |
|
804
|
0
|
|
|
|
|
0
|
++$count{$tag}; |
|
805
|
0
|
0
|
|
|
|
0
|
if ($frag) { |
|
806
|
0
|
0
|
|
|
|
0
|
$raf->Seek(4 - $frag, 1) or last; |
|
807
|
0
|
|
|
|
|
0
|
$n += 4 - $frag; # pad to multiple of 4 bytes (PH NC) |
|
808
|
|
|
|
|
|
|
} |
|
809
|
|
|
|
|
|
|
} else { |
|
810
|
0
|
0
|
|
|
|
0
|
$n += 4 - $frag if $frag; |
|
811
|
0
|
0
|
|
|
|
0
|
$raf->Seek($n, 1) or last; |
|
812
|
|
|
|
|
|
|
} |
|
813
|
0
|
|
|
|
|
0
|
$pos += $n; # step to start of next structure |
|
814
|
|
|
|
|
|
|
} |
|
815
|
0
|
|
|
|
|
0
|
$pos = $nxt; |
|
816
|
|
|
|
|
|
|
} |
|
817
|
4
|
|
|
|
|
19
|
$$et{INDENT} = $oldIndent; |
|
818
|
4
|
|
|
|
|
9
|
return 1; |
|
819
|
|
|
|
|
|
|
} |
|
820
|
|
|
|
|
|
|
|
|
821
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
822
|
|
|
|
|
|
|
# Process Photoshop ImageSourceData |
|
823
|
|
|
|
|
|
|
# Inputs: 0) ExifTool ref, 1) dirInfo ref, 2) tag table ref |
|
824
|
|
|
|
|
|
|
# Returns: 1 on success |
|
825
|
|
|
|
|
|
|
sub ProcessDocumentData($$$) |
|
826
|
|
|
|
|
|
|
{ |
|
827
|
0
|
|
|
0
|
0
|
0
|
my ($et, $dirInfo, $tagTablePtr) = @_; |
|
828
|
0
|
|
|
|
|
0
|
my $verbose = $$et{OPTIONS}{Verbose}; |
|
829
|
0
|
|
|
|
|
0
|
my $raf = $$dirInfo{RAF}; |
|
830
|
0
|
|
|
|
|
0
|
my $dirLen = $$dirInfo{DirLen}; |
|
831
|
0
|
|
|
|
|
0
|
my $pos = 36; # length of header |
|
832
|
0
|
|
|
|
|
0
|
my ($buff, $n, $err); |
|
833
|
|
|
|
|
|
|
|
|
834
|
0
|
|
|
|
|
0
|
$et->VerboseDir('Photoshop Document Data', undef, $dirLen); |
|
835
|
0
|
0
|
|
|
|
0
|
unless ($raf) { |
|
836
|
0
|
|
|
|
|
0
|
my $dataPt = $$dirInfo{DataPt}; |
|
837
|
0
|
|
0
|
|
|
0
|
my $start = $$dirInfo{DirStart} || 0; |
|
838
|
0
|
|
|
|
|
0
|
$raf = new File::RandomAccess($dataPt); |
|
839
|
0
|
0
|
|
|
|
0
|
$raf->Seek($start, 0) if $start; |
|
840
|
0
|
0
|
|
|
|
0
|
$dirLen = length $$dataPt - $start unless defined $dirLen; |
|
841
|
0
|
|
|
|
|
0
|
$et->VerboseDump($dataPt, Start => $start, Len => $dirLen, Base => $$dirInfo{Base}); |
|
842
|
|
|
|
|
|
|
} |
|
843
|
0
|
0
|
0
|
|
|
0
|
unless ($raf->Read($buff, $pos) == $pos and |
|
844
|
|
|
|
|
|
|
$buff =~ /^Adobe Photoshop Document Data (Block|V0002)\0/) |
|
845
|
|
|
|
|
|
|
{ |
|
846
|
0
|
|
|
|
|
0
|
$et->Warn('Invalid Photoshop Document Data'); |
|
847
|
0
|
|
|
|
|
0
|
return 0; |
|
848
|
|
|
|
|
|
|
} |
|
849
|
0
|
|
|
|
|
0
|
my $psb = ($1 eq 'V0002'); |
|
850
|
0
|
|
|
|
|
0
|
my %dinfo = ( DataPt => \$buff ); |
|
851
|
0
|
|
|
|
|
0
|
$$et{IsPSB} = $psb; # set PSB flag (needed when handling Layers directory) |
|
852
|
0
|
|
|
|
|
0
|
while ($pos + 12 <= $dirLen) { |
|
853
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, 8) == 8 or $err = 'Error reading document data', last; |
|
854
|
|
|
|
|
|
|
# set byte order according to byte order of first signature |
|
855
|
0
|
0
|
|
|
|
0
|
SetByteOrder($buff =~ /^(8BIM|8B64)/ ? 'MM' : 'II') if $pos == 36; |
|
|
|
0
|
|
|
|
|
|
|
856
|
0
|
0
|
|
|
|
0
|
$buff = pack 'N*', unpack 'V*', $buff if GetByteOrder() eq 'II'; |
|
857
|
0
|
|
|
|
|
0
|
my $sig = substr($buff, 0, 4); |
|
858
|
0
|
0
|
0
|
|
|
0
|
$sig eq '8BIM' or $sig eq '8B64' or $err = 'Bad photoshop resource', last; # verify signature |
|
859
|
0
|
|
|
|
|
0
|
my $tag = substr($buff, 4, 4); |
|
860
|
0
|
0
|
0
|
|
|
0
|
if ($psb and $tag =~ /^(LMsk|Lr16|Lr32|Layr|Mt16|Mt32|Mtrn|Alph|FMsk|lnk2|FEid|FXid|PxSD)$/) { |
|
861
|
0
|
0
|
|
|
|
0
|
$pos + 16 > $dirLen and $err = 'Short PSB resource', last; |
|
862
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, 8) == 8 or $err = 'Error reading PSB resource', last; |
|
863
|
0
|
|
|
|
|
0
|
$n = Get64u(\$buff, 0); |
|
864
|
0
|
|
|
|
|
0
|
$pos += 4; |
|
865
|
|
|
|
|
|
|
} else { |
|
866
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, 4) == 4 or $err = 'Error reading PSD resource', last; |
|
867
|
0
|
|
|
|
|
0
|
$n = Get32u(\$buff, 0); |
|
868
|
|
|
|
|
|
|
} |
|
869
|
0
|
|
|
|
|
0
|
$pos += 12; |
|
870
|
0
|
0
|
|
|
|
0
|
$pos + $n > $dirLen and $err = 'Truncated photoshop resource', last; |
|
871
|
0
|
|
|
|
|
0
|
my $pad = (4 - ($n & 3)) & 3; # number of padding bytes |
|
872
|
0
|
|
|
|
|
0
|
my $tagInfo = $$tagTablePtr{$tag}; |
|
873
|
0
|
0
|
0
|
|
|
0
|
if ($tagInfo or $verbose) { |
|
874
|
0
|
0
|
0
|
|
|
0
|
if ($tagInfo and $$tagInfo{SubDirectory}) { |
|
875
|
0
|
|
|
|
|
0
|
my $fpos = $raf->Tell() + $n + $pad; |
|
876
|
0
|
|
|
|
|
0
|
my $subTable = GetTagTable($$tagInfo{SubDirectory}{TagTable}); |
|
877
|
0
|
|
|
|
|
0
|
$et->ProcessDirectory({ RAF => $raf, DirLen => $n }, $subTable); |
|
878
|
0
|
0
|
|
|
|
0
|
$raf->Seek($fpos, 0) or $err = 'Seek error', last; |
|
879
|
|
|
|
|
|
|
} else { |
|
880
|
0
|
|
|
|
|
0
|
$dinfo{DataPos} = $raf->Tell(); |
|
881
|
0
|
|
|
|
|
0
|
$dinfo{Start} = 0; |
|
882
|
0
|
|
|
|
|
0
|
$dinfo{Size} = $n; |
|
883
|
0
|
0
|
|
|
|
0
|
$raf->Read($buff, $n) == $n or $err = 'Error reading photoshop resource', last; |
|
884
|
0
|
|
|
|
|
0
|
$et->HandleTag($tagTablePtr, $tag, undef, %dinfo); |
|
885
|
0
|
0
|
|
|
|
0
|
$raf->Seek($pad, 1) or $err = 'Seek error', last; |
|
886
|
|
|
|
|
|
|
} |
|
887
|
|
|
|
|
|
|
} else { |
|
888
|
0
|
0
|
|
|
|
0
|
$raf->Seek($n + $pad, 1) or $err = 'Seek error', last; |
|
889
|
|
|
|
|
|
|
} |
|
890
|
0
|
|
|
|
|
0
|
$pos += $n + $pad; # step to start of next structure |
|
891
|
|
|
|
|
|
|
} |
|
892
|
0
|
0
|
|
|
|
0
|
$err and $et->Warn($err); |
|
893
|
0
|
|
|
|
|
0
|
return 1; |
|
894
|
|
|
|
|
|
|
} |
|
895
|
|
|
|
|
|
|
|
|
896
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
897
|
|
|
|
|
|
|
# Process Photoshop APP13 record |
|
898
|
|
|
|
|
|
|
# Inputs: 0) ExifTool object reference, 1) Reference to directory information |
|
899
|
|
|
|
|
|
|
# 2) Tag table reference |
|
900
|
|
|
|
|
|
|
# Returns: 1 on success |
|
901
|
|
|
|
|
|
|
sub ProcessPhotoshop($$$) |
|
902
|
|
|
|
|
|
|
{ |
|
903
|
93
|
|
|
93
|
0
|
243
|
my ($et, $dirInfo, $tagTablePtr) = @_; |
|
904
|
93
|
|
|
|
|
211
|
my $dataPt = $$dirInfo{DataPt}; |
|
905
|
93
|
|
|
|
|
206
|
my $pos = $$dirInfo{DirStart}; |
|
906
|
93
|
|
|
|
|
200
|
my $dirEnd = $pos + $$dirInfo{DirLen}; |
|
907
|
93
|
|
|
|
|
301
|
my $verbose = $et->Options('Verbose'); |
|
908
|
93
|
|
|
|
|
224
|
my $success = 0; |
|
909
|
|
|
|
|
|
|
|
|
910
|
|
|
|
|
|
|
# ignore non-standard XMP while in strict MWG compatibility mode |
|
911
|
93
|
100
|
66
|
|
|
453
|
if (($Image::ExifTool::MWG::strict or $et->Options('Validate')) and |
|
|
|
|
66
|
|
|
|
|
|
912
|
|
|
|
|
|
|
$$et{FILE_TYPE} =~ /^(JPEG|TIFF|PSD)$/) |
|
913
|
|
|
|
|
|
|
{ |
|
914
|
5
|
|
|
|
|
21
|
my $path = $et->MetadataPath(); |
|
915
|
5
|
50
|
|
|
|
31
|
unless ($path =~ /^(JPEG-APP13-Photoshop|TIFF-IFD0-Photoshop|PSD)$/) { |
|
916
|
0
|
0
|
|
|
|
0
|
if ($Image::ExifTool::MWG::strict) { |
|
917
|
0
|
|
|
|
|
0
|
$et->Warn("Ignored non-standard Photoshop at $path"); |
|
918
|
0
|
|
|
|
|
0
|
return 1; |
|
919
|
|
|
|
|
|
|
} else { |
|
920
|
0
|
|
|
|
|
0
|
$et->Warn("Non-standard Photoshop at $path", 1); |
|
921
|
|
|
|
|
|
|
} |
|
922
|
|
|
|
|
|
|
} |
|
923
|
|
|
|
|
|
|
} |
|
924
|
93
|
50
|
66
|
|
|
498
|
if ($$et{FILE_TYPE} eq 'JPEG' and $$dirInfo{Parent} ne 'APP13') { |
|
925
|
0
|
|
|
|
|
0
|
$$et{LOW_PRIORITY_DIR}{'*'} = 1; # lower priority of all these tags |
|
926
|
|
|
|
|
|
|
} |
|
927
|
93
|
|
|
|
|
352
|
SetByteOrder('MM'); # Photoshop is always big-endian |
|
928
|
93
|
50
|
|
|
|
284
|
$verbose and $et->VerboseDir('Photoshop', 0, $$dirInfo{DirLen}); |
|
929
|
|
|
|
|
|
|
|
|
930
|
|
|
|
|
|
|
# scan through resource blocks: |
|
931
|
|
|
|
|
|
|
# Format: 0) Type, 4 bytes - '8BIM' (or the rare 'PHUT', 'DCSR', 'AgHg' or 'MeSa') |
|
932
|
|
|
|
|
|
|
# 1) TagID,2 bytes |
|
933
|
|
|
|
|
|
|
# 2) Name, pascal string padded to even no. bytes |
|
934
|
|
|
|
|
|
|
# 3) Size, 4 bytes - N |
|
935
|
|
|
|
|
|
|
# 4) Data, N bytes |
|
936
|
93
|
|
|
|
|
321
|
while ($pos + 8 < $dirEnd) { |
|
937
|
1182
|
|
|
|
|
2006
|
my $type = substr($$dataPt, $pos, 4); |
|
938
|
1182
|
|
|
|
|
1633
|
my ($ttPtr, $extra, $val, $name); |
|
939
|
1182
|
50
|
|
|
|
1940
|
if ($type eq '8BIM') { |
|
|
|
0
|
|
|
|
|
|
|
940
|
1182
|
|
|
|
|
1455
|
$ttPtr = $tagTablePtr; |
|
941
|
|
|
|
|
|
|
} elsif ($type =~ /^(PHUT|DCSR|AgHg|MeSa)$/) { # (PHUT~ImageReady, MeSa~PhotoDeluxe) |
|
942
|
0
|
|
|
|
|
0
|
$ttPtr = GetTagTable('Image::ExifTool::Photoshop::Unknown'); |
|
943
|
|
|
|
|
|
|
} else { |
|
944
|
0
|
|
|
|
|
0
|
$type =~ s/([^\w])/sprintf("\\x%.2x",ord($1))/ge; |
|
|
0
|
|
|
|
|
0
|
|
|
945
|
0
|
|
|
|
|
0
|
$et->Warn(qq{Bad Photoshop IRB resource "$type"}); |
|
946
|
0
|
|
|
|
|
0
|
last; |
|
947
|
|
|
|
|
|
|
} |
|
948
|
1182
|
|
|
|
|
2533
|
my $tag = Get16u($dataPt, $pos + 4); |
|
949
|
1182
|
|
|
|
|
1763
|
$pos += 6; # point to start of name |
|
950
|
1182
|
|
|
|
|
2155
|
my $nameLen = Get8u($dataPt, $pos); |
|
951
|
1182
|
|
|
|
|
1735
|
my $namePos = ++$pos; |
|
952
|
|
|
|
|
|
|
# skip resource block name (pascal string, padded to an even # of bytes) |
|
953
|
1182
|
|
|
|
|
1397
|
$pos += $nameLen; |
|
954
|
1182
|
50
|
|
|
|
2097
|
++$pos unless $nameLen & 0x01; |
|
955
|
1182
|
50
|
|
|
|
1958
|
if ($pos + 4 > $dirEnd) { |
|
956
|
0
|
|
|
|
|
0
|
$et->Warn("Bad Photoshop resource block"); |
|
957
|
0
|
|
|
|
|
0
|
last; |
|
958
|
|
|
|
|
|
|
} |
|
959
|
1182
|
|
|
|
|
2154
|
my $size = Get32u($dataPt, $pos); |
|
960
|
1182
|
|
|
|
|
1570
|
$pos += 4; |
|
961
|
1182
|
50
|
|
|
|
2111
|
if ($size + $pos > $dirEnd) { |
|
962
|
0
|
|
|
|
|
0
|
$et->Warn("Bad Photoshop resource data size $size"); |
|
963
|
0
|
|
|
|
|
0
|
last; |
|
964
|
|
|
|
|
|
|
} |
|
965
|
1182
|
|
|
|
|
1460
|
$success = 1; |
|
966
|
1182
|
50
|
|
|
|
1842
|
if ($nameLen) { |
|
967
|
0
|
|
|
|
|
0
|
$name = substr($$dataPt, $namePos, $nameLen); |
|
968
|
0
|
|
|
|
|
0
|
$extra = qq{, Name="$name"}; |
|
969
|
|
|
|
|
|
|
} else { |
|
970
|
1182
|
|
|
|
|
1573
|
$name = ''; |
|
971
|
|
|
|
|
|
|
} |
|
972
|
1182
|
|
|
|
|
2399
|
my $tagInfo = $et->GetTagInfo($ttPtr, $tag); |
|
973
|
|
|
|
|
|
|
# append resource name to value if requested (braced by "/#...#/") |
|
974
|
1182
|
0
|
66
|
|
|
3596
|
if ($tagInfo and defined $$tagInfo{SetResourceName} and |
|
|
|
|
33
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
975
|
|
|
|
|
|
|
$$tagInfo{SetResourceName} eq '1' and $name !~ m{/#}) |
|
976
|
|
|
|
|
|
|
{ |
|
977
|
0
|
|
|
|
|
0
|
$val = substr($$dataPt, $pos, $size) . '/#' . $name . '#/'; |
|
978
|
|
|
|
|
|
|
} |
|
979
|
|
|
|
|
|
|
$et->HandleTag($ttPtr, $tag, $val, |
|
980
|
|
|
|
|
|
|
TagInfo => $tagInfo, |
|
981
|
|
|
|
|
|
|
Extra => $extra, |
|
982
|
|
|
|
|
|
|
DataPt => $dataPt, |
|
983
|
|
|
|
|
|
|
DataPos => $$dirInfo{DataPos}, |
|
984
|
|
|
|
|
|
|
Size => $size, |
|
985
|
|
|
|
|
|
|
Start => $pos, |
|
986
|
|
|
|
|
|
|
Base => $$dirInfo{Base}, |
|
987
|
|
|
|
|
|
|
Parent => $$dirInfo{DirName}, |
|
988
|
1182
|
|
|
|
|
4710
|
); |
|
989
|
1182
|
100
|
|
|
|
2978
|
$size += 1 if $size & 0x01; # size is padded to an even # bytes |
|
990
|
1182
|
|
|
|
|
2701
|
$pos += $size; |
|
991
|
|
|
|
|
|
|
} |
|
992
|
|
|
|
|
|
|
# warn about incorrect IPTCDigest |
|
993
|
93
|
100
|
100
|
|
|
666
|
if ($$et{VALUE}{IPTCDigest} and $$et{VALUE}{CurrentIPTCDigest} and |
|
|
|
|
100
|
|
|
|
|
|
994
|
|
|
|
|
|
|
$$et{VALUE}{IPTCDigest} ne $$et{VALUE}{CurrentIPTCDigest}) |
|
995
|
|
|
|
|
|
|
{ |
|
996
|
32
|
|
|
|
|
156
|
$et->WarnOnce('IPTCDigest is not current. XMP may be out of sync'); |
|
997
|
|
|
|
|
|
|
} |
|
998
|
93
|
|
|
|
|
236
|
delete $$et{LOW_PRIORITY_DIR}{'*'}; |
|
999
|
93
|
|
|
|
|
252
|
return $success; |
|
1000
|
|
|
|
|
|
|
} |
|
1001
|
|
|
|
|
|
|
|
|
1002
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
1003
|
|
|
|
|
|
|
# extract information from Photoshop PSD file |
|
1004
|
|
|
|
|
|
|
# Inputs: 0) ExifTool object reference, 1) dirInfo reference |
|
1005
|
|
|
|
|
|
|
# Returns: 1 if this was a valid PSD file, -1 on write error |
|
1006
|
|
|
|
|
|
|
sub ProcessPSD($$) |
|
1007
|
|
|
|
|
|
|
{ |
|
1008
|
5
|
|
|
5
|
0
|
16
|
my ($et, $dirInfo) = @_; |
|
1009
|
5
|
|
|
|
|
13
|
my $raf = $$dirInfo{RAF}; |
|
1010
|
5
|
|
|
|
|
13
|
my $outfile = $$dirInfo{OutFile}; |
|
1011
|
5
|
|
|
|
|
8
|
my ($data, $err, $tagTablePtr); |
|
1012
|
|
|
|
|
|
|
|
|
1013
|
5
|
50
|
|
|
|
16
|
$raf->Read($data, 30) == 30 or return 0; |
|
1014
|
5
|
50
|
|
|
|
33
|
$data =~ /^8BPS\0([\x01\x02])/ or return 0; |
|
1015
|
5
|
|
|
|
|
22
|
SetByteOrder('MM'); |
|
1016
|
5
|
50
|
|
|
|
33
|
$et->SetFileType($1 eq "\x01" ? 'PSD' : 'PSB'); # set the FileType tag |
|
1017
|
5
|
|
|
|
|
26
|
my %dirInfo = ( |
|
1018
|
|
|
|
|
|
|
DataPt => \$data, |
|
1019
|
|
|
|
|
|
|
DirStart => 0, |
|
1020
|
|
|
|
|
|
|
DirName => 'Photoshop', |
|
1021
|
|
|
|
|
|
|
); |
|
1022
|
5
|
|
|
|
|
18
|
my $len = Get32u(\$data, 26); |
|
1023
|
5
|
100
|
|
|
|
16
|
if ($outfile) { |
|
1024
|
1
|
50
|
|
|
|
5
|
Write($outfile, $data) or $err = 1; |
|
1025
|
1
|
50
|
|
|
|
7
|
$raf->Read($data, $len) == $len or return -1; |
|
1026
|
1
|
50
|
|
|
|
10
|
Write($outfile, $data) or $err = 1; # write color mode data |
|
1027
|
|
|
|
|
|
|
# initialize map of where things are written |
|
1028
|
1
|
|
|
|
|
6
|
$et->InitWriteDirs(\%psdMap); |
|
1029
|
|
|
|
|
|
|
} else { |
|
1030
|
|
|
|
|
|
|
# process the header |
|
1031
|
4
|
|
|
|
|
11
|
$tagTablePtr = GetTagTable('Image::ExifTool::Photoshop::Header'); |
|
1032
|
4
|
|
|
|
|
14
|
$dirInfo{DirLen} = 30; |
|
1033
|
4
|
|
|
|
|
17
|
$et->ProcessDirectory(\%dirInfo, $tagTablePtr); |
|
1034
|
4
|
50
|
|
|
|
16
|
$raf->Seek($len, 1) or $err = 1; # skip over color mode data |
|
1035
|
|
|
|
|
|
|
} |
|
1036
|
|
|
|
|
|
|
# read image resource section |
|
1037
|
5
|
50
|
|
|
|
21
|
$raf->Read($data, 4) == 4 or $err = 1; |
|
1038
|
5
|
|
|
|
|
21
|
$len = Get32u(\$data, 0); |
|
1039
|
5
|
50
|
|
|
|
16
|
$raf->Read($data, $len) == $len or $err = 1; |
|
1040
|
5
|
|
|
|
|
18
|
$tagTablePtr = GetTagTable('Image::ExifTool::Photoshop::Main'); |
|
1041
|
5
|
|
|
|
|
14
|
$dirInfo{DirLen} = $len; |
|
1042
|
5
|
|
|
|
|
11
|
my $rtnVal = 1; |
|
1043
|
5
|
100
|
|
|
|
20
|
if ($outfile) { |
|
|
|
50
|
|
|
|
|
|
|
1044
|
|
|
|
|
|
|
# rewrite IRB resources |
|
1045
|
1
|
|
|
|
|
9
|
$data = WritePhotoshop($et, \%dirInfo, $tagTablePtr); |
|
1046
|
1
|
50
|
|
|
|
6
|
if ($data) { |
|
1047
|
1
|
|
|
|
|
4
|
$len = Set32u(length $data); |
|
1048
|
1
|
50
|
|
|
|
5
|
Write($outfile, $len, $data) or $err = 1; |
|
1049
|
|
|
|
|
|
|
# look for trailer and edit if necessary |
|
1050
|
1
|
|
|
|
|
6
|
my $trailInfo = Image::ExifTool::IdentifyTrailer($raf); |
|
1051
|
1
|
50
|
|
|
|
5
|
if ($trailInfo) { |
|
1052
|
1
|
|
|
|
|
2
|
my $tbuf = ''; |
|
1053
|
1
|
|
|
|
|
4
|
$$trailInfo{OutFile} = \$tbuf; # rewrite trailer(s) |
|
1054
|
|
|
|
|
|
|
# rewrite all trailers to buffer |
|
1055
|
1
|
50
|
|
|
|
6
|
if ($et->ProcessTrailers($trailInfo)) { |
|
1056
|
1
|
|
|
|
|
5
|
my $copyBytes = $$trailInfo{DataPos} - $raf->Tell(); |
|
1057
|
1
|
50
|
|
|
|
4
|
if ($copyBytes >= 0) { |
|
1058
|
|
|
|
|
|
|
# copy remaining PSD file up to start of trailer |
|
1059
|
1
|
|
|
|
|
4
|
while ($copyBytes) { |
|
1060
|
1
|
50
|
|
|
|
3
|
my $n = ($copyBytes > 65536) ? 65536 : $copyBytes; |
|
1061
|
1
|
50
|
|
|
|
4
|
$raf->Read($data, $n) == $n or $err = 1; |
|
1062
|
1
|
50
|
|
|
|
5
|
Write($outfile, $data) or $err = 1; |
|
1063
|
1
|
|
|
|
|
4
|
$copyBytes -= $n; |
|
1064
|
|
|
|
|
|
|
} |
|
1065
|
|
|
|
|
|
|
# write the trailer (or not) |
|
1066
|
1
|
50
|
|
|
|
5
|
$et->WriteTrailerBuffer($trailInfo, $outfile) or $err = 1; |
|
1067
|
|
|
|
|
|
|
} else { |
|
1068
|
0
|
|
|
|
|
0
|
$et->Warn('Overlapping trailer'); |
|
1069
|
0
|
|
|
|
|
0
|
undef $trailInfo; |
|
1070
|
|
|
|
|
|
|
} |
|
1071
|
|
|
|
|
|
|
} else { |
|
1072
|
0
|
|
|
|
|
0
|
undef $trailInfo; |
|
1073
|
|
|
|
|
|
|
} |
|
1074
|
|
|
|
|
|
|
} |
|
1075
|
1
|
50
|
|
|
|
8
|
unless ($trailInfo) { |
|
1076
|
|
|
|
|
|
|
# copy over the rest of the file |
|
1077
|
0
|
|
|
|
|
0
|
while ($raf->Read($data, 65536)) { |
|
1078
|
0
|
0
|
|
|
|
0
|
Write($outfile, $data) or $err = 1; |
|
1079
|
|
|
|
|
|
|
} |
|
1080
|
|
|
|
|
|
|
} |
|
1081
|
|
|
|
|
|
|
} else { |
|
1082
|
0
|
|
|
|
|
0
|
$err = 1; |
|
1083
|
|
|
|
|
|
|
} |
|
1084
|
1
|
50
|
|
|
|
4
|
$rtnVal = -1 if $err; |
|
1085
|
|
|
|
|
|
|
} elsif ($err) { |
|
1086
|
0
|
|
|
|
|
0
|
$et->Warn('File format error'); |
|
1087
|
|
|
|
|
|
|
} else { |
|
1088
|
|
|
|
|
|
|
# read IRB resources |
|
1089
|
4
|
|
|
|
|
19
|
ProcessPhotoshop($et, \%dirInfo, $tagTablePtr); |
|
1090
|
|
|
|
|
|
|
# read layer and mask information section |
|
1091
|
4
|
|
|
|
|
10
|
$dirInfo{RAF} = $raf; |
|
1092
|
4
|
|
|
|
|
14
|
$tagTablePtr = GetTagTable('Image::ExifTool::Photoshop::Layers'); |
|
1093
|
4
|
|
|
|
|
14
|
my $oldIndent = $$et{INDENT}; |
|
1094
|
4
|
|
|
|
|
14
|
$$et{INDENT} .= '| '; |
|
1095
|
4
|
50
|
33
|
|
|
20
|
if (ProcessLayersAndMask($et, \%dirInfo, $tagTablePtr) and |
|
1096
|
|
|
|
|
|
|
# read compression mode from image data section |
|
1097
|
|
|
|
|
|
|
$raf->Read($data,2) == 2) |
|
1098
|
|
|
|
|
|
|
{ |
|
1099
|
4
|
|
|
|
|
17
|
my %dirInfo = ( |
|
1100
|
|
|
|
|
|
|
DataPt => \$data, |
|
1101
|
|
|
|
|
|
|
DataPos => $raf->Tell() - 2, |
|
1102
|
|
|
|
|
|
|
); |
|
1103
|
4
|
|
|
|
|
16
|
$tagTablePtr = GetTagTable('Image::ExifTool::Photoshop::ImageData'); |
|
1104
|
4
|
|
|
|
|
15
|
$et->ProcessDirectory(\%dirInfo, $tagTablePtr); |
|
1105
|
|
|
|
|
|
|
} |
|
1106
|
4
|
|
|
|
|
18
|
$$et{INDENT} = $oldIndent; |
|
1107
|
|
|
|
|
|
|
# process trailers if they exist |
|
1108
|
4
|
|
|
|
|
19
|
my $trailInfo = Image::ExifTool::IdentifyTrailer($raf); |
|
1109
|
4
|
50
|
|
|
|
27
|
$et->ProcessTrailers($trailInfo) if $trailInfo; |
|
1110
|
|
|
|
|
|
|
} |
|
1111
|
5
|
|
|
|
|
27
|
return $rtnVal; |
|
1112
|
|
|
|
|
|
|
} |
|
1113
|
|
|
|
|
|
|
|
|
1114
|
|
|
|
|
|
|
1; # end |
|
1115
|
|
|
|
|
|
|
|
|
1116
|
|
|
|
|
|
|
|
|
1117
|
|
|
|
|
|
|
__END__ |