Mercurial > emacs
annotate lisp/cedet/data-debug.el @ 107521:54f3a4d055ee
Document font-use-system-font.
* cmdargs.texi (Font X): Move most content to Fonts.
* frames.texi (Fonts): New node. Document font-use-system-font.
* emacs.texi (Top):
* xresources.texi (Table of Resources):
* mule.texi (Defining Fontsets, Charsets): Update xrefs.
| author | Chong Yidong <cyd@stupidchicken.com> |
|---|---|
| date | Sat, 20 Mar 2010 13:24:06 -0400 |
| parents | f6b8c73548b3 |
| children | 280c8ae2476d 376148b31b5e |
| rev | line source |
|---|---|
| 105241 | 1 ;;; data-debug.el --- Datastructure Debugger |
| 2 | |
| 106815 | 3 ;; Copyright (C) 2007, 2008, 2009, 2010 Free Software Foundation, Inc. |
| 105241 | 4 |
| 5 ;; Author: Eric M. Ludlam <zappo@gnu.org> | |
| 6 ;; Version: 0.2 | |
| 7 ;; Keywords: OO, lisp | |
| 8 | |
| 9 ;; This file is part of GNU Emacs. | |
| 10 | |
| 11 ;; GNU Emacs is free software: you can redistribute it and/or modify | |
| 12 ;; it under the terms of the GNU General Public License as published by | |
| 13 ;; the Free Software Foundation, either version 3 of the License, or | |
| 14 ;; (at your option) any later version. | |
| 15 | |
| 16 ;; GNU Emacs is distributed in the hope that it will be useful, | |
| 17 ;; but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 18 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 19 ;; GNU General Public License for more details. | |
| 20 | |
| 21 ;; You should have received a copy of the GNU General Public License | |
| 22 ;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>. | |
| 23 | |
| 24 ;;; Commentary: | |
| 25 ;; | |
| 26 ;; Provide a simple way to investigate particularly large and complex | |
| 27 ;; data structures. | |
| 28 ;; | |
| 29 ;; The best way to get started is to bind M-: to 'data-debug-eval-expression. | |
| 30 ;; | |
| 31 ;; (global-set-key "\M-:" 'data-debug-eval-expression) | |
| 32 ;; | |
| 33 ;; If you write functions with complex output that need debugging, you | |
| 34 ;; can make them interactive with data-debug-show-stuff. For example: | |
| 35 ;; | |
| 36 ;; (defun my-complex-output-fcn () | |
| 37 ;; "Calculate something complicated at point, and return it." | |
| 38 ;; (interactive) ;; function not normally interactive | |
| 39 ;; (let ((stuff (do-stuff))) | |
| 40 ;; (when (interactive-p) | |
| 41 ;; (data-debug-show-stuff stuff "myStuff")) | |
| 42 ;; stuff)) | |
| 43 | |
| 44 (require 'font-lock) | |
| 45 (require 'ring) | |
| 46 | |
| 47 ;;; Code: | |
| 48 | |
| 49 ;;; Compatibility | |
| 50 ;; | |
| 51 (if (featurep 'xemacs) | |
| 52 (eval-and-compile | |
| 53 (defalias 'data-debug-overlay-properties 'extent-properties) | |
| 54 (defalias 'data-debug-overlay-p 'extentp) | |
| 55 (if (not (fboundp 'propertize)) | |
| 56 (defun dd-propertize (string &rest properties) | |
| 57 "Mimic 'propertize' in from Emacs 23." | |
| 58 (add-text-properties 0 (length string) properties string) | |
| 59 string | |
| 60 ) | |
| 61 (defalias 'dd-propertize 'propertize)) | |
| 62 ) | |
| 63 ;; Regular Emacs | |
| 64 (eval-and-compile | |
| 65 (defalias 'data-debug-overlay-properties 'overlay-properties) | |
| 66 (defalias 'data-debug-overlay-p 'overlayp) | |
| 67 (defalias 'dd-propertize 'propertize) | |
| 68 ) | |
| 69 ) | |
| 70 | |
| 71 ;;; GENERIC STUFF | |
| 72 ;; | |
| 73 (defun data-debug-insert-property-list (proplist prefix &optional parent) | |
| 74 "Insert the property list PROPLIST. | |
| 75 Each line starts with PREFIX. | |
| 76 The attributes belong to the tag PARENT." | |
| 77 (while proplist | |
| 78 (let ((pretext (concat (symbol-name (car proplist)) " : "))) | |
| 79 (data-debug-insert-thing (car (cdr proplist)) | |
| 80 prefix | |
| 81 pretext | |
| 82 parent)) | |
| 83 (setq proplist (cdr (cdr proplist))))) | |
| 84 | |
| 85 ;;; overlays | |
| 86 ;; | |
| 87 (defun data-debug-insert-overlay-props (overlay prefix) | |
| 88 "Insert all the parts of OVERLAY. | |
| 89 PREFIX specifies what to insert at the start of each line." | |
| 90 (let ((attrprefix (concat (make-string (length prefix) ? ) "# ")) | |
| 91 (proplist (data-debug-overlay-properties overlay))) | |
| 92 (data-debug-insert-property-list | |
| 93 proplist attrprefix) | |
| 94 ) | |
| 95 ) | |
| 96 | |
| 97 (defun data-debug-insert-overlay-from-point (point) | |
| 98 "Insert the overlay found at the overlay button at POINT." | |
| 99 (let ((overlay (get-text-property point 'ddebug)) | |
| 100 (indent (get-text-property point 'ddebug-indent)) | |
| 101 start | |
| 102 ) | |
| 103 (end-of-line) | |
| 104 (setq start (point)) | |
| 105 (forward-char 1) | |
| 106 (data-debug-insert-overlay-props overlay | |
| 107 (concat (make-string indent ? ) | |
| 108 "| ")) | |
| 109 (goto-char start) | |
| 110 )) | |
| 111 | |
| 112 (defun data-debug-insert-overlay-button (overlay prefix prebuttontext) | |
| 113 "Insert a button representing OVERLAY. | |
| 105325 | 114 PREFIX is the text that precedes the button. |
| 105241 | 115 PREBUTTONTEXT is some text between prefix and the overlay button." |
| 116 (let ((start (point)) | |
| 117 (end nil) | |
| 118 (str (format "%s" overlay)) | |
| 119 (tip nil)) | |
| 120 (insert prefix prebuttontext str) | |
| 121 (setq end (point)) | |
| 122 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face) | |
| 123 (put-text-property start end 'ddebug overlay) | |
| 124 (put-text-property start end 'ddebug-indent(length prefix)) | |
| 125 (put-text-property start end 'ddebug-prefix prefix) | |
| 126 (put-text-property start end 'help-echo tip) | |
| 127 (put-text-property start end 'ddebug-function | |
| 128 'data-debug-insert-overlay-from-point) | |
| 129 (insert "\n") | |
| 130 ) | |
| 131 ) | |
| 132 | |
| 133 ;;; overlay list | |
| 134 ;; | |
| 135 (defun data-debug-insert-overlay-list (overlaylist prefix) | |
| 136 "Insert all the parts of OVERLAYLIST. | |
| 137 PREFIX specifies what to insert at the start of each line." | |
| 138 (while overlaylist | |
| 139 (data-debug-insert-overlay-button (car overlaylist) | |
| 140 prefix | |
| 141 "") | |
| 142 (setq overlaylist (cdr overlaylist)))) | |
| 143 | |
| 144 (defun data-debug-insert-overlay-list-from-point (point) | |
| 145 "Insert the overlay found at the overlay list button at POINT." | |
| 146 (let ((overlaylist (get-text-property point 'ddebug)) | |
| 147 (indent (get-text-property point 'ddebug-indent)) | |
| 148 start | |
| 149 ) | |
| 150 (end-of-line) | |
| 151 (setq start (point)) | |
| 152 (forward-char 1) | |
| 153 (data-debug-insert-overlay-list overlaylist | |
| 154 (concat (make-string indent ? ) | |
| 155 "* ")) | |
| 156 (goto-char start) | |
| 157 )) | |
| 158 | |
| 159 (defun data-debug-insert-overlay-list-button (overlaylist | |
| 160 prefix | |
| 161 prebuttontext) | |
| 162 "Insert a button representing OVERLAYLIST. | |
| 105325 | 163 PREFIX is the text that precedes the button. |
| 105241 | 164 PREBUTTONTEXT is some text between prefix and the overlay list button." |
| 165 (let ((start (point)) | |
| 166 (end nil) | |
| 167 (str (format "#<overlay list: %d entries>" (length overlaylist))) | |
| 168 (tip nil)) | |
| 169 (insert prefix prebuttontext str) | |
| 170 (setq end (point)) | |
| 171 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face) | |
| 172 (put-text-property start end 'ddebug overlaylist) | |
| 173 (put-text-property start end 'ddebug-indent(length prefix)) | |
| 174 (put-text-property start end 'ddebug-prefix prefix) | |
| 175 (put-text-property start end 'help-echo tip) | |
| 176 (put-text-property start end 'ddebug-function | |
| 177 'data-debug-insert-overlay-list-from-point) | |
| 178 (insert "\n") | |
| 179 ) | |
| 180 ) | |
| 181 | |
| 182 ;;; buffers | |
| 183 ;; | |
| 184 (defun data-debug-insert-buffer-props (buffer prefix) | |
| 185 "Insert all the parts of BUFFER. | |
| 186 PREFIX specifies what to insert at the start of each line." | |
| 187 (let ((attrprefix (concat (make-string (length prefix) ? ) "# ")) | |
| 188 (proplist | |
| 189 (list :filename (buffer-file-name buffer) | |
| 190 :live (buffer-live-p buffer) | |
| 191 :modified (buffer-modified-p buffer) | |
| 192 :size (buffer-size buffer) | |
| 193 :process (get-buffer-process buffer) | |
| 194 :localvars (buffer-local-variables buffer) | |
| 195 ))) | |
| 196 (data-debug-insert-property-list | |
| 197 proplist attrprefix) | |
| 198 ) | |
| 199 ) | |
| 200 | |
| 201 (defun data-debug-insert-buffer-from-point (point) | |
| 202 "Insert the buffer found at the buffer button at POINT." | |
| 203 (let ((buffer (get-text-property point 'ddebug)) | |
| 204 (indent (get-text-property point 'ddebug-indent)) | |
| 205 start | |
| 206 ) | |
| 207 (end-of-line) | |
| 208 (setq start (point)) | |
| 209 (forward-char 1) | |
| 210 (data-debug-insert-buffer-props buffer | |
| 211 (concat (make-string indent ? ) | |
| 212 "| ")) | |
| 213 (goto-char start) | |
| 214 )) | |
| 215 | |
| 216 (defun data-debug-insert-buffer-button (buffer prefix prebuttontext) | |
| 217 "Insert a button representing BUFFER. | |
| 105325 | 218 PREFIX is the text that precedes the button. |
| 105241 | 219 PREBUTTONTEXT is some text between prefix and the buffer button." |
| 220 (let ((start (point)) | |
| 221 (end nil) | |
| 222 (str (format "%S" buffer)) | |
| 223 (tip nil)) | |
| 224 (insert prefix prebuttontext str) | |
| 225 (setq end (point)) | |
| 226 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face) | |
| 227 (put-text-property start end 'ddebug buffer) | |
| 228 (put-text-property start end 'ddebug-indent(length prefix)) | |
| 229 (put-text-property start end 'ddebug-prefix prefix) | |
| 230 (put-text-property start end 'help-echo tip) | |
| 231 (put-text-property start end 'ddebug-function | |
| 232 'data-debug-insert-buffer-from-point) | |
| 233 (insert "\n") | |
| 234 ) | |
| 235 ) | |
| 236 | |
| 237 ;;; buffer list | |
| 238 ;; | |
| 239 (defun data-debug-insert-buffer-list (bufferlist prefix) | |
| 240 "Insert all the parts of BUFFERLIST. | |
| 241 PREFIX specifies what to insert at the start of each line." | |
| 242 (while bufferlist | |
| 243 (data-debug-insert-buffer-button (car bufferlist) | |
| 244 prefix | |
| 245 "") | |
| 246 (setq bufferlist (cdr bufferlist)))) | |
| 247 | |
| 248 (defun data-debug-insert-buffer-list-from-point (point) | |
| 249 "Insert the buffer found at the buffer list button at POINT." | |
| 250 (let ((bufferlist (get-text-property point 'ddebug)) | |
| 251 (indent (get-text-property point 'ddebug-indent)) | |
| 252 start | |
| 253 ) | |
| 254 (end-of-line) | |
| 255 (setq start (point)) | |
| 256 (forward-char 1) | |
| 257 (data-debug-insert-buffer-list bufferlist | |
| 258 (concat (make-string indent ? ) | |
| 259 "* ")) | |
| 260 (goto-char start) | |
| 261 )) | |
| 262 | |
| 263 (defun data-debug-insert-buffer-list-button (bufferlist | |
| 264 prefix | |
| 265 prebuttontext) | |
| 266 "Insert a button representing BUFFERLIST. | |
| 105325 | 267 PREFIX is the text that precedes the button. |
| 105241 | 268 PREBUTTONTEXT is some text between prefix and the buffer list button." |
| 269 (let ((start (point)) | |
| 270 (end nil) | |
| 271 (str (format "#<buffer list: %d entries>" (length bufferlist))) | |
| 272 (tip nil)) | |
| 273 (insert prefix prebuttontext str) | |
| 274 (setq end (point)) | |
| 275 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face) | |
| 276 (put-text-property start end 'ddebug bufferlist) | |
| 277 (put-text-property start end 'ddebug-indent(length prefix)) | |
| 278 (put-text-property start end 'ddebug-prefix prefix) | |
| 279 (put-text-property start end 'help-echo tip) | |
| 280 (put-text-property start end 'ddebug-function | |
| 281 'data-debug-insert-buffer-list-from-point) | |
| 282 (insert "\n") | |
| 283 ) | |
| 284 ) | |
| 285 | |
| 286 ;;; processes | |
| 287 ;; | |
| 288 (defun data-debug-insert-process-props (process prefix) | |
| 289 "Insert all the parts of PROCESS. | |
| 290 PREFIX specifies what to insert at the start of each line." | |
| 291 (let ((attrprefix (concat (make-string (length prefix) ? ) "# ")) | |
| 292 (id (process-id process)) | |
| 293 (tty (process-tty-name process)) | |
| 294 (pcontact (process-contact process t)) | |
| 295 (proplist (process-plist process))) | |
| 296 (data-debug-insert-property-list | |
| 297 (append | |
| 298 (if id (list 'id id)) | |
| 299 (if tty (list 'tty tty)) | |
| 300 (if pcontact pcontact) | |
| 301 proplist) | |
| 302 attrprefix) | |
| 303 ) | |
| 304 ) | |
| 305 | |
| 306 (defun data-debug-insert-process-from-point (point) | |
| 307 "Insert the process found at the process button at POINT." | |
| 308 (let ((process (get-text-property point 'ddebug)) | |
| 309 (indent (get-text-property point 'ddebug-indent)) | |
| 310 start | |
| 311 ) | |
| 312 (end-of-line) | |
| 313 (setq start (point)) | |
| 314 (forward-char 1) | |
| 315 (data-debug-insert-process-props process | |
| 316 (concat (make-string indent ? ) | |
| 317 "| ")) | |
| 318 (goto-char start) | |
| 319 )) | |
| 320 | |
| 321 (defun data-debug-insert-process-button (process prefix prebuttontext) | |
| 322 "Insert a button representing PROCESS. | |
| 105325 | 323 PREFIX is the text that precedes the button. |
| 105241 | 324 PREBUTTONTEXT is some text between prefix and the process button." |
| 325 (let ((start (point)) | |
| 326 (end nil) | |
| 327 (str (format "%S : %s" process (process-status process))) | |
| 328 (tip nil)) | |
| 329 (insert prefix prebuttontext str) | |
| 330 (setq end (point)) | |
| 331 (put-text-property (- end (length str)) end 'face 'font-lock-comment-face) | |
| 332 (put-text-property start end 'ddebug process) | |
| 333 (put-text-property start end 'ddebug-indent(length prefix)) | |
| 334 (put-text-property start end 'ddebug-prefix prefix) | |
| 335 (put-text-property start end 'help-echo tip) | |
| 336 (put-text-property start end 'ddebug-function | |
| 337 'data-debug-insert-process-from-point) | |
| 338 (insert "\n") | |
| 339 ) | |
| 340 ) | |
| 341 | |
| 342 ;;; Rings | |
| 343 ;; | |
| 344 ;; A ring (like kill-ring, or whatever.) | |
| 345 (defun data-debug-insert-ring-contents (ring prefix) | |
| 346 "Insert all the parts of RING. | |
| 347 PREFIX specifies what to insert at the start of each line." | |
| 348 (let ((len (ring-length ring)) | |
| 349 (idx 0) | |
| 350 ) | |
| 351 (while (< idx len) | |
| 352 (data-debug-insert-thing (ring-ref ring idx) prefix "") | |
| 353 (setq idx (1+ idx)) | |
| 354 ))) | |
| 355 | |
| 356 (defun data-debug-insert-ring-items-from-point (point) | |
| 357 "Insert the ring found at the ring button at POINT." | |
| 358 (let ((ring (get-text-property point 'ddebug)) | |
| 359 (indent (get-text-property point 'ddebug-indent)) | |
| 360 start | |
| 361 ) | |
| 362 (end-of-line) | |
| 363 (setq start (point)) | |
| 364 (forward-char 1) | |
| 365 (data-debug-insert-ring-contents ring | |
| 366 (concat (make-string indent ? ) | |
| 367 "} ")) | |
| 368 (goto-char start) | |
| 369 )) | |
| 370 | |
| 371 (defun data-debug-insert-ring-button (ring | |
| 372 prefix | |
| 373 prebuttontext) | |
| 374 "Insert a button representing RING. | |
| 105325 | 375 PREFIX is the text that precedes the button. |
| 105241 | 376 PREBUTTONTEXT is some text between prefix and the stuff list button." |
| 377 (let* ((start (point)) | |
| 378 (end nil) | |
| 379 (str (format "#<RING: %d, %d max>" | |
| 380 (ring-length ring) | |
| 381 (ring-size ring))) | |
| 382 (ringthing | |
| 383 (if (= (ring-length ring) 0) nil (ring-ref ring 0))) | |
| 384 (tip (format "Ring max-size %d, length %d." | |
| 385 (ring-size ring) | |
| 386 (ring-length ring))) | |
| 387 ) | |
| 388 (insert prefix prebuttontext str) | |
| 389 (setq end (point)) | |
| 390 (put-text-property (- end (length str)) end 'face 'font-lock-type-face) | |
| 391 (put-text-property start end 'ddebug ring) | |
| 392 (put-text-property start end 'ddebug-indent(length prefix)) | |
| 393 (put-text-property start end 'ddebug-prefix prefix) | |
| 394 (put-text-property start end 'help-echo tip) | |
| 395 (put-text-property start end 'ddebug-function | |
| 396 'data-debug-insert-ring-items-from-point) | |
| 397 (insert "\n") | |
| 398 ) | |
| 399 ) | |
| 400 | |
| 401 | |
| 402 ;;; Hash-table | |
| 403 ;; | |
| 404 | |
| 405 (defun data-debug-insert-hash-table (hash-table prefix) | |
| 406 "Insert the contents of HASH-TABLE inserting PREFIX before each element." | |
| 407 (maphash | |
| 408 (lambda (key value) | |
| 409 (data-debug-insert-thing | |
| 410 key prefix | |
| 411 (dd-propertize "key " 'face font-lock-comment-face)) | |
| 412 (data-debug-insert-thing | |
| 413 value prefix | |
| 414 (dd-propertize "val " 'face font-lock-comment-face))) | |
| 415 hash-table)) | |
| 416 | |
| 417 (defun data-debug-insert-hash-table-from-point (point) | |
| 418 "Insert the contents of the hash-table button at POINT." | |
| 419 (let ((hash-table (get-text-property point 'ddebug)) | |
| 420 (indent (get-text-property point 'ddebug-indent)) | |
| 421 start) | |
| 422 (end-of-line) | |
| 423 (setq start (point)) | |
| 424 (forward-char 1) | |
| 425 (data-debug-insert-hash-table | |
| 426 hash-table | |
| 427 (concat (make-string indent ? ) "> ")) | |
| 428 (goto-char start)) | |
| 429 ) | |
| 430 | |
| 431 (defun data-debug-insert-hash-table-button (hash-table prefix prebuttontext) | |
| 432 "Insert HASH-TABLE as expandable button with recursive prefix PREFIX and PREBUTTONTEXT in front of the button text." | |
| 433 (let ((string (dd-propertize (format "%s" hash-table) | |
| 434 'face 'font-lock-keyword-face))) | |
| 435 (insert (dd-propertize | |
| 436 (concat prefix prebuttontext string) | |
| 437 'ddebug hash-table | |
| 438 'ddebug-indent (length prefix) | |
| 439 'ddebug-prefix prefix | |
| 440 'help-echo | |
| 441 (format "Hash-table\nTest: %s\nWeakness: %s\nElements: %d (of %d)" | |
| 442 (hash-table-test hash-table) | |
| 443 (if (hash-table-weakness hash-table) "yes" "no") | |
| 444 (hash-table-count hash-table) | |
| 445 (hash-table-size hash-table)) | |
| 446 'ddebug-function | |
| 447 'data-debug-insert-hash-table-from-point) | |
| 448 "\n")) | |
| 449 ) | |
| 450 | |
| 451 ;;; Widget | |
| 452 ;; | |
| 453 ;; Widgets have a long list of properties | |
| 454 (defun data-debug-insert-widget-properties (widget prefix) | |
| 455 "Insert the contents of WIDGET inserting PREFIX before each element." | |
| 456 (let ((type (car widget)) | |
| 457 (rest (cdr widget))) | |
| 458 (while rest | |
| 459 (data-debug-insert-thing (car (cdr rest)) | |
| 460 prefix | |
| 461 (concat | |
| 462 (dd-propertize (format "%s" (car rest)) | |
| 463 'face font-lock-comment-face) | |
| 464 " : ")) | |
| 465 (setq rest (cdr (cdr rest)))) | |
| 466 )) | |
| 467 | |
| 468 (defun data-debug-insert-widget-from-point (point) | |
| 469 "Insert the contents of the widget button at POINT." | |
| 470 (let ((widget (get-text-property point 'ddebug)) | |
| 471 (indent (get-text-property point 'ddebug-indent)) | |
| 472 start) | |
| 473 (end-of-line) | |
| 474 (setq start (point)) | |
| 475 (forward-char 1) | |
| 476 (data-debug-insert-widget-properties | |
| 477 widget (concat (make-string indent ? ) "# ")) | |
| 478 (goto-char start)) | |
| 479 ) | |
| 480 | |
| 481 (defun data-debug-insert-widget (widget prefix prebuttontext) | |
| 482 "Insert one WIDGET. | |
| 483 A Symbol is a simple thing, but this provides some face and prefix rules. | |
| 105325 | 484 PREFIX is the text that precedes the button. |
| 105241 | 485 PREBUTTONTEXT is some text between prefix and the thing." |
| 486 (let ((string (dd-propertize (format "#<WIDGET %s>" (car widget)) | |
| 487 'face 'font-lock-keyword-face))) | |
| 488 (insert (dd-propertize | |
| 489 (concat prefix prebuttontext string) | |
| 490 'ddebug widget | |
| 491 'ddebug-indent (length prefix) | |
| 492 'ddebug-prefix prefix | |
| 493 'help-echo | |
| 494 (format "Widget\nType: %s\n# Properties: %d" | |
| 495 (car widget) | |
| 496 (/ (1- (length widget)) 2)) | |
| 497 'ddebug-function | |
| 498 'data-debug-insert-widget-from-point) | |
| 499 "\n"))) | |
| 500 | |
| 501 ;;; list of stuff | |
| 502 ;; | |
| 503 ;; just a list. random stuff inside. | |
| 504 (defun data-debug-insert-stuff-list (stufflist prefix) | |
| 505 "Insert all the parts of STUFFLIST. | |
| 506 PREFIX specifies what to insert at the start of each line." | |
| 507 (while stufflist | |
| 508 (data-debug-insert-thing | |
| 509 ;; Some lists may put a value in the CDR | |
| 510 (if (listp stufflist) (car stufflist) stufflist) | |
| 511 prefix | |
| 512 "") | |
| 513 (setq stufflist | |
| 514 (if (listp stufflist) | |
| 515 (cdr-safe stufflist) | |
| 516 nil)))) | |
| 517 | |
| 518 (defun data-debug-insert-stuff-list-from-point (point) | |
| 519 "Insert the stuff found at the stuff list button at POINT." | |
| 520 (let ((stufflist (get-text-property point 'ddebug)) | |
| 521 (indent (get-text-property point 'ddebug-indent)) | |
| 522 start | |
| 523 ) | |
| 524 (end-of-line) | |
| 525 (setq start (point)) | |
| 526 (forward-char 1) | |
| 527 (data-debug-insert-stuff-list stufflist | |
| 528 (concat (make-string indent ? ) | |
| 529 "> ")) | |
| 530 (goto-char start) | |
| 531 )) | |
| 532 | |
| 533 (defun data-debug-insert-stuff-list-button (stufflist | |
| 534 prefix | |
| 535 prebuttontext) | |
| 536 "Insert a button representing STUFFLIST. | |
| 105325 | 537 PREFIX is the text that precedes the button. |
| 105241 | 538 PREBUTTONTEXT is some text between prefix and the stuff list button." |
| 539 (let ((start (point)) | |
| 540 (end nil) | |
| 541 (str | |
| 542 (condition-case nil | |
| 543 (format "#<list o' stuff: %d entries>" (safe-length stufflist)) | |
| 544 (error "#<list o' stuff>"))) | |
| 545 (tip (if (or (listp (car stufflist)) | |
| 546 (vectorp (car stufflist))) | |
| 547 "" | |
| 548 (format "%s" stufflist)))) | |
| 549 (insert prefix prebuttontext str) | |
| 550 (setq end (point)) | |
| 551 (put-text-property (- end (length str)) end 'face 'font-lock-variable-name-face) | |
| 552 (put-text-property start end 'ddebug stufflist) | |
| 553 (put-text-property start end 'ddebug-indent (length prefix)) | |
| 554 (put-text-property start end 'ddebug-prefix prefix) | |
| 555 (put-text-property start end 'help-echo tip) | |
| 556 (put-text-property start end 'ddebug-function | |
| 557 'data-debug-insert-stuff-list-from-point) | |
| 558 (insert "\n") | |
| 559 ) | |
| 560 ) | |
| 561 | |
| 562 ;;; vector of stuff | |
| 563 ;; | |
| 564 ;; just a vector. random stuff inside. | |
| 565 (defun data-debug-insert-stuff-vector (stuffvector prefix) | |
| 566 "Insert all the parts of STUFFVECTOR. | |
| 567 PREFIX specifies what to insert at the start of each line." | |
| 568 (let ((idx 0)) | |
| 569 (while (< idx (length stuffvector)) | |
| 570 (data-debug-insert-thing | |
| 571 ;; Some vectors may put a value in the CDR | |
| 572 (aref stuffvector idx) | |
| 573 prefix | |
| 574 "") | |
| 575 (setq idx (1+ idx))))) | |
| 576 | |
| 577 (defun data-debug-insert-stuff-vector-from-point (point) | |
| 578 "Insert the stuff found at the stuff vector button at POINT." | |
| 579 (let ((stuffvector (get-text-property point 'ddebug)) | |
| 580 (indent (get-text-property point 'ddebug-indent)) | |
| 581 start | |
| 582 ) | |
| 583 (end-of-line) | |
| 584 (setq start (point)) | |
| 585 (forward-char 1) | |
| 586 (data-debug-insert-stuff-vector stuffvector | |
| 587 (concat (make-string indent ? ) | |
| 588 "[ ")) | |
| 589 (goto-char start) | |
| 590 )) | |
| 591 | |
| 592 (defun data-debug-insert-stuff-vector-button (stuffvector | |
| 593 prefix | |
| 594 prebuttontext) | |
| 595 "Insert a button representing STUFFVECTOR. | |
| 105325 | 596 PREFIX is the text that precedes the button. |
| 105241 | 597 PREBUTTONTEXT is some text between prefix and the stuff vector button." |
| 598 (let* ((start (point)) | |
| 599 (end nil) | |
| 600 (str (format "#<vector o' stuff: %d entries>" (length stuffvector))) | |
| 601 (tip str)) | |
| 602 (insert prefix prebuttontext str) | |
| 603 (setq end (point)) | |
| 604 (put-text-property (- end (length str)) end 'face 'font-lock-variable-name-face) | |
| 605 (put-text-property start end 'ddebug stuffvector) | |
| 606 (put-text-property start end 'ddebug-indent (length prefix)) | |
| 607 (put-text-property start end 'ddebug-prefix prefix) | |
| 608 (put-text-property start end 'help-echo tip) | |
| 609 (put-text-property start end 'ddebug-function | |
| 610 'data-debug-insert-stuff-vector-from-point) | |
| 611 (insert "\n") | |
| 612 ) | |
| 613 ) | |
| 614 | |
| 615 ;;; Symbol | |
| 616 ;; | |
| 617 | |
| 618 (defun data-debug-insert-symbol-from-point (point) | |
| 619 "Insert attached properties and possibly the value of symbol at POINT." | |
| 620 (let ((symbol (get-text-property point 'ddebug)) | |
| 621 (indent (get-text-property point 'ddebug-indent)) | |
| 622 start) | |
| 623 (end-of-line) | |
| 624 (setq start (point)) | |
| 625 (forward-char 1) | |
| 626 (when (and (not (fboundp symbol)) (boundp symbol)) | |
| 627 (data-debug-insert-thing | |
| 628 (symbol-value symbol) | |
| 629 (concat (make-string indent ? ) "> ") | |
| 630 (concat | |
| 631 (dd-propertize "value" | |
| 632 'face 'font-lock-comment-face) | |
| 633 " "))) | |
| 634 (data-debug-insert-property-list | |
| 635 (symbol-plist symbol) | |
| 636 (concat (make-string indent ? ) "> ")) | |
| 637 (goto-char start)) | |
| 638 ) | |
| 639 | |
| 640 (defun data-debug-insert-symbol-button (symbol prefix prebuttontext) | |
| 641 "Insert a button representing SYMBOL. | |
| 105325 | 642 PREFIX is the text that precedes the button. |
| 643 PREBUTTONTEXT is some text between prefix and the symbol button." | |
| 105241 | 644 (let ((string |
| 645 (cond ((fboundp symbol) | |
| 646 (dd-propertize (concat "#'" (symbol-name symbol)) | |
| 647 'face 'font-lock-function-name-face)) | |
| 648 ((boundp symbol) | |
| 649 (dd-propertize (concat "'" (symbol-name symbol)) | |
| 650 'face 'font-lock-variable-name-face)) | |
| 651 (t (format "'%s" symbol))))) | |
| 652 (insert (dd-propertize | |
| 653 (concat prefix prebuttontext string) | |
| 654 'ddebug symbol | |
| 655 'ddebug-indent (length prefix) | |
| 656 'ddebug-prefix prefix | |
| 657 'help-echo "" | |
| 658 'ddebug-function | |
| 659 'data-debug-insert-symbol-from-point) | |
| 660 "\n")) | |
| 661 ) | |
| 662 | |
| 663 ;;; String | |
| 664 (defun data-debug-insert-string (thing prefix prebuttontext) | |
| 665 "Insert one symbol THING. | |
| 666 A Symbol is a simple thing, but this provides some face and prefix rules. | |
| 105325 | 667 PREFIX is the text that precedes the button. |
| 105241 | 668 PREBUTTONTEXT is some text between prefix and the thing." |
| 669 (let ((newstr thing)) | |
| 670 (while (string-match "\n" newstr) | |
| 671 (setq newstr (replace-match "\\n" t t newstr))) | |
| 672 (while (string-match "\t" newstr) | |
| 673 (setq newstr (replace-match "\\t" t t newstr))) | |
| 674 (insert prefix prebuttontext | |
| 675 (dd-propertize (format "\"%s\"" newstr) | |
| 676 'face font-lock-string-face) | |
| 677 "\n" ))) | |
| 678 | |
| 679 ;;; Number | |
| 680 (defun data-debug-insert-number (thing prefix prebuttontext) | |
| 681 "Insert one symbol THING. | |
| 682 A Symbol is a simple thing, but this provides some face and prefix rules. | |
| 105325 | 683 PREFIX is the text that precedes the button. |
| 105241 | 684 PREBUTTONTEXT is some text between prefix and the thing." |
| 685 (insert prefix prebuttontext | |
| 686 (dd-propertize (format "%S" thing) | |
| 687 'face font-lock-string-face) | |
| 688 "\n")) | |
| 689 | |
| 690 ;;; Lambda Expression | |
| 691 (defun data-debug-insert-lambda-expression (thing prefix prebuttontext) | |
| 692 "Insert one lambda expression THING. | |
| 693 A Symbol is a simple thing, but this provides some face and prefix rules. | |
| 105325 | 694 PREFIX is the text that precedes the button. |
| 105241 | 695 PREBUTTONTEXT is some text between prefix and the thing." |
| 696 (let ((txt (prin1-to-string thing))) | |
| 697 (data-debug-insert-simple-thing | |
| 698 txt prefix prebuttontext 'font-lock-keyword-face)) | |
| 699 ) | |
| 700 | |
| 701 ;;; nil thing | |
| 702 (defun data-debug-insert-nil (thing prefix prebuttontext) | |
| 703 "Insert one simple THING with a face. | |
| 105325 | 704 PREFIX is the text that precedes the button. |
| 105241 | 705 PREBUTTONTEXT is some text between prefix and the thing. |
| 706 FACE is the face to use." | |
| 707 (insert prefix prebuttontext) | |
| 708 (insert ": ") | |
| 709 (let ((start (point)) | |
| 710 (end nil)) | |
| 711 (insert "nil") | |
| 712 (setq end (point)) | |
| 713 (insert "\n" ) | |
| 714 (put-text-property start end 'face 'font-lock-variable-name-face) | |
| 715 )) | |
| 716 | |
| 717 ;;; simple thing | |
| 718 (defun data-debug-insert-simple-thing (thing prefix prebuttontext face) | |
| 719 "Insert one simple THING with a face. | |
| 105325 | 720 PREFIX is the text that precedes the button. |
| 105241 | 721 PREBUTTONTEXT is some text between prefix and the thing. |
| 722 FACE is the face to use." | |
| 723 (insert prefix prebuttontext) | |
| 724 (let ((start (point)) | |
| 725 (end nil)) | |
| 726 (insert (format "%s" thing)) | |
| 727 (setq end (point)) | |
| 728 (insert "\n" ) | |
| 729 (put-text-property start end 'face face) | |
| 730 )) | |
| 731 | |
| 732 ;;; custom thing | |
| 733 (defun data-debug-insert-custom (thingstring prefix prebuttontext face) | |
| 734 "Insert one simple THINGSTRING with a face. | |
| 735 Use for simple items that need a custom insert. | |
| 105325 | 736 PREFIX is the text that precedes the button. |
| 105241 | 737 PREBUTTONTEXT is some text between prefix and the thing. |
| 738 FACE is the face to use." | |
| 739 (insert prefix prebuttontext) | |
| 740 (let ((start (point)) | |
| 741 (end nil)) | |
| 742 (insert thingstring) | |
| 743 (setq end (point)) | |
| 744 (insert "\n" ) | |
| 745 (put-text-property start end 'face face) | |
| 746 )) | |
| 747 | |
| 748 | |
| 749 (defvar data-debug-thing-alist | |
| 750 '( | |
| 751 ;; nil | |
| 752 (null . data-debug-insert-nil) | |
| 753 | |
| 754 ;; Overlay | |
| 755 (data-debug-overlay-p . data-debug-insert-overlay-button) | |
| 756 | |
| 757 ;; Overlay list | |
| 758 ((lambda (thing) (and (consp thing) (data-debug-overlay-p (car thing)))) . | |
| 759 data-debug-insert-overlay-list-button) | |
| 760 | |
| 761 ;; Buffer | |
| 762 (bufferp . data-debug-insert-buffer-button) | |
| 763 | |
| 764 ;; Buffer list | |
| 765 ((lambda (thing) (and (consp thing) (bufferp (car thing)))) . | |
| 766 data-debug-insert-buffer-list-button) | |
| 767 | |
| 768 ;; Process | |
| 769 (processp . data-debug-insert-process-button) | |
| 770 | |
| 771 ;; String | |
| 772 (stringp . data-debug-insert-string) | |
| 773 | |
| 774 ;; Number | |
| 775 (numberp . data-debug-insert-number) | |
| 776 | |
| 777 ;; Symbol | |
| 778 (symbolp . data-debug-insert-symbol-button) | |
| 779 | |
| 780 ;; Ring | |
| 781 (ring-p . data-debug-insert-ring-button) | |
| 782 | |
| 783 ;; Lambda Expression | |
| 784 ((lambda (thing) (and (consp thing) (eq (car thing) 'lambda))) . | |
| 785 data-debug-insert-lambda-expression) | |
| 786 | |
| 787 ;; Hash-table | |
| 788 (hash-table-p . data-debug-insert-hash-table-button) | |
| 789 | |
| 790 ;; Widgets | |
| 791 (widgetp . data-debug-insert-widget) | |
| 792 | |
| 793 ;; List of stuff | |
| 794 (listp . data-debug-insert-stuff-list-button) | |
| 795 | |
| 796 ;; Vector of stuff | |
| 797 (vectorp . data-debug-insert-stuff-vector-button) | |
| 798 ) | |
| 799 "Alist of methods used to insert things into an Ddebug buffer.") | |
| 800 | |
| 801 ;; An augmentation function for the thing alist. | |
| 802 (defun data-debug-add-specialized-thing (predicate fcn) | |
| 803 "Add a new specialized thing to display with data-debug. | |
| 804 PREDICATE is a function that returns t if a thing is this new type. | |
| 805 FCN is a function that will display stuff in the data debug buffer." | |
| 806 (let ((entry (cons predicate fcn)) | |
| 807 ;; Specialized entries show up AFTER nil, | |
| 808 ;; but before listp, vectorp, symbolp, and | |
| 809 ;; other general things. Splice it into | |
| 810 ;; the beginning. | |
| 811 (first (nthcdr 0 data-debug-thing-alist)) | |
| 812 (second (nthcdr 1 data-debug-thing-alist)) | |
| 813 ) | |
| 814 (when (not (member entry data-debug-thing-alist)) | |
| 815 (setcdr first (cons entry second))))) | |
| 816 | |
| 817 ;; uber insert method | |
| 818 (defun data-debug-insert-thing (thing prefix prebuttontext &optional parent) | |
| 819 "Insert THING with PREFIX. | |
| 820 PREBUTTONTEXT is some text to insert between prefix and the thing | |
| 821 that is not included in the indentation calculation of any children. | |
| 822 If PARENT is non-nil, it is somehow related as a parent to thing." | |
| 823 (when (catch 'done | |
| 824 (dolist (test data-debug-thing-alist) | |
| 825 (when (funcall (car test) thing) | |
| 826 (condition-case nil | |
| 827 (funcall (cdr test) thing prefix prebuttontext parent) | |
| 828 (error | |
| 829 (funcall (cdr test) thing prefix prebuttontext))) | |
| 830 (throw 'done nil)) | |
| 831 ) | |
| 832 nil) | |
| 833 (data-debug-insert-simple-thing (format "%S" thing) | |
| 834 prefix | |
| 835 prebuttontext | |
| 836 'bold))) | |
| 837 | |
| 838 ;;; MAJOR MODE | |
| 839 ;; | |
| 840 ;; The Ddebug major mode provides an interactive space to explore | |
| 841 ;; complicated data structures. | |
| 842 ;; | |
| 843 (defgroup data-debug nil | |
| 844 "data-debug group." | |
|
107167
f6b8c73548b3
Minor tweaks to custom groups.
Chong Yidong <cyd@stupidchicken.com>
parents:
106815
diff
changeset
|
845 :group 'extensions) |
| 105241 | 846 |
| 847 (defvar data-debug-mode-syntax-table | |
| 848 (let ((table (make-syntax-table (standard-syntax-table)))) | |
| 849 (modify-syntax-entry ?\; ". 12" table) ;; SEMI, Comment start ;; | |
| 850 (modify-syntax-entry ?\n ">" table) ;; Comment end | |
| 851 (modify-syntax-entry ?\" "\"" table) ;; String | |
| 852 (modify-syntax-entry ?\- "_" table) ;; Symbol | |
| 853 (modify-syntax-entry ?\\ "\\" table) ;; Quote | |
| 854 (modify-syntax-entry ?\` "'" table) ;; Prefix ` (backquote) | |
| 855 (modify-syntax-entry ?\' "'" table) ;; Prefix ' (quote) | |
| 856 (modify-syntax-entry ?\, "'" table) ;; Prefix , (comma) | |
| 857 | |
| 858 table) | |
| 859 "Syntax table used in data-debug macro buffers.") | |
| 860 | |
| 861 (defvar data-debug-map | |
| 862 (let ((km (make-sparse-keymap))) | |
| 863 (define-key km [mouse-2] 'data-debug-expand-or-contract-mouse) | |
| 864 (define-key km " " 'data-debug-expand-or-contract) | |
| 865 (define-key km "\C-m" 'data-debug-expand-or-contract) | |
| 866 (define-key km "n" 'data-debug-next) | |
| 867 (define-key km "p" 'data-debug-prev) | |
| 868 (define-key km "N" 'data-debug-next-expando) | |
| 869 (define-key km "P" 'data-debug-prev-expando) | |
| 870 km) | |
| 871 "Keymap used in data-debug.") | |
| 872 | |
| 873 (defcustom data-debug-mode-hook nil | |
| 874 "*Hook run when data-debug starts." | |
| 875 :group 'data-debug | |
| 876 :type 'hook) | |
| 877 | |
| 878 (defun data-debug-mode () | |
| 879 "Major-mode for the Analyzer debugger. | |
| 880 | |
| 881 \\{data-debug-map}" | |
| 882 (interactive) | |
| 883 (kill-all-local-variables) | |
| 884 (setq major-mode 'data-debug-mode | |
| 885 mode-name "DATA-DEBUG" | |
| 886 comment-start ";;" | |
| 887 comment-end "") | |
| 888 (set (make-local-variable 'comment-start-skip) | |
| 889 "\\(\\(^\\|[^\\\\\n]\\)\\(\\\\\\\\\\)*\\);+ *") | |
| 890 (set-syntax-table data-debug-mode-syntax-table) | |
| 891 (use-local-map data-debug-map) | |
| 892 (run-hooks 'data-debug-hook) | |
| 893 (buffer-disable-undo) | |
| 894 (set (make-local-variable 'font-lock-global-modes) nil) | |
| 895 (font-lock-mode -1) | |
| 896 ) | |
| 897 | |
| 898 ;;;###autoload | |
| 899 (defun data-debug-new-buffer (name) | |
| 900 "Create a new data-debug buffer with NAME." | |
| 901 (let ((b (get-buffer-create name))) | |
| 902 (pop-to-buffer b) | |
| 903 (set-buffer b) | |
| 904 (erase-buffer) | |
| 905 (data-debug-mode) | |
| 906 b)) | |
| 907 | |
| 908 ;;; Ddebug mode commands | |
| 909 ;; | |
| 910 (defun data-debug-next () | |
| 911 "Go to the next line in the Ddebug buffer." | |
| 912 (interactive) | |
| 913 (forward-line 1) | |
| 914 (beginning-of-line) | |
| 915 (skip-chars-forward " *-><[]" (point-at-eol))) | |
| 916 | |
| 917 (defun data-debug-prev () | |
| 918 "Go to the next line in the Ddebug buffer." | |
| 919 (interactive) | |
| 920 (forward-line -1) | |
| 921 (beginning-of-line) | |
| 922 (skip-chars-forward " *-><[]" (point-at-eol))) | |
| 923 | |
| 924 (defun data-debug-next-expando () | |
| 925 "Go to the next line in the Ddebug buffer. | |
| 926 Contract the current line (if open) and expand the line | |
| 927 we move to." | |
| 928 (interactive) | |
| 929 (data-debug-contract-current-line) | |
| 930 (data-debug-next) | |
| 931 (data-debug-expand-current-line) | |
| 932 ) | |
| 933 | |
| 934 (defun data-debug-prev-expando () | |
| 935 "Go to the previous line in the Ddebug buffer. | |
| 936 Contract the current line (if open) and expand the line | |
| 937 we move to." | |
| 938 (interactive) | |
| 939 (data-debug-contract-current-line) | |
| 940 (data-debug-prev) | |
| 941 (data-debug-expand-current-line) | |
| 942 ) | |
| 943 | |
| 944 (defun data-debug-current-line-expanded-p () | |
| 945 "Return non-nil if the current line is expanded." | |
| 946 (let ((ti (current-indentation)) | |
| 947 (ni (condition-case nil | |
| 948 (save-excursion | |
| 949 (end-of-line) | |
| 950 (forward-char 1) | |
| 951 (current-indentation)) | |
| 952 (error 0)))) | |
| 953 (> ni ti))) | |
| 954 | |
| 955 (defun data-debug-line-expandable-p () | |
| 956 "Return non-nil if the current line is expandable. | |
| 957 Lines that are not expandable are assumed to not be contractable." | |
| 958 (not (get-text-property (point) 'ddebug-noexpand))) | |
| 959 | |
| 960 (defun data-debug-expand-current-line () | |
| 961 "Expand the current line (if possible). | |
| 962 Do nothing if already expanded." | |
| 963 (when (or (not (data-debug-line-expandable-p)) | |
| 964 (not (data-debug-current-line-expanded-p))) | |
| 965 ;; If the next line is the same or less indentation, expand. | |
| 966 (let ((fcn (get-text-property (point) 'ddebug-function))) | |
| 967 (when fcn | |
| 968 (funcall fcn (point)) | |
| 969 (beginning-of-line) | |
| 970 )))) | |
| 971 | |
| 972 (defun data-debug-contract-current-line () | |
| 973 "Contract the current line (if possible). | |
| 974 Do nothing if already expanded." | |
| 975 (when (and (data-debug-current-line-expanded-p) | |
| 976 ;; Don't contract if the current line is not expandable. | |
| 977 (get-text-property (point) 'ddebug-function)) | |
| 978 (let ((ti (current-indentation)) | |
| 979 ) | |
| 980 ;; If next indentation is larger, collapse. | |
| 981 (end-of-line) | |
| 982 (forward-char 1) | |
| 983 (let ((start (point)) | |
| 984 (end nil)) | |
| 985 (condition-case nil | |
| 986 (progn | |
| 987 ;; Keep checking indentation | |
| 988 (while (or (> (current-indentation) ti) | |
| 989 (looking-at "^\\s-*$")) | |
| 990 (end-of-line) | |
| 991 (forward-char 1)) | |
| 992 (setq end (point)) | |
| 993 ) | |
| 994 (error (setq end (point-max)))) | |
| 995 (delete-region start end) | |
| 996 (forward-char -1) | |
| 997 (beginning-of-line))))) | |
| 998 | |
| 999 (defun data-debug-expand-or-contract () | |
| 1000 "Expand or contract anything at the current point." | |
| 1001 (interactive) | |
| 1002 (if (and (data-debug-line-expandable-p) | |
| 1003 (data-debug-current-line-expanded-p)) | |
| 1004 (data-debug-contract-current-line) | |
| 1005 (data-debug-expand-current-line)) | |
| 1006 (skip-chars-forward " *-><[]" (point-at-eol))) | |
| 1007 | |
| 1008 (defun data-debug-expand-or-contract-mouse (event) | |
| 1009 "Expand or contract anything at event EVENT." | |
| 1010 (interactive "e") | |
| 1011 (let* ((win (car (car (cdr event)))) | |
| 1012 ) | |
| 1013 (select-window win t) | |
| 1014 (save-excursion | |
| 1015 ;(goto-char (window-start win)) | |
| 1016 (mouse-set-point event) | |
| 1017 (data-debug-expand-or-contract)) | |
| 1018 )) | |
| 1019 | |
| 1020 ;;; GENERIC STRUCTURE DUMP | |
| 1021 ;; | |
| 1022 (defun data-debug-show-stuff (stuff name) | |
| 1023 "Data debug STUFF in a buffer named *NAME DDebug*." | |
| 1024 (data-debug-new-buffer (concat "*" name " DDebug*")) | |
| 1025 (data-debug-insert-thing stuff "?" "") | |
| 1026 (goto-char (point-min)) | |
| 1027 (when (data-debug-line-expandable-p) | |
| 1028 (data-debug-expand-current-line))) | |
| 1029 | |
| 1030 ;;; DEBUG COMMANDS | |
| 1031 ;; | |
| 1032 ;; Various commands for displaying complex data structures. | |
| 1033 | |
| 1034 (defun data-debug-edebug-expr (expr) | |
| 105325 | 1035 "Dump out the contents of some expression EXPR in edebug with ddebug." |
| 105241 | 1036 (interactive |
| 1037 (list (let ((minibuffer-completing-symbol t)) | |
| 1038 (read-from-minibuffer "Eval: " | |
| 1039 nil read-expression-map t | |
| 1040 'read-expression-history)) | |
| 1041 )) | |
| 1042 (let ((v (eval expr))) | |
| 1043 (if (not v) | |
| 1044 (message "Expression %s is nil." expr) | |
| 1045 (data-debug-show-stuff v "expression")))) | |
| 1046 | |
| 1047 (defun data-debug-eval-expression (expr) | |
| 1048 "Evaluate EXPR and display the value. | |
| 1049 If the result is something simple, show it in the echo area. | |
| 1050 If the result is a list or vector, then use the data debugger to display it." | |
| 1051 (interactive | |
| 1052 (list (let ((minibuffer-completing-symbol t)) | |
| 1053 (read-from-minibuffer "Eval: " | |
| 1054 nil read-expression-map t | |
| 1055 'read-expression-history)) | |
| 1056 )) | |
| 1057 | |
| 1058 (if (null eval-expression-debug-on-error) | |
| 1059 (setq values (cons (eval expr) values)) | |
| 1060 (let ((old-value (make-symbol "t")) new-value) | |
| 1061 ;; Bind debug-on-error to something unique so that we can | |
| 1062 ;; detect when evaled code changes it. | |
| 1063 (let ((debug-on-error old-value)) | |
| 1064 (setq values (cons (eval expr) values)) | |
| 1065 (setq new-value debug-on-error)) | |
| 1066 ;; If evaled code has changed the value of debug-on-error, | |
| 1067 ;; propagate that change to the global binding. | |
| 1068 (unless (eq old-value new-value) | |
| 1069 (setq debug-on-error new-value)))) | |
| 1070 | |
| 1071 (if (or (consp (car values)) (vectorp (car values))) | |
| 1072 (let ((v (car values))) | |
| 1073 (data-debug-show-stuff v "Expression")) | |
| 1074 ;; Old style | |
| 1075 (prog1 | |
| 1076 (prin1 (car values) t) | |
| 1077 (let ((str (eval-expression-print-format (car values)))) | |
| 1078 (if str (princ str t)))))) | |
| 1079 | |
| 1080 (provide 'data-debug) | |
| 1081 | |
| 1082 (if (featurep 'eieio) | |
| 1083 (require 'eieio-datadebug)) | |
| 1084 | |
| 105377 | 1085 ;; arch-tag: 4807227d-08e7-45c4-8ea5-9e4595c3bfb1 |
| 105241 | 1086 ;;; data-debug.el ends here |
