123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622 |
- #include "visual_script_editor.h"
- #include "tools/editor/editor_node.h"
- #include "visual_script_nodes.h"
- #include "visual_script_flow_control.h"
- #include "visual_script_func_nodes.h"
- #include "os/input.h"
- #include "os/keyboard.h"
- class VisualScriptEditorSignalEdit : public Object {
- OBJ_TYPE(VisualScriptEditorSignalEdit,Object)
- StringName sig;
- public:
- UndoRedo *undo_redo;
- Ref<VisualScript> script;
- protected:
- static void _bind_methods() {
- ObjectTypeDB::bind_method("_sig_changed",&VisualScriptEditorSignalEdit::_sig_changed);
- }
- void _sig_changed() {
- _change_notify();
- }
- bool _set(const StringName& p_name, const Variant& p_value) {
- if (sig==StringName())
- return false;
- if (p_name=="argument_count") {
- int new_argc=p_value;
- int argc = script->custom_signal_get_argument_count(sig);
- if (argc==new_argc)
- return true;
- undo_redo->create_action("Change Signal Arguments");
- if (new_argc < argc) {
- for(int i=new_argc;i<argc;i++) {
- undo_redo->add_do_method(script.ptr(),"custom_signal_remove_argument",sig,new_argc);
- undo_redo->add_undo_method(script.ptr(),"custom_signal_add_argument",sig,script->custom_signal_get_argument_name(sig,i),script->custom_signal_get_argument_type(sig,i),-1);
- }
- } else if (new_argc>argc) {
- for(int i=argc;i<new_argc;i++) {
- undo_redo->add_do_method(script.ptr(),"custom_signal_add_argument",sig,Variant::NIL,"arg"+itos(i+1),-1);
- undo_redo->add_undo_method(script.ptr(),"custom_signal_remove_argument",sig,argc);
- }
- }
- undo_redo->add_do_method(this,"_sig_changed");
- undo_redo->add_undo_method(this,"_sig_changed");
- undo_redo->commit_action();
- return true;
- }
- if (String(p_name).begins_with("argument/")) {
- int idx = String(p_name).get_slice("/",1).to_int()-1;
- ERR_FAIL_INDEX_V(idx,script->custom_signal_get_argument_count(sig),false);
- String what = String(p_name).get_slice("/",2);
- if (what=="type") {
- int old_type = script->custom_signal_get_argument_type(sig,idx);
- int new_type=p_value;
- undo_redo->create_action("Change Argument Type");
- undo_redo->add_do_method(script.ptr(),"custom_signal_set_argument_type",sig,idx,new_type);
- undo_redo->add_undo_method(script.ptr(),"custom_signal_set_argument_type",sig,idx,old_type);
- undo_redo->commit_action();
- return true;
- }
- if (what=="name") {
- String old_name = script->custom_signal_get_argument_name(sig,idx);
- String new_name=p_value;
- undo_redo->create_action("Change Argument name");
- undo_redo->add_do_method(script.ptr(),"custom_signal_set_argument_name",sig,idx,new_name);
- undo_redo->add_undo_method(script.ptr(),"custom_signal_set_argument_name",sig,idx,old_name);
- undo_redo->commit_action();
- return true;
- }
- }
- return false;
- }
- bool _get(const StringName& p_name,Variant &r_ret) const {
- if (sig==StringName())
- return false;
- if (p_name=="argument_count") {
- r_ret = script->custom_signal_get_argument_count(sig);
- return true;
- }
- if (String(p_name).begins_with("argument/")) {
- int idx = String(p_name).get_slice("/",1).to_int()-1;
- ERR_FAIL_INDEX_V(idx,script->custom_signal_get_argument_count(sig),false);
- String what = String(p_name).get_slice("/",2);
- if (what=="type") {
- r_ret = script->custom_signal_get_argument_type(sig,idx);
- return true;
- }
- if (what=="name") {
- r_ret = script->custom_signal_get_argument_name(sig,idx);
- return true;
- }
- }
- return false;
- }
- void _get_property_list( List<PropertyInfo> *p_list) const {
- if (sig==StringName())
- return;
- p_list->push_back(PropertyInfo(Variant::INT,"argument_count",PROPERTY_HINT_RANGE,"0,256"));
- String argt="Variant";
- for(int i=1;i<Variant::VARIANT_MAX;i++) {
- argt+=","+Variant::get_type_name(Variant::Type(i));
- }
- for(int i=0;i<script->custom_signal_get_argument_count(sig);i++) {
- p_list->push_back(PropertyInfo(Variant::INT,"argument/"+itos(i+1)+"/type",PROPERTY_HINT_ENUM,argt));
- p_list->push_back(PropertyInfo(Variant::STRING,"argument/"+itos(i+1)+"/name"));
- }
- }
- public:
- void edit(const StringName& p_sig) {
- sig=p_sig;
- _change_notify();
- }
- VisualScriptEditorSignalEdit() { undo_redo=NULL; }
- };
- class VisualScriptEditorVariableEdit : public Object {
- OBJ_TYPE(VisualScriptEditorVariableEdit,Object)
- StringName var;
- public:
- UndoRedo *undo_redo;
- Ref<VisualScript> script;
- protected:
- static void _bind_methods() {
- ObjectTypeDB::bind_method("_var_changed",&VisualScriptEditorVariableEdit::_var_changed);
- ObjectTypeDB::bind_method("_var_value_changed",&VisualScriptEditorVariableEdit::_var_value_changed);
- }
- void _var_changed() {
- _change_notify();
- }
- void _var_value_changed() {
- _change_notify("value"); //so the whole tree is not redrawn, makes editing smoother in general
- }
- bool _set(const StringName& p_name, const Variant& p_value) {
- if (var==StringName())
- return false;
- if (String(p_name)=="value") {
- undo_redo->create_action("Set Variable Default Value");
- Variant current=script->get_variable_default_value(var);
- undo_redo->add_do_method(script.ptr(),"set_variable_default_value",var,p_value);
- undo_redo->add_undo_method(script.ptr(),"set_variable_default_value",var,current);
- undo_redo->add_do_method(this,"_var_value_changed");
- undo_redo->add_undo_method(this,"_var_value_changed");
- undo_redo->commit_action();
- return true;
- }
- Dictionary d = script->call("get_variable_info",var);
- if (String(p_name)=="type") {
- Dictionary dc=d.copy();
- dc["type"]=p_value;
- undo_redo->create_action("Set Variable Type");
- undo_redo->add_do_method(script.ptr(),"set_variable_info",var,dc);
- undo_redo->add_undo_method(script.ptr(),"set_variable_info",var,d);
- undo_redo->add_do_method(this,"_var_changed");
- undo_redo->add_undo_method(this,"_var_changed");
- undo_redo->commit_action();
- return true;
- }
- if (String(p_name)=="hint") {
- Dictionary dc=d.copy();
- dc["hint"]=p_value;
- undo_redo->create_action("Set Variable Type");
- undo_redo->add_do_method(script.ptr(),"set_variable_info",var,dc);
- undo_redo->add_undo_method(script.ptr(),"set_variable_info",var,d);
- undo_redo->add_do_method(this,"_var_changed");
- undo_redo->add_undo_method(this,"_var_changed");
- undo_redo->commit_action();
- return true;
- }
- if (String(p_name)=="hint_string") {
- Dictionary dc=d.copy();
- dc["hint_string"]=p_value;
- undo_redo->create_action("Set Variable Type");
- undo_redo->add_do_method(script.ptr(),"set_variable_info",var,dc);
- undo_redo->add_undo_method(script.ptr(),"set_variable_info",var,d);
- undo_redo->add_do_method(this,"_var_changed");
- undo_redo->add_undo_method(this,"_var_changed");
- undo_redo->commit_action();
- return true;
- }
- return false;
- }
- bool _get(const StringName& p_name,Variant &r_ret) const {
- if (var==StringName())
- return false;
- if (String(p_name)=="value") {
- r_ret=script->get_variable_default_value(var);
- return true;
- }
- PropertyInfo pinfo = script->get_variable_info(var);
- if (String(p_name)=="type") {
- r_ret=pinfo.type;
- return true;
- }
- if (String(p_name)=="hint") {
- r_ret=pinfo.hint;
- return true;
- }
- if (String(p_name)=="hint_string") {
- r_ret=pinfo.hint_string;
- return true;
- }
- return false;
- }
- void _get_property_list( List<PropertyInfo> *p_list) const {
- if (var==StringName())
- return;
- String argt="Variant";
- for(int i=1;i<Variant::VARIANT_MAX;i++) {
- argt+=","+Variant::get_type_name(Variant::Type(i));
- }
- p_list->push_back(PropertyInfo(Variant::INT,"type",PROPERTY_HINT_ENUM,argt));
- p_list->push_back(PropertyInfo(script->get_variable_info(var).type,"value",script->get_variable_info(var).hint,script->get_variable_info(var).hint_string,PROPERTY_USAGE_DEFAULT));
- p_list->push_back(PropertyInfo(Variant::INT,"hint",PROPERTY_HINT_ENUM,"None,Range,ExpRange,Enum,ExpEasing,Length,SpriteFrame,KeyAccel,BitFlags,AllFlags,File,Dir,GlobalFile,GlobalDir,ResourceType,MultilineText"));
- p_list->push_back(PropertyInfo(Variant::STRING,"hint_string"));
- }
- public:
- void edit(const StringName& p_var) {
- var=p_var;
- _change_notify();
- }
- VisualScriptEditorVariableEdit() { undo_redo=NULL; }
- };
- static Color _color_from_type(Variant::Type p_type) {
- Color color;
- color.set_hsv(p_type/float(Variant::VARIANT_MAX),0.7,0.7);
- return color;
- }
- void VisualScriptEditor::_update_graph_connections() {
- graph->clear_connections();
- List<VisualScript::SequenceConnection> sequence_conns;
- script->get_sequence_connection_list(edited_func,&sequence_conns);
- for (List<VisualScript::SequenceConnection>::Element *E=sequence_conns.front();E;E=E->next()) {
- graph->connect_node(itos(E->get().from_node),E->get().from_output,itos(E->get().to_node),0);
- }
- List<VisualScript::DataConnection> data_conns;
- script->get_data_connection_list(edited_func,&data_conns);
- for (List<VisualScript::DataConnection>::Element *E=data_conns.front();E;E=E->next()) {
- VisualScript::DataConnection dc=E->get();
- Ref<VisualScriptNode> from_node = script->get_node(edited_func,E->get().from_node);
- Ref<VisualScriptNode> to_node = script->get_node(edited_func,E->get().to_node);
- if (to_node->has_input_sequence_port()) {
- dc.to_port++;
- }
- dc.from_port+=from_node->get_output_sequence_port_count();
- graph->connect_node(itos(E->get().from_node),dc.from_port,itos(E->get().to_node),dc.to_port);
- }
- }
- void VisualScriptEditor::_update_graph(int p_only_id) {
- updating_graph=true;
- //byebye all nodes
- if (p_only_id>=0) {
- if (graph->has_node(itos(p_only_id))) {
- Node* gid = graph->get_node(itos(p_only_id));
- if (gid)
- memdelete(gid);
- }
- } else {
- for(int i=0;i<graph->get_child_count();i++) {
- if (graph->get_child(i)->cast_to<GraphNode>()) {
- memdelete(graph->get_child(i));
- i--;
- }
- }
- }
- if (!script->has_function(edited_func)) {
- graph->hide();
- select_func_text->show();
- updating_graph=false;
- return;
- }
- graph->show();
- select_func_text->hide();
- Ref<Texture> type_icons[Variant::VARIANT_MAX]={
- Control::get_icon("MiniVariant","EditorIcons"),
- Control::get_icon("MiniBoolean","EditorIcons"),
- Control::get_icon("MiniInteger","EditorIcons"),
- Control::get_icon("MiniFloat","EditorIcons"),
- Control::get_icon("MiniString","EditorIcons"),
- Control::get_icon("MiniVector2","EditorIcons"),
- Control::get_icon("MiniRect2","EditorIcons"),
- Control::get_icon("MiniVector3","EditorIcons"),
- Control::get_icon("MiniMatrix2","EditorIcons"),
- Control::get_icon("MiniPlane","EditorIcons"),
- Control::get_icon("MiniQuat","EditorIcons"),
- Control::get_icon("MiniAabb","EditorIcons"),
- Control::get_icon("MiniMatrix3","EditorIcons"),
- Control::get_icon("MiniTransform","EditorIcons"),
- Control::get_icon("MiniColor","EditorIcons"),
- Control::get_icon("MiniImage","EditorIcons"),
- Control::get_icon("MiniPath","EditorIcons"),
- Control::get_icon("MiniRid","EditorIcons"),
- Control::get_icon("MiniObject","EditorIcons"),
- Control::get_icon("MiniInput","EditorIcons"),
- Control::get_icon("MiniDictionary","EditorIcons"),
- Control::get_icon("MiniArray","EditorIcons"),
- Control::get_icon("MiniRawArray","EditorIcons"),
- Control::get_icon("MiniIntArray","EditorIcons"),
- Control::get_icon("MiniFloatArray","EditorIcons"),
- Control::get_icon("MiniStringArray","EditorIcons"),
- Control::get_icon("MiniVector2Array","EditorIcons"),
- Control::get_icon("MiniVector3Array","EditorIcons"),
- Control::get_icon("MiniColorArray","EditorIcons")
- };
- Ref<Texture> seq_port = Control::get_icon("VisualShaderPort","EditorIcons");
- List<int> ids;
- script->get_node_list(edited_func,&ids);
- StringName editor_icons="EditorIcons";
- for(List<int>::Element *E=ids.front();E;E=E->next()) {
- if (p_only_id>=0 && p_only_id!=E->get())
- continue;
- Ref<VisualScriptNode> node = script->get_node(edited_func,E->get());
- Vector2 pos = script->get_node_pos(edited_func,E->get());
- GraphNode *gnode = memnew( GraphNode );
- gnode->set_title(node->get_caption());
- if (error_line==E->get()) {
- gnode->set_overlay(GraphNode::OVERLAY_POSITION);
- } else if (node->is_breakpoint()) {
- gnode->set_overlay(GraphNode::OVERLAY_BREAKPOINT);
- }
- if (EditorSettings::get_singleton()->has("visual_script_editor/color_"+node->get_category())) {
- gnode->set_modulate(EditorSettings::get_singleton()->get("visual_script_editor/color_"+node->get_category()));
- }
- gnode->set_meta("__vnode",node);
- gnode->set_name(itos(E->get()));
- gnode->connect("dragged",this,"_node_moved",varray(E->get()));
- gnode->connect("close_request",this,"_remove_node",varray(E->get()),CONNECT_DEFERRED);
- if (E->get()!=script->get_function_node_id(edited_func)) {
- //function can't be erased
- gnode->set_show_close_button(true);
- }
- Label *text = memnew( Label );
- text->set_text(node->get_text());
- gnode->add_child(text);
- int slot_idx=0;
- bool single_seq_output = node->get_output_sequence_port_count()==1 && node->get_output_sequence_port_text(0)==String();
- gnode->set_slot(0,node->has_input_sequence_port(),TYPE_SEQUENCE,Color(1,1,1,1),single_seq_output,TYPE_SEQUENCE,Color(1,1,1,1),seq_port,seq_port);
- gnode->set_offset(pos*EDSCALE);
- slot_idx++;
- if (!single_seq_output) {
- for(int i=0;i<node->get_output_sequence_port_count();i++) {
- Label *text2 = memnew( Label );
- text2->set_text(node->get_output_sequence_port_text(i));
- text2->set_align(Label::ALIGN_RIGHT);
- gnode->add_child(text2);
- gnode->set_slot(slot_idx,false,0,Color(),true,TYPE_SEQUENCE,Color(1,1,1,1),seq_port,seq_port);
- slot_idx++;
- }
- }
- for(int i=0;i<MAX(node->get_output_value_port_count(),node->get_input_value_port_count());i++) {
- bool left_ok=false;
- Variant::Type left_type=Variant::NIL;
- String left_name;
- if (i<node->get_input_value_port_count()) {
- PropertyInfo pi = node->get_input_value_port_info(i);
- left_ok=true;
- left_type=pi.type;
- left_name=pi.name;
- }
- bool right_ok=false;
- Variant::Type right_type=Variant::NIL;
- String right_name;
- if (i<node->get_output_value_port_count()) {
- PropertyInfo pi = node->get_output_value_port_info(i);
- right_ok=true;
- right_type=pi.type;
- right_name=pi.name;
- }
- HBoxContainer *hbc = memnew( HBoxContainer);
- if (left_ok) {
- Ref<Texture> t;
- if (left_type>=0 && left_type<Variant::VARIANT_MAX) {
- t=type_icons[left_type];
- }
- if (t.is_valid()) {
- TextureFrame *tf = memnew(TextureFrame);
- tf->set_texture(t);
- tf->set_stretch_mode(TextureFrame::STRETCH_KEEP_CENTERED);
- hbc->add_child(tf);
- }
- hbc->add_child(memnew(Label(left_name)));
- if (left_type!=Variant::NIL && !script->is_input_value_port_connected(edited_func,E->get(),i)) {
- Button *button = memnew( Button );
- Variant value = node->get_default_input_value(i);
- if (value.get_type()!=left_type) {
- //different type? for now convert
- //not the same, reconvert
- Variant::CallError ce;
- const Variant *existingp=&value;
- value = Variant::construct(left_type,&existingp,1,ce,false);
- }
- button->set_text(value);
- button->connect("pressed",this,"_default_value_edited",varray(button,E->get(),i));
- hbc->add_child(button);
- }
- } else {
- Control *c = memnew(Control);
- c->set_custom_minimum_size(Size2(10,0)*EDSCALE);
- hbc->add_child(c);
- }
- hbc->add_spacer();
- if (right_ok) {
- hbc->add_child(memnew(Label(right_name)));
- Ref<Texture> t;
- if (right_type>=0 && right_type<Variant::VARIANT_MAX) {
- t=type_icons[right_type];
- }
- if (t.is_valid()) {
- TextureFrame *tf = memnew(TextureFrame);
- tf->set_texture(t);
- tf->set_stretch_mode(TextureFrame::STRETCH_KEEP_CENTERED);
- hbc->add_child(tf);
- }
- }
- gnode->add_child(hbc);
- gnode->set_slot(slot_idx,left_ok,left_type,_color_from_type(left_type),right_ok,right_type,_color_from_type(right_type));
- slot_idx++;
- }
- graph->add_child(gnode);
- }
- _update_graph_connections();
- graph->call_deferred("set_scroll_ofs",script->get_function_scroll(edited_func)*EDSCALE); //may need to adapt a bit, let it do so
- updating_graph=false;
- }
- void VisualScriptEditor::_update_members() {
- updating_members=true;
- members->clear();
- TreeItem *root = members->create_item();
- TreeItem *functions = members->create_item(root);
- functions->set_selectable(0,false);
- functions->set_text(0,TTR("Functions:"));
- functions->add_button(0,Control::get_icon("Override","EditorIcons"),1);
- functions->add_button(0,Control::get_icon("Add","EditorIcons"),0);
- functions->set_custom_bg_color(0,Control::get_color("prop_section","Editor"));
- List<StringName> func_names;
- script->get_function_list(&func_names);
- for (List<StringName>::Element *E=func_names.front();E;E=E->next()) {
- TreeItem *ti = members->create_item(functions) ;
- ti->set_text(0,E->get());
- ti->set_selectable(0,true);
- ti->set_editable(0,true);
- //ti->add_button(0,Control::get_icon("Edit","EditorIcons"),0); function arguments are in the node now
- ti->add_button(0,Control::get_icon("Del","EditorIcons"),1);
- ti->set_metadata(0,E->get());
- if (E->get()==edited_func) {
- ti->set_custom_bg_color(0,get_color("prop_category","Editor"));
- ti->set_custom_color(0,Color(1,1,1,1));
- }
- if (selected==E->get())
- ti->select(0);
- }
- TreeItem *variables = members->create_item(root);
- variables->set_selectable(0,false);
- variables->set_text(0,TTR("Variables:"));
- variables->add_button(0,Control::get_icon("Add","EditorIcons"));
- variables->set_custom_bg_color(0,Control::get_color("prop_section","Editor"));
- List<StringName> var_names;
- script->get_variable_list(&var_names);
- for (List<StringName>::Element *E=var_names.front();E;E=E->next()) {
- TreeItem *ti = members->create_item(variables);
- ti->set_text(0,E->get());
- ti->set_selectable(0,true);
- ti->set_editable(0,true);
- ti->add_button(0,Control::get_icon("Edit","EditorIcons"),0);
- ti->add_button(0,Control::get_icon("Del","EditorIcons"),1);
- ti->set_metadata(0,E->get());
- if (selected==E->get())
- ti->select(0);
- }
- TreeItem *_signals = members->create_item(root);
- _signals->set_selectable(0,false);
- _signals->set_text(0,TTR("Signals:"));
- _signals->add_button(0,Control::get_icon("Add","EditorIcons"));
- _signals->set_custom_bg_color(0,Control::get_color("prop_section","Editor"));
- List<StringName> signal_names;
- script->get_custom_signal_list(&signal_names);
- for (List<StringName>::Element *E=signal_names.front();E;E=E->next()) {
- TreeItem *ti = members->create_item(_signals);
- ti->set_text(0,E->get());
- ti->set_selectable(0,true);
- ti->set_editable(0,true);
- ti->add_button(0,Control::get_icon("Edit","EditorIcons"),0);
- ti->add_button(0,Control::get_icon("Del","EditorIcons"),1);
- ti->set_metadata(0,E->get());
- if (selected==E->get())
- ti->select(0);
- }
- String base_type=script->get_instance_base_type();
- String icon_type=base_type;
- if (!Control::has_icon(base_type,"EditorIcons")) {
- icon_type="Object";
- }
- base_type_select->set_text(base_type);
- base_type_select->set_icon(Control::get_icon(icon_type,"EditorIcons"));
- updating_members=false;
- }
- void VisualScriptEditor::_member_selected() {
- if (updating_members)
- return;
- TreeItem *ti=members->get_selected();
- ERR_FAIL_COND(!ti);
- selected=ti->get_metadata(0);
- // print_line("selected: "+String(selected));
- if (ti->get_parent()==members->get_root()->get_children()) {
- if (edited_func!=selected) {
- revert_on_drag=edited_func;
- edited_func=selected;
- _update_members();
- _update_graph();
- }
- return; //or crash because it will become invalid
- }
- }
- void VisualScriptEditor::_member_edited() {
- if (updating_members)
- return;
- TreeItem *ti=members->get_edited();
- ERR_FAIL_COND(!ti);
- String name = ti->get_metadata(0);
- String new_name = ti->get_text(0);
- if (name==new_name)
- return;
- if (!new_name.is_valid_identifier()) {
- EditorNode::get_singleton()->show_warning(TTR("Name is not a valid identifier:")+" "+new_name);
- updating_members=true;
- ti->set_text(0,name);
- updating_members=false;
- return;
- }
- if (script->has_function(new_name) || script->has_variable(new_name) || script->has_custom_signal(new_name)) {
- EditorNode::get_singleton()->show_warning(TTR("Name already in use by another func/var/signal:")+" "+new_name);
- updating_members=true;
- ti->set_text(0,name);
- updating_members=false;
- return;
- }
- TreeItem *root=members->get_root();
- if (ti->get_parent()==root->get_children()) {
- if (edited_func==selected) {
- edited_func=new_name;
- }
- selected=new_name;
- _update_graph();
- undo_redo->create_action(TTR("Rename Function"));
- undo_redo->add_do_method(script.ptr(),"rename_function",name,new_name);
- undo_redo->add_undo_method(script.ptr(),"rename_function",new_name,name);
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- return; //or crash because it will become invalid
- }
- if (ti->get_parent()==root->get_children()->get_next()) {
- selected=new_name;
- undo_redo->create_action(TTR("Rename Variable"));
- undo_redo->add_do_method(script.ptr(),"rename_variable",name,new_name);
- undo_redo->add_undo_method(script.ptr(),"rename_variable",new_name,name);
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- return; //or crash because it will become invalid
- }
- if (ti->get_parent()==root->get_children()->get_next()->get_next()) {
- selected=new_name;
- undo_redo->create_action(TTR("Rename Signal"));
- undo_redo->add_do_method(script.ptr(),"rename_custom_signal",name,new_name);
- undo_redo->add_undo_method(script.ptr(),"rename_custom_signal",new_name,name);
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- return; //or crash because it will become invalid
- }
- }
- void VisualScriptEditor::_override_pressed(int p_id) {
- //override a virtual function or method from base type
- ERR_FAIL_COND(!virtuals_in_menu.has(p_id));
- VirtualInMenu vim=virtuals_in_menu[p_id];
- String name = _validate_name(vim.name);
- selected=name;
- edited_func=selected;
- Ref<VisualScriptFunction> func_node;
- func_node.instance();
- func_node->set_name(vim.name);
- undo_redo->create_action(TTR("Add Function"));
- undo_redo->add_do_method(script.ptr(),"add_function",name);
- for(int i=0;i<vim.args.size();i++) {
- func_node->add_argument(vim.args[i].first,vim.args[i].second);
- }
- undo_redo->add_do_method(script.ptr(),"add_node",name,script->get_available_id(),func_node);
- if (vim.ret!=Variant::NIL || vim.ret_variant) {
- Ref<VisualScriptReturn> ret_node;
- ret_node.instance();
- ret_node->set_return_type(vim.ret);
- ret_node->set_enable_return_value(true);
- ret_node->set_name(vim.name);
- undo_redo->add_do_method(script.ptr(),"add_node",name,script->get_available_id()+1,ret_node,Vector2(500,0));
- }
- undo_redo->add_undo_method(script.ptr(),"remove_function",name);
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- _update_graph();
- }
- void VisualScriptEditor::_member_button(Object *p_item, int p_column, int p_button) {
- TreeItem *ti=p_item->cast_to<TreeItem>();
- TreeItem *root=members->get_root();
- if (ti->get_parent()==root) {
- //main buttons
- if (ti==root->get_children()) {
- //add function, this one uses menu
- if (p_button==1) {
- new_function_menu->clear();
- new_function_menu->set_size(Size2(0,0));
- int idx=0;
- virtuals_in_menu.clear();
- List<MethodInfo> mi;
- ObjectTypeDB::get_method_list(script->get_instance_base_type(),&mi);
- for (List<MethodInfo>::Element *E=mi.front();E;E=E->next()) {
- MethodInfo mi=E->get();
- if (mi.flags&METHOD_FLAG_VIRTUAL) {
- VirtualInMenu vim;
- vim.name=mi.name;
- vim.ret=mi.return_val.type;
- if (mi.return_val.name!=String())
- vim.ret_variant=true;
- else
- vim.ret_variant=false;
- String desc;
- if (mi.return_val.type==Variant::NIL)
- desc="var";
- else
- desc=Variant::get_type_name(mi.return_val.type);
- desc+=" "+mi.name+" ( ";
- for(int i=0;i<mi.arguments.size();i++) {
- if (i>0)
- desc+=", ";
- if (mi.arguments[i].type==Variant::NIL)
- desc+="var ";
- else
- desc+=Variant::get_type_name(mi.arguments[i].type)+" ";
- desc+=mi.arguments[i].name;
- Pair<Variant::Type,String> p;
- p.first=mi.arguments[i].type;
- p.second=mi.arguments[i].name;
- vim.args.push_back( p );
- }
- desc+=" )";
- virtuals_in_menu[idx]=vim;
- new_function_menu->add_item(desc,idx);
- idx++;
- }
- }
- Rect2 pos = members->get_item_rect(ti);
- new_function_menu->set_pos(members->get_global_pos()+pos.pos+Vector2(0,pos.size.y));
- new_function_menu->popup();
- return;
- } else if (p_button==0) {
- String name = _validate_name("new_function");
- selected=name;
- edited_func=selected;
- Ref<VisualScriptFunction> func_node;
- func_node.instance();
- func_node->set_name(name);
- undo_redo->create_action(TTR("Add Function"));
- undo_redo->add_do_method(script.ptr(),"add_function",name);
- undo_redo->add_do_method(script.ptr(),"add_node",name,script->get_available_id(),func_node);
- undo_redo->add_undo_method(script.ptr(),"remove_function",name);
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- _update_graph();
- }
- return; //or crash because it will become invalid
- }
- if (ti==root->get_children()->get_next()) {
- //add variable
- String name = _validate_name("new_variable");
- selected=name;
- undo_redo->create_action(TTR("Add Variable"));
- undo_redo->add_do_method(script.ptr(),"add_variable",name);
- undo_redo->add_undo_method(script.ptr(),"remove_variable",name);
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- return; //or crash because it will become invalid
- }
- if (ti==root->get_children()->get_next()->get_next()) {
- //add variable
- String name = _validate_name("new_signal");
- selected=name;
- undo_redo->create_action(TTR("Add Signal"));
- undo_redo->add_do_method(script.ptr(),"add_custom_signal",name);
- undo_redo->add_undo_method(script.ptr(),"remove_custom_signal",name);
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- return; //or crash because it will become invalid
- }
- } else {
- if (ti->get_parent()==root->get_children()) {
- //edit/remove function
- String name = ti->get_metadata(0);
- if (p_button==1) {
- //delete the function
- undo_redo->create_action(TTR("Remove Function"));
- undo_redo->add_do_method(script.ptr(),"remove_function",name);
- undo_redo->add_undo_method(script.ptr(),"add_function",name);
- List<int> nodes;
- script->get_node_list(name,&nodes);
- for (List<int>::Element *E=nodes.front();E;E=E->next()) {
- undo_redo->add_undo_method(script.ptr(),"add_node",name,E->get(),script->get_node(name,E->get()),script->get_node_pos(name,E->get()));
- }
- List<VisualScript::SequenceConnection> seq_connections;
- script->get_sequence_connection_list(name,&seq_connections);
- for (List<VisualScript::SequenceConnection>::Element *E=seq_connections.front();E;E=E->next()) {
- undo_redo->add_undo_method(script.ptr(),"sequence_connect",name,E->get().from_node,E->get().from_output,E->get().to_node);
- }
- List<VisualScript::DataConnection> data_connections;
- script->get_data_connection_list(name,&data_connections);
- for (List<VisualScript::DataConnection>::Element *E=data_connections.front();E;E=E->next()) {
- undo_redo->add_undo_method(script.ptr(),"data_connect",name,E->get().from_node,E->get().from_port,E->get().to_node,E->get().to_port);
- }
- //for(int i=0;i<script->function_get_argument_count(name);i++) {
- //// undo_redo->add_undo_method(script.ptr(),"function_add_argument",name,script->function_get_argument_name(name,i),script->function_get_argument_type(name,i));
- //}
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- } else if (p_button==0) {
- }
- return; //or crash because it will become invalid
- }
- if (ti->get_parent()==root->get_children()->get_next()) {
- //edit/remove variable
- String name = ti->get_metadata(0);
- if (p_button==1) {
- undo_redo->create_action(TTR("Remove Variable"));
- undo_redo->add_do_method(script.ptr(),"remove_variable",name);
- undo_redo->add_undo_method(script.ptr(),"add_variable",name,script->get_variable_default_value(name));
- undo_redo->add_undo_method(script.ptr(),"set_variable_info",name,script->call("get_variable_info",name)); //return as dict
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- return; //or crash because it will become invalid
- } else if (p_button==0) {
- variable_editor->edit(name);
- edit_variable_dialog->set_title(TTR("Editing Variable:")+" "+name);
- edit_variable_dialog->popup_centered_minsize(Size2(400,200)*EDSCALE);
- }
- }
- if (ti->get_parent()==root->get_children()->get_next()->get_next()) {
- //edit/remove variable
- String name = ti->get_metadata(0);
- if (p_button==1) {
- undo_redo->create_action(TTR("Remove Signal"));
- undo_redo->add_do_method(script.ptr(),"remove_custom_signal",name);
- undo_redo->add_undo_method(script.ptr(),"add_custom_signal",name);
- for(int i=0;i<script->custom_signal_get_argument_count(name);i++) {
- undo_redo->add_undo_method(script.ptr(),"custom_signal_add_argument",name,script->custom_signal_get_argument_name(name,i),script->custom_signal_get_argument_type(name,i));
- }
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- } else if (p_button==0) {
- signal_editor->edit(name);
- edit_signal_dialog->set_title(TTR("Editing Signal:")+" "+name);
- edit_signal_dialog->popup_centered_minsize(Size2(400,300)*EDSCALE);
- }
- return; //or crash because it will become invalid
- }
- }
- }
- void VisualScriptEditor::_available_node_doubleclicked() {
- TreeItem *item = nodes->get_selected();
- if (!item)
- return;
- String which = item->get_metadata(0);
- if (which==String())
- return;
- Vector2 ofs = graph->get_scroll_ofs() + graph->get_size() * 0.5;
- if (graph->is_using_snap()) {
- int snap = graph->get_snap();
- ofs = ofs.snapped(Vector2(snap,snap));
- }
- ofs/=EDSCALE;
- while(true) {
- bool exists=false;
- List<int> existing;
- script->get_node_list(edited_func,&existing);
- for (List<int>::Element *E=existing.front();E;E=E->next()) {
- Point2 pos = script->get_node_pos(edited_func,E->get());
- if (pos.distance_to(ofs)<15) {
- ofs+=Vector2(graph->get_snap(),graph->get_snap());
- exists=true;
- break;
- }
- }
- if (exists)
- continue;
- break;
- }
- Ref<VisualScriptNode> vnode = VisualScriptLanguage::singleton->create_node_from_name(which);
- int new_id = script->get_available_id();
- undo_redo->create_action(TTR("Add Node"));
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,new_id,vnode,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,new_id);
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- Node* node = graph->get_node(itos(new_id));
- if (node) {
- graph->set_selected(node);
- _node_selected(node);
- }
- }
- void VisualScriptEditor::_update_available_nodes() {
- nodes->clear();
- TreeItem *root = nodes->create_item();
- Map<String,TreeItem*> path_cache;
- String filter = node_filter->get_text();
- List<String> fnodes;
- VisualScriptLanguage::singleton->get_registered_node_names(&fnodes);
- for (List<String>::Element *E=fnodes.front();E;E=E->next()) {
- Vector<String> path = E->get().split("/");
- if (filter!=String() && path.size() && path[path.size()-1].findn(filter)==-1)
- continue;
- String sp;
- TreeItem* parent=root;
- for(int i=0;i<path.size()-1;i++) {
- if (i>0)
- sp+=",";
- sp+=path[i];
- if (!path_cache.has(sp)) {
- TreeItem* pathn = nodes->create_item(parent);
- pathn->set_selectable(0,false);
- pathn->set_text(0,path[i].capitalize());
- path_cache[sp]=pathn;
- parent=pathn;
- if (filter==String()) {
- pathn->set_collapsed(true); //should remember state
- }
- } else {
- parent=path_cache[sp];
- }
- }
- TreeItem *item = nodes->create_item(parent);
- item->set_text(0,path[path.size()-1].capitalize());
- item->set_selectable(0,true);
- item->set_metadata(0,E->get());
- }
- }
- String VisualScriptEditor::_validate_name(const String& p_name) const {
- String valid=p_name;
- int counter=1;
- while(true) {
- bool exists = script->has_function(valid) || script->has_variable(valid) || script->has_custom_signal(valid);
- if (exists) {
- counter++;
- valid=p_name+"_"+itos(counter);
- continue;
- }
- break;
- }
- return valid;
- }
- void VisualScriptEditor::_on_nodes_delete() {
- List<int> to_erase;
- for(int i=0;i<graph->get_child_count();i++) {
- GraphNode *gn = graph->get_child(i)->cast_to<GraphNode>();
- if (gn) {
- if (gn->is_selected() && gn->is_close_button_visible()) {
- to_erase.push_back(gn->get_name().operator String().to_int());
- }
- }
- }
- if (to_erase.empty())
- return;
- undo_redo->create_action("Remove VisualScript Nodes");
- for(List<int>::Element*F=to_erase.front();F;F=F->next()) {
- undo_redo->add_do_method(script.ptr(),"remove_node",edited_func,F->get());
- undo_redo->add_undo_method(script.ptr(),"add_node",edited_func,F->get(),script->get_node(edited_func,F->get()),script->get_node_pos(edited_func,F->get()));
- List<VisualScript::SequenceConnection> sequence_conns;
- script->get_sequence_connection_list(edited_func,&sequence_conns);
- for (List<VisualScript::SequenceConnection>::Element *E=sequence_conns.front();E;E=E->next()) {
- if (E->get().from_node==F->get() || E->get().to_node==F->get()) {
- undo_redo->add_undo_method(script.ptr(),"sequence_connect",edited_func,E->get().from_node,E->get().from_output,E->get().to_node);
- }
- }
- List<VisualScript::DataConnection> data_conns;
- script->get_data_connection_list(edited_func,&data_conns);
- for (List<VisualScript::DataConnection>::Element *E=data_conns.front();E;E=E->next()) {
- if (E->get().from_node==F->get() || E->get().to_node==F->get()) {
- undo_redo->add_undo_method(script.ptr(),"data_connect",edited_func,E->get().from_node,E->get().from_port,E->get().to_node,E->get().to_port);
- }
- }
- }
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- }
- void VisualScriptEditor::_on_nodes_duplicate() {
- List<int> to_duplicate;
- for(int i=0;i<graph->get_child_count();i++) {
- GraphNode *gn = graph->get_child(i)->cast_to<GraphNode>();
- if (gn) {
- if (gn->is_selected() && gn->is_close_button_visible()) {
- to_duplicate.push_back(gn->get_name().operator String().to_int());
- }
- }
- }
- if (to_duplicate.empty())
- return;
- undo_redo->create_action("Duplicate VisualScript Nodes");
- int idc=script->get_available_id()+1;
- Set<int> to_select;
- for(List<int>::Element*F=to_duplicate.front();F;F=F->next()) {
- Ref<VisualScriptNode> node = script->get_node(edited_func,F->get());
- Ref<VisualScriptNode> dupe = node->duplicate();
- int new_id = idc++;
- to_select.insert(new_id);
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,new_id,dupe,script->get_node_pos(edited_func,F->get())+Vector2(20,20));
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,new_id);
- }
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- for(int i=0;i<graph->get_child_count();i++) {
- GraphNode *gn = graph->get_child(i)->cast_to<GraphNode>();
- if (gn) {
- int id = gn->get_name().operator String().to_int();
- gn->set_selected(to_select.has(id));
- }
- }
- if (to_select.size()) {
- EditorNode::get_singleton()->push_item(script->get_node(edited_func,to_select.front()->get()).ptr());
- }
- }
- void VisualScriptEditor::_input(const InputEvent& p_event) {
- if (p_event.type==InputEvent::MOUSE_BUTTON && !p_event.mouse_button.pressed && p_event.mouse_button.button_index==BUTTON_LEFT) {
- revert_on_drag=String(); //so we can still drag functions
- }
- }
- Variant VisualScriptEditor::get_drag_data_fw(const Point2& p_point,Control* p_from) {
- if (p_from==nodes) {
- TreeItem *it = nodes->get_item_at_pos(p_point);
- if (!it)
- return Variant();
- String type=it->get_metadata(0);
- if (type==String())
- return Variant();
- Dictionary dd;
- dd["type"]="visual_script_node_drag";
- dd["node_type"]=type;
- Label *label = memnew(Label);
- label->set_text(it->get_text(0));
- set_drag_preview(label);
- return dd;
- }
- if (p_from==members) {
- TreeItem *it = members->get_item_at_pos(p_point);
- if (!it)
- return Variant();
- String type=it->get_metadata(0);
- if (type==String())
- return Variant();
- Dictionary dd;
- TreeItem *root=members->get_root();
- if (it->get_parent()==root->get_children()) {
- dd["type"]="visual_script_function_drag";
- dd["function"]=type;
- if (revert_on_drag!=String()) {
- edited_func=revert_on_drag; //revert so function does not change
- revert_on_drag=String();
- _update_graph();
- }
- } else if (it->get_parent()==root->get_children()->get_next()) {
- dd["type"]="visual_script_variable_drag";
- dd["variable"]=type;
- } else if (it->get_parent()==root->get_children()->get_next()->get_next()) {
- dd["type"]="visual_script_signal_drag";
- dd["signal"]=type;
- } else {
- return Variant();
- }
- Label *label = memnew(Label);
- label->set_text(it->get_text(0));
- set_drag_preview(label);
- return dd;
- }
- return Variant();
- }
- bool VisualScriptEditor::can_drop_data_fw(const Point2& p_point,const Variant& p_data,Control* p_from) const{
- if (p_from==graph) {
- Dictionary d = p_data;
- if (d.has("type") &&
- (
- String(d["type"])=="visual_script_node_drag" ||
- String(d["type"])=="visual_script_function_drag" ||
- String(d["type"])=="visual_script_variable_drag" ||
- String(d["type"])=="visual_script_signal_drag" ||
- String(d["type"])=="obj_property" ||
- String(d["type"])=="nodes"
- ) ) {
- if (String(d["type"])=="obj_property") {
- #ifdef OSX_ENABLED
- const_cast<VisualScriptEditor*>(this)->_show_hint("Hold Meta to drop a Setter, Shift+Meta to drop a Setter and copy the value.");
- #else
- const_cast<VisualScriptEditor*>(this)->_show_hint("Hold Ctrl to drop a Setter, Shift+Ctrl to drop a Setter and copy the value.");
- #endif
- }
- if (String(d["type"])=="visual_script_variable_drag") {
- #ifdef OSX_ENABLED
- const_cast<VisualScriptEditor*>(this)->_show_hint("Hold Meta to drop a Variable Setter.");
- #else
- const_cast<VisualScriptEditor*>(this)->_show_hint("Hold Ctrl to drop a Variable Setter.");
- #endif
- }
- return true;
- }
- }
- return false;
- }
- #ifdef TOOLS_ENABLED
- static Node* _find_script_node(Node* p_edited_scene,Node* p_current_node,const Ref<Script> &script) {
- if (p_edited_scene!=p_current_node && p_current_node->get_owner()!=p_edited_scene)
- return NULL;
- Ref<Script> scr = p_current_node->get_script();
- if (scr.is_valid() && scr==script)
- return p_current_node;
- for(int i=0;i<p_current_node->get_child_count();i++) {
- Node *n = _find_script_node(p_edited_scene,p_current_node->get_child(i),script);
- if (n)
- return n;
- }
- return NULL;
- }
- #endif
- void VisualScriptEditor::drop_data_fw(const Point2& p_point,const Variant& p_data,Control* p_from){
- if (p_from==graph) {
- Dictionary d = p_data;
- if (d.has("type") && String(d["type"])=="visual_script_node_drag") {
- Vector2 ofs = graph->get_scroll_ofs() + p_point;
- if (graph->is_using_snap()) {
- int snap = graph->get_snap();
- ofs = ofs.snapped(Vector2(snap,snap));
- }
- ofs/=EDSCALE;
- Ref<VisualScriptNode> vnode = VisualScriptLanguage::singleton->create_node_from_name(d["node_type"]);
- int new_id = script->get_available_id();
- undo_redo->create_action(TTR("Add Node"));
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,new_id,vnode,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,new_id);
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- Node* node = graph->get_node(itos(new_id));
- if (node) {
- graph->set_selected(node);
- _node_selected(node);
- }
- }
- if (d.has("type") && String(d["type"])=="visual_script_variable_drag") {
- #ifdef OSX_ENABLED
- bool use_set = Input::get_singleton()->is_key_pressed(KEY_META);
- #else
- bool use_set = Input::get_singleton()->is_key_pressed(KEY_CONTROL);
- #endif
- Vector2 ofs = graph->get_scroll_ofs() + p_point;
- if (graph->is_using_snap()) {
- int snap = graph->get_snap();
- ofs = ofs.snapped(Vector2(snap,snap));
- }
- ofs/=EDSCALE;
- Ref<VisualScriptNode> vnode;
- if (use_set) {
- Ref<VisualScriptVariableSet> vnodes;
- vnodes.instance();
- vnodes->set_variable(d["variable"]);
- vnode=vnodes;
- } else {
- Ref<VisualScriptVariableGet> vnodeg;
- vnodeg.instance();
- vnodeg->set_variable(d["variable"]);
- vnode=vnodeg;
- }
- int new_id = script->get_available_id();
- undo_redo->create_action(TTR("Add Node"));
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,new_id,vnode,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,new_id);
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- Node* node = graph->get_node(itos(new_id));
- if (node) {
- graph->set_selected(node);
- _node_selected(node);
- }
- }
- if (d.has("type") && String(d["type"])=="visual_script_function_drag") {
- Vector2 ofs = graph->get_scroll_ofs() + p_point;
- if (graph->is_using_snap()) {
- int snap = graph->get_snap();
- ofs = ofs.snapped(Vector2(snap,snap));
- }
- ofs/=EDSCALE;
- Ref<VisualScriptScriptCall> vnode;
- vnode.instance();
- vnode->set_call_mode(VisualScriptScriptCall::CALL_MODE_SELF);
- vnode->set_function(d["function"]);
- int new_id = script->get_available_id();
- undo_redo->create_action(TTR("Add Node"));
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,new_id,vnode,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,new_id);
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- Node* node = graph->get_node(itos(new_id));
- if (node) {
- graph->set_selected(node);
- _node_selected(node);
- }
- }
- if (d.has("type") && String(d["type"])=="visual_script_signal_drag") {
- Vector2 ofs = graph->get_scroll_ofs() + p_point;
- if (graph->is_using_snap()) {
- int snap = graph->get_snap();
- ofs = ofs.snapped(Vector2(snap,snap));
- }
- ofs/=EDSCALE;
- Ref<VisualScriptEmitSignal> vnode;
- vnode.instance();
- vnode->set_signal(d["signal"]);
- int new_id = script->get_available_id();
- undo_redo->create_action(TTR("Add Node"));
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,new_id,vnode,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,new_id);
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- Node* node = graph->get_node(itos(new_id));
- if (node) {
- graph->set_selected(node);
- _node_selected(node);
- }
- }
- if (d.has("type") && String(d["type"])=="nodes") {
- Node* sn = _find_script_node(get_tree()->get_edited_scene_root(),get_tree()->get_edited_scene_root(),script);
- if (!sn) {
- EditorNode::get_singleton()->show_warning("Can't drop nodes because script '"+get_name()+"' is not used in this scene.");
- return;
- }
- Array nodes = d["nodes"];
- Vector2 ofs = graph->get_scroll_ofs() + p_point;
- if (graph->is_using_snap()) {
- int snap = graph->get_snap();
- ofs = ofs.snapped(Vector2(snap,snap));
- }
- ofs/=EDSCALE;
- undo_redo->create_action(TTR("Add Node(s) From Tree"));
- int base_id = script->get_available_id();
- for(int i=0;i<nodes.size();i++) {
- NodePath np = nodes[i];
- Node *node = get_node(np);
- if (!node) {
- continue;
- }
- Ref<VisualScriptSceneNode> scene_node;
- scene_node.instance();
- scene_node->set_node_path(sn->get_path_to(node));
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,base_id,scene_node,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,base_id);
- base_id++;
- ofs+=Vector2(25,25);
- }
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- }
- if (d.has("type") && String(d["type"])=="obj_property") {
- Node* sn = _find_script_node(get_tree()->get_edited_scene_root(),get_tree()->get_edited_scene_root(),script);
- if (!sn) {
- //EditorNode::get_singleton()->show_warning("Can't drop properties because script '"+get_name()+"' is not used in this scene.");
- //return;
- }
- Object *obj=d["object"];
- if (!obj)
- return;
- Node *node = obj->cast_to<Node>();
- Vector2 ofs = graph->get_scroll_ofs() + p_point;
- if (graph->is_using_snap()) {
- int snap = graph->get_snap();
- ofs = ofs.snapped(Vector2(snap,snap));
- }
- ofs/=EDSCALE;
- #ifdef OSX_ENABLED
- bool use_set = Input::get_singleton()->is_key_pressed(KEY_META);
- #else
- bool use_set = Input::get_singleton()->is_key_pressed(KEY_CONTROL);
- #endif
- bool use_value = Input::get_singleton()->is_key_pressed(KEY_SHIFT);
- if (!node) {
- if (use_set)
- undo_redo->create_action(TTR("Add Setter Property"));
- else
- undo_redo->create_action(TTR("Add Getter Property"));
- int base_id = script->get_available_id();
- Ref<VisualScriptNode> vnode;
- if (use_set) {
- Ref<VisualScriptPropertySet> pset;
- pset.instance();
- pset->set_call_mode(VisualScriptPropertySet::CALL_MODE_INSTANCE);
- pset->set_base_type(obj->get_type());
- pset->set_property(d["property"]);
- if (use_value) {
- pset->set_use_builtin_value(true);
- pset->set_builtin_value(d["value"]);
- }
- vnode=pset;
- } else {
- Ref<VisualScriptPropertyGet> pget;
- pget.instance();
- pget->set_call_mode(VisualScriptPropertyGet::CALL_MODE_INSTANCE);
- pget->set_base_type(obj->get_type());
- pget->set_property(d["property"]);
- vnode=pget;
- }
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,base_id,vnode,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,base_id);
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- } else {
- if (use_set)
- undo_redo->create_action(TTR("Add Setter Property"));
- else
- undo_redo->create_action(TTR("Add Getter Property"));
- int base_id = script->get_available_id();
- Ref<VisualScriptNode> vnode;
- if (use_set) {
- Ref<VisualScriptPropertySet> pset;
- pset.instance();
- pset->set_call_mode(VisualScriptPropertySet::CALL_MODE_NODE_PATH);
- pset->set_base_path(sn->get_path_to(sn));
- pset->set_property(d["property"]);
- if (use_value) {
- pset->set_use_builtin_value(true);
- pset->set_builtin_value(d["value"]);
- }
- vnode=pset;
- } else {
- Ref<VisualScriptPropertyGet> pget;
- pget.instance();
- pget->set_call_mode(VisualScriptPropertyGet::CALL_MODE_NODE_PATH);
- pget->set_base_path(sn->get_path_to(sn));
- pget->set_property(d["property"]);
- vnode=pget;
- }
- undo_redo->add_do_method(script.ptr(),"add_node",edited_func,base_id,vnode,ofs);
- undo_redo->add_undo_method(script.ptr(),"remove_node",edited_func,base_id);
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- }
- }
- }
- }
- /////////////////////////
- void VisualScriptEditor::apply_code() {
- }
- Ref<Script> VisualScriptEditor::get_edited_script() const{
- return script;
- }
- Vector<String> VisualScriptEditor::get_functions(){
- return Vector<String>();
- }
- void VisualScriptEditor::set_edited_script(const Ref<Script>& p_script){
- script=p_script;
- signal_editor->script=p_script;
- signal_editor->undo_redo=undo_redo;
- variable_editor->script=p_script;
- variable_editor->undo_redo=undo_redo;
- script->connect("node_ports_changed",this,"_node_ports_changed");
- _update_members();
- _update_available_nodes();
- }
- void VisualScriptEditor::reload_text(){
- }
- String VisualScriptEditor::get_name(){
- String name;
- if (script->get_path().find("local://")==-1 && script->get_path().find("::")==-1) {
- name=script->get_path().get_file();
- if (is_unsaved()) {
- name+="(*)";
- }
- } else if (script->get_name()!="")
- name=script->get_name();
- else
- name=script->get_type()+"("+itos(script->get_instance_ID())+")";
- return name;
- }
- Ref<Texture> VisualScriptEditor::get_icon(){
- return Control::get_icon("VisualScript","EditorIcons");
- }
- bool VisualScriptEditor::is_unsaved(){
- #ifdef TOOLS_ENABLED
- return script->is_edited();
- #else
- return false;
- #endif
- }
- Variant VisualScriptEditor::get_edit_state(){
- Dictionary d;
- d["function"]=edited_func;
- d["scroll"]=graph->get_scroll_ofs();
- d["zoom"]=graph->get_zoom();
- d["using_snap"]=graph->is_using_snap();
- d["snap"]=graph->get_snap();
- return d;
- }
- void VisualScriptEditor::set_edit_state(const Variant& p_state){
- Dictionary d = p_state;
- if (d.has("function")) {
- edited_func=p_state;
- selected=edited_func;
- }
- _update_graph();
- _update_members();
- if (d.has("scroll")) {
- graph->set_scroll_ofs(d["scroll"]);
- }
- if (d.has("zoom")) {
- graph->set_zoom(d["zoom"]);
- }
- if (d.has("snap")) {
- graph->set_snap(d["snap"]);
- }
- if (d.has("snap_enabled")) {
- graph->set_use_snap(d["snap_enabled"]);
- }
- }
- void VisualScriptEditor::_center_on_node(int p_id) {
- Node *n = graph->get_node(itos(p_id));
- if (!n)
- return;
- GraphNode *gn = n->cast_to<GraphNode>();
- if (gn) {
- gn->set_selected(true);
- Vector2 new_scroll = gn->get_offset() - graph->get_size()*0.5 + gn->get_size()*0.5;
- graph->set_scroll_ofs( new_scroll );
- script->set_function_scroll(edited_func,new_scroll/EDSCALE);
- script->set_edited(true); //so it's saved
- }
- }
- void VisualScriptEditor::goto_line(int p_line, bool p_with_error){
- p_line+=1; //add one because script lines begin from 0.
- if (p_with_error)
- error_line=p_line;
- List<StringName> functions;
- script->get_function_list(&functions);
- for (List<StringName>::Element *E=functions.front();E;E=E->next()) {
- if (script->has_node(E->get(),p_line)) {
- edited_func=E->get();
- selected=edited_func;
- _update_graph();
- _update_members();
- call_deferred("_center_on_node",p_line); //editor might be just created and size might not exist yet
- return;
- }
- }
- }
- void VisualScriptEditor::trim_trailing_whitespace(){
- }
- void VisualScriptEditor::ensure_focus(){
- graph->grab_focus();
- }
- void VisualScriptEditor::tag_saved_version(){
- }
- void VisualScriptEditor::reload(bool p_soft){
- }
- void VisualScriptEditor::get_breakpoints(List<int> *p_breakpoints){
- List<StringName> functions;
- script->get_function_list(&functions);
- for (List<StringName>::Element *E=functions.front();E;E=E->next()) {
- List<int> nodes;
- script->get_node_list(E->get(),&nodes);
- for (List<int>::Element *F=nodes.front();F;F=F->next()) {
- Ref<VisualScriptNode> vsn = script->get_node(E->get(),F->get());
- if (vsn->is_breakpoint()) {
- p_breakpoints->push_back(F->get()-1); //subtract 1 because breakpoints in text start from zero
- }
- }
- }
- }
- bool VisualScriptEditor::goto_method(const String& p_method){
- if (!script->has_function(p_method))
- return false;
- edited_func=p_method;
- selected=edited_func;
- _update_members();
- _update_graph();
- return true;
- }
- void VisualScriptEditor::add_callback(const String& p_function,StringArray p_args){
- if (script->has_function(p_function)) {
- edited_func=p_function;
- selected=edited_func;
- _update_members();
- _update_graph();
- return;
- }
- Ref<VisualScriptFunction> func;
- func.instance();
- for(int i=0;i<p_args.size();i++) {
- String name = p_args[i];
- Variant::Type type=Variant::NIL;
- if (name.find(":")!=-1) {
- String tt = name.get_slice(":",1);
- name=name.get_slice(":",0);
- for(int j=0;j<Variant::VARIANT_MAX;j++) {
- String tname = Variant::get_type_name(Variant::Type(j));
- if (tname==tt) {
- type=Variant::Type(j);
- break;
- }
- }
- }
- func->add_argument(type,name);
- }
- func->set_name(p_function);
- script->add_function(p_function);
- script->add_node(p_function,script->get_available_id(),func);
- edited_func=p_function;
- selected=edited_func;
- _update_members();
- _update_graph();
- graph->call_deferred("set_scroll_ofs",script->get_function_scroll(edited_func)); //for first time it might need to be later
- //undo_redo->clear_history();
- }
- void VisualScriptEditor::update_settings(){
- _update_graph();
- }
- void VisualScriptEditor::set_debugger_active(bool p_active) {
- if (!p_active) {
- error_line=-1;
- _update_graph(); //clear line break
- }
- }
- void VisualScriptEditor::set_tooltip_request_func(String p_method,Object* p_obj){
- }
- Control *VisualScriptEditor::get_edit_menu(){
- return edit_menu;
- }
- void VisualScriptEditor::_change_base_type() {
- select_base_type->popup(true);
- }
- void VisualScriptEditor::_change_base_type_callback() {
- String bt = select_base_type->get_selected_type();
- ERR_FAIL_COND(bt==String());
- undo_redo->create_action("Change Base Type");
- undo_redo->add_do_method(script.ptr(),"set_instance_base_type",bt);
- undo_redo->add_undo_method(script.ptr(),"set_instance_base_type",script->get_instance_base_type());
- undo_redo->add_do_method(this,"_update_members");
- undo_redo->add_undo_method(this,"_update_members");
- undo_redo->commit_action();
- }
- void VisualScriptEditor::_node_selected(Node* p_node) {
- Ref<VisualScriptNode> vnode = p_node->get_meta("__vnode");
- if (vnode.is_null())
- return;
- EditorNode::get_singleton()->push_item(vnode.ptr()); //edit node in inspector
- }
- static bool _get_out_slot(const Ref<VisualScriptNode>& p_node,int p_slot,int& r_real_slot,bool& r_sequence) {
- if (p_slot<p_node->get_output_sequence_port_count()) {
- r_sequence=true;
- r_real_slot=p_slot;
- return true;
- }
- r_real_slot=p_slot-p_node->get_output_sequence_port_count();
- r_sequence=false;
- return (r_real_slot<p_node->get_output_value_port_count());
- }
- static bool _get_in_slot(const Ref<VisualScriptNode>& p_node,int p_slot,int& r_real_slot,bool& r_sequence) {
- if (p_slot==0 && p_node->has_input_sequence_port()) {
- r_sequence=true;
- r_real_slot=0;
- return true;
- }
- r_real_slot=p_slot-(p_node->has_input_sequence_port()?1:0);
- r_sequence=false;
- return r_real_slot<p_node->get_input_value_port_count();
- }
- void VisualScriptEditor::_begin_node_move() {
- undo_redo->create_action("Move Node(s)");
- }
- void VisualScriptEditor::_end_node_move() {
- undo_redo->commit_action();
- }
- void VisualScriptEditor::_move_node(String func,int p_id,const Vector2& p_to) {
- if (func==String(edited_func)) {
- Node* node = graph->get_node(itos(p_id));
- if (node && node->cast_to<GraphNode>())
- node->cast_to<GraphNode>()->set_offset(p_to);
- }
- script->set_node_pos(edited_func,p_id,p_to/EDSCALE);
- }
- void VisualScriptEditor::_node_moved(Vector2 p_from,Vector2 p_to, int p_id) {
- undo_redo->add_do_method(this,"_move_node",String(edited_func),p_id,p_to);
- undo_redo->add_undo_method(this,"_move_node",String(edited_func),p_id,p_from);
- }
- void VisualScriptEditor::_remove_node(int p_id) {
- undo_redo->create_action("Remove VisualScript Node");
- undo_redo->add_do_method(script.ptr(),"remove_node",edited_func,p_id);
- undo_redo->add_undo_method(script.ptr(),"add_node",edited_func,p_id,script->get_node(edited_func,p_id),script->get_node_pos(edited_func,p_id));
- List<VisualScript::SequenceConnection> sequence_conns;
- script->get_sequence_connection_list(edited_func,&sequence_conns);
- for (List<VisualScript::SequenceConnection>::Element *E=sequence_conns.front();E;E=E->next()) {
- if (E->get().from_node==p_id || E->get().to_node==p_id) {
- undo_redo->add_undo_method(script.ptr(),"sequence_connect",edited_func,E->get().from_node,E->get().from_output,E->get().to_node);
- }
- }
- List<VisualScript::DataConnection> data_conns;
- script->get_data_connection_list(edited_func,&data_conns);
- for (List<VisualScript::DataConnection>::Element *E=data_conns.front();E;E=E->next()) {
- if (E->get().from_node==p_id || E->get().to_node==p_id) {
- undo_redo->add_undo_method(script.ptr(),"data_connect",edited_func,E->get().from_node,E->get().from_port,E->get().to_node,E->get().to_port);
- }
- }
- undo_redo->add_do_method(this,"_update_graph");
- undo_redo->add_undo_method(this,"_update_graph");
- undo_redo->commit_action();
- }
- void VisualScriptEditor::_node_ports_changed(const String& p_func,int p_id) {
- if (p_func!=String(edited_func))
- return;
- _update_graph(p_id);
- }
- void VisualScriptEditor::_graph_connected(const String& p_from,int p_from_slot,const String& p_to,int p_to_slot) {
- Ref<VisualScriptNode> from_node = script->get_node(edited_func,p_from.to_int());
- ERR_FAIL_COND(!from_node.is_valid());
- bool from_seq;
- int from_port;
- if (!_get_out_slot(from_node,p_from_slot,from_port,from_seq))
- return; //can't connect this, it' s invalid
- Ref<VisualScriptNode> to_node = script->get_node(edited_func,p_to.to_int());
- ERR_FAIL_COND(!to_node.is_valid());
- bool to_seq;
- int to_port;
- if (!_get_in_slot(to_node,p_to_slot,to_port,to_seq))
- return; //can't connect this, it' s invalid
- ERR_FAIL_COND(from_seq!=to_seq);
- undo_redo->create_action("Connect Nodes");
- if (from_seq) {
- undo_redo->add_do_method(script.ptr(),"sequence_connect",edited_func,p_from.to_int(),from_port,p_to.to_int());
- undo_redo->add_undo_method(script.ptr(),"sequence_disconnect",edited_func,p_from.to_int(),from_port,p_to.to_int());
- } else {
- undo_redo->add_do_method(script.ptr(),"data_connect",edited_func,p_from.to_int(),from_port,p_to.to_int(),to_port);
- undo_redo->add_undo_method(script.ptr(),"data_disconnect",edited_func,p_from.to_int(),from_port,p_to.to_int(),to_port);
- //update nodes in sgraph
- undo_redo->add_do_method(this,"_update_graph",p_from.to_int());
- undo_redo->add_do_method(this,"_update_graph",p_to.to_int());
- undo_redo->add_undo_method(this,"_update_graph",p_from.to_int());
- undo_redo->add_undo_method(this,"_update_graph",p_to.to_int());
- }
- undo_redo->add_do_method(this,"_update_graph_connections");
- undo_redo->add_undo_method(this,"_update_graph_connections");
- undo_redo->commit_action();
- }
- void VisualScriptEditor::_graph_disconnected(const String& p_from,int p_from_slot,const String& p_to,int p_to_slot){
- Ref<VisualScriptNode> from_node = script->get_node(edited_func,p_from.to_int());
- ERR_FAIL_COND(!from_node.is_valid());
- bool from_seq;
- int from_port;
- if (!_get_out_slot(from_node,p_from_slot,from_port,from_seq))
- return; //can't connect this, it' s invalid
- Ref<VisualScriptNode> to_node = script->get_node(edited_func,p_to.to_int());
- ERR_FAIL_COND(!to_node.is_valid());
- bool to_seq;
- int to_port;
- if (!_get_in_slot(to_node,p_to_slot,to_port,to_seq))
- return; //can't connect this, it' s invalid
- ERR_FAIL_COND(from_seq!=to_seq);
- undo_redo->create_action("Connect Nodes");
- if (from_seq) {
- undo_redo->add_do_method(script.ptr(),"sequence_disconnect",edited_func,p_from.to_int(),from_port,p_to.to_int());
- undo_redo->add_undo_method(script.ptr(),"sequence_connect",edited_func,p_from.to_int(),from_port,p_to.to_int());
- } else {
- undo_redo->add_do_method(script.ptr(),"data_disconnect",edited_func,p_from.to_int(),from_port,p_to.to_int(),to_port);
- undo_redo->add_undo_method(script.ptr(),"data_connect",edited_func,p_from.to_int(),from_port,p_to.to_int(),to_port);
- //update nodes in sgraph
- undo_redo->add_do_method(this,"_update_graph",p_from.to_int());
- undo_redo->add_do_method(this,"_update_graph",p_to.to_int());
- undo_redo->add_undo_method(this,"_update_graph",p_from.to_int());
- undo_redo->add_undo_method(this,"_update_graph",p_to.to_int());
- }
- undo_redo->add_do_method(this,"_update_graph_connections");
- undo_redo->add_undo_method(this,"_update_graph_connections");
- undo_redo->commit_action();
- }
- void VisualScriptEditor::_default_value_changed() {
- Ref<VisualScriptNode> vsn = script->get_node(edited_func,editing_id);
- if (vsn.is_null())
- return;
- undo_redo->create_action("Change Input Value");
- undo_redo->add_do_method(vsn.ptr(),"set_default_input_value",editing_input,default_value_edit->get_variant());
- undo_redo->add_undo_method(vsn.ptr(),"set_default_input_value",editing_input,vsn->get_default_input_value(editing_input));
- undo_redo->add_do_method(this,"_update_graph",editing_id);
- undo_redo->add_undo_method(this,"_update_graph",editing_id);
- undo_redo->commit_action();
- }
- void VisualScriptEditor::_default_value_edited(Node * p_button,int p_id,int p_input_port) {
- Ref<VisualScriptNode> vsn = script->get_node(edited_func,p_id);
- if (vsn.is_null())
- return;
- PropertyInfo pinfo = vsn->get_input_value_port_info(p_input_port);
- Variant existing = vsn->get_default_input_value(p_input_port);
- if (pinfo.type!=Variant::NIL && existing.get_type()!=pinfo.type) {
- Variant::CallError ce;
- const Variant *existingp=&existing;
- existing = Variant::construct(pinfo.type,&existingp,1,ce,false);
- }
- default_value_edit->set_pos(p_button->cast_to<Control>()->get_global_pos()+Vector2(0,p_button->cast_to<Control>()->get_size().y));
- default_value_edit->set_size(Size2(1,1));
- if (default_value_edit->edit(NULL,pinfo.name,pinfo.type,existing,pinfo.hint,pinfo.hint_string))
- default_value_edit->popup();
- editing_id = p_id;
- editing_input=p_input_port;
- }
- void VisualScriptEditor::_show_hint(const String& p_hint) {
- hint_text->set_text(p_hint);
- hint_text->show();
- hint_text_timer->start();
- }
- void VisualScriptEditor::_hide_timer() {
- hint_text->hide();
- }
- void VisualScriptEditor::_node_filter_changed(const String& p_text) {
- _update_available_nodes();
- }
- void VisualScriptEditor::_notification(int p_what) {
- if (p_what==NOTIFICATION_READY) {
- node_filter_icon->set_texture(Control::get_icon("Zoom","EditorIcons"));
- }
- }
- void VisualScriptEditor::_graph_ofs_changed(const Vector2& p_ofs) {
- if (updating_graph)
- return;
- updating_graph=true;
- if (script->has_function(edited_func)) {
- script->set_function_scroll(edited_func,graph->get_scroll_ofs()/EDSCALE);
- script->set_edited(true);
- }
- updating_graph=false;
- }
- void VisualScriptEditor::_menu_option(int p_what) {
- switch(p_what) {
- case EDIT_DELETE_NODES: {
- _on_nodes_delete();
- } break;
- case EDIT_TOGGLE_BREAKPOINT: {
- List<String> reselect;
- for(int i=0;i<graph->get_child_count();i++) {
- GraphNode *gn = graph->get_child(i)->cast_to<GraphNode>();
- if (gn) {
- if (gn->is_selected()) {
- int id = String(gn->get_name()).to_int();
- Ref<VisualScriptNode> vsn = script->get_node(edited_func,id);
- if (vsn.is_valid()) {
- vsn->set_breakpoint(!vsn->is_breakpoint());
- reselect.push_back(gn->get_name());
- }
- }
- }
- }
- _update_graph();
- for(List<String>::Element *E=reselect.front();E;E=E->next()) {
- GraphNode *gn = graph->get_node(E->get())->cast_to<GraphNode>();
- gn->set_selected(true);
- }
- } break;
- case EDIT_FIND_NODE_TYPE: {
- //popup disappearing grabs focus to owner, so use call deferred
- node_filter->call_deferred("grab_focus");
- node_filter->call_deferred("select_all");
- } break;
- }
- }
- void VisualScriptEditor::_bind_methods() {
- ObjectTypeDB::bind_method("_member_button",&VisualScriptEditor::_member_button);
- ObjectTypeDB::bind_method("_member_edited",&VisualScriptEditor::_member_edited);
- ObjectTypeDB::bind_method("_member_selected",&VisualScriptEditor::_member_selected);
- ObjectTypeDB::bind_method("_update_members",&VisualScriptEditor::_update_members);
- ObjectTypeDB::bind_method("_change_base_type",&VisualScriptEditor::_change_base_type);
- ObjectTypeDB::bind_method("_change_base_type_callback",&VisualScriptEditor::_change_base_type_callback);
- ObjectTypeDB::bind_method("_override_pressed",&VisualScriptEditor::_override_pressed);
- ObjectTypeDB::bind_method("_node_selected",&VisualScriptEditor::_node_selected);
- ObjectTypeDB::bind_method("_node_moved",&VisualScriptEditor::_node_moved);
- ObjectTypeDB::bind_method("_move_node",&VisualScriptEditor::_move_node);
- ObjectTypeDB::bind_method("_begin_node_move",&VisualScriptEditor::_begin_node_move);
- ObjectTypeDB::bind_method("_end_node_move",&VisualScriptEditor::_end_node_move);
- ObjectTypeDB::bind_method("_remove_node",&VisualScriptEditor::_remove_node);
- ObjectTypeDB::bind_method("_update_graph",&VisualScriptEditor::_update_graph,DEFVAL(-1));
- ObjectTypeDB::bind_method("_node_ports_changed",&VisualScriptEditor::_node_ports_changed);
- ObjectTypeDB::bind_method("_available_node_doubleclicked",&VisualScriptEditor::_available_node_doubleclicked);
- ObjectTypeDB::bind_method("_default_value_edited",&VisualScriptEditor::_default_value_edited);
- ObjectTypeDB::bind_method("_default_value_changed",&VisualScriptEditor::_default_value_changed);
- ObjectTypeDB::bind_method("_menu_option",&VisualScriptEditor::_menu_option);
- ObjectTypeDB::bind_method("_graph_ofs_changed",&VisualScriptEditor::_graph_ofs_changed);
- ObjectTypeDB::bind_method("_center_on_node",&VisualScriptEditor::_center_on_node);
- ObjectTypeDB::bind_method("get_drag_data_fw",&VisualScriptEditor::get_drag_data_fw);
- ObjectTypeDB::bind_method("can_drop_data_fw",&VisualScriptEditor::can_drop_data_fw);
- ObjectTypeDB::bind_method("drop_data_fw",&VisualScriptEditor::drop_data_fw);
- ObjectTypeDB::bind_method("_input",&VisualScriptEditor::_input);
- ObjectTypeDB::bind_method("_on_nodes_delete",&VisualScriptEditor::_on_nodes_delete);
- ObjectTypeDB::bind_method("_on_nodes_duplicate",&VisualScriptEditor::_on_nodes_duplicate);
- ObjectTypeDB::bind_method("_hide_timer",&VisualScriptEditor::_hide_timer);
- ObjectTypeDB::bind_method("_graph_connected",&VisualScriptEditor::_graph_connected);
- ObjectTypeDB::bind_method("_graph_disconnected",&VisualScriptEditor::_graph_disconnected);
- ObjectTypeDB::bind_method("_update_graph_connections",&VisualScriptEditor::_update_graph_connections);
- ObjectTypeDB::bind_method("_node_filter_changed",&VisualScriptEditor::_node_filter_changed);
- }
- VisualScriptEditor::VisualScriptEditor() {
- updating_graph=false;
- edit_menu = memnew( MenuButton );
- edit_menu->set_text(TTR("Edit"));
- edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("visual_script_editor/delete_selected"), EDIT_DELETE_NODES);
- edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("visual_script_editor/toggle_breakpoint"), EDIT_TOGGLE_BREAKPOINT);
- edit_menu->get_popup()->add_shortcut(ED_GET_SHORTCUT("visual_script_editor/find_node_type"), EDIT_FIND_NODE_TYPE);
- edit_menu->get_popup()->connect("item_pressed",this,"_menu_option");
- main_hsplit = memnew( HSplitContainer );
- add_child(main_hsplit);
- main_hsplit->set_area_as_parent_rect();
- left_vsplit = memnew( VSplitContainer );
- main_hsplit->add_child(left_vsplit);
- VBoxContainer *left_vb = memnew( VBoxContainer );
- left_vsplit->add_child(left_vb);
- left_vb->set_v_size_flags(SIZE_EXPAND_FILL);
- left_vb->set_custom_minimum_size(Size2(180,1)*EDSCALE);
- base_type_select = memnew( Button );
- left_vb->add_margin_child(TTR("Base Type:"),base_type_select);
- base_type_select->connect("pressed",this,"_change_base_type");
- members = memnew( Tree );
- left_vb->add_margin_child(TTR("Members:"),members,true);
- members->set_hide_root(true);
- members->connect("button_pressed",this,"_member_button");
- members->connect("item_edited",this,"_member_edited");
- members->connect("cell_selected",this,"_member_selected",varray(),CONNECT_DEFERRED);
- members->set_single_select_cell_editing_only_when_already_selected(true);
- members->set_hide_folding(true);
- members->set_drag_forwarding(this);
- VBoxContainer *left_vb2 = memnew( VBoxContainer );
- left_vsplit->add_child(left_vb2);
- left_vb2->set_v_size_flags(SIZE_EXPAND_FILL);
- VBoxContainer *vbc_nodes = memnew( VBoxContainer );
- HBoxContainer *hbc_nodes = memnew( HBoxContainer );
- node_filter = memnew (LineEdit);
- node_filter->connect("text_changed",this,"_node_filter_changed");
- hbc_nodes->add_child(node_filter);
- node_filter->set_h_size_flags(SIZE_EXPAND_FILL);
- node_filter_icon = memnew( TextureFrame );
- node_filter_icon->set_stretch_mode(TextureFrame::STRETCH_KEEP_CENTERED);
- hbc_nodes->add_child(node_filter_icon);
- vbc_nodes->add_child(hbc_nodes);
- nodes = memnew( Tree );
- vbc_nodes->add_child(nodes);
- nodes->set_v_size_flags(SIZE_EXPAND_FILL);
- left_vb2->add_margin_child(TTR("Available Nodes:"),vbc_nodes,true);
- nodes->set_hide_root(true);
- nodes->connect("item_activated",this,"_available_node_doubleclicked");
- nodes->set_drag_forwarding(this);
- graph = memnew( GraphEdit );
- main_hsplit->add_child(graph);
- graph->set_h_size_flags(SIZE_EXPAND_FILL);
- graph->connect("node_selected",this,"_node_selected");
- graph->connect("_begin_node_move",this,"_begin_node_move");
- graph->connect("_end_node_move",this,"_end_node_move");
- graph->connect("delete_nodes_request",this,"_on_nodes_delete");
- graph->connect("duplicate_nodes_request",this,"_on_nodes_duplicate");
- graph->set_drag_forwarding(this);
- graph->hide();
- graph->connect("scroll_offset_changed",this,"_graph_ofs_changed");
- select_func_text = memnew( Label );
- select_func_text->set_text(TTR("Select or create a function to edit graph"));
- select_func_text->set_align(Label::ALIGN_CENTER);
- select_func_text->set_valign(Label::VALIGN_CENTER);
- select_func_text->set_h_size_flags(SIZE_EXPAND_FILL);
- main_hsplit->add_child(select_func_text);
- hint_text = memnew( Label );
- hint_text->set_anchor_and_margin(MARGIN_TOP,ANCHOR_END,100);
- hint_text->set_anchor_and_margin(MARGIN_BOTTOM,ANCHOR_END,0);
- hint_text->set_anchor_and_margin(MARGIN_RIGHT,ANCHOR_END,0);
- hint_text->set_align(Label::ALIGN_CENTER);
- hint_text->set_valign(Label::VALIGN_CENTER);
- graph->add_child(hint_text);
- hint_text_timer = memnew( Timer );
- hint_text_timer->set_wait_time(4);
- hint_text_timer->connect("timeout",this,"_hide_timer");
- add_child(hint_text_timer);
- //allowed casts (connections)
- for(int i=0;i<Variant::VARIANT_MAX;i++) {
- graph->add_valid_connection_type(Variant::NIL,i);
- graph->add_valid_connection_type(i,Variant::NIL);
- for(int j=0;j<Variant::VARIANT_MAX;j++) {
- if (Variant::can_convert(Variant::Type(i),Variant::Type(j))) {
- graph->add_valid_connection_type(i,j);
- }
- }
- graph->add_valid_right_disconnect_type(i);
- }
- graph->add_valid_left_disconnect_type(TYPE_SEQUENCE);
- graph->connect("connection_request",this,"_graph_connected");
- graph->connect("disconnection_request",this,"_graph_disconnected");
- edit_signal_dialog = memnew( AcceptDialog );
- edit_signal_dialog->get_ok()->set_text(TTR("Close"));
- add_child(edit_signal_dialog);
- edit_signal_dialog->set_title(TTR("Edit Signal Arguments:"));
- signal_editor = memnew( VisualScriptEditorSignalEdit );
- edit_signal_edit = memnew( PropertyEditor );
- edit_signal_edit->hide_top_label();
- edit_signal_dialog->add_child(edit_signal_edit);
- edit_signal_dialog->set_child_rect(edit_signal_edit);
- edit_signal_edit->edit(signal_editor);
- edit_variable_dialog = memnew( AcceptDialog );
- edit_variable_dialog->get_ok()->set_text(TTR("Close"));
- add_child(edit_variable_dialog);
- edit_variable_dialog->set_title(TTR("Edit Variable:"));
- variable_editor = memnew( VisualScriptEditorVariableEdit );
- edit_variable_edit = memnew( PropertyEditor );
- edit_variable_edit->hide_top_label();
- edit_variable_dialog->add_child(edit_variable_edit);
- edit_variable_dialog->set_child_rect(edit_variable_edit);
- edit_variable_edit->edit(variable_editor);
- select_base_type=memnew(CreateDialog);
- select_base_type->set_base_type("Object"); //anything goes
- select_base_type->connect("create",this,"_change_base_type_callback");
- select_base_type->get_ok()->set_text(TTR("Change"));
- add_child(select_base_type);
- undo_redo = EditorNode::get_singleton()->get_undo_redo();
- new_function_menu = memnew( PopupMenu );
- new_function_menu->connect("item_pressed",this,"_override_pressed");
- add_child(new_function_menu);
- updating_members=false;
- set_process_input(true); //for revert on drag
- set_process_unhandled_input(true); //for revert on drag
- default_value_edit= memnew( CustomPropertyEditor);
- add_child(default_value_edit);
- default_value_edit->connect("variant_changed",this,"_default_value_changed");
- error_line=-1;
- }
- VisualScriptEditor::~VisualScriptEditor() {
- undo_redo->clear_history(); //avoid crashes
- memdelete(signal_editor);
- memdelete(variable_editor);
- }
- static ScriptEditorBase * create_editor(const Ref<Script>& p_script) {
- if (p_script->cast_to<VisualScript>()) {
- return memnew( VisualScriptEditor );
- }
- return NULL;
- }
- static void register_editor_callback() {
- ScriptEditor::register_create_script_editor_function(create_editor);
- EditorSettings::get_singleton()->set("visual_script_editor/color_functions",Color(1,0.9,0.9));
- EditorSettings::get_singleton()->set("visual_script_editor/color_data",Color(0.9,1.0,0.9));
- EditorSettings::get_singleton()->set("visual_script_editor/color_operators",Color(0.9,0.9,1.0));
- EditorSettings::get_singleton()->set("visual_script_editor/color_flow_control",Color(1.0,1.0,0.8));
- EditorSettings::get_singleton()->set("visual_script_editor/color_custom",Color(0.8,1.0,1.0));
- ED_SHORTCUT("visual_script_editor/delete_selected", TTR("Delete Selected"));
- ED_SHORTCUT("visual_script_editor/toggle_breakpoint", TTR("Toggle Breakpoint"), KEY_F9);
- ED_SHORTCUT("visual_script_editor/find_node_type", TTR("Find Node Tyoe"), KEY_MASK_CMD+KEY_F);
- }
- void VisualScriptEditor::register_editor() {
- //too early to register stuff here, request a callback
- EditorNode::add_plugin_init_callback(register_editor_callback);
- }
|