Files
resume/cv/services/dowload/pdf.py
Pavel Sobolev c4bb087aaf First commite
2025-11-12 23:49:00 +03:00

108 lines
4.4 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
from __future__ import annotations
from io import BytesIO
from weasyprint import HTML
from cv.services.dowload.base import ProfileSerializer
class PdfRenderer:
"""Формирует HTML на лету (без шаблона) и конвертирует в PDF."""
def __init__(self):
# template не используется, оставлен для совместимости интерфейса
self.serializer = ProfileSerializer()
def render(self, profile) -> BytesIO:
data = self.serializer.serialize(profile)
# Минимальный, печатный HTML с безопасными стилями
parts = []
parts.append("<!DOCTYPE html><html lang='ru'><head><meta charset='utf-8'>")
parts.append(f"<title>{data['full_name']} — Резюме (PDF)</title>")
parts.append("""
<style>
@page { size: A4; margin: 18mm 16mm; }
body { font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Arial, sans-serif; color: #111; }
h1 { margin: 0 0 6mm 0; font-size: 20pt; }
h2 { margin: 8mm 0 3mm 0; font-size: 13pt; border-bottom: 1px solid #ccc; padding-bottom: 2mm; }
p { margin: 0 0 3mm 0; line-height: 1.4; }
.meta { color: #555; }
.tag { display: inline-block; border: 1px solid #ddd; padding: 2px 6px; border-radius: 10px; font-size: 9pt; color: #444; margin-right: 4px; }
.item { margin: 0 0 5mm 0; }
.item-title { font-weight: 700; }
.item-period { color: #666; }
ul { margin: 2mm 0 2mm 6mm; }
</style>
</head><body>
""")
# Header
parts.append(f"<h1>{data['full_name']}</h1>")
if data["role"]:
parts.append(f"<p class='meta'><strong>{data['role']}</strong></p>")
if data["summary"]:
parts.append(f"<p>{data['summary']}</p>")
meta_tags = []
if data["location"]:
meta_tags.append(f"<span class='tag'>{data['location']}</span>")
if data["languages"]:
meta_tags.append(f"<span class='tag'>Языки: {', '.join(data['languages'])}</span>")
if meta_tags:
parts.append(f"<p class='meta'>{' '.join(meta_tags)}</p>")
# Contacts
contacts = data["contacts"]
parts.append("<h2>Контакты</h2><p>")
if contacts.get("email"):
parts.append(f"<strong>Email:</strong> {contacts['email']}<br>")
if contacts.get("phone"):
parts.append(f"<strong>Телефон:</strong> {contacts['phone']}<br>")
if contacts.get("telegram"):
parts.append(f"<strong>Telegram:</strong> {contacts['telegram']}")
parts.append("</p>")
# Experience
parts.append("<h2>Опыт работы</h2>")
exp = data["experience"]
if exp:
for e in exp:
parts.append("<div class='item'>")
parts.append(f"<div class='item-title'>{e.company}</div>")
period = ""
if getattr(e, 'start_date', None):
period += e.start_date.strftime("%B %Y")
period += ""
period += e.end_date.strftime("%B %Y") if getattr(e, 'end_date', None) else "настоящее время"
parts.append(f"<div class='item-period'>{period}</div>")
if e.summary:
parts.append(f"<p>{e.summary}</p>")
if e.achievements:
parts.append("<ul>")
for a in e.achievements:
if a:
parts.append(f"<li>{a}</li>")
parts.append("</ul>")
if e.tech:
parts.append(f"<p class='meta'>Технологии: {', '.join(e.tech)}</p>")
parts.append("</div>")
else:
parts.append("<p class='meta'>Нет записей.</p>")
# Skills
parts.append("<h2>Навыки</h2>")
skills = data["skills_map"]
if skills:
for g in skills:
if g.items:
parts.append(f"<p><strong>{g.group}:</strong> {', '.join([i for i in g.items if i])}</p>")
else:
parts.append("<p class='meta'>Нет данных.</p>")
parts.append("</body></html>")
html = "".join(parts)
pdf = HTML(string=html).write_pdf()
out = BytesIO(pdf)
out.seek(0)
return out