(* Copyright (C) 1992, Digital Equipment Corporation *) (* All rights reserved. *) (* See the file COPYRIGHT for a full description. *) (* *) (* ZSplit.i3, coded Fri Oct 31 11:24:53 1986 by cgn *) (* Last modified on Mon Feb 24 13:55:25 PST 1992 by muller *) (* modified on Wed Dec 11 18:29:58 PST 1991 by gnelson *) (* modified on Fri Feb 2 14:08:01 PST 1990 by glassman *) (* modified on Mon Jan 12 17:12:01 1987 by msm *) <*PRAGMA LL*> (* A "ZSplit.T" is a parent window with overlapping child windows. Each child has a stacking order given (conceptually) by a "z" coordinate. A pixel of the parent's screen that is in the domain of more than one child is controlled by whichever of these children is highest in the "z" coordinate. The portions of the domains of the children that extend outside the parent domain will be clipped. "Split.Succ" enumerates the children from top to bottom. The bottom child is called the {\it background}. An initial background can be specified when the "ZSplit" is created; usually it remains the background throughout the life of the "ZSplit". Usually the background has the same domain as the parent, and therefore controls all pixels that are not controlled by any other child. In the unusual case that the background child has a domain different from the parent domain, there may be some parent pixels that are not controlled by any child. The "ZSplit" will ignore these pixels when asked to repaint. The shape of a "ZSplit" is the shape of its background child (if it has no children its shape is the default shape for a "VBT"). When the preferred shape of a non-background child changes, the "ZSplit" reshapes the child to its new preferred shape, preserving its {\it offset}, which is the vector between the northwest corners of the parent and child. *) INTERFACE ZSplit; IMPORT VBT, Rect, Split, Point; TYPE T <: Public; Private <: Split.T; Public = Private OBJECT METHODS <* LL <= VBT.mu *> init(bg: VBT.T := NIL; saveBits := FALSE; parlim: INTEGER := -1): T END; (* The call "v.init(...)" initializes "v" as a "ZSplit". *) (* It is only legal to call the "init" method for a newly-allocated "ZSplit" (as in the definition of the procedure "New" below) or from the "init" method of a subclass. This restriction applies to all the "init" methods in Trestle, although it will not be repeated for each one. \index{init method, rules for calling} The "ZSplit" will be given the initial background child "bg" if "bg#NIL"; it will be given no children if "bg=NIL". If "bg" is non-"NIL" it will be mapped initially. If "saveBits" is "TRUE", the split will try to save the children's old bits when reformatting; if the children don't use them anyway, it is faster to let "saveBits" default to "FALSE". The value of "parlim" is the minimum area of a child for which a separate thread will be forked to reshape or repaint it; if it is "-1", it is set to an appropriate default (see the "VBTTuning" interface). *) PROCEDURE New( bg: VBT.T := NIL; saveBits := FALSE; parlim: INTEGER := -1) : T; <* LL <= VBT.mu *> (* "New(...)" is equivalent to "NEW(T).init(...)". *) (* \subsubsection{Inserting children} *) (* The default "Split.Insert" call is rarely useful for a "ZSplit": it inserts the new child at the parent's northwest corner, unmapped. "Split.AddChild" is even less useful, since it adds children as the background, which is almost certainly not what you want. The following procedures are more useful for inserting children into a "ZSplit": *) PROCEDURE InsertAfter( v: T; pred, ch: VBT.T; READONLY dom: Rect.T; alsoMap: BOOLEAN := TRUE) RAISES {Split.NotAChild}; <* LL.sup = VBT.mu *> (* Insert "ch" as a new child of "v" with domain "dom", and mark "v" for redisplay. *) (* The new child is inserted immediately after (that is, below) "pred"; if "pred=NIL" the new child is inserted first (that is, on top). If the height or width of "dom" does not satisfy "ch"'s size contraints, then the height and width of the child are projected into range; its offset is preserved. This is a checked runtime error if "ch" is not detached. If "alsoMap" is "TRUE", "ch" is mapped, otherwise it is unmapped. It is occasionally useful to insert a new child below all existing children except the background, in which case the following procedure is handy: *) TYPE Altitude = {Top, Bot}; PROCEDURE Insert( v: T; ch: VBT.T; READONLY dom: Rect.T; alt := Altitude.Top; alsoMap: BOOLEAN := TRUE); <* LL.sup = VBT.mu *> (* Insert "ch" at the top if "alt = Altitude.Top"; insert "ch" just above the background if "alt = Altitude.Bot". *) (* That is, "Insert" is equivalent to | IF alt = Altitude.Top THEN | pred := NIL | ELSE | pred := Split.Pred(v, Split.Pred(v, NIL)) | END; | InsertAfter(v, pred, ch, dom, alsoMap) Finally, instead of providing the new child's domain it can be useful to provide only the northwest corner and let the child's domain be determined by its shape constraints: *) PROCEDURE InsertAt( v: T; ch: VBT.T; nw: Point.T; alt := Altitude.Top; alsoMap: BOOLEAN := TRUE); <* LL.sup = VBT.mu *> (* Insert "ch" with its preferred shape and its northwest corner at "nw". The "alt" and "alsoMap" parameters are interpreted as in "Insert". *) (* \subsubsection{Moving, lifting, and lowering children} *) PROCEDURE Move(ch: VBT.T; READONLY dom: Rect.T); <* LL.sup = VBT.mu *> (* Change the domain of "ch" to be "dom" and mark "ch"'s parent for redisplay. *) (* If the height or width of "dom" do not satisfy "ch"'s size constraints, then they are projected into range, preserving the northwest corner of "dom". The stacking order of "ch" is not changed. "Move" is a checked runtime error if "ch"'s parent is not a "ZSplit". Note that this has nothing to do with "Split.Move", unlike the next procedure. *) PROCEDURE Lift(ch: VBT.T; alt := Altitude.Top); <* LL.sup = VBT.mu *> (* Lift "ch" to the top or lower it to be just above the background, depending on "alt". "Lift" is equivalent to: | v := VBT.Parent(ch); | IF alt = Altitude.Top THEN | pred := NIL | ELSE | pred := Split.Pred(v, Split.Pred(v, NIL)) | END; | Split.Move(v, pred, ch) *) (* \subsubsection{Mapping and unmapping children} *) (* You can {\it unmap} a child of a "ZSplit", which reshapes the child to be empty after recording the child's shape and offset. When you later {\it map} the child, the recorded shape and offset are restored. An unmapped child is rescreened when the parent is rescreened, and its recorded shape and offset are updated when the parent is reshaped, just like the domains of the mapped children. *) PROCEDURE Unmap(ch: VBT.T); <* LL.sup = VBT.mu *> (* If "ch" is mapped, unmap it and mark its parent for redisplay. *) PROCEDURE Map(ch: VBT.T); <* LL.sup = VBT.mu *> (* If "ch" is unmapped, map it and mark its parent for redisplay. *) PROCEDURE IsMapped(ch: VBT.T): BOOLEAN; <* LL.sup = VBT.mu *> (* Return "TRUE" if ch is "mapped" and "FALSE" if "ch" is unmapped. *) (* "Map", "Unmap", and "IsMapped" are checked runtime errors if "ch"'s parent is not a "ZSplit". *) (* \subsubsection{Getting domains} *) PROCEDURE GetDomain(ch: VBT.T): Rect.T; <* LL.sup = VBT.mu *> (* Return the effective domain of "ch". *) (* The effective domain is the same as the normal domain, except (1) if the parent has been marked for redisplay, "GetDomain" returns the domain that "ch" will receive when the redisplay happens, or (2) if the domain of the parent is "Rect.Empty", "GetDomain" returns the domain "ch" would receive if the parent were reshaped to its last non-empty domain, or (3) if the child is unmapped, "GetDomain" returns the domain the child would have if it were mapped. "GetDomain" is a checked runtime error if the parent of "ch" is not a "ZSplit". *) PROCEDURE GetParentDomain(v: T): Rect.T; <* LL.sup = VBT.mu *> (* Return the last non-empty value of "v.domain", or "Rect.Empty" if "v.domain" has always been empty. *) (* \subsubsection{Moving children when the parent is reshaped} *) (* You can supply procedures to control what happens to the children when a "ZSplit" is reshaped. If you don't supply a procedure, the default behavior is as follows: the initial background child is always reshaped to have the same domain as the parent. The other children are reshaped so as to preserve their shape and their offsets (even if this makes them extend outside the parent domain). The rule is different if the parent is reshaped to "Rect.Empty": in this case the "ZSplit" records its children's shapes and offsets and reshapes them all to "Rect.Empty". When the "ZSplit" is later reshaped to a non-empty domain, it reshapes the initial background child to have the same domain as the parent, and restores the saved dimensions and offsets of the other children. In the unusual case that the initial background child is deleted, subsequent background children do not automatically inherit the special reshaping behavior of the initial background child. To override the default behavior, use "SetReshapeControl": *) PROCEDURE SetReshapeControl( ch: VBT.T; rc: ReshapeControl); <* LL.sup = VBT.mu *> (* Set the reshape control object for the child "ch" to be "rc". *) TYPE ReshapeControl = OBJECT METHODS apply(ch:VBT.T; READONLY old, new, prev: Rect.T) : Rect.T <* LL.sup = VBT.mu.ch *> END; (* "SetReshapeControl" arranges that whenever the "ZSplit" parent "v" of "ch" is reshaped from domain "old" to domain "new", then if the previous domain of "ch" is "prev", the new domain of "ch" will become "rc.apply(ch, old, new, prev)" (if this rectangle doesn't satisfy "ch"'s size constraints, its height and width will be projected into range, preserving its offset). These methods of the "ReshapeControl" objects may be called concurrently for different children. (This is why the apply method has only a share of "VBT.mu".) The stacking order is not changed by reshaping. When a "ZSplit" child is replaced by "Split.Replace", the new child inherits the old child's reshape control object. "SetReshapeControl" is a checked runtime error if the parent of "ch" is not a "ZSplit". If the "ZSplit" is reshaped to "Rect.Empty", it will reshape its children to "Rect.Empty" without calling their reshape control methods. Similarly, if the parent is subsequently reshaped to its original rectangle, it will restore the children's previous domains without calling the methods. One useful reshape control method provided by this interface is "ChainReshape", in which some set of the child's west, east, north, and south edges are ``chained'' to the corresponding edges of the parent. Chaining an edge means that the distance between the child edge and the corresponding parent edge will be preserved. For example, if both the west and east edges are chained, then the child's horizontal extent will be inset into the parent's horizontal extent by fixed amounts on both sides. For another example, suppose that the the east edge is chained and the west edge is not. In this case the distance between the east edges of the child and parent will be preserved, but the west edge of the child will move so as to preserve the width of the child. The north and south edges control the vertical extent in a similar manner. *) TYPE Ch = {W, E, N, S}; ChainSet = SET OF Ch; ChainReshapeControl = ReshapeControl OBJECT chains: ChainSet OVERRIDES apply := ChainedReshape END; VAR (*CONST*) NoChains, WChains, EChains, WEChains, NChains, WNChains, ENChains, WENChains, SChains, WSChains, ESChains, WESChains, NSChains, WNSChains, ENSChains, WENSChains: ChainReshapeControl; (* The ``variables'' above are constants for the following reshape control objects: | NEW(ChainReshapeControl, chains := ChainSet{}), | NEW(ChainReshapeControl, chains := ChainSet{Ch.W}), | | ... | | NEW(ChainReshapeControl, | chains := ChainSet{Ch.W,Ch.E,Ch.N,Ch.S}) *) PROCEDURE ChainedReshape( self: ChainReshapeControl; ch: VBT.T; READONLY oldParentDomain, newParentDomain, oldChildDomain: Rect.T): Rect.T; (* Return the rectangle that results from chaining each edge in "self.chains" to the corresponding edge of the parent domain, and leaving the other edges unconstrained. *) (* If both edges in a dimension are chained, the offset and extent of the child will both vary to satisfy the chain constraints; if one edge is chained, the offset will vary and the extent will be fixed; if both edges are unchained, the offset and the extent will both be fixed. *) (* For example, the default behavior for the initial background child is "WENSChains", and the default behavior for all other children is "WNChains". One final reshape control method is sometimes useful: *) PROCEDURE ScaledReshape( self: ReshapeControl; ch: VBT.T; READONLY oldParentDomain, newParentDomain, oldChildDomain: Rect.T) : Rect.T; (* Return the integer approximation to the rectangle that results from scaling the old child domain to occupy the same relative position of the changing parent domain. *) VAR (*CONST*) Scaled: ReshapeControl; (* This ``variable'' is really a constant for the following reshape control object: | NEW(ReshapeControl, apply := ScaledReshape) *) END ZSplit.