function Inspector_Custom_Renderer(draw) : widget() constructor { h = 64; self.draw = draw; } function Panel_Inspector() : PanelContent() constructor { context_str = "Inspector"; inspecting = noone; top_bar_h = ui(96); prop_hover = noone; prop_selecting = noone; function initSize() { content_w = w - ui(28); content_h = h - top_bar_h - ui(12); } initSize(); keyframe_dragging = noone; keyframe_drag_st = 0; anim_toggling = false; min_w = ui(160); lineBreak = true; tb_node_name = new textBox(TEXTBOX_INPUT.text, function(txt) { if(inspecting) inspecting.name = txt; }) tb_prop_filter = new textBox(TEXTBOX_INPUT.text, function(txt) { filter_text = txt; }) filter_text = ""; addHotkey("Inspector", "Copy property", "C", MOD_KEY.ctrl, function() { propSelectCopy(); }); addHotkey("Inspector", "Paste property", "V", MOD_KEY.ctrl, function() { propSelectPaste(); }); addHotkey("Inspector", "Toggle animation", "I", MOD_KEY.none, function() { anim_toggling = true; }); group_menu = [ [ "Expand all", function() { if(inspecting == noone) return; if(inspecting.input_display_list == -1) return; var dlist = inspecting.input_display_list; for( var i = 0; i < array_length(dlist); i++ ) { if(!is_array(dlist[i])) continue; dlist[i][@ 1] = false; } }], [ "Collapse all", function() { if(inspecting == noone) return; if(inspecting.input_display_list == -1) return; var dlist = inspecting.input_display_list; for( var i = 0; i < array_length(dlist); i++ ) { if(!is_array(dlist[i])) continue; dlist[i][@ 1] = true; } }], ] function onResize() { initSize(); contentPane.resize(content_w, content_h); } contentPane = new scrollPane(content_w, content_h, function(_y, _m) { var con_w = contentPane.surface_w - ui(4); draw_clear_alpha(COLORS.panel_bg_clear, 0); if(point_in_rectangle(_m[0], _m[1], 0, 0, con_w, content_h) && mouse_press(mb_left, pFOCUS)) prop_selecting = noone; if(inspecting == noone) return 0; inspecting.inspecting = true; prop_hover = noone; var jun = noone; var amo = inspecting.input_display_list == -1? ds_list_size(inspecting.inputs) : array_length(inspecting.input_display_list); var hh = ui(8); var _hover = pHOVER && contentPane.hover; for(var i = 0; i < amo; i++) { var xc = con_w / 2; var yy = hh + _y; if(inspecting.input_display_list == -1) { jun = inspecting.inputs[| i]; } else { if(i >= array_length(inspecting.input_display_list)) break; var jun_disp = inspecting.input_display_list[i]; if(is_array(jun_disp)) { var txt = jun_disp[0]; var coll = jun_disp[1] && filter_text == ""; if(_hover && point_in_rectangle(_m[0], _m[1], 0, yy, con_w, yy + ui(32))) { draw_sprite_stretched_ext(THEME.group_label, 0, 0, yy, con_w, ui(32), COLORS.panel_inspector_group_hover, 1); if(mouse_press(mb_left, pFOCUS)) jun_disp[@ 1] = !coll; if(mouse_press(mb_right, pFOCUS)) menuCall(, , group_menu); } else draw_sprite_stretched_ext(THEME.group_label, 0, 0, yy, con_w, ui(32), COLORS.panel_inspector_group_bg, 1); if(filter_text == "") { draw_sprite_ui(THEME.arrow, 0, ui(16), yy + ui(32) / 2, 1, 1, -90 + coll * 90, COLORS.panel_inspector_group_bg, 1); } draw_set_text(f_p0, fa_left, fa_center, COLORS._main_text); draw_text(ui(32), yy + ui(32) / 2, txt); hh += ui(32 + 8); if(coll) { var j = i + 1; while(j < amo) { if(j >= array_length(inspecting.input_display_list)) break; var j_jun = inspecting.input_display_list[j]; if(is_array(j_jun)) break; else j++; } i = j - 1; continue; } continue; } else if(is_struct(jun_disp) && instanceof(jun_disp) == "Inspector_Custom_Renderer") { if(pFOCUS) jun_disp.register(contentPane); jun_disp.rx = ui(16) + x; jun_disp.ry = top_bar_h + y; hh += jun_disp.draw(ui(6), yy, con_w - ui(12), _m, _hover, pFOCUS) + ui(8); continue; } jun = inspecting.inputs[| inspecting.input_display_list[i]]; } if(!is_struct(jun) || instanceof(jun) != "NodeValue") continue; if(!jun.show_in_inspector || jun.type == VALUE_TYPE.object) continue; if(filter_text != "") { var pos = string_pos(filter_text, string_lower(jun.name)); if(pos == 0) continue; } var lb_h = line_height(f_p0) + ui(8); var lb_y = yy + lb_h / 2; var butx = ui(16); if(jun.isAnimable()) { var index = jun.value_from == noone? jun.animator.is_anim : 2; draw_sprite_ui_uniform(THEME.animate_clock, index, butx, lb_y, 1,, 0.8); if(_hover && point_in_circle(_m[0], _m[1], butx, lb_y, ui(10))) { draw_sprite_ui_uniform(THEME.animate_clock, index, butx, lb_y, 1,, 1); TOOLTIP = jun.value_from == noone? "Toggle animation" : "Remove link"; if(mouse_press(mb_left, pFOCUS)) { if(jun.value_from != noone) jun.removeFrom(); else jun.animator.is_anim = !jun.animator.is_anim; PANEL_ANIMATION.updatePropertyList(); } } } butx += ui(20); index = jun.visible; draw_sprite_ui_uniform(THEME.junc_visible, index, butx, lb_y, 1,, 0.8); if(_hover && point_in_circle(_m[0], _m[1], butx, lb_y, ui(10))) { draw_sprite_ui_uniform(THEME.junc_visible, index, butx, lb_y, 1,, 1); TOOLTIP = "Visibility"; if(mouse_press(mb_left, pFOCUS)) jun.visible = !jun.visible; } draw_set_text(f_p0, fa_left, fa_center, COLORS._main_text); draw_text_add(ui(56), lb_y - ui(2), jun.name); var lb_w = string_width(jun.name) + ui(32); #region tooltip if(jun.tooltip != "") { var tx = ui(56) + string_width(jun.name) + ui(16); var ty = lb_y - ui(1); if(point_in_circle(_m[0], _m[1], tx, ty, ui(10))) { TOOLTIP = jun.tooltip; draw_sprite_ui(THEME.info, 0, tx, ty,,,, COLORS._main_icon_light, 1); } else draw_sprite_ui(THEME.info, 0, tx, ty,,,, COLORS._main_icon_light, 0.75); } #endregion #region anim if(lineBreak && jun.animator.is_anim) { var bx = w - ui(64); var by = lb_y; if(buttonInstant(THEME.button_hide, bx - ui(12), by - ui(12), ui(24), ui(24), _m, pFOCUS, _hover, "", THEME.prop_keyframe, 2) == 2) { for(var j = 0; j < ds_list_size(jun.animator.values); j++) { var _key = jun.animator.values[| j]; if(_key.time > ANIMATOR.current_frame) { ANIMATOR.setFrame(_key.time); break; } } } bx -= ui(26); var cc = COLORS.panel_animation_keyframe_unselected; var kfFocus = false; for(var j = 0; j < ds_list_size(jun.animator.values); j++) { if(jun.animator.values[| j].time == ANIMATOR.current_frame) { cc = COLORS.panel_animation_keyframe_selected; kfFocus = true; break; } } if(buttonInstant(THEME.button_hide, bx - ui(12), by - ui(12), ui(24), ui(24), _m, pFOCUS, _hover, kfFocus? "Remove keyframe" : "Add keyframe", THEME.prop_keyframe, 1, cc) == 2) { var _add = false; for(var j = 0; j < ds_list_size(jun.animator.values); j++) { var _key = jun.animator.values[| j]; if(_key.time == ANIMATOR.current_frame) { if(ds_list_size(jun.animator.values) > 1) ds_list_delete(jun.animator.values, j); _add = true; break; } else if(_key.time > ANIMATOR.current_frame) { ds_list_insert(jun.animator.values, j, new valueKey(ANIMATOR.current_frame, jun.showValue(), jun.animator)); _add = true; break; } } if(!_add) ds_list_add(jun.animator.values, new valueKey(ANIMATOR.current_frame, jun.showValue(), jun.animator)); } bx -= ui(26); if(buttonInstant(THEME.button_hide, bx - ui(12), by - ui(12), ui(24), ui(24), _m, pFOCUS, _hover, "", THEME.prop_keyframe, 0) == 2) { var _t = -1; for(var j = 0; j < ds_list_size(jun.animator.values); j++) { var _key = jun.animator.values[| j]; if(_key.time < ANIMATOR.current_frame) { _t = _key.time; } } if(_t > -1) ANIMATOR.setFrame(_t); } var lhf = lb_h / 2 - 4; draw_set_color(COLORS.panel_inspector_key_separator); draw_line(bx - ui(20), by - lhf, bx - ui(20), by + lhf); draw_set_color(COLORS.panel_inspector_key_separator); draw_line(bx - ui(20), by - lhf, bx - ui(20), by + lhf); bx -= ui(26 + 12); if(buttonInstant(THEME.button_hide, bx - ui(12), by - ui(12), ui(24), ui(24), _m, pFOCUS, _hover, "Looping mode " + ON_END_NAME[jun.on_end], THEME.prop_on_end, jun.on_end) == 2) jun.on_end = safe_mod(jun.on_end + 1, sprite_get_number(THEME.prop_on_end)); } #endregion var _hsy = yy + lb_h; var padd = ui(8); var labelWidth = max(lb_w, min(ui(80) + w * 0.2, ui(200))); var editBoxW = w - ui(16 + 48) - !lineBreak * labelWidth; var editBoxH = lineBreak? TEXTBOX_HEIGHT : lb_h; var editBoxX = ui(16) + !lineBreak * labelWidth; var editBoxY = lineBreak? _hsy : yy; var widH = lineBreak? editBoxH : 0; if(jun.editWidget) { jun.editWidget.setInteract(jun.value_from == noone); jun.editWidget.setActiveFocus(pFOCUS, _hover); if(pFOCUS) jun.editWidget.register(contentPane); switch(jun.display_type) { case VALUE_DISPLAY.button : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, _m); break; default : switch(jun.type) { case VALUE_TYPE.float : case VALUE_TYPE.integer : switch(jun.display_type) { case VALUE_DISPLAY._default : case VALUE_DISPLAY.range : case VALUE_DISPLAY.vector : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m); break; case VALUE_DISPLAY.vector_range : var ebH = jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m); widH = lineBreak? ebH : ebH - lb_h; break; case VALUE_DISPLAY.enum_scroll : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, array_safe_get(jun.display_data, jun.showValue()), _m, ui(16) + x, top_bar_h + y); break; case VALUE_DISPLAY.enum_button : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m, ui(16) + x, top_bar_h + y); break; case VALUE_DISPLAY.padding : jun.editWidget.draw(xc, _hsy + ui(32), jun.showValue(), _m); widH = ui(192); break; case VALUE_DISPLAY.rotation : case VALUE_DISPLAY.rotation_range : jun.editWidget.draw(xc, _hsy, jun.showValue(), _m); widH = ui(96); break; case VALUE_DISPLAY.slider : case VALUE_DISPLAY.slider_range : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m); break; case VALUE_DISPLAY.area : jun.editWidget.draw(xc, _hsy + ui(40), jun.showValue(), jun.extra_data, _m); widH = ui(204); break; case VALUE_DISPLAY.puppet_control : widH = jun.editWidget.draw(editBoxX, editBoxY, editBoxW, jun.showValue(), _m, ui(16) + x, top_bar_h + y); break; case VALUE_DISPLAY.kernel : var ebH = jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m); widH = lineBreak? ebH : ebH - lb_h; break; } break; case VALUE_TYPE.boolean : editBoxX = lineBreak? editBoxX : (labelWidth + con_w) / 2; jun.editWidget.draw(editBoxX, editBoxY, jun.showValue(), _m, editBoxH); break; case VALUE_TYPE.color : switch(jun.display_type) { case VALUE_DISPLAY.gradient : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), jun.extra_data, _m); break; default : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m); break; } break; case VALUE_TYPE.path : switch(jun.display_type) { case VALUE_DISPLAY.path_load : case VALUE_DISPLAY.path_save : jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m); break; case VALUE_DISPLAY.path_array : var val = jun.showValue(), txt = ""; var pathExist = jun.value_validation == VALIDATION.pass; if(is_array(val) && array_length(val)) txt = "[" + string(array_length(val)) + "] " + val[0]; else txt = string(val); jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, _m,, pathExist? COLORS._main_text : COLORS._main_value_negative); var icx = editBoxX + editBoxW - ui(16); var icy = editBoxY + editBoxH / 2; draw_sprite_ui_uniform(pathExist? THEME.button_path_icon : THEME.button_path_not_found_icon, 0, icx, icy, 1,, 1); draw_set_text(f_p0, fa_left, fa_center, COLORS._main_text); draw_text_cut(editBoxX + ui(8), editBoxY + editBoxH / 2, txt, editBoxW - ui(60)); if(!pathExist && _hover && point_in_rectangle(_m[0], _m[1], icx - ui(17), icy - ui(17), icx + ui(17), icy + ui(17))) TOOLTIP = "File not exist"; break; case VALUE_DISPLAY.path_font : var val = jun.showValue(); if(file_exists(val)) val = filename_name(val); jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, val, _m, ui(16) + x, top_bar_h + y); break; } break; case VALUE_TYPE.surface : editBoxH = ui(96); jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m, ui(16) + x, top_bar_h + y); widH = lineBreak? editBoxH : editBoxH - lb_h; break; case VALUE_TYPE.curve : editBoxH = ui(132); jun.editWidget.draw(ui(32), _hsy, w - ui(80), editBoxH, jun.showValue(), _m); widH = editBoxH; break; case VALUE_TYPE.text : var _hh = 0; if(instanceof(jun.editWidget) == "textBox") _hh = jun.editWidget.draw(editBoxX, editBoxY, editBoxW, editBoxH, jun.showValue(), _m, jun.display_type); else if(instanceof(jun.editWidget) == "textArea") _hh = jun.editWidget.draw(ui(16), _hsy, w - ui(16 + 48), editBoxH, jun.showValue(), _m, jun.display_type); widH = _hh; break; } } } else if(jun.display_type == VALUE_DISPLAY.label) { draw_set_text(f_p1, fa_left, fa_top, COLORS._main_text_sub); draw_text_add(ui(32), _hsy, jun.display_data); widH = string_height(jun.display_data); } else { widH = 0; } if(false) { var __hsy = yy; draw_set_alpha(0.5); draw_set_color(c_aqua); draw_rectangle(ui(8), __hsy, contentPane.w - ui(16), __hsy + lb_h, 0); __hsy += lb_h; draw_set_color(c_red); draw_rectangle(ui(8), __hsy, contentPane.w - ui(16), __hsy + widH, 0); __hsy += widH; draw_set_color(c_lime); draw_rectangle(ui(8), __hsy, contentPane.w - ui(16), __hsy + padd, 0); draw_set_alpha(1); } hh += lb_h + widH + padd; var _selY = yy - ui(0); var _selY1 = yy + lb_h + widH + ui(2); var _selH = _selY1 - _selY; if(_hover && point_in_rectangle(_m[0], _m[1], 4, _selY, contentPane.surface_w - ui(4), _selY + _selH)) { draw_sprite_stretched_ext(THEME.prop_selecting, 0, 4, _selY, contentPane.surface_w - ui(8), _selH, COLORS._main_accent, 1); if(anim_toggling) { jun.animator.is_anim = !jun.animator.is_anim; PANEL_ANIMATION.updatePropertyList(); anim_toggling = false; } prop_hover = jun; if(mouse_press(mb_left, pFOCUS)) prop_selecting = jun; if(mouse_press(mb_right, pFOCUS)) { __dialog_junction = jun; var dia = dialogCall(o_dialog_menubox, mouse_mx, mouse_my); var menuItem = [ [ "Reset value", function() { __dialog_junction.setValue(__dialog_junction.def_val); }], [ __dialog_junction.animator.is_anim? "Remove animation" : "Add animation", function() { __dialog_junction.animator.is_anim = !__dialog_junction.animator.is_anim; PANEL_ANIMATION.updatePropertyList(); }], -1, [ "Copy", function() { clipboard_set_text(__dialog_junction.getShowString()); }, ["Inspector", "Copy property"]], [ "Paste", function() { __dialog_junction.setString(clipboard_get_text()); }, ["Inspector", "Paste property"]], ]; if(jun.extract_node != "") { array_insert(menuItem, 2, [ "Extract to node", function() { __dialog_junction.extractNode(); }]); } dia.setMenu( menuItem ); } } } return hh; }); function propSelectCopy() { if(!prop_selecting) return; clipboard_set_text(prop_selecting.getShowString()); } function propSelectPaste() { if(!prop_selecting) return; prop_selecting.setString(clipboard_get_text()); } function drawInspectingNode() { tb_node_name.font = f_h5; tb_node_name.hide = true; tb_node_name.active = pFOCUS; tb_node_name.hover = pHOVER; tb_node_name.align = fa_center; tb_node_name.draw(ui(64), ui(14), w - ui(128), ui(32), inspecting.name, [mx, my]); tb_prop_filter.auto_update = true; tb_prop_filter.font = f_p0; tb_prop_filter.color = COLORS._main_text_sub; tb_prop_filter.active = pFOCUS; tb_prop_filter.hover = pHOVER; tb_prop_filter.align = fa_center; tb_prop_filter.draw(ui(64), ui(52), w - ui(128), ui(28), filter_text, [mx, my]); draw_sprite_ui(THEME.search, 0, ui(80), ui(52 + 14), 1, 1, 0, COLORS._main_icon, 1); var bx = ui(8); var by = ui(12); if(buttonInstant(THEME.button_hide, bx, by, ui(32), ui(32), [mx, my], pFOCUS, pHOVER, "Presets", THEME.preset, 1) == 2) dialogCall(o_dialog_preset, x + bx, y + by + ui(36), { "node": inspecting }); var bx = w - ui(44); var by = ui(12); if(inspecting.inspectorUpdate != noone) { if(buttonInstant(THEME.button_hide, bx, by, ui(32), ui(32), [mx, my], pFOCUS, pHOVER, "Execute node", THEME.sequence_control, 1, COLORS._main_value_positive) == 2) inspecting.inspectorUpdate(); } else draw_sprite_ui(THEME.sequence_control, 1, bx + ui(16), by + ui(16),,,, COLORS._main_icon_dark); if(inspecting.use_cache) { var bx = w - ui(44); var by = ui(12 + 36); if(buttonInstant(THEME.button_hide, bx, by, ui(32), ui(32), [mx, my], pFOCUS, pHOVER, "This node cache output for performance.\nClick to clear all cached frames in this node.", THEME.cache) = 2) inspecting.clearCache(); } } function drawContent(panel) { draw_clear_alpha(COLORS.panel_bg_clear, 0); lineBreak = w < PREF_MAP[? "inspector_line_break_width"]; draw_sprite_stretched(THEME.ui_panel_bg, 1, ui(8), top_bar_h - ui(8), w - ui(16), h - top_bar_h); if(inspecting) drawInspectingNode(); contentPane.active = pHOVER; contentPane.draw(ui(16), top_bar_h, mx - ui(16), my - top_bar_h); if(PANEL_GRAPH.node_focus && inspecting != PANEL_GRAPH.node_focus) { inspecting = PANEL_GRAPH.node_focus; if(inspecting != noone) inspecting.onInspect(); contentPane.scroll_y = 0; contentPane.scroll_y_to = 0; } } }