forked from jdee-emacs/jdee
-
Notifications
You must be signed in to change notification settings - Fork 0
/
jdee-open-source.el
624 lines (553 loc) · 26.2 KB
/
jdee-open-source.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
;; jdee-open-source.el -- Open class source files
;;
;; Author: Klaus Berndl
;; Maintainer: Paul Landes <landes <at> mailc dt net>
;; Keywords: java, open files
;; Copyright (C) 2002, 2003, 2004 Klaus Berndl
;; Copyright (C) 2009 by Paul Landes
;; This package follows the GNU General Public Licence (GPL), see the
;; COPYING file that comes along with GNU Emacs. This is free software,
;; you can redistribute it and/or modify it under the GNU GPL terms.
;;
;; Java is a registered trademark of Sun Microsystem, Inc.
;;
;;; Commentary:
;; This package allows to open the class at point.
;;; Code:
(require 'cl-lib)
(require 'etags);; find-tag-marker-ring
(require 'efc)
(require 'jdee-archive)
(require 'jdee-backend)
(require 'jdee-classpath)
(require 'jdee-files)
(require 'jdee-complete);; jdee-complete-private
(require 'jdee-import);; jdee-import-get-import
(require 'jdee-parse)
(require 'jdee-util)
(require 'semantic/senator)
(defcustom jdee-open-class-at-point-find-file-function 'find-file-other-window
"Define the function for opening the class at point. See
`jdee-open-class-at-point'`. Default is `find-file-other-window'. A function
defined here must have the same signature as `find-file' means the first
argument is the filename and the second optional argument is a
wildcard-pattern."
:group 'jdee-project
:type '(function :tag "Function to open class at point"))
(defvar jdee-open-cap-ff-function-temp-override nil
"Maybe some tools needs to temporally override the value of
`jdee-open-class-at-point-find-file-function'. Cause of the auto. resetting
mechanism of JDE for defcustom-variables this is not possible with the
defcustom version. So, if you need to override the value of
`jdee-open-class-at-point-find-file-function' from within your elisp code you
can use the variable `jdee-open-cap-ff-function-temp-override'.
`jdee-open-class-at-point' checks first if this variable is not nil and uses
then this value. Only if this variable is nil it uses the value of
`jdee-open-class-at-point'!
This variable is NOT for user customizing, but only for use within elisp!")
(defmacro jdee-with-file-contents (file &rest body)
"If FILE exists and is readable creates a temporary buffer with the contents
of FILE, points to beginning of buffer, evaluates BODY and return the value of
the last form of BODY. If FILE does not exist or is not readable nil is
returned.
Note: No major/minor-mode is activated and no local variables are evaluated
for FILE, but proper EOL-conversion and charcater interpretation is done!"
(let ((exp-filename (make-symbol "exp-filename")))
`(let ((,exp-filename (expand-file-name ,file)))
(if (and (file-exists-p ,exp-filename)
(file-readable-p ,exp-filename))
(with-temp-buffer
(insert-file-contents ,exp-filename)
(goto-char (point-min))
,@body)
nil))))
(defun jdee-open-get-class-to-open (pair parsed-symbol)
"Evaluates PARSE-SYMBOL to check if it is a variable name or a class name.
If this fails point is on a method or an attribute of a class in the current
buffer or in a superclass. In this cases we check first if the parsed-symbol
is a possible member of the current class(\"this\") and if this fails it
checks if it is a member of the base class(\"super\")."
(cond ((and (stringp (car pair))
(> (length (car pair)) 0))
;; if we got a pair all should work fine.
(concat (car pair) "." (cadr pair)))
((condition-case ()
(jdee-parse-eval-type-of parsed-symbol)
(error nil)) )
((jdee-parse-find-completion-for-pair
`("this" ,parsed-symbol) nil jdee-complete-private)
(jdee-parse-eval-type-of "this"))
((jdee-parse-find-completion-for-pair
`("super" ,parsed-symbol) nil jdee-complete-private)
(jdee-parse-eval-type-of "super"))))
(defun jdee-open-functions-exist ()
"Checks if the functions `jdee-parse-java-variable-at-point',
`jdee-parse-eval-type-of', and `jdee-parse-find-completion-for-pair' are defined"
(and (fboundp 'jdee-parse-java-variable-at-point)
(fboundp 'jdee-parse-eval-type-of)
(fboundp 'jdee-parse-find-completion-for-pair)))
(defun jdee-open-jump-to-class (parsed-symbol class-name)
"Place the cursor in the parsed variable"
(let* (tags super-class (first-time t))
;; Searching only for the symbol '{' is not good enough. We can
;; have, for example, '{@link }' in the javadoc before the class
;; definition.
(search-forward-regexp "^[^\\*]*?{" nil t)
(setq tags (jdee-get-parents))
(setq super-class (car tags))
(message "Superclass of %s is %s" class-name super-class)
;; Now let´s jump to the thing-of-interest. If this is a
;; variable-name then we will not find this with senator in
;; the opened java-file so we search for the definiton of
;; the class itself. This feature is only available if we
;; have senator!
(when (and (fboundp 'senator-search-forward) (not (string= parsed-symbol "")))
(goto-char (point-min))
(semantic-fetch-tags)
(setq parsed-symbol (concat "\\b" parsed-symbol "\\b"))
(while (not (senator-re-search-forward parsed-symbol nil t))
(message "Could not find %s in %s" parsed-symbol (buffer-name))
;; searching for the thing-of-interest has failed
;; let's try in the base class
(progn
(if (not super-class)
(error "Method not found"))
(let ((jdee-open-cap-ff-function-temp-override 'find-file))
(jdee-show-superclass-source-2 tags))
(goto-char (point-min))
(search-forward-regexp "^[^\\*]*?{" nil t)
(setq tags (jdee-get-parents))
;;if it is the first time try in the class definition
;;itself.
(if first-time
(progn
(setq first-time nil)
(senator-re-search-forward
(progn
(string-match ".*\\.\\([^.]+\\)$"
(concat "." class-name))
(match-string 1 (concat "." class-name)))
nil t)))
(setq super-class (car tags)))))))
(defun jdee-get-parents ()
"Returns a list with all the parents (super class and interfaces,
if any) of the current class or interface."
(jdee-remove-type
(append (semantic-tag-type-superclasses
(semantic-current-tag-of-class 'type))
(semantic-tag-type-interfaces (semantic-current-tag-of-class
'type)))))
(defun jdee-remove-type (list)
"Removes generics '<Type>' declaration from every given
class/interface name."
(mapcar #'(lambda(s) (replace-regexp-in-string "<.*>" "" s)) list))
(defun jdee-open-class-at-event (event)
"Like `jdee-open-class-at-point', but is mouse-bindable.
Preserves point."
(interactive "e")
(jdee-open-class-at-point
(posn-point (event-end event))))
(defun jdee-open-class-at-point (&optional position)
"Opens the source file that defines the class of the symbol at point and
scrolls the source file to the definition of the symbol, which can be the name of
a variable, class, method, or attribute. This function has the
same requirements as the JDEE's field/method completion commands. See, for example,
`jdee-complete-menu'. The JDEE searches for the source file first in
`jdee-sourcepath', then in `jdee-global-classpath', then in
$CLASSPATH, then in the current directory."
(interactive)
(if (jdee-open-functions-exist)
(let* ((old-point (if position
(prog1
(point)
(goto-char position))))
(thing-of-interest (thing-at-point 'symbol))
(pair (save-excursion
(end-of-thing 'symbol)
(jdee-parse-java-variable-at-point)))
(class-to-open (jdee-open-get-class-to-open
pair thing-of-interest)))
(if old-point
(goto-char old-point))
(if (and class-to-open
(stringp class-to-open))
;; Handle the case where the definition of the symbol is in the current buffer.
(let ((pos
(and
(string= (car pair) "")
(jdee-parse-find-declaration-of thing-of-interest))))
(ring-insert find-tag-marker-ring (point-marker))
(if pos
(goto-char pos)
;; Handle the case where the definition is in another buffer or an
;; unopened source file.
(let ((source
(jdee-find-class-source-file class-to-open)))
(if source
;; we have found the source file. So let´s open it and
;; then jump to the thing-of-interest
(progn
(if (typep source 'buffer)
(let ((pop-up-frames t))
(set-buffer source)
(display-buffer source)
;; (jdee-mode)
;; (semantic-new-buffer-fcn)
;; (semantic-fetch-tags)
)
;; (switch-to-buffer source)
;; (pop-to-buffer source other-window)
;; if the current buffer contains java-file-name do not try to
;; open the file
(if (not (string-equal (buffer-file-name) source))
(funcall (or jdee-open-cap-ff-function-temp-override
jdee-open-class-at-point-find-file-function)
source)))
(jdee-open-jump-to-class thing-of-interest class-to-open))
(message "Can not find the source for \"%s\"." class-to-open)))))
(message "Cannot determine the class of \"%s\"." thing-of-interest)))
(message "You need JDEE >= 2.2.6 and Senator to use this command.")))
(defun jdee-open-class-source ( &optional unqual-class )
"Displays source of the class whose name appears at point in the current
Java buffer. This command finds only classes that reside in the source paths
specified by `jdee-sourcepath'. You should provide a global setting
for this variable in your .emacs file to accommodate source files that are
not associated with any project."
(interactive)
(condition-case err
(let* ((unqualified-name
(or unqual-class
(read-from-minibuffer "Class: " (thing-at-point 'symbol))))
(class-names
(jdee-backend-get-qualified-name unqualified-name)))
;;Check return value of QualifiedName
(if (or (eq class-names nil)
(not (listp class-names)))
(error "Cannot find %s" unqualified-name))
(ring-insert find-tag-marker-ring (point-marker))
;; Turn off switching project settings to avoid
;; resetting jdee-sourcepath.
(let ((old-value jdee-project-context-switching-enabled-p))
(setq jdee-project-context-switching-enabled-p nil)
;;If the list is only one long
(if (eq 1 (length class-names))
;;then show it
(progn(other-window 1)
(jdee-find-class-source (car class-names)))
;;else let the user choose
(let ((class (efc-query-options class-names "Which class?")))
(if class
(jdee-find-class-source class))))
(setq jdee-project-context-switching-enabled-p old-value)))
(error
(message "%s" (error-message-string err)))))
(defalias 'jdee-show-class-source 'jdee-open-class-source)
;; Thanks to Sandip Chitale <[email protected]>
(defun jdee-show-superclass-source-2 (tags)
(if tags
(if (= (length tags) 1)
(jdee-show-class-source (car tags))
(let ((parent (efc-query-options tags "Which super class?")))
(if parent
(jdee-show-class-source parent))))
(error "Superclass not available")))
(defun jdee-show-superclass-source ()
"Show the source for the parent of the class at point."
(interactive)
(let ((tags (semantic-tag-type-superclasses
(semantic-current-tag-of-class 'type))))
(jdee-show-superclass-source-2 tags)))
;; Thanks to Sandip Chitale <[email protected]>
(defun jdee-show-interface-source ()
"Show the source for the interface implemented by the class at point.
If the class implements more than one interface, this command prompts
you to select one of the interfaces to show."
(interactive)
(let ((tags (semantic-tag-type-interfaces
(semantic-current-tag-of-class 'type))))
(if tags
(if (= (length tags) 1)
(jdee-show-class-source (car tags))
(let ((interface (efc-query-options tags "Which interface?")))
(if interface
(jdee-show-class-source interface)))))))
(defvar jdee-open-source-archive nil
"Will be set to the name of the archive (jar, zip, etc) that is
the real source of a buffer. See `jdee-open-source-resource'
and `jdee-find-class-source-file'.")
(defvar jdee-open-source-resource nil
"Will be set to the resource path within
`jdee-open-source-archive' that is the real source of the
buffer.")
;; (defun jdee-open-source-find-file (marker filename directory &rest formats)
;; "See if there is a buffer matching FILENAME that was opened via
;; `jdee-find-class-source-file'. Return that buffer or nil.
;; This function is designed as :before-until advice for
;; `compilation-find-file'.
;; "
;; ;; FIXME: If the FILENAME looks like <path to archive>:<path to source>,
;; ;; try and open it
;; (let ((buffer (get-file-buffer filename)))
;; (when (and buffer
;; (with-current-buffer buffer
;; jdee-open-source-archive))
;; buffer)))
;;
;; Add support for finding files in archives.
;;
;;(advice-add 'compilation-find-file :before-until #'jdee-open-source-find-file)
(defadvice compilation-find-file (around jdee-open-source-find-file activate)
"See if there is a buffer matching FILENAME that was opened via
`jdee-find-class-source-file'. Return that buffer or nil.
This function is designed as :before-until advice for
`compilation-find-file'.
"
;; FIXME: If the FILENAME looks like <path to archive>:<path to source>,
;; try and open it
(let ((buffer (get-file-buffer filename)))
(if (and buffer (with-current-buffer buffer jdee-open-source-archive))
buffer
ad-do-it)))
;; (defun jdee-open-source-find-file-of-fqn (fn marker filename directory &rest formats)
;; "Check if FILENAME matches an FQN and load it. Return that buffer or pass onto FN.
;; This function is designed as :around advice for
;; `compilation-find-file'.
;; "
;; (if (string-match (format "^%s$" (jdee-parse-java-fqn-re)) filename)
;; (let ((path (jdee-find-class-source-file filename)))
;; (cond
;; ((bufferp path) path)
;; ((stringp path) (apply fn marker path directory formats))
;; (t (apply fn marker filename directory formats))))
;; (apply fn marker filename directory formats)))
;;(advice-add 'compilation-find-file :around #'jdee-open-source-find-file-of-fqn)
(defadvice compilation-find-file (around jdee-open-source-find-file-of-fqn activate)
"Check if FILENAME matches an FQN and load it. Return that buffer or pass onto FN.
This function is designed as :around advice for
`compilation-find-file'."
(if (string-match (format "^%s$" (jdee-parse-java-fqn-re)) filename)
(let ((path (jdee-find-class-source-file filename)))
(cond
((bufferp path) path)
((stringp path)
(progn
(ad-set-arg 1 path)
ad-do-it))
(t ad-do-it)))
ad-do-it))
(defun jdee--open-source-set-archive-buffer-properties (path class-file-name)
(jdee-mode)
(set (make-local-variable 'jdee-open-source-archive) path)
(set (make-local-variable 'jdee-open-source-resource) class-file-name)
(goto-char (point-min))
(setq buffer-undo-list nil)
(setq buffer-saved-size (buffer-size))
(set-buffer-modified-p nil)
(setq buffer-read-only t))
(defun jdee--open-class-from-archive-into-buffer (path bufname)
"Open class source from archive given in `PATH' into buffer `BUFNAME'."
(let* ((pkg-path (subst-char-in-string ?. ?/ package))
(class-file-name (concat pkg-path "/" file))
success)
(setq buffer (get-buffer-create bufname))
(with-current-buffer buffer
(setq buffer-file-name (expand-file-name (concat path ":" class-file-name)))
(setq buffer-file-truename file)
(let ((exit-status
(archive-extract-by-stdout path class-file-name archive-zip-extract)))
(if (and (numberp exit-status) (= exit-status 0))
(progn
(jdee--open-source-set-archive-buffer-properties path class-file-name)
(throw 'found buffer))
(progn
(set-buffer-modified-p nil)
(kill-buffer buffer)))))))
(defun jdee--open-class-from-archive (path)
"Open class source from archive given in `PATH'."
(let* ((bufname (concat file " (" (file-name-nondirectory path) ")"))
(buffer (get-buffer bufname)))
(when buffer
(throw 'found buffer))
(jdee--open-class-from-archive-into-buffer path bufname)))
(defun jdee-find-class-source-file (class)
"Find the source file for a specified class.
CLASS is the fully qualified name of the class. This function searches
the directories and source file archives (i.e., jar or zip files)
specified by `jdee-sourcepath' for the source file corresponding to
CLASS. If it finds the source file in a directory, it returns the
file's path. If it finds the source file in an archive, it returns a
buffer containing the contents of the file. If this function does not
find the source for the class, it returns nil.
If CLASS is found in an archive, set both
`jdee-open-source-archive' and `jdee-open-source-resource' buffer local."
(let* ((outer-class (car (split-string class "[$]")))
(file (concat
(jdee-parse-get-unqualified-name outer-class)
".java"))
(package (jdee-parse-get-package-from-name outer-class)))
(catch 'found
(loop for path in (jdee-expand-wildcards-and-normalize jdee-sourcepath 'jdee-sourcepath) do
(let* ((pkg-path (subst-char-in-string ?. ?/ package))
(pkg-dir (expand-file-name pkg-path path))
(file-path (expand-file-name file pkg-dir)))
(cond
((jdee-archive-file-p path)
(jdee--open-class-from-archive path))
((file-exists-p (expand-file-name file path))
(throw 'found (expand-file-name file path)))
((file-exists-p file-path)
(throw 'found file-path))))))))
(defcustom jdee-preferred-packages
'("java.util" "java" "javax")
"Classes from these packages will appear first when reading from user input."
:group 'jdee-project
:type '(repeat string))
(defvar jdee-read-class-fq-items nil
"History for `jdee-choose-class'.")
(defun jdee-choose-class (classes &optional prompt uq-name confirm-fq-p)
"Choose a class from user input.
CLASSES are a list of fully qualified classes that are presetned to user as
choices for input.
PROMPT the prompt the user sees. Don't add the `: ' at the end to this.
UQ-NAME the unqualified name, which is used for the initial input if found as
an import in the buffer.
CONFIRM-FQ-P, if non-nil, confirm the class name even when there
is only one unique fully qualified class found for the simple
class name \(that is the class without the package part in the
name)."
(let ((sort-helper
(lambda (a b)
(dolist (pkg jdee-preferred-packages)
(let ((len (length pkg)))
(cond ((eq t (compare-strings pkg 0 len a 0 len))
(return t))
((eq t (compare-strings pkg 0 len b 0 len))
(return nil))
(t (string< a b))))))))
(setq classes (sort classes sort-helper))
(setq prompt (or prompt "Class"))
(let ((default (if uq-name
(jdee-import-get-import uq-name))))
(setq default (or default (car classes)))
(if (and (not confirm-fq-p) (= 1 (length classes)))
(car classes)
(efc-query-options classes prompt "Class" jdee-read-class-fq-items default)))))
(defvar jdee-read-class-items nil
"*History for `jdee-read-class' read items.")
(defvar jdee-read-class-fq-items nil
"*History for `jdee-read-class' read items (second part of fully
qualified classes).")
;;;###autoload
(defun jdee-read-class (&optional prompt fq-prompt
this-class-p confirm-fq-p no-confirm-nfq-p
validate-fn)
"Select a class interactively. PROMPT is used to prompt the user for the
first class name, FQ-PROMPT is used only if the class name expands into more
than one fully qualified name.
PROMPT text used to prompt the user for the simple class name, or
\"Class\" as the default. Don't add the colon/space at the end
of this prompt as a default will be added if it exists.
FQ-PROMPT text used to prompt the fully qualified class name, or
\"Select qualified class\" as the default. Don't add the
colon/space at the end of this prompt as a default will be added
if it exists.
THIS-CLASS-P, if non-nil, use the current class name if no class name at point
and we are in a JDEE buffer.
CONFIRM-FQ-P, if non-nil, confirm the class name even when there
is only one unique fully qualified class found for the simple
class name \(that is the class without the package part in the
name).
NO-CONFIRM-NFQ-P, if non-nil, don't confirm the class to check for fully
qualified classes if it is obtainable from either the point or this class (see
THIS-CLASS-P). If obtained from the point, then the class name is parsed with
`jdee-parse-class-name' for the input.
VALIDATE-FN is a function to validate the class. This function takes an
argument the output from `jdee-parse-class-name' given from the first user input
class name query. The function should raise an error if there is anything
wrong with the class. If this is `nil', then no validate is done.
When called interactively, select the class and copy it to the kill ring."
(interactive (list nil nil t))
(setq fq-prompt (or fq-prompt "Select qualified class"))
(let ((ctup (jdee-parse-class-name 'point))
uinput uq-name classes initial-input fqc default)
(if (and (null ctup)
(eq major-mode 'jdee-mode)
this-class-p)
(setq ctup
(jdee-parse-class-name (jdee-parse-get-buffer-class))))
(setq default (cond ((null ctup) nil)
((first ctup) (first ctup))
(t (third ctup))))
(setq prompt (concat (or prompt "Class")
(if default
(format " (default %s): " default)
": ")))
(setq uinput
(if (and default no-confirm-nfq-p)
(prog1
default
(setq fq-prompt (format "%s" fq-prompt)))
(read-string prompt nil 'jdee-read-class-items default)))
(setq ctup (jdee-parse-class-name uinput))
(if (null ctup)
(error "Doesn't appear to be a classname: `%s'" uinput))
(if validate-fn
(setq fqc (funcall validate-fn ctup)))
(when (not (eq 'pass fqc))
(setq fqc (first ctup)
uq-name (third ctup))
(if fqc
(if (not (jdee-backend-class-exists-p fqc))
(error "No match for %s" uq-name))
(setq classes (jdee-backend-get-qualified-name uq-name))
(if (= 0 (length classes))
(error "Not match for %s" uq-name))
(setq fqc (jdee-choose-class classes fq-prompt uq-name confirm-fq-p))))
(when (called-interactively-p 'interactive)
(kill-new fqc)
(message "Copied `%s'" fqc))
fqc))
;;;###autoload
(defun jdee-find-class-source (class &optional other-window)
"*Find the source file for a specified fully qualified class.
Calls `jdee-find-class-source-file' to do the search.
If it finds the source file, it opens the file in a buffer."
(interactive (list (jdee-read-class "Class") current-prefix-arg))
(let ((source (jdee-find-class-source-file class)))
(if source
(progn
(if (typep source 'buffer)
(switch-to-buffer source)
;; (pop-to-buffer source other-window)
(if (not (string-equal (buffer-file-name) source))
(if other-window
(find-file-other-window source)
(find-file source))))
(if (fboundp 'senator-re-search-forward)
(let ((inner-class-pos (string-match "\\$" class)))
(if inner-class-pos
(let ((inner-class (substring class (+ 1 inner-class-pos))))
(when inner-class
(goto-char (point-min))
(senator-re-search-forward
(concat "\\b" inner-class "\\b") nil t)))))))
(message "JDE error: Could not find source for \"%s\" in this
project's source path. See `jdee-sourcepath' for more information." class))))
(defun jdee-browse-class-at-point ()
"Displays the class of the object at point in the BeanShell Class
Browser. Point can be in a variable name, class name, method name, or field name).
This command has the same requirements to work as the field/method-completion
feature in JDEE (see `jdee-complete-at-point')."
(interactive)
(if (jdee-open-functions-exist)
(let* ((thing-of-interest (thing-at-point 'symbol))
(pair (save-excursion (end-of-thing 'symbol)
(jdee-parse-java-variable-at-point)))
(class-to-open (jdee-open-get-class-to-open
pair thing-of-interest)))
(if (and class-to-open (stringp class-to-open))
(jdee-backend-explore-class class-to-open)
(error "Can not parse the thing at point!")))
(message "You need JDEE >= 2.2.6 and Senator for using this feature!")))
(provide 'jdee-open-source)
;;; jdee-open-source.el ends here