Targa image file loading and saving for RGB and ARGB uncompressed images
parent
6306d58f77
commit
66aaebe08d
|
@ -0,0 +1 @@
|
|||
Erik Charlebois
|
|
@ -0,0 +1,290 @@
|
|||
! Copyright (C) 2010 Erik Charlebois
|
||||
! See http://factorcode.org/license.txt for BSD license.
|
||||
USING: accessors images images.loader io io.binary kernel
|
||||
locals math sequences io.encodings.ascii io.encodings.string
|
||||
calendar math.ranges math.parser colors arrays hashtables
|
||||
ui.pixel-formats combinators continuations ;
|
||||
IN: images.tga
|
||||
|
||||
SINGLETON: tga-image
|
||||
"tga" tga-image register-image-class
|
||||
|
||||
ERROR: bad-tga-header ;
|
||||
ERROR: bad-tga-footer ;
|
||||
ERROR: bad-tga-extension-size ;
|
||||
ERROR: bad-tga-timestamp ;
|
||||
ERROR: bad-tga-unsupported ;
|
||||
|
||||
: read-id-length ( -- byte )
|
||||
1 read le> ; inline
|
||||
|
||||
: read-color-map-type ( -- byte )
|
||||
1 read le> dup
|
||||
{ 0 1 } member? [ bad-tga-header ] unless ;
|
||||
|
||||
: read-image-type ( -- byte )
|
||||
1 read le> dup
|
||||
{ 0 1 2 3 9 10 11 } member? [ bad-tga-header ] unless ; inline
|
||||
|
||||
: read-color-map-first ( -- short )
|
||||
2 read le> ; inline
|
||||
|
||||
: read-color-map-length ( -- short )
|
||||
2 read le> ; inline
|
||||
|
||||
: read-color-map-entry-size ( -- byte )
|
||||
1 read le> ; inline
|
||||
|
||||
: read-x-origin ( -- short )
|
||||
2 read le> ; inline
|
||||
|
||||
: read-y-origin ( -- short )
|
||||
2 read le> ; inline
|
||||
|
||||
: read-image-width ( -- short )
|
||||
2 read le> ; inline
|
||||
|
||||
: read-image-height ( -- short )
|
||||
2 read le> ; inline
|
||||
|
||||
: read-pixel-depth ( -- byte )
|
||||
1 read le> ; inline
|
||||
|
||||
: read-image-descriptor ( -- alpha-bits pixel-order )
|
||||
1 read le>
|
||||
[ 7 bitand ] [ 24 bitand -3 shift ] bi ; inline
|
||||
|
||||
: read-image-id ( length -- image-id )
|
||||
read ; inline
|
||||
|
||||
: read-color-map ( type length elt-size -- color-map )
|
||||
pick 1 = [ 8 align 8 / * read ] [ 2drop f ] if swap drop ; inline
|
||||
|
||||
: read-image-data ( width height depth -- image-data )
|
||||
8 align 8 / * * read ; inline
|
||||
|
||||
: read-extension-area-offset ( -- offset )
|
||||
4 read le> ; inline
|
||||
|
||||
: read-developer-directory-offset ( -- offset )
|
||||
4 read le> ; inline
|
||||
|
||||
: read-signature ( -- )
|
||||
18 read ascii decode "TRUEVISION-XFILE.\0" = [ bad-tga-footer ] unless ; inline
|
||||
|
||||
: read-extension-size ( -- )
|
||||
2 read le> 495 = [ bad-tga-extension-size ] unless ; inline
|
||||
|
||||
: read-author-name ( -- string )
|
||||
41 read ascii decode [ 0 = ] trim ; inline
|
||||
|
||||
: read-author-comments ( -- string )
|
||||
4 iota [ drop 81 read ascii decode [ 0 = ] trim ] map concat ; inline
|
||||
|
||||
: read-date-timestamp ( -- timestamp )
|
||||
timestamp new
|
||||
2 read le> dup 12 [1,b] member? [ bad-tga-timestamp ] unless >>month
|
||||
2 read le> dup 31 [1,b] member? [ bad-tga-timestamp ] unless >>day
|
||||
2 read le> >>year
|
||||
2 read le> dup 23 [0,b] member? [ bad-tga-timestamp ] unless >>hour
|
||||
2 read le> dup 59 [0,b] member? [ bad-tga-timestamp ] unless >>minute
|
||||
2 read le> dup 59 [0,b] member? [ bad-tga-timestamp ] unless >>second ; inline
|
||||
|
||||
: read-job-name ( -- string )
|
||||
41 read ascii decode [ 0 = ] trim ; inline
|
||||
|
||||
: read-job-time ( -- duration )
|
||||
duration new
|
||||
2 read le> >>hour
|
||||
2 read le> dup 59 [0,b] member? [ bad-tga-timestamp ] unless >>minute
|
||||
2 read le> dup 59 [0,b] member? [ bad-tga-timestamp ] unless >>second ; inline
|
||||
|
||||
: read-software-id ( -- string )
|
||||
41 read ascii decode [ 0 = ] trim ; inline
|
||||
|
||||
: read-software-version ( -- string )
|
||||
2 read le> 100 /f number>string
|
||||
1 read ascii decode append [ " " = ] trim ; inline
|
||||
|
||||
:: read-key-color ( -- color )
|
||||
1 read le> 255 /f :> alpha
|
||||
1 read le> 255 /f
|
||||
1 read le> 255 /f
|
||||
1 read le> 255 /f
|
||||
alpha <rgba> ; inline
|
||||
|
||||
: read-pixel-aspect-ratio ( -- aspect-ratio )
|
||||
2 read le> 2 read le> /f ; inline
|
||||
|
||||
: read-gamma-value ( -- gamma-value )
|
||||
2 read le> 2 read le> /f ; inline
|
||||
|
||||
: read-color-correction-offset ( -- offset )
|
||||
4 read le> ; inline
|
||||
|
||||
: read-postage-stamp-offset ( -- offset )
|
||||
4 read le> ; inline
|
||||
|
||||
: read-scan-line-offset ( -- offset )
|
||||
4 read le> ; inline
|
||||
|
||||
: read-premultiplied-alpha ( -- boolean )
|
||||
1 read le> 4 = ; inline
|
||||
|
||||
: read-scan-line-table ( height -- scan-offsets )
|
||||
iota [ drop 4 read le> ] map ; inline
|
||||
|
||||
: read-postage-stamp-image ( depth -- postage-data )
|
||||
8 align 8 / 1 read le> 1 read le> * * read ; inline
|
||||
|
||||
:: read-color-correction-table ( -- correction-table )
|
||||
256 iota
|
||||
[
|
||||
drop
|
||||
4 iota
|
||||
[
|
||||
drop
|
||||
2 read le> 65535 /f :> alpha
|
||||
2 read le> 65535 /f
|
||||
2 read le> 65535 /f
|
||||
2 read le> 65535 /f
|
||||
alpha <rgba>
|
||||
] map
|
||||
] map ; inline
|
||||
|
||||
: read-developer-directory ( -- developer-directory )
|
||||
2 read le> iota
|
||||
[
|
||||
drop
|
||||
2 read le>
|
||||
4 read le>
|
||||
4 read le>
|
||||
3array
|
||||
] map ; inline
|
||||
|
||||
: read-developer-areas ( developer-directory -- developer-area-map )
|
||||
[
|
||||
[ first ]
|
||||
[ dup third second seek-absolute seek-input read ] bi 2array
|
||||
] map >hashtable ; inline
|
||||
|
||||
:: read-tga ( -- image )
|
||||
#! Read header
|
||||
read-id-length :> id-length
|
||||
read-color-map-type :> map-type
|
||||
read-image-type :> image-type
|
||||
read-color-map-first :> map-first
|
||||
read-color-map-length :> map-length
|
||||
read-color-map-entry-size :> map-entry-size
|
||||
read-x-origin :> x-origin
|
||||
read-y-origin :> y-origin
|
||||
read-image-width :> image-width
|
||||
read-image-height :> image-height
|
||||
read-pixel-depth :> pixel-depth
|
||||
read-image-descriptor :> ( alpha-bits pixel-order )
|
||||
id-length read-image-id :> image-id
|
||||
map-type map-length map-entry-size read-color-map :> color-map-data
|
||||
image-width image-height pixel-depth read-image-data :> image-data
|
||||
|
||||
[
|
||||
#! Read optional footer
|
||||
26 seek-end seek-input
|
||||
read-extension-area-offset :> extension-offset
|
||||
read-developer-directory-offset :> directory-offset
|
||||
read-signature
|
||||
|
||||
#! Read optional extension section
|
||||
extension-offset 0 =
|
||||
[
|
||||
extension-offset seek-absolute seek-input
|
||||
read-extension-size
|
||||
read-author-name :> author-name
|
||||
read-author-comments :> author-comments
|
||||
read-date-timestamp :> date-timestamp
|
||||
read-job-name :> job-name
|
||||
read-job-time :> job-time
|
||||
read-software-id :> software-id
|
||||
read-software-version :> software-version
|
||||
read-key-color :> key-color
|
||||
read-pixel-aspect-ratio :> aspect-ratio
|
||||
read-gamma-value :> gamma-value
|
||||
read-color-correction-offset :> color-correction-offset
|
||||
read-postage-stamp-offset :> postage-stamp-offset
|
||||
read-scan-line-offset :> scan-line-offset
|
||||
read-premultiplied-alpha :> premultiplied-alpha
|
||||
|
||||
color-correction-offset 0 =
|
||||
[
|
||||
color-correction-offset seek-absolute seek-input
|
||||
read-color-correction-table :> color-correction-table
|
||||
] unless
|
||||
|
||||
postage-stamp-offset 0 =
|
||||
[
|
||||
postage-stamp-offset seek-absolute seek-input
|
||||
pixel-depth read-postage-stamp-image :> postage-data
|
||||
] unless
|
||||
|
||||
scan-line-offset seek-absolute seek-input
|
||||
image-height read-scan-line-table :> scan-offsets
|
||||
|
||||
#! Read optional developer section
|
||||
directory-offset 0 =
|
||||
[ f ]
|
||||
[
|
||||
directory-offset seek-absolute seek-input
|
||||
read-developer-directory read-developer-areas
|
||||
] if :> developer-areas
|
||||
] unless
|
||||
] ignore-errors
|
||||
|
||||
#! Only 24-bit uncompressed RGB and 32-bit uncompressed ARGB are supported.
|
||||
#! Other formats would need to be converted to work within the image class.
|
||||
map-type 0 = [ bad-tga-unsupported ] unless
|
||||
image-type 2 = [ bad-tga-unsupported ] unless
|
||||
pixel-depth { 24 32 } member? [ bad-tga-unsupported ] unless
|
||||
pixel-order { 0 2 } member? [ bad-tga-unsupported ] unless
|
||||
|
||||
#! Create image instance
|
||||
image new
|
||||
alpha-bits 0 = [ RGB ] [ ARGB ] if >>component-order
|
||||
{ image-width image-height } >>dim
|
||||
pixel-order 0 = >>upside-down?
|
||||
image-data >>bitmap
|
||||
ubyte-components >>component-type ;
|
||||
|
||||
M: tga-image stream>image
|
||||
drop [ read-tga ] with-input-stream ;
|
||||
|
||||
M: tga-image image>stream
|
||||
drop
|
||||
[
|
||||
component-order>> { RGB ARGB } member? [ bad-tga-unsupported ] unless
|
||||
] keep
|
||||
|
||||
B{ 0 } write #! id-length
|
||||
B{ 0 } write #! map-type
|
||||
B{ 2 } write #! image-type
|
||||
B{ 0 0 0 0 0 } write #! color map first, length, entry size
|
||||
B{ 0 0 0 0 } write #! x-origin, y-origin
|
||||
{
|
||||
[ dim>> first 2 >le write ]
|
||||
[ dim>> second 2 >le write ]
|
||||
[ component-order>>
|
||||
{
|
||||
{ RGB [ B{ 24 } write ] }
|
||||
{ ARGB [ B{ 32 } write ] }
|
||||
} case
|
||||
]
|
||||
[
|
||||
dup component-order>>
|
||||
{
|
||||
{ RGB [ 0 ] }
|
||||
{ ARGB [ 8 ] }
|
||||
} case swap
|
||||
upside-down?>> [ 0 ] [ 2 ] if 3 shift bitor
|
||||
1 >le write
|
||||
]
|
||||
[ bitmap>> write ]
|
||||
} cleave ;
|
||||
|
Loading…
Reference in New Issue