diff --git a/django/newforms/widgets.py b/django/newforms/widgets.py
index 5ec27653cf..b24f91f4c6 100644
--- a/django/newforms/widgets.py
+++ b/django/newforms/widgets.py
@@ -16,9 +16,9 @@ try:
except NameError:
from sets import Set as set # Python 2.3 fallback
-# Converts a dictionary to a single string with key="value", XML-style.
-# Assumes keys do not need to be XML-escaped.
-flatatt = lambda attrs: ' '.join(['%s="%s"' % (k, escape(v)) for k, v in attrs.items()])
+# Converts a dictionary to a single string with key="value", XML-style with
+# a leading space. Assumes keys do not need to be XML-escaped.
+flatatt = lambda attrs: ''.join([' %s="%s"' % (k, escape(v)) for k, v in attrs.items()])
class Widget(object):
requires_data_list = False # Determines whether render()'s 'value' argument should be a list.
@@ -28,16 +28,20 @@ class Widget(object):
def render(self, name, value):
raise NotImplementedError
+ def build_attrs(self, extra_attrs=None, **kwargs):
+ attrs = dict(self.attrs, **kwargs)
+ if extra_attrs:
+ attrs.update(extra_attrs)
+ return attrs
+
class Input(Widget):
"Base class for all widgets (except type='checkbox', which is special)"
input_type = None # Subclasses must define this.
def render(self, name, value, attrs=None):
if value is None: value = ''
- final_attrs = dict(self.attrs, type=self.input_type, name=name)
- if attrs:
- final_attrs.update(attrs)
+ final_attrs = self.build_attrs(attrs, type=self.input_type, name=name)
if value != '': final_attrs['value'] = value # Only add the 'value' attribute if a value is non-empty.
- return u'' % flatatt(final_attrs)
+ return u'' % flatatt(final_attrs)
class TextInput(Input):
input_type = 'text'
@@ -54,18 +58,14 @@ class FileInput(Input):
class Textarea(Widget):
def render(self, name, value, attrs=None):
if value is None: value = ''
- final_attrs = dict(self.attrs, name=name)
- if attrs:
- final_attrs.update(attrs)
- return u'' % (flatatt(final_attrs), escape(value))
+ final_attrs = self.build_attrs(attrs, name=name)
+ return u'' % (flatatt(final_attrs), escape(value))
class CheckboxInput(Widget):
def render(self, name, value, attrs=None):
- final_attrs = dict(self.attrs, type='checkbox', name=name)
- if attrs:
- final_attrs.update(attrs)
+ final_attrs = self.build_attrs(attrs, type='checkbox', name=name)
if value: final_attrs['checked'] = 'checked'
- return u'' % flatatt(final_attrs)
+ return u'' % flatatt(final_attrs)
class Select(Widget):
def __init__(self, attrs=None, choices=()):
@@ -75,10 +75,8 @@ class Select(Widget):
def render(self, name, value, attrs=None, choices=()):
if value is None: value = ''
- final_attrs = dict(self.attrs, name=name)
- if attrs:
- final_attrs.update(attrs)
- output = [u'