File size: 3,575 Bytes
941c846
77f33ba
2c2e55b
efe5371
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
941c846
2c2e55b
 
77f33ba
2c2e55b
 
 
 
941c846
2c2e55b
 
 
 
941c846
2c2e55b
 
941c846
2c2e55b
 
77f33ba
2c2e55b
 
941c846
2c2e55b
941c846
 
 
2c2e55b
 
 
 
 
 
 
 
 
 
941c846
 
2c2e55b
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
import gradio as gr
from PIL import Image
import os
from datetime import datetime
import struct

def create_xmp_block(width, height):
    """Create XMP metadata block following ExifTool's exact format."""
    xmp = (
        f'<?xpacket begin="" id="W5M0MpCehiHzreSzNTczkc9d"?>\n'
        f'<x:xmpmeta xmlns:x="adobe:ns:meta/" x:xmptk="ExifTool">\n'
        f'<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">\n'
        f'<rdf:Description rdf:about=""\n'
        f'xmlns:GPano="http://ns.google.com/photos/1.0/panorama/"\n'
        f'GPano:ProjectionType="equirectangular"\n'
        f'GPano:UsePanoramaViewer="True"\n'
        f'GPano:FullPanoWidthPixels="{width}"\n'
        f'GPano:FullPanoHeightPixels="{height}"\n'
        f'GPano:CroppedAreaImageWidthPixels="{width}"\n'
        f'GPano:CroppedAreaImageHeightPixels="{height}"\n'
        f'GPano:CroppedAreaLeftPixels="0"\n'
        f'GPano:CroppedAreaTopPixels="0"/>\n'
        f'</rdf:RDF>\n'
        f'</x:xmpmeta>\n'
        f'<?xpacket end="w"?>'
    )
    return xmp

def write_xmp_to_jpg(input_path, output_path, width, height):
    """Write XMP metadata to JPEG file following ExifTool's method."""
    # Read the original JPEG
    with open(input_path, 'rb') as f:
        data = f.read()
    
    # Find the start of image marker
    if data[0:2] != b'\xFF\xD8':
        raise ValueError("Not a valid JPEG file")
    
    # Create XMP data
    xmp_data = create_xmp_block(width, height)
    
    # Create APP1 segment for XMP
    app1_marker = b'\xFF\xE1'
    xmp_header = b'http://ns.adobe.com/xap/1.0/\x00'
    xmp_bytes = xmp_data.encode('utf-8')
    length = len(xmp_header) + len(xmp_bytes) + 2  # +2 for length bytes
    length_bytes = struct.pack('>H', length)
    
    # Construct new file content
    output = bytearray()
    output.extend(data[0:2])  # SOI marker
    output.extend(app1_marker)
    output.extend(length_bytes)
    output.extend(xmp_header)
    output.extend(xmp_bytes)
    output.extend(data[2:])  # Rest of the original file
    
    # Write the new file
    with open(output_path, 'wb') as f:
        f.write(output)

def add_360_metadata(input_image):
    """Add 360 photo metadata to an image file."""
    try:
        # Open and verify the image
        img = Image.open(input_image)
        if img.width != 2 * img.height:
            raise gr.Error("Image must have 2:1 aspect ratio for equirectangular projection")
        
        # Create output filename
        timestamp = datetime.now().strftime("%Y%m%d_%H%M%S")
        output_filename = f"360_photo_{timestamp}.jpg"
        output_path = os.path.join("/tmp", output_filename)
        
        # First save as high-quality JPEG
        img.save(output_path, "JPEG", quality=95)
        
        # Then inject XMP metadata directly into JPEG file
        write_xmp_to_jpg(output_path, output_path, img.width, img.height)
        
        return output_path
    
    except Exception as e:
        raise gr.Error(f"Error processing image: {str(e)}")

# Create Gradio interface
iface = gr.Interface(
    fn=add_360_metadata,
    inputs=gr.Image(type="filepath", label="Upload 360° Photo"),
    outputs=gr.Image(type="filepath", label="360° Photo with Metadata"),
    title="360° Photo Metadata Adder",
    description=(
        "Upload an equirectangular 360° photo to add metadata for Google Photos and other 360° viewers.\n"
        "Important: Image must have 2:1 aspect ratio (width = 2 × height)."
    ),
    examples=[],
    cache_examples=False
)

# Launch the interface
iface.launch()