Spaces:
Sleeping
Sleeping
Update app.py
Browse files
app.py
CHANGED
|
@@ -7,6 +7,7 @@ import json
|
|
| 7 |
import tempfile
|
| 8 |
from typing import List, Dict, Tuple, Optional
|
| 9 |
from dataclasses import dataclass
|
|
|
|
| 10 |
|
| 11 |
@dataclass
|
| 12 |
class Question:
|
|
@@ -111,29 +112,115 @@ class QuizGenerator:
|
|
| 111 |
all(isinstance(opt, str) for opt in question["options"])
|
| 112 |
)
|
| 113 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 114 |
class CertificateGenerator:
|
| 115 |
def __init__(self):
|
| 116 |
self.certificate_size = (1200, 800)
|
| 117 |
self.border_color = '#4682B4'
|
| 118 |
self.background_color = '#F0F8FF'
|
| 119 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 120 |
def generate(
|
| 121 |
self,
|
| 122 |
-
name: str,
|
| 123 |
score: float,
|
|
|
|
| 124 |
course_name: str,
|
| 125 |
company_logo: Optional[str] = None,
|
| 126 |
participant_photo: Optional[str] = None
|
| 127 |
) -> str:
|
| 128 |
-
|
| 129 |
-
|
| 130 |
-
|
| 131 |
-
|
| 132 |
-
|
| 133 |
-
|
| 134 |
-
|
| 135 |
-
|
| 136 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 137 |
|
| 138 |
def _create_base_certificate(self) -> Image.Image:
|
| 139 |
return Image.new('RGB', self.certificate_size, self.background_color)
|
|
@@ -183,20 +270,24 @@ class CertificateGenerator:
|
|
| 183 |
draw: ImageDraw.Draw,
|
| 184 |
fonts: Dict[str, ImageFont.FreeTypeFont],
|
| 185 |
name: str,
|
| 186 |
-
|
| 187 |
-
|
| 188 |
):
|
| 189 |
# Title and headers
|
| 190 |
draw.text((600, 100), "CertifyMe AI", font=fonts['title'], fill=self.border_color, anchor="mm")
|
| 191 |
draw.text((600, 160), "Certificate of Achievement", font=fonts['subtitle'], fill=self.border_color, anchor="mm")
|
| 192 |
|
|
|
|
|
|
|
|
|
|
|
|
|
| 193 |
# Main content
|
| 194 |
content = [
|
| 195 |
(300, "This is to certify that", 'black'),
|
| 196 |
-
(380, name
|
| 197 |
(460, "has successfully completed", 'black'),
|
| 198 |
-
(540, course_name
|
| 199 |
-
(620, f"with a score of {score:.1f}%", 'black'),
|
| 200 |
(700, datetime.now().strftime("%B %d, %Y"), 'black')
|
| 201 |
]
|
| 202 |
|
|
|
|
| 7 |
import tempfile
|
| 8 |
from typing import List, Dict, Tuple, Optional
|
| 9 |
from dataclasses import dataclass
|
| 10 |
+
import subprocess
|
| 11 |
|
| 12 |
@dataclass
|
| 13 |
class Question:
|
|
|
|
| 112 |
all(isinstance(opt, str) for opt in question["options"])
|
| 113 |
)
|
| 114 |
|
| 115 |
+
class FontManager:
|
| 116 |
+
"""Manages font installation and loading for the certificate generator"""
|
| 117 |
+
|
| 118 |
+
@staticmethod
|
| 119 |
+
def install_fonts():
|
| 120 |
+
"""Install required fonts if they're not already present"""
|
| 121 |
+
try:
|
| 122 |
+
# Install fonts package
|
| 123 |
+
subprocess.run([
|
| 124 |
+
"apt-get", "update", "-y"
|
| 125 |
+
], check=True)
|
| 126 |
+
subprocess.run([
|
| 127 |
+
"apt-get", "install", "-y",
|
| 128 |
+
"fonts-liberation", # Provides Liberation fonts (Arial alternative)
|
| 129 |
+
"fontconfig" # Font configuration
|
| 130 |
+
], check=True)
|
| 131 |
+
|
| 132 |
+
# Clear font cache
|
| 133 |
+
subprocess.run(["fc-cache", "-f"], check=True)
|
| 134 |
+
print("Fonts installed successfully")
|
| 135 |
+
except subprocess.CalledProcessError as e:
|
| 136 |
+
print(f"Warning: Could not install fonts: {e}")
|
| 137 |
+
except Exception as e:
|
| 138 |
+
print(f"Warning: Unexpected error installing fonts: {e}")
|
| 139 |
+
|
| 140 |
+
@staticmethod
|
| 141 |
+
def get_font_paths() -> Dict[str, str]:
|
| 142 |
+
"""Get the paths to the required fonts"""
|
| 143 |
+
# Liberation Sans is a metric-compatible replacement for Arial
|
| 144 |
+
font_paths = {
|
| 145 |
+
'regular': '/usr/share/fonts/truetype/liberation/LiberationSans-Regular.ttf',
|
| 146 |
+
'bold': '/usr/share/fonts/truetype/liberation/LiberationSans-Bold.ttf'
|
| 147 |
+
}
|
| 148 |
+
|
| 149 |
+
# Fallback paths for different distributions
|
| 150 |
+
fallback_paths = {
|
| 151 |
+
'regular': [
|
| 152 |
+
'/usr/share/fonts/liberation-sans/LiberationSans-Regular.ttf',
|
| 153 |
+
'/usr/share/fonts/truetype/liberation/LiberationSans-Regular.ttf',
|
| 154 |
+
'/usr/share/fonts/TTF/LiberationSans-Regular.ttf'
|
| 155 |
+
],
|
| 156 |
+
'bold': [
|
| 157 |
+
'/usr/share/fonts/liberation-sans/LiberationSans-Bold.ttf',
|
| 158 |
+
'/usr/share/fonts/truetype/liberation/LiberationSans-Bold.ttf',
|
| 159 |
+
'/usr/share/fonts/TTF/LiberationSans-Bold.ttf'
|
| 160 |
+
]
|
| 161 |
+
}
|
| 162 |
+
|
| 163 |
+
# Check fallback paths
|
| 164 |
+
for style in ['regular', 'bold']:
|
| 165 |
+
if not os.path.exists(font_paths[style]):
|
| 166 |
+
for path in fallback_paths[style]:
|
| 167 |
+
if os.path.exists(path):
|
| 168 |
+
font_paths[style] = path
|
| 169 |
+
break
|
| 170 |
+
|
| 171 |
+
return font_paths
|
| 172 |
+
|
| 173 |
class CertificateGenerator:
|
| 174 |
def __init__(self):
|
| 175 |
self.certificate_size = (1200, 800)
|
| 176 |
self.border_color = '#4682B4'
|
| 177 |
self.background_color = '#F0F8FF'
|
| 178 |
|
| 179 |
+
# Install fonts if needed
|
| 180 |
+
FontManager.install_fonts()
|
| 181 |
+
self.font_paths = FontManager.get_font_paths()
|
| 182 |
+
|
| 183 |
+
def _load_fonts(self) -> Dict[str, ImageFont.FreeTypeFont]:
|
| 184 |
+
"""Load fonts with fallbacks"""
|
| 185 |
+
fonts = {}
|
| 186 |
+
try:
|
| 187 |
+
fonts['title'] = ImageFont.truetype(self.font_paths['bold'], 60)
|
| 188 |
+
fonts['text'] = ImageFont.truetype(self.font_paths['regular'], 40)
|
| 189 |
+
fonts['subtitle'] = ImageFont.truetype(self.font_paths['regular'], 30)
|
| 190 |
+
except Exception as e:
|
| 191 |
+
print(f"Font loading error: {e}. Using default font.")
|
| 192 |
+
default = ImageFont.load_default()
|
| 193 |
+
fonts = {
|
| 194 |
+
'title': default,
|
| 195 |
+
'text': default,
|
| 196 |
+
'subtitle': default
|
| 197 |
+
}
|
| 198 |
+
return fonts
|
| 199 |
+
|
| 200 |
def generate(
|
| 201 |
self,
|
|
|
|
| 202 |
score: float,
|
| 203 |
+
name: str,
|
| 204 |
course_name: str,
|
| 205 |
company_logo: Optional[str] = None,
|
| 206 |
participant_photo: Optional[str] = None
|
| 207 |
) -> str:
|
| 208 |
+
"""
|
| 209 |
+
Generate a certificate with custom styling and optional logo/photo
|
| 210 |
+
"""
|
| 211 |
+
try:
|
| 212 |
+
certificate = self._create_base_certificate()
|
| 213 |
+
draw = ImageDraw.Draw(certificate)
|
| 214 |
+
|
| 215 |
+
fonts = self._load_fonts()
|
| 216 |
+
self._add_borders(draw)
|
| 217 |
+
self._add_content(draw, fonts, str(name), str(course_name), float(score))
|
| 218 |
+
self._add_images(certificate, company_logo, participant_photo)
|
| 219 |
+
|
| 220 |
+
return self._save_certificate(certificate)
|
| 221 |
+
except Exception as e:
|
| 222 |
+
print(f"Error generating certificate: {e}")
|
| 223 |
+
return None
|
| 224 |
|
| 225 |
def _create_base_certificate(self) -> Image.Image:
|
| 226 |
return Image.new('RGB', self.certificate_size, self.background_color)
|
|
|
|
| 270 |
draw: ImageDraw.Draw,
|
| 271 |
fonts: Dict[str, ImageFont.FreeTypeFont],
|
| 272 |
name: str,
|
| 273 |
+
course_name: str,
|
| 274 |
+
score: float
|
| 275 |
):
|
| 276 |
# Title and headers
|
| 277 |
draw.text((600, 100), "CertifyMe AI", font=fonts['title'], fill=self.border_color, anchor="mm")
|
| 278 |
draw.text((600, 160), "Certificate of Achievement", font=fonts['subtitle'], fill=self.border_color, anchor="mm")
|
| 279 |
|
| 280 |
+
# Clean inputs
|
| 281 |
+
name = str(name).strip() or "Participant"
|
| 282 |
+
course_name = str(course_name).strip() or "Assessment"
|
| 283 |
+
|
| 284 |
# Main content
|
| 285 |
content = [
|
| 286 |
(300, "This is to certify that", 'black'),
|
| 287 |
+
(380, name, self.border_color),
|
| 288 |
(460, "has successfully completed", 'black'),
|
| 289 |
+
(540, course_name, self.border_color),
|
| 290 |
+
(620, f"with a score of {float(score):.1f}%", 'black'),
|
| 291 |
(700, datetime.now().strftime("%B %d, %Y"), 'black')
|
| 292 |
]
|
| 293 |
|