not quite there
trying to set EXIF_TAG_USER_COMMENT but not working
This commit is contained in:
parent
1cc7dd6af7
commit
c12cd6d2cb
@ -706,6 +706,69 @@ vips_exif_set_double( ExifData *ed,
|
|||||||
typedef void (*write_fn)( ExifData *ed,
|
typedef void (*write_fn)( ExifData *ed,
|
||||||
ExifEntry *entry, unsigned long component, void *data );
|
ExifEntry *entry, unsigned long component, void *data );
|
||||||
|
|
||||||
|
/* special header required for EXIF_TAG_USER_COMMENT.
|
||||||
|
*/
|
||||||
|
#define ASCII_COMMENT "ASCII\0\0\0"
|
||||||
|
|
||||||
|
/* data is a null-terminated utf-8 string.
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
vips_exif_set_string( ExifData *ed,
|
||||||
|
ExifEntry *entry, unsigned long component, const char *data )
|
||||||
|
{
|
||||||
|
ExifMem *mem = exif_mem_new_default();
|
||||||
|
|
||||||
|
void *buf;
|
||||||
|
int len;
|
||||||
|
const char *p;
|
||||||
|
char *q;
|
||||||
|
|
||||||
|
/* A copy of data which we may have to free.
|
||||||
|
*/
|
||||||
|
q = NULL;
|
||||||
|
|
||||||
|
/* The final " (xx, ASCII, yy, zz)" part of the string (if present) was
|
||||||
|
* added by us in _to_s(), we must remove it before setting the string
|
||||||
|
* back again.
|
||||||
|
*
|
||||||
|
* It may not be there if the user has changed the string.
|
||||||
|
*
|
||||||
|
* Leave p pointing to the trimmed string.
|
||||||
|
*/
|
||||||
|
len = strlen( data );
|
||||||
|
if( len > 0 &&
|
||||||
|
data[len - 1] == ')' &&
|
||||||
|
(p = strrchr( data, '(' )) &&
|
||||||
|
p - data > 0 &&
|
||||||
|
p[-1] == ' ' ) {
|
||||||
|
q = g_strdup( data );
|
||||||
|
q[p - data - 1] = '\0';
|
||||||
|
p = q;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
p = data;
|
||||||
|
len = strlen( p );
|
||||||
|
|
||||||
|
/* The string in the entry must be allocated with the same allocator
|
||||||
|
* that was used to allocate the entry itself. We assume the entry was
|
||||||
|
* created with the default one.
|
||||||
|
*
|
||||||
|
* We have to write the ASCII charset tag and not write a
|
||||||
|
* null-terminator.
|
||||||
|
*/
|
||||||
|
buf = exif_mem_alloc( mem, len + sizeof( ASCII_COMMENT ) - 1 );
|
||||||
|
entry->data = buf;
|
||||||
|
entry->size = len;
|
||||||
|
entry->components = len;
|
||||||
|
entry->format = EXIF_FORMAT_ASCII;
|
||||||
|
|
||||||
|
memcpy( entry->data, ASCII_COMMENT, sizeof( ASCII_COMMENT ) - 1 );
|
||||||
|
memcpy( entry->data + sizeof(ASCII_COMMENT) - 1, p, len - 1 );
|
||||||
|
|
||||||
|
VIPS_FREEF( g_free, q );
|
||||||
|
VIPS_FREEF( exif_mem_unref, mem );
|
||||||
|
}
|
||||||
|
|
||||||
/* Write a tag. Update what's there, or make a new one.
|
/* Write a tag. Update what's there, or make a new one.
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
@ -722,13 +785,20 @@ vips_exif_set_tag( ExifData *ed, int ifd, ExifTag tag, write_fn fn, void *data )
|
|||||||
/* tag must be set before calling exif_content_add_entry.
|
/* tag must be set before calling exif_content_add_entry.
|
||||||
*/
|
*/
|
||||||
entry->tag = tag;
|
entry->tag = tag;
|
||||||
|
|
||||||
exif_content_add_entry( ed->ifd[ifd], entry );
|
exif_content_add_entry( ed->ifd[ifd], entry );
|
||||||
exif_entry_initialize( entry, tag );
|
|
||||||
exif_entry_unref( entry );
|
exif_entry_unref( entry );
|
||||||
|
|
||||||
|
/* libexif makes us have a special path for string-valued
|
||||||
|
* fields :(
|
||||||
|
*/
|
||||||
|
if( tag == EXIF_TAG_USER_COMMENT ) {
|
||||||
|
vips_exif_set_string( ed, entry, 0, data );
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
exif_entry_initialize( entry, tag );
|
||||||
fn( ed, entry, 0, data );
|
fn( ed, entry, 0, data );
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set the EXIF resolution from the vips xres/yres tags.
|
/* Set the EXIF resolution from the vips xres/yres tags.
|
||||||
@ -869,13 +939,38 @@ vips_exif_set_thumbnail( ExifData *ed, VipsImage *im )
|
|||||||
return( 0 );
|
return( 0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* See also vips_exif_to_s() ... keep in sync.
|
/* Skip any spaces.
|
||||||
|
*/
|
||||||
|
static const char *
|
||||||
|
skip_space( const char *p )
|
||||||
|
{
|
||||||
|
while( p && *p == ' ' )
|
||||||
|
p += 1;
|
||||||
|
|
||||||
|
return( p );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Skip to the end of this non-space sequence.
|
||||||
|
*/
|
||||||
|
static const char *
|
||||||
|
skip_nonspace( const char *p )
|
||||||
|
{
|
||||||
|
while( p && *p && *p != ' ' )
|
||||||
|
p += 1;
|
||||||
|
|
||||||
|
return( p );
|
||||||
|
}
|
||||||
|
|
||||||
|
/* See also vips_exif_to_s() ... keep in sync. Only the numeric types are
|
||||||
|
* handled here, since they can be updated. For string types, we have to
|
||||||
|
* destroy and recreate, see above.
|
||||||
*/
|
*/
|
||||||
static void
|
static void
|
||||||
vips_exif_from_s( ExifData *ed, ExifEntry *entry, const char *value )
|
vips_exif_from_s( ExifData *ed, ExifEntry *entry, const char *value )
|
||||||
{
|
{
|
||||||
unsigned long i;
|
unsigned long i;
|
||||||
const char *p;
|
const char *p;
|
||||||
|
int v;
|
||||||
|
|
||||||
if( entry->format == EXIF_FORMAT_SHORT ||
|
if( entry->format == EXIF_FORMAT_SHORT ||
|
||||||
entry->format == EXIF_FORMAT_SSHORT ||
|
entry->format == EXIF_FORMAT_SSHORT ||
|
||||||
@ -884,26 +979,15 @@ vips_exif_from_s( ExifData *ed, ExifEntry *entry, const char *value )
|
|||||||
if( entry->components >= 10 )
|
if( entry->components >= 10 )
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* Skip any leading spaces.
|
|
||||||
*/
|
|
||||||
p = value;
|
p = value;
|
||||||
while( *p == ' ' )
|
|
||||||
p += 1;
|
|
||||||
|
|
||||||
for( i = 0; i < entry->components; i++ ) {
|
for( i = 0; i < entry->components; i++ ) {
|
||||||
int value = atof( p );
|
if( !(p = skip_space( p )) )
|
||||||
|
|
||||||
vips_exif_set_int( ed, entry, i, &value );
|
|
||||||
|
|
||||||
/* Skip to the next set of spaces, then to the
|
|
||||||
* beginning of the next item.
|
|
||||||
*/
|
|
||||||
while( *p && *p != ' ' )
|
|
||||||
p += 1;
|
|
||||||
while( *p == ' ' )
|
|
||||||
p += 1;
|
|
||||||
if( !*p )
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
v = atof( p );
|
||||||
|
vips_exif_set_int( ed, entry, i, &v );
|
||||||
|
|
||||||
|
p = skip_nonspace( p );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if( entry->format == EXIF_FORMAT_RATIONAL ||
|
else if( entry->format == EXIF_FORMAT_RATIONAL ||
|
||||||
@ -911,57 +995,16 @@ vips_exif_from_s( ExifData *ed, ExifEntry *entry, const char *value )
|
|||||||
if( entry->components >= 10 )
|
if( entry->components >= 10 )
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* Skip any leading spaces.
|
|
||||||
*/
|
|
||||||
p = value;
|
p = value;
|
||||||
while( *p == ' ' )
|
|
||||||
p += 1;
|
|
||||||
|
|
||||||
for( i = 0; i < entry->components; i++ ) {
|
for( i = 0; i < entry->components; i++ ) {
|
||||||
|
if( !(p = skip_space( p )) )
|
||||||
|
break;
|
||||||
|
|
||||||
vips_exif_set_rational( ed, entry, i, (void *) p );
|
vips_exif_set_rational( ed, entry, i, (void *) p );
|
||||||
|
|
||||||
/* Skip to the next set of spaces, then to the
|
p = skip_nonspace( p );
|
||||||
* beginning of the next item.
|
|
||||||
*/
|
|
||||||
while( *p && *p != ' ' )
|
|
||||||
p += 1;
|
|
||||||
while( *p == ' ' )
|
|
||||||
p += 1;
|
|
||||||
if( !*p )
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if( entry->format == EXIF_FORMAT_ASCII ) {
|
|
||||||
int len;
|
|
||||||
char *q;
|
|
||||||
|
|
||||||
/* A copy of value which we may have to free.
|
|
||||||
*/
|
|
||||||
q = NULL;
|
|
||||||
|
|
||||||
/* The final " (xx, ASCII, yy, zz)" part of the string was
|
|
||||||
* added by us in _to_s(), we must remove it before setting
|
|
||||||
* the string back again.
|
|
||||||
*
|
|
||||||
* It may not be there if the user has changed the string.
|
|
||||||
*
|
|
||||||
* Leave p as poointing to the trimmed string.
|
|
||||||
*/
|
|
||||||
len = strlen( value );
|
|
||||||
if( len > 0 &&
|
|
||||||
value[len - 1] == ')' &&
|
|
||||||
(p = strrchr( value, '(' )) &&
|
|
||||||
p - value > 0 &&
|
|
||||||
p[-1] == ' ' ) {
|
|
||||||
q = g_strdup( value );
|
|
||||||
q[p - value - 1] = '\0';
|
|
||||||
p = q;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
p = value;
|
|
||||||
|
|
||||||
for( p = value + strlen( value ) - 1; p >= value; p-- )
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1048,6 +1091,12 @@ vips_exif_exif_entry( ExifEntry *entry, VipsExifRemove *ve )
|
|||||||
if( strcmp( tag_name, "Orientation" ) == 0 &&
|
if( strcmp( tag_name, "Orientation" ) == 0 &&
|
||||||
vips_image_get_typeof( ve->image, VIPS_META_ORIENTATION ) )
|
vips_image_get_typeof( ve->image, VIPS_META_ORIENTATION ) )
|
||||||
ve->to_remove = g_slist_prepend( ve->to_remove, entry );
|
ve->to_remove = g_slist_prepend( ve->to_remove, entry );
|
||||||
|
|
||||||
|
/* If this is a string tag, we must also remove it ready for
|
||||||
|
* recreation, see the comment below.
|
||||||
|
*/
|
||||||
|
if( entry->format == EXIF_FORMAT_ASCII )
|
||||||
|
ve->to_remove = g_slist_prepend( ve->to_remove, entry );
|
||||||
}
|
}
|
||||||
|
|
||||||
static void *
|
static void *
|
||||||
@ -1091,14 +1140,17 @@ vips_exif_update( ExifData *ed, VipsImage *image )
|
|||||||
|
|
||||||
VIPS_DEBUG_MSG( "vips_exif_update: \n" );
|
VIPS_DEBUG_MSG( "vips_exif_update: \n" );
|
||||||
|
|
||||||
/* Walk the image and add any exif- that's set in image metadata.
|
|
||||||
*/
|
|
||||||
vips_image_map( image, vips_exif_image_field, ed );
|
|
||||||
|
|
||||||
/* If this exif came from the image (rather than being an exif block we
|
/* If this exif came from the image (rather than being an exif block we
|
||||||
* have made afresh), then any fields which are in the block but not on
|
* have made afresh), then any fields which are in the block but not on
|
||||||
* the image must have been deliberately removed. Remove them from the
|
* the image must have been deliberately removed. Remove them from the
|
||||||
* block as well.
|
* block as well.
|
||||||
|
*
|
||||||
|
* If there are any string-valued fields (eg. comment etc.) which
|
||||||
|
* exist as libvips metadata tags, we must also remove those from the
|
||||||
|
* exif block.
|
||||||
|
*
|
||||||
|
* libexif does not allow you to change string lengths, you must make
|
||||||
|
* new tags, so we have to remove ready to re-add.
|
||||||
*/
|
*/
|
||||||
if( vips_image_get_typeof( image, VIPS_META_EXIF_NAME ) ) {
|
if( vips_image_get_typeof( image, VIPS_META_EXIF_NAME ) ) {
|
||||||
ve.image = image;
|
ve.image = image;
|
||||||
@ -1107,6 +1159,10 @@ vips_exif_update( ExifData *ed, VipsImage *image )
|
|||||||
(ExifDataForeachContentFunc) vips_exif_exif_content,
|
(ExifDataForeachContentFunc) vips_exif_exif_content,
|
||||||
&ve );
|
&ve );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Walk the image and add any exif- that's set in image metadata.
|
||||||
|
*/
|
||||||
|
vips_image_map( image, vips_exif_image_field, ed );
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Examine the metadata tags on the image and update the EXIF block.
|
/* Examine the metadata tags on the image and update the EXIF block.
|
||||||
|
Loading…
Reference in New Issue
Block a user