def Field_Template_Factory(field, many=None, manyStyle=None): field_name = field.name field_ver_name = str(field.verbose_name) if not contain_zh(field_ver_name): field_ver_name = trans(field.name) # 进行字段名,国际化 rules = rules_factory(field, field_ver_name) default = default_factory(field) valueEnum = valueEnum_factory(field) hideIn = hideIn_factory(field) value_type = value_type_factory(field) render = render_factory(field, many, manyStyle) BASE_TEMPLATE = f"""{{ title: '{field_ver_name}', {hideIn} {default} dataIndex: '{field_name}', {value_type} rules: [ {rules} ], {render} {valueEnum} }}, """ return BASE_TEMPLATE
def gen_route(project_name_settings, user_label_list): init_django_env(project_name_settings) import django from django.conf import settings if not user_label_list: user_label_list = settings.TY_ADMIN_CONFIG["GEN_APPS"] path_list = [] gen_labels = SYS_LABELS + user_label_list gen_labels = [one for one in gen_labels if one != "contenttypes"] for one in django.apps.apps.get_models(): if one._meta.app_label in gen_labels: model_name = trans(one._meta.verbose_name) url = "/xadmin/" + get_lower_case_name(one._meta.model.__name__) one = "{" + f""" name: '{model_name}', icon: 'smile', path: '{url}', component: './AutoGenPage/{one._meta.model.__name__}List', """ + "}" # print(one) path_list.append(one) inner_before_list = """ [ { name: '首页', path: '/xadmin/index', icon: 'dashboard', component: './TyAdminBuiltIn/DashBoard' }, """ inner_after_list = """ ,{ name: 'Tyadmin内置', icon: 'VideoCamera', path: '/xadmin/sys', routes: [ { name: 'TyAdmin日志', icon: 'smile', path: '/xadmin/sys/ty_admin_sys_log', component: './TyAdminBuiltIn/TyAdminSysLogList' }, { name: 'TyAdmin验证', icon: 'smile', path: '/xadmin/sys/ty_admin_email_verify_record', component: './TyAdminBuiltIn/TyAdminEmailVerifyRecordList' } ] }] """ with open(f'{settings.BASE_DIR}/tyadmin_api/menu.json', 'w', encoding='utf-8') as fw: fw.write(inner_before_list + ",".join(path_list) + inner_after_list) with open(f'{settings.BASE_DIR}/tyadmin/config/auto_menu.json', 'w', encoding='utf-8') as fw: fw.write(inner_before_list + ",".join(path_list) + inner_after_list)
def gen_antd_pages(project_name_settings, user_label_list, focus_model=None, template_type="base"): # focus_model = 'HistoricalNetworkDevice' init_django_env(project_name_settings) import django from django.conf import settings if not user_label_list: user_label_list = settings.TY_ADMIN_CONFIG["GEN_APPS"] gen_labels = SYS_LABELS + user_label_list # focus_model = "CourseResource" model_pic_dict = {} model_date_dict = {} model_service_dict = {} user = get_user_model() from django.utils.translation import gettext as _ model_list = django.apps.apps.get_models() for one in model_list: columns = [] model_name = one._meta.model.__name__ model_ver_name = trans(one._meta.verbose_name_raw) # if(isinstance(model_ver_name,__proxy__)): if focus_model and model_name != focus_model: continue if one._meta.app_label in gen_labels: print("^^" * 20) print(one._meta.app_label) print("^^" * 20) img_field_list = [] date_row_list = [] model_add_item_list = [] model_many_to_many_list = [] model_import_item_list = [] model_service_item_list = [] date_field_list = [] fileds_num = len(one.objects.model._meta.fields) for filed in one.objects.model._meta.fields: name = filed.name ver_name = trans(filed.verbose_name) if not contain_zh(ver_name): ver_name = trans(filed.name) print(ver_name) if import_adapter_priority(one._meta.app_label,one._meta.model.__name__, name): model_import_item_list.append(FIELD_ADAPTER[one._meta.app_label][one._meta.model.__name__][name]["index_import"]) if effect_state_adapter_priority(one._meta.app_label,one._meta.model.__name__, name): model_add_item_list.append(FIELD_ADAPTER[one._meta.app_label][one._meta.model.__name__][name]["effect_state"]) if services_adapter_priority(one._meta.app_label, one._meta.model.__name__, name): model_service_item_list.append(FIELD_ADAPTER[one._meta.app_label][one._meta.model.__name__][name]["services"]) if field_adapter_priority(one._meta.app_label,one._meta.model.__name__, name): one_c = FIELD_ADAPTER[one._meta.app_label][one._meta.model.__name__][name]["field"] elif isinstance(filed, ForeignKey): rela_model = filed.related_model._meta.object_name if filed.blank: one_c = """{ title: '%s', dataIndex: '%s', backendType: 'foreignKey', rules: [ ], renderFormItem: (item, {value, onChange}) => { return dealForeignKeyField(item, value, onChange, %sForeignKeyList); },render: (text, record) => { return renderForeignKey(text, %sVerboseNameMap); }, },""" % (ver_name, name, name, name) else: one_c = """{ title: '%s', dataIndex: '%s', backendType: 'foreignKey', rules: [ { required: true, message: '%s为必填项', }, ], renderFormItem: (item, {value, onChange}) => { return dealForeignKeyField(item, value, onChange, %sForeignKeyList); }, render: (text, record) => { return renderForeignKey(text, %sVerboseNameMap); }, },""" % (ver_name, name, ver_name, name, name) one_c = one_c.replace("$模型名字$", name) print(name) print(filed) if isinstance(filed.verbose_name, str): one_c = one_c.replace("$模型显示名字$", filed.verbose_name) else: one_c = one_c.replace("$模型显示名字$", name) model_add_item = """const [$模型名字$ForeignKeyList, set$模型名字首字母大写$ForeignKeyList] = useState([]); useEffect(() => { query$关联Model$({all: 1}).then(value => { set$模型名字首字母大写$ForeignKeyList(value); }); }, []); const [$模型名字$VerboseNameMap, set$模型名字首字母大写$VerboseNameMap] = useState([]); useEffect(() => { query$关联Model$VerboseName().then(value => { set$模型名字首字母大写$VerboseNameMap(value); }); }, []); """.replace("$模型名字$", name).replace("$模型名字首字母大写$", name[0].upper() + name[1:]).replace("$关联Model$", rela_model) model_add_item_list.append(model_add_item) model_import_item = """import {query$关联Model$, query$关联Model$VerboseName} from '@/pages/AutoGenPage/$关联Model$List/service';""" \ .replace("$关联Model$", rela_model) if model_import_item not in model_import_item_list and model_name != rela_model: model_import_item_list.append(model_import_item) elif isinstance(filed, CharField): if filed.default != NOT_PROVIDED: if filed.default is not None: default_replace = f' initialValue: "{filed.default}",' else: default_replace = "" else: default_replace = "" if filed.choices: filed_choices_list = [] for filed_one in filed.choices: one_line = f'"{filed_one[0]}":"{filed_one[1]}"' filed_choices_list.append(one_line) if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ ], valueEnum: { $valueEnum$ }, },""" % (default_replace, ver_name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ { required: true, message: '%s为必填项', }, ], valueEnum: { $valueEnum$ }, },""" % (default_replace, ver_name, name, ver_name) one_c = one_c.replace("$valueEnum$", ",".join(filed_choices_list)) elif name == "password": one_c = """{ %s title: '%s', dataIndex: '%s', hideInTable: true, hideInSearch: true, rules: [ { required: true, message: '%s为必填项', }, ], },""" % (default_replace, ver_name, name, ver_name) else: if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ ], },""" % (default_replace, ver_name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ { required: true, message: '%s为必填项', }, ], },""" % (default_replace, ver_name, name, ver_name) elif name == "id": if filed.default != NOT_PROVIDED: default_replace = f' initialValue: "{filed.default}",' else: default_replace = "" one_c = """{ %s title: '%s', dataIndex: '%s', hideInForm: true, hideInSearch: true, rules: [ { required: true, message: '%s为必填项', }, ], },""" % (default_replace, ver_name, name, ver_name) elif isinstance(filed, DateTimeField): print(filed.default) hideInform = None if filed.default != django.db.models.NOT_PROVIDED or filed.auto_now == True: hideInform = True one_c = """{ title: '%s', dataIndex: '%s', valueType: 'dateTime', $隐藏form$ rules: [ { required: true, message: '%s为必填项', }, ], render: (text, record) => { return dealDateTimeDisplay(text); }, },""" % (ver_name, name, ver_name) if hideInform: one_c = one_c.replace("$隐藏form$", "hideInForm: true,") else: one_c = one_c.replace("$隐藏form$", "") date_field_list.append('"' + name + '"') date_row_list.append(f'record.{name} = moment(record.{name});') elif isinstance(filed, DateField): if filed.auto_now: pass one_c = """{ title: '%s', dataIndex: '%s', valueType: 'date', rules: [ { required: true, message: '%s为必填项', }, ], },""" % (ver_name, name, ver_name) date_field_list.append('"' + name + '"') date_row_list.append(f'record.{name} = moment(record.{name});') elif isinstance(filed, BooleanField): if filed.default != NOT_PROVIDED: if filed.default: default_replace = f' initialValue: true,' else: default_replace = f' initialValue: false,' else: default_replace = "" if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ ], render: (text, record) => { return BooleanDisplay(text); }, renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const is_value = form.getFieldValue('%s'); if (type === "form" && !is_value) { form.setFieldsValue({'%s': false}); } return <Switch checked={is_value} onClick={(checked) => { form.setFieldsValue({'%s': checked}); }} />; }, },""" % (default_replace, ver_name, name, name, name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ { required: true, message: '%s为必填项', }, ], render: (text, record) => { return BooleanDisplay(text); }, renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const is_value = form.getFieldValue('%s'); if (type === "form" && !is_value) { form.setFieldsValue({'%s': false}); } return <Switch checked={is_value} onClick={(checked) => { form.setFieldsValue({'%s': checked}); }} />; }, },""" % (default_replace, ver_name, name, ver_name, name, name, name) elif isinstance(filed, IntegerField) or isinstance(filed, FloatField): if filed.default != NOT_PROVIDED: if filed.default: default_replace = f' initialValue: {filed.default},' else: default_replace = "" else: default_replace = "" if filed.choices: filed_choices_list = [] for filed_one in filed.choices: one_line = f'{filed_one[0]}:"{filed_one[1]}"' filed_choices_list.append(one_line) if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ ], valueEnum: { $valueEnum$ }, },""" % (default_replace, ver_name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ { required: true, message: '%s为必填项', }, ], valueEnum: { $valueEnum$ }, },""" % (default_replace, ver_name, name, ver_name) one_c = one_c.replace("$valueEnum$", ",".join(filed_choices_list)) else: if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', valueType: 'digit', rules: [ ], },""" % (default_replace, ver_name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', valueType: 'digit', rules: [ { required: true, message: '%s为必填项', }, ], },""" % (default_replace, ver_name, name, ver_name) elif isinstance(filed, ImageField) or isinstance(filed, SImageField): img_field_list.append('"' + name + '"') help_text = filed.help_text if "头像" in filed.verbose_name or filed.name == "avatar" or help_text == MAIN_AVATAR: if filed.blank: one_c = """{ title: '%s', dataIndex: '%s', valueType: 'avatar', hideInSearch: true, rules: [ ], renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const imageUrl = form.getFieldValue('%s'); return <UploadAvatar img={imageUrl}/> }, },""" % (ver_name, name, name) else: one_c = """{ title: '%s', dataIndex: '%s', valueType: 'avatar', hideInSearch: true, rules: [ { required: true, message: '%s为必填项', }, ], renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const imageUrl = form.getFieldValue('%s'); return <UploadAvatar img={imageUrl}/> }, },""" % (ver_name, name, ver_name, name) else: if filed.blank: one_c = """{ title: '%s', dataIndex: '%s', render: (text, record) => { return <img src={text} width="80px" alt=""/> }, hideInSearch: true, rules: [ ], renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const imageUrl = form.getFieldValue('%s'); return <UploadAvatar img={imageUrl}/> }, },""" % (ver_name, name, name) else: one_c = """{ title: '%s', dataIndex: '%s', render: (text, record) => { return <img src={text} width="80px" alt=""/> }, hideInSearch: true, rules: [ { required: true, message: '%s为必填项', }, ], renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const imageUrl = form.getFieldValue('%s'); return <UploadAvatar img={imageUrl}/> }, },""" % (ver_name, name, ver_name, name) elif isinstance(filed, FileField): img_field_list.append('"' + name + '"') if filed.blank: one_c = """{ title: '%s', dataIndex: '%s', hideInSearch: true, rules: [ ], render: (text, record) => { return <a download={text.split('/').slice(-1)} href={text}>{text.split('/').slice(-1)}</a>; }, renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const downloadUrl = form.getFieldValue('download'); return fileUpload(downloadUrl); }, },""" % (ver_name, name) else: one_c = """{ title: '%s', dataIndex: '%s', hideInSearch: true, rules: [ { required: true, message: '%s为必填项', }, ], render: (text, record) => { return <a download={text.split('/').slice(-1)} href={text}>{text.split('/').slice(-1)}</a>; }, renderFormItem: (_, {type, defaultRender, ...rest}, form) => { const downloadUrl = form.getFieldValue('download'); return fileUpload(downloadUrl); }, },""" % (ver_name, name, ver_name) elif filed.__class__.__name__ == "TextField": if filed.default != NOT_PROVIDED: default_replace = f' initialValue: "{filed.default}",' else: default_replace = "" if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', valueType: 'textarea', ellipsis: true, rules: [ ], },""" % (default_replace, ver_name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', valueType: 'textarea', ellipsis: true, rules: [ { required: true, message: '%s为必填项', }, ], },""" % (default_replace, ver_name, name, ver_name) elif filed.__class__.__name__ == "UEditorField" or isinstance(filed, richTextField): if filed.default != NOT_PROVIDED: default_replace = f' initialValue: "{filed.default}",' else: default_replace = "" if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', customCol:richCol, ellipsis: true, hideInSearch: true, rules: [ ], renderFormItem: (_, {type, defaultRender, ...rest}, form) => { return richForm(form, rest.id); }, },""" % (default_replace, ver_name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', customCol:richCol, ellipsis: true, hideInSearch: true, rules: [ { required: true, message: '%s为必填项', }, ], renderFormItem: (_, {type, defaultRender, ...rest}, form) => { return richForm(form, rest.id); }, },""" % (default_replace, ver_name, name, ver_name) elif filed.__class__.__name__ == "TimeZoneField": if filed.default != NOT_PROVIDED: if filed.default is not None: default_replace = f' initialValue: "{filed.default}",' else: default_replace = "" else: default_replace = "" if filed.CHOICES: filed_choices_list = [] for filed_one in filed.CHOICES: one_line = f'"{filed_one[0]}":"{filed_one[1]}"' filed_choices_list.append(one_line) if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ ], valueEnum: { $valueEnum$ }, },""" % (default_replace, ver_name, name) else: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ { required: true, message: '%s为必填项', }, ], valueEnum: { $valueEnum$ }, },""" % (default_replace, ver_name, name, ver_name) one_c = one_c.replace("$valueEnum$", ",".join(filed_choices_list)) else: if filed.default != NOT_PROVIDED: default_replace = f' initialValue: "{filed.default}",' else: default_replace = "" if filed.blank: one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ ], },""" % (default_replace, ver_name, name) else: print(filed.__class__.__name__) one_c = """{ %s title: '%s', dataIndex: '%s', rules: [ { required: true, message: '%s为必填项', }, ], },""" % (default_replace, ver_name, name, ver_name) model_pic_dict[model_name] = img_field_list model_date_dict[model_name] = date_field_list columns.append(one_c) model_service_dict[model_name] = model_service_item_list for filed in one.objects.model._meta.many_to_many: black_many_to_many = [] ver_name = filed.verbose_name if filed.name in black_many_to_many: continue else: name = filed.name rela_model = filed.related_model._meta.object_name print("&&&" * 30) print(one._meta.app_label, name) print("&&&" * 30) if filed.blank: if filed.name == "permissions" and one._meta.app_label == "auth" and rela_model == "Group": width_help = " width: '70%'," else: width_help = "" one_c = """{ title: '%s', dataIndex: '%s', %s rules: [ ], renderFormItem: (item, {value, onChange, type, defaultRender}) => { return dealManyToManyField(item, value,onChange,type, %sManyToManyList) }, render: (text, record) => { return renderManyToMany(text) }, },""" % (ver_name, name, width_help, name,) else: one_c = """{ title: '%s', dataIndex: '%s', rules: [ { required: true, message: '%s为必填项', }, ], renderFormItem: (item, {value, onChange, type, defaultRender}) => { return dealManyToManyField(item, value,onChange,type, %sManyToManyList) }, render: (text, record) => { return renderManyToMany(text) }, },""" % (ver_name, name, ver_name, name) one_c = one_c.replace("$百分$", "%") model_many_to_many_item = """const [$模型名字$ManyToManyList, set$模型名字首字母大写$ManyToManyList] = useState([]); useEffect(() => { query$关联Model$({all:1}).then(value => { set$模型名字首字母大写$ManyToManyList(value); }); }, []);""".replace("$模型名字$", name).replace("$模型名字首字母大写$", name[0].upper() + name[1:]) \ .replace("$关联Model$", rela_model) model_many_to_many_list.append(model_many_to_many_item) model_import_item = """import {query$关联Model$} from '@/pages/AutoGenPage/$关联Model$List/service';""".replace("$关联Model$", rela_model) if model_import_item not in model_import_item_list and model_name != rela_model: model_import_item_list.append(model_import_item) columns.append(one_c) if one._meta.app_label == user._meta.app_label and model_name == user._meta.object_name: opera = """ { title: '操作', dataIndex: 'option', valueType: 'option', fixed: 'right', width: 100, render: (text, record) => ( <> <EditOutlined title="编辑" className="icon" onClick={async () => { $时间处理占位$ handleUpdateModalVisible(true); setUpdateFormValues(record); }} /> <Divider type="vertical" /> <KeyOutlined onClick={() => { handleUpdatePassWordModalVisible(true); setUpdateFormValues(record); }} /> <Divider type="vertical" /> <Popconfirm title="您确定要删除$模型名字$吗?" placement="topRight" onConfirm={() => { handleRemove([record]) actionRef.current.reloadAndRest(); }} okText="确定" cancelText="取消" > <DeleteOutlined title="删除" className="icon" /> </Popconfirm> </> ), },""".replace("$模型名字$", model_ver_name) else: opera = """ { title: '操作', dataIndex: 'option', valueType: 'option', fixed: 'right', width: 100, render: (text, record) => ( <> <EditOutlined title="编辑" className="icon" onClick={async () => { $时间处理占位$ handleUpdateModalVisible(true); setUpdateFormValues(record); }} /> <Divider type="vertical" /> <Popconfirm title="您确定要删除$模型名字$吗?" placement="topRight" onConfirm={() => { handleRemove([record]) actionRef.current.reloadAndRest(); }} okText="确定" cancelText="取消" > <DeleteOutlined title="删除" className="icon" /> </Popconfirm> </> ), },""".replace("$模型名字$", model_ver_name) opera = opera.replace("$时间处理占位$", "".join(date_row_list)) columns.append(opera) dest_path = f'{os.path.dirname(__file__)}/antd_page_templates/{template_type}' with open(f'{dest_path}/index.jsx', encoding='utf-8') as fr: content = fr.read() if fileds_num > 8: new_content = content.replace("$两列布局占位$", """ search={{ span: { lg: 12, md: 12, xxl: 12, xl: 12, sm: 12, xs: 24, }, }}""") else: new_content = content.replace("$两列布局占位$", "") new_content = new_content.replace("$占位列数据$", "".join(columns)) new_content = new_content.replace("$占位模型名$", model_name) new_content = new_content.replace("$占位模型显示名$", str(model_ver_name)) new_content = new_content.replace("$外键占位$", "".join(model_add_item_list)) new_content = new_content.replace("$多对多占位$", "".join(model_many_to_many_list)) new_content = new_content.replace("$import占位$", "".join(model_import_item_list)) new_content = new_content.replace("$时间占位$", ",".join(model_date_dict[model_name])) if one._meta.app_label == user._meta.app_label and model_name == user._meta.object_name: print("生成密码", one._meta.app_label, model_name) new_content = new_content.replace("$passwordform引入$", """import {updateUserPassword} from './service';\nimport UpdatePasswordForm from './components/UpdatePasswordForm';""") new_content = new_content.replace("$Password占位$", 'const [updatePassWordModalVisible, handleUpdatePassWordModalVisible] = useState(false);\nconst [updatePasswordForm] = Form.useForm();') new_content = new_content.replace("$更新密码方法占位$", """ const handlePassWordUpdate = () => { if (updatePasswordForm.getFieldValue('password') !== updatePasswordForm.getFieldValue('re_password')) { updatePasswordForm.setFields([{ name: 're_password', errors: ['两次密码不一致'], }]); } else { updatePasswordForm.validateFields().then( value => { updateUserPassword({ ...value, username: updateFormValues["username"], }).then( message.success('密码修改成功'), handleUpdatePassWordModalVisible(false), ); }, ); updatePasswordForm.submit; } };""") new_content = new_content.replace("$Password更新Form$", """ { <UpdatePasswordForm onCancel={() => { handleUpdatePassWordModalVisible(false); }} handleUpdate={handlePassWordUpdate} updateModalVisible={updatePassWordModalVisible} userName={updateFormValues["username"]} > <Form form={updatePasswordForm}> <FormItem labelCol={{ span: 5, }} wrapperCol={{ span: 15, }} label="密码" name="password" rules={[ { required: true, message: '请输入密码!', }, ]} > <Input.Password placeholder="请输入密码" type="password" /> </FormItem> <FormItem labelCol={{ span: 5, }} wrapperCol={{ span: 15, }} label="重复密码" name="re_password" rules={[ { required: true, message: '请输入重复密码', }, ]} > <Input.Password placeholder="请再次输入密码" type="password" /> </FormItem> </Form> </UpdatePasswordForm> }""") else: new_content = new_content.replace("$passwordform引入$", "") new_content = new_content.replace("$Password占位$", '') new_content = new_content.replace("$更新密码方法占位$", '') new_content = new_content.replace("$Password更新Form$", '') with open(f'{dest_path}/service.js', encoding='utf-8') as fr: content = fr.read() new_services = content.replace("$占位path$", get_lower_case_name(model_name)) new_services = new_services.replace("$占位模型名$", model_name) new_services = new_services.replace("$图片字段列表$", ",".join(model_pic_dict[model_name])) new_services = new_services.replace("$适配service占位$", "\n".join(model_service_dict[model_name])) if one._meta.app_label == user._meta.app_label: new_services = new_services.replace("$更新密码service占位$", """ export async function updateUserPassword(params) { return request('/api/xadmin/v1/list_change_password', { method: 'POST', data: { ...params}, }); }""") else: new_services = new_services.replace("$更新密码service占位$", "") with open(f'{dest_path}/components/CreateForm.jsx', encoding='utf-8') as fr: create_form = fr.read() create_form = create_form.replace("$占位模型显示名$", str(model_ver_name)) if fileds_num > 8: create_form = create_form.replace("$宽度占位$", 'width={1200}') else: create_form = create_form.replace("$宽度占位$", "width={800}") with open(f'{dest_path}/components/UpdateForm.jsx', encoding='utf-8') as fr: update_form = fr.read() update_form = update_form.replace("$占位模型显示名$", str(model_ver_name)) if fileds_num > 8: update_form = update_form.replace("$宽度占位$", 'width={1200}') else: update_form = update_form.replace("$宽度占位$", "width={800}") with open(f'{dest_path}/components/UpdatePasswordForm.jsx', encoding='utf-8') as fr: change_password_form = fr.read() target_path = f'{settings.BASE_DIR}/tyadmin/src/pages/AutoGenPage' cur_path = f'{target_path}/{model_name}List' if not os.path.exists(cur_path): os.makedirs(cur_path) cur_path_co = f'{target_path}/{model_name}List/components' if not os.path.exists(cur_path_co): os.mkdir(cur_path_co) with open(f'{target_path}/{model_name}List/index.jsx', 'w', encoding='utf-8') as fw: fw.write(new_content) with open(f'{target_path}/{model_name}List/service.js', 'w', encoding='utf-8') as fw: fw.write(new_services) with open(f'{target_path}/{model_name}List/components/CreateForm.jsx', 'w', encoding='utf-8') as fw: fw.write(create_form) with open(f'{target_path}/{model_name}List/components/UpdateForm.jsx', 'w', encoding='utf-8') as fw: fw.write(update_form) if one._meta.app_label == user._meta.app_label: with open(f'{target_path}/{model_name}List/components/UpdatePasswordForm.jsx', 'w', encoding='utf-8') as fw: fw.write(change_password_form)
def gen_antd_pages(project_name_settings, user_label_list, focus_model=None, template_type="base"): init_django_env(project_name_settings) import django from django.conf import settings if not user_label_list: user_label_list = settings.TY_ADMIN_CONFIG["GEN_APPS"] try: force_cover = settings.TY_ADMIN_CONFIG['FORCED_COVER'] except KeyError: force_cover = False gen_labels = SYS_LABELS + user_label_list model_pic_dict = {} model_date_dict = {} model_service_dict = {} user = get_user_model() model_list = django.apps.apps.get_models() for one in model_list: columns = [] model_name = one._meta.model.__name__ model_ver_name = trans(one._meta.verbose_name_raw) app_name = one._meta.app_label model_meta = one.objects.model._meta if focus_model and model_name != focus_model: continue if app_name in gen_labels: img_field_list = [] date_row_list = [] model_add_item_list = [] model_many_to_many_list = [] model_import_item_list = [] model_service_item_list = [] date_field_list = [] fields_count = len(model_meta.fields) # 处理字段类型 for field in model_meta.fields: field_name = field.name # 进行第三方Field重写,优先级高 try: cur_adapter = FIELD_ADAPTER[app_name][model_name][ field_name] except KeyError: cur_adapter = None if cur_adapter and adapter_priority( app_name, model_name, field_name, "index_import"): model_import_item_list.append(cur_adapter["index_import"]) if cur_adapter and adapter_priority( app_name, model_name, field_name, "effect_state"): model_add_item_list.append(cur_adapter["effect_state"]) if cur_adapter and adapter_priority(app_name, model_name, field_name, "services"): model_service_item_list.append(cur_adapter["services"]) if cur_adapter and adapter_priority(app_name, model_name, field_name, "field"): current_field = cur_adapter["field"] elif isinstance(field, ForeignKey): associated_model = field.related_model._meta.object_name current_field = Field_Template_Factory(field) # print(current_field) field_first_up = field_name[0].upper() + field_name[1:] model_add_item = f""" const [{field_name}ForeignKeyList, set{field_first_up}ForeignKeyList] = useState([]); useEffect(() => {{ query{associated_model}({{all: 1}}).then(value => {{ set{field_first_up}ForeignKeyList(value); }}); }}, []); const [{field_name}VerboseNameMap, set{field_first_up}VerboseNameMap] = useState([]); useEffect(() => {{ query{associated_model}VerboseName().then(value => {{ set{field_first_up}VerboseNameMap(value); }}); }}, []);""" model_add_item_list.append(model_add_item) model_import_item = f"""import {{query{associated_model}, query{associated_model}VerboseName}} from '@/pages/AutoGenPage/{associated_model}List/service';""" if model_import_item not in model_import_item_list and model_name != associated_model: # 外键self 处理 model_import_item_list.append(model_import_item) elif isinstance(field, CharField): current_field = Field_Template_Factory(field) elif isinstance(field, DateTimeField): current_field = Field_Template_Factory(field) date_field_list.append('"' + field_name + '"') date_row_list.append( f'record.{field_name} = record.{field_name} === null ? record.{field_name} : moment(record.{field_name});' ) elif isinstance(field, DateField): current_field = Field_Template_Factory(field) date_field_list.append('"' + field_name + '"') date_row_list.append( f'record.{field_name} = record.{field_name} === null ? record.{field_name} : moment(record.{field_name});' ) elif isinstance(field, BooleanField): current_field = Field_Template_Factory(field) elif isinstance(field, IntegerField): current_field = Field_Template_Factory(field) elif isinstance(field, FloatField): current_field = Field_Template_Factory(field) elif isinstance(field, ImageField) or isinstance( field, SImageField): img_field_list.append('"' + field_name + '"') current_field = Field_Template_Factory(field) elif isinstance(field, FileField): img_field_list.append('"' + field_name + '"') current_field = Field_Template_Factory(field) elif isinstance(field, TextField): current_field = Field_Template_Factory(field) elif field.__class__.__name__ == "UEditorField" or isinstance( field, richTextField): current_field = Field_Template_Factory(field) elif field.__class__.__name__ == "TimeZoneField": current_field = Field_Template_Factory(field) else: current_field = Field_Template_Factory(field) # print(current_field) model_pic_dict[model_name] = img_field_list model_date_dict[model_name] = date_field_list columns.append(current_field) model_service_dict[model_name] = model_service_item_list # 处理多对多关系 for field in model_meta.many_to_many: associated_model = field.related_model._meta.object_name if style_adapter_priority(app_name, model_name, field.name, "many_to_many"): try: many_style = settings.TY_ADMIN_CONFIG["REWRITE"][ app_name][model_name][field.name]["many_to_many"] except KeyError: many_style = ty_admin_settings.TY_ADMIN_CONFIG[ "REWRITE"][app_name][model_name][ field.name]["many_to_many"] else: many_style = None current_field = Field_Template_Factory(field, many=True, manyStyle=many_style) field_name = field.name field_first_up = field_name[0].upper() + field_name[1:] model_many_to_many_item = f"""const [{field.name}ManyToManyList, set{field_first_up}ManyToManyList] = useState([]); useEffect(() => {{ query{associated_model}({{all:1}}).then(value => {{ set{field_first_up}ManyToManyList(value); }}); }}, []);""" model_many_to_many_list.append(model_many_to_many_item) model_import_item = f"""import {{query{associated_model}}} from '@/pages/AutoGenPage/{associated_model}List/service';""" if model_import_item not in model_import_item_list and model_name != associated_model: model_import_item_list.append(model_import_item) columns.append(current_field) if app_name == user._meta.app_label and model_name == user._meta.object_name: opera = UserOptionColumn.replace(">>MODEL_NAME<<", model_ver_name) else: opera = GeneralOptionColumn.replace(">>MODEL_NAME<<", model_ver_name) opera = opera.replace(">>TIME_DEAL_PLACEHOLDER<<", "".join(date_row_list)) columns.append(opera) dest_path = f'{os.path.dirname(__file__)}/antd_page_templates/{template_type}' with open(f'{dest_path}/index.jsx', encoding='utf-8') as fr: content = fr.read() if fields_count > 8: new_content = content.replace(">>TWO_COLUMNS_COL<<", "twoColumns") else: new_content = content.replace(">>TWO_COLUMNS_COL<<", "{}") new_content = new_content.replace(">>COLUMNS_LIST<<", "".join(columns)) new_content = new_content.replace(">>MODEL_NAME<<", model_name) new_content = new_content.replace(">>MODEL_VERBOSE_NAME<<", str(model_ver_name)) new_content = new_content.replace(">>FOREIGNKEY_PLACE<<", "".join(model_add_item_list)) new_content = new_content.replace( ">>MANY_TO_MANY_PLACE<<", "".join(model_many_to_many_list)) new_content = new_content.replace( ">>IMPORT_PLACE<<", "".join(model_import_item_list)) new_content = new_content.replace( ">>TIME_PLACE<<", ",".join(model_date_dict[model_name])) if app_name == user._meta.app_label and model_name == user._meta.object_name: # 更新自己的密码 new_content = new_content.replace(">>PASSWORD_FORM<<", PASSWORD_FORM) new_content = new_content.replace(">>PASSWORD_PLACE<<", PASSWORD_PLACE) new_content = new_content.replace( ">>UPDATE_PASSWORD_METHOD<<", UPDATE_PASSWORD_METHOD) new_content = new_content.replace( ">>PASSWORD_UPDATE_FORM<<", PASSWORD_UPDATE_FORM) else: new_content = new_content.replace(">>PASSWORD_FORM<<", "") new_content = new_content.replace(">>PASSWORD_PLACE<<", '') new_content = new_content.replace( ">>UPDATE_PASSWORD_METHOD<<", '') new_content = new_content.replace( ">>PASSWORD_UPDATE_FORM<<", '') with open(f'{dest_path}/service.js', encoding='utf-8') as fr: content = fr.read() new_services = content.replace(">>MODEL_NAME_LOWER_CASE<<", get_lower_case_name(model_name)) new_services = new_services.replace(">>MODEL_NAME<<", model_name) new_services = new_services.replace( ">>IMAGE_FIELD_LIST<<", ",".join(model_pic_dict[model_name])) new_services = new_services.replace( ">>ADAPTER_SERVICE<<", "\n".join(model_service_dict[model_name])) if app_name == user._meta.app_label: new_services = new_services.replace( ">>UPDATE_PASSWORD_SERVICE<<", """ export async function updateUserPassword(params) { return request('/api/xadmin/v1/list_change_password', { method: 'POST', data: { ...params}, }); }""") else: new_services = new_services.replace( ">>UPDATE_PASSWORD_SERVICE<<", "") with open(f'{dest_path}/components/CreateForm.jsx', encoding='utf-8') as fr: create_form = fr.read() create_form = create_form.replace(">>MODEL_VERBOSE_NAME<<", str(model_ver_name)) if fields_count > 8: create_form = create_form.replace(">>WIDTH_PLACEHOLDER<<", 'width={1200}') else: create_form = create_form.replace(">>WIDTH_PLACEHOLDER<<", "width={800}") with open(f'{dest_path}/components/UpdateForm.jsx', encoding='utf-8') as fr: update_form = fr.read() update_form = update_form.replace(">>MODEL_VERBOSE_NAME<<", str(model_ver_name)) if fields_count > 8: update_form = update_form.replace(">>WIDTH_PLACEHOLDER<<", 'width={1200}') else: update_form = update_form.replace(">>WIDTH_PLACEHOLDER<<", "width={800}") with open(f'{dest_path}/components/UpdatePasswordForm.jsx', encoding='utf-8') as fr: change_password_form = fr.read() target_path = f'{settings.BASE_DIR}/tyadmin/src/pages/AutoGenPage' cur_path = f'{target_path}/{model_name}List' if not os.path.exists(cur_path): os.makedirs(cur_path) cur_path_co = f'{target_path}/{model_name}List/components' if not os.path.exists(cur_path_co): os.makedirs(cur_path_co) index_jsx_path = f'{target_path}/{model_name}List/index.jsx' if not force_cover and os.path.exists(index_jsx_path): pass else: with open(index_jsx_path, 'w', encoding='utf-8') as fw: fw.write(new_content) service_jsx_path = f'{target_path}/{model_name}List/service.js' if not force_cover and os.path.exists(service_jsx_path): pass else: with open(service_jsx_path, 'w', encoding='utf-8') as fw: fw.write(new_services) create_form_path = f'{target_path}/{model_name}List/components/CreateForm.jsx' if not force_cover and os.path.exists(create_form_path): pass else: with open(create_form_path, 'w', encoding='utf-8') as fw: fw.write(create_form) update_form_path = f'{target_path}/{model_name}List/components/UpdateForm.jsx' if not force_cover and os.path.exists(update_form_path): pass else: with open(update_form_path, 'w', encoding='utf-8') as fw: fw.write(update_form) if app_name == user._meta.app_label: update_password_form_path = f'{target_path}/{model_name}List/components/UpdatePasswordForm.jsx' if not force_cover and os.path.exists( update_password_form_path): pass else: with open(update_password_form_path, 'w', encoding='utf-8') as fw: fw.write(change_password_form)