1 Introduction
2 Installation
3 First simple example
3.1 Example directory
4 Description
4.1 The .rkt file
4.2 The .rktd file
5 Usage ideas
6 License
Version: 5.3.0.2

Script Plugin for DrRacket

Laurent Orseau <laurent dot orseauREMOVEME at gmail dot com>

1 Introduction

The Script Plugin’s purpose is to make it easy to extend DrRacket with small Racket scripts that can be used in the definition (or interaction) window, or interact with the user.

Creating a new script is as easy as a click on a menu item. Each script is automatically added as an item to the Scripts menu, without needing to restart DrRacket. A keyboard shortcut can be assigned to a script (via the menu item). By default, a script takes as input the currently selected text, and outputs the replacement text. There is also direct access to some elements of DrRacket for advanced (though simplified since there is no need to create a dedicated plugin) scripting, like DrRacket’s frame and the definition or interaction editor.

2 Installation

To install, simply evaluate:

(require (planet orseau/script-plugin/tool))

Wait for the installation process to finish, and then restart DrRacket. You should now see a new Scripts menu.

3 First simple example

Click on the Scripts/New Script menu item, and enter Reverse for the script name. This creates and opens the files reverse.rkt and reverse.rktd in the script directory. Also, a new item automatically appears in the Scripts menu.

In the .rkt file, modify the item-callback function to the following:
(define (item-callback str)
  (list->string (reverse (string->list str))))

Then go to a new tab, type some text, select it, and click on Scripts/Reverse.

3.1 Example directory

There are several other examples in the examples directory.

To use an example, just copy both the .rkt and .rktd files of the same name from the example directory to the user’s script directory. It should then automatically appear in the Scripts menu.

To find the example directory, evaluate (once the plugin is installed):
(require racket planet/resolver)
(build-path (path-only
             (resolve-planet-path
              '(planet orseau/script-plugin/tool)))
            "examples")

The default location of the user’s scripts is in a sub-folder of (find-system-path 'home-path). The directory of the user scripts can be changed through DrRacket’s preferences (in Edit/Preferences/Scripts). Important: The scripts directory must have write-access for the user (which should be the case for the default settings).

(I will try to simplify this process in the future, I’m just not sure exactly what would be best.)

4 Description

This DrRacket plugin adds a Script menu to the main window. This menu has several items, followed by the (initially empty) list of active scripts.

The New Script item asks for a script name and creates 2 files:
  • a .rkt file, the script itself (with a sample program)

  • a .rktd file, the metadata of the script with the default values

These two files are automatically opened.

The script menu is rebuilt each time the user activates it, so that changes are taken into account as soon as possible.

4.1 The .rkt file

This is the script file. It must provide the item-callback function, as in the sample code. It is meant to be executable by itself, as a normal module, to ease the testing process.

(item-callback str)  (or/c string? (is-a?/c snip%) #f)
  str : string?
Returns the string meant to be inserted in place of the current selection, or at the cursor if there is no selection. If the returned value is not a string or a snip%, the selection is not modified (i.e., the file remains in a saved state if it was already saved).

This function also accepts (optional or mandatory) special keyword arguments (the exact signature is determined with procedure-keywords):

The name of the function can also be changed, but this requires to change it also in the functions entry of the .rktd file, and the function must be provided.

4.2 The .rktd file

This is the metadata file. It contains an association list that defines the configuration of the script.

Note: This file being a data file, it should almost never contain quotes. The quotes in the following definitions must thus not appear in the file.

Most options (label, shortcut, shortcut-prefix, help-string) are the same as for the menu-item% constructor. In particular, a keyboard shortcut can be assigned to an item.

Additionally, If the label is 'separator, then a separator is added in the menu.

There are some additional options:
  • functions : (or/c symbol? (listof (list/c symbol? string?))) = item-callback

    If a symbol, the name of the function to call (which must be provided), and must follow item-callback’s signature.

    If a list, each symbol is the name of a function, and each string is a label for that function. In this case, a sub-menu holding all these functions is created, and the label option is used as the parent menu name.

    Note that a sub-menu can be shared among scripts.

    Example:

    The following .rktd file will create a submenu named My Functions (with the letter F for keyboard access), containing 3 items, one for each function and its associated letter-accessor.
    ((label . "My &Functions")
     (functions . ((my-function1 "My Function #&1")
                   (my-function2 "My Function #&2")
                   (my-function3 "My Function #&3")
                   ))
     (output-to . message-box)
     )

    And the associated .rkt example file:
    #lang racket/base
     
    (provide my-function1)
    (define (my-function1 str) "1")
     
    (provide my-function2)
    (define (my-function2 str) "2")
     
    (provide my-function3)
    (define (my-function3 str) "3")
     
     

    Note: The label can also be 'separator.

  • output-to : (one-of/c 'selection 'new-tab 'message-box #f) = 'selection

    If 'selection, the output of the transform function replaces the selection in the current tab (or insert at the cursor if there is no selection). If 'new-tab, a new tab is created and the output of the script is written to it. If 'message-box, the output is displayed in a message-box. If #f, no output is generated.

  • active : boolean? = #t

    If set to #f, no menu item is generated for this dictionary.

Finally, one .rktd file can contain several such dictionaries (one after the other), which allows for multiple sub-menus and menu items and in a single script. This would have roughly the same effect as splitting such a script into several script, each one with its own .rktd file and its dictionary.

5 Usage ideas

Note that racket/gui can be used to ask the user for more information and more.

Remark: Code snippets should probably be of rare usage, as one should better take advantage of Racket’s wonderful macro system. In some cases however, snippets might be useful, e.g., to require your common module where all your usual macros and functions are defined, or for automatic comments.

6 License

Copyright (c) 2012 by Laurent Orseau <laurent.orseauREMOVEME@gmail.com>.

This package is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.

This program 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 Lesser General Public License for more details.

You can find a copy of the GNU Lesser General Public License at http://www.gnu.org/licenses/lgpl-3.0.html.