(async function(){
var RAW_SCHEMA = "{"elements":[{"type":"Header","text":"Калькулятор фундаментной плиты","bgColor":"#e62","textColor":"#FFF"},{"type":"Input","key":"LEN_M","name":"Длина плиты, м","default":10,"valueMode":"number"},{"type":"Input","key":"WID_M","name":"Ширина плиты, м","default":10,"valueMode":"number"},{"type":"Input","key":"THK_MM","name":"Толщина плиты, мм","default":350,"valueMode":"number"},{"type":"Radio","key":"PU_NEEDED","name":"П-образные элементы","default":1,"cols":2,"variants":[{"name":"Нужны","value":1},{"name":"Не нужны","value":0}],"valueMode":"number"},{"type":"Radio","key":"MESH_ROWS","name":"Число рядов арматурной сетки","default":2,"cols":2,"variants":[{"name":"1 ряд","value":1},{"name":"2 ряда","value":2}],"valueMode":"number"},{"type":"Select","key":"CELL_STEP_MM","name":"Шаг ячейки, мм","default":250,"variants":[{"name":"100 мм","value":100},{"name":"150 мм","value":150},{"name":"200 мм","value":200},{"name":"250 мм","value":250},{"name":"300 мм","value":300},{"name":"350 мм","value":350},{"name":"400 мм","value":400},{"name":"450 мм","value":450},{"name":"500 мм","value":500},{"name":"550 мм","value":550},{"name":"600 мм","value":600}],"valueMode":"number"},{"type":"DataSelect","key":"REBAR_DIA_MM","name":"Диаметр арматуры, мм","default":"A500C d14","variants":[{"name":"Арматура А240 d3","value":3},{"name":"Арматура А240 d4","value":4},{"name":"Арматура А240 d5","value":5},{"name":"Арматура А240 d6","value":6},{"name":"Арматура А240 d8","value":8},{"name":"d10","value":10},{"name":"d12","value":12},{"name":"d14","value":14},{"name":"d16","value":16},{"name":"d18","value":18},{"name":"d20","value":20},{"name":"d22","value":22},{"name":"d25","value":25},{"name":"d28","value":28},{"name":"d32","value":32},{"name":"d36","value":36},{"name":"d40","value":40},{"name":"d45","value":45},{"name":"d50","value":50},{"name":"d55","value":55},{"name":"d60","value":60},{"name":"d70","value":70},{"name":"d80","value":80}],"data":"{\n  \"A240 d3\": {\"MAIN_W\": 0.055, \"MAIN_D\": 3},\n  \"A240 d4\": {\"MAIN_W\": 0.099, \"MAIN_D\": 4},\n  \"A240 d5\": {\"MAIN_W\": 0.154, \"MAIN_D\": 5},\n  \"A240 d6\": {\"MAIN_W\": 0.222, \"MAIN_D\": 6},\n  \"A240 d8\": {\"MAIN_W\": 0.395, \"MAIN_D\": 8},\n\n  \"A500C d10\": {\"MAIN_W\": 0.617, \"MAIN_D\": 10},\n  \"A500C d12\": {\"MAIN_W\": 0.888, \"MAIN_D\": 12},\n  \"A500C d14\": {\"MAIN_W\": 1.208, \"MAIN_D\": 14},\n  \"A500C d16\": {\"MAIN_W\": 1.578, \"MAIN_D\": 16},\n  \"A500C d18\": {\"MAIN_W\": 1.998, \"MAIN_D\": 18},\n  \"A500C d20\": {\"MAIN_W\": 2.466, \"MAIN_D\": 20},\n  \"A500C d22\": {\"MAIN_W\": 2.984, \"MAIN_D\": 22},\n  \"A500C d25\": {\"MAIN_W\": 3.853, \"MAIN_D\": 25},\n  \"A500C d28\": {\"MAIN_W\": 4.834, \"MAIN_D\": 28},\n  \"A500C d32\": {\"MAIN_W\": 6.313, \"MAIN_D\": 32},\n  \"A500C d36\": {\"MAIN_W\": 7.99, \"MAIN_D\": 36},\n  \"A500C d40\": {\"MAIN_W\": 9.865, \"MAIN_D\": 40},\n  \"A500C d45\": {\"MAIN_W\": 12.485, \"MAIN_D\": 45},\n  \"A500C d50\": {\"MAIN_W\": 15.413, \"MAIN_D\": 50},\n  \"A500C d55\": {\"MAIN_W\": 18.65, \"MAIN_D\": 55},\n  \"A500C d60\": {\"MAIN_W\": 22.195, \"MAIN_D\": 60},\n  \"A500C d70\": {\"MAIN_W\": 30.21, \"MAIN_D\": 70},\n  \"A500C d80\": {\"MAIN_W\": 39.458, \"MAIN_D\": 80}\n}"},{"type":"Radio","key":"FORMWORK_TYPE","name":"Опалубка","default":1,"cols":2,"variants":[{"name":"Обрезная доска","value":1,"image":"https://ins.bestnovels.ru/api/getFile?name=efbf77cf-990b-4592-a19b-a20f8142cd44"},{"name":"Ламинированная фанера","value":2,"image":"https://ins.bestnovels.ru/api/getFile?name=58bc8fd9-a159-4a66-bce4-54b8f0adccbf"}],"valueMode":"number"},{"type":"Select","key":"BOARD_THK_M","name":"Толщина доски (для опалубки), м","default":0.04,"variants":[{"name":"30 мм","value":0.03},{"name":"40 мм","value":0.04},{"name":"50 мм","value":0.05}],"valueMode":"number"},{"type":"Select","key":"INSUL_THK_M","name":"Толщина утеплителя, м","default":0.05,"variants":[{"name":"30 мм","value":0.03},{"name":"40 мм","value":0.04},{"name":"50 мм","value":0.05},{"name":"60 мм","value":0.06},{"name":"70 мм","value":0.07},{"name":"80 мм","value":0.08},{"name":"90 мм","value":0.09},{"name":"100 мм","value":0.1}],"valueMode":"number"},{"type":"Header","text":"Цены","bgColor":"#008000","textColor":""},{"type":"Input","key":"PRICE_CONCRETE_M3","name":"Стоимость бетона, ₽/м³","default":6000,"valueMode":"number"},{"type":"Input","key":"PRICE_REBAR","name":"Стоимость арматуры (по выбранной единице), ₽","default":70000,"valueMode":"number"},{"type":"Radio","key":"REBAR_PRICE_MODE","name":"Единица цены арматуры","default":1,"cols":2,"variants":[{"name":"₽/тонна","value":1},{"name":"₽/м.п.","value":2}],"valueMode":"number"},{"type":"Input","key":"PRICE_INSUL_M2","name":"Стоимость утеплителя, ₽/м²","default":750,"valueMode":"number"},{"type":"Input","key":"PRICE_FORMWORK","name":"Стоимость опалубки, ₽","default":21000,"valueMode":"number"},{"type":"Radio","key":"WORK_PRICE_MODE","name":"Единица цены бетонных работ","default":2,"cols":2,"variants":[{"name":"₽/м² плиты","value":1},{"name":"₽/м³ бетона","value":2}],"valueMode":"raw"},{"type":"Input","key":"PRICE_WORK_CONC","name":"Стоимость бетонных работ, ₽","default":12000,"valueMode":"number"},{"type":"Input","key":"PRICE_WORK_INSUL","name":"Стоимость монтажа утеплителя, ₽/м²","default":1000,"valueMode":"number"},{"type":"Calc","key":"THK_M","name":"Толщина, м","formula":"THK_MM/1000","valueMode":"number"},{"type":"Calc","key":"LEN_CM","name":"Длина, см","formula":"LEN_M*100","valueMode":"number"},{"type":"Calc","key":"WID_CM","name":"Ширина, см","formula":"WID_M*100","valueMode":"number"},{"type":"Calc","key":"LEN_MM","name":"Длина, мм","formula":"LEN_M*1000","valueMode":"number"},{"type":"Calc","key":"WID_MM","name":"Ширина, мм","formula":"WID_M*1000","valueMode":"number"},{"type":"Calc","key":"PERIM_M","name":"Периметр, м.п.","formula":"(LEN_M*2)+(WID_M*2)","valueMode":"number"},{"type":"Calc","key":"AREA_M2","name":"Площадь плиты, м²","formula":"LEN_M*WID_M","valueMode":"number"},{"type":"Calc","key":"VOL_M3","name":"Объём бетона, м³","formula":"(LEN_M*WID_M)*(THK_MM/1000)","valueMode":"number"},{"type":"Calc","key":"WEIGHT_T","name":"Вес плиты, т","formula":"VOL_M3*2.5","valueMode":"number"},{"type":"Calc","key":"LOAD_KG_CM2","name":"Нагрузка, кг/см²","formula":"(WEIGHT_T*1000)/(LEN_CM*WID_CM)","valueMode":"number"},{"type":"Calc","key":"SIDE_AREA_M2","name":"Площадь боковой поверхности, м²","formula":"PERIM_M*THK_M","valueMode":"number"},{"type":"Calc","key":"REBAR_W_KG_M","name":"Вес 1 м арматуры, кг/м","formula":"REBAR_DIA_MM===3?0.055:REBAR_DIA_MM===4?0.099:REBAR_DIA_MM===5?0.154:REBAR_DIA_MM===6?0.222:REBAR_DIA_MM===8?0.395:REBAR_DIA_MM===10?0.617:REBAR_DIA_MM===12?0.888:REBAR_DIA_MM===14?1.208:REBAR_DIA_MM===16?1.578:REBAR_DIA_MM===18?1.998:REBAR_DIA_MM===20?2.466:REBAR_DIA_MM===22?2.984:REBAR_DIA_MM===25?3.853:REBAR_DIA_MM===28?4.834:REBAR_DIA_MM===32?6.313:REBAR_DIA_MM===36?7.99:REBAR_DIA_MM===40?9.865:REBAR_DIA_MM===45?12.485:REBAR_DIA_MM===50?15.413:REBAR_DIA_MM===55?18.65:REBAR_DIA_MM===60?22.195:REBAR_DIA_MM===70?30.21:REBAR_DIA_MM===80?39.458:0","valueMode":"number"},{"type":"Calc","key":"REBAR_MAIN_M","name":"Основная арматура, м.п.","formula":"(((Math.ceil(WID_MM/CELL_STEP_MM)+1)*((LEN_M||0)+((REBAR_SEG_L||1)-1)*(REBAR_LAP_M||0)))+((Math.ceil(LEN_MM/CELL_STEP_MM)+1)*((WID_M||0)+((REBAR_SEG_W||1)-1)*(REBAR_LAP_M||0))))*(MESH_ROWS||1)","valueMode":"number"},{"type":"Calc","key":"REBAR_MAIN_KG","name":"Вес основной арматуры, кг","formula":"REBAR_MAIN_M*MAIN_W","valueMode":"number"},{"type":"Calc","key":"PU_COUNT","name":"Кол-во П-элементов, шт","formula":"(Math.ceil((LEN_MM*2+WID_MM*2))/CELL_STEP_MM)","valueMode":"number"},{"type":"Calc","key":"PU_LEN_1","name":"Длина 1 П-элемента, м","formula":"THK_M+(MAIN_D*(32/1000)*2)","valueMode":"number"},{"type":"Calc","key":"PU_TOTAL_M","name":"П-элементы, всего м.п.","formula":"PU_NEEDED===1?PU_LEN_1*PU_COUNT:0","valueMode":"number"},{"type":"Calc","key":"PU_TOTAL_KG","name":"Вес П-элементов, кг","formula":"PU_TOTAL_M*MAIN_W","valueMode":"number"},{"type":"Calc","key":"FORMWORK_M2","name":"Опалубка, площадь м²","formula":"SIDE_AREA_M2","valueMode":"number"},{"type":"Calc","key":"FORMWORK_M3","name":"Опалубка, объём леса м³","formula":"FORMWORK_M2*BOARD_THK_M","valueMode":"number"},{"type":"Calc","key":"INSUL_M2","name":"Утеплитель, м²","formula":"SIDE_AREA_M2","valueMode":"number"},{"type":"Calc","key":"INSUL_M3","name":"Утеплитель, м³","formula":"INSUL_M2*INSUL_THK_M","valueMode":"number"},{"type":"Calc","key":"COST_CONCRETE","name":"Стоимость бетона, ₽","formula":"VOL_M3*PRICE_CONCRETE_M3","valueMode":"number"},{"type":"Calc","key":"REBAR_QTY_FOR_PRICE","name":"Арматура, кол-во (для цены)","formula":"REBAR_PRICE_MODE===1?((REBAR_MAIN_KG+PU_TOTAL_KG)/1000):(REBAR_MAIN_M+PU_TOTAL_M)","valueMode":"number"},{"type":"Calc","key":"COST_REBAR","name":"Стоимость арматуры, ₽","formula":"REBAR_QTY_FOR_PRICE*PRICE_REBAR","valueMode":"number"},{"type":"Calc","key":"FORMWORK_QTY_FOR_PRICE","name":"Опалубка, кол-во (для цены)","formula":"FORMWORK_TYPE===1?FORMWORK_M3:FORMWORK_M2","valueMode":"number"},{"type":"Calc","key":"COST_FORMWORK","name":"Стоимость опалубки, ₽","formula":"FORMWORK_QTY_FOR_PRICE*PRICE_FORMWORK","valueMode":"number"},{"type":"Calc","key":"COST_INSUL","name":"Стоимость утеплителя, ₽","formula":"INSUL_M2*PRICE_INSUL_M2","valueMode":"number"},{"type":"Calc","key":"COST_MAT_TOTAL","name":"Итого материалы, ₽","formula":"COST_CONCRETE+COST_REBAR+COST_FORMWORK+COST_INSUL","valueMode":"number"},{"type":"Calc","key":"WORK_QTY_CONC","name":"Бетонные работы, кол-во (для цены)","formula":"WORK_PRICE_MODE===1?AREA_M2:VOL_M3","valueMode":"number"},{"type":"Calc","key":"COST_WORK_CONC","name":"Бетонные работы, ₽","formula":"WORK_QTY_CONC*PRICE_WORK_CONC","valueMode":"number"},{"type":"Calc","key":"COST_WORK_INSUL","name":"Монтаж утеплителя, ₽","formula":"OBUTEPL*PRICE_WORK_INSUL","valueMode":"number"},{"type":"Calc","key":"COST_WORK_TOTAL","name":"Итого работа, ₽","formula":"COST_WORK_CONC+COST_WORK_INSUL","valueMode":"number"},{"type":"Calc","key":"COST_TOTAL","name":"ВСЕГО, ₽","formula":"COST_MAT_TOTAL+COST_WORK_TOTAL","valueMode":"number"},{"type":"Result","title":"Результаты","data":[["Показатель","Значение"],["Периметр, м.п.","PERIM_M"],["Площадь плиты, м²","AREA_M2"],["Объём бетона, м³","VOL_M3"],["Вес плиты, т","WEIGHT_T"],["Нагрузка, кг/см²","LOAD_KG_CM2"],["Боковая поверхность, м²","SIDE_AREA_M2"],["Арматура основная, м.п.","REBAR_MAIN_M"],["Арматура П-элементы, м.п.","PU_TOTAL_M"],["Опалубка (площадь), м²","FORMWORK_M2"],["Опалубка (лес), м³","FORMWORK_M3"],["Утеплитель, м²","INSUL_M2"],["Итого материалы, ₽","COST_MAT_TOTAL"],["Итого работа, ₽","COST_WORK_TOTAL"],["ВСЕГО, ₽","COST_TOTAL"]]},{"type":"Result","key":"","name":"","default":0,"title":"Промежуточные результаты","data":[["Длина","LEN_M","м","Периметр","PERIM_M","м"],["Ширина","WID_M","м","Площадь","AREA_M2","м2"],["Толщина","THK_MM","мм","Вес","WEIGHT_T","тонн"]]},{"type":"Header","key":"","name":"","default":0,"text":"Расчет армирования плиты","bgColor":"#008000","textColor":"#fff"},{"type":"Image","key":"","name":"Фундаментная плита","default":0,"src":"https://ins.bestnovels.ru/api/getFile?name=79efe629-f8ea-4fa4-9d91-a0b72c710208"},{"type":"Image","key":"","name":"П-образные","default":0,"src":"https://ins.bestnovels.ru/api/getFile?name=6dc783e7-393c-43d8-8210-d958ed1c3a40"},{"type":"Result","key":"","name":"","default":0,"data":[["Наименование","Материал","Длина, м.п.","Вес, кг","Прочее"],["Основная арматура","REBAR_DIA_MM","REBAR_MAIN_M","REBAR_MAIN_KG",""]],"title":"Основная арматура"},{"type":"Result","key":"","name":"","default":0,"title":"Арматура на П- образные элементы","data":[["Наименование","Материал","Длина, м.п.","Вес, кг","Прочее"],["П-образные элементы","REBAR_DIA_MM","PU_TOTAL_M","PU_TOTAL_KG",""]]},{"type":"Calc","key":"TOTOAL_ARM_LEN","name":"Всего арматуры","default":0,"formula":"REBAR_MAIN_M+PU_TOTAL_M"},{"type":"Result","key":"","name":"","default":0,"title":"Всего арматуры","data":[["Наименование","Материал","Длина, м.п.","Вес, кг"],["ВСЕГО","REBAR_DIA_MM","TOTOAL_ARM_LEN","TOTAL_ARM_WEITH"]]},{"type":"Calc","key":"TOTAL_ARM_WEITH","name":"Всего арматуры, кг","default":0,"formula":"REBAR_MAIN_KG+PU_TOTAL_KG"},{"type":"Header","key":"","name":"","default":0,"text":"Расчет опалубки","bgColor":"#008000","textColor":"#fff"},{"type":"Result","key":"","name":"","default":0,"data":[["Наименование","Толщина, мм","Кол-во","Ед. изм."],["Обрезная доска","wood_tinkness","FORMWORK_M3","м3"]],"title":"Необходимое кол-во обрезной доски для опалубки"},{"type":"Calc","key":"wood_tinkness","name":"Толщина опалубки","default":0,"formula":"BOARD_THK_M*1000"},{"type":"Result","key":"","name":"","default":0,"title":"Необходимое кол-во ламинированной фанеры","data":[["Наименование","Кол-во","Ед. изм."],["Ламинированная фанера","FORMWORK_M2","м2"]]},{"type":"Header","key":"","name":"","default":0,"text":"Расчет стоимости","bgColor":"#008000","textColor":"#fff"},{"type":"Header","key":"","name":"","default":0,"text":"Материалы","bgColor":"#808080","textColor":"#fff"},{"type":"Header","key":"","name":"","default":0,"text":"Работа","bgColor":"#808080","textColor":"#fff"},{"type":"Header","key":"","name":"","default":0,"text":"ИТОГОВАЯ СМЕТА","bgColor":"#008000","textColor":"#fff"},{"type":"Header","key":"","name":"","default":0,"text":"Расчет утеплителя на боковую поверхность плиты","bgColor":"#008000","textColor":"#fff"},{"type":"Result","key":"","name":"","default":0,"data":[["Наименование","Толщина, мм","Площадь, м2","Объем, м3"],["Утеплитель","THINKE_UT","INSUL_M2","INSUL_M3"]],"title":"Расчет утеплителя"},{"type":"Calc","key":"THINKE_UT","name":"Толщина утеплителя","default":0,"formula":"INSUL_THK_M*1000","valueMode":"number"},{"type":"Result","key":"","name":"","default":0,"title":"СТОИМОСТЬ МАТЕРИАЛОВ","data":[["","","МАТЕРИАЛЫ","",""],["Наименование","кол-во","ед. изм.","цена","стоимость"],["","","","",""],["REBAR_DIA_MM","TOTAL_ARM_WEITH_T","тонн","PRICE_REBAR","COST_REBAR"],["Бетон","VOL_M3","м3","PRICE_CONCRETE_M3","COST_CONCRETE"],["Опалубка","FORMWORK_QTY_FOR_PRICE","м2","PRICE_FORMWORK","COST_FORMWORK"],["Утеплитель","INSUL_M2","м2","PRICE_INSUL_M2","COST_INSUL"],["","","","",""],["ИТОГО НА МАТЕРИАЛЫ","","","","TOTAL_MATER_1"]]},{"type":"Calc","key":"TOTAL_ARM_WEITH_T","name":"Всего арматуры","default":0,"formula":"TOTAL_ARM_WEITH/1000","valueMode":"number"},{"type":"Calc","key":"TOTAL_MATER_1","name":"ИТОГО МАТЕРИАЛЫ","default":0,"formula":"COST_REBAR+COST_CONCRETE+COST_FORMWORK+COST_INSUL","valueMode":"number"},{"type":"Result","key":"","name":"","default":0,"data":[["","","РАБОТА","",""],["Наименование","кол-во","ед.изм.","цена","стоимость"],["Монтаж арматуры и бетонирование","WORK_QTY_CONC","ED_IZM","PRICE_WORK_CONC","COST_WORK_CONC"],["Монтаж утеплителя","OBUTEPL","EDIZMUTEP","PRICE_WORK_INSUL","COST_WORK_INSUL"],["","","","",""],["ИТОГО НА РАБОТЫ","","","","COST_WORK_TOTAL"],["","","","",""],["ВСЕГО","","","","COST_TOTAL"]],"title":"СТОИМОСТЬ РАБОТ"},{"type":"Calc","key":"ED_IZM","name":"Единица объема работ ","default":0,"formula":"WORK_PRICE_MODE===1 ? \"м2\" : WORK_PRICE_MODE===2 ? \"м3\" : \"\""},{"type":"Image","key":"","name":"логотип","default":0,"src":"https://ins.bestnovels.ru/api/getFile?name=6740cb78-ed51-4e6e-9a1b-3c6955038a57"},{"type":"Radio","key":"PARAMUTEP","name":"Единица цены монтажа утеплителя","default":"1","valueMode":"raw","formula":"","variants":[{"name":"За м.п.","value":1,"image":""},{"name":"За м2","value":2,"image":""}],"cols":2},{"type":"Calc","key":"OBUTEPL","name":"Объем утепления","default":0,"valueMode":"number","formula":"PARAMUTEP===1 ? (PERIM_M||0) : PARAMUTEP===2 ? (INSUL_M2||0) : 0"},{"type":"Calc","key":"EDIZMUTEP","name":"Единица измерения утепления","default":0,"valueMode":"raw","formula":"PARAMUTEP===1?\"м.п.\":PARAMUTEP===2?\"м²\":\"\""},{"type":"Select","key":"REBAR_PIECE_LEN_M","name":"Длина 1-го прутка, м","default":"11.75","valueMode":"number","variants":[{"name":"5.875","value":5.875},{"name":"11.75","value":11.75}]},{"type":"Calc","key":"REBAR_LAP_M","name":"Куски","default":0,"valueMode":"number","formula":"(32*(MAIN_D||0))/1000"},{"type":"Calc","key":"REBAR_SEG_L","name":"Поле: количество сегментов стержня по длине","default":0,"valueMode":"number","formula":"(LEN_M||0)<=((REBAR_PIECE_LEN_M||0))?1:(((REBAR_PIECE_LEN_M||0)-(REBAR_LAP_M||0))>0?(Math.ceil(((LEN_M||0)-(REBAR_PIECE_LEN_M||0))/((REBAR_PIECE_LEN_M||0)-(REBAR_LAP_M||0)))+1):1)"},{"type":"Calc","key":"REBAR_SEG_W","name":"количество сегментов стержня по ширине","default":0,"valueMode":"number","formula":"WID_M<=REBAR_PIECE_LEN_M?1:Math.ceil((WID_M-REBAR_PIECE_LEN_M)/(REBAR_PIECE_LEN_M-REBAR_LAP_M))+1"}],"layout":[{"cols":[{"span":2,"items":[0]},{"span":1,"items":[80]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[57]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":1,"items":[1]},{"span":1,"items":[2]},{"span":1,"items":[3]}]},{"cols":[{"span":3,"items":[55]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[56]},{"span":0,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":1,"items":[7,6,5,84]},{"span":2,"items":[58,4]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[59]},{"span":1,"items":[]},{"span":1,"items":[]}],"showMode":"expr","showExpr":"PU_NEEDED===1"},{"cols":[{"span":3,"items":[60]},{"span":1,"items":[]},{"span":1,"items":[]}],"showMode":"expr","showExpr":"PU_NEEDED===1"},{"cols":[{"span":3,"items":[62]},{"span":1,"items":[]},{"span":1,"items":[]}],"showMode":"always","showExpr":""},{"cols":[{"span":3,"items":[64]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[8]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[9]},{"span":1,"items":[]},{"span":1,"items":[]}],"showMode":"expr","showExpr":"FORMWORK_TYPE===1"},{"cols":[{"span":3,"items":[65]},{"span":1,"items":[]},{"span":1,"items":[]}],"showMode":"expr","showExpr":"FORMWORK_TYPE===1"},{"cols":[{"span":3,"items":[67]},{"span":1,"items":[]},{"span":1,"items":[]}],"showMode":"expr","showExpr":"FORMWORK_TYPE===2"},{"cols":[{"span":3,"items":[72]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[10]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[73]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[68]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[69]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":2,"items":[13]},{"span":1,"items":[14]},{"span":0,"items":[]}]},{"cols":[{"span":1,"items":[12]},{"span":1,"items":[15]},{"span":1,"items":[16]}]},{"cols":[{"span":3,"items":[70]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":2,"items":[18,19]},{"span":2,"items":[17,81]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[71]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[75]},{"span":1,"items":[]},{"span":1,"items":[]}]},{"cols":[{"span":3,"items":[78]},{"span":1,"items":[]},{"span":1,"items":[]}]}]}";
var DEV = false;
var providers={"QueryParams":{render:(param)=>{
let { item , state } = param;
if (item.type !== "QueryParams") return null;
try {
var _window_location, _window_history;
if (!(window === null || window === void 0 ? void 0 : (_window_location = window.location) === null || _window_location === void 0 ? void 0 : _window_location.href)) return null;
if (!(window === null || window === void 0 ? void 0 : (_window_history = window.history) === null || _window_history === void 0 ? void 0 : _window_history.replaceState)) return null;
const url = new URL(window.location.href);
let changed = false;
(item.data || []).forEach((row)=>{
var _row_;
const paramKey = String((_row_ = row === null || row === void 0 ? void 0 : row[0]) !== null && _row_ !== void 0 ? _row_ : "").trim();
if (!paramKey) return;
const rawValue = row === null || row === void 0 ? void 0 : row[1];
let nextValue = "";
if (typeof rawValue === "string" && Object.prototype.hasOwnProperty.call(state || {}, rawValue)) {
nextValue = state[rawValue];
} else {
nextValue = rawValue;
}
nextValue = nextValue == null ? "" : String(nextValue);
if (url.searchParams.get(paramKey) !== nextValue) {
url.searchParams.set(paramKey, nextValue);
changed = true;
}
});
if (changed) {
window.history.replaceState(null, "", url.toString());
}
} catch (e) {
console.log("[CALC] QueryParams error", e);
}
return null;
}},"Header":{render:(param)=>{
let { item , h } = param;
if (item.type !== "Header") return null;
const style = {};
if (item.bgColor) {
style.background = item.bgColor;
style.padding = ".5rem .6rem";
style.borderRadius = ".6rem";
}
if (item.textColor) {
style.color = item.textColor;
}
return h("div", {
class: "header",
style
}, item.text || "");
}},"SaveResult":{render:(param)=>{
let { item , h } = param;
if (item.type !== "SaveResult") return null;
const style = {};
if (item.bgColor) style.background = item.bgColor;
if (item.textColor) style.color = item.textColor;
return h("button", {
class: "save-btn",
style,
onMouseEnter: (e)=>{
if (item.hoverBg) e.currentTarget.style.background = item.hoverBg;
},
onMouseLeave: (e)=>{
if (item.bgColor) e.currentTarget.style.background = item.bgColor;
else e.currentTarget.style.background = "";
},
onClick: handleClick
}, item.text || "Скачать");
async function handleClick() {
const root = document.querySelector(".cbs-calc-runtime");
if (!root) return;
try {
const html2canvas = await loadHtml2Canvas();
const calcCanvas = await html2canvas(root, {
backgroundColor: "#ffffff",
useCORS: true
});
const topImg = await loadImage(item.topImage);
const bottomImg = await loadImage(item.bottomImage);
const width = calcCanvas.width;
const topH = topImg ? Math.round(topImg.height * (width / topImg.width)) : 0;
const bottomH = bottomImg ? Math.round(bottomImg.height * (width / bottomImg.width)) : 0;
const height = topH + calcCanvas.height + bottomH;
const finalCanvas = document.createElement("canvas");
finalCanvas.width = width;
finalCanvas.height = height;
const ctx = finalCanvas.getContext("2d");
let y = 0;
if (topImg) {
ctx.drawImage(topImg, 0, y, width, topH);
y += topH;
}
ctx.drawImage(calcCanvas, 0, y);
y += calcCanvas.height;
if (bottomImg) {
ctx.drawImage(bottomImg, 0, y, width, bottomH);
}
const link = document.createElement("a");
link.download = "result.png";
link.href = finalCanvas.toDataURL("image/png");
link.click();
} catch (e) {
console.error(e);
alert("Ошибка сохранения");
}
}
function loadImage(src) {
return new Promise((res)=>{
if (!src) return res(null);
const img = new Image();
img.crossOrigin = "anonymous";
img.onload = ()=>res(img);
img.onerror = ()=>res(null);
img.src = src;
});
}
function loadHtml2Canvas() {
if (window.html2canvas) return Promise.resolve(window.html2canvas);
return new Promise((resolve, reject)=>{
const s = document.createElement("script");
s.src = "https://unpkg.com/html2canvas@1.4.1/dist/html2canvas.min.js";
s.onload = ()=>resolve(window.html2canvas);
s.onerror = reject;
document.head.appendChild(s);
});
}
}},"Image":{render:(param)=>{
let { item , h } = param;
if (item.type !== "Image") return null;
if (!item.src) return null;
let aspectClass = "";
if (item.aspect === "wide") aspectClass = "image-wide";
else if (item.aspect === "square") aspectClass = "image-square";
else aspectClass = "image-free";
const imgClass = item.aspect ? "image" : "image";
return h("div", {
class: "image-wrap " + aspectClass
}, h("img", {
class: imgClass,
src: item.src
}));
}},"Text":{render:(param)=>{
let { item , h } = param;
if (item.type !== "Text") return null;
if (!item.text) return null;
const parts = String(item.text).split(/\n+/);
return h("div", {
class: "text-block"
}, parts.map((line)=>h("p", {}, line)));
}},"DataSelect":{render:(param)=>{
let { item , state , set , get , h } = param;
if (item.type !== "DataSelect") return null;
let data = {};
try {
data = JSON.parse(item.data || "{}");
} catch (e) {}
const keys = Object.keys(data || {});
const def = item.default || keys[0] || "";
const value = get(item.key, def);
const row = data[value];
if (row && typeof row === "object") {
Object.entries(row).forEach((param)=>{
let [k, v] = param;
set(k, v);
});
}
return h("div", {
class: "field"
}, item.name ? h("div", {
class: "field-label"
}, item.name) : null, h("select", {
value: value || "",
onChange: (e)=>set(item.key, e.target.value)
}, keys.map((k)=>h("option", {
value: k
}, k))));
}},"Input":{render:(param)=>{
let { item , state , set , get , h } = param;
if (item.type !== "Input") return null;
var _item_default;
const value = get(item.key, (_item_default = item.default) !== null && _item_default !== void 0 ? _item_default : 0);
return h("div", {
class: "field"
}, item.name ? h("div", {
class: "field-label"
}, item.name) : null, h("input", {
value: value !== null && value !== void 0 ? value : "",
onInput: (e)=>set(item.key, e.target.value, item.valueMode)
}));
}},"Select":{render:(param)=>{
let { item , state , set , get , h } = param;
var _item_variants, _item_variants_;
if (item.type !== "Select") return null;
var _item_variants__value, _item_default;
const def = (_item_default = item.default) !== null && _item_default !== void 0 ? _item_default : (_item_variants__value = (_item_variants = item.variants) === null || _item_variants === void 0 ? void 0 : (_item_variants_ = _item_variants[0]) === null || _item_variants_ === void 0 ? void 0 : _item_variants_.value) !== null && _item_variants__value !== void 0 ? _item_variants__value : 0;
const value = get(item.key, def);
return h("div", {
class: "field"
}, item.name ? h("div", {
class: "field-label"
}, item.name) : null, h("select", {
value,
onChange: (e)=>set(item.key, e.target.value, item.valueMode)
}, (item.variants || []).map((v)=>h("option", {
value: v.value
}, v.name))));
}},"Radio":{render:(param)=>{
let { item , state , set , get , h } = param;
var _item_variants, _item_variants_;
if (item.type !== "Radio") return null;
var _item_variants__value, _item_default;
const def = (_item_default = item.default) !== null && _item_default !== void 0 ? _item_default : (_item_variants__value = (_item_variants = item.variants) === null || _item_variants === void 0 ? void 0 : (_item_variants_ = _item_variants[0]) === null || _item_variants_ === void 0 ? void 0 : _item_variants_.value) !== null && _item_variants__value !== void 0 ? _item_variants__value : 0;
const value = get(item.key, def);
return h("div", {}, item.name ? h("div", {
class: "field-label"
}, item.name) : null, h("div", {
class: "radio-group",
style: {
gridTemplateColumns: "repeat(" + (item.cols || 4) + ",1fr)"
}
}, (item.variants || []).map((v)=>{
const ratioClass = v.imageRatio === "wide" ? "radio-ratio-wide" : v.imageRatio === "square" ? "radio-ratio-square" : "";
return h("div", {
class: "radio-card" + (value === v.value ? " active" : ""),
onClick: ()=>set(item.key, v.value, item.valueMode)
}, v.image ? h("img", {
class: "radio-img " + ratioClass,
src: v.image
}) : null, h("div", {
class: "radio-card-label"
}, v.name));
})));
}},"Calc":{render:(param)=>{
let { item , calc , set } = param;
if (item.type !== "Calc") return null;
const v = calc(item.formula || "");
set(item.key, v, item.valueMode);
return null;
}},"Range":{render:(param)=>{
let { item , state , set , get , h } = param;
if (item.type !== "Range") return null;
var _item_min;
const min = Number((_item_min = item.min) !== null && _item_min !== void 0 ? _item_min : 0);
var _item_max;
const max = Number((_item_max = item.max) !== null && _item_max !== void 0 ? _item_max : 100);
var _item_step;
const step = Number((_item_step = item.step) !== null && _item_step !== void 0 ? _item_step : 1);
var _item_default;
const def = (_item_default = item.default) !== null && _item_default !== void 0 ? _item_default : min;
const stateValue = get(item.key, def);
const snapped = normalize(stateValue, min, max, step);
const inputValue = clampEdge(snapped, min, max, step);
return h("div", {
class: "field"
}, h("div", {
class: "range-head"
}, item.name ? h("div", {
class: "field-label"
}, item.name) : h("div", {}), h("div", {
class: "range-value"
}, inputValue)), h("input", {
type: "range",
class: "range-input",
min: min,
max: max,
step: "any",
value: inputValue,
onInput: (e)=>{
const raw = parseFloat(e.target.value) || 0;
const v = normalize(raw, min, max, step);
set(item.key, v);
}
}));
function normalize(v, min, max, step) {
if (v <= min) return min;
if (v >= max) return max;
const steps = Math.round((v - min) / step);
let snapped = min + steps * step;
return Number(snapped.toFixed(6));
}
function clampEdge(v, min, max, step) {
if (Math.abs(v - min) <= step) return min;
if (Math.abs(v - max) <= step) return max;
return v;
}
}},"Result":{render:(param)=>{
let { item , state , h } = param;
if (item.type !== "Result") return null;
const rows = [];
(item.data || []).forEach((row)=>{
const cols = (row || []).map((cell)=>{
const val = state[cell];
if (typeof val === "number" && isFinite(val)) {
const out = Number.isInteger(val) ? val : Number(val.toFixed(2));
return h("td", {}, out);
}
if (val !== undefined && val !== null) {
return h("td", {}, String(val));
}
return h("td", {}, cell);
});
rows.push(h("tr", {}, cols));
});
return h("div", {
class: "results-card"
}, item.title ? h("div", {
class: "header"
}, item.title) : null, h("table", {}, rows));
}}};
var EXEC_ORDER = [0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,56,57,58,64,68,69,70,71,72,80,81,84,20,21,22,23,24,25,26,27,31,66,74,79,83,85,35,34,30,28,42,49,86,87,36,38,40,29,50,32,37,39,41,47,82,33,61,45,51,43,63,46,52,44,76,48,77,53,54,55,59,60,62,65,67,73,75,78];
const ctx = await bootstrap();
if(!ctx) return;
startApp(ctx);
async function bootstrap(){
const ctx={
host:null,
assets:null,
mount:null,
schema:null,
dev:createDevAdapter()
};
if(!prepareHost()) return null;
await loadDeps();
ctx.schema=parseSchema();
if(!ctx.schema){
ctx.mount.textContent="Schema parse error";
return null;
}
if(DEV) initDevTools();
return ctx;
function prepareHost(){
const cs=document.currentScript;
const host=cs && cs.parentElement;
if(!host || host.__calc_initialized) return false;
host.__calc_initialized=true;
ctx.host=host;
function ensureZone(cls){
let el=host.querySelector("."+cls);
if(!el){
el=document.createElement("div");
el.className=cls;
host.appendChild(el);
}else if(cls==="calc-app"){
el.innerHTML="";
}
return el;
}
ctx.assets=ensureZone("calc-assets");
ctx.mount=ensureZone("calc-app");
return true;
}
async function loadDeps(){
const deps=[
{key:"preact",url:"https://unpkg.com/preact@10.19.3/dist/preact.umd.js"},
{key:"preactHooks",url:"https://unpkg.com/preact@10.19.3/hooks/dist/hooks.umd.js"}
];
async function load(src){
return new Promise(function(res,rej){
const s=document.createElement("script");
s.src=src;
s.onload=res;
s.onerror=rej;
ctx.assets.appendChild(s);
});
}
for(const d of deps){
if(!window[d.key]) await load(d.url);
}
}
function parseSchema(){
if(!RAW_SCHEMA) return null;
try{
return JSON.parse(decodeURIComponent(escape(atob(RAW_SCHEMA))));
}catch(e){
console.error("[CALC] schema parse error",e);
return null;
}
}
function createDevAdapter(){
return {
elementClick(){},
blockClick(){},
showFormulaErrors(){}
};
}
function initDevTools(){
function highlight(el){
if(!el) return;
const prev=ctx.host.querySelector(".calc-highlight");
if(prev) prev.classList.remove("calc-highlight");
el.classList.add("calc-highlight");
setTimeout(()=>el.classList.remove("calc-highlight"),1200);
}
function scrollTo(el){
const rect=el.getBoundingClientRect();
const y=window.scrollY || document.documentElement.scrollTop;
const top=rect.top+y-(window.innerHeight/2)+(rect.height/2);
window.scrollTo({top,behavior:"smooth"});
highlight(el);
}
function findScrollParent(el){
let node=el;
while(node){
if(node.scrollHeight>node.clientHeight) return node;
node=node.parentElement||node.parentNode;
}
return document.scrollingElement||document.documentElement;
}
ctx.dev.elementClick=i=>window.parent.postMessage({type:"calc-click",index:i},"*");
ctx.dev.blockClick=i=>window.parent.postMessage({type:"calc-block-click",index:i},"*");
ctx.dev.showFormulaErrors=function(errors){
const id="calc-dev-banner";
const old=ctx.assets.querySelector("#"+id);
if(old) old.remove();
if(!errors?.length) return;
const first=errors[0];
const bar=document.createElement("div");
bar.id=id;
bar.className="calc-dev-banner";
bar.textContent="⚠ Ошибка формулы ("+errors.length+") — нажмите чтобы перейти";
bar.onclick=()=>window.parent.postMessage({
type:"calc-dev-error",
index:first.index
},"*");
ctx.assets.appendChild(bar);
};
window.addEventListener("message",function(ev){
const msg=ev?.data||{};
if(msg.type==="calc-scroll"){
const el=ctx.host.querySelector('[data-el-index="'+msg.index+'"]');
if(el) scrollTo(el);
}
if(msg.type==="calc-block-scroll"){
const el=ctx.host.querySelector('[data-block-index="'+msg.index+'"]');
if(!el) return;
const box=findScrollParent(el);
if(box){
box.scrollTo({
top:el.offsetTop-box.clientHeight/2+el.clientHeight/2,
behavior:"smooth"
});
}
highlight(el);
}
});
}
}
function startApp(ctx){
const {h,render} = window.preact;
const {useState} = window.preactHooks;
const schema = ctx.schema;
function App(){
const [state,setState] = useState({});
function set(key,val,mode){
const v = mode==="number" ? normalizeNumber(val) : val;
setState(prev=>{
if(prev[key]===v) return prev;
const next={...prev};
next[key]=v;
return next;
});
}
function get(key,def){
if(state[key]===undefined && def!==undefined){
set(key,def,typeof def==="number"?"number":"raw");
return def;
}
return state[key];
}
const elements = runRenders();
return renderLayout(elements);
function runRenders(){
const out={};
const list=schema.elements||[];
for(const i of EXEC_ORDER){
const item=list[i];
const p=item && providers[item.type];
if(!p||!p.render) continue;
const vnode=p.render({
item,state,set,get,
num:k=>normalizeNumber(state[k]),
calc:e=>calcExpr(e,state),
h
});
if(vnode){
out[i]=h('div',{
class:'el-wrap',
'data-el-index':i,
onClickCapture: DEV ? ()=>ctx.dev.elementClick(i) : null
},vnode);
}
}
return out;
}
function renderLayout(elements){
return h('div',{class:'cbs-calc-runtime'},
(schema.layout||[]).map((b,i)=>renderBlock(b,i,elements))
);
}
function renderBlock(block,i,elements){
if(block.showMode==="expr" && block.showExpr){
try{ if(!calcExpr(block.showExpr,state)) return null; }
catch{ return null; }
}
const cols=(block.cols&&block.cols.length)?block.cols:[{span:1,items:[]}];
const grid='position:relative;grid-template-columns:repeat('+cols.length+',1fr);';
const hit = DEV ? h('div',{
class:'calc-dev-block-hit',
onClickCapture:e=>{
e.stopPropagation();
ctx.dev.blockClick(i);
}
}) : null;
const gridVNode = h('div',{class:'grid',style:grid},
cols.map((c,k)=>renderCol(c,k,elements))
);
if((block.viewMode||"full")==="dropdown"){
return h('details',{
key:i,class:'calc-block','data-block-index':i,style:'position:relative;'
},
hit,
h('summary',{class:'calc-dd-summary calc-dd-'+i},
block.dropdownTitle||"Подробнее"
),
gridVNode
);
}
return h('div',{
class:'grid calc-block',
key:i,
'data-block-index':i,
style:grid
},hit,cols.map((c,k)=>renderCol(c,k,elements)));
}
function renderCol(col,c,elements){
let span = Number(col?.span ?? 1);
if(span<1) span=1;
return h('div',{class:'col-span-'+span,key:c},
(col.items||[]).map(i=>elements[i]||null)
);
}
}
render(h(App),ctx.mount);
function calcExpr(expr,state){
try{
if(!expr) return 0;
if(expr[0]==='=') expr=expr.slice(1);
const k=Object.keys(state);
const v=k.map(x=>state[x]);
return Function(...k,"return "+expr)(...v);
}catch(e){
console.error("[CALC] formula error",expr,e);
return 0;
}
}
function normalizeNumber(v){
if(v==null) return 0;
if(typeof v==="number") return isFinite(v)?v:0;
const n=parseFloat(String(v).replace(",","."));
return isFinite(n)?n:0;
}
}
})();
Lorem
test
Показать калькулятор