|
@@ -18,78 +18,6 @@ function print(msg){
|
|
|
el.value+="\n"+msg;
|
|
|
}
|
|
|
|
|
|
-function makeQuery(containerName,queryName,fieldName,filterArray){
|
|
|
- let e=new Object();
|
|
|
- e.containerName=containerName;
|
|
|
- e.queryName=queryName;
|
|
|
- e.fieldName=fieldName;
|
|
|
- e.filterArray=filterArray;
|
|
|
- return e;
|
|
|
-}
|
|
|
-
|
|
|
-function getDataFromQueries(queryArray,cb){
|
|
|
- //queryArray should contain elements with
|
|
|
- //- fieldName to set the data variable
|
|
|
- //- containerName to select container (data,config,CRF)
|
|
|
- //- queryName to select query
|
|
|
- //- filterArray to perform filtering, empty array works
|
|
|
- //- callback cb to be called with no arguments
|
|
|
- //
|
|
|
- afterQuery(new Object(),-1,queryArray,cb);
|
|
|
-}
|
|
|
-
|
|
|
-function afterQuery(data,id,queryArray,cb){
|
|
|
- //queryArray should contain elements with
|
|
|
- //- fieldName to set the data variable
|
|
|
- //- containerName to select container (data,config,CRF)
|
|
|
- //- queryName to select query
|
|
|
- //- filterArray to perform filtering, empty array works
|
|
|
- //- callback cb to be called with no arguments
|
|
|
- //
|
|
|
- //it should be called with id -1.
|
|
|
- //
|
|
|
- print('afterQuery['+id+']: ');
|
|
|
-
|
|
|
- if (id>-1){
|
|
|
- let fieldName=queryArray[id].fieldName;
|
|
|
- print('afterQuery['+fieldName+']: '+data.rows.length);
|
|
|
- config.formConfig[fieldName]=data;
|
|
|
- }
|
|
|
- id+=1;
|
|
|
- if (id==queryArray.length) {
|
|
|
- cb();
|
|
|
- return;
|
|
|
- }
|
|
|
-
|
|
|
-
|
|
|
- let e=queryArray[id];
|
|
|
- let qconfig=new Object();
|
|
|
- qconfig.containerPath=getContainer(e.containerName);
|
|
|
- qconfig.schemaName="lists";
|
|
|
- if ("schemaName" in e){
|
|
|
- print('afterQuery: schemaName='+e.schemaName);
|
|
|
- qconfig.schemaName=e.schemaName;
|
|
|
- }
|
|
|
-
|
|
|
- if ("columns" in e){
|
|
|
- print('afterQuery: columns='+e.columns);
|
|
|
- qconfig.columns=e.columns;
|
|
|
- }
|
|
|
- qconfig.queryName=e.queryName;
|
|
|
- //this should point to configuration container
|
|
|
- //don't filter -> so we can pick up other forms (say registration) later on
|
|
|
- //qconfig.filterArray=[LABKEY.Filter.create('Key',config.formId)];
|
|
|
- if ("filterArray" in e)
|
|
|
- qconfig.filterArray=e.filterArray;
|
|
|
-
|
|
|
- //qconfig.filterArray=[LABKEY.Filter.create('formStatus',1)]
|
|
|
- qconfig.success=function(data){afterQuery(data,id,queryArray,cb);};
|
|
|
- qconfig.failure=doNothing;
|
|
|
- LABKEY.Query.selectRows(qconfig);
|
|
|
-
|
|
|
-}
|
|
|
-
|
|
|
-
|
|
|
function getCRFrefFirst(){
|
|
|
//crfRef is part of html call and gets stored in the page
|
|
|
return config.document.getElementById(config.crfRefId).innerHTML;
|
|
@@ -173,40 +101,60 @@ function generateDebugSection(){
|
|
|
|
|
|
|
|
|
function getAdditionalData(formSetupEntry){
|
|
|
+ //return information on additional data associated with the form
|
|
|
+ //additionalData is a sub-list with multiple entries per patient/visit
|
|
|
+
|
|
|
|
|
|
+ //argument is the row of the formSetup setup list
|
|
|
let queryName=config.formConfig.queryMap[formSetupEntry['queryName']];
|
|
|
let fName='[getAdditionalData/'+queryName+']';
|
|
|
print(fName);
|
|
|
|
|
|
+ //additionalData holds a reference to all queries already parsed
|
|
|
+ //this helps in reducing number of calls to the database (I assume)
|
|
|
if (queryName in config.formConfig.additionalData){
|
|
|
print(fName+': Returning preset value');
|
|
|
return config.formConfig.additionalData[queryName];
|
|
|
}
|
|
|
+
|
|
|
+ //first time we see this query, so we have to do the setup
|
|
|
print(fName+': generating');
|
|
|
config.formConfig.additionalData[queryName]=new Object();
|
|
|
- //takes address, so further changes will be to the newly created object
|
|
|
+
|
|
|
+ //takes address, so further changes will be to the newly created object
|
|
|
+ //in fact, ad is just a short alias of the long variable name on the right
|
|
|
let ad=config.formConfig.additionalData[queryName];
|
|
|
|
|
|
+ //no additional data
|
|
|
if (formSetupEntry["showFlag"]==="NONE") {
|
|
|
print(fName+": empty");
|
|
|
return ad;
|
|
|
}
|
|
|
+
|
|
|
+ //use showFlag to setup report section of the CRF list
|
|
|
if (formSetupEntry["showFlag"]==="REVIEW") {
|
|
|
//abuse additionalData to signal different segment
|
|
|
print(fName+": generateReport");
|
|
|
ad.isReview=true;
|
|
|
return ad;
|
|
|
}
|
|
|
+
|
|
|
+ //setup the additionalData memory object
|
|
|
print(fName+': setting values');
|
|
|
ad.showFlag=formSetupEntry["showFlag"];
|
|
|
ad.showFlagValue=formSetupEntry["showFlagValue"];
|
|
|
ad.queryName=formSetupEntry["showQuery"];
|
|
|
+
|
|
|
+ //for data queries, limit to present CRF only
|
|
|
ad.filters=new Object();
|
|
|
ad.filters['crfRef']=getCRFref();
|
|
|
+
|
|
|
+ //compose a long debug message
|
|
|
let msg=fName+": flag "+ad.showFlag;
|
|
|
msg+=" value "+ad.showFlagValue;
|
|
|
msg+=" query "+ad.queryName;
|
|
|
print(msg);
|
|
|
+
|
|
|
return ad;
|
|
|
}
|
|
|
|
|
@@ -260,12 +208,7 @@ function getSetup(listName,writeAccess=true){
|
|
|
// return readonlySetup(listName);
|
|
|
return fullAccessSetup(listName);
|
|
|
}
|
|
|
-
|
|
|
-//afterFormConfig replaced by afterFormSetup
|
|
|
-
|
|
|
-//afterFormSetupLookup replaced by afterFormDatasets
|
|
|
|
|
|
-//function generateSection(listName, sectionTitle, accessMode, additionalData){
|
|
|
function generateSection(formSetupEntry){
|
|
|
|
|
|
let listName=config.formConfig.queryMap[formSetupEntry['queryName']];
|
|
@@ -1055,19 +998,36 @@ function addFieldRow(tb,field,setup,additionalData){
|
|
|
let vType=field.type;
|
|
|
let isLookup=("lookup" in field);
|
|
|
print(fName+": ["+vName+"/"+vType+'/'+isLookup+"]");
|
|
|
+
|
|
|
+ let detailsRow=config.formConfig['variableDetails'].rows;
|
|
|
+ let details=undefined;
|
|
|
+ for (let i=0;i<detailsRow.length;i++){
|
|
|
+ if (detailsRow[i]['variableName']==field.name){
|
|
|
+ print(fName+": ["+vName+"]: details "+detailsRow[i].description);
|
|
|
+ details=detailsRow[i].description;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
let row=tb.insertRow();
|
|
|
let cell=config.document.createElement('th');
|
|
|
+ cell.style.width='300px';
|
|
|
row.appendChild(cell);
|
|
|
|
|
|
let text = config.document.createTextNode(field.shortCaption);
|
|
|
cell.appendChild(text);
|
|
|
- let cell1=row.insertCell();
|
|
|
|
|
|
|
|
|
let input=null;
|
|
|
- cell1.colSpan="3";
|
|
|
-
|
|
|
+ let colSpan="3";
|
|
|
+ if (details!=undefined){
|
|
|
+ let cell2=row.insertCell();
|
|
|
+ cell2.colSpan="2";
|
|
|
+ cell2.innerText=details;
|
|
|
+ colSpan="1";
|
|
|
+ }
|
|
|
+ let cell1=row.insertCell();
|
|
|
+ cell1.colSpan=colSpan;
|
|
|
let readonlyFlag=setup.readonlyFlag(vName);
|
|
|
|
|
|
|
|
@@ -2421,22 +2381,54 @@ function generateMasterForm(){
|
|
|
//(fields defined in html file)
|
|
|
function populateBasicData(){
|
|
|
|
|
|
- config.document.getElementById('version').innerText=config.formConfig.softwareVersion;
|
|
|
- config.document.getElementById('eudraCTNumber').innerText=
|
|
|
- config.formConfig.crfEntry.EudraCTNumber;
|
|
|
- config.document.getElementById('studyCoordinator').innerText=
|
|
|
- config.formConfig.crfEntry.StudyCoordinator;
|
|
|
- config.document.getElementById('studySponsor').innerText=
|
|
|
- config.formConfig.crfEntry.StudySponsor;
|
|
|
- config.document.getElementById('siteName').innerText=
|
|
|
- config.formConfig.currentSite['siteName'];
|
|
|
- config.document.getElementById('sitePhone').innerText=
|
|
|
- config.formConfig.currentSite['sitePhone'];
|
|
|
- config.document.getElementById('investigatorName').innerText=
|
|
|
- config.formConfig.user['DisplayName'];
|
|
|
+ let staticData=new Object();
|
|
|
+ let titles=new Object();
|
|
|
+ staticData['version']=config.formConfig.softwareVersion;
|
|
|
+ titles['version']='Software version';
|
|
|
+ let varRows=config.formConfig['crfStaticVariables'].rows;
|
|
|
+ for (let i=0;i<varRows.length;i++){
|
|
|
+ let vName=varRows[i].staticVariable;
|
|
|
+ let val=config.formConfig.crfEntry[vName];
|
|
|
+ if (val==undefined) continue;
|
|
|
+ staticData[vName]=val;
|
|
|
+ titles[vName]=varRows[i].Title;
|
|
|
+ }
|
|
|
+ staticData['investigatorName']=config.formConfig.user['DisplayName'];
|
|
|
+ titles['investigatorName']='Investigator';
|
|
|
+ staticData['email']=config.formConfig.user['Email'];
|
|
|
+ titles['email']='Email';
|
|
|
+ staticData['siteName']=config.formConfig.currentSite['siteName'];
|
|
|
+ titles['siteName']='Site';
|
|
|
+ staticData['sitePhone']=config.formConfig.currentSite['sitePhone'];
|
|
|
+ titles['sitePhone']='Telephone(site)';
|
|
|
+
|
|
|
+ for (f in staticData){
|
|
|
+ addStaticData(f,titles[f],staticData[f]);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+function addStaticData(f,title,value){
|
|
|
+ let el=config.document.getElementById(f);
|
|
|
+
|
|
|
+ //populate only
|
|
|
+ if (el!=undefined){
|
|
|
+ el.innerText=value;
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ //add row to table if element cannot be found
|
|
|
+ let table=config.document.getElementById('staticTable');
|
|
|
+ let row=table.insertRow();
|
|
|
+ let cell=row.insertCell();
|
|
|
+ cell.innerText=title;
|
|
|
+ let cell1=row.insertCell();
|
|
|
+ cell1.id=f;
|
|
|
+ cell1.style.fontWeight='bold';
|
|
|
+
|
|
|
+ //populate
|
|
|
+ cell1.innerText=value;
|
|
|
}
|
|
|
|
|
|
-
|
|
|
//come here after the layout is read from labkey page
|
|
|
//
|
|
|
function generateErrorMsg(msg){
|
|
@@ -2581,6 +2573,24 @@ function afterData(){
|
|
|
//operatorBasedAccessMode
|
|
|
let accessMode=config.formConfig.operator+'Mode';
|
|
|
let rowsSetup=config.formConfig.formSetupRows;
|
|
|
+
|
|
|
+ print(fName+': '+'generatePariticpantEntry');
|
|
|
+ //generateParticipantEntryField();
|
|
|
+ //add print to config so participantManager can use it
|
|
|
+ config.print=print;
|
|
|
+ let pM=getParticipantManagerObject(config);
|
|
|
+
|
|
|
+ print(fName+': pariticpantManager: '+pM);
|
|
|
+ //check if ParticipantId is set in crfEntry
|
|
|
+ let pId=config.formConfig.crfEntry['ParticipantId'];
|
|
|
+ print(fName+' pId '+pId);
|
|
|
+ if (!pId){
|
|
|
+ pM.setParticipantManagerToEditMode();
|
|
|
+ }
|
|
|
+ else{
|
|
|
+ pM.setParticipantManagerToLabelMode(pId,'Record');
|
|
|
+ }
|
|
|
+
|
|
|
for (let i=0;i<rowsSetup.length;i++){
|
|
|
let entry=rowsSetup[i];
|
|
|
let queryName=config.formConfig.queryMap[entry['queryName']];
|
|
@@ -2602,6 +2612,7 @@ function afterData(){
|
|
|
//let additionalData=new Object();
|
|
|
//setAdditionalData(additionalData,entry);
|
|
|
//section fits one dataset/list
|
|
|
+
|
|
|
generateSection(entry);
|
|
|
//generateSection(queryName,entry["title"],entry[accessMode],
|
|
|
// additionalData);
|
|
@@ -2928,7 +2939,7 @@ function setFormConfig(){
|
|
|
//add object to store form related data
|
|
|
config.formConfig=new Object();
|
|
|
|
|
|
- config.formConfig.softwareVersion='0.14.18';
|
|
|
+ config.formConfig.softwareVersion='0.15.08';
|
|
|
let debug=true;
|
|
|
|
|
|
if (debug)
|
|
@@ -3004,50 +3015,98 @@ function afterCRFEntry(data){
|
|
|
}
|
|
|
|
|
|
function collectData(){
|
|
|
-
|
|
|
- let varLabel='sourceFormStatus';
|
|
|
- let formStatus=config.formConfig.crfEntry['FormStatus'];
|
|
|
-
|
|
|
+
|
|
|
+ let targetObject=config.formConfig;
|
|
|
+ targetObject.print=print;
|
|
|
+ targetObject.getContainer=getContainer;
|
|
|
let queryArray=new Array();
|
|
|
+ //k
|
|
|
//site
|
|
|
- queryArray.push(makeQuery('config','site','siteData',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'config','site','siteData',[]));
|
|
|
//users
|
|
|
- queryArray.push(makeQuery('CRF','users','userData',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'CRF','users','userData',[]));
|
|
|
queryArray[queryArray.length-1].schemaName='core';
|
|
|
//crfEditors
|
|
|
- queryArray.push(makeQuery('config','crfEditors','crfEditorsData',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'config','crfEditors','crfEditorsData',[]));
|
|
|
//crfMonitors
|
|
|
- queryArray.push(makeQuery('config','crfMonitors','crfMonitorsData',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'config','crfMonitors','crfMonitorsData',[]));
|
|
|
//crfSponsors
|
|
|
- queryArray.push(makeQuery('config','crfSponsors','crfSponsorsData',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'config','crfSponsors','crfSponsorsData',[]));
|
|
|
+
|
|
|
+ //study static data
|
|
|
+ queryArray.push(
|
|
|
+ makeQuery(targetObject,'data','crfStaticVariables','crfStaticVariables',[]));
|
|
|
+
|
|
|
+ queryArray.push(makeQuery(targetObject,'data','variableDetails','variableDetails',[]));
|
|
|
//study
|
|
|
- queryArray.push(makeQuery('data','Study','studyDataAll',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'data','Study','studyDataAll1',[]));
|
|
|
let e=queryArray[queryArray.length-1];
|
|
|
+ //overload schema name
|
|
|
e.schemaName='study';
|
|
|
+ //make sure variables not part of default view are loaded
|
|
|
+ //here we should already have read crfStaticVariables table
|
|
|
e.columns="SubjectColumnName,EudraCTNumber,StudySponsor";
|
|
|
e.columns+=",StudyCoordinator,RegulatoryNumber";
|
|
|
|
|
|
//formStatus
|
|
|
+ let varLabel='sourceFormStatus';
|
|
|
+ let formStatus=config.formConfig.crfEntry['FormStatus'];
|
|
|
let formFilter=LABKEY.Filter.create('Key',formStatus);
|
|
|
- queryArray.push(makeQuery('config','FormStatus','formStatusData',[formFilter]));
|
|
|
+ queryArray.push(
|
|
|
+ makeQuery(targetObject,'config','FormStatus','formStatusData',[formFilter]));
|
|
|
//crfButtons
|
|
|
let statusFilter=LABKEY.Filter.create(varLabel,formStatus);
|
|
|
- queryArray.push(makeQuery('config','crfButtons','crfButtons',[statusFilter]));
|
|
|
+ queryArray.push(
|
|
|
+ makeQuery(targetObject,'config','crfButtons','crfButtons',[statusFilter]));
|
|
|
//Forms
|
|
|
- queryArray.push(makeQuery('config','Forms','formData',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'config','Forms','formData',[]));
|
|
|
//FormSetup
|
|
|
- queryArray.push(makeQuery('config','FormSetup','formSetup',[]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'config','FormSetup','formSetup',[]));
|
|
|
//generateConfig
|
|
|
- queryArray.push(makeQuery('config','generateConfig','generateConfigData',[]));
|
|
|
+ queryArray.push(
|
|
|
+ makeQuery(targetObject,'config','generateConfig','generateConfigData',[]));
|
|
|
//parentCrf
|
|
|
let parentCrf=config.formConfig.crfEntry['parentCrf'];
|
|
|
if (parentCrf!=undefined){
|
|
|
let crfFilter=LABKEY.Filter.create('entryId',parentCrf);
|
|
|
- queryArray.push(makeQuery('data','crfEntry','parentCrfData',[crfFilter]));
|
|
|
+ queryArray.push(makeQuery(targetObject,'data','crfEntry','parentCrfData',[crfFilter]));
|
|
|
}
|
|
|
|
|
|
print('running getDataFromQueries');
|
|
|
+ getDataFromQueries(queryArray,addStudyData);
|
|
|
+}
|
|
|
+
|
|
|
+function addStudyData(){
|
|
|
+ let queryArray=new Array();
|
|
|
+
|
|
|
+ let targetObject=config.formConfig;
|
|
|
+ targetObject.print=print;
|
|
|
+ targetObject.getContainer=getContainer;
|
|
|
+ //study
|
|
|
+ queryArray.push(makeQuery(targetObject,'data','Study','studyDataAll',[]));
|
|
|
+ //queryArray.push(makeQuery('data','Study','studyDataAll',[]));
|
|
|
+ let e=queryArray[queryArray.length-1];
|
|
|
+ //overload schema name
|
|
|
+ e.schemaName='study';
|
|
|
+ //make sure variables not part of default view are loaded
|
|
|
+ //here we should already have read crfStaticVariables table
|
|
|
+ let staticVarRows=config.formConfig['crfStaticVariables'].rows;
|
|
|
+ let columnModel=""
|
|
|
+ for (let i=0;i<staticVarRows.length;i++){
|
|
|
+ if (i>0) columnModel+=',';
|
|
|
+ columnModel+=staticVarRows[i]['staticVariable'];
|
|
|
+ }
|
|
|
+ e.columns=columnModel;
|
|
|
+
|
|
|
+ //also collect ids already in study
|
|
|
+ let demoQuery=config.formConfig.settings['demographicQuery'];
|
|
|
+
|
|
|
+ queryArray.push(makeQuery(targetObject,'data',demoQuery,'demographicData',[]));
|
|
|
+ queryArray[queryArray.length-1].schemaName='study';
|
|
|
+
|
|
|
+
|
|
|
getDataFromQueries(queryArray,fcontinue);
|
|
|
+
|
|
|
}
|
|
|
|
|
|
function selectFormSetupRows(formId){
|
|
@@ -3062,6 +3121,16 @@ function selectFormSetupRows(formId){
|
|
|
}
|
|
|
|
|
|
function fcontinue(){
|
|
|
+
|
|
|
+ //debug
|
|
|
+ let fName='[fcontinue]';
|
|
|
+ let varRows=config.formConfig['crfStaticVariables'].rows;
|
|
|
+ let studyVars=config.formConfig['studyDataAll'].rows[0];
|
|
|
+ for (let i=0;i<varRows.length;i++){
|
|
|
+ let vName=varRows[i].staticVariable;
|
|
|
+ print(fName+' '+vName+': '+studyVars[vName]);
|
|
|
+ }
|
|
|
+
|
|
|
//parse site
|
|
|
config.formConfig.siteRows=config.formConfig.siteData.rows;
|
|
|
let sRows=config.formConfig.siteRows;
|
|
@@ -3363,7 +3432,7 @@ function dataLayoutSet(){
|
|
|
|
|
|
function setData(cb){
|
|
|
fName='[setData]';
|
|
|
- print(fName+': cb '+cb);
|
|
|
+ //print(fName+': cb '+cb);
|
|
|
let crfMatch=getCRFref();
|
|
|
let parentCrf=config.formConfig.crfEntry['parentCrf'];
|
|
|
if (parentCrf!=undefined) crfMatch=parentCrf;
|