You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

738 lines
30 KiB

6 months ago
import dash
import time
import uuid
from dash import dcc
from dash.dependencies import Input, Output, State, ALL
from dash.exceptions import PreventUpdate
import feffery_antd_components as fac
import feffery_utils_components as fuc
from server import app
from utils.common import validate_data_not_empty
from api.role import get_role_list_api, get_role_detail_api, add_role_api, edit_role_api, delete_role_api, export_role_list_api
from api.menu import get_menu_tree_api
@app.callback(
output=dict(
role_table_data=Output('role-list-table', 'data', allow_duplicate=True),
role_table_pagination=Output('role-list-table', 'pagination', allow_duplicate=True),
role_table_key=Output('role-list-table', 'key'),
role_table_selectedrowkeys=Output('role-list-table', 'selectedRowKeys'),
api_check_token_trigger=Output('api-check-token', 'data', allow_duplicate=True)
),
inputs=dict(
search_click=Input('role-search', 'nClicks'),
refresh_click=Input('role-refresh', 'nClicks'),
pagination=Input('role-list-table', 'pagination'),
operations=Input('role-operations-store', 'data')
),
state=dict(
role_name=State('role-role_name-input', 'value'),
role_key=State('role-role_key-input', 'value'),
status_select=State('role-status-select', 'value'),
create_time_range=State('role-create_time-range', 'value'),
button_perms=State('role-button-perms-container', 'data')
),
prevent_initial_call=True
)
def get_role_table_data(search_click, refresh_click, pagination, operations, role_name, role_key, status_select, create_time_range, button_perms):
"""
获取角色表格数据回调进行表格相关增删查改操作后均会触发此回调
"""
create_time_start = None
create_time_end = None
if create_time_range:
create_time_start = create_time_range[0]
create_time_end = create_time_range[1]
query_params = dict(
role_name=role_name,
role_key=role_key,
status=status_select,
create_time_start=create_time_start,
create_time_end=create_time_end,
page_num=1,
page_size=10
)
triggered_id = dash.ctx.triggered_id
if triggered_id == 'role-list-table':
query_params = dict(
role_name=role_name,
role_key=role_key,
status=status_select,
create_time_start=create_time_start,
create_time_end=create_time_end,
page_num=pagination['current'],
page_size=pagination['pageSize']
)
if search_click or refresh_click or pagination or operations:
table_info = get_role_list_api(query_params)
if table_info['code'] == 200:
table_data = table_info['data']['rows']
table_pagination = dict(
pageSize=table_info['data']['page_size'],
current=table_info['data']['page_num'],
showSizeChanger=True,
pageSizeOptions=[10, 30, 50, 100],
showQuickJumper=True,
total=table_info['data']['total']
)
for item in table_data:
if item['status'] == '0':
item['status'] = dict(checked=True, disabled=item['role_id'] == 1)
else:
item['status'] = dict(checked=False, disabled=item['role_id'] == 1)
item['key'] = str(item['role_id'])
if item['role_id'] == 1:
item['operation'] = []
else:
item['operation'] = fac.AntdSpace(
[
fac.AntdButton(
'修改',
id={
'type': 'role-operation-table',
'operation': 'edit',
'index': str(item['role_id'])
},
type='link',
icon=fac.AntdIcon(
icon='antd-edit'
),
style={
'padding': 0
}
) if 'system:role:edit' in button_perms else [],
fac.AntdButton(
'删除',
id={
'type': 'role-operation-table',
'operation': 'delete',
'index': str(item['role_id'])
},
type='link',
icon=fac.AntdIcon(
icon='antd-delete'
),
style={
'padding': 0
}
) if 'system:role:remove' in button_perms else [],
fac.AntdPopover(
fac.AntdButton(
'更多',
type='link',
icon=fac.AntdIcon(
icon='antd-more'
),
style={
'padding': 0
}
),
content=fac.AntdSpace(
[
fac.AntdButton(
'数据权限',
id={
'type': 'role-operation-table',
'operation': 'datascope',
'index': str(item['role_id'])
},
type='text',
block=True,
icon=fac.AntdIcon(
icon='antd-check-circle'
),
style={
'padding': 0
}
),
fac.AntdButton(
'分配用户',
id={
'type': 'role-operation-table',
'operation': 'allocation',
'index': str(item['role_id'])
},
type='text',
block=True,
icon=fac.AntdIcon(
icon='antd-user'
),
style={
'padding': 0
}
),
],
direction='vertical'
),
placement='bottomRight'
) if 'system:role:edit' in button_perms else []
]
)
return dict(
role_table_data=table_data,
role_table_pagination=table_pagination,
role_table_key=str(uuid.uuid4()),
role_table_selectedrowkeys=None,
api_check_token_trigger={'timestamp': time.time()}
)
return dict(
role_table_data=dash.no_update,
role_table_pagination=dash.no_update,
role_table_key=dash.no_update,
role_table_selectedrowkeys=dash.no_update,
api_check_token_trigger={'timestamp': time.time()}
)
raise PreventUpdate
# 重置角色搜索表单数据回调
app.clientside_callback(
'''
(reset_click) => {
if (reset_click) {
return [null, null, null, null, {'type': 'reset'}]
}
return window.dash_clientside.no_update;
}
''',
[Output('role-role_name-input', 'value'),
Output('role-role_key-input', 'value'),
Output('role-status-select', 'value'),
Output('role-create_time-range', 'value'),
Output('role-operations-store', 'data')],
Input('role-reset', 'nClicks'),
prevent_initial_call=True
)
# 隐藏/显示角色搜索表单回调
app.clientside_callback(
'''
(hidden_click, hidden_status) => {
if (hidden_click) {
return [
!hidden_status,
hidden_status ? '隐藏搜索' : '显示搜索'
]
}
return window.dash_clientside.no_update;
}
''',
[Output('role-search-form-container', 'hidden'),
Output('role-hidden-tooltip', 'title')],
Input('role-hidden', 'nClicks'),
State('role-search-form-container', 'hidden'),
prevent_initial_call=True
)
@app.callback(
Output({'type': 'role-operation-button', 'operation': 'edit'}, 'disabled'),
Input('role-list-table', 'selectedRowKeys'),
prevent_initial_call=True
)
def change_role_edit_button_status(table_rows_selected):
"""
根据选择的表格数据行数控制编辑按钮状态回调
"""
outputs_list = dash.ctx.outputs_list
if outputs_list:
if table_rows_selected:
if len(table_rows_selected) > 1 or '1' in table_rows_selected:
return True
return False
return True
return dash.no_update
@app.callback(
Output({'type': 'role-operation-button', 'operation': 'delete'}, 'disabled'),
Input('role-list-table', 'selectedRowKeys'),
prevent_initial_call=True
)
def change_role_delete_button_status(table_rows_selected):
"""
根据选择的表格数据行数控制删除按钮状态回调
"""
outputs_list = dash.ctx.outputs_list
if outputs_list:
if table_rows_selected:
if '1' in table_rows_selected:
return True
return False
return True
return dash.no_update
@app.callback(
Output('role-menu-perms', 'expandedKeys', allow_duplicate=True),
Input('role-menu-perms-radio-fold-unfold', 'checked'),
State('role-menu-store', 'data'),
prevent_initial_call=True
)
def fold_unfold_role_menu(fold_unfold, menu_info):
"""
新增和编辑表单中展开/折叠checkbox回调
"""
if menu_info:
default_expanded_keys = []
for item in menu_info:
if item.get('parent_id') == 0:
default_expanded_keys.append(str(item.get('menu_id')))
if fold_unfold:
return default_expanded_keys
else:
return []
return dash.no_update
@app.callback(
Output('role-menu-perms', 'checkedKeys', allow_duplicate=True),
Input('role-menu-perms-radio-all-none', 'checked'),
State('role-menu-store', 'data'),
prevent_initial_call=True
)
def all_none_role_menu_mode(all_none, menu_info):
"""
新增和编辑表单中全选/全不选checkbox回调
"""
if menu_info:
default_expanded_keys = []
for item in menu_info:
if item.get('parent_id') == 0:
default_expanded_keys.append(str(item.get('menu_id')))
if all_none:
return [str(item.get('menu_id')) for item in menu_info]
else:
return []
return dash.no_update
@app.callback(
[Output('role-menu-perms', 'checkStrictly'),
Output('role-menu-perms', 'checkedKeys', allow_duplicate=True)],
Input('role-menu-perms-radio-parent-children', 'checked'),
State('current-role-menu-store', 'data'),
prevent_initial_call=True
)
def change_role_menu_mode(parent_children, current_role_menu):
"""
新增和编辑表单中父子联动checkbox回调
"""
checked_menu = []
if parent_children:
if current_role_menu:
for item in current_role_menu:
has_children = False
for other_item in current_role_menu:
if other_item['parent_id'] == item['menu_id']:
has_children = True
break
if not has_children:
checked_menu.append(str(item.get('menu_id')))
return [False, checked_menu]
else:
if current_role_menu:
checked_menu = [str(item.get('menu_id')) for item in current_role_menu if item] or []
return [True, checked_menu]
@app.callback(
output=dict(
modal_visible=Output('role-modal', 'visible', allow_duplicate=True),
modal_title=Output('role-modal', 'title'),
form_value=Output({'type': 'role-form-value', 'index': ALL, 'required': ALL}, 'value'),
form_label_validate_status=Output({'type': 'role-form-label', 'index': ALL, 'required': True}, 'validateStatus', allow_duplicate=True),
form_label_validate_info=Output({'type': 'role-form-label', 'index': ALL, 'required': True}, 'help', allow_duplicate=True),
menu_perms_tree=Output('role-menu-perms', 'treeData'),
menu_perms_expandedkeys=Output('role-menu-perms', 'expandedKeys', allow_duplicate=True),
menu_perms_checkedkeys=Output('role-menu-perms', 'checkedKeys', allow_duplicate=True),
menu_perms_halfcheckedkeys=Output('role-menu-perms', 'halfCheckedKeys', allow_duplicate=True),
role_menu=Output('role-menu-store', 'data'),
current_role_menu=Output('current-role-menu-store', 'data'),
api_check_token_trigger=Output('api-check-token', 'data', allow_duplicate=True),
edit_row_info=Output('role-edit-id-store', 'data'),
modal_type=Output('role-operations-store-bk', 'data')
),
inputs=dict(
operation_click=Input({'type': 'role-operation-button', 'operation': ALL}, 'nClicks'),
button_click=Input({'type': 'role-operation-table', 'operation': ALL, 'index': ALL}, 'nClicks')
),
state=dict(
selected_row_keys=State('role-list-table', 'selectedRowKeys')
),
prevent_initial_call=True
)
def add_edit_role_modal(operation_click, button_click, selected_row_keys):
"""
显示新增或编辑角色弹窗回调
"""
trigger_id = dash.ctx.triggered_id
if trigger_id.operation in ['add', 'edit']:
# 获取所有输出表单项对应value的index
form_value_list = [x['id']['index'] for x in dash.ctx.outputs_list[2]]
# 获取所有输出表单项对应label的index
form_label_list = [x['id']['index'] for x in dash.ctx.outputs_list[3]]
menu_params = dict(menu_name='', type='role')
tree_info = get_menu_tree_api(menu_params)
if tree_info.get('code') == 200:
tree_data = tree_info['data']
if trigger_id.type == 'role-operation-button' and trigger_id.operation == 'add':
role_info = dict(role_name=None, role_key=None, role_sort=None, status='0', remark=None)
return dict(
modal_visible=True,
modal_title='新增角色',
form_value=[role_info.get(k) for k in form_value_list],
form_label_validate_status=[None] * len(form_label_list),
form_label_validate_info=[None] * len(form_label_list),
menu_perms_tree=tree_data[0],
menu_perms_expandedkeys=[],
menu_perms_checkedkeys=None,
menu_perms_halfcheckedkeys=None,
role_menu=tree_data[1],
current_role_menu=None,
api_check_token_trigger={'timestamp': time.time()},
edit_row_info=None,
modal_type={'type': 'add'}
)
elif trigger_id.operation == 'edit':
if trigger_id.type == 'role-operation-button':
role_id = int(','.join(selected_row_keys))
else:
role_id = int(trigger_id.index)
role_info_res = get_role_detail_api(role_id=role_id)
if role_info_res['code'] == 200:
role_info = role_info_res['data']
checked_menu = []
checked_menu_all = []
if role_info.get('menu')[0]:
for item in role_info.get('menu'):
checked_menu_all.append(str(item.get('menu_id')))
has_children = False
for other_item in role_info.get('menu'):
if other_item['parent_id'] == item['menu_id']:
has_children = True
break
if not has_children:
checked_menu.append(str(item.get('menu_id')))
half_checked_menu = [x for x in checked_menu_all if x not in checked_menu]
return dict(
modal_visible=True,
modal_title='编辑角色',
form_value=[role_info.get('role').get(k) for k in form_value_list],
form_label_validate_status=[None] * len(form_label_list),
form_label_validate_info=[None] * len(form_label_list),
menu_perms_tree=tree_data[0],
menu_perms_expandedkeys=[],
menu_perms_checkedkeys=checked_menu,
menu_perms_halfcheckedkeys=half_checked_menu,
role_menu=tree_data[1],
current_role_menu=role_info.get('menu'),
api_check_token_trigger={'timestamp': time.time()},
edit_row_info=role_info.get('role') if role_info else None,
modal_type={'type': 'edit'}
)
return dict(
modal_visible=dash.no_update,
modal_title=dash.no_update,
form_value=[dash.no_update] * len(form_value_list),
form_label_validate_status=[dash.no_update] * len(form_value_list),
form_label_validate_info=[dash.no_update] * len(form_value_list),
menu_perms_tree=dash.no_update,
menu_perms_expandedkeys=dash.no_update,
menu_perms_checkedkeys=dash.no_update,
menu_perms_halfcheckedkeys=dash.no_update,
role_menu=dash.no_update,
current_role_menu=dash.no_update,
api_check_token_trigger={'timestamp': time.time()},
edit_row_info=None,
modal_type=None
)
raise PreventUpdate
@app.callback(
output=dict(
form_label_validate_status=Output({'type': 'role-form-label', 'index': ALL, 'required': True}, 'validateStatus',
allow_duplicate=True),
form_label_validate_info=Output({'type': 'role-form-label', 'index': ALL, 'required': True}, 'help',
allow_duplicate=True),
modal_visible=Output('role-modal', 'visible'),
operations=Output('role-operations-store', 'data', allow_duplicate=True),
api_check_token_trigger=Output('api-check-token', 'data', allow_duplicate=True),
global_message_container=Output('global-message-container', 'children', allow_duplicate=True)
),
inputs=dict(
confirm_trigger=Input('role-modal', 'okCounts')
),
state=dict(
modal_type=State('role-operations-store-bk', 'data'),
edit_row_info=State('role-edit-id-store', 'data'),
form_value=State({'type': 'role-form-value', 'index': ALL, 'required': ALL}, 'value'),
form_label=State({'type': 'role-form-value', 'index': ALL, 'required': True}, 'placeholder'),
menu_checked_keys=State('role-menu-perms', 'checkedKeys'),
menu_half_checked_keys=State('role-menu-perms', 'halfCheckedKeys'),
parent_checked=State('role-menu-perms-radio-parent-children', 'checked')
),
prevent_initial_call=True
)
def role_confirm(confirm_trigger, modal_type, edit_row_info, form_value, form_label, menu_checked_keys, menu_half_checked_keys, parent_checked):
"""
新增或编辑角色弹窗确认回调实现新增或编辑操作
"""
if confirm_trigger:
# 获取所有输出表单项对应label的index
form_label_output_list = [x['id']['index'] for x in dash.ctx.outputs_list[0]]
# 获取所有输入表单项对应的value及label
form_value_state = {x['id']['index']: x.get('value') for x in dash.ctx.states_list[2]}
form_label_state = {x['id']['index']: x.get('value') for x in dash.ctx.states_list[3]}
if all(validate_data_not_empty(item) for item in [form_value_state.get(k) for k in form_label_output_list]):
menu_half_checked_keys = menu_half_checked_keys if menu_half_checked_keys else []
menu_checked_keys = menu_checked_keys if menu_checked_keys else []
if parent_checked:
menu_perms = menu_half_checked_keys + menu_checked_keys
else:
menu_perms = menu_checked_keys
params_add = form_value_state
params_add['menu_id'] = ','.join(menu_perms) if menu_perms else None
params_edit = params_add.copy()
params_edit['role_id'] = edit_row_info.get('role_id') if edit_row_info else None
api_res = {}
modal_type = modal_type.get('type')
if modal_type == 'add':
api_res = add_role_api(params_add)
if modal_type == 'edit':
api_res = edit_role_api(params_edit)
if api_res.get('code') == 200:
if modal_type == 'add':
return dict(
form_label_validate_status=[None] * len(form_label_output_list),
form_label_validate_info=[None] * len(form_label_output_list),
modal_visible=False,
operations={'type': 'add'},
api_check_token_trigger={'timestamp': time.time()},
global_message_container=fuc.FefferyFancyMessage('新增成功', type='success')
)
if modal_type == 'edit':
return dict(
form_label_validate_status=[None] * len(form_label_output_list),
form_label_validate_info=[None] * len(form_label_output_list),
modal_visible=False,
operations={'type': 'edit'},
api_check_token_trigger={'timestamp': time.time()},
global_message_container=fuc.FefferyFancyMessage('编辑成功', type='success')
)
return dict(
form_label_validate_status=[None] * len(form_label_output_list),
form_label_validate_info=[None] * len(form_label_output_list),
modal_visible=dash.no_update,
operations=dash.no_update,
api_check_token_trigger={'timestamp': time.time()},
global_message_container=fuc.FefferyFancyMessage('处理失败', type='error')
)
return dict(
form_label_validate_status=[None if validate_data_not_empty(form_value_state.get(k)) else 'error' for k in form_label_output_list],
form_label_validate_info=[None if validate_data_not_empty(form_value_state.get(k)) else form_label_state.get(k) for k in form_label_output_list],
modal_visible=dash.no_update,
operations=dash.no_update,
api_check_token_trigger=dash.no_update,
global_message_container=fuc.FefferyFancyMessage('处理失败', type='error')
)
raise PreventUpdate
@app.callback(
[Output('role-operations-store', 'data', allow_duplicate=True),
Output('api-check-token', 'data', allow_duplicate=True),
Output('global-message-container', 'children', allow_duplicate=True)],
[Input('role-list-table', 'recentlySwitchDataIndex'),
Input('role-list-table', 'recentlySwitchStatus'),
Input('role-list-table', 'recentlySwitchRow')],
prevent_initial_call=True
)
def table_switch_role_status(recently_switch_data_index, recently_switch_status, recently_switch_row):
"""
表格内切换角色状态回调
"""
if recently_switch_data_index:
if recently_switch_status:
params = dict(role_id=int(recently_switch_row['key']), status='0', type='status')
else:
params = dict(role_id=int(recently_switch_row['key']), status='1', type='status')
edit_button_result = edit_role_api(params)
if edit_button_result['code'] == 200:
return [
{'type': 'switch-status'},
{'timestamp': time.time()},
fuc.FefferyFancyMessage('修改成功', type='success')
]
return [
{'type': 'switch-status'},
{'timestamp': time.time()},
fuc.FefferyFancyMessage('修改失败', type='error')
]
raise PreventUpdate
@app.callback(
[Output('role-delete-text', 'children'),
Output('role-delete-confirm-modal', 'visible'),
Output('role-delete-ids-store', 'data')],
[Input({'type': 'role-operation-button', 'operation': ALL}, 'nClicks'),
Input({'type': 'role-operation-table', 'operation': ALL, 'index': ALL}, 'nClicks')],
State('role-list-table', 'selectedRowKeys'),
prevent_initial_call=True
)
def role_delete_modal(operation_click, button_click, selected_row_keys):
"""
显示删除角色二次确认弹窗回调
"""
trigger_id = dash.ctx.triggered_id
if trigger_id.operation == 'delete':
if trigger_id.type == 'role-operation-button':
role_ids = ','.join(selected_row_keys)
else:
if trigger_id.type == 'role-operation-table':
role_ids = trigger_id.index
else:
raise PreventUpdate
return [
f'是否确认删除角色编号为{role_ids}的角色?',
True,
{'role_ids': role_ids}
]
raise PreventUpdate
@app.callback(
[Output('role-operations-store', 'data', allow_duplicate=True),
Output('api-check-token', 'data', allow_duplicate=True),
Output('global-message-container', 'children', allow_duplicate=True)],
Input('role-delete-confirm-modal', 'okCounts'),
State('role-delete-ids-store', 'data'),
prevent_initial_call=True
)
def role_delete_confirm(delete_confirm, role_ids_data):
"""
删除角色弹窗确认回调实现删除操作
"""
if delete_confirm:
params = role_ids_data
delete_button_info = delete_role_api(params)
if delete_button_info['code'] == 200:
return [
{'type': 'delete'},
{'timestamp': time.time()},
fuc.FefferyFancyMessage('删除成功', type='success')
]
return [
dash.no_update,
{'timestamp': time.time()},
fuc.FefferyFancyMessage('删除失败', type='error')
]
raise PreventUpdate
@app.callback(
[Output('role_to_allocated_user-modal', 'visible'),
Output({'type': 'allocate_user-search', 'index': 'allocated'}, 'nClicks'),
Output('allocate_user-role_id-container', 'data')],
Input({'type': 'role-operation-table', 'operation': ALL, 'index': ALL}, 'nClicks'),
State({'type': 'allocate_user-search', 'index': 'allocated'}, 'nClicks'),
prevent_initial_call=True
)
def role_to_allocated_user_modal(allocated_click, allocated_user_search_nclick):
"""
显示角色分配用户弹窗回调
"""
trigger_id = dash.ctx.triggered_id
if trigger_id.operation == 'allocation':
return [
True,
allocated_user_search_nclick + 1 if allocated_user_search_nclick else 1,
trigger_id.index
]
raise PreventUpdate
@app.callback(
[Output('role-export-container', 'data', allow_duplicate=True),
Output('role-export-complete-judge-container', 'data'),
Output('api-check-token', 'data', allow_duplicate=True),
Output('global-message-container', 'children', allow_duplicate=True)],
Input('role-export', 'nClicks'),
prevent_initial_call=True
)
def export_role_list(export_click):
"""
导出角色信息回调
"""
if export_click:
export_role_res = export_role_list_api({})
if export_role_res.status_code == 200:
export_role = export_role_res.content
return [
dcc.send_bytes(export_role, f'角色信息_{time.strftime("%Y%m%d%H%M%S", time.localtime())}.xlsx'),
{'timestamp': time.time()},
{'timestamp': time.time()},
fuc.FefferyFancyMessage('导出成功', type='success')
]
return [
dash.no_update,
dash.no_update,
{'timestamp': time.time()},
fuc.FefferyFancyMessage('导出失败', type='error')
]
raise PreventUpdate
@app.callback(
Output('role-export-container', 'data', allow_duplicate=True),
Input('role-export-complete-judge-container', 'data'),
prevent_initial_call=True
)
def reset_role_export_status(data):
"""
导出完成后重置下载组件数据回调防止重复下载文件
"""
time.sleep(0.5)
if data:
return None
raise PreventUpdate