crfReviewSection.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. function generateReviewSection(listName,id,callback){
  2. //callback should be generateReviewSectionCB and it takes no arguments
  3. print("generateReviewSection");
  4. //need base path
  5. config.loadFileConfig=new Object();
  6. config.loadFileConfig.cb=callback;
  7. config.loadFileConfig.id=id;
  8. config.loadFileConfig.url=getBasePath()+'/@files/reportSetup/'+listName+'.json';
  9. loadFile();
  10. //load file and continue in the next function
  11. }
  12. function getParticipantCode(pid){
  13. let filters=[LABKEY.Filter.create("crfRef",getCRFref())];
  14. let mfId=config.formConfig.form['masterQuery'];
  15. let queryName=config.formConfig.queryMap[mfId];
  16. pid.afterId=setParticipantCode;
  17. pid.participantField=config.formConfig.studyData["SubjectColumnName"];
  18. let cb=function(data){afterRegistration(pid,data);}
  19. //untested
  20. cvSelectRows('lists',queryName,filters,cb,getContainer('data'));
  21. }
  22. function visitCodeFromVisitId(visitId){
  23. if (visitId<0) return "NONE";
  24. let project=getContainer('data');
  25. print('visitCodeFromVisitId: '+project.search('retro'));
  26. if (project.search('retro')>-1)
  27. visitId-=1;
  28. return 'VISIT_'+visitId.toString();
  29. }
  30. function replaceSlash(x){
  31. return x.replace(/\//,'_');
  32. }
  33. function setParticipantCode(pid){
  34. let fName='[setParticipantCode]';
  35. let rows=pid.registration.rows;
  36. //pick from study
  37. let participantField=config.formConfig.studyData["SubjectColumnName"];
  38. if (rows.length==1){
  39. print(fName+': '+rows[0][participantField]+'/'+rows[0].visitId);
  40. let visitCode=visitCodeFromVisitId(rows[0].visitId);
  41. print('setParticipantCode: '+pid.participantId+'/'+visitCode);
  42. pid.participantCode=replaceSlash(pid.participantId);
  43. pid.visitCode=visitCode;
  44. }
  45. generateReviewSection2(pid);
  46. }
  47. function generateReviewSectionCB(){
  48. let listName=config.loadFileConfig.listName;
  49. let id=config.loadFileConfig.id;
  50. clearErrorMessage(listName);
  51. let pid=new Object();
  52. pid.participantCode="NONE";
  53. pid.visitCode="NONE";
  54. getParticipantCode(pid);
  55. print('Get participant code sent');
  56. //involves database search, continue after callback
  57. }
  58. function getValueFromElement(id,defaultValue){
  59. let e=config.document.getElementById(id);
  60. if (e!=null){
  61. defaultValue=e.innerHTML;
  62. }
  63. return defaultValue;
  64. }
  65. function pickParticipantCodeFromPage(){
  66. let pid=new Object();
  67. pid.participantCode=getValueFromElement("participantCode","NIX-LJU-D2002-IRAE-A000");
  68. pid.visitCode=getValueFromElement("visitCode","VISIT_1");
  69. generateReviewSection2(pid);
  70. }
  71. function patternReplace(src,replacements,values){
  72. for (rep in replacements){
  73. let txt1=src.replace(new RegExp(rep),values[replacements[rep]]);
  74. src=txt1;
  75. }
  76. return src;
  77. }
  78. function plotImage(cell,k,row,rowVariable,obj,pid){
  79. let baseDir=patternReplace(obj.imageDir,obj.replacements,pid);
  80. print('Base dir: '+pid.basePath);
  81. pid[obj.variable]=obj.values[k];
  82. cell.id=pid[obj.variable]+"_"+rowVariable+pid[rowVariable];
  83. let img=null;
  84. let imgId=cell.id+'_img_';
  85. img=config.document.getElementById(imgId);
  86. if (img===null){
  87. img=config.document.createElement('img');
  88. img.id=imgId;
  89. cell.appendChild(img);
  90. }
  91. let imgSrc=patternReplace(obj.file,obj.replacements,pid);
  92. print('Image: '+imgSrc);
  93. let imagePath=pid.basePath+'/'+baseDir+'/'+imgSrc;
  94. img.src=imagePath;
  95. img.width="300";
  96. }
  97. function showReport(cell,k,row,rowVariable,obj,pid){
  98. cell.width="300px";
  99. cell.id='report_'+obj.values[k]+"_"+rowVariable+pid[rowVariable];
  100. let reportConfig=new Object();
  101. reportConfig.partName="Report";
  102. reportConfig.renderTo=cell.id;
  103. //reportConfig.showFrame=false;
  104. //reportConfig.width="300";
  105. reportConfig.frame="none";
  106. reportConfig.partConfig=new Object();
  107. reportConfig.partConfig.width="300";
  108. reportConfig.partConfig.title="R Report";
  109. reportConfig.partConfig.reportName=obj.values[k];
  110. for (f in obj.parameters){
  111. reportConfig.partConfig[f]=pid[f];
  112. }
  113. reportConfig.partConfig.showSection="myscatterplot";
  114. let reportWebPartRenderer = new LABKEY.WebPart(reportConfig);
  115. print('Render to: '+reportConfig.renderTo);
  116. reportWebPartRenderer.render();
  117. }
  118. function showProbability(cell,k,row,rowSetup,j,obj,pid){
  119. print('showProbability: '+rowSetup);
  120. let rowVariable=rowSetup.variable;
  121. cell.id='prob_'+obj.values[k]+"_"+rowVariable+pid[rowVariable];
  122. let probDensity=new Object();
  123. probDensity.mean=rowSetup.mean[j];
  124. probDensity.sigma=rowSetup.sigma[j];
  125. print('showProbability: mean '+probDensity.mean+' sigma '+probDensity.sigma);
  126. probDensity.func=obj.values[k];
  127. probDensity.organCode=pid.organCode;
  128. pid[obj.variable]=rowSetup[obj.variable][j];
  129. probDensity.percentile=pid.percentile;
  130. let selectRows=new Object();
  131. selectRows.queryName=obj.queryName;
  132. selectRows.schemaName="study";
  133. selectRows.filterArray=[];
  134. selectRows.containerPath=getContainer('data');
  135. for (let f in obj.filters){
  136. selectRows.filterArray.push(
  137. LABKEY.Filter.create(f,pid[obj.filters[f]]));
  138. print('Filter ['+f+']: '+pid[obj.filters[f]]);
  139. }
  140. selectRows.success=function(data){
  141. drawProbability(data,cell,obj,pid,probDensity);}
  142. LABKEY.Query.selectRows(selectRows);
  143. }
  144. function erf(x){
  145. let fx=[0,0.02,0.04,0.06,0.08,0.1,0.2,0.3,0.4,0.5,0.6,0.7,0.8,0.9,
  146. 1,1.1,1.2,1.3,1.4,1.5,1.6,1.7,1.8,1.9,2,
  147. 2.1,2.2,2.3,2.4,2.5,3,3.5];
  148. let fy=[0,0.222702589,0.328626759,0.428392355,0.520499878,
  149. 0.603856091,0.677801194,0.742100965,0.796908212,
  150. 0.842700793,0.880205070,0.910313978,0.934007945,
  151. 0.952285120,0.966105146,0.976348383,
  152. 0.983790459,0.989090502,0.992790429,0.995322265,
  153. 0.997020533,0.998137154,0.998856823,0.999311486,
  154. 0.999593048,0.999977910,0.999999257];
  155. let n=32;
  156. let i0=n-1;
  157. for (let i=1;i<n;i++){
  158. if (Math.abs(x)>fx[i]) continue;
  159. i0=i-1;
  160. break;
  161. }
  162. let fval=1;
  163. if (i0<n-1){
  164. //interpolate
  165. let y1=fy[i0+1];
  166. let y0=fy[i0];
  167. let x1=fx[i0+1];
  168. let x0=fx[i0];
  169. fval=y0+(y1-y0)/(x1-x0)*(Math.abs(x)-x0);
  170. }
  171. print('Erf: '+fval);
  172. if (x<0) return -fval;
  173. return fval;
  174. }
  175. function setLine(fbox,name,value,fontSize){
  176. let fpId=fbox.id+name;
  177. let fp=config.document.getElementById(fpId);
  178. if (fp===null){
  179. fp=config.document.createElement("p");
  180. fp.id=fpId;
  181. fbox.appendChild(fp);
  182. }
  183. fp.classList.add("center");
  184. fp.style.textAlign="center";
  185. fp.style.fontSize=fontSize;
  186. fp.innerText=value;
  187. }
  188. function drawProbability(data,cell,obj,pid,probDensity){
  189. print('drawProbability');
  190. if (data.rows.length!=1){
  191. print("drawProbability row length mismatch: "+data.rows.length);
  192. return;
  193. }
  194. //possible mismatch; I assume the dataset will have a field called value
  195. let val=data.rows[0].value;
  196. let prob=0;
  197. let fz=-100;
  198. if (probDensity.func=="gaus"){
  199. fz=(val-probDensity.mean)/probDensity.sigma/Math.sqrt(2);
  200. prob=0.5+0.5*erf(fz);
  201. }
  202. let color="red";
  203. let fzx=fz*Math.sqrt(2);
  204. print('drawProbability '+fzx);
  205. for (let i=1;i<obj.intervals.n;i++){
  206. if (fzx>obj.intervals.zlimits[i]) continue;
  207. color=obj.intervals.colors[i-1];
  208. break;
  209. }
  210. let fboxId=cell.id+'_fbox_';
  211. let fbox=config.document.getElementById(fboxId);
  212. if (fbox===null){
  213. fbox=config.document.createElement("div");
  214. fbox.id=fboxId;
  215. cell.appendChild(fbox);
  216. }
  217. fbox.style.backgroundColor=color;
  218. fbox.style.width="180px";
  219. fbox.style.height="180px";
  220. print('organCode '+probDensity.organCode);
  221. let organName="Lung";
  222. if (probDensity.organCode==4){
  223. organName="Thyroid";
  224. }
  225. if (probDensity.organCode==5){
  226. organName="Bowel";
  227. }
  228. setLine(fbox,'_fp4_',organName,"16px");
  229. setLine(fbox,'_fp_',val.toPrecision(3),"25px");
  230. setLine(fbox,'_fp1_',"SUV("+probDensity.percentile+"%)","16px");
  231. setLine(fbox,'_fp2_',fzx.toPrecision(3),"25px");
  232. setLine(fbox,'_fp3_',"z-value","16px");
  233. }
  234. function generateReviewSection2(pid){
  235. let listName=config.loadFileConfig.listName;
  236. let id=config.loadFileConfig.id;
  237. print('generateReviewSection2: '+pid.participantCode+'/'+
  238. pid.visitCode);
  239. if (pid.participantCode=="NONE" || pid.visitCode=="NONE"){
  240. generateErrorMessage(id,listName,
  241. "ParticipantId/visitId not set");
  242. return;
  243. }
  244. print('JSON: '+config.loadFileConfig.json);
  245. let json=config.loadFileConfig.json;
  246. let nrows=json.rows.values.length;
  247. let ncol=json.columns.length;
  248. pid.basePath=getBasePath()+"/@files";
  249. let el=config.document.getElementById(id);
  250. let tableId=id+'_Table';
  251. let table=config.document.getElementById(tableId);
  252. if (table==null){
  253. table=config.document.createElement('table');
  254. table.id=tableId;
  255. el.appendChild(table);
  256. }
  257. table.style.tableLayout="fixed";
  258. table.style.columnWidth="300px";
  259. for (let i=0;i<nrows;i++){
  260. pid[json.rows.variable]=json.rows.values[i];
  261. //let organ=organs[i];
  262. let row=null;
  263. if (i<table.rows.length)
  264. row=table.rows[i];
  265. else
  266. row=table.insertRow();
  267. let ic=0;
  268. for (let j=0;j<ncol;j++){
  269. let obj=json.columns[j];
  270. let nv=obj.values.length;
  271. for (let k=0;k<nv;k++){
  272. let cell=null;
  273. if (ic<row.cells.length)
  274. cell=row.cells[ic];
  275. else
  276. cell=row.insertCell();
  277. if (obj.display=="image")
  278. plotImage(cell,k,row,json.rows.variable,obj,pid);
  279. if (obj.display=="report")
  280. showReport(cell,k,row,json.rows.variable,obj,pid);
  281. if (obj.display=="probability"){
  282. showProbability(cell,k,row,json.rows,i,obj,pid);
  283. }
  284. ic++;
  285. }
  286. }
  287. }
  288. }
  289. ///>>>>>>>>>>>>>>end of reviewSection(REPORT)
  290. function afterRegistration(data,fc){
  291. let fName='[afterRegistration/'+data.queryName+']';
  292. print(fName+": rows:"+data.rows.length);
  293. fc.registration=data;
  294. let registrationData=fc.registration;
  295. clearErr();
  296. if (registrationData.rows.length!=1){
  297. let msg=fName+": ERROR: Found "+registrationData.rows.length;
  298. msg+=" registration entries for crfrefid "+getCRFref();
  299. print(msg);
  300. fc.afterId(fc);
  301. return;
  302. }
  303. print(fName+'registration participant field: '+fc.participantField);
  304. fc.participantId=registrationData.rows[0][fc.participantField];
  305. //could be a lookup field (particularly for studies)
  306. print('ID: '+fc.participantId);
  307. let fields=registrationData.metaData.fields;
  308. let field="NONE";
  309. for (f in fields){
  310. if (fields[f]["name"]==fc.participantField)
  311. field=fields[f];
  312. }
  313. if ("lookup" in field){
  314. let pid=fc.participantId;
  315. print("Using lookup for participantId: "+pid);
  316. let lookup=field["lookup"];
  317. print("Lookup: ["+lookup.schemaName+','+lookup.queryName+']');
  318. //load lookup
  319. let cb=function(data){afterRegistrationLookup(data,lookup.displayColumn,fc)};
  320. let filters=[LABKEY.Filter.create(lookup.keyColumn,pid)];
  321. cvSelectRows(lookup.schemaName,lookup.queryName,filters,cb,lookup.containerPath);
  322. }
  323. else{
  324. //afterParticipantId(configUpload);
  325. fc.afterId(fc);
  326. }
  327. }
  328. function afterRegistrationLookup(data,displayColumn,fc){
  329. print("afterRegistrationLookup");
  330. let entry=data.rows[0];
  331. fc.participantId=entry[displayColumn];
  332. print('Setting to '+fc.participantId);
  333. fc.afterId(fc);
  334. //afterParticipantId(configUpload);
  335. }