mirror of
https://github.com/wasrusgen/zov-tech.git
synced 2026-06-03 21:44:48 +00:00
94 lines
4.9 KiB
JavaScript
94 lines
4.9 KiB
JavaScript
/**
|
||
* ЗОВ — База: одноразовый setup всех 8 листов с заголовками и формулами.
|
||
*
|
||
* Как запустить:
|
||
* 1. Открыть таблицу «ЗОВ — База» в Google Sheets.
|
||
* 2. Extensions (Расширения) → Apps Script.
|
||
* 3. В редакторе скопировать сюда содержимое этого файла.
|
||
* 4. Сохранить (Ctrl+S).
|
||
* 5. В верхней панели выбрать функцию `setupDatabase` → нажать Run.
|
||
* 6. Authorize при первом запуске (Google спросит разрешение на работу со Sheet).
|
||
* 7. Вернуться в таблицу — все 8 листов готовы.
|
||
*/
|
||
|
||
const SHEETS = {
|
||
Users: ["tg_id","tg_username","first_name","last_name","role","created_at","last_seen_at","invite_code_used"],
|
||
Managers: ["tg_id","full_name","email","phone","salon","city","is_zov_employee","status","last_order_date","active_until","total_leads","total_deals","conversion_rate","invite_code"],
|
||
Clients: ["tg_id","full_name","phone","email","address","city","budget_total","manager_tg_id","source","last_measurement_id"],
|
||
Measurements: ["id","created_at","client_tg_id","manager_tg_id","filled_by","layout","area_m2","ceiling_mm","walls_json","openings_json","infra_json","niches_json","photos_urls","notes","status"],
|
||
Leads: ["id","created_at","manager_tg_id","client_tg_id","client_name","measurement_id","checklist_json","ai_response","ai_model","ai_tokens_used","sent_to_tg","deal_status","deal_amount"],
|
||
Logs: ["timestamp","event","tg_id","payload"],
|
||
Settings: ["key","value","description"],
|
||
Dashboard: ["metric","value"],
|
||
};
|
||
|
||
const SETTINGS_DEFAULTS = [
|
||
["ACTIVE_PERIOD_DAYS", 90, "Дней active-статуса после сделки через куратора"],
|
||
["GRACE_PERIOD_DAYS", 14, "Grace-период перед переводом в lapsed"],
|
||
["AI_MODEL", "claude-haiku-4-5-20251001", "Модель Anthropic для подбора"],
|
||
["AI_TEMPERATURE", 0.3, "Температура генерации"],
|
||
["ADMIN_TG_ID", 5937498515, "Tg_id куратора (Руслан Васильев)"],
|
||
["PAID_PRICE_PER_LEAD", 500, "Цена pay-per-use для lapsed-менеджеров, ₽"],
|
||
["PAID_SUBSCRIPTION", 3000, "Цена месячной подписки, ₽"],
|
||
];
|
||
|
||
const DASHBOARD_ROWS = [
|
||
["Активных менеджеров", '=COUNTIF(Managers!H2:H; "active")'],
|
||
["Lapsed-менеджеров", '=COUNTIF(Managers!H2:H; "lapsed")'],
|
||
["Заявок за 30 дней", '=COUNTIFS(Leads!B2:B; ">="&(TODAY()-30))'],
|
||
["Сделок won", '=COUNTIF(Leads!L2:L; "won")'],
|
||
["Конверсия в сделку", '=IFERROR(COUNTIF(Leads!L2:L; "won")/COUNTA(Leads!A2:A); 0)'],
|
||
["Средний чек won-сделок, ₽", '=IFERROR(AVERAGEIF(Leads!L2:L; "won"; Leads!M2:M); 0)'],
|
||
["Замеров всего", '=COUNTA(Measurements!A2:A)'],
|
||
["Расход на AI, токенов", '=SUM(Leads!J2:J)'],
|
||
];
|
||
|
||
function setupDatabase() {
|
||
const ss = SpreadsheetApp.getActiveSpreadsheet();
|
||
|
||
// Удалить дефолтный пустой лист, если есть
|
||
const defaultSheet = ss.getSheetByName("Sheet1") || ss.getSheetByName("Лист1");
|
||
|
||
Object.keys(SHEETS).forEach((name, idx) => {
|
||
let sheet = ss.getSheetByName(name);
|
||
if (!sheet) {
|
||
sheet = ss.insertSheet(name, idx);
|
||
} else {
|
||
sheet.clear();
|
||
}
|
||
const headers = SHEETS[name];
|
||
sheet.getRange(1, 1, 1, headers.length).setValues([headers]).setFontWeight("bold").setBackground("#F0F9E8");
|
||
sheet.setFrozenRows(1);
|
||
sheet.autoResizeColumns(1, headers.length);
|
||
});
|
||
|
||
// Settings — заполнить дефолтами
|
||
const settings = ss.getSheetByName("Settings");
|
||
settings.getRange(2, 1, SETTINGS_DEFAULTS.length, 3).setValues(SETTINGS_DEFAULTS);
|
||
|
||
// Dashboard — заполнить формулами
|
||
const dashboard = ss.getSheetByName("Dashboard");
|
||
for (let i = 0; i < DASHBOARD_ROWS.length; i++) {
|
||
const [metric, formula] = DASHBOARD_ROWS[i];
|
||
dashboard.getRange(i + 2, 1).setValue(metric);
|
||
dashboard.getRange(i + 2, 2).setFormula(formula);
|
||
}
|
||
dashboard.setColumnWidth(1, 280);
|
||
dashboard.setColumnWidth(2, 140);
|
||
|
||
// Удалить пустой Sheet1 в самом конце (если был)
|
||
if (defaultSheet && ss.getSheets().length > 1) {
|
||
try { ss.deleteSheet(defaultSheet); } catch (e) {}
|
||
}
|
||
|
||
SpreadsheetApp.getActive().toast("✅ База готова — 8 листов созданы");
|
||
}
|
||
|
||
/** Удобный вызов через меню — необязательно */
|
||
function onOpen() {
|
||
SpreadsheetApp.getUi()
|
||
.createMenu("ЗОВ")
|
||
.addItem("Setup database", "setupDatabase")
|
||
.addToUi();
|
||
}
|