上一篇文章介紹了 Dashboard 的基本結構框架,那接下來的問題就是如何在這個框架中加入咱們本身想要的內容了。在真正動手以前,讓咱們先來看看官方的頁面是怎麼作出來的。首先咱們進入 /usr/share/openstack-dashboard/openstack_dashboard/dashboards/admin/networks 文件夾下面,能夠看到有這幾個文件和子文件夾:ajax
../networks:django
- __init__.pyapi
- ports/網絡
- subnets/框架
- templates/網站
- forms.pyui
- tables.pyurl
- panel.pyspa
- urls.py.net
- views.py
- tests.py
從上一篇文章咱們已經知道了 views.py 和 urls.py 是幹什麼的,那咱們就先來看看 panel.py 好了。
from django.utils.translation import ugettext_lazy as _ import horizon from openstack_dashboard.dashboards.admin import dashboard class Networks(horizon.Panel): name = _("Networks") slug = 'networks' permissions = ('openstack.services.network',) dashboard.Admin.register(Networks)
這個文件很簡單,最重要的是兩個參數: name 和 slug。 name 是這個panel 在網頁上顯示的名字,而 slug 則是這個 panel 的相似於 ID 的東西。所以,在咱們本身的 panel.py 中也能夠依葫蘆畫瓢,定義一個 PluginPanel 的類:
import horizon class PluginPanel(horizon.Panel): name = "MyPlugin" slug = 'plugin_panel'
注意這裏的 slug 要和在以前提到的 _50_admin_add_panel.py 中的 PANEL 變量的值相同。
接下來咱們再看看networks文件夾下面的 form.py:
import logging from django.core.urlresolvers import reverse from django.utils.translation import ugettext_lazy as _ from horizon import exceptions from horizon import forms from horizon import messages from openstack_dashboard import api LOG = logging.getLogger(__name__) class CreateNetwork(forms.SelfHandlingForm): name = forms.CharField(max_length=255, label=_("Name"), required=False) tenant_id = forms.ChoiceField(label=_("Project")) if api.neutron.is_port_profiles_supported(): net_profile_id = forms.ChoiceField(label=_("Network Profile")) admin_state = forms.BooleanField(label=_("Admin State"), initial=True, required=False) shared = forms.BooleanField(label=_("Shared"), initial=False, required=False) external = forms.BooleanField(label=_("External Network"), initial=False, required=False) ...... 這裏省略了一些其餘代碼 ......
def handle(self, request, data): try: params = {'name': data['name'], 'tenant_id': data['tenant_id'], 'admin_state_up': data['admin_state'], 'shared': data['shared'],} #'router:external': data['external']} if api.neutron.is_port_profiles_supported(): params['net_profile_id'] = data['net_profile_id'] network = api.neutron.network_create(request, **params) msg = _('Network %s was successfully created.') % data['name'] LOG.debug(msg) messages.success(request, msg) return network except Exception: redirect = reverse('horizon:admin:networks:index') msg = _('Failed to create network %s') % data['name'] exceptions.handle(request, msg, redirect=redirect) class UpdateNetwork(forms.SelfHandlingForm): name = forms.CharField(label=_("Name"), required=False) tenant_id = forms.CharField(widget=forms.HiddenInput) network_id = forms.CharField(label=_("ID"), widget=forms.TextInput( attrs={'readonly': 'readonly'})) admin_state = forms.BooleanField(label=_("Admin State"), required=False) shared = forms.BooleanField(label=_("Shared"), required=False) external = forms.BooleanField(label=_("External Network"), required=False) failure_url = 'horizon:admin:networks:index' def handle(self, request, data): try: params = {'name': data['name'], 'admin_state_up': data['admin_state'], 'shared': data['shared'], 'router:external': data['external']} network = api.neutron.network_update(request, data['network_id'], **params) msg = _('Network %s was successfully updated.') % data['name'] LOG.debug(msg) messages.success(request, msg) return network except Exception: msg = _('Failed to update network %s') % data['name'] LOG.info(msg) redirect = reverse(self.failure_url) exceptions.handle(request, msg, redirect=redirect)
能夠看到裏面有兩個類,一個 CreateNetwork , 一個 UpdateNetwork 。這兩個類其實很好理解,就是在網頁上你點擊 "Create a new network" 或者 "Update Network" 時會彈出來的一個對話框,類的屬性一一對應對話框中你要設置的參數。若是瞭解 Django 的人就會知道這個東西其實和 Django 裏面的 Form 是一回事情。 對於這兩個類須要額外注意的是分別須要定義一個 handle() 的方法,告訴 Horizon 怎麼處理表格提交的數據。那若是咱們也須要有建立或者更新咱們的 plugin resources 的話,能夠跟着這個文件畫瓢。
接下來咱們再看 tables.py:
import logging from django.core.urlresolvers import reverse from django.template import defaultfilters as filters from django.utils.translation import ugettext_lazy as _ from horizon import exceptions from horizon import tables from openstack_dashboard import api from openstack_dashboard.dashboards.project.networks \ import tables as project_tables LOG = logging.getLogger(__name__) class DeleteNetwork(tables.DeleteAction): data_type_singular = _("Network") data_type_plural = _("Networks") def delete(self, request, obj_id): try: api.neutron.network_delete(request, obj_id) except Exception: msg = _('Failed to delete network %s') % obj_id LOG.info(msg) redirect = reverse('horizon:admin:networks:index') exceptions.handle(request, msg, redirect=redirect) class CreateNetwork(tables.LinkAction): name = "create" verbose_name = _("Create Network") url = "horizon:admin:networks:create" classes = ("ajax-modal", "btn-create") class EditNetwork(tables.LinkAction): name = "update" verbose_name = _("Edit Network") url = "horizon:admin:networks:update" classes = ("ajax-modal", "btn-edit") #def _get_subnets(network): # cidrs = [subnet.get('cidr') for subnet in network.subnets] # return ','.join(cidrs) class NetworksTable(tables.DataTable): tenant = tables.Column("tenant_name", verbose_name=_("Project")) name = tables.Column("name", verbose_name=_("Network Name"), link='horizon:admin:networks:detail') subnets = tables.Column(project_tables.get_subnets, verbose_name=_("Subnets Associated"),) shared = tables.Column("shared", verbose_name=_("Shared"), filters=(filters.yesno, filters.capfirst)) status = tables.Column("status", verbose_name=_("Status")) admin_state = tables.Column("admin_state", verbose_name=_("Admin State")) class Meta: name = "networks" verbose_name = _("Networks") table_actions = (CreateNetwork, DeleteNetwork) row_actions = (EditNetwork, DeleteNetwork)
首先咱們看到最底下的 NetworksTable 類,這個類也很好理解,直接對應的就是你在主頁上 Network 下面會看到的一張表格。類的屬性就是表格的列。 Meta 就是一些額外信息。而後還有三個類分別對應建立、刪除和修改網絡的操做。你們能夠對照上面的代碼和下面的網頁顯示進一步理解代碼的做用。其中用藍色圈出來的就是個人 plugin panel 在 Admin 這個 Dashboard 中的顯示。
好了,Dashboard 的改造就介紹到這裏了。若是有什麼不清楚的地方或者我沒有講到的地方,一能夠參考 Django 的官方網站;二能夠直接打開一個其餘的 Dashboard 的文件夾看看,依葫蘆畫瓢;三也歡迎你們在底下直接提問。