Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
fd1d5c4f31
|
@@ -5,7 +5,7 @@ import os
|
|||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
from fastapi import FastAPI, Request
|
from fastapi import FastAPI, Request
|
||||||
from fastapi.responses import HTMLResponse, JSONResponse
|
from fastapi.responses import HTMLResponse, JSONResponse, FileResponse
|
||||||
from fastapi.staticfiles import StaticFiles
|
from fastapi.staticfiles import StaticFiles
|
||||||
from fastapi.templating import Jinja2Templates
|
from fastapi.templating import Jinja2Templates
|
||||||
|
|
||||||
@@ -49,6 +49,44 @@ static_dir.mkdir(exist_ok=True)
|
|||||||
app.mount("/static", StaticFiles(directory=str(static_dir)), name="static")
|
app.mount("/static", StaticFiles(directory=str(static_dir)), name="static")
|
||||||
|
|
||||||
|
|
||||||
|
@app.get("/apple-touch-icon.png")
|
||||||
|
async def apple_touch_icon():
|
||||||
|
"""Serve Apple Touch Icon with proper headers."""
|
||||||
|
icon_path = static_dir / "apple-touch-icon.png"
|
||||||
|
return FileResponse(
|
||||||
|
path=icon_path,
|
||||||
|
media_type="image/png",
|
||||||
|
headers={
|
||||||
|
"Cache-Control": "public, max-age=31536000",
|
||||||
|
"Content-Type": "image/png"
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@app.get("/favicon.ico")
|
||||||
|
async def favicon():
|
||||||
|
"""Serve favicon."""
|
||||||
|
icon_path = static_dir / "apple-touch-icon.png"
|
||||||
|
return FileResponse(
|
||||||
|
path=icon_path,
|
||||||
|
media_type="image/png"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@app.get("/manifest.json")
|
||||||
|
async def manifest():
|
||||||
|
"""Serve Web App Manifest with proper headers."""
|
||||||
|
manifest_path = static_dir / "manifest.json"
|
||||||
|
return FileResponse(
|
||||||
|
path=manifest_path,
|
||||||
|
media_type="application/manifest+json",
|
||||||
|
headers={
|
||||||
|
"Cache-Control": "public, max-age=86400",
|
||||||
|
"Content-Type": "application/manifest+json"
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@app.get("/health")
|
@app.get("/health")
|
||||||
async def health() -> JSONResponse:
|
async def health() -> JSONResponse:
|
||||||
"""Health check endpoint for Kubernetes/Docker.
|
"""Health check endpoint for Kubernetes/Docker.
|
||||||
|
|||||||
BIN
apps/ui/static/apple-touch-icon-114x114.png
Normal file
|
After Width: | Height: | Size: 5.5 KiB |
BIN
apps/ui/static/apple-touch-icon-120x120.png
Normal file
|
After Width: | Height: | Size: 5.9 KiB |
BIN
apps/ui/static/apple-touch-icon-144x144.png
Normal file
|
After Width: | Height: | Size: 8.0 KiB |
BIN
apps/ui/static/apple-touch-icon-152x152.png
Normal file
|
After Width: | Height: | Size: 8.7 KiB |
BIN
apps/ui/static/apple-touch-icon-16x16.png
Normal file
|
After Width: | Height: | Size: 449 B |
BIN
apps/ui/static/apple-touch-icon-180x180.png
Normal file
|
After Width: | Height: | Size: 11 KiB |
BIN
apps/ui/static/apple-touch-icon-32x32.png
Normal file
|
After Width: | Height: | Size: 933 B |
BIN
apps/ui/static/apple-touch-icon-57x57.png
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
apps/ui/static/apple-touch-icon-60x60.png
Normal file
|
After Width: | Height: | Size: 2.0 KiB |
BIN
apps/ui/static/apple-touch-icon-72x72.png
Normal file
|
After Width: | Height: | Size: 2.7 KiB |
BIN
apps/ui/static/apple-touch-icon-76x76.png
Normal file
|
After Width: | Height: | Size: 2.9 KiB |
BIN
apps/ui/static/garage-icon-114x114.png
Normal file
|
After Width: | Height: | Size: 4.8 KiB |
BIN
apps/ui/static/garage-icon-120x120.png
Normal file
|
After Width: | Height: | Size: 5.2 KiB |
BIN
apps/ui/static/garage-icon-144x144.png
Normal file
|
After Width: | Height: | Size: 7.0 KiB |
BIN
apps/ui/static/garage-icon-152x152.png
Normal file
|
After Width: | Height: | Size: 7.6 KiB |
BIN
apps/ui/static/garage-icon-16x16.png
Normal file
|
After Width: | Height: | Size: 447 B |
BIN
apps/ui/static/garage-icon-180x180.png
Normal file
|
After Width: | Height: | Size: 8.9 KiB |
BIN
apps/ui/static/garage-icon-32x32.png
Normal file
|
After Width: | Height: | Size: 897 B |
BIN
apps/ui/static/garage-icon-57x57.png
Normal file
|
After Width: | Height: | Size: 1.8 KiB |
BIN
apps/ui/static/garage-icon-60x60.png
Normal file
|
After Width: | Height: | Size: 1.9 KiB |
BIN
apps/ui/static/garage-icon-72x72.png
Normal file
|
After Width: | Height: | Size: 2.5 KiB |
BIN
apps/ui/static/garage-icon-76x76.png
Normal file
|
After Width: | Height: | Size: 2.6 KiB |
43
apps/ui/static/manifest.json
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
{
|
||||||
|
"name": "Home Automation",
|
||||||
|
"short_name": "Home",
|
||||||
|
"description": "Smart Home Automation System",
|
||||||
|
"start_url": "/",
|
||||||
|
"display": "standalone",
|
||||||
|
"background_color": "#667eea",
|
||||||
|
"theme_color": "#667eea",
|
||||||
|
"orientation": "portrait",
|
||||||
|
"icons": [
|
||||||
|
{
|
||||||
|
"src": "/static/apple-touch-icon-180x180.png",
|
||||||
|
"sizes": "180x180",
|
||||||
|
"type": "image/png",
|
||||||
|
"purpose": "any maskable"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "/static/apple-touch-icon-152x152.png",
|
||||||
|
"sizes": "152x152",
|
||||||
|
"type": "image/png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "/static/apple-touch-icon-120x120.png",
|
||||||
|
"sizes": "120x120",
|
||||||
|
"type": "image/png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "/static/apple-touch-icon-76x76.png",
|
||||||
|
"sizes": "76x76",
|
||||||
|
"type": "image/png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "/static/apple-touch-icon-32x32.png",
|
||||||
|
"sizes": "32x32",
|
||||||
|
"type": "image/png"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"src": "/static/apple-touch-icon-16x16.png",
|
||||||
|
"sizes": "16x16",
|
||||||
|
"type": "image/png"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -4,7 +4,19 @@
|
|||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>Home Automation</title>
|
<title>Home Automation</title>
|
||||||
|
|
||||||
|
<!-- Apple Touch Icon -->
|
||||||
|
<link rel="apple-touch-icon" sizes="180x180" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="152x152" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/static/apple-touch-icon.png">
|
||||||
<link rel="icon" type="image/svg+xml" href="/static/favicon.svg">
|
<link rel="icon" type="image/svg+xml" href="/static/favicon.svg">
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
|
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
||||||
|
<meta name="apple-mobile-web-app-title" content="Dashboard">
|
||||||
|
<meta name="theme-color" content="#667eea">
|
||||||
|
<link rel="manifest" href="/static/manifest.json">
|
||||||
<style>
|
<style>
|
||||||
* {
|
* {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|||||||
@@ -4,6 +4,18 @@
|
|||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>Gerät - Home Automation</title>
|
<title>Gerät - Home Automation</title>
|
||||||
|
|
||||||
|
<!-- Apple Touch Icon -->
|
||||||
|
<link rel="apple-touch-icon" sizes="180x180" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="152x152" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/static/apple-touch-icon.png">
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
|
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
||||||
|
<meta name="apple-mobile-web-app-title" content="Gerät">
|
||||||
|
<meta name="theme-color" content="#667eea">
|
||||||
|
<link rel="manifest" href="/static/manifest.json">
|
||||||
<style>
|
<style>
|
||||||
* {
|
* {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|||||||
@@ -6,11 +6,17 @@
|
|||||||
<title>Garage - Home Automation</title>
|
<title>Garage - Home Automation</title>
|
||||||
|
|
||||||
<!-- Apple Touch Icon -->
|
<!-- Apple Touch Icon -->
|
||||||
<link rel="apple-touch-icon" sizes="180x180" href="/static/garage-icon.png">
|
<link rel="apple-touch-icon" sizes="180x180" href="/static/garage-icon-180x180.png">
|
||||||
<link rel="icon" type="image/png" sizes="32x32" href="/static/garage-icon.png">
|
<link rel="apple-touch-icon" sizes="152x152" href="/static/garage-icon-152x152.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/static/garage-icon-120x120.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="76x76" href="/static/garage-icon-76x76.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/static/garage-icon-32x32.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/static/garage-icon-16x16.png">
|
||||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
||||||
<meta name="apple-mobile-web-app-title" content="Garage">
|
<meta name="apple-mobile-web-app-title" content="Garage">
|
||||||
|
<meta name="theme-color" content="#667eea">
|
||||||
|
<link rel="manifest" href="/static/manifest.json">
|
||||||
<style>
|
<style>
|
||||||
* {
|
* {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|||||||
@@ -4,6 +4,19 @@
|
|||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>Home Automation</title>
|
<title>Home Automation</title>
|
||||||
|
|
||||||
|
<!-- Apple Touch Icon -->
|
||||||
|
<link rel="apple-touch-icon" sizes="180x180" href="/static/apple-touch-icon-180x180.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="152x152" href="/static/apple-touch-icon-152x152.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/static/apple-touch-icon-120x120.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="76x76" href="/static/apple-touch-icon-76x76.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/static/apple-touch-icon-32x32.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/static/apple-touch-icon-16x16.png">
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
|
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
||||||
|
<meta name="apple-mobile-web-app-title" content="Home Automation">
|
||||||
|
<meta name="theme-color" content="#667eea">
|
||||||
|
<link rel="manifest" href="/static/manifest.json">
|
||||||
<style>
|
<style>
|
||||||
* {
|
* {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|||||||
@@ -4,6 +4,18 @@
|
|||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>{{ room_name }} - Home Automation</title>
|
<title>{{ room_name }} - Home Automation</title>
|
||||||
|
|
||||||
|
<!-- Apple Touch Icon -->
|
||||||
|
<link rel="apple-touch-icon" sizes="180x180" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="152x152" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/static/apple-touch-icon.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/static/apple-touch-icon.png">
|
||||||
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
|
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
||||||
|
<meta name="apple-mobile-web-app-title" content="{{ room_name }}">
|
||||||
|
<meta name="theme-color" content="#667eea">
|
||||||
|
<link rel="manifest" href="/static/manifest.json">
|
||||||
<style>
|
<style>
|
||||||
* {
|
* {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|||||||
@@ -6,11 +6,17 @@
|
|||||||
<title>Räume - Home Automation</title>
|
<title>Räume - Home Automation</title>
|
||||||
|
|
||||||
<!-- Apple Touch Icon -->
|
<!-- Apple Touch Icon -->
|
||||||
<link rel="apple-touch-icon" sizes="180x180" href="/static/apple-touch-icon.png">
|
<link rel="apple-touch-icon" sizes="180x180" href="/static/apple-touch-icon-180x180.png">
|
||||||
<link rel="icon" type="image/png" sizes="32x32" href="/static/apple-touch-icon.png">
|
<link rel="apple-touch-icon" sizes="152x152" href="/static/apple-touch-icon-152x152.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="120x120" href="/static/apple-touch-icon-120x120.png">
|
||||||
|
<link rel="apple-touch-icon" sizes="76x76" href="/static/apple-touch-icon-76x76.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="32x32" href="/static/apple-touch-icon-32x32.png">
|
||||||
|
<link rel="icon" type="image/png" sizes="16x16" href="/static/apple-touch-icon-16x16.png">
|
||||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
<meta name="apple-mobile-web-app-status-bar-style" content="default">
|
||||||
<meta name="apple-mobile-web-app-title" content="Home Automation">
|
<meta name="apple-mobile-web-app-title" content="Räume">
|
||||||
|
<meta name="theme-color" content="#667eea">
|
||||||
|
<link rel="manifest" href="/static/manifest.json">
|
||||||
<style>
|
<style>
|
||||||
* {
|
* {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|||||||
30
create_icons.py
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
"""
|
||||||
|
Script to create additional PNG icon sizes for better iOS compatibility
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
from pathlib import Path
|
||||||
|
from PIL import Image
|
||||||
|
|
||||||
|
def create_icon_sizes():
|
||||||
|
static_dir = Path("/Users/wn/Workspace/home-automation/apps/ui/static")
|
||||||
|
|
||||||
|
# Sizes that iOS might need
|
||||||
|
sizes = [16, 32, 57, 60, 72, 76, 114, 120, 144, 152, 180]
|
||||||
|
|
||||||
|
# Create home icons
|
||||||
|
base_icon = Image.open(static_dir / "apple-touch-icon.png")
|
||||||
|
for size in sizes:
|
||||||
|
resized = base_icon.resize((size, size), Image.Resampling.LANCZOS)
|
||||||
|
resized.save(static_dir / f"apple-touch-icon-{size}x{size}.png")
|
||||||
|
print(f"Created apple-touch-icon-{size}x{size}.png")
|
||||||
|
|
||||||
|
# Create garage icons
|
||||||
|
garage_icon = Image.open(static_dir / "garage-icon.png")
|
||||||
|
for size in sizes:
|
||||||
|
resized = garage_icon.resize((size, size), Image.Resampling.LANCZOS)
|
||||||
|
resized.save(static_dir / f"garage-icon-{size}x{size}.png")
|
||||||
|
print(f"Created garage-icon-{size}x{size}.png")
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
create_icon_sizes()
|
||||||
29
test-icons.sh
Executable file
@@ -0,0 +1,29 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
# Script to test icon accessibility over HTTPS/mTLS
|
||||||
|
|
||||||
|
echo "Testing Apple Touch Icon accessibility..."
|
||||||
|
|
||||||
|
# Test base icons
|
||||||
|
echo "1. Testing main apple-touch-icon.png:"
|
||||||
|
curl -I "https://your-domain.com/static/apple-touch-icon-180x180.png" || echo "FAILED"
|
||||||
|
|
||||||
|
echo "2. Testing garage icon:"
|
||||||
|
curl -I "https://your-domain.com/static/garage-icon-180x180.png" || echo "FAILED"
|
||||||
|
|
||||||
|
echo "3. Testing manifest:"
|
||||||
|
curl -I "https://your-domain.com/manifest.json" || echo "FAILED"
|
||||||
|
|
||||||
|
echo "4. Testing favicon route:"
|
||||||
|
curl -I "https://your-domain.com/favicon.ico" || echo "FAILED"
|
||||||
|
|
||||||
|
echo "5. Testing apple-touch-icon route:"
|
||||||
|
curl -I "https://your-domain.com/apple-touch-icon.png" || echo "FAILED"
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "Testing mTLS with client certificate:"
|
||||||
|
echo "6. Testing with client cert:"
|
||||||
|
curl -I --cert client.crt --key client.key "https://your-domain.com/static/apple-touch-icon-180x180.png" || echo "FAILED"
|
||||||
|
|
||||||
|
echo ""
|
||||||
|
echo "Note: Replace 'your-domain.com' with your actual domain"
|
||||||
|
echo "Note: Use actual client certificate files if testing mTLS"
|
||||||