Index: includes/common.inc
===================================================================
RCS file: /cvs/drupal/drupal/includes/common.inc,v
retrieving revision 1.923
diff -u -r1.923 common.inc
--- includes/common.inc	17 Jun 2009 11:08:46 -0000	1.923
+++ includes/common.inc	17 Jun 2009 17:17:00 -0000
@@ -2822,6 +2822,94 @@
 }
 
 /**
+ * Adds multiple JavaScript files and CSS files for a JavaScript plugin.
+ *
+ * @param $plugin
+ *   The name of the plugin to add.
+ * @param ...
+ *   Additional arguments to pass to the call to hook_js_plugin_add().
+ * @return
+ *   The results from invoking hook_js_plugin_add().
+ * @see hook_js_plugin()
+ * @see hook_js_plugin_add()
+ * @see hook_js_plugin_alter()
+ */
+function drupal_add_plugin($plugin) {
+  $plugin = drupal_get_plugin($plugin);
+  if (!isset($plugin)) {
+    return NULL;
+  }
+
+  // Add all the JavaScript, stylesheets and dependent plugins.
+  foreach (array('js', 'css', 'plugin') as $type) {
+    if (isset($plugin[$type]) && is_array($plugin[$type])) {
+      foreach ($plugin[$type] as $data => $options) {
+        // If the value is not an array, it's a filename or plugin name and
+        // passed as the first (and only) argument.
+        if (!is_array($options)) {
+          $data = $options;
+          $options = NULL;
+        }
+        // When drupal_add_js with 'type' => 'setting' is called, the first
+        // parameter ($data) is an array. Arrays can't be keys in PHP, so we
+        // have to get $data from the value array.
+        if (is_numeric($data)) {
+          $data = $options['data'];
+          unset($options['data']);
+        }
+        call_user_func('drupal_add_' . $type, $data, $options);
+      }
+    }
+  }
+
+  // Invoke hook_js_plugin_add() with extra parameters to allow any additional
+  // actions.
+  $args = func_get_args();
+  array_unshift($args, 'js_plugin_add');
+  return call_user_func_array('module_invoke_all', $args);
+}
+
+/**
+ * Retrieves information from the JavaScript plugin registry.
+ *
+ * @param $plugin
+ *   If given, will provide the JavaScript plugin information for just the given
+ *   plugin name.
+ * @return
+ *   Returns all available JavaScript plugins, or the requested one in $plugin.
+ *   Otherwise, returns NULL.
+ * @see drupal_add_plugin()
+ */
+function drupal_get_plugin($plugin = NULL) {
+  $plugins = &drupal_static(__FUNCTION__);
+
+  if (!isset($plugins)) {
+    if ($cache = cache_get('js_plugin')) {
+      $plugins = $cache->data;
+    }
+    else {
+      // Invoke hook_js_plugin() to create the plugin registry.
+      $plugins = module_invoke_all('js_plugin');
+
+      // Invoke hook_js_plugin_alter() to allow modification of the
+      // registry. This is helpful to allow modules to update any plugins
+      // to later versions.
+      drupal_alter('js_plugin', $plugins);
+
+      // Save the registry to the cache.
+      cache_set('js_plugin', $plugins);
+    }
+  }
+
+  if (isset($plugin)) {
+    return isset($plugins[$plugin]) ? $plugins[$plugin] : NULL;
+  }
+  else {
+    return $plugins;
+  }
+}
+
+/**
  * Constructs an array of the defaults that are used for JavaScript items.
  *
  * @param $data
Index: includes/form.inc
===================================================================
RCS file: /cvs/drupal/drupal/includes/form.inc,v
retrieving revision 1.341
diff -u -r1.341 form.inc
--- includes/form.inc	15 Jun 2009 09:51:49 -0000	1.341
+++ includes/form.inc	17 Jun 2009 17:17:00 -0000
@@ -2000,7 +2000,7 @@
   // Adding the same javascript settings twice will cause a recursion error,
   // we avoid the problem by checking if the javascript has already been added.
   if ((isset($element['#ahah']['callback']) || isset($element['#ahah']['path'])) && isset($element['#ahah']['event']) && !isset($js_added[$element['#id']])) {
-    drupal_add_js('misc/jquery.form.js', array('weight' => JS_LIBRARY));
+    drupal_add_plugin('form');
     drupal_add_js('misc/ahah.js');
 
     $ahah_binding = array(
Index: modules/system/system.api.php
===================================================================
RCS file: /cvs/drupal/drupal/modules/system/system.api.php,v
retrieving revision 1.41
diff -u -r1.41 system.api.php
--- modules/system/system.api.php	8 Jun 2009 09:23:54 -0000	1.41
+++ modules/system/system.api.php	17 Jun 2009 17:17:00 -0000
@@ -183,6 +183,109 @@
 }
 
 /**
+ * Registers any JavaScript plugins associated with the module.
+ *
+ * @return
+ *   An array defining any JavaScript plugins associated with the module.
+ * @see system_js_plugin()
+ */
+function hook_js_plugin() {
+  // Plugin One.
+  $plugins['plugin-1'] = array(
+    'title' => 'Plugin One',
+    'project_page' => 'http://example.com/plugin-1',
+    'version' => 1.2,
+    'download' => 'http://example.com/plugin-1/plugin-1.js',
+    'js' => array(
+      // Items that are just strings are interpreted as stright files. 
+      drupal_get_path('module', 'my_module') . '/plugin-1.js',
+    ),
+    'css' => array(
+      // By providing an array, it becomes the $options parameter during the
+      // drupal_add_js/drupal_add_css call.
+      drupal_get_path('module', 'my_module') . '/plugin-2.css' => array(
+        'type' => 'file',
+        'media' => 'screen',
+      ),
+    ),
+  );
+  // Plugin Two.
+  $plugins['plugin-2'] = array(
+    'title' => 'Plugin Two',
+    'project_page' => 'http://example.com/plugin-2',
+    'version' => 1.2,
+    'download' => 'http://example.com/plugin-2/plugin-2.js',
+    'js' => array(
+      // JavaScript settings can be added by sending in an array.
+      array(
+        'type' => 'setting',
+        'data' => array('plugin-2' => TRUE),
+      ),
+    ),
+    // Any depending plugins can be added by using the "plugin" key.
+    'plugin' => array(
+      // Sending in a string will just add the depending plugin to the page.
+      'plugin-1',
+      // Sending in the plugin name along with an array of parameters will be
+      // sent in as additional arguments to drupal_add_plugin().
+      'plugin-3' => array(
+        'additional arguments',
+      )
+    ),
+  );
+  return $plugins;
+}
+
+/**
+ * Alters the JavaScript plugin registry.
+ *
+ * @param $plugins
+ *   The JavaScript plugin registry.
+ * @return
+ *   An array with the updated JavaScript plugin registry.
+ * @see hook_js_plugin()
+ */
+function hook_js_plugin_alter(&$plugins) {
+  // Update Plugin One to version 1.6.
+  if (isset($plugins['plugin-1'])) {
+    if ($plugins['plugin-1plugin-1']['version'] < 1.6) {
+      // The version number and CSS/JS files will change.
+      $plugins['plugin-1']['version'] = 1.6;
+      $plugins['plugin-1']['js'] = array(
+        drupal_get_path('module', 'plugin_1_update') . '/plugin-1.js',
+      );
+      $plugins['plugin-1']['css'] = array(
+        drupal_get_path('module', 'plugin_1_update') . '/plugin-1.css',
+      );
+    }
+  }
+  return $plugins;
+}
+
+/**
+ * A JavaScript plugin is being added to the page.
+ *
+ * This is helpful when special actions are to be taken when the plugin is
+ * added. Some plugins, for example, require special inline CSS, or settings
+ * to be added to the page.
+ *
+ * @param $plugin
+ *   The JavaScript plugin being added.
+ * @param ...
+ *   Any additional arguments passed in when calling drupal_add_plugin().
+ * @see drupal_add_plugin()
+ */
+function hook_js_plugin_add($plugin) {
+  // When Plugin One is added, also add some settings to the page.
+  if ($plugin == 'plugin-1') {
+    $settings = array(
+      'plugin-1' => variable_get('plugin-1', 'November 19th, 1978'),
+    );
+    drupal_add_js($settings, 'setting'); 
+  }
+}
+
+/**
  * Perform alterations before a page is rendered.
  *
  * Use this hook when you want to add, remove, or alter elements at the page
Index: modules/system/system.module
===================================================================
RCS file: /cvs/drupal/drupal/modules/system/system.module,v
retrieving revision 1.712
diff -u -r1.712 system.module
--- modules/system/system.module	16 Jun 2009 08:41:35 -0000	1.712
+++ modules/system/system.module	17 Jun 2009 17:17:00 -0000
@@ -789,6 +789,38 @@
 }
 
 /**
+ * Implementation of hook_js_plugin().
+ */
+function system_js_plugin() {
+  // jQuery Form Plugin.
+  $plugins['form'] = array(
+    'title' => 'jQuery Form Plugin',
+    'project_page' => 'http://malsup.com/jquery/form/',
+    'version' => 2.16,
+    'download' => 'http://github.com/malsup/form/tarball/e52ed46bff664431a51d9562803acbcfbbd2e3b4',
+    'js' => array(
+      'misc/jquery.form.js',
+    ),
+  );
+
+  // Farbtastic.
+  $plugins['farbtastic'] = array(
+    'title' => 'Farbtastic',
+    'project_page' => 'http://acko.net/dev/farbtastic',
+    'version' => 1.2,
+    'download' => 'http://acko.net/files/farbtastic_/farbtastic12.zip',
+    'js' => array(
+      'misc/farbtastic/farbtastic.js',
+    ),
+    'css' => array(
+      'misc/farbtastic/farbtastic.css',
+    ),
+  );
+
+  return $plugins;
+}
+
+/**
  * Retrieve a blocked IP address from the database.
  *
  * @param $iid integer
Index: modules/simpletest/tests/common.test
===================================================================
RCS file: /cvs/drupal/drupal/modules/simpletest/tests/common.test,v
retrieving revision 1.46
diff -u -r1.46 common.test
--- modules/simpletest/tests/common.test	12 Jun 2009 08:39:39 -0000	1.46
+++ modules/simpletest/tests/common.test	17 Jun 2009 17:17:00 -0000
@@ -443,8 +443,9 @@
     $this->preprocess_js = variable_get('preprocess_js', 0);
     variable_set('preprocess_js', 0);
 
-    // Reset drupal_add_js() before each test.
+    // Reset drupal_add_js() and the plugin registry before each test.
     drupal_static_reset('drupal_add_js');
+    drupal_static_reset('drupal_get_plugin');
   }
 
   function tearDown() {
@@ -541,6 +542,25 @@
   }
 
   /**
+   * Checks to make sure that JavaScript plugins registry is built correctly.
+   */
+  function testPluginRegistry() {
+    $plugins = drupal_get_plugin();
+    $this->assertTrue(isset($plugins['farbtastic']) && isset($plugins['form']), t('The JavaScript plugin registry is built correctly.'));
+  }
+
+  /**
+   * Adds a JavaScript plugin to the page and tests for both it and its CSS.
+   */
+  function testRenderPlugin() {
+    drupal_add_plugin('farbtastic');
+    $javascript = drupal_get_js();
+    $stylesheets = drupal_get_css();
+    $this->assertTrue(strpos($javascript, 'misc/farbtastic/farbtastic.js') > 0, t('JavaScript plugins are rendered to the page.'));
+    $this->assertTrue(strpos($stylesheets, 'misc/farbtastic/farbtastic.css') > 0, t('JavaScript plugins render their CSS to the page.'));
+  }
+
+  /**
    * Test adding a JavaScript file with a different weight.
    */
   function testDifferentWeight() {
Index: modules/color/color.module
===================================================================
RCS file: /cvs/drupal/drupal/modules/color/color.module,v
retrieving revision 1.59
diff -u -r1.59 color.module
--- modules/color/color.module	27 May 2009 18:33:55 -0000	1.59
+++ modules/color/color.module	17 Jun 2009 17:17:00 -0000
@@ -169,8 +169,7 @@
   $info = color_get_info($theme);
 
   // Add Farbtastic color picker.
-  drupal_add_css('misc/farbtastic/farbtastic.css', array('preprocess' => FALSE));
-  drupal_add_js('misc/farbtastic/farbtastic.js', array('weight' => JS_LIBRARY));
+  drupal_add_plugin('farbtastic');
 
   // Add custom CSS and JS.
   drupal_add_css($base . '/color.css', array('preprocess' => FALSE));
