curvedraw
changeset 8:8bf96e11ed1f
curves file format parsing (untested)
author | John Tsiombikas <nuclear@member.fsf.org> |
---|---|
date | Sat, 19 Dec 2015 08:00:09 +0200 (2015-12-19) |
parents | 6e980fddbf3b |
children | c2008ba88d03 |
files | src/curvefile.cc |
diffstat | 1 files changed, 134 insertions(+), 3 deletions(-) [+] |
line diff
1.1 --- a/src/curvefile.cc Fri Dec 18 07:07:19 2015 +0200 1.2 +++ b/src/curvefile.cc Sat Dec 19 08:00:09 2015 +0200 1.3 @@ -1,4 +1,6 @@ 1.4 #include <stdlib.h> 1.5 +#include <ctype.h> 1.6 +#include <string> 1.7 #include "curvefile.h" 1.8 1.9 static bool save_curve(FILE *fp, const Curve *curve); 1.10 @@ -15,7 +17,7 @@ 1.11 1.12 bool save_curves(FILE *fp, const Curve * const *curves, int count) 1.13 { 1.14 - fprintf(stderr, "GCURVES\ncount %d\n", count); 1.15 + fprintf(stderr, "GCURVES\n"); 1.16 1.17 for(int i=0; i<count; i++) { 1.18 if(!save_curve(fp, curves[i])) { 1.19 @@ -53,7 +55,7 @@ 1.20 1.21 Curve **load_curves(const char *fname, int *countret) 1.22 { 1.23 - FILE *fp = fopen(fname, "rb"); 1.24 + FILE *fp = fopen(fname, "r"); 1.25 if(!fp) return false; 1.26 1.27 Curve **res = load_curves(fp, countret); 1.28 @@ -61,8 +63,137 @@ 1.29 return res; 1.30 } 1.31 1.32 +static std::string next_token(FILE *fp) 1.33 +{ 1.34 + std::string s; 1.35 + int c; 1.36 + while((c = fgetc(fp)) != -1 && isspace(c)); // skip whitespace 1.37 + if(feof(fp)) return s; 1.38 + while((c = fgetc(fp)) != -1 && !isspace(c)) { 1.39 + s.push_back(c); 1.40 + } 1.41 + return s; 1.42 +} 1.43 + 1.44 +static bool expect_str(FILE *fp, const char *s) 1.45 +{ 1.46 + std::string tok = next_token(fp); 1.47 + if(tok != std::string(s)) { 1.48 + if(!(tok.empty() && feof(fp))) { 1.49 + fprintf(stderr, "expected: %s\n", s); 1.50 + } 1.51 + return false; 1.52 + } 1.53 + return true; 1.54 +} 1.55 + 1.56 +static bool expect_float(FILE *fp, float *ret) 1.57 +{ 1.58 + std::string tok = next_token(fp); 1.59 + const char *cs = tok.c_str(); 1.60 + char *endp; 1.61 + *ret = strtod(cs, &endp); 1.62 + if(endp != cs + tok.length()) { 1.63 + if(!(tok.empty() && feof(fp))) { 1.64 + fprintf(stderr, "number expected\n"); 1.65 + } 1.66 + return false; 1.67 + } 1.68 + return true; 1.69 +} 1.70 + 1.71 +static bool expect_int(FILE *fp, int *ret) 1.72 +{ 1.73 + std::string tok = next_token(fp); 1.74 + const char *cs = tok.c_str(); 1.75 + char *endp; 1.76 + *ret = strtol(cs, &endp, 0); 1.77 + if(endp != cs + tok.length()) { 1.78 + if(!(tok.empty() && feof(fp))) { 1.79 + fprintf(stderr, "integer expected\n"); 1.80 + } 1.81 + return false; 1.82 + } 1.83 + return true; 1.84 +} 1.85 + 1.86 +static Curve *curve_block(FILE *fp) 1.87 +{ 1.88 + if(!expect_str(fp, "curve") || !expect_str(fp, "{")) { 1.89 + return 0; 1.90 + } 1.91 + 1.92 + Curve *curve = new Curve; 1.93 + int cpcount = -1; 1.94 + std::string tok; 1.95 + while(!(tok = next_token(fp)).empty() && tok != "}") { 1.96 + if(tok == "cpcount") { 1.97 + if(cpcount != -1 || !expect_int(fp, &cpcount) || cpcount <= 0) { 1.98 + goto err; 1.99 + } 1.100 + } else if(tok == "type") { 1.101 + tok = next_token(fp); 1.102 + if(tok == "polyline") { 1.103 + curve->set_type(CURVE_LINEAR); 1.104 + } else if(tok == "hermite") { 1.105 + curve->set_type(CURVE_HERMITE); 1.106 + } else if(tok == "bspline") { 1.107 + curve->set_type(CURVE_BSPLINE); 1.108 + } else { 1.109 + goto err; 1.110 + } 1.111 + } else { 1.112 + if(!expect_str(fp, "cp")) { 1.113 + goto err; 1.114 + } 1.115 + Vector3 cp; 1.116 + for(int i=0; i<3; i++) { 1.117 + if(!expect_float(fp, &cp[i])) { 1.118 + goto err; 1.119 + } 1.120 + } 1.121 + curve->add_point(Vector2(cp.x, cp.y), cp.z); 1.122 + } 1.123 + } 1.124 + 1.125 + if(curve->size() != cpcount) { 1.126 + fprintf(stderr, "warning: curve cpcount was %d, but read %d control points\n", cpcount, curve->size()); 1.127 + } 1.128 + 1.129 + return curve; 1.130 +err: 1.131 + fprintf(stderr, "failed to parse curve block\n"); 1.132 + delete curve; 1.133 + return 0; 1.134 +} 1.135 + 1.136 Curve **load_curves(FILE *fp, int *countret) 1.137 { 1.138 - return 0; // TODO 1.139 + if(!expect_str(fp, "GCURVES")) { 1.140 + fprintf(stderr, "load_curves: failed to load, invalid file format\n"); 1.141 + return 0; 1.142 + } 1.143 + 1.144 + std::vector<Curve*> curves; 1.145 + Curve *curve; 1.146 + 1.147 + while((curve = curve_block(fp))) { 1.148 + curves.push_back(curve); 1.149 + } 1.150 + 1.151 + int ncurves = (int)curves.size(); 1.152 + if(!feof(fp)) { 1.153 + for(int i=0; i<ncurves; i++) { 1.154 + delete curves[i]; 1.155 + } 1.156 + return 0; 1.157 + } 1.158 + 1.159 + Curve **carr = new Curve*[ncurves]; 1.160 + for(int i=0; i<ncurves; i++) { 1.161 + carr[i] = curves[i]; 1.162 + } 1.163 + *countret = ncurves; 1.164 + return carr; 1.165 } 1.166