From cb8f3ddf4e43ab786a2dc56ee392b322c8f2a33c Mon Sep 17 00:00:00 2001 From: Gabriel Hurley Date: Tue, 13 Mar 2012 20:35:23 -0700 Subject: [PATCH] Added IDs and identifiable classes to all action buttons. Fixes bug 953483. Change-Id: Ie6b858a9a595d024f71ca372a11b97a454b3b1e8 --- docs/source/index.rst | 2 +- docs/source/quickstart.rst | 6 +- docs/source/topics/branding.rst | 29 -------- docs/source/topics/customizing.rst | 72 +++++++++++++++++++ .../floating_ips/tables.py | 7 +- .../access_and_security/keypairs/tables.py | 4 +- .../security_groups/tables.py | 4 +- horizon/dashboards/nova/containers/tables.py | 9 +-- .../images_and_snapshots/images/tables.py | 4 +- .../volume_snapshots/tables.py | 1 - .../instances_and_volumes/instances/tables.py | 13 ++-- .../instances_and_volumes/volumes/tables.py | 9 ++- horizon/dashboards/syspanel/flavors/tables.py | 2 +- .../dashboards/syspanel/projects/tables.py | 6 +- horizon/dashboards/syspanel/quotas/tables.py | 3 - horizon/dashboards/syspanel/users/tables.py | 6 +- horizon/tables/actions.py | 28 +++++++- .../common/_data_table_table_actions.html | 4 +- horizon/usage/tables.py | 1 + horizon/utils/html.py | 6 +- 20 files changed, 148 insertions(+), 68 deletions(-) delete mode 100644 docs/source/topics/branding.rst create mode 100644 docs/source/topics/customizing.rst diff --git a/docs/source/index.rst b/docs/source/index.rst index ae0d6f368..49f0cde9e 100644 --- a/docs/source/index.rst +++ b/docs/source/index.rst @@ -41,7 +41,7 @@ How to use Horizon in your own projects. intro quickstart topics/deployment - topics/branding + topics/customizing Developer Docs ============== diff --git a/docs/source/quickstart.rst b/docs/source/quickstart.rst index df64abe8c..c744b52af 100644 --- a/docs/source/quickstart.rst +++ b/docs/source/quickstart.rst @@ -28,8 +28,10 @@ Alternately specify the listen IP and port:: Once the Horizon server is running point a web browser to http://localhost:8000 or to the IP and port the server is listening. -.. note:: The ``DevStack`` project (http://devstack.org/) can be used to install -an OpenStack development environment from scratch. +.. note:: + + The ``DevStack`` project (http://devstack.org/) can be used to install + an OpenStack development environment from scratch. .. note:: diff --git a/docs/source/topics/branding.rst b/docs/source/topics/branding.rst deleted file mode 100644 index a1ba83fff..000000000 --- a/docs/source/topics/branding.rst +++ /dev/null @@ -1,29 +0,0 @@ -============================== -Change the branding of Horizon -============================== - -Changing the Page Title -======================= - -The OpenStack Dashboard Page Title branding (i.e. "**OpenStack** Dashboard") -can be overwritten by adding the attribute ``SITE_BRANDING`` -to ``local_settings.py`` with the value being the desired name. - -The file ``local_settings.py`` can be found at the Horizon directory path of -``horizon/openstack-dashboard/local/local_settings.py``. - -Changing the Page Logo -======================= - -The OpenStack Logo is pulled in through ``style.css``:: - - #splash .modal { - background: #fff url(../images/logo.png) no-repeat center 35px; - - h1.brand a { - background: url(../images/logo.png) top left no-repeat; - -To override the OpenStack Logo image, replace the image at the directory path -``horizon/openstack-dashboard/dashboard/static/dashboard/images/logo.png``. - -The dimensions should be ``width: 108px, height: 121px``. diff --git a/docs/source/topics/customizing.rst b/docs/source/topics/customizing.rst new file mode 100644 index 000000000..7d377ddb0 --- /dev/null +++ b/docs/source/topics/customizing.rst @@ -0,0 +1,72 @@ +=================== +Customizing Horizon +=================== + +Changing the Site Title +======================= + +The OpenStack Dashboard Site Title branding (i.e. "**OpenStack** Dashboard") +can be overwritten by adding the attribute ``SITE_BRANDING`` +to ``local_settings.py`` with the value being the desired name. + +The file ``local_settings.py`` can be found at the Horizon directory path of +``horizon/openstack-dashboard/local/local_settings.py``. + +Changing the Logo +================= + +The OpenStack Logo is pulled in through ``style.css``:: + + #splash .modal { + background: #fff url(../images/logo.png) no-repeat center 35px; + + h1.brand a { + background: url(../images/logo.png) top left no-repeat; + +To override the OpenStack Logo image, replace the image at the directory path +``horizon/openstack-dashboard/dashboard/static/dashboard/images/logo.png``. + +The dimensions should be ``width: 108px, height: 121px``. + +Button Icons +============ + +Horizon provides hooks for customizing the look and feel of each class of +button on the site. The following classes are used to identify each type of +button: + +* Generic Classes + * btn-search + * btn-delete + * btn-upload + * btn-download + * btn-create + * btn-edit + * btn-list + * btn-copy + * btn-camera + * btn-stats + * btn-enable + * btn-disable + +* Floating IP-specific Classes + * btn-allocate + * btn-release + * btn-associate + * btn-disassociate + +* Instance-specific Classes + * btn-launch + * btn-terminate + * btn-reboot + * btn-pause + * btn-suspend + * btn-console + * btn-log + +* Volume-specific classes + * btn-detach + +Additionally, the site-wide default button classes can be configured by +setting ``ACTION_CSS_CLASSES`` to a tuple of the classes you wish to appear +on all action buttons in your ``local_settings.py`` file. diff --git a/horizon/dashboards/nova/access_and_security/floating_ips/tables.py b/horizon/dashboards/nova/access_and_security/floating_ips/tables.py index e050c5b60..74fe0664a 100644 --- a/horizon/dashboards/nova/access_and_security/floating_ips/tables.py +++ b/horizon/dashboards/nova/access_and_security/floating_ips/tables.py @@ -32,7 +32,7 @@ LOG = logging.getLogger(__name__) class AllocateIP(tables.LinkAction): name = "allocate" verbose_name = _("Allocate IP To Project") - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-allocate") url = "horizon:nova:access_and_security:floating_ips:allocate" def single(self, data_table, request, *args): @@ -45,7 +45,7 @@ class ReleaseIPs(tables.BatchAction): action_past = _("Released") data_type_singular = _("Floating IP") data_type_plural = _("Floating IPs") - classes = ('btn-danger',) + classes = ('btn-danger', 'btn-release') def action(self, request, obj_id): api.tenant_floating_ip_release(request, obj_id) @@ -55,7 +55,7 @@ class AssociateIP(tables.LinkAction): name = "associate" verbose_name = _("Associate IP") url = "horizon:nova:access_and_security:floating_ips:associate" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-associate") def allowed(self, request, fip): if fip.instance_id: @@ -66,6 +66,7 @@ class AssociateIP(tables.LinkAction): class DisassociateIP(tables.Action): name = "disassociate" verbose_name = _("Disassociate IP") + classes = ("btn-disassociate", "btn-danger") def allowed(self, request, fip): if fip.instance_id: diff --git a/horizon/dashboards/nova/access_and_security/keypairs/tables.py b/horizon/dashboards/nova/access_and_security/keypairs/tables.py index dcd5d1e12..54306ae65 100644 --- a/horizon/dashboards/nova/access_and_security/keypairs/tables.py +++ b/horizon/dashboards/nova/access_and_security/keypairs/tables.py @@ -37,14 +37,14 @@ class ImportKeyPair(tables.LinkAction): name = "import" verbose_name = _("Import Keypair") url = "horizon:nova:access_and_security:keypairs:import" - attrs = {"class": "ajax-modal btn"} + classes = ("ajax-modal", "btn-upload") class CreateKeyPair(tables.LinkAction): name = "create" verbose_name = _("Create Keypair") url = "horizon:nova:access_and_security:keypairs:create" - attrs = {"class": "ajax-modal btn"} + classes = ("ajax-modal", "btn-create") class KeypairsTable(tables.DataTable): diff --git a/horizon/dashboards/nova/access_and_security/security_groups/tables.py b/horizon/dashboards/nova/access_and_security/security_groups/tables.py index bf58e135d..eb3974cb9 100644 --- a/horizon/dashboards/nova/access_and_security/security_groups/tables.py +++ b/horizon/dashboards/nova/access_and_security/security_groups/tables.py @@ -43,14 +43,14 @@ class CreateGroup(tables.LinkAction): name = "create" verbose_name = _("Create Security Group") url = "horizon:nova:access_and_security:security_groups:create" - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-create") class EditRules(tables.LinkAction): name = "edit_rules" verbose_name = _("Edit Rules") url = "horizon:nova:access_and_security:security_groups:edit_rules" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-edit") class SecurityGroupsTable(tables.DataTable): diff --git a/horizon/dashboards/nova/containers/tables.py b/horizon/dashboards/nova/containers/tables.py index aa58f763c..049dd3e29 100644 --- a/horizon/dashboards/nova/containers/tables.py +++ b/horizon/dashboards/nova/containers/tables.py @@ -25,7 +25,6 @@ from django.utils import http from django.utils.translation import ugettext as _ from horizon import api -from horizon import exceptions from horizon import tables @@ -63,20 +62,21 @@ class CreateContainer(tables.LinkAction): name = "create" verbose_name = _("Create Container") url = "horizon:nova:containers:create" - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-create") class ListObjects(tables.LinkAction): name = "list_objects" verbose_name = _("List Objects") url = "horizon:nova:containers:object_index" + classes = ("btn-list",) class UploadObject(tables.LinkAction): name = "upload" verbose_name = _("Upload Object") url = "horizon:nova:containers:object_upload" - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-upload") def get_link_url(self, datum=None): # Usable for both the container and object tables @@ -130,7 +130,7 @@ class CopyObject(tables.LinkAction): name = "copy" verbose_name = _("Copy") url = "horizon:nova:containers:object_copy" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-copy") def get_link_url(self, obj): return reverse(self.url, args=(http.urlquote(obj.container.name), @@ -141,6 +141,7 @@ class DownloadObject(tables.LinkAction): name = "download" verbose_name = _("Download") url = "horizon:nova:containers:object_download" + classes = ("btn-download",) def get_link_url(self, obj): #assert False, obj.__dict__['_apiresource'].__dict__ diff --git a/horizon/dashboards/nova/images_and_snapshots/images/tables.py b/horizon/dashboards/nova/images_and_snapshots/images/tables.py index 529ffd1bb..343d24bf6 100644 --- a/horizon/dashboards/nova/images_and_snapshots/images/tables.py +++ b/horizon/dashboards/nova/images_and_snapshots/images/tables.py @@ -43,14 +43,14 @@ class LaunchImage(tables.LinkAction): name = "launch" verbose_name = _("Launch") url = "horizon:nova:images_and_snapshots:images:launch" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-launch") class EditImage(tables.LinkAction): name = "edit" verbose_name = _("Edit") url = "horizon:nova:images_and_snapshots:images:update" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-edit") def get_image_type(image): diff --git a/horizon/dashboards/nova/images_and_snapshots/volume_snapshots/tables.py b/horizon/dashboards/nova/images_and_snapshots/volume_snapshots/tables.py index 0a2ba15c4..c73b1ab44 100644 --- a/horizon/dashboards/nova/images_and_snapshots/volume_snapshots/tables.py +++ b/horizon/dashboards/nova/images_and_snapshots/volume_snapshots/tables.py @@ -29,7 +29,6 @@ LOG = logging.getLogger(__name__) class DeleteVolumeSnapshot(tables.DeleteAction): data_type_singular = _("Volume Snapshot") data_type_plural = _("Volume Snapshots") - classes = ('btn-danger',) def delete(self, request, obj_id): api.volume_snapshot_delete(request, obj_id) diff --git a/horizon/dashboards/nova/instances_and_volumes/instances/tables.py b/horizon/dashboards/nova/instances_and_volumes/instances/tables.py index baff28924..d178249c5 100644 --- a/horizon/dashboards/nova/instances_and_volumes/instances/tables.py +++ b/horizon/dashboards/nova/instances_and_volumes/instances/tables.py @@ -55,7 +55,7 @@ class TerminateInstance(tables.BatchAction): action_past = _("Terminated") data_type_singular = _("Instance") data_type_plural = _("Instances") - classes = ('btn-danger',) + classes = ('btn-danger', 'btn-terminate') def action(self, request, obj_id): api.server_delete(request, obj_id) @@ -67,7 +67,7 @@ class RebootInstance(tables.BatchAction): action_past = _("Rebooted") data_type_singular = _("Instance") data_type_plural = _("Instances") - classes = ('btn-danger',) + classes = ('btn-danger', 'btn-reboot') def allowed(self, request, instance=None): return instance.status in ACTIVE_STATES or instance.status == 'SHUTOFF' @@ -82,6 +82,7 @@ class TogglePause(tables.BatchAction): action_past = (_("Paused"), _("Unpaused")) data_type_singular = _("Instance") data_type_plural = _("Instances") + classes = ("btn-pause") def allowed(self, request, instance=None): self.paused = False @@ -107,6 +108,7 @@ class ToggleSuspend(tables.BatchAction): action_past = (_("Suspended"), _("Resumed")) data_type_singular = _("Instance") data_type_plural = _("Instances") + classes = ("btn-suspend") def allowed(self, request, instance=None): self.suspended = False @@ -130,20 +132,21 @@ class LaunchLink(tables.LinkAction): name = "launch" verbose_name = _("Launch Instance") url = "horizon:nova:images_and_snapshots:index" + classes = ("btn-launch",) class EditInstance(tables.LinkAction): name = "edit" verbose_name = _("Edit Instance") url = "horizon:nova:instances_and_volumes:instances:update" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-edit") class SnapshotLink(tables.LinkAction): name = "snapshot" verbose_name = _("Snapshot") url = "horizon:nova:images_and_snapshots:snapshots:create" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-camera") def allowed(self, request, instance=None): return instance.status in ACTIVE_STATES @@ -153,6 +156,7 @@ class ConsoleLink(tables.LinkAction): name = "console" verbose_name = _("VNC Console") url = "horizon:nova:instances_and_volumes:instances:vnc" + classes = ("btn-console",) def allowed(self, request, instance=None): return instance.status in ACTIVE_STATES @@ -162,6 +166,7 @@ class LogLink(tables.LinkAction): name = "log" verbose_name = _("View Log") url = "horizon:nova:instances_and_volumes:instances:console" + classes = ("btn-log",) def allowed(self, request, instance=None): return instance.status in ACTIVE_STATES diff --git a/horizon/dashboards/nova/instances_and_volumes/volumes/tables.py b/horizon/dashboards/nova/instances_and_volumes/volumes/tables.py index 4d5ea6ac9..5d0decb1f 100644 --- a/horizon/dashboards/nova/instances_and_volumes/volumes/tables.py +++ b/horizon/dashboards/nova/instances_and_volumes/volumes/tables.py @@ -34,7 +34,6 @@ DELETABLE_STATES = ("available", "error") class DeleteVolume(tables.DeleteAction): data_type_singular = _("Volume") data_type_plural = _("Volumes") - classes = ('btn-danger',) def delete(self, request, obj_id): api.volume_delete(request, obj_id) @@ -58,14 +57,14 @@ class CreateVolume(tables.LinkAction): name = "create" verbose_name = _("Create Volume") url = "%s:volumes:create" % URL_PREFIX - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-create") class EditAttachments(tables.LinkAction): name = "attachments" verbose_name = _("Edit Attachments") url = "%s:volumes:attach" % URL_PREFIX - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-edit") def allowed(self, request, volume=None): return volume.status in ("available", "in-use") @@ -75,7 +74,7 @@ class CreateSnapshot(tables.LinkAction): name = "snapshots" verbose_name = _("Create Snapshot") url = "%s:volumes:create_snapshot" % URL_PREFIX - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-camera") def allowed(self, request, volume=None): return volume.status == "available" @@ -154,7 +153,7 @@ class DetachVolume(tables.BatchAction): action_past = _("Detached") data_type_singular = _("Volume") data_type_plural = _("Volumes") - classes = ('btn-danger',) + classes = ('btn-danger', 'btn-detach') def action(self, request, obj_id): instance_id = self.table.get_object_by_id(obj_id)['serverId'] diff --git a/horizon/dashboards/syspanel/flavors/tables.py b/horizon/dashboards/syspanel/flavors/tables.py index 05e35e02d..3380725ad 100644 --- a/horizon/dashboards/syspanel/flavors/tables.py +++ b/horizon/dashboards/syspanel/flavors/tables.py @@ -21,7 +21,7 @@ class CreateFlavor(tables.LinkAction): name = "create" verbose_name = _("Create Flavor") url = "horizon:syspanel:flavors:create" - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-create") class FlavorsTable(tables.DataTable): diff --git a/horizon/dashboards/syspanel/projects/tables.py b/horizon/dashboards/syspanel/projects/tables.py index 598ed6e85..3b6b81fd3 100644 --- a/horizon/dashboards/syspanel/projects/tables.py +++ b/horizon/dashboards/syspanel/projects/tables.py @@ -16,26 +16,28 @@ class ModifyQuotasLink(tables.LinkAction): name = "quotas" verbose_name = _("Modify Quotas") url = "horizon:syspanel:projects:quotas" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-edit") class ViewMembersLink(tables.LinkAction): name = "users" verbose_name = _("Modify Users") url = "horizon:syspanel:projects:users" + classes = ("btn-download",) class UsageLink(tables.LinkAction): name = "usage" verbose_name = _("View Usage") url = "horizon:syspanel:projects:usage" + classes = ("btn-stats",) class EditLink(tables.LinkAction): name = "update" verbose_name = _("Edit Project") url = "horizon:syspanel:projects:update" - attrs = {"class": "ajax-modal"} + classes = ("ajax-modal", "btn-edit") class CreateLink(tables.LinkAction): diff --git a/horizon/dashboards/syspanel/quotas/tables.py b/horizon/dashboards/syspanel/quotas/tables.py index 122451501..18b52922f 100644 --- a/horizon/dashboards/syspanel/quotas/tables.py +++ b/horizon/dashboards/syspanel/quotas/tables.py @@ -1,10 +1,7 @@ import logging -from django import shortcuts -from django.contrib import messages from django.utils.translation import ugettext_lazy as _ -from horizon import api from horizon import tables diff --git a/horizon/dashboards/syspanel/users/tables.py b/horizon/dashboards/syspanel/users/tables.py index 41f05015e..4c813bb45 100644 --- a/horizon/dashboards/syspanel/users/tables.py +++ b/horizon/dashboards/syspanel/users/tables.py @@ -15,20 +15,21 @@ class CreateUserLink(tables.LinkAction): name = "create" verbose_name = _("Create User") url = "horizon:syspanel:users:create" - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-create") class EditUserLink(tables.LinkAction): name = "edit" verbose_name = _("Edit") url = "horizon:syspanel:users:update" - classes = ("ajax-modal",) + classes = ("ajax-modal", "btn-edit") class EnableUsersAction(tables.Action): name = "enable" verbose_name = _("Enable") verbose_name_plural = _("Enable Users") + classes = ("btn-enable",) def allowed(self, request, user): return not user.enabled @@ -57,6 +58,7 @@ class DisableUsersAction(tables.Action): name = "disable" verbose_name = _("Disable") verbose_name_plural = _("Disable Users") + classes = ("btn-disable",) def allowed(self, request, user): return user.enabled diff --git a/horizon/tables/actions.py b/horizon/tables/actions.py index 3ed40fd04..2988853a2 100644 --- a/horizon/tables/actions.py +++ b/horizon/tables/actions.py @@ -32,6 +32,7 @@ LOG = logging.getLogger(__name__) # For Bootstrap integration; can be overridden in settings. ACTION_CSS_CLASSES = ("btn", "btn-small") +STRING_SEPARATOR = "__" class BaseAction(html.HTMLElement): @@ -41,6 +42,10 @@ class BaseAction(html.HTMLElement): requires_input = False preempt = False + def __init__(self): + super(BaseAction, self).__init__() + self.id_counter = 0 + def allowed(self, request, datum): """ Determine whether this action is allowed for the current request. @@ -64,11 +69,21 @@ class BaseAction(html.HTMLElement): def get_default_classes(self): """ - Returns a list of the default classes for the tab. Defaults to + Returns a list of the default classes for the action. Defaults to ``["btn", "btn-small"]``. """ return getattr(settings, "ACTION_CSS_CLASSES", ACTION_CSS_CLASSES) + def get_default_attrs(self): + """ + Returns a list of the default HTML attributes for the action. Defaults + to returning an ``id`` attribute with the value + ``{{ table.name }}__action_{{ action.name }}__{{ creation counter }}``. + """ + bits = (self.table.name, "action_%s" % self.name, str(self.id_counter)) + self.id_counter += 1 + return {"id": STRING_SEPARATOR.join(bits)} + def __repr__(self): return "<%s: %s>" % (self.__class__.__name__, self.name) @@ -286,6 +301,11 @@ class FilterAction(BaseAction): """ return "__".join([self.table.name, self.name, self.param_name]) + def get_default_classes(self): + classes = super(FilterAction, self).get_default_classes() + classes += ("btn-search",) + return classes + def filter(self, table, data, filter_string): """ Provides the actual filtering logic. @@ -452,10 +472,14 @@ class DeleteAction(BatchAction): name = "delete" action_present = _("Delete") action_past = _("Deleted") - classes = ('btn-danger',) def action(self, request, obj_id): return self.delete(request, obj_id) def delete(self, request, obj_id): raise NotImplementedError("DeleteAction must define a delete method.") + + def get_default_classes(self): + classes = super(DeleteAction, self).get_default_classes() + classes += ("btn-danger", "btn-delete") + return classes diff --git a/horizon/templates/horizon/common/_data_table_table_actions.html b/horizon/templates/horizon/common/_data_table_table_actions.html index 0988f902f..169406fd2 100644 --- a/horizon/templates/horizon/common/_data_table_table_actions.html +++ b/horizon/templates/horizon/common/_data_table_table_actions.html @@ -2,13 +2,13 @@ {% if filter %} {% endif %} {% for action in table_actions %} {% if action != filter %} {% if action.method != "GET" %} - + {% else %} {{ action.verbose_name }} {% endif %} diff --git a/horizon/usage/tables.py b/horizon/usage/tables.py index b3305ab4a..802b8c2a7 100644 --- a/horizon/usage/tables.py +++ b/horizon/usage/tables.py @@ -8,6 +8,7 @@ from horizon.templatetags.sizeformat import mbformat class CSVSummary(tables.LinkAction): name = "csv_summary" verbose_name = _("Download CSV Summary") + classes = ("btn-download",) def get_link_url(self, usage=None): return self.table.kwargs['usage'].csv_link() diff --git a/horizon/utils/html.py b/horizon/utils/html.py index ee1b44029..72cf5df79 100644 --- a/horizon/utils/html.py +++ b/horizon/utils/html.py @@ -16,13 +16,17 @@ class HTMLElement(object): """ return [] + def get_default_attrs(self): + return {} + @property def attr_string(self): """ Returns a flattened string of HTML attributes based on the ``attrs`` dict provided to the class. """ - final_attrs = copy.copy(self.attrs) + final_attrs = copy.copy(self.get_default_attrs()) + final_attrs.update(self.attrs) # Handle css class concatenation default = " ".join(self.get_default_classes()) defined = self.attrs.get('class', '')