gigafibre-fsm/services/targo-hub/lib/contracts.js
louispaulb 41d9b5f316 feat: flow editor, Gemini QR scanner with offline queue, dispatch planning v2
Major additions accumulated over 9 days — single commit per request.

Flow editor (new):
- Generic visual editor for step trees, usable by project wizard + agent flows
- PROJECT_KINDS / AGENT_KINDS catalogs decouple UI from domain
- Drag-and-drop reorder via vuedraggable with scope isolation per peer group
- Chain-aware depends_on rewrite on reorder (sequential only — DAGs preserved)
- Variable picker with per-applies_to catalog (Customer / Quotation /
  Service Contract / Issue / Subscription), insert + copy-clipboard modes
- trigger_condition helper with domain-specific JSONLogic examples
- Global FlowEditorDialog mounted once in MainLayout, Odoo inline pattern
- Server: targo-hub flow-runtime.js, flow-api.js, flow-templates.js
- ERPNext: Flow Template/Run doctypes, scheduler, 5 seeded system templates
- depends_on chips resolve to step labels instead of opaque "s4" ids

QR/OCR scanner (field app):
- Camera capture → Gemini Vision via targo-hub with 8s timeout
- IndexedDB offline queue retries photos when signal returns
- Watcher merges late-arriving scan results into the live UI

Dispatch:
- Planning mode (draft → publish) with offer pool for unassigned jobs
- Shared presets, recurrence selector, suggested-slots dialog
- PublishScheduleModal, unassign confirmation

Ops app:
- ClientDetailPage composables extraction (useClientData, useDeviceStatus,
  useWifiDiagnostic, useModemDiagnostic)
- Project wizard: shared detail sections, wizard catalog/publish composables
- Address pricing composable + pricing-mock data
- Settings redesign hosting flow templates

Targo-hub:
- Contract acceptance (JWT residential + DocuSeal commercial tracks)
- Referral system
- Modem-bridge diagnostic normalizer
- Device extractors consolidated

Migration scripts:
- Invoice/quote print format setup, Jinja rendering
- Additional import + fix scripts (reversals, dates, customers, payments)

Docs:
- Consolidated: old scattered MDs → HANDOFF, ARCHITECTURE, DATA_AND_FLOWS,
  FLOW_EDITOR_ARCHITECTURE, BILLING_AND_PAYMENTS, CPE_MANAGEMENT,
  APP_DESIGN_GUIDELINES
- Archived legacy wizard PHP for reference
- STATUS snapshots for 2026-04-18/19

Cleanup:
- Removed ~40 generated PDFs/HTMLs (invoice_preview*, rendered_jinja*)
- .gitignore now covers invoice preview output + nested .DS_Store

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-04-22 10:44:17 -04:00

619 lines
26 KiB
JavaScript
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.

'use strict'
const cfg = require('./config')
const { log, json, parseBody, erpFetch } = require('./helpers')
const { signJwt, verifyJwt } = require('./magic-link')
// Lazy-loaded flow runtime. Kept local so the require() cost isn't paid on
// modules that don't trigger flows.
let _flowRuntime
function _fireFlowTrigger (event, ctx) {
if (!_flowRuntime) _flowRuntime = require('./flow-runtime')
return _flowRuntime.dispatchEvent(event, ctx)
}
// Résidentiel:
// - Pénalité = valeur résiduelle des avantages non compensés
// - Chaque mois d'abonnement "reconnaît" (compense) benefit_value / duration_months
// - Ex: Installation 288$ gratuite sur 24 mois → 12$/mois compensé
// - Résiliation au mois 18 → 6 × 12$ = 72$ de pénalité
//
// Commercial:
// - Pénalité = toutes les mensualités restantes au contrat
// - Ex: 79.95$/mois × 6 mois restants = 479.70$
function calculateTerminationFee (contract) {
const now = new Date()
const start = new Date(contract.start_date)
const monthsElapsed = Math.max(0, monthsBetween(start, now))
const monthsRemaining = Math.max(0, (contract.duration_months || 24) - monthsElapsed)
const benefits = (contract.benefits || []).map(b => {
const benefitValue = (b.regular_price || 0) - (b.granted_price || 0)
const monthlyRecognition = benefitValue / (contract.duration_months || 24)
const recognized = monthlyRecognition * monthsElapsed
const remaining = Math.max(0, benefitValue - recognized)
return {
description: b.description,
regular_price: b.regular_price,
granted_price: b.granted_price,
benefit_value: round2(benefitValue),
monthly_recognition: round2(monthlyRecognition),
months_recognized: monthsElapsed,
remaining_value: round2(remaining),
}
})
const totalBenefitRemaining = round2(benefits.reduce((s, b) => s + b.remaining_value, 0))
const monthlyRate = contract.monthly_rate || 0
let fee = 0
let feeBreakdown = {}
if (contract.contract_type === 'Commercial') {
// Commercial: all remaining monthly payments
fee = round2(monthsRemaining * monthlyRate)
feeBreakdown = {
type: 'commercial',
months_remaining: monthsRemaining,
monthly_rate: monthlyRate,
termination_fee_remaining: fee,
termination_fee_benefits: 0,
termination_fee_total: fee,
}
} else {
// Résidentiel: only benefit residual value (+ current month max)
fee = totalBenefitRemaining
feeBreakdown = {
type: 'residential',
months_remaining: monthsRemaining,
monthly_rate: monthlyRate,
termination_fee_benefits: totalBenefitRemaining,
termination_fee_remaining: 0,
termination_fee_total: totalBenefitRemaining,
}
}
return {
months_elapsed: monthsElapsed,
months_remaining: monthsRemaining,
benefits,
total_benefit_remaining: totalBenefitRemaining,
...feeBreakdown,
}
}
function generateInvoiceNote (contract) {
const benefits = contract.benefits || []
const lines = []
for (const b of benefits) {
const benefitValue = (b.regular_price || 0) - (b.granted_price || 0)
if (benefitValue <= 0) continue
const monthly = round2(benefitValue / (contract.duration_months || 24))
lines.push(
`${b.description} ${b.granted_price || 0}$ sur entente de ${contract.duration_months || 24} mois. ` +
`(Prix régulier ${b.regular_price}$) Chaque mois d'abonnement reconnaît ${monthly}$ de compensé.`
)
}
return lines.join('\n')
}
function generateContractToken (contractName, customer, phone, ttlHours = 168) {
return signJwt({
sub: customer,
doc: contractName,
type: 'contract_accept',
phone: phone || '',
iat: Math.floor(Date.now() / 1000),
exp: Math.floor(Date.now() / 1000) + ttlHours * 3600,
})
}
function generateContractLink (contractName, customer, phone) {
const token = generateContractToken(contractName, customer, phone)
return `${cfg.HUB_PUBLIC_URL || 'https://msg.gigafibre.ca'}/contract/accept/${token}`
}
async function handle (req, res, method, path) {
// GET /contract/list?customer=CUST-001
if (path === '/contract/list' && method === 'GET') {
const url = new URL(req.url, `http://localhost:${cfg.PORT}`)
const customer = url.searchParams.get('customer')
const filters = customer ? [['customer', '=', customer]] : []
filters.push(['status', '!=', 'Brouillon'])
const params = new URLSearchParams({
fields: JSON.stringify(['name', 'customer', 'customer_name', 'contract_type', 'status',
'start_date', 'end_date', 'duration_months', 'monthly_rate', 'total_benefit_value',
'total_remaining_value', 'months_elapsed', 'months_remaining']),
filters: JSON.stringify(filters),
limit_page_length: 200,
order_by: 'start_date desc',
})
const r = await erpFetch('/api/resource/Service%20Contract?' + params)
return json(res, 200, { contracts: r.status === 200 ? (r.data?.data || []) : [] })
}
// GET /contract/detail/CTR-00001
if (path.startsWith('/contract/detail/') && method === 'GET') {
const name = decodeURIComponent(path.replace('/contract/detail/', ''))
const r = await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(name)}`)
if (r.status !== 200) return json(res, 404, { error: 'Contract not found' })
return json(res, 200, { contract: r.data.data })
}
// POST /contract/create — create a new contract
if (path === '/contract/create' && method === 'POST') {
const body = await parseBody(req)
if (!body.customer || !body.duration_months) {
return json(res, 400, { error: 'customer and duration_months required' })
}
// Build benefits with calculated fields
const duration = body.duration_months || 24
const benefits = (body.benefits || []).map(b => ({
description: b.description || '',
regular_price: b.regular_price || 0,
granted_price: b.granted_price || 0,
benefit_value: round2((b.regular_price || 0) - (b.granted_price || 0)),
monthly_recognition: round2(((b.regular_price || 0) - (b.granted_price || 0)) / duration),
months_recognized: 0,
remaining_value: round2((b.regular_price || 0) - (b.granted_price || 0)),
}))
const totalBenefitValue = round2(benefits.reduce((s, b) => s + b.benefit_value, 0))
const startDate = body.start_date || new Date().toISOString().slice(0, 10)
const endDate = addMonths(startDate, duration)
// Generate invoice note
const invoiceNote = generateInvoiceNote({
benefits: body.benefits || [],
duration_months: duration,
})
const payload = {
customer: body.customer,
contract_type: body.contract_type || 'Résidentiel',
status: 'Brouillon',
start_date: startDate,
end_date: endDate,
duration_months: duration,
monthly_rate: body.monthly_rate || 0,
service_location: body.service_location || '',
quotation: body.quotation || '',
subscription: body.subscription || '',
benefits,
total_benefit_value: totalBenefitValue,
total_remaining_value: totalBenefitValue,
months_elapsed: 0,
months_remaining: duration,
invoice_note: invoiceNote,
acceptance_method: body.acceptance_method || '',
docuseal_template_id: body.docuseal_template_id || 0,
}
const r = await erpFetch('/api/resource/Service%20Contract', {
method: 'POST',
body: JSON.stringify(payload),
})
if (r.status !== 200) {
return json(res, 500, { error: 'Failed to create contract', detail: r.data })
}
log(`Contract created: ${r.data.data.name} for ${body.customer}`)
return json(res, 200, { ok: true, contract: r.data.data })
}
// POST /contract/calculate-termination — preview termination fee without saving
if (path === '/contract/calculate-termination' && method === 'POST') {
const body = await parseBody(req)
if (!body.name) return json(res, 400, { error: 'contract name required' })
const r = await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(body.name)}`)
if (r.status !== 200) return json(res, 404, { error: 'Contract not found' })
const contract = r.data.data
const calc = calculateTerminationFee(contract)
return json(res, 200, { contract_name: body.name, ...calc })
}
// POST /contract/terminate — execute termination + create invoice
if (path === '/contract/terminate' && method === 'POST') {
const body = await parseBody(req)
if (!body.name) return json(res, 400, { error: 'contract name required' })
const r = await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(body.name)}`)
if (r.status !== 200) return json(res, 404, { error: 'Contract not found' })
const contract = r.data.data
if (contract.status === 'Résilié' || contract.status === 'Complété') {
return json(res, 400, { error: 'Contract already terminated' })
}
const calc = calculateTerminationFee(contract)
const today = new Date().toISOString().slice(0, 10)
// Update contract with termination data
const updateData = {
status: 'Résilié',
terminated_at: today,
termination_reason: body.reason || '',
termination_fee_benefits: calc.termination_fee_benefits,
termination_fee_remaining: calc.termination_fee_remaining,
termination_fee_total: calc.termination_fee_total,
months_elapsed: calc.months_elapsed,
months_remaining: calc.months_remaining,
}
// Update benefits rows with current recognition
if (contract.benefits && contract.benefits.length) {
updateData.benefits = calc.benefits.map((b, i) => ({
...contract.benefits[i],
months_recognized: b.months_recognized,
remaining_value: b.remaining_value,
benefit_value: b.benefit_value,
monthly_recognition: b.monthly_recognition,
}))
updateData.total_remaining_value = calc.total_benefit_remaining
}
await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(body.name)}`, {
method: 'PUT',
body: JSON.stringify(updateData),
})
// Create termination invoice if fee > 0
let invoiceName = null
if (calc.termination_fee_total > 0 && body.create_invoice !== false) {
invoiceName = await createTerminationInvoice(contract, calc, body.reason)
if (invoiceName) {
await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(body.name)}`, {
method: 'PUT',
body: JSON.stringify({ termination_invoice: invoiceName }),
})
}
}
log(`Contract ${body.name} terminated. Fee: ${calc.termination_fee_total}$. Invoice: ${invoiceName || 'none'}`)
return json(res, 200, {
ok: true,
...calc,
termination_invoice: invoiceName,
})
}
// POST /contract/send — send contract for acceptance (DocuSeal or JWT SMS)
if (path === '/contract/send' && method === 'POST') {
const body = await parseBody(req)
if (!body.name) return json(res, 400, { error: 'contract name required' })
const r = await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(body.name)}`)
if (r.status !== 200) return json(res, 404, { error: 'Contract not found' })
const contract = r.data.data
const phone = body.phone
const email = body.email
let result = {}
// Try DocuSeal first if configured
if (body.use_docuseal && cfg.DOCUSEAL_URL && cfg.DOCUSEAL_API_KEY) {
const { createDocuSealSubmission } = require('./acceptance')
const acceptLink = generateContractLink(body.name, contract.customer, phone)
const ds = await createDocuSealSubmission({
templateId: contract.docuseal_template_id || body.docuseal_template_id || 1,
email: email || '',
name: contract.customer_name || contract.customer,
phone: phone || '',
values: {
'Nom': contract.customer_name || '',
'Contrat': body.name,
'Mensualité': `${contract.monthly_rate}$`,
'Durée': `${contract.duration_months} mois`,
'Avantages': (contract.benefits || []).map(b => b.description).join(', '),
},
completedRedirectUrl: acceptLink + '?signed=1',
})
if (ds) {
await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(body.name)}`, {
method: 'PUT',
body: JSON.stringify({
status: 'Envoyé',
acceptance_method: 'DocuSeal',
docuseal_submission_id: ds.submissionId,
}),
})
result = { method: 'docuseal', sign_url: ds.signUrl, submission_id: ds.submissionId }
}
}
// Fallback: JWT SMS acceptance
if (!result.method && phone) {
const link = generateContractLink(body.name, contract.customer, phone)
const { sendSmsInternal } = require('./twilio')
const note = (contract.benefits || []).map(b =>
`${b.description}: ${b.granted_price || 0}$ (rég. ${b.regular_price}$)`
).join('\n')
const msg = `Gigafibre — Entente de service\n` +
`${contract.monthly_rate}$/mois × ${contract.duration_months} mois\n` +
(note ? note + '\n' : '') +
`Accepter: ${link}`
const sid = await sendSmsInternal(phone, msg)
await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(body.name)}`, {
method: 'PUT',
body: JSON.stringify({ status: 'Envoyé', acceptance_method: 'JWT SMS' }),
})
result = { method: 'sms', sent_to: phone, sms_sid: sid, accept_link: link }
}
if (!result.method) {
return json(res, 400, { error: 'No phone or DocuSeal configured for sending' })
}
log(`Contract ${body.name} sent via ${result.method}`)
return json(res, 200, { ok: true, ...result })
}
// GET /contract/accept/:token — accept contract via JWT
if (path.startsWith('/contract/accept/') && method === 'GET') {
const token = path.replace('/contract/accept/', '')
const payload = verifyJwt(token)
if (!payload || payload.type !== 'contract_accept') {
return json(res, 401, { error: 'Lien expiré ou invalide.' })
}
// Render simple acceptance page
const contractName = payload.doc
const r = await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(contractName)}`)
if (r.status !== 200) return json(res, 404, { error: 'Contrat introuvable' })
const contract = r.data.data
const html = renderAcceptancePage(contract, token)
res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' })
return res.end(html)
}
// POST /contract/confirm — confirm acceptance
if (path === '/contract/confirm' && method === 'POST') {
const body = await parseBody(req)
if (!body.token) return json(res, 400, { error: 'token required' })
const payload = verifyJwt(body.token)
if (!payload || payload.type !== 'contract_accept') {
return json(res, 401, { error: 'Lien expiré' })
}
const contractName = payload.doc
const ip = req.headers['x-forwarded-for'] || req.socket?.remoteAddress || ''
const ua = req.headers['user-agent'] || ''
const now = new Date().toISOString()
await erpFetch(`/api/resource/Service%20Contract/${encodeURIComponent(contractName)}`, {
method: 'PUT',
body: JSON.stringify({
status: 'Actif',
signed_at: now,
signed_by: payload.sub,
signature_proof: `IP: ${ip}\nUser-Agent: ${ua}\nDate: ${now}\nPhone: ${payload.phone || ''}`,
}),
})
log(`Contract ${contractName} accepted by ${payload.sub}`)
// Fire flow trigger: on_contract_signed
// Non-blocking — flow runtime handles its own errors so the acceptance
// endpoint never fails because of downstream automation.
_fireFlowTrigger('on_contract_signed', {
doctype: 'Service Contract',
docname: contractName,
customer: payload.sub,
variables: {
contract_type: payload.contract_type,
signed_at: now,
},
}).catch(e => log('flow trigger on_contract_signed failed:', e.message))
return json(res, 200, { ok: true, contract: contractName })
}
return json(res, 404, { error: 'Contract endpoint not found' })
}
async function createTerminationInvoice (contract, calc, reason) {
const items = []
const today = new Date().toISOString().slice(0, 10)
if (contract.contract_type === 'Commercial') {
items.push({
item_name: 'Frais de résiliation anticipée',
description: `Résiliation anticipée du contrat ${contract.name}. ` +
`${calc.months_remaining} mois restants × ${contract.monthly_rate}$/mois.` +
(reason ? ` Raison: ${reason}` : ''),
qty: calc.months_remaining,
rate: contract.monthly_rate,
income_account: '', // will use default
})
} else {
// Residential: charge benefit residual
for (const b of calc.benefits) {
if (b.remaining_value <= 0) continue
items.push({
item_name: `Résiliation — ${b.description}`,
description: `Valeur résiduelle: ${b.benefit_value}$ - ${b.months_recognized} mois reconnus ` +
`(${b.monthly_recognition}$/mois). Restant: ${b.remaining_value}$`,
qty: 1,
rate: b.remaining_value,
})
}
}
if (!items.length) return null
const payload = {
customer: contract.customer,
posting_date: today,
due_date: today,
items,
remarks: `Facture de résiliation anticipée — Contrat ${contract.name}`,
}
try {
const r = await erpFetch('/api/resource/Sales%20Invoice', {
method: 'POST',
body: JSON.stringify(payload),
})
if (r.status === 200 && r.data?.data) {
return r.data.data.name
}
} catch (e) {
log('Termination invoice creation failed:', e.message)
}
return null
}
function renderAcceptancePage (contract, token) {
const benefitRows = (contract.benefits || []).map(b => {
return `<tr>
<td>${b.description}</td>
<td class="r"><s style="color:#9ca3af">${b.regular_price || 0} $</s> → <strong>${b.granted_price || 0} $</strong></td>
</tr>`
}).join('')
const duration = contract.duration_months || 24
const totalBenefit = (contract.benefits || []).reduce(
(s, b) => s + ((b.regular_price || 0) - (b.granted_price || 0)), 0,
)
const TARGO_LOGO_SVG = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 326.39 70.35" width="130" height="28"><path fill="#019547" d="M25.83,15H8.41a4.14,4.14,0,0,1-3.89-2.71L0,1.18H66.59L62.07,12.27A4.14,4.14,0,0,1,58.18,15H40.76V69.19H25.81Z"/><path fill="#019547" d="M90.74.68h18.55a5.63,5.63,0,0,1,5.63,5.6l.26,62.91H99.55L99.76,54H71.87L66.41,65.51a6.45,6.45,0,0,1-5.84,3.68H49.35L73.53,12.11A18.7,18.7,0,0,1,90.74.68M100,40.73l.07-26h-8a6.75,6.75,0,0,0-6.26,4.18L76.73,40.73Z"/><path fill="#019547" d="M124.51,6.81a5.64,5.64,0,0,1,5.65-5.65H155.6c8.64,0,15.37,2.44,19.81,6.91,3.81,3.78,5.84,9.14,5.84,15.53v.18c0,11-5.92,17.87-14.59,21.1l16.61,24.29h-14a6.51,6.51,0,0,1-5.39-2.87L151.21,47.41H139.44V69.17h-15Zm30.12,27.41c7.28,0,11.45-3.89,11.45-9.62v-.21c0-6.42-4.46-9.7-11.74-9.7H139.46V34.22Z"/><path fill="#019547" d="M184.74,35.37v-.18C184.74,15.85,199.8,0,220.4,0,232.65,0,240,3.31,247.13,9.33l-6.28,7.57a5.18,5.18,0,0,1-6.84,1,23.71,23.71,0,0,0-14.11-4.13c-10.88,0-19.52,9.62-19.52,21.18v.19c0,12.43,8.54,21.57,20.6,21.57a24,24,0,0,0,14.09-4.07V42.91h-8.68A6.41,6.41,0,0,1,220,36.53V30h29.54V59.52a44,44,0,0,1-29,10.78c-21.18,0-35.74-14.8-35.74-34.93"/><path fill="#019547" d="M254.09,35.37v-.18C254.09,15.85,269.33,0,290.33,0s36.06,15.66,36.06,35v.18c0,19.34-15.27,35.19-36.24,35.19s-36.06-15.64-36.06-35m56.63,0v-.18c0-11.67-8.54-21.39-20.6-21.39S269.73,23.34,269.73,35v.18c0,11.67,8.54,21.37,20.6,21.37S310.72,47,310.72,35.37"/></svg>`
const isCommercial = contract.contract_type === 'Commercial'
const penaltyText = isCommercial
? `la totalité des mensualités restantes au terme (${contract.monthly_rate || 0} $/mois × mois restants)`
: `la portion non étalée de la promotion, au prorata des mois restants sur ${duration} mois`
return `<!DOCTYPE html><html lang="fr"><head>
<meta charset="utf-8"><meta name="viewport" content="width=device-width,initial-scale=1">
<title>Entente de service — TARGO</title>
<style>
*{box-sizing:border-box;margin:0;padding:0}
body{font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,sans-serif;background:#f5f7fa;color:#1a1a1a;padding:16px;line-height:1.5}
.card{background:#fff;border-radius:14px;padding:24px;max-width:560px;margin:0 auto;box-shadow:0 4px 16px rgba(0,0,0,.06)}
.brand{display:flex;align-items:center;gap:12px;padding-bottom:16px;border-bottom:2px solid #019547;margin-bottom:18px}
.brand .num{margin-left:auto;font-size:12px;color:#888;text-align:right}
h1{font-size:20px;font-weight:800;color:#019547;margin-bottom:4px;letter-spacing:.2px}
.sub{color:#6b7280;font-size:13px;margin-bottom:18px}
.field{display:flex;justify-content:space-between;gap:12px;padding:9px 0;border-bottom:1px solid #f1f5f9}
.field:last-child{border-bottom:none}
.field .label{color:#6b7280;font-size:13px}
.field .value{font-weight:600;font-size:14px;text-align:right}
h3{font-size:14px;color:#019547;margin-top:18px;margin-bottom:8px;font-weight:700;text-transform:uppercase;letter-spacing:.3px}
table{width:100%;border-collapse:collapse;margin:6px 0 10px 0;font-size:12.5px}
th{text-align:left;padding:6px 8px;background:#f4fff6;color:#00733a;font-weight:700;border-bottom:2px solid #019547}
th.r,td.r{text-align:right}
td{padding:6px 8px;border-bottom:1px solid #f1f5f9}
tfoot td{font-weight:700;color:#00733a;background:#f4fff6;border-top:1px solid #019547;border-bottom:none}
.callout{background:#fffbeb;border:1px solid #fde68a;border-radius:10px;padding:12px 14px;margin:14px 0;font-size:13px;line-height:1.55}
.callout .head{color:#b45309;font-weight:700;margin-bottom:4px}
.legal{font-size:12px;color:#6b7280;margin-top:14px;line-height:1.55}
.legal a{color:#019547;text-decoration:none;font-weight:600}
.btn{display:block;width:100%;padding:16px;background:#019547;color:#fff;border:none;border-radius:12px;font-size:16px;font-weight:700;cursor:pointer;margin-top:18px;letter-spacing:.3px}
.btn:active{background:#01733a}
.btn:disabled{background:#9ca3af;cursor:wait}
.ok{text-align:center;padding:40px 20px}
.ok .check{width:64px;height:64px;border-radius:50%;background:#019547;color:#fff;display:flex;align-items:center;justify-content:center;margin:0 auto 16px;font-size:32px;font-weight:700}
.ok h2{color:#019547;margin-bottom:8px;font-size:22px}
.ok p{color:#4b5563;font-size:14px}
#result{display:none}
</style></head><body>
<div class="card" id="contract">
<div class="brand">
${TARGO_LOGO_SVG}
<div class="num">Référence<br><strong>${contract.name || ''}</strong></div>
</div>
<h1>Récapitulatif de votre service</h1>
<div class="sub">Un résumé clair des modalités convenues — rien de plus.</div>
<div class="intro" style="background:#f4fff6;border:1px solid #bde5cb;border-radius:10px;padding:12px 14px;margin-bottom:16px;font-size:13px;color:#00733a;line-height:1.55">
Voici un récapitulatif de votre service Gigafibre. Il reprend ce que nous avons convenu ensemble pour que tout soit clair des deux côtés. Aucune surprise — tout est écrit noir sur blanc ci-dessous.
</div>
<div class="field"><span class="label">Client</span><span class="value">${escapeHtml(contract.customer_name || contract.customer || '')}</span></div>
<div class="field"><span class="label">Mensualité</span><span class="value">${contract.monthly_rate || 0} $/mois <span style="color:#9ca3af;font-weight:400;font-size:12px">(+taxes)</span></span></div>
<div class="field"><span class="label">Durée</span><span class="value">${duration} mois</span></div>
<div class="field"><span class="label">Début prévu</span><span class="value">${contract.start_date || 'À déterminer'}</span></div>
${benefitRows ? `
<h3>Promotions appliquées</h3>
<table>
<tbody>${benefitRows}</tbody>
<tfoot>
<tr><td>Valeur totale de la promotion</td><td class="r">${round2(totalBenefit)} $</td></tr>
<tr><td colspan="2" style="font-weight:400;color:#00733a;font-size:12px;text-align:left;">Étalée sur ${duration} mois</td></tr>
</tfoot>
</table>` : ''}
<div class="callout">
<div class="head">Changement avant ${duration} mois ?</div>
Pas de pénalité. On récupère simplement ${penaltyText}. Rien de plus.
</div>
<div class="legal">
En cliquant <strong>« J'accepte »</strong>, vous confirmez que les informations ci-dessus correspondent à ce que nous avons convenu. Les <a href="https://www.targo.ca/conditions" target="_blank">conditions complètes du service</a> (facturation, équipement, Loi 25, juridiction du Québec) s'appliquent également. Votre confirmation est horodatée.
</div>
<button class="btn" onclick="accept()">J'accepte ce récapitulatif</button>
</div>
<div class="card ok" id="result">
<div class="check">✓</div>
<h2>C'est confirmé !</h2>
<p>Merci ! Une copie du récapitulatif vous sera envoyée par courriel sous peu.</p>
</div>
<script>
async function accept(){
const btn=document.querySelector('.btn');btn.disabled=true;btn.textContent='Enregistrement...';
try{
const r=await fetch('/contract/confirm',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({token:'${token}'})});
const d=await r.json();
if(d.ok){document.getElementById('contract').style.display='none';document.getElementById('result').style.display='block';}
else{btn.textContent='Erreur : '+(d.error||'Réessayez');btn.disabled=false;}
}catch(e){btn.textContent='Erreur réseau';btn.disabled=false;}
}
</script></body></html>`
}
function escapeHtml (s) {
return String(s || '').replace(/[&<>"']/g, c => ({
'&': '&amp;', '<': '&lt;', '>': '&gt;', '"': '&quot;', "'": '&#39;',
}[c]))
}
function monthsBetween (d1, d2) {
return (d2.getFullYear() - d1.getFullYear()) * 12 + (d2.getMonth() - d1.getMonth())
}
function addMonths (dateStr, months) {
const d = new Date(dateStr)
d.setMonth(d.getMonth() + months)
return d.toISOString().slice(0, 10)
}
function round2 (v) { return Math.round(v * 100) / 100 }
module.exports = { handle, calculateTerminationFee, generateInvoiceNote, generateContractLink, renderAcceptancePage }