[mcclim-cvs] CVS mcclim/Experimental/tree-with-cross-edges

rgoldman rgoldman at common-lisp.net
Fri Mar 9 23:42:34 UTC 2007


Update of /project/mcclim/cvsroot/mcclim/Experimental/tree-with-cross-edges
In directory clnet:/tmp/cvs-serv24198

Added Files:
	mcclim-tree-with-cross-edges.asd tree-with-cross-edges.lisp 
Log Message:
First draft version of an experimental extension to the graph-formatting protocol.


--- /project/mcclim/cvsroot/mcclim/Experimental/tree-with-cross-edges/mcclim-tree-with-cross-edges.asd	2007/03/09 23:42:34	NONE
+++ /project/mcclim/cvsroot/mcclim/Experimental/tree-with-cross-edges/mcclim-tree-with-cross-edges.asd	2007/03/09 23:42:34	1.1
;;;; -*- Lisp -*-

;;;---------------------------------------------------------------------------
;;; Copyright (c) 2005-2007 Robert P. Goldman and Smart Information
;;; Flow Technologies, d/b/a SIFT, LLC
;;;
;;; This library is free software; you can redistribute it and/or
;;; modify it under the terms of the GNU Library General Public
;;; License as published by the Free Software Foundation; either
;;; version 2 of the License, or (at your option) any later version.
;;;
;;; This library is distributed in the hope that it will be useful,
;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
;;; Library General Public License for more details.
;;;
;;; You should have received a copy of the GNU Library General Public
;;; License along with this library; if not, write to the 
;;; Free Software Foundation, Inc., 59 Temple Place - Suite 330, 
;;; Boston, MA  02111-1307  USA.
;;;
;;; All rights reserved.
;;;
;;;---------------------------------------------------------------------------
;;; File Description:
;;;
;;;    A system that adds a new type of graph to the
;;;    format-graph-from-roots protocol for McCLIM.
;;;
;;;
;;;---------------------------------------------------------------------------


(defpackage :mcclim-tree-with-cross-edges-system (:use :cl :asdf))
(in-package :mcclim-tree-with-cross-edges-system)

(defsystem :mcclim-tree-with-cross-edges
    :depends-on (:mcclim)
    :serial t
    :components
    ((:file "tree-with-cross-edges")))




--- /project/mcclim/cvsroot/mcclim/Experimental/tree-with-cross-edges/tree-with-cross-edges.lisp	2007/03/09 23:42:34	NONE
+++ /project/mcclim/cvsroot/mcclim/Experimental/tree-with-cross-edges/tree-with-cross-edges.lisp	2007/03/09 23:42:34	1.1
;;; -*- Mode: Lisp; Package: CLIM-INTERNALS -*-

;;;---------------------------------------------------------------------------
;;; Copyright (c) 2005-2007 Robert P. Goldman and Smart Information
;;; Flow Technologies, d/b/a SIFT, LLC
;;;
;;; This library is free software; you can redistribute it and/or
;;; modify it under the terms of the GNU Library General Public
;;; License as published by the Free Software Foundation; either
;;; version 2 of the License, or (at your option) any later version.
;;;
;;; This library is distributed in the hope that it will be useful,
;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
;;; Library General Public License for more details.
;;;
;;; You should have received a copy of the GNU Library General Public
;;; License along with this library; if not, write to the 
;;; Free Software Foundation, Inc., 59 Temple Place - Suite 330, 
;;; Boston, MA  02111-1307  USA.
;;;
;;; All rights reserved.
;;;
;;;---------------------------------------------------------------------------
;;; File Description:
;;;
;;; File for definitions of a new graph type that should allow tree
;;; style layouts with edges across in a level. [2005/05/05:rpg]
;;;
;;; History/Bugs/Notes:
;;;
;;;   [2005/05/05:rpg] Created.
;;;
;;;---------------------------------------------------------------------------

(in-package "CLIM-INTERNALS")

;;;---------------------------------------------------------------------------
;;; A graph with cross trees will have an additional type option: a
;;; cross-edge-producer
;;;---------------------------------------------------------------------------


(eval-when (:compile-toplevel :load-toplevel :execute)
  (define-graph-type :tree-with-cross-edges cross-tree-output-record))

(defun standard-cross-arc-drawer (stream from-node to-node x1 y1 x2 y2
				  &rest drawing-options
				  &key edge-type &allow-other-keys)
  "The standard cross-arc-drawer simply ignores the edge-type keyword argument."
  (declare (ignore edge-type))
  (remf drawing-options :edge-type)
  (apply #'standard-arc-drawer stream from-node to-node x1 y1 x2 y2 drawing-options))

(defclass cross-tree-output-record (tree-graph-output-record)
     ((cross-arc-drawer
       :initarg :cross-arc-drawer
       :reader cross-arc-drawer
       :documentation
       "This slot should be bound to a function that 
takes all the arguments accepted by a normal arc-drawer,
but also an edge-type keyword argument, which it is free
to ignore."
       :initform #'standard-cross-arc-drawer
       )
      (cross-arc-producer
       :initarg :cross-arc-producer
       ;; by default, this just acts like a tree...
       :initform nil 
       :reader cross-arc-producer
       :documentation
       "This should be bound to a function that 
takes a graph-node as argument, like inferior-producer,
but that returns two values:  a list of destination
nodes and (optionally) a list of type-designators, that
can be passed to the cross-arc-drawer, as the value of the
:edge-type keyword argument."
       )
      (cross-arc-drawing-options
       :reader cross-arc-drawing-options
       )
      )
  )

;;;---------------------------------------------------------------------------
;;; This is very yucky.  It will be expensive on large graphs (perhaps
;;; a mixin for using a hash-table would be better), and needs some
;;; kind of good way of specifying the test in your graph class, which
;;; will be difficult... [2005/05/06:rpg]
;;;---------------------------------------------------------------------------

(defmethod lookup-node (source-node (graph graph-output-record)
			&key (test #'eql)
			     (default :error))
  (let ((hash-table (make-hash-table :test #'eq)))
    (flet ((visitedp (node)
	     (gethash node hash-table nil))
	   (mark (node)
	     (setf (gethash node hash-table) t)))
      (or
       (loop with openlist = (graph-root-nodes graph)
	   for node = (pop openlist)
	   while node
	   unless (visitedp node)
	   when (funcall test source-node (graph-node-object node))
	   return node
	   end
	   and do (mark node)
		  (setf openlist (append openlist (graph-node-children node))))
       (when (eq default :error)
	 (error "Unable to find graph node for ~S in ~S"
		source-node graph))
       default))))
	    
(defmethod initialize-instance :after ((obj cross-tree-output-record) &key cross-arc-drawing-options
				       arc-drawing-options)
  "A possibly reasonable default is to draw cross-arcs as if they were 
normal tree edge arcs."
  (unless cross-arc-drawing-options
    (setf (slot-value obj 'cross-arc-drawing-options)
	  arc-drawing-options)))

;;; note that this could later be made into a function argument, so
;;; that programmers could customize [2005/05/06:rpg]
(defgeneric cross-arc-routing (from to orientation)
  (:documentation "Return four values, x1, y1, x2, y2 for
the arc-drawing for a cross-arc.  More complex than for
the tree case."))

(defun middle (dim1 dim2)
  (/ (+ dim1 dim2) 2))
  
(defmethod cross-arc-routing (from to (orientation (eql :horizontal)))
  (with-bounding-rectangle* (x1 y1 x2 y2) from
    (with-bounding-rectangle* (u1 v1 u2 v2) to
      (cond ((< x2 u1)
	     ;; node entirely to the left of k
	     (values x2 (middle y1 y2) u1 (middle v1 v2)))
	    ((< u2 x1)
	     ;; node entirely to the right of k
	     ;; draw from the top or bottom to make distinguishable...
	     (if (<= v1 y1)
		 ;; draw from the top to the x middle of TO on the
		 ;; bottom
		 (values x1 y1 (middle u1 u2) v2)
	       ;; draw from the bottom to the x middle of TO on the
	       ;; top...
	       (values x1 y2 (middle u1 u2) v1)))
	    ;; overlapping in X -- as long as this is a tree, means
	    ;; they are siblings.
	    ((< y2 v1)
	     ;; FROM above: middle x of FROM to middle x of TO, bottom to top...
	     (values (middle x1 x2) y2 (middle u1 u2) v1))
	    ((< v2 y1)
	     ;; TO above: middle x of FROM to middle x of TO, top to bottom...
	     (values (middle x1 x2) y1 (middle u1 u2) v2))
	    (t
	     (error "Unforeseen node positioning."))))))

;;; copied from original layout-graph-edges and enhanced to add cross
;;; edges.
(defmethod layout-graph-edges :after ((graph cross-tree-output-record)
				      stream arc-drawer arc-drawing-options)
  "After the main method has drawn the tree, add the cross-edges."
  (declare (ignore arc-drawer arc-drawing-options))
;;;  (format excl:*initial-terminal-io* "~&Invoking after method to layout cross-edges.~%")
;;;  (unless (cross-arc-producer graph)
;;;    (format excl:*initial-terminal-io* "~&Uh-oh!  No cross-arc-producer!~%"))
  (with-slots (orientation) graph
    ;; We tranformed the position of the nodes when we inserted them into
    ;; output history, so the bounding rectangles queried below will be
    ;; transformed. Therefore, disable the transformation now, otherwise
    ;; the transformation is effectively applied twice to the edges.
    (when (cross-arc-producer graph)
      (with-identity-transformation (stream)
	;; for some damn reason, this graph traversal isn't working....
	(traverse-graph-nodes graph
			      (lambda (node children continuation)
;;;				(format excl:*initial-terminal-io*
;;;					"~&Invoking traverse function on ~S and ~S!~%" node children)
				(unless (eq node graph)
				  (multiple-value-bind (source-siblings types)
				      (funcall (cross-arc-producer graph)
					       (graph-node-object node))
				    ;; there's a kind of odd loop here
				    ;; because types might be NIL.  Using
				    ;; a built-in stepper would cause the
				    ;; loop to terminate too soon if types
				    ;; was nil [2005/05/06:rpg]
				    (loop for ss in source-siblings
					for k = (lookup-node ss graph)
					for typelist = types then (cdr typelist)
					for type = (when typelist (car typelist))
					do (multiple-value-bind (fromx fromy tox toy)
					       (cross-arc-routing node k orientation)
					     (apply (cross-arc-drawer graph) stream node k
						    fromx fromy 
						    tox  toy
						    :edge-type type
						    (cross-arc-drawing-options graph))))))
				(map nil continuation children)))))))



More information about the Mcclim-cvs mailing list