Annotation of freem/mlib/%SYSPOLY.m, revision 1.1
1.1 ! snw 1: %SYSPOLY ;V4W/DLW-FREEM POLYFILLS; 9/2/2021 7:52 PM
! 2: ;0.0;FreeM;****FREEM**;John P Willis @2021
! 3: ;
! 4: ; *
! 5: ; * *
! 6: ; * *
! 7: ; ***************
! 8: ; * * * *
! 9: ; * MUMPS *
! 10: ; * * * *
! 11: ; ***************
! 12: ; * *
! 13: ; * *
! 14: ; *
! 15: ;
! 16: ; %ZUTILS.m
! 17: ; FreeM Polyfills
! 18: ;
! 19: ;
! 20: ; Author: David Wicksell <dlwicksell@fourthwatchsoftware.com>
! 21: ; Copyright (C) 1998 MUG Deutschland
! 22: ; Copyright (C) 2020, 2021, 2023 Fourth Watch Software LC
! 23: ;
! 24: ;
! 25: ; This file is part of FreeM.
! 26: ;
! 27: ; FreeM is free software: you can redistribute it and/or modify
! 28: ; it under the terms of the GNU Affero Public License as published by
! 29: ; the Free Software Foundation, either version 3 of the License, or
! 30: ; (at your option) any later version.
! 31: ;
! 32: ; FreeM is distributed in the hope that it will be useful,
! 33: ; but WITHOUT ANY WARRANTY; without even the implied warranty of
! 34: ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
! 35: ; GNU Affero Public License for more details.
! 36: ;
! 37: ; You should have received a copy of the GNU Affero Public License
! 38: ; along with FreeM. If not, see <https://www.gnu.org/licenses/>.
! 39: ;
! 40: MERGE(SOURCE,DESTINATION) ; Implement the MERGE command
! 41: ; SOURCE = M array to merge from passed by name indirection
! 42: ; DESTINATION = M array to merge to passed by name indirection
! 43: ; returns 0 on success, -1 on error
! 44: ;
! 45: ; SOURCE is an array descendant of DESTINATION; throw M19 (error code 158)
! 46: IF $EXTRACT(SOURCE,1,$LENGTH(DESTINATION))=DESTINATION QUIT -1
! 47: ; DESTINATION is an array descendant of SOURCE; throw M19 (error code 158)
! 48: IF $EXTRACT(DESTINATION,1,$LENGTH(SOURCE))=SOURCE QUIT -1
! 49: ;
! 50: ; save SOURCE and DESTINATION in order to reset the naked indicator at the end
! 51: NEW SAVESRC,SAVEDST
! 52: SET SAVESRC=SOURCE,SAVEDST=DESTINATION
! 53: ;
! 54: ; set DESTINATION clear base name
! 55: NEW DST
! 56: SET DST=DESTINATION
! 57: ;
! 58: ; set SOURCE break flag
! 59: NEW SRC
! 60: SET SRC=SOURCE,$QSUBSCRIPT(SRC,0)="",$EXTRACT(SRC,$LENGTH(SRC))=""
! 61: ;
! 62: ; set the SOURCE POP offset
! 63: NEW POP
! 64: IF $QLENGTH(SOURCE)>$QLENGTH(DESTINATION) SET POP=$QLENGTH(SOURCE)-$QLENGTH(DESTINATION)
! 65: ELSE SET POP=0
! 66: ;
! 67: ; set the DESTINATION PUSH offset
! 68: NEW PUSH
! 69: IF $QLENGTH(DESTINATION)>$QLENGTH(SOURCE) SET PUSH=$QLENGTH(DESTINATION)-$QLENGTH(SOURCE)
! 70: ELSE SET PUSH=0
! 71: ;
! 72: ; merge initial node, if it exists
! 73: SET:$DATA(@SOURCE)#2 @DESTINATION=@SOURCE
! 74: ;
! 75: ; merge the rest of the nodes
! 76: NEW I,TST
! 77: FOR SET SOURCE=$QUERY(@SOURCE) QUIT:(SOURCE="")!(SRC=-1) DO
! 78: . ; quit at the end of a SOURCE sub-tree
! 79: . SET TST=SOURCE,$QSUBSCRIPT(TST,0)="",TST=$EXTRACT(TST,1,$LENGTH(SRC))
! 80: . IF TST'=SRC SET SRC=-1 QUIT
! 81: . ;
! 82: . ; clear DESTINATION keys
! 83: . SET DESTINATION=DST
! 84: . ; rebuild the DESTINATION array keys
! 85: . FOR I=1:1:($QLENGTH(SOURCE)-POP) DO
! 86: . . SET $QSUBSCRIPT(DESTINATION,I+PUSH)=$QSUBSCRIPT(SOURCE,I+POP)
! 87: . ;
! 88: . ; set the next node
! 89: . SET @DESTINATION=@SOURCE
! 90: ;
! 91: ; reset the naked indicator
! 92: SET SAVESRC=$GET(@SAVESRC),SAVEDST=$GET(@SAVEDST)
! 93: ;
! 94: QUIT 0
! 95: REVQUERY(CURRENT,PREVIOUS) ; Implement reverse $QUERY
! 96: ; CURRENT = global or local query reference passed by name indirection
! 97: ; PREVIOUS = previous global or local node, regardless of depth, passed by reference
! 98: ; returns 0
! 99: ;
! 100: ; if root node, return the empty string
! 101: IF $QLENGTH(CURRENT)=0 SET PREVIOUS="" QUIT 0
! 102: ELSE SET PREVIOUS=CURRENT
! 103: ;
! 104: NEW FLAG SET FLAG=0
! 105: NEW I,SUB
! 106: ; loop backwards, stripping off subscript levels on the same line
! 107: FOR I=1:1:$QLENGTH(CURRENT) QUIT:FLAG DO
! 108: . SET SUB=$ORDER(@PREVIOUS,-1)
! 109: . SET PREVIOUS=$NAME(@PREVIOUS,$QLENGTH(PREVIOUS)-1)
! 110: . ;
! 111: . ; if we moved to a new line, start looking forward
! 112: . IF SUB'="" DO SET FLAG=1
! 113: . . SET PREVIOUS=$NAME(@PREVIOUS@(SUB))
! 114: . . ;
! 115: . . NEW SAVE
! 116: . . ; loop forwards until we hit the original node
! 117: . . FOR DO QUIT:(PREVIOUS="") QUIT:($NAME(@PREVIOUS)=$NAME(@CURRENT))
! 118: . . . ; save off the correct previous node to return
! 119: . . . SET SAVE=PREVIOUS
! 120: . . . SET PREVIOUS=$QUERY(@PREVIOUS)
! 121: . . ;
! 122: . . SET PREVIOUS=SAVE
! 123: . ELSE IF $DATA(@PREVIOUS)#2 SET FLAG=1
! 124: ;
! 125: ; if root node doesn't exist, return the empty string
! 126: SET:(($QLENGTH(PREVIOUS)=0)&($DATA(@PREVIOUS)#2=0)) PREVIOUS=""
! 127: ;
! 128: QUIT 0
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>