Re: Script for creating Application Menu



Christopher Roy Bratusek <zanghar freenet de> writes:

> Am Montag, den 17.08.2009, 18:26 +0200 schrieb Christopher Roy Bratusek:
>> Am Sonntag, den 16.08.2009, 16:40 -0600 schrieb matthew:
>> > matthew <matth love gmail com> writes:
>> > 
>> > > Christopher Roy Bratusek <zanghar freenet de> writes:
>> > >
>> > >> Am Sonntag, den 16.08.2009, 20:18 +0200 schrieb Christopher Roy
>> > >> Bratusek:
>> > >>> Am Sonntag, den 16.08.2009, 20:09 +0200 schrieb Christopher Roy
>> > >>> Bratusek:
>> > >>> > Am Sonntag, den 16.08.2009, 10:18 -0600 schrieb matthew:
>> > >>> > > Hi.
>> > >>> > > I made a librep menu generator for .desktop files in a
>> > >>> > > /usr/share/applications directory, it is attached. 
>> > >>> > > Let me know if it works for any of you.
>> > >>> > > 
>> > >>> > > Put it in a load path and add the following into your .sawfishrc:
>> > >>> > > 
>> > >>> > > (require 'mk-saw-menu)
>> > >>> > > (write-saw-menu)
>> > >>> > > (require 'saw-menu)
>> > >>> > > (setq apps-menu saw-apps-menu)
>> > >>> > > 
>> > >>> > > You can make it an executable script by uncommenting the last line in the
>> > >>> > > mk-saw-menu.jl, i.e. (write-saw-menu) and adding the following to the
>> > >>> > > begining and chmoding the file to +x:
>> > >>> > > 
>> > >>> > > #! /bin/sh
>> > >>> > > exec rep --batch "$0" "$@"
>> > >>> > > !#
>> > >>> > > 
>> > >>> > > Cheers
>> > >>> > > 
>> > >>> > 
>> > >>> > Hi Matthew,
>> > >>> > 
>> > >>> > you're script is great and works like a charm.
>> > >>> > 
>> > >>> > I also did a few small changes:
>> > >>> > 
>> > >>> > - Added Settings Category (Otherwise System contains too many entries
>> > >>> > and becomes too huge to be usefull (at least on my box))
>> > >>> > - Re-orderd the Categories (System + Settings at the end for example)
>> > >>> > - Bumped version to 0.5.2 (just for fun)
>> > >>> > 
>> > >>> > Two thoughts:
>> > >>> > 
>> > >>> > Also I personally don't see the point for having both Network and
>> > >>> > Internet Submenus. And there should be an option for ignoring NoDisplay
>> > >>> > Setting if Categories=GNOME* , as GNOME hides several stuff by default
>> > >>> > from the menu (say: nautilus, bug-buddy, evince, eog, file-roller and
>> > >>> > more -- Ugly, I know)
>> > >>> > 
>> > >>> > If there are no complaints about it, I would integrate this into sawfish
>> > >>> > (as sawfish.wm.ext.fdo-menu) and then let the user choose wether to use
>> > >>> > the apps-menu (the updated version I've posted to the ML yesterday) or
>> > >>> > this new menu. But I guess It's o.k.
>> > >>> > 
>> > >>> > Updated script + Praising Screenshot (with mk-saw-menu + revamped
>> > >>> > rootmenu (see my yesterdays post)) attached.
>> > >>> > 
>> > >>> > Thanks a lot for your efforts,
>> > >>> > Chris
>> > >>> > ________________________________________________________________________
>> > >>> > Sawfish rules!
>> > >>> 
>> > >>> ... it does not check for duplicated entries, eg Synaptic twice in the
>> > >>> menu, but this is just a minor glitch, not a show-stopper :)
>> > >>> 
>> > >>> Thanks,
>> > >>> Chris
>> > >>> 
>> > >>
>> > >> Ah and it should sort entries alphabetically (I promise this was the
>> > >> last one :p)
>> > >>
>> > >> Thanks,
>> > >> Chris
>> > >>
>> > >> _______________________________________________________________________
>> > >> Sawfish rules!
>> > >
>> > > Thanks for the feedback and contributions.
>> > >
>> > > I have always felt the same way about Network and Internet menus, it may
>> > > be good to merge them perhaps.
>> > >
>> > > I added two new variables at the begining, 
>> > >
>> > > ignore-no-display
>> > > want-alphebetize
>> > >
>> > > Set these as 't to use; 
>> > > - ignore-no-display will, intuitively, ignore the
>> > > no-display entries in .desktop files, it is not tied to GNOME* entries,
>> > > but that can be changed in the future.  
>> > > - want-alphebetize, when set to 't, will alphebetize the resulting
>> > > entries inside the individual categories.
>> > > The alphebetize-entries function just does a simple (sort) using
>> > > 'string<, which doesn't ignore case, so capital letters will come
>> > > before lower-case ones, for some reason the 'string-lessp predicate
>> > > doesn't want to work for me...
>> > >
>> > > ignore-no-display is defaulted to '() - false in this file,
>> > > while want-alphebetize is defaulted to 't
>> > >
>> > > bumped to 0.5.3 :)
>> > >
>> > > Thanks again,
>> > 
>> > Made some updates, and a bug fix for the alph*a*betize function, which,
>> > other than being spelled wrong, cut off the "Settings" menu.
>> > 
>> > I also added a function that parses out the non-regular files from the
>> > .desktop file list, so it wont read ~ and # files anymore.
>> > 
>> > Also, began to add a function in (parse-cat-list) to do specific things
>> > for specific categories, etc.
>> > 
>> > Cheers
>> > 
>> > -- 
>> > Mattew Love
>> > 
>> 
>> ... the parse-directory-files function cuts off several *valid* files
>> from the list on my system. Also default-directory is already used in
>> sawfish i guess desktop-directory to be a better choice, also could you
>> make desktop-directory, ignore-no-display and want-alphabetize
>> changeable via .sawfishrc? I tried but somehow it failed, maybe just a
>> typo I oversaw, but well.
>> 
>> Thanks for your efforts,
>> Chris
>> 
>> 
>> _______________________________________________________________________
>> Sawfish rules!
>
> v0.5.5
>
> - (if (not (boundp'ed ignore-no-display (now changeable via rc)
> - (if (not (boundp'ed want-alphabetize (now changeable via rc)
> - temporarly disable parse-directory-files (not working correctly, as I
> said previously)
> - added "Emulator" to Game (oddly wine also has Emulator Category)
> - added "X-GNOME-PersonalSettings to Desktop
> - merged network into internet (as I don't see the point for both)
> - s/file is not a part of sawfish/file will be part of sawfish/
>
> I'm pretty sure where are more Categories outwhere we might have missed.
> Time will tell us, if so ;)
>
> Regards,
> Chris
>
>
> _______________________________________________________________________
> Sawfish rules!
>

v0.5.6

- Removed (parse-directory-files) for a number of reasons, i guess it is
  a bit pointless as well as destructive, so it is gone :)

- Changed the use of default-directory to desktop-directory, and
  consequently added a new line in (parse-desktop-file).

- Added the functions (desktop-file-p) and (backup-file-p) for possible
  future use, though, perhaps they too aren't necessary, as from my
  experience, the /usr/share/applications directory usually stays pretty
  tidy, though I guess we can't count on that :)

- Began adding in some comments for the various functions.

- Re-arranged the functions.

- Moved "X-Xfce-Toplevel" to "Desktop" from "Settings"

;-*-sawfish-*-
;; (mk-saw-menu.jl (v0.5.6)--- sawfish wm menu generation -- librep)

;; This file will be part of Sawfish

;;; Description:
;;
;; Create a sawfish wm menu from .desktop files
;; in your /usr/share/applications folder.

;;; Usage:
;;
;; Make sure the mk-saw-menu.jl is in your load path (i.e. ~/.sawfish/lisp)
;; in your .sawfishrc file add:
;; (setq my-term-string "xterm -e ") ;; change xterm -e to your appropriate \
;; string (should have a space at the end), you can leave this out if you use \
;; xterm, as that is the default.
;; (require 'mk-saw-menu)
;; (write-saw-menu)
;; (require 'saw-menu)
;; (setq apps-menu saw-apps-menu)
;; Doing the above will update the menu each time sawfish is started

;;; TODO:

;; adhere to the desktop entry file specifications.
;; http://standards.freedesktop.org/desktop-entry-spec/latest/
;; add support for field codes: http://standards.freedesktop.org/desktop-entry-spec/latest/ar01s06.html

;;; Code:

;; Some defaults
(setq desktop-directory "/usr/share/applications"
      desk-files (directory-files desktop-directory)
      my-name ()
      my-disp ()
      my-term ()
      my-exec ())

;; Variables that can be set in .sawfishrc
(if (not (boundp 'ignore-no-display))
    (setq ignore-no-display '()))

(if (not (boundp 'want-alphabetize))
    (setq want-alphabetize 't))

(if (not (boundp 'my-term-string))
    (setq my-term-string "xterm -e "))

;; The Master Category List
(setq menu-cat-alist
      '(("Desktop" .  ("X-Desktop" "X-DesktopApplets" "X-DesktopCountry" \
		       "DesktopSettings" "GNOME" "KDE" "X-GNOME-PersonalSettings" \
		       "X-Xfce-Toplevel"))
	("Personal" . ("X-Personal" "X-PersonalUtility" "Calendar" "ContactManagement"))
	("Office" . ("Office" "WordProcessor" "Presentation" "X-Document" \
		     "TextEditor" "SpreadSheet" "Calculator" "X-Calculate" \
		     "Chart" "FlowChart" "Finance"))
	("Internet" . ("Telephony" "Network" "Dialup" "VideoConference" \
		      "RemoteAccess" "News" "HamRadio" "FileTransfer" \
		      "X-Internet" "P2P" "Email" "WebBrowser" "IRCClient" "Chat" \
		      "InstantMessaging" "Chat" "WebDevelopment"))
	("Games" . ("Game" "ActionGame" "AdventureGame" "ArcadeGame" "BoardGame" "Emulator"\
		    "BlocksGame" "CardGame" "KidsGame" "LogicGame" "RolePlaying" "Simulation"))
	("Graphics" . ("RasterGraphics" "VectorGraphics" "X-GraphicUtility" \
		       "2DGraphics" "3dGraphics" "3DGraphics" "Scanning" "OCR" "Photography" \
		       "Viewer" "Publishing" "Art" "ImageProcessing"))
	("Media" . ("AudioVideo" "Audio", "Video" "Midi" "Mixer" "Sequencer" "Tuner" \
		    "TV" "AudioVideoEditing" "Player" "Recorder" "DiscBurning" "Music"))
	("Science" . ("Science" "Astrology" "ArtificialIntelligence" "Astronomy" \
		      "Biology" "Chemistry" "ComputerScience" "DataVisualization" \
		      "Electricity" "Robotics" "Physics" "Math" "Education" "Geography"))
	("Development" . ("GUIDesigner" "IDE" "Profiling" "RevisionControl" \
			  "ProjectManagement" "Translation" "GTK" "Development" \
			  "Qt" "Development" "Documentation"))
	("Utility" . ("X-SystemMemory" "Security" "Utility" \
		      "X-SetupEntry" "X-SetupUtility" "X-SystemMemory" \
		      "TextTools" "TelephonyTools" "Accessibility" "Clock" \
		      "ConsoleOnly"))
	("Filesystem" .  ("X-FileSystemFind" "X-FileSystemUtility" "Archiving" \
			  "FileManager" "X-FileSystemMount" "Compression"))
	("System" . ("X-SystemSchedule" "System" "X-SystemMemory" \
		     "TerminalEmulator" "Dictionary" "Puppy" "Printing" "Monitor" "Security"))
	("Settings" . ("Settings" "HardwareSettings" "PackageManager"))))

;; Some file-verifiers
(defun desktop-file-p (directory-file)
  (let ((this-file (open-file directory-file 'read)))
    (let ((this-line (read-line this-file)))
      (if (string= (substring this-line 0 15) "[Desktop Entry]")
	  't
	'()))))

(defun backup-file-p (input-file-name)
  (if (or (string= "~" (substring input-file-name (- (length input-file-name) 1)))
	  (string= "#" (substring input-file-name 0 1)))
      't
    '()))

;; Parse a .desktop file into a list suitable for a menu
;; ex. (parse-desktop-file "emacs.desktop") ==> ("Development" "Emacs" (system "emacs &"))
(defun parse-desktop-file (desktop-file)
  (let ((desktop-file-name (expand-file-name desktop-file desktop-directory)))
    desktop-file-name
    (if (and (file-exists-p desktop-file-name) 
	     (not (file-directory-p desktop-file-name)))
	     ;(not (backup-file-p desktop-file-name))
	     ;(file-regular-p desktop-file-name))
	(let ((new-file (open-file desktop-file-name 'read)))
	  (while (setq file-line (read-line new-file))
	    (create-menu file-line))
	  (if ignore-no-display
	      (setq my-disp '()))
	  (if (not (string= my-disp "true"))
	      (if (not (string= (string-downcase my-term) "true"))
		  (cons my-cat (cons my-name (cons (list 'system my-exec) nil)))
		(cons my-cat (cons my-name (cons (list 'system (concat my-term-string my-exec)) nil))))
	    (setq my-disp ()))))))

;; begining of parsing for (parse-desktop-file), feed a .desktop file line into this.
;; (create-menu "Name=Emacs ") ==> "Emacs"
;; (create-menu "Categories=Development;TextEditor;") ==> "Development"
(defun create-menu (line)
  (cond
   ((parse-desk-line line "Categories=")
    (setq my-cat (parse-desk-line line "Categories=")))
   ((parse-desk-line line "Name=")
    (setq my-name (parse-desk-line line "Name=")))
   ((parse-desk-line line "Exec=")
    (setq my-exec (concat (parse-desk-line line "Exec=") " &")))
   ((parse-desk-line line "Terminal=")
    (setq my-term (parse-desk-line line "Terminal=")))
   ((parse-desk-line line "NoDisplay=")
    (setq my-disp (parse-desk-line line "NoDisplay=")))))

;; Helper function for (create-menu), which will parse the string input there.
;; Will only give an output for specified lines (i.e. category, name, etc.)
(defun parse-desk-line (line desk-value)
  (let ((line-len (length line)))
    (cond
     ; this section is for the exec string
     ((and (> line-len 5) (string= desk-value (substring line 0 5)))
      (if (string= (aref line (- line-len 3)) 37)
	  (substring line 5 (- line-len 4))
	(substring line 5 (- line-len 1))))

     ; this section is for the category string
     ((and (> line-len 10) (string= desk-value (substring line 0 11)))
      (let ((cat-string (parse-cat-list (build-cat-list (substring line 11)))))
	cat-string))

     ; this section is for the terminal string
     ((and (> line-len 8) (string= desk-value (substring line 0 9)))
      (substring line 9 (- line-len 1)))

     ; this section is for the nodisplay string
     ((and (> line-len 9) (string= desk-value (substring line 0 10)))
      (substring line 10 (- line-len 1))))))

;; Helper for (parse-desk-line) - specifically, for categories.
(defun build-cat-list (line) ; line must be excluding the \
					; categories= part -> (substring line 11)
  (if (> (length line) 1)
      (let ((line-len (length line))
	    (this-cat (prin1-to-string (read-from-string line))))
	(cons this-cat (build-cat-list (substring line (+ 1 (length this-cat))))))))

;; Helper for (parse-desk-line)
;; Determine best category to use... :|
(defun parse-cat-list (cat-list)
  (if (cdr cat-list)
      (let ((this-cat (car cat-list)))
	(if (or
	     (string= this-cat "GNOME")
	     (string= this-cat "GTK")
	     (string= this-cat "KDE")
	     (string= this-cat "X-XFCE")
	     (string= this-cat "Qt")
	     (string= this-cat "Application"))
	    (parse-cat-list (cdr cat-list))
	    ; to do specific things for the above entries (uncomment below, \
					; and comment out the above (parse-cat-list (cdr cat-list)):
	    ;(let ((dm-specific this-cat))
	    ;  (if (or (string= dm-specific "Application") 
		;      (string= dm-specific "Qt")
		;      (string= dm-specific "GTK"))
		;  (parse-cat-list (cdr cat-list))
		;"Settings"))
	  this-cat))
    (car cat-list)))

;; Second Part of process, after (parse-desktop-file) has run \
					; through and generated the *loc-menu*.
;; Will run through the Category alist and assign menu values accordingly, and \
					; will output the base of the menu, to \
					; be fed into (build-saw-menu)
(defun fix-cats (cat-list)
  (if cat-list
      (let ((cat-val (car (car cat-list)))
	    (c-list (fix-sub-cats (car cat-list) *loc-menu*)))
	(if (car c-list)
	    (cons (cons cat-val c-list) (fix-cats (cdr cat-list)))
	  (fix-cats (cdr cat-list))))))

;; Helper function for (fix-cats)
(defun fix-sub-cats (cat-list loc-list)
  (if cat-list
      (let ((cat-val (car cat-list)))
	(if (assoc cat-val loc-list)
	    (cons (cdr (assoc cat-val loc-list))
		  (fix-sub-cats cat-list (remove (assoc cat-val loc-list) loc-list)))
	  (fix-sub-cats (cdr cat-list) loc-list)))))

;; Format the menu for sawfish
(defun build-saw-menu (entry)
  `(defvar saw-apps-menu ',entry))

;; Alphabetize the entries in the category menus
(defun alphabetize-entries (saw-menu)
  (if saw-menu
      (cons (cons (car (car saw-menu)) (sort (cdr (car saw-menu)) string<)) (alphabetize-entries (cdr saw-menu)))))

;; Write the sawfish menu to file
(defun write-saw-menu ()
  (setq *loc-menu* '())
  ; generate a list (*loc-menu*) from a (parse-desktop-file) call on all the .desktop files
  (mapc (lambda (x)
	  (setq *loc-menu* (append *loc-menu* (list (parse-desktop-file x))))) desk-files)
  ; check if ~/.sawfish/lisp exists, and if not, create it
  (if (not (file-exists-p "~/.sawfish/lisp"))
      (if (file-exists-p "~/.sawfish")
	  (make-directory "~/.sawfish/lisp")
 	(lambda () (make-directory "~/.sawfish") (make-directory "~/.sawfish/lisp"))))
  (setq menu-file (open-file "~/.sawfish/lisp/saw-menu.jl" 'write))
  ; generate the sawfish compatible menu file from the *loc-menu* list
  (if want-alphabetize
      (prin1 (build-saw-menu (alphabetize-entries (fix-cats menu-cat-alist))) menu-file)
    (prin1 (build-saw-menu (fix-cats menu-cat-alist)) menu-file))
  (close-file menu-file))

;(write-saw-menu)

-- 
Matthew Love


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]