Improve .tscn VCS

Serialize dictionaries adding newlines between key-value pairs
Serialize group lists also with newlines in between
Serialize string properties escaping only " and \ (needed for a good diff experience with built-in scripts and shaders)

Bonus:
Make AnimationPlayer serialize its blend times always sorted so their order is predictable in the .tscn file.

This PR is back-compat; won't break the load of existing files.
This commit is contained in:
Pedro J. Estébanez 2017-01-16 18:03:38 +01:00
parent 681575fa71
commit 7dbb1c0571
6 changed files with 32 additions and 21 deletions

View file

@ -3286,8 +3286,17 @@ String String::c_escape() const {
escaped=escaped.replace("\t","\\t"); escaped=escaped.replace("\t","\\t");
escaped=escaped.replace("\v","\\v"); escaped=escaped.replace("\v","\\v");
escaped=escaped.replace("\'","\\'"); escaped=escaped.replace("\'","\\'");
escaped=escaped.replace("\"","\\\"");
escaped=escaped.replace("\?","\\?"); escaped=escaped.replace("\?","\\?");
escaped=escaped.replace("\"","\\\"");
return escaped;
}
String String::c_escape_multiline() const {
String escaped=*this;
escaped=escaped.replace("\\","\\\\");
escaped=escaped.replace("\"","\\\"");
return escaped; return escaped;
} }

View file

@ -216,6 +216,7 @@ public:
String http_escape() const; String http_escape() const;
String http_unescape() const; String http_unescape() const;
String c_escape() const; String c_escape() const;
String c_escape_multiline() const;
String c_unescape() const; String c_unescape() const;
String json_escape() const; String json_escape() const;
String word_wrap(int p_chars_per_line) const; String word_wrap(int p_chars_per_line) const;

View file

@ -1878,7 +1878,7 @@ Error VariantWriter::write(const Variant& p_variant, StoreStringFunc p_store_str
String str=p_variant; String str=p_variant;
str="\""+str.c_escape()+"\""; str="\""+str.c_escape_multiline()+"\"";
p_store_string_func(p_store_string_ud, str ); p_store_string_func(p_store_string_ud, str );
} break; } break;
case Variant::VECTOR2: { case Variant::VECTOR2: {
@ -2123,7 +2123,7 @@ Error VariantWriter::write(const Variant& p_variant, StoreStringFunc p_store_str
dict.get_key_list(&keys); dict.get_key_list(&keys);
keys.sort(); keys.sort();
p_store_string_func(p_store_string_ud,"{ "); p_store_string_func(p_store_string_ud,"{\n");
for(List<Variant>::Element *E=keys.front();E;E=E->next()) { for(List<Variant>::Element *E=keys.front();E;E=E->next()) {
//if (!_check_type(dict[E->get()])) //if (!_check_type(dict[E->get()]))
@ -2132,11 +2132,11 @@ Error VariantWriter::write(const Variant& p_variant, StoreStringFunc p_store_str
p_store_string_func(p_store_string_ud,": "); p_store_string_func(p_store_string_ud,": ");
write(dict[E->get()],p_store_string_func,p_store_string_ud,p_encode_res_func,p_encode_res_ud); write(dict[E->get()],p_store_string_func,p_store_string_ud,p_encode_res_func,p_encode_res_ud);
if (E->next()) if (E->next())
p_store_string_func(p_store_string_ud,", "); p_store_string_func(p_store_string_ud,",\n");
} }
p_store_string_func(p_store_string_ud," }"); p_store_string_func(p_store_string_ud,"\n}");
} break; } break;

View file

@ -118,17 +118,20 @@ bool AnimationPlayer::_get(const StringName& p_name,Variant &r_ret) const {
} else if (name=="blend_times") { } else if (name=="blend_times") {
Array array; Vector<BlendKey> keys;
array.resize(blend_times.size()*3);
int idx=0;
for(Map<BlendKey, float >::Element *E=blend_times.front();E;E=E->next()) { for(Map<BlendKey, float >::Element *E=blend_times.front();E;E=E->next()) {
array.set(idx*3+0,E->key().from); keys.ordered_insert(E->key());
array.set(idx*3+1,E->key().to);
array.set(idx*3+2,E->get());
idx++;
} }
Array array;
for(int i=0;i<keys.size();i++) {
array.push_back(keys[i].from);
array.push_back(keys[i].to);
array.push_back(blend_times[keys[i]]);
}
r_ret=array; r_ret=array;
} else if (name=="autoplay") { } else if (name=="autoplay") {
r_ret=autoplay; r_ret=autoplay;

View file

@ -142,7 +142,7 @@ private:
StringName from; StringName from;
StringName to; StringName to;
bool operator<(const BlendKey& bk) const { return from==bk.from?to<bk.to:from<bk.from; } bool operator<(const BlendKey& bk) const { return from==bk.from?String(to)<String(bk.to):String(from)<String(bk.from); }
}; };

View file

@ -1158,7 +1158,7 @@ void ResourceFormatSaverTextInstance::_find_resources(const Variant& p_variant,b
static String _valprop(const String& p_name) { static String _valprop(const String& p_name) {
if (p_name.find("\"")!=-1 || p_name.find("=")!=-1 || p_name.find(" ")!=-1) if (p_name.find("\"")!=-1 || p_name.find("=")!=-1 || p_name.find(" ")!=-1)
return "\""+p_name.c_escape()+"\""; return "\""+p_name.c_escape_multiline()+"\"";
return p_name; return p_name;
} }
@ -1358,11 +1358,9 @@ Error ResourceFormatSaverTextInstance::save(const String &p_path,const RES& p_re
} }
if (groups.size()) { if (groups.size()) {
String sgroups=" groups=[ "; String sgroups=" groups=[\n";
for(int j=0;j<groups.size();j++) { for(int j=0;j<groups.size();j++) {
if (j>0) sgroups+="\""+String(groups[j]).c_escape()+"\",\n";
sgroups+=", ";
sgroups+="\""+groups[j].operator String().c_escape()+"\"";
} }
sgroups+="]"; sgroups+="]";
header+=sgroups; header+=sgroups;