[623] | 1 | PXRMDATE ; SLC/PKR - Clinical Reminders date utilities. ;06/20/2006
|
---|
| 2 | ;;2.0;CLINICAL REMINDERS;**4**;Feb 04, 2005;Build 21
|
---|
| 3 | ;
|
---|
| 4 | ;==================================================
|
---|
| 5 | CEFD(FDA) ;Called by the Exchange Utility only if the input packed
|
---|
| 6 | ;reminder was packed under v1.5 Move Effective Date to Beginning Date.
|
---|
| 7 | N IND
|
---|
| 8 | S IND=""
|
---|
| 9 | F S IND=$O(FDA(811.902,IND)) Q:IND="" D
|
---|
| 10 | . I '$D(FDA(811.902,IND,12)) Q
|
---|
| 11 | .;If the EFFECTIVE PERIOD exists don't do anything.
|
---|
| 12 | . I $D(FDA(811.902,IND,9)) Q
|
---|
| 13 | . S FDA(811.902,IND,9)=FDA(811.902,IND,12)
|
---|
| 14 | . K FDA(811.902,IND,12)
|
---|
| 15 | Q
|
---|
| 16 | ;
|
---|
| 17 | ;==================================================
|
---|
| 18 | COMPARE(X) ;Compare beginning and ending dates, give a warning if
|
---|
| 19 | ;Ending Date comes before Beginning Date. Called by ADATE xref in
|
---|
| 20 | ;definitions and terms.
|
---|
| 21 | ;Do not execute as part of exchange.
|
---|
| 22 | I $G(PXRMEXCH) Q
|
---|
| 23 | N BDT,EDT
|
---|
| 24 | S BDT=$S(X(1)'="":$$CTFMD^PXRMDATE(X(1)),1:0)
|
---|
| 25 | S EDT=X(2)
|
---|
| 26 | I EDT="" S EDT="T"
|
---|
| 27 | S EDT=$$CTFMD^PXRMDATE(EDT)
|
---|
| 28 | ;If EDT does not contain a time set it to the end of the day.
|
---|
| 29 | I EDT'["." S EDT=EDT_".235959"
|
---|
| 30 | I EDT<BDT D
|
---|
| 31 | . S BDT=$S(X(1)'="":X(1),1:"")
|
---|
| 32 | . S EDT=$S(X(2)'="":X(2),1:"T@2400")
|
---|
| 33 | . S TEXT="Warning the ending date ("_EDT_") is before the beginning date ("_BDT_")"
|
---|
| 34 | . D EN^DDIOL(TEXT)
|
---|
| 35 | Q
|
---|
| 36 | ;
|
---|
| 37 | ;==================================================
|
---|
| 38 | COTN(EFP) ;Convert an Effective Period to the new date/time format.
|
---|
| 39 | ;Possible effective periods are ND, NM, or NY where N is an integer.
|
---|
| 40 | S EFP=$$UP^XLFSTR(EFP)
|
---|
| 41 | I (EFP?1N.N1"D")!(EFP?1N.N1"M")!(EFP?1N.N1"Y") D
|
---|
| 42 | . S NUM=+EFP
|
---|
| 43 | . S EFP=$S(NUM=0:"T",1:"T-"_EFP)
|
---|
| 44 | Q EFP
|
---|
| 45 | ;
|
---|
| 46 | ;==================================================
|
---|
| 47 | CTFMD(DATE) ;Convert DATE which may be in any of the FileMan acceptable
|
---|
| 48 | ;forms as well as T-NY to a FileMan date. Also understands LAD for
|
---|
| 49 | ;Last Admission Date.
|
---|
| 50 | N %DT,X,Y
|
---|
| 51 | ;Check for a date FileMan understands.
|
---|
| 52 | S X=DATE,%DT="ST"
|
---|
| 53 | D ^%DT
|
---|
| 54 | ;If it is not a FileMan date check for a symbolic date.
|
---|
| 55 | I Y=-1 S Y=$$SYMDATE(DATE)
|
---|
| 56 | ;If it is not a date that is understood by SYMDATE return -1
|
---|
| 57 | I Y=-1 Q -1
|
---|
| 58 | I $G(PXRMDATE)'="",$$ISVSYMD(DATE) D
|
---|
| 59 | . N DIFFS
|
---|
| 60 | . S DIFFS=-$$FMDIFF^XLFDT(DT,PXRMDATE,2)
|
---|
| 61 | . S Y=$$FMADD^XLFDT(Y,0,0,0,DIFFS)
|
---|
| 62 | I DATE["LAD" D
|
---|
| 63 | . I $G(PXRMLAD)="" S Y=0
|
---|
| 64 | . E D
|
---|
| 65 | .. N DIFFS
|
---|
| 66 | .. S DIFFS=-$$FMDIFF^XLFDT(DT,$G(PXRMLAD),2)
|
---|
| 67 | .. S Y=$$FMADD^XLFDT(Y,0,0,0,DIFFS)
|
---|
| 68 | Q Y
|
---|
| 69 | ;
|
---|
| 70 | ;=================================================
|
---|
| 71 | DCHECK(DATE) ;Trap for special characters before calling CTFMD^PXRMDATE.
|
---|
| 72 | ;Used in DIR("PRE") for date inputs.
|
---|
| 73 | I $D(DTOUT) Q DATE
|
---|
| 74 | I DATE="" Q DATE
|
---|
| 75 | I DATE["^" Q DATE
|
---|
| 76 | I DATE["?" Q DATE
|
---|
| 77 | Q $$CTFMD^PXRMDATE(DATE)
|
---|
| 78 | ;
|
---|
| 79 | ;==================================================
|
---|
| 80 | DUE(DEFARR,RESDATE,FREQ,DUE,DUEDATE,FIEVAL) ;Compute the due date.
|
---|
| 81 | ;This is the date of the resolution finding + the reminder frequency.
|
---|
| 82 | ;Subtract the due in advance time to see if the reminder should be
|
---|
| 83 | ;marked as due soon.
|
---|
| 84 | ;
|
---|
| 85 | N DATE,DIAT,DIATOK,LDATE,PXRMITEM,TDDUE,TODAY
|
---|
| 86 | S PXRMITEM=DEFARR("IEN")
|
---|
| 87 | ;If the final frequency is 0Y then the reminder is not due.
|
---|
| 88 | I FREQ="0Y" S DUE=0,DUEDATE="" Q
|
---|
| 89 | ;
|
---|
| 90 | S DUEDATE=""
|
---|
| 91 | ;Check for custom date due.
|
---|
| 92 | I DEFARR(45)'="" S DUEDATE=$$CDUEDATE^PXRMCDUE(.DEFARR,.FIEVAL)
|
---|
| 93 | I DUEDATE'="",DUEDATE'=-1 G SETDUE
|
---|
| 94 | ;
|
---|
| 95 | ;No custom date due, do regular date calculation.
|
---|
| 96 | I (FREQ="")!(FREQ=-1) D Q
|
---|
| 97 | . S ^TMP(PXRMPID,$J,PXRMITEM,"WARNING","NOFREQ")="No reminder frequency - cannot compute due date!"
|
---|
| 98 | . S (DUE,DUEDATE)="CNBD"
|
---|
| 99 | ;
|
---|
| 100 | S LDATE=$S(RESDATE["X":0,1:+RESDATE)
|
---|
| 101 | I LDATE=0 S (DUE,DUEDATE)="DUE NOW" Q
|
---|
| 102 | S DATE=$$FULLDATE(LDATE),DUEDATE=$$NEWDATE(DATE,FREQ)
|
---|
| 103 | ;
|
---|
| 104 | SETDUE ;If the due date is less than or equal to today's date the reminder
|
---|
| 105 | ;is due.
|
---|
| 106 | S TODAY=$$NOW^PXRMDATE
|
---|
| 107 | I +DUEDATE'>TODAY S DUE="DUE NOW" Q
|
---|
| 108 | ;
|
---|
| 109 | S DIAT="-"_$P(DEFARR(0),U,4)
|
---|
| 110 | I DIAT="-" D
|
---|
| 111 | . S DIATOK=0
|
---|
| 112 | . S ^TMP(PXRMPID,$J,PXRMITEM,"WARNING","DIAT")="Warning no do in advance time"
|
---|
| 113 | E S DIATOK=1
|
---|
| 114 | ;
|
---|
| 115 | S TDDUE=$S(DIATOK=1:$$NEWDATE(DUEDATE,DIAT),1:DUEDATE)
|
---|
| 116 | S DUE=$S(TDDUE'>TODAY:"DUE SOON",1:"RESOLVED")
|
---|
| 117 | Q
|
---|
| 118 | ;
|
---|
| 119 | ;==================================================
|
---|
| 120 | DURATION(START,STOP) ;Return the number days between the Start Date and
|
---|
| 121 | ;Stop Date.
|
---|
| 122 | I +START=0 Q 0
|
---|
| 123 | N PXRMNOW
|
---|
| 124 | S PXRMNOW=$$NOW^PXRMDATE
|
---|
| 125 | I START>PXRMNOW Q 0
|
---|
| 126 | I (STOP="")!(STOP>PXRMNOW) S STOP=PXRMNOW
|
---|
| 127 | Q $$FMDIFF^XLFDT(STOP,START)
|
---|
| 128 | ;
|
---|
| 129 | ;==================================================
|
---|
| 130 | EDATE(DATE) ;Check for an historical (event) date, format as appropriate.
|
---|
| 131 | Q $$FMTE^XLFDT(DATE,"5DZ")
|
---|
| 132 | ;
|
---|
| 133 | ;==================================================
|
---|
| 134 | FULLDATE(DATE) ;See if DATE is a full date, i.e., it has a month and
|
---|
| 135 | ;a day along with a year. If the month is missing assume Jan. If the
|
---|
| 136 | ;day is missing assume the first. Issue a warning so the user knows
|
---|
| 137 | ;what happened. DATE should be in Fileman format.
|
---|
| 138 | N DAY,MISSING,MONTH,TDATE,YEAR
|
---|
| 139 | S TDATE=DATE
|
---|
| 140 | S MISSING=0
|
---|
| 141 | S DAY=$E(DATE,6,7)
|
---|
| 142 | S MONTH=$E(DATE,4,5)
|
---|
| 143 | S YEAR=$E(DATE,1,3)
|
---|
| 144 | I +DAY=0 D
|
---|
| 145 | . S DAY=1
|
---|
| 146 | . S MISSING=1
|
---|
| 147 | . S ^TMP(PXRMPID,$J,PXRMITEM,"INFO","NO DAY")="Encounter date missing the day, using the first for the date due calculation."
|
---|
| 148 | I +MONTH=0 D
|
---|
| 149 | . S MONTH=1
|
---|
| 150 | . S MISSING=1
|
---|
| 151 | . S ^TMP(PXRMPID,$J,PXRMITEM,"INFO","NO MONTH")="Encounter date missing the month, using January for the date due calculation."
|
---|
| 152 | I MISSING D
|
---|
| 153 | . S TDATE=(YEAR*1E4)+(MONTH*1E2)+DAY
|
---|
| 154 | . I DATE["E" S TDATE=TDATE_"E"
|
---|
| 155 | Q TDATE
|
---|
| 156 | ;
|
---|
| 157 | ;==================================================
|
---|
| 158 | FRQINDAY(FREQ) ;Given a frequency in the form ND, NM, or NY where N is a
|
---|
| 159 | ;number and D stands for days, M for months, and Y for years return
|
---|
| 160 | ;the value in days.
|
---|
| 161 | I FREQ="" Q ""
|
---|
| 162 | N CODE,LEN,MULT,NUM
|
---|
| 163 | S LEN=$L(FREQ)
|
---|
| 164 | S NUM=$E(FREQ,1,LEN-1)
|
---|
| 165 | S CODE=$E(FREQ,LEN,LEN)
|
---|
| 166 | S MULT=1.0
|
---|
| 167 | I CODE="M" S MULT=30.42
|
---|
| 168 | I CODE="Y" S MULT=365.25
|
---|
| 169 | Q +(MULT*NUM)
|
---|
| 170 | ;
|
---|
| 171 | ;==================================================
|
---|
| 172 | ISVSYMD(DATE) ;Return true if DATE is a valid symbolic date.
|
---|
| 173 | N P1,P1OK,P2,P2OK,OP,PAT
|
---|
| 174 | S DATE=$P(DATE,"@",1)
|
---|
| 175 | S OP=$S(DATE["+":"+",1:"-")
|
---|
| 176 | S P1=$P(DATE,OP,1),P1OK=0
|
---|
| 177 | F PAT="T","TODAY","N","NOW" I P1=PAT S P1OK=1 Q:P1OK
|
---|
| 178 | I PAT=DATE Q 1
|
---|
| 179 | S P2=$P(DATE,OP,2),P2OK=0
|
---|
| 180 | F PAT="1N.N","1N.N1""D""","1N.N1""M""","1N.N1""Y""" I P2?@PAT S P2OK=1 Q:P2OK
|
---|
| 181 | Q P1OK&P2OK
|
---|
| 182 | ;
|
---|
| 183 | ;==================================================
|
---|
| 184 | NEWDATE(FMDATE,OFFSET) ;Given a date in VA Fileman format (FMDATE) and an
|
---|
| 185 | ;offset of the form NY, NM, ND where N is a number and Y stands for
|
---|
| 186 | ;years, M for months, and D for days return the new date in VA Fileman
|
---|
| 187 | ;format.
|
---|
| 188 | I FMDATE=0 Q 0
|
---|
| 189 | N LEN,NEWDATE,NUM,UNIT
|
---|
| 190 | S LEN=$L(OFFSET)
|
---|
| 191 | S NUM=+$E(OFFSET,1,LEN-1)
|
---|
| 192 | S UNIT=$E(OFFSET,LEN)
|
---|
| 193 | I UNIT="D" G DAY
|
---|
| 194 | I UNIT="M" G MONTH
|
---|
| 195 | I UNIT="Y" G YEAR
|
---|
| 196 | ;Unknown unit just return the original date
|
---|
| 197 | Q FMDATE
|
---|
| 198 | DAY ;
|
---|
| 199 | S NEWDATE=+$$FMADD^XLFDT(FMDATE,NUM)
|
---|
| 200 | Q NEWDATE
|
---|
| 201 | MONTH ;
|
---|
| 202 | ;Convert the months to days and then add the days using the DAY code.
|
---|
| 203 | ;Multiply the number of months by the average number of days in a month.
|
---|
| 204 | N INT,FRAC
|
---|
| 205 | S NUM=30.42*NUM
|
---|
| 206 | ;Round the number of days, FMADD^XLFDT has problems with non-integer
|
---|
| 207 | ;days.
|
---|
| 208 | S INT=+$P(NUM,".",1)
|
---|
| 209 | S FRAC=NUM-INT
|
---|
| 210 | I FRAC<0.5 S NUM=INT
|
---|
| 211 | E S NUM=INT+1
|
---|
| 212 | G DAY
|
---|
| 213 | Q
|
---|
| 214 | YEAR ;
|
---|
| 215 | Q FMDATE+(10000*NUM)
|
---|
| 216 | ;
|
---|
| 217 | ;==================================================
|
---|
| 218 | NOW() ;If the reminder global PXRMDATE is defined return it, otherwise
|
---|
| 219 | ;return the current date and time.
|
---|
| 220 | Q $S(+$G(PXRMDATE)>0:PXRMDATE,1:$$NOW^XLFDT)
|
---|
| 221 | ;
|
---|
| 222 | ;==================================================
|
---|
| 223 | SYMDATE(DATE) ;Convert a symbolic date into a FileMan date.
|
---|
| 224 | N %DT,OPER,PFSTACK,SYM,TIME,X,Y
|
---|
| 225 | S TIME=$P(DATE,"@",2),DATE=$P(DATE,"@",1)
|
---|
| 226 | S X=$S(DATE="LAD":$G(PXRMLAD),1:"")
|
---|
| 227 | I X="" D
|
---|
| 228 | . S OPER="+-"
|
---|
| 229 | . D POSTFIX^PXRMSTAC(DATE,OPER,.PFSTACK)
|
---|
| 230 | I PFSTACK(0)=3 D
|
---|
| 231 | . S SYM=PFSTACK(1)
|
---|
| 232 | . S SYM=$S(SYM="LAD":"T",SYM="N":"N",SYM="NOW":"N",SYM="T":"T",SYM="TODAY":"T",1:"")
|
---|
| 233 | . I SYM="" S Y=-1 Q
|
---|
| 234 | .;FileMan only handles D, W, or M so convert Y to months.
|
---|
| 235 | . I PFSTACK(2)["Y" S PFSTACK(2)=+PFSTACK(2)*12_"M"
|
---|
| 236 | . S X=SYM_PFSTACK(3)_PFSTACK(2)
|
---|
| 237 | I PFSTACK(0)=1 S X=PFSTACK(1)
|
---|
| 238 | I TIME'="" S X=X_"@"_TIME
|
---|
| 239 | S %DT="ST"
|
---|
| 240 | D ^%DT
|
---|
| 241 | Q Y
|
---|
| 242 | ;
|
---|
| 243 | ;==================================================
|
---|
| 244 | VDATE(VIEN) ;Given a visit ien return the visit date.
|
---|
| 245 | N DATE
|
---|
| 246 | I +VIEN>0 S DATE=$P($G(^AUPNVSIT(VIEN,0)),U,1)
|
---|
| 247 | E S DATE=0
|
---|
| 248 | I $L(DATE)=0 S DATE=0
|
---|
| 249 | ;Check for historical encounter.
|
---|
| 250 | I $$ISHIST^PXRMVSIT(VIEN) S DATE=DATE_"E"
|
---|
| 251 | Q DATE
|
---|
| 252 | ;
|
---|