function readObj(path) { if(!file_exists(path)) return noone; var _VB = []; var _VBT = []; var _VBN = []; var mats = []; var matIndex = []; var tris = []; var mtlPath = ""; var use_normal = true; var v = ds_list_create(); var vt = ds_list_create(); var vn = ds_list_create(); var f = ds_list_create(); var ft = ds_list_create(); var fn = ds_list_create(); var tri = 0; var file = file_text_open_read(path); while(!file_text_eof(file)) { #region reading file var l = file_text_readln(file); l = string_trim(l); var sep = string_split(l, " "); if(array_length(sep) == 0) continue; switch(sep[0]) { case "v" : ds_list_add(v, [ toNumber(sep[1]), toNumber(sep[2]), toNumber(sep[3]) ]); break; case "vt" : var _u = toNumber(sep[1]); var _v = toNumber(sep[2]); ds_list_add(vt, [ _u, _v ]); break; case "vn" : var _nx = toNumber(sep[1]); var _ny = toNumber(sep[2]); var _nz = toNumber(sep[3]); var _di = sqrt(_nx * _nx + _ny * _ny + _nz * _nz); _nx /= _di; _ny /= _di; _nz /= _di; ds_list_add(vn, [ _nx, _ny, _nz ]); break; case "f" : var _len = array_length(sep); var _f = array_create(_len - 1); var _ft = array_create(_len - 1); var _fn = array_create(_len - 1); for( var i = 1; i < _len; i++ ) { var _sp = string_split(sep[i], "/"); if(array_length(_sp) < 2) continue; _f[i - 1] = toNumber(array_safe_get(_sp, 0, 1)); _ft[i - 1] = toNumber(array_safe_get(_sp, 1, 1)); _fn[i - 1] = toNumber(array_safe_get(_sp, 2, 1)); if(array_length(_sp) < 3) use_normal = false; } tri += _len - 2; ds_list_add(f, _f ); //get position ds_list_add(ft, _ft); //get texture map ds_list_add(fn, _fn); //get normal break; case "usemtl" : var mname = ""; for( var i = 1; i < array_length(sep); i++ ) mname += (i == 1? "" : " ") + sep[i]; mname = string_trim(mname); array_push_unique(mats, mname); array_push(matIndex, array_find(mats, mname)); if(!ds_list_empty(f)) { array_push(_VB, f); array_push(_VBT, ft); array_push(_VBN, fn); array_push(tris, tri); f = ds_list_create(); ft = ds_list_create(); fn = ds_list_create(); } tri = 0; break; case "mtllib" : mtlPath = ""; for( var i = 1; i < array_length(sep); i++ ) mtlPath += (i == 1? "" : " ") + sep[i]; mtlPath = string_trim(mtlPath); break; case "o" : //print("Reading vertex group: " + sep[1]) break; } } if(!ds_list_empty(f)) { array_push(_VB, f); array_push(_VBT, ft); array_push(_VBN, fn); array_push(tris, tri); } file_text_close(file); #endregion if(use_normal) vn[| 0] = [ 0, 0, 0 ]; //var txt = "OBJ summary"; //txt += $"\n\tVerticies : {ds_list_size(v)}"; //txt += $"\n\tTexture Verticies : {ds_list_size(vt)}"; //txt += $"\n\tNormal Verticies : {ds_list_size(vn)}"; //txt += $"\n\tVertex groups : {array_length(_VB)}"; //txt += $"\n\tTriangles : {tris}"; //print(txt); #region centralize vertex var _bmin = v[| 0]; var _bmax = v[| 0]; var cv = [0, 0, 0]; var vertex = ds_list_size(v); for( var i = 0; i < vertex; i++ ) { var _v = v[| i]; cv[0] += _v[0]; cv[1] += _v[1]; cv[2] += _v[2]; _bmin = [ min(_bmin[0], _v[0]), min(_bmin[1], _v[1]), min(_bmin[2], _v[2]), ]; _bmax = [ max(_bmax[0], _v[0]), max(_bmax[1], _v[1]), max(_bmax[2], _v[2]), ]; } cv[0] /= vertex; cv[1] /= vertex; cv[2] /= vertex; for( var i = 0, n = ds_list_size(v); i < n; i++ ) { v[| i][0] -= cv[0]; v[| i][1] -= cv[1]; v[| i][2] -= cv[2]; } var _size = new __vec3( _bmax[0] - _bmin[0], _bmax[1] - _bmin[1], _bmax[2] - _bmin[2], ); #endregion #region vertex buffer creation var _vblen = array_length(_VB); var VBS = array_create(_vblen); var V = array_create(_vblen); for(var i = 0; i < _vblen; i++) { var VB = vertex_create_buffer(); vertex_begin(VB, global.VF_POS_NORM_TEX_COL); var face = _VB[i]; var facet = _VBT[i]; var facen = _VBN[i]; var tri = tris[i]; var _flen = ds_list_size(face); var _v = ds_list_create(); for(var j = 0; j < _flen; j++) { var _f = face[| j]; var _ft = facet[| j]; var _fn = facen[| j]; var _vlen = array_length(_f); var _pf = array_create(_vlen); var _pft = array_create(_vlen); var _pfn = array_create(_vlen); for( var k = 0; k < _vlen; k++ ) { var _vPindex = _f[k] - 1; _pf[k] = v[| _vPindex]; var _vNindex = _fn[k] - 1; _pfn[k] = vn[| _vNindex]; var _vTindex = _ft[k] - 1; _pft[k] = vt[| _vTindex]; } if(_vlen >= 3) { vertex_add_pntc(VB, _pf[0], _pfn[0], _pft[0]); vertex_add_pntc(VB, _pf[2], _pfn[2], _pft[2]); vertex_add_pntc(VB, _pf[1], _pfn[1], _pft[1]); ds_list_add(_v, new __vertex(_pf[0][0], _pf[0][1], _pf[0][2]).setNormal(_pfn[0][0], _pfn[0][1]).setUV(_pft[0][0], _pft[0][1])); ds_list_add(_v, new __vertex(_pf[2][0], _pf[2][1], _pf[2][2]).setNormal(_pfn[2][0], _pfn[2][1]).setUV(_pft[2][0], _pft[2][1])); ds_list_add(_v, new __vertex(_pf[1][0], _pf[1][1], _pf[1][2]).setNormal(_pfn[1][0], _pfn[1][1]).setUV(_pft[1][0], _pft[1][1])); } if(_vlen >= 4) { vertex_add_pntc(VB, _pf[0], _pfn[0], _pft[0]); vertex_add_pntc(VB, _pf[3], _pfn[3], _pft[3]); vertex_add_pntc(VB, _pf[2], _pfn[2], _pft[2]); ds_list_add(_v, new __vertex(_pf[0][0], _pf[0][1], _pf[0][2]).setNormal(_pfn[0][0], _pfn[0][1]).setUV(_pft[0][0], _pft[0][1])); ds_list_add(_v, new __vertex(_pf[3][0], _pf[3][1], _pf[3][2]).setNormal(_pfn[3][0], _pfn[3][1]).setUV(_pft[3][0], _pft[3][1])); ds_list_add(_v, new __vertex(_pf[2][0], _pf[2][1], _pf[2][2]).setNormal(_pfn[2][0], _pfn[2][1]).setUV(_pft[2][0], _pft[2][1])); } } vertex_end(VB); vertex_freeze(VB); VBS[i] = VB; V[i] = ds_list_to_array(_v); ds_list_destroy(_v); } #endregion #region clean array_foreach(_VB, function(val, ind) { ds_list_destroy(val); }); array_foreach(_VBT, function(val, ind) { ds_list_destroy(val); }); array_foreach(_VBN, function(val, ind) { ds_list_destroy(val); }); ds_list_destroy(v); ds_list_destroy(vn); ds_list_destroy(vt); #endregion return { vertex: V, vertex_groups: VBS, object_counts: _vblen, materials: mats, material_index: matIndex, use_normal: use_normal, mtl_path: mtlPath, model_size: _size, }; }