psgml/psgml-xpointer.el

84 lines
2.6 KiB
EmacsLisp
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

;; psgml-xpointer.el -- generate XPointers for element in XML and SGML docs.
;; Copyright (c) 1998 Megginson Technologies Ltd.
;; Author: David Megginson (david@megginson.com)
;; $Id: psgml-xpointer.el,v 1.1 1998/08/22 03:14:36 david Exp $
;;; Commentary:
;; This mode includes a single user-level command for generating an
;; XPointer to a single element in an XML or SGML document (by
;; default, the element containing point). You must be using Lennart
;; Staflin's PSGML mode (optionally with XML patches). To obtain an
;; XPointer for the current point in a document, issue the following
;; command (which may be bound to a key sequence for convenience):
;;
;; M-x sgml-xpointer
;;
;; The program will climb the element tree until it finds an element
;; with an ID attribute, then will use child() statements to locate
;; the closest element that contains point.
;;
;; Installation:
;;
;; Put the following in one of your startup files:
;;
;; (autoload 'sgml-xpointer "psgml-xpointer" nil t)
;;
;;; Code:
(require 'psgml-parse)
;;;###autoload
(defun sgml-xpointer (loc)
"Display an XPointer for the current point in an XML or SGML document.
The XPointer will appear in a temporary buffer."
(interactive "d")
(with-output-to-temp-buffer "*XPOINTER*"
(princ (sgml-xpointer-string (sgml-find-element-of loc)))))
(defun sgml-xpointer-string (el)
"Return a string containing an XPointer for the element containing point.
The XPointer will begin with the nearest ancestor that possesses an ID
attribute, or with the document root if no ancestor has an ID."
(cond ((= (sgml-element-level el) 0)
(error "XPointer Outside of document element!"))
((= (sgml-element-level el) 1)
"root")
((sgml-element-id el)
(format "id('%s')" (sgml-element-id el)))
(t
(concat
(sgml-xpointer-string (sgml-element-parent el))
(format ".child(%d,#element,'%s')"
(sgml-element-child-number el)
(sgml-element-gi el))))))
(defun sgml-element-id (el)
"Return the value of the ID attribute for this element, if any."
(let ((id (sgml-attribute-with-declared-value
(sgml-element-attlist el) 'id)))
(if id
(sgml-element-attval el (sgml-attspec-name id))
nil)))
(defun sgml-element-child-number (el)
"Return the child number of the current element.
The child number counts only elements with the same GI."
(let ((sibling (sgml-element-content (sgml-element-parent el)))
(gi (sgml-element-gi el))
(n 1))
(while (not (equal sibling el))
(if (equal gi (sgml-element-gi sibling))
(setq n (1+ n)))
(setq sibling (sgml-element-next sibling)))
n))
(provide 'psgml-xpointer)
;;; psgml-xpointer.el ends here.