formPortal.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540
  1. //global config variable
  2. const config=new Object();
  3. function print(msg){
  4. config.document.getElementById(config.debugArea).value+="\n"+msg;
  5. }
  6. function clear(){
  7. config.document.getElementById(config.debugArea).value="";
  8. }
  9. function getMode(){
  10. if ("role" in config){
  11. return config.role;
  12. }
  13. return "crfEditor";
  14. }
  15. function doNothing(){
  16. print('doNothing called');
  17. }
  18. function makeQuery(containerName,queryName,fieldName,filterArray){
  19. //queryArray should contain elements with
  20. //- fieldName to set the data variable
  21. //- containerName to select container (data,config,CRF)
  22. //- queryName to select query
  23. //- filterArray to perform filtering, empty array works
  24. //- callback cb to be called with no arguments
  25. let e=new Object();
  26. e.containerName=containerName;
  27. e.queryName=queryName;
  28. e.fieldName=fieldName;
  29. e.filterArray=filterArray;
  30. return e;
  31. }
  32. function getDataFromQueries(queryArray,cb){
  33. afterQuery(new Object(),-1,queryArray,cb);
  34. }
  35. function afterQuery(data,id,queryArray,cb){
  36. print('afterQuery['+id+']: ');
  37. if (id>-1){
  38. let fieldName=queryArray[id].fieldName;
  39. print('afterQuery['+fieldName+']: '+data.rows.length);
  40. //uses config.formConfig
  41. config.formConfig[fieldName]=data;
  42. }
  43. id+=1;
  44. if (id==queryArray.length) {
  45. cb();
  46. return;
  47. }
  48. let e=queryArray[id];
  49. let qconfig=new Object();
  50. qconfig.containerPath=getContainer(e.containerName);
  51. qconfig.schemaName="lists";
  52. if ("schemaName" in e){
  53. print('afterQuery: schemaName='+e.schemaName);
  54. qconfig.schemaName=e.schemaName;
  55. }
  56. if ("columns" in e){
  57. print('afterQuery: columns='+e.columns);
  58. qconfig.columns=e.columns;
  59. }
  60. qconfig.queryName=e.queryName;
  61. //this should point to configuration container
  62. //don't filter -> so we can pick up other forms (say registration) later on
  63. //qconfig.filterArray=[LABKEY.Filter.create('Key',config.formId)];
  64. if ("filterArray" in e)
  65. qconfig.filterArray=e.filterArray;
  66. //qconfig.filterArray=[LABKEY.Filter.create('formStatus',1)]
  67. qconfig.success=function(data){afterQuery(data,id,queryArray,cb);};
  68. qconfig.failure=doNothing;
  69. LABKEY.Query.selectRows(qconfig);
  70. }
  71. function printMessage(msg){
  72. let txt=config.document.createElement("p");
  73. config.document.getElementById(config.div).appendChild(txt);
  74. txt.innerText=msg;
  75. }
  76. function userName(id){
  77. let formConfig=config.formConfig;
  78. for (let i=0;i<formConfig.users.rows.length;i++){
  79. if (formConfig.users.rows[i].UserId!=id)
  80. continue;
  81. return formConfig.users.rows[i].DisplayName;
  82. }
  83. return "NONE";
  84. }
  85. function setContainer(label,container){
  86. if (!(config.formConfig.hasOwnProperty('container'))){
  87. config.formConfig.container=new Array();
  88. }
  89. config.formConfig.container[label]=container;
  90. }
  91. function getContainer(label){
  92. return config.formConfig.container[label];
  93. }
  94. function generateFormArray(){
  95. print("generateFormArray "+getMode());
  96. config.formConfig=new Object();
  97. config.formConfig.softwareVersion='T.1.11';
  98. //report software version
  99. config.document.getElementById('version').innerText=config.formConfig.softwareVersion;
  100. setContainer('data',LABKEY.ActionURL.getContainer());
  101. setContainer('config',LABKEY.ActionURL.getContainer());
  102. setContainer('CRF',LABKEY.ActionURL.getContainer());
  103. let selectRows=new Object();
  104. //this is local data
  105. selectRows.containerPath=getContainer('CRF');
  106. selectRows.schemaName='lists';
  107. selectRows.queryName='crfSettings';
  108. //store form related data to this object
  109. selectRows.success=afterSettings;
  110. LABKEY.Query.selectRows(selectRows);
  111. }
  112. function afterSettings(data){
  113. config.formConfig.settings=new Array();
  114. for (let i=0;i<data.rows.length;i++){
  115. let n=data.rows[i]['name'];
  116. let v=data.rows[i]['value'];
  117. config.formConfig.settings[n]=v;
  118. }
  119. let st=config.formConfig.settings;
  120. print('afterSettings');
  121. for (let k in st){
  122. print('\t'+k+'='+st[k]);
  123. }
  124. //if ('dataContainer' in st){
  125. // setContainer('data',st['dataContainer']);
  126. //}
  127. let vname='configContainer';
  128. if (vname in st){
  129. setContainer('config',st[vname]);
  130. }
  131. print('Config: '+getContainer('config'));
  132. print('Data: '+getContainer('data'));
  133. //setup queryArray
  134. let queryArray=new Array();
  135. //static variables
  136. queryArray.push(makeQuery('data','crfStaticVariables','crfStaticVariables',[]));
  137. //Forms
  138. queryArray.push(makeQuery('config','Forms','dataForms',[]));
  139. //users
  140. queryArray.push(makeQuery('data','users','users',[]));
  141. queryArray[queryArray.length-1].schemaName='core';
  142. //inputLists
  143. queryArray.push(makeQuery('config','inputLists','inputLists',[]));
  144. //crfEditors
  145. queryArray.push(makeQuery('config','crfEditors','crfEditors',[]));
  146. //crfMonitors
  147. queryArray.push(makeQuery('config','crfMonitors','crfMonitors',[]));
  148. //crfSponsors
  149. queryArray.push(makeQuery('config','crfSponsors','crfSponsors',[]));
  150. //crfManagers
  151. queryArray.push(makeQuery('config','crfManagers','crfManagers',[]));
  152. //FormStatus
  153. queryArray.push(makeQuery('config','FormStatus','formStatusg',[]));
  154. //site
  155. queryArray.push(makeQuery('config','site','siteData',[]));
  156. //crfEntry
  157. queryArray.push(makeQuery('data','crfEntry','crfEntries',[]));
  158. getDataFromQueries(queryArray,addStudyData);
  159. //getDataFromQueries(queryArray,fcontinue);
  160. }
  161. function addStudyData(){
  162. //setup queryArray
  163. let queryArray=new Array();
  164. queryArray.push(makeQuery('data','StudyProperties','studyData',[]));
  165. let e=queryArray[queryArray.length-1];
  166. e.schemaName='study';
  167. let columnModel="";
  168. let varRows=config.formConfig['crfStaticVariables'].rows;
  169. for (let i=0;i<varRows.length;i++){
  170. if (i>0) columnModel+=',';
  171. columnModel+=varRows[i]['staticVariable'];
  172. }
  173. e.columns=columnModel;
  174. getDataFromQueries(queryArray,fcontinue);
  175. }
  176. function filterEntry(entry,filter,settings){
  177. if (entry.Form!=filter.form)
  178. return false;
  179. //only select forms where status matches the target status
  180. if (entry.FormStatus!=filter.formStatus){
  181. return false;
  182. }
  183. print('Candidate '+entry.entryId);
  184. //TODO: smart filter on user (now we get to see all)
  185. //
  186. //for editors
  187. if ("filterUser" in settings && filter.role=='crfEditor' && entry.UserId!=filter.userId){
  188. print('Skipping identity mismatch: '+entry.UserId+'/'+filter.userId);
  189. return false;
  190. }
  191. //for others
  192. let matchingSite=-1;
  193. let potentialSiteNumbers="[";
  194. for (let k=0;k<filter.sites.length;k++){
  195. if (k>0) potentialSiteNumbers+=',';
  196. potentialSiteNumbers+=filter.sites[k].siteNumber;
  197. //skip mismatching sites
  198. if (entry.Site!=filter.sites[k].siteNumber) continue;
  199. matchingSite=filter.sites[k].siteNumber;
  200. break;
  201. }
  202. potentialSiteNumbers+=']';
  203. if (matchingSite==-1){
  204. print('Skipping wrong site: '+entry.Site+'/'+potentialSiteNumbers);
  205. return false;
  206. }
  207. return true;
  208. }
  209. function fcontinue(){
  210. let fName='[fcontinue]';
  211. let formConfig=config.formConfig;
  212. print("Number of study data entries: "+formConfig.studyData.rows.length);
  213. print("ParticipantId: "+formConfig.studyData.rows[0].SubjectColumnName);
  214. let dataForms=formConfig.dataForms.rows;
  215. formConfig.table=config.document.createElement("table");
  216. config.document.getElementById(config.div).appendChild(formConfig.table);
  217. let accessModeColumn=getMode()+'Status';
  218. print('accessModeColumn '+accessModeColumn);
  219. //cutting down on number of fields
  220. //let creatorModeColumn=getMode()+'Creator';
  221. //switch from status based to form based access
  222. print("Forms: "+dataForms.length);
  223. print("Entries: "+formConfig.crfEntries.rows.length);
  224. let fEntries=formConfig.crfEntries.rows;
  225. let users=formConfig.users.rows;
  226. let currentUserId=LABKEY.Security.currentUser.id;
  227. let currentUser=undefined;
  228. for (let i=0;i<users.length;i++){
  229. if (users[i].UserId!=currentUserId) continue;
  230. currentUser=users[i];
  231. }
  232. //determine the role filter
  233. let fList=config.role+'s';
  234. //check for users that fit the role,
  235. //fRows lists all users for role
  236. let fRows=config.formConfig[fList].rows;
  237. print(fName+' candidates: '+fRows.length)
  238. //current user must be in the list
  239. let currentUserRoles=new Array();
  240. //the same user can act for multiple sites
  241. for (let i=0;i<fRows.length;i++){
  242. if (fRows[i].User!=currentUser.UserId) continue;
  243. currentUserRoles.push(fRows[i]);
  244. }
  245. //cludge for public sites where all users can act as anything
  246. let sts=config.formConfig.settings;
  247. let vName='allowAllForSite';
  248. if (vName in sts){
  249. let tempUserRole=new Object();
  250. tempUserRole.User=currentUser.UserId;
  251. tempUserRole.Site=parseInt(sts[vName]);
  252. currentUserRoles.push(tempUserRole);
  253. }
  254. //currentUser was not matched in fRows
  255. if (currentUserRoles.length==0){
  256. printMessage('User '+currentUser.DisplayName+" can't act as "+config.role);
  257. return;
  258. }
  259. //currentUser should be also attached to the site of the document
  260. let currentSites=new Array();
  261. let siteRows=config.formConfig.siteData.rows;
  262. for (let i=0;i<siteRows.length;i++){
  263. for (let j=0;j<currentUserRoles.length;j++){
  264. if (siteRows[i].siteNumber!=currentUserRoles[j].Site) continue;
  265. currentSites.push(siteRows[i]);
  266. }
  267. }
  268. config.formConfig.currentSites=currentSites;
  269. let msg='User '+currentUser.DisplayName+' acting as '+config.role+' for (';
  270. for (let i=0;i<currentSites.length;i++){
  271. if (i>0) msg+=', ';
  272. msg+=currentSites[i].siteName;
  273. }
  274. msg+=')';
  275. printMessage(msg);
  276. let filter=new Object();
  277. filter.role=config.role;
  278. filter.userId=currentUser.UserId;
  279. filter.sites=currentSites;
  280. //browse through forms
  281. for (let i=0;i<dataForms.length;i++){
  282. //dataForms is Forms
  283. let qForm=dataForms[i];
  284. let formKey=qForm.Key;
  285. filter.form=qForm.Key;
  286. //add row for each form
  287. let row=formConfig.table.insertRow(i);
  288. let formName=qForm.formName;
  289. print("["+i+"/"+formKey+']: '+formName);
  290. //column counter
  291. let k=0;
  292. //get the target status
  293. let formStatus=qForm[accessModeColumn];
  294. filter.formStatus=qForm[accessModeColumn];
  295. print('target formStatus '+formStatus);
  296. for (let j=0;j<fEntries.length;j++){
  297. let entry=fEntries[j];
  298. if (!filterEntry(entry,filter,config.formConfig.settings))
  299. continue;
  300. //insert form
  301. //
  302. let fbox=config.document.createElement("div");
  303. fbox.classList.add("box","gold");
  304. let fp=config.document.createElement("p");
  305. let id=entry.entryId;
  306. fp.innerHTML=id;
  307. //it would be great if this were patientId if available
  308. //fp.classList.add("large","center");
  309. fp.classList.add("center");
  310. fbox.appendChild(fp);
  311. let fp1=config.document.createElement("p");
  312. let user="NONE";
  313. for (let ii=0;ii<users.length;ii++){
  314. if (users[ii].UserId!=entry.UserId)
  315. continue;
  316. user=users[ii].DisplayName;
  317. break;
  318. }
  319. fp1.innerHTML=user;
  320. fp1.classList.add("center");
  321. fbox.appendChild(fp1);
  322. let fp2=config.document.createElement("p");
  323. fp2.innerHTML=formName;
  324. fp2.classList.add("center");
  325. fbox.appendChild(fp2);
  326. let fp3=config.document.createElement("p");
  327. fp3.id="pid"+id;
  328. let pid=entry['participantStudyId'];
  329. let loc=entry['participantLocalId'];
  330. let label='';
  331. if (pid) label+=pid+' ';
  332. if (loc) label+='(Local: '+loc+')';
  333. if (label.length==0) label="NONE";
  334. fp3.innerHTML=label;
  335. fp3.classList.add("center");
  336. fbox.appendChild(fp3);
  337. let cell=row.insertCell(k);
  338. cell.classList.add("stretch");
  339. cell.id="box"+entry.crfRef;
  340. let button=config.document.createElement("button");
  341. button.appendChild(fbox);
  342. button.onclick=function(){openForm(entry)};
  343. cell.appendChild(button);
  344. k++;
  345. }
  346. print('finished checking existing forms');
  347. //only those that are allowed to create forms
  348. //print('Status: '+qForm[creatorModeColumn]);
  349. let creator=qForm['creator'];
  350. if (!creator) continue;
  351. if (creator!=getMode()) continue;
  352. //if (qForm[creatorModeColumn]!='TRUE') continue;
  353. let fbox=config.document.createElement("div");
  354. fbox.classList.add("box","red");
  355. let fp=config.document.createElement("p");
  356. fp.innerHTML="Create new";
  357. fbox.appendChild(fp);
  358. let fp2=config.document.createElement("p");
  359. fp2.innerHTML=formName;
  360. fp2.classList.add("center");
  361. fbox.appendChild(fp2);
  362. let cell=row.insertCell(k);
  363. cell.classList.add("stretch");
  364. let button=config.document.createElement("button");
  365. button.appendChild(fbox);
  366. button.onclick=function(){createForm(formKey)};
  367. cell.appendChild(button);
  368. }
  369. }
  370. function openForm(crfEntry){
  371. let formConfig=config.formConfig;
  372. let crfRef=crfEntry.entryId;
  373. print("Clicked for "+crfRef);
  374. let formId=crfEntry.Form;
  375. for (let i=0;i<formConfig.dataForms.rows.length;i++){
  376. if (formConfig.dataForms.rows[i].Key!=formId) continue;
  377. print("Setting form "+formConfig.dataForms.rows[i].formName);
  378. formEntry=formConfig.dataForms.rows[i];
  379. break;
  380. }
  381. if (formEntry==undefined) return;
  382. //select between review and view
  383. //let formUrl=formEntry["formUrl"];
  384. //if ("reviewMode" in config) formUrl=formEntry["reviewFormUrl"];
  385. //print("Setting url "+formUrl);
  386. //direct all to the same html
  387. let formUrl="visit";
  388. reviewMode="EDIT";
  389. if ("reviewMode" in config) reviewMode=config.reviewMode;
  390. let params = {
  391. "name": formUrl,
  392. // The destination wiki page. The name of this parameter is not arbitrary.
  393. "entryId": crfRef,
  394. "formId":formId,
  395. "role" : config.role
  396. };
  397. //"formSetupQuery":formEntry["setupQuery"],
  398. let containerPath= LABKEY.ActionURL.getContainer();
  399. // This changes the page after building the URL.
  400. //Note that the wiki page destination name is set in params.
  401. var wikiURL = LABKEY.ActionURL.buildURL("crf_tecant", formUrl , containerPath, params);
  402. print("Redirecting to "+wikiURL);
  403. window.location = wikiURL;
  404. }
  405. function createForm(formId){
  406. let formConfig=config.formConfig;
  407. print("Create form w/id "+formId);
  408. let crfEntry=new Object();
  409. crfEntry.entryId=Date.now();
  410. crfEntry["Date"]=new Date();
  411. crfEntry["View"]="[VIEW]";
  412. crfEntry.formStatus=1;//In progress
  413. //set other variables
  414. //requires studyData as part of formConfig
  415. let studyData=formConfig.studyData.rows[0];
  416. let varRows=formConfig['crfStaticVariables'].rows;
  417. for (let i=0;i<varRows.length;i++){
  418. let varName=varRows[i].staticVariable;
  419. crfEntry[varName]=studyData[varName];
  420. }
  421. crfEntry.UserId=LABKEY.Security.currentUser.id;
  422. crfEntry.Site=config.formConfig.currentSites[0].siteNumber;
  423. print("Setting site to id="+crfEntry.Site);
  424. //from argument list
  425. crfEntry.Form=formId;
  426. let qconfig=new Object();
  427. qconfig.containerPath=getContainer('data');
  428. qconfig.schemaName='lists';
  429. qconfig.queryName='crfEntry';
  430. qconfig.success=function(data){openForm(crfEntry)};
  431. qconfig.rows=[crfEntry];
  432. LABKEY.Query.insertRows(qconfig);
  433. }