效果圖:css
編輯按鈕是默認的list_editable屬性對應的插件(xadmin.plugins.editable)html
放大按鈕對應的是自定義插件。ajax
自定義按鈕源碼:django
xplugin.py(保證可以直接或者間接被adminx.py引用到便可)app
# -*- coding:utf-8 -*- import xadmin from xadmin.views import BaseAdminPlugin, ListAdminView from xadmin.views.edit import ModelFormAdminUtil from xadmin.util import label_for_field from django.utils.translation import ugettext as _ class CustomDetailPlugin(BaseAdminPlugin): custom_details={} def __init__(self, admin_view): super(CustomDetailPlugin, self).__init__(admin_view) self.editable_need_fields = {} def init_request(self, *args, **kwargs): active = bool(self.request.method == 'GET' and self.admin_view.has_view_permission() and self.custom_details) if active: self.model_form = self.get_model_view(ModelFormAdminUtil, self.model).form_obj return active def result_item(self, item, obj, field_name, row): if self.custom_details and item.field and (field_name in self.custom_details.keys()): pk = getattr(obj, obj._meta.pk.attname) field_label = label_for_field(field_name, obj, model_admin=self.admin_view, return_attr=False) item.wraps.insert(0, '<span class="editable-field">%s</span>') title=self.custom_details.get(field_name,{}).get('title',_(u"Details of %s") % field_label) default_load_url=self.admin_view.model_admin_url('patch', pk) + '?fields=' + field_name load_url = self.custom_details.get(field_name,{}).get('load_url',default_load_url) if load_url!=default_load_url: concator='?' if load_url.find('?')==-1 else '&' load_url=load_url+concator+'pk='+str(pk) item.btns.append(( '<a class="editable-handler" title="%s" data-editable-field="%s" data-editable-loadurl="%s">'+ '<i class="fa fa-search"></i></a>') % (title, field_name,load_url) ) if field_name not in self.editable_need_fields: self.editable_need_fields[field_name] = item.field return item # Media def get_media(self, media): if self.editable_need_fields: media = media + self.model_form.media + \ self.vendor( 'xadmin.plugin.editable.js', 'xadmin.widget.editable.css') return media xadmin.site.register_plugin(CustomDetailPlugin, ListAdminView)
基本抄寫了xadmin.plugins.editable的源碼,並作少許修改。this
1. 增長custom_details 字段 {字段名:{'title':自定義彈框標題, 'load_url':自定義彈框所加載的url地址},...}url
此url後續增長get參數: pk=n , 一塊兒被xadmin.plugin.editable.js作get方式的ajax讀取,並將讀取到的html賦值到彈框裏面。spa
所以自定義功能很大程度依賴於url對應的頁面的實現。prototype
2. 默認的編輯按鈕改爲了放大鏡(能夠提取爲屬性供adminx中的類自定義)插件
js/css都沿用原來的。並對 xadmin.plugin.editable.js作了少許閹割(註釋如下兩行代碼,避免自定義icon被恢復爲編輯圖標)。
Editpop.prototype.beforeToggle = function() { var $el = this.$element if(this.content == null){ var that = this //$el.find('>i').removeClass('fa fa-edit').addClass('fa-spinner fa-spin') $.ajax({ url: $el.data('editable-loadurl'), success: function(content){ //$el.find('>i').removeClass('fa-spinner fa-spin').addClass('fa fa-edit') that.content = content that.toggle() }, dataType: 'html' }) } else { this.toggle() } }
備註:
自定義彈框原理:
ajax get方式加載url: cust_details[字段名]['load_url']?pk=n(n爲對應模型實例的主鍵id)
將加載獲得的html顯示在彈出的窗體中。
應用示例
adminx.py
class SimCardPoolAdmin(object): #... custom_details={'card_in':{'title':u'xx明細', 'load_url':'detail2'}}
views.py (details2的試圖&菜單註冊)
# -*- coding:utf-8 -*- from xadmin.sites import site from xadmin.views.base import CommAdminView,csrf_protect_m from django.template.response import TemplateResponse from .models import * class CardPoolCardsView(CommAdminView): #base_template = 'xadmin/base_site.html' @csrf_protect_m def get(self, request, *args, **kwargs): pid = request.GET['pk'] pool_obj = SimCardPool.objects.get(id=pid) card_objs=SimCardPhy.objects.filter(pool_no=pool_obj.pool_no) data = [[{'x':x+1,'y':y+1,'v':0} for x in range(pool_obj.cols)] for y in range(pool_obj.rows)] for c in card_objs: data[c.row-1][c.col-1]['v']=1 import pdb #context = self.get_context() context={'cp_details':data, 'rows':range(1,pool_obj.rows+1), 'cols':range(1,pool_obj.cols+1)} #pdb.set_trace() return TemplateResponse(self.request, [ 'card_pool/cp_details.html' ], context, current_app=self.admin_site.name) site.register_view(r'^card_pool/simcardpool/detail2/$',CardPoolCardsView, name='cp_detail')
xadmin此版本貌似不能加載views.py,做爲破解在 __init__.py中增長了 import views一行,以下:
__init__.py
import views
相關模板
<html> <head> <style type="text/css"> .in222{ background-color: green; width: 14px;height: 14px; float:left; border:solid #add9c0; border-width:0px 1px 1px 0px; } .empty{ background-color: #eeeeff; border:1px #ff000 solid; width: 14px;height: 14px; float:left; border:solid #add9c0; border-width:0px 1px 1px 0px; } </style> </head> <body> <div style="border:solid 1px #add9c0; width:auto;text-align: center;display: inline-block;border-width:1px 0px 0px 1px;"> {% for row in cp_details %} <div style="clear:left; float:left;width:auto;"> {% for o in row %} {% if o.v %} <div class="in222" title="(第{{o.y}}行,第{{o.x}}列)已插卡"> </div> {% else %} <div class="empty" title="(第{{o.y}}行,第{{o.x}}列)未插卡" style="border:1px #ff000 solid"> </div> {% endif %} {% endfor %} </div> {% endfor %} </div> </body> </html>
最終效果:
轉載請著明來自:http://www.cnblogs.com/Tommy-Yu/p/5443127.html,謝謝!