| 1 | PXRMUTIL ; SLC/PKR/PJH - Utility routines for use by PXRM. ;10/02/2007 | 
|---|
| 2 | ;;2.0;CLINICAL REMINDERS;**4,6**;Feb 04, 2005;Build 123 | 
|---|
| 3 | ; | 
|---|
| 4 | ;================================= | 
|---|
| 5 | ATTVALUE(STRING,ATTR,SEP,AVSEP) ;STRING contains a list of attribute value | 
|---|
| 6 | ;pairs. Each pair is separated by SEP and the attribute value pair | 
|---|
| 7 | ;is separated by AVSEP. Return the value for the attribute ATTR. | 
|---|
| 8 | N AVPAIR,IND,NUMAVP,VALUE | 
|---|
| 9 | S NUMAVP=$L(STRING,SEP) | 
|---|
| 10 | S VALUE="" | 
|---|
| 11 | F IND=1:1:NUMAVP Q:VALUE'=""  D | 
|---|
| 12 | . S AVPAIR=$P(STRING,SEP,IND) | 
|---|
| 13 | . I AVPAIR[ATTR S VALUE=$P(AVPAIR,AVSEP,2) | 
|---|
| 14 | Q VALUE | 
|---|
| 15 | ; | 
|---|
| 16 | ;================================= | 
|---|
| 17 | ACOPY(REF,OUTPUT) ;Copy all the descendants of the array reference into a linear | 
|---|
| 18 | ;array. REF is the starting array reference, for example A or | 
|---|
| 19 | ;^TMP("PXRM",$J). OUTPUT is the linear array for the output. It | 
|---|
| 20 | ;should be in the form of a closed root, i.e., A() or ^TMP($J,). | 
|---|
| 21 | ;Note OUTPUT cannot be used as the name of the output array. | 
|---|
| 22 | N DONE,IND,LEN,NL,OROOT,OUT,PROOT,ROOT,START,TEMP | 
|---|
| 23 | I REF="" Q | 
|---|
| 24 | S NL=0 | 
|---|
| 25 | S OROOT=$P(OUTPUT,")",1) | 
|---|
| 26 | S PROOT=$P(REF,")",1) | 
|---|
| 27 | ;Build the root so we can tell when we are done. | 
|---|
| 28 | S TEMP=$NA(@REF) | 
|---|
| 29 | S ROOT=$P(TEMP,")",1) | 
|---|
| 30 | S REF=$Q(@REF) | 
|---|
| 31 | I REF'[ROOT Q | 
|---|
| 32 | S DONE=0 | 
|---|
| 33 | F  Q:(REF="")!(DONE)  D | 
|---|
| 34 | . S START=$F(REF,ROOT) | 
|---|
| 35 | . S LEN=$L(REF) | 
|---|
| 36 | . S IND=$E(REF,START,LEN) | 
|---|
| 37 | . S NL=NL+1 | 
|---|
| 38 | . S OUT=OROOT_NL_")" | 
|---|
| 39 | . S @OUT=PROOT_IND_"="_@REF | 
|---|
| 40 | . S REF=$Q(@REF) | 
|---|
| 41 | . I REF'[ROOT S DONE=1 | 
|---|
| 42 | Q | 
|---|
| 43 | ; | 
|---|
| 44 | ;================================= | 
|---|
| 45 | AWRITE(REF) ;Write all the descendants of the array reference. | 
|---|
| 46 | ;REF is the starting array reference, for example A or ^TMP("PXRM",$J). | 
|---|
| 47 | N DONE,IND,LEN,PROOT,ROOT,START,TEMP | 
|---|
| 48 | I REF="" Q | 
|---|
| 49 | S PROOT=$P(REF,")",1) | 
|---|
| 50 | ;Build the root so we can tell when we are done. | 
|---|
| 51 | S TEMP=$NA(@REF) | 
|---|
| 52 | S ROOT=$P(TEMP,")",1) | 
|---|
| 53 | S REF=$Q(@REF) | 
|---|
| 54 | I REF'[ROOT Q | 
|---|
| 55 | S DONE=0 | 
|---|
| 56 | F  Q:(REF="")!(DONE)  D | 
|---|
| 57 | . S START=$F(REF,ROOT) | 
|---|
| 58 | . S LEN=$L(REF) | 
|---|
| 59 | . S IND=$E(REF,START,LEN) | 
|---|
| 60 | . W !,PROOT_IND,"=",@REF | 
|---|
| 61 | . S REF=$Q(@REF) | 
|---|
| 62 | . I REF'[ROOT S DONE=1 | 
|---|
| 63 | Q | 
|---|
| 64 | ; | 
|---|
| 65 | ;================================= | 
|---|
| 66 | DIP(VAR,IEN,PXRMROOT,FLDS) ;Do general inquiry for IEN return formatted | 
|---|
| 67 | ;output in VAR. VAR can be either a local variable or a global. | 
|---|
| 68 | ;If it is a local it is indexed for the broker. If it is a global | 
|---|
| 69 | ;it should be passed in closed form i.e., ^TMP("PXRMTEST",$J). | 
|---|
| 70 | ;It will be returned formatted for ListMan i.e., | 
|---|
| 71 | ;^TMP("PXRMTEST",$J,N,0). | 
|---|
| 72 | N %ZIS,ARRAY,BY,DC,DHD,DIC,DONE,FF,FILENAME,FILESPEC,FR,GBL,HFNAME | 
|---|
| 73 | N IND,IOP,L,NOW,PATH,SUCCESS,TO,UNIQN | 
|---|
| 74 | S BY="NUMBER",(FR,TO)=+$P(IEN,U,1),DHD="@@" | 
|---|
| 75 | ;Make sure the PXRM WORKSTATION device exists. | 
|---|
| 76 | D MKWSDEV^PXRMHOST | 
|---|
| 77 | ;Set up the output file before DIP is called. | 
|---|
| 78 | S PATH=$$PWD^%ZISH | 
|---|
| 79 | S NOW=$$NOW^XLFDT | 
|---|
| 80 | S NOW=$TR(NOW,".","") | 
|---|
| 81 | S UNIQN=$J_NOW | 
|---|
| 82 | S FILENAME="PXRMWSD"_UNIQN_".DAT" | 
|---|
| 83 | S HFNAME=PATH_FILENAME | 
|---|
| 84 | S IOP="PXRM WORKSTATION;80" | 
|---|
| 85 | S %ZIS("HFSMODE")="W" | 
|---|
| 86 | S %ZIS("HFSNAME")=HFNAME | 
|---|
| 87 | S L=0,DIC=PXRMROOT | 
|---|
| 88 | D EN1^DIP | 
|---|
| 89 | ;Move the host file into a global. | 
|---|
| 90 | S GBL="^TMP(""PXRMUTIL"",$J,1,0)" | 
|---|
| 91 | S GBL=$NA(@GBL) | 
|---|
| 92 | K ^TMP("PXRMUTIL",$J) | 
|---|
| 93 | S SUCCESS=$$FTG^%ZISH(PATH,FILENAME,GBL,3) | 
|---|
| 94 | ;Look for a form feed, remove it and all subsequent lines. | 
|---|
| 95 | S FF=$C(12) | 
|---|
| 96 | I $G(VAR)["^" D | 
|---|
| 97 | . S VAR=$NA(@VAR) | 
|---|
| 98 | . S VAR=$P(VAR,")",1) | 
|---|
| 99 | . S VAR=VAR_",IND,0)" | 
|---|
| 100 | . S (DONE,IND)=0 | 
|---|
| 101 | . F  Q:DONE  S IND=$O(^TMP("PXRMUTIL",$J,IND)) Q:+IND=0  D | 
|---|
| 102 | .. I ^TMP("PXRMUTIL",$J,IND,0)=FF S DONE=1 Q | 
|---|
| 103 | .. S @VAR=^TMP("PXRMUTIL",$J,IND,0) | 
|---|
| 104 | E  D | 
|---|
| 105 | . S (DONE,IND)=0 | 
|---|
| 106 | . F  Q:DONE  S IND=$O(^TMP("PXRMUTIL",$J,IND)) Q:+IND=0  D | 
|---|
| 107 | .. S VAR(IND)=^TMP("PXRMUTIL",$J,IND,0) | 
|---|
| 108 | .. I VAR(IND)=FF K ARRAY(IND) S DONE=1 | 
|---|
| 109 | K ^TMP("PXRMUTIL",$J) | 
|---|
| 110 | ;Delete the host file. | 
|---|
| 111 | S FILESPEC(FILENAME)="" | 
|---|
| 112 | S SUCCESS=$$DEL^%ZISH(PATH,$NA(FILESPEC)) | 
|---|
| 113 | Q | 
|---|
| 114 | ; | 
|---|
| 115 | ;================================= | 
|---|
| 116 | FNFR(ROOT) ;Given the root of a file return the file number. | 
|---|
| 117 | Q +$P(@(ROOT_"0)"),U,2) | 
|---|
| 118 | ; | 
|---|
| 119 | ;================================= | 
|---|
| 120 | NTOAN(NUMBER) ;Given an integer N return an alphabetic string that can be | 
|---|
| 121 | ;used for sorting. This will be modulus 26. For example N=0 returns | 
|---|
| 122 | ;A, N=26 returns BA etc. | 
|---|
| 123 | N ALPH | 
|---|
| 124 | S ALPH(0)="A",ALPH(1)="B",ALPH(2)="C",ALPH(3)="D",ALPH(4)="E" | 
|---|
| 125 | S ALPH(5)="F",ALPH(6)="G",ALPH(7)="H",ALPH(8)="I",ALPH(9)="J" | 
|---|
| 126 | S ALPH(10)="K",ALPH(11)="L",ALPH(12)="M",ALPH(13)="N",ALPH(14)="O" | 
|---|
| 127 | S ALPH(15)="P",ALPH(16)="Q",ALPH(17)="R",ALPH(18)="S",ALPH(19)="T" | 
|---|
| 128 | S ALPH(20)="U",ALPH(21)="V",ALPH(22)="W",ALPH(23)="X",ALPH(24)="Y" | 
|---|
| 129 | S ALPH(25)="Z" | 
|---|
| 130 | ; | 
|---|
| 131 | N ANUM,DIGIT,NUM,P26,PC,PWR | 
|---|
| 132 | S ANUM="",NUM=NUMBER,PWR=0 | 
|---|
| 133 | S P26(PWR)=1 | 
|---|
| 134 | F PWR=1:1 S P26(PWR)=26*P26(PWR-1) I P26(PWR)>NUMBER Q | 
|---|
| 135 | S PWR=PWR-1 | 
|---|
| 136 | F PC=PWR:-1:0 D | 
|---|
| 137 | . S DIGIT=NUM\P26(PC) | 
|---|
| 138 | . S ANUM=ANUM_ALPH(DIGIT) | 
|---|
| 139 | . S NUM=NUM-(DIGIT*P26(PC)) | 
|---|
| 140 | Q ANUM | 
|---|
| 141 | ; | 
|---|
| 142 | ;================================= | 
|---|
| 143 | RMEHIST(FILENUM,IEN) ;Remove the edit history for a reminder file. | 
|---|
| 144 | I (FILENUM<800)!(FILENUM>811.9)!(FILENUM=811.8) Q | 
|---|
| 145 | N DA,DIK,GLOBAL,ROOT | 
|---|
| 146 | S GLOBAL=$$GET1^DID(FILENUM,"","","GLOBAL NAME") | 
|---|
| 147 | ;Edit History is stored in node 110 for all files. | 
|---|
| 148 | S DA(1)=IEN | 
|---|
| 149 | S DIK=GLOBAL_IEN_",110," | 
|---|
| 150 | S ROOT=GLOBAL_IEN_",110,DA)" | 
|---|
| 151 | S DA=0 | 
|---|
| 152 | F  S DA=+$O(@ROOT) Q:DA=0  D ^DIK | 
|---|
| 153 | Q | 
|---|
| 154 | ; | 
|---|
| 155 | ;================================= | 
|---|
| 156 | SEHIST(FILENUM,ROOT,IEN) ;Set the edit date and edit by and prompt the | 
|---|
| 157 | ;user for the edit comment. | 
|---|
| 158 | N DIC,DIR,DWLW,DWPK,ENTRY,FDA,FDAIEN,IENS,IND,MSG,SFN,TARGET,X,Y | 
|---|
| 159 | K ^TMP("PXRMWP",$J) | 
|---|
| 160 | D FIELD^DID(FILENUM,"EDIT HISTORY","","SPECIFIER","TARGET") | 
|---|
| 161 | S SFN=+$G(TARGET("SPECIFIER")) | 
|---|
| 162 | I SFN=0 Q | 
|---|
| 163 | S ENTRY=ROOT_IEN_",110)" | 
|---|
| 164 | S IND=$O(@ENTRY@("B"),-1) | 
|---|
| 165 | S IND=IND+1 | 
|---|
| 166 | S IENS="+"_IND_","_IEN_"," | 
|---|
| 167 | S FDAIEN(IEN)=IEN | 
|---|
| 168 | S FDA(SFN,IENS,.01)=$$FMTE^XLFDT($$NOW^XLFDT,"5Z") | 
|---|
| 169 | S FDA(SFN,IENS,1)=$$GET1^DIQ(200,DUZ,.01) | 
|---|
| 170 | ;Prompt the user for edit comments. | 
|---|
| 171 | S DIC="^TMP(""PXRMWP"",$J," | 
|---|
| 172 | S DWLW=72 | 
|---|
| 173 | S DWPK=1 | 
|---|
| 174 | W !,"Input your edit comments." | 
|---|
| 175 | S DIR(0)="Y"_U_"AO" | 
|---|
| 176 | S DIR("A")="Edit" | 
|---|
| 177 | S DIR("B")="NO" | 
|---|
| 178 | D ^DIR | 
|---|
| 179 | I Y D | 
|---|
| 180 | . D EN^DIWE | 
|---|
| 181 | . K ^TMP("PXRMWP",$J,0) | 
|---|
| 182 | . I $D(^TMP("PXRMWP",$J)) S FDA(SFN,IENS,2)="^TMP(""PXRMWP"",$J)" | 
|---|
| 183 | D UPDATE^DIE("E","FDA","FDAIEN","MSG") | 
|---|
| 184 | I $D(MSG) D AWRITE^PXRMUTIL("MSG") | 
|---|
| 185 | K ^TMP("PXRMWP",$J) | 
|---|
| 186 | Q | 
|---|
| 187 | ; | 
|---|
| 188 | ;================================= | 
|---|
| 189 | SFRES(SDIR,NRES,FIEVAL) ;Save the finding result. | 
|---|
| 190 | I NRES=0 S FIEVAL=0 Q | 
|---|
| 191 | N DATE,IND,OA,SUB,TF | 
|---|
| 192 | F IND=1:1:NRES S OA(FIEVAL(IND,"DATE"),FIEVAL(IND),IND)="" | 
|---|
| 193 | ;If SDIR is positive get the oldest date otherwise get the most | 
|---|
| 194 | ;recent date. | 
|---|
| 195 | S DATE=$S(SDIR>0:$O(OA("")),1:$O(OA(""),-1)) | 
|---|
| 196 | ;If there is a true finding on DATE get it. | 
|---|
| 197 | S TF=$O(OA(DATE,""),-1) | 
|---|
| 198 | S IND=$O(OA(DATE,TF,"")) | 
|---|
| 199 | S FIEVAL=TF | 
|---|
| 200 | S SUB="" | 
|---|
| 201 | F  S SUB=$O(FIEVAL(IND,SUB)) Q:SUB=""  M FIEVAL(SUB)=FIEVAL(IND,SUB) | 
|---|
| 202 | Q | 
|---|
| 203 | ; | 
|---|
| 204 | ;================================= | 
|---|
| 205 | SSPAR(FIND0,NOCC,BDT,EDT) ;Set the finding search parameters. | 
|---|
| 206 | S BDT=$P(FIND0,U,8),EDT=$P(FIND0,U,11),NOCC=$P(FIND0,U,14) | 
|---|
| 207 | I +NOCC=0 S NOCC=1 | 
|---|
| 208 | ;Convert the dates to FileMan dates. | 
|---|
| 209 | S BDT=$S(BDT="":0,BDT=0:0,1:$$CTFMD^PXRMDATE(BDT)) | 
|---|
| 210 | I EDT="" S EDT="T" | 
|---|
| 211 | S EDT=$$CTFMD^PXRMDATE(EDT) | 
|---|
| 212 | ;If EDT does not contain a time set it to the end of the day. | 
|---|
| 213 | I EDT'["." S EDT=EDT_".235959" | 
|---|
| 214 | I $G(PXRMDDOC)'=1 Q | 
|---|
| 215 | S ^TMP("PXRMDDOC",$J,$P(FIND0,U,1,11))=BDT_U_EDT | 
|---|
| 216 | Q | 
|---|
| 217 | ; | 
|---|
| 218 | ;================================= | 
|---|
| 219 | STRREP(STRING,TS,RS) ;Replace every occurrence of the target string (TS) | 
|---|
| 220 | ;in STRING with the replacement string (RS). | 
|---|
| 221 | ;Example 9.19 (page 220) in "The Complete Mumps" by John Lewkowicz: | 
|---|
| 222 | ;  F  Q:STRING'[TS  S STRING=$P(STRING,TS)_RS_$P(STRING,TS,2,999) | 
|---|
| 223 | ;fails if any portion of the target string is contained in the with | 
|---|
| 224 | ;string. Therefore a more elaborate version is required. | 
|---|
| 225 | ; | 
|---|
| 226 | N IND,NPCS,STR | 
|---|
| 227 | I STRING'[TS Q STRING | 
|---|
| 228 | ;Count the number of pieces using the target string as the delimiter. | 
|---|
| 229 | S NPCS=$L(STRING,TS) | 
|---|
| 230 | ;Extract the pieces and concatenate RS | 
|---|
| 231 | S STR="" | 
|---|
| 232 | F IND=1:1:NPCS-1 S STR=STR_$P(STRING,TS,IND)_RS | 
|---|
| 233 | S STR=STR_$P(STRING,TS,NPCS) | 
|---|
| 234 | Q STR | 
|---|
| 235 | ; | 
|---|
| 236 | ;================================= | 
|---|
| 237 | VEDIT(ROOT,IEN) ;This is used as a DIC("S") screen to select which entries | 
|---|
| 238 | ;a user can edit. | 
|---|
| 239 | N CLASS,ENTRY,VALID | 
|---|
| 240 | S ENTRY=ROOT_IEN_")" | 
|---|
| 241 | S CLASS=$P($G(@ENTRY@(100)),U,1) | 
|---|
| 242 | I CLASS="N" D | 
|---|
| 243 | . I ($G(PXRMINST)=1),(DUZ(0)="@") S VALID=1 | 
|---|
| 244 | . E  S VALID=0 | 
|---|
| 245 | E  S VALID=1 | 
|---|
| 246 | Q VALID | 
|---|
| 247 | ; | 
|---|