review: 修改 footer.
This commit is contained in:
@@ -1,143 +1,85 @@
|
|||||||
// package com.orion.ops.module.asset.controller;
|
package com.orion.ops.module.asset.controller;
|
||||||
//
|
|
||||||
// import com.orion.lang.define.wrapper.DataGrid;
|
import com.orion.ops.framework.biz.operator.log.core.annotation.OperatorLog;
|
||||||
// import com.orion.ops.framework.biz.operator.log.core.annotation.OperatorLog;
|
import com.orion.ops.framework.log.core.annotation.IgnoreLog;
|
||||||
// import com.orion.ops.framework.common.validator.group.Page;
|
import com.orion.ops.framework.log.core.enums.IgnoreLogMode;
|
||||||
// import com.orion.ops.framework.log.core.annotation.IgnoreLog;
|
import com.orion.ops.framework.web.core.annotation.RestWrapper;
|
||||||
// import com.orion.ops.framework.log.core.enums.IgnoreLogMode;
|
import com.orion.ops.module.asset.define.operator.HostOperatorType;
|
||||||
// import com.orion.ops.framework.web.core.annotation.RestWrapper;
|
import com.orion.ops.module.asset.service.HostService;
|
||||||
// import com.orion.ops.module.asset.define.operator.HostOperatorType;
|
import com.orion.ops.module.infra.api.DataGroupApi;
|
||||||
// import com.orion.ops.module.asset.entity.request.host.*;
|
import com.orion.ops.module.infra.entity.dto.data.DataGroupCreateDTO;
|
||||||
// import com.orion.ops.module.asset.entity.vo.HostConfigVO;
|
import com.orion.ops.module.infra.entity.dto.data.DataGroupDTO;
|
||||||
// import com.orion.ops.module.asset.entity.vo.HostVO;
|
import com.orion.ops.module.infra.entity.dto.data.DataGroupUpdateDTO;
|
||||||
// import com.orion.ops.module.asset.service.HostConfigService;
|
import com.orion.ops.module.infra.enums.DataGroupTypeEnum;
|
||||||
// import com.orion.ops.module.asset.service.HostService;
|
import io.swagger.v3.oas.annotations.Operation;
|
||||||
// import io.swagger.v3.oas.annotations.Operation;
|
import io.swagger.v3.oas.annotations.tags.Tag;
|
||||||
// import io.swagger.v3.oas.annotations.Parameter;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
// import io.swagger.v3.oas.annotations.tags.Tag;
|
import org.springframework.security.access.prepost.PreAuthorize;
|
||||||
// import lombok.extern.slf4j.Slf4j;
|
import org.springframework.validation.annotation.Validated;
|
||||||
// import org.springframework.security.access.prepost.PreAuthorize;
|
import org.springframework.web.bind.annotation.*;
|
||||||
// import org.springframework.validation.annotation.Validated;
|
|
||||||
// import org.springframework.web.bind.annotation.*;
|
import javax.annotation.Resource;
|
||||||
//
|
import java.util.List;
|
||||||
// import javax.annotation.Resource;
|
|
||||||
// import java.util.List;
|
/**
|
||||||
//
|
* 主机分组 api
|
||||||
// /**
|
*
|
||||||
// * 主机 api
|
* @author Jiahang Li
|
||||||
// *
|
* @version 1.0.0
|
||||||
// * @author Jiahang Li
|
* @since 2023-11-09 16:16
|
||||||
// * @version 1.0.0
|
*/
|
||||||
// * @since 2023-9-11 14:16
|
@Tag(name = "asset - 主机分组服务")
|
||||||
// */
|
@Slf4j
|
||||||
// @Tag(name = "asset - 主机服务")
|
@Validated
|
||||||
// @Slf4j
|
@RestWrapper
|
||||||
// @Validated
|
@RestController
|
||||||
// @RestWrapper
|
@RequestMapping("/asset/host-group")
|
||||||
// @RestController
|
@SuppressWarnings({"ELValidationInJSP", "SpringElInspection"})
|
||||||
// @RequestMapping("/asset/host-group")
|
public class HostGroupController {
|
||||||
// @SuppressWarnings({"ELValidationInJSP", "SpringElInspection"})
|
|
||||||
// public class HostGroupController {
|
@Resource
|
||||||
//
|
private HostService hostService;
|
||||||
// @Resource
|
|
||||||
// private HostService hostService;
|
@Resource
|
||||||
//
|
private DataGroupApi dataGroupApi;
|
||||||
// @Resource
|
|
||||||
// private HostConfigService hostConfigService;
|
// TODO 配置权限
|
||||||
//
|
// TODO 配置操作日志类型
|
||||||
// @OperatorLog(HostOperatorType.CREATE)
|
// TODO 拖拽
|
||||||
// @PostMapping("/create")
|
// TODO http
|
||||||
// @Operation(summary = "创建主机")
|
// TODO 聚合查询关联
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:create')")
|
|
||||||
// public Long createHost(@Validated @RequestBody HostCreateRequest request) {
|
@OperatorLog(HostOperatorType.CREATE)
|
||||||
// return hostService.createHost(request);
|
@PostMapping("/create")
|
||||||
// }
|
@Operation(summary = "创建主机分组")
|
||||||
//
|
@PreAuthorize("@ss.hasPermission('asset:host-group:create')")
|
||||||
// @OperatorLog(HostOperatorType.UPDATE)
|
public Long createHostGroup(@Validated @RequestBody DataGroupCreateDTO request) {
|
||||||
// @PutMapping("/update")
|
return dataGroupApi.createDataGroup(DataGroupTypeEnum.HOST, request);
|
||||||
// @Operation(summary = "通过 id 更新主机")
|
}
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:update')")
|
|
||||||
// public Integer updateHost(@Validated @RequestBody HostUpdateRequest request) {
|
@IgnoreLog(IgnoreLogMode.RET)
|
||||||
// return hostService.updateHostById(request);
|
@GetMapping("/tree")
|
||||||
// }
|
@Operation(summary = "创建主机分组")
|
||||||
//
|
@PreAuthorize("@ss.hasPermission('asset:host-group:query')")
|
||||||
// @IgnoreLog(IgnoreLogMode.RET)
|
public List<DataGroupDTO> queryHostGroup() {
|
||||||
// @GetMapping("/get")
|
return dataGroupApi.getDataGroupTree(DataGroupTypeEnum.HOST);
|
||||||
// @Operation(summary = "通过 id 查询主机")
|
}
|
||||||
// @Parameter(name = "id", description = "id", required = true)
|
|
||||||
// @Parameter(name = "extra", description = "是否查询额外信息")
|
@OperatorLog(HostOperatorType.UPDATE)
|
||||||
// @Parameter(name = "config", description = "是否查询配置信息")
|
@PutMapping("/rename")
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:query')")
|
@Operation(summary = "修改名称")
|
||||||
// public HostVO getHost(@RequestParam("id") Long id,
|
@PreAuthorize("@ss.hasPermission('asset:host-group:update')")
|
||||||
// @RequestParam(name = "extra", required = false) boolean extra,
|
public Integer updateHostGroupName(@Validated @RequestBody DataGroupUpdateDTO request) {
|
||||||
// @RequestParam(name = "config", required = false) boolean config) {
|
return dataGroupApi.renameDataGroup(request);
|
||||||
// HostQueryRequest request = new HostQueryRequest();
|
}
|
||||||
// request.setId(id);
|
|
||||||
// request.setExtra(extra);
|
@OperatorLog(HostOperatorType.DELETE)
|
||||||
// request.setConfig(config);
|
@DeleteMapping("/delete")
|
||||||
// return hostService.getHostById(request);
|
@Operation(summary = "删除主机分组")
|
||||||
// }
|
@PreAuthorize("@ss.hasPermission('asset:host-group:delete')")
|
||||||
//
|
public Integer deleteHostGroup(@RequestParam("id") Long id) {
|
||||||
// @IgnoreLog(IgnoreLogMode.RET)
|
return dataGroupApi.deleteDataGroupById(id);
|
||||||
// @PostMapping("/list")
|
}
|
||||||
// @Operation(summary = "查询主机")
|
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:query')")
|
}
|
||||||
// public List<HostVO> getHostList() {
|
|
||||||
// return hostService.getHostListByCache();
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// @IgnoreLog(IgnoreLogMode.RET)
|
|
||||||
// @PostMapping("/query")
|
|
||||||
// @Operation(summary = "分页查询主机")
|
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:query')")
|
|
||||||
// public DataGrid<HostVO> getHostPage(@Validated(Page.class) @RequestBody HostQueryRequest request) {
|
|
||||||
// return hostService.getHostPage(request);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// @OperatorLog(HostOperatorType.DELETE)
|
|
||||||
// @DeleteMapping("/delete")
|
|
||||||
// @Operation(summary = "通过 id 删除主机")
|
|
||||||
// @Parameter(name = "id", description = "id", required = true)
|
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:delete')")
|
|
||||||
// public Integer deleteHost(@RequestParam("id") Long id) {
|
|
||||||
// return hostService.deleteHostById(id);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// @IgnoreLog(IgnoreLogMode.RET)
|
|
||||||
// @GetMapping("/get-config")
|
|
||||||
// @Operation(summary = "查询主机配置")
|
|
||||||
// @Parameter(name = "hostId", description = "hostId", required = true)
|
|
||||||
// @Parameter(name = "type", description = "配置类型", required = true)
|
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:query')")
|
|
||||||
// public HostConfigVO getHostConfig(@RequestParam("hostId") Long hostId,
|
|
||||||
// @RequestParam(name = "type") String type) {
|
|
||||||
// return hostConfigService.getHostConfig(hostId, type);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// @IgnoreLog(IgnoreLogMode.RET)
|
|
||||||
// @GetMapping("/get-config-all")
|
|
||||||
// @Operation(summary = "查询主机配置 - 全部")
|
|
||||||
// @Parameter(name = "hostId", description = "hostId", required = true)
|
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:query')")
|
|
||||||
// public List<HostConfigVO> getHostConfig(@RequestParam("hostId") Long hostId) {
|
|
||||||
// return hostConfigService.getHostConfig(hostId);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// @OperatorLog(HostOperatorType.UPDATE_CONFIG)
|
|
||||||
// @PutMapping("/update-config")
|
|
||||||
// @Operation(summary = "更新主机配置")
|
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:update-config')")
|
|
||||||
// public Integer updateHostConfig(@Validated @RequestBody HostConfigUpdateRequest request) {
|
|
||||||
// return hostConfigService.updateHostConfig(request);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// @OperatorLog(HostOperatorType.UPDATE_CONFIG_STATUS)
|
|
||||||
// @PutMapping("/update-config-status")
|
|
||||||
// @Operation(summary = "更新主机配置状态")
|
|
||||||
// @PreAuthorize("@ss.hasPermission('asset:host:update-config')")
|
|
||||||
// public Integer updateHostConfigStatus(@Validated @RequestBody HostConfigUpdateStatusRequest request) {
|
|
||||||
// return hostConfigService.updateHostConfigStatus(request);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
|
|||||||
114
orion-ops-ui/src/api/asset/host-group-rel.ts
Normal file
114
orion-ops-ui/src/api/asset/host-group-rel.ts
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
import type { DataGrid, Pagination } from '@/types/global';
|
||||||
|
import type { TableData } from '@arco-design/web-vue/es/table/interface';
|
||||||
|
import axios from 'axios';
|
||||||
|
import qs from 'query-string';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 数据分组关联创建请求
|
||||||
|
*/
|
||||||
|
export interface DataGroupRelCreateRequest {
|
||||||
|
groupId?: number;
|
||||||
|
relId?: number;
|
||||||
|
type?: string;
|
||||||
|
sort?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 数据分组关联更新请求
|
||||||
|
*/
|
||||||
|
export interface DataGroupRelUpdateRequest extends DataGroupRelCreateRequest {
|
||||||
|
id?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 数据分组关联查询请求
|
||||||
|
*/
|
||||||
|
export interface DataGroupRelQueryRequest extends Pagination {
|
||||||
|
searchValue?: string;
|
||||||
|
id?: number;
|
||||||
|
groupId?: number;
|
||||||
|
relId?: number;
|
||||||
|
type?: string;
|
||||||
|
sort?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 数据分组关联查询响应
|
||||||
|
*/
|
||||||
|
export interface DataGroupRelQueryResponse extends TableData {
|
||||||
|
id: number;
|
||||||
|
groupId: number;
|
||||||
|
relId: number;
|
||||||
|
type: string;
|
||||||
|
sort: number;
|
||||||
|
createTime: number;
|
||||||
|
updateTime: number;
|
||||||
|
creator: string;
|
||||||
|
updater: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 创建数据分组关联
|
||||||
|
*/
|
||||||
|
export function createDataGroupRel(request: DataGroupRelCreateRequest) {
|
||||||
|
return axios.post('/infra/data-group-rel/create', request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 更新数据分组关联
|
||||||
|
*/
|
||||||
|
export function updateDataGroupRel(request: DataGroupRelUpdateRequest) {
|
||||||
|
return axios.put('/infra/data-group-rel/update', request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 查询数据分组关联
|
||||||
|
*/
|
||||||
|
export function getDataGroupRel(id: number) {
|
||||||
|
return axios.get<DataGroupRelQueryResponse>('/infra/data-group-rel/get', { params: { id } });
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 批量查询数据分组关联
|
||||||
|
*/
|
||||||
|
export function batchGetDataGroupRelList(idList: Array<number>) {
|
||||||
|
return axios.get<DataGroupRelQueryResponse[]>('/infra/data-group-rel/batch-get', {
|
||||||
|
params: { idList },
|
||||||
|
paramsSerializer: params => {
|
||||||
|
return qs.stringify(params, { arrayFormat: 'comma' });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 查询全部数据分组关联
|
||||||
|
*/
|
||||||
|
export function getDataGroupRelList(request: DataGroupRelQueryRequest) {
|
||||||
|
return axios.post<Array<DataGroupRelQueryResponse>>('/infra/data-group-rel/list', request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 分页查询数据分组关联
|
||||||
|
*/
|
||||||
|
export function getDataGroupRelPage(request: DataGroupRelQueryRequest) {
|
||||||
|
return axios.post<DataGrid<DataGroupRelQueryResponse>>('/infra/data-group-rel/query', request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除数据分组关联
|
||||||
|
*/
|
||||||
|
export function deleteDataGroupRel(id: number) {
|
||||||
|
return axios.delete('/infra/data-group-rel/delete', { params: { id } });
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 批量删除数据分组关联
|
||||||
|
*/
|
||||||
|
export function batchDeleteDataGroupRel(idList: Array<number>) {
|
||||||
|
return axios.delete('/infra/data-group-rel/batch-delete', {
|
||||||
|
params: { idList },
|
||||||
|
paramsSerializer: params => {
|
||||||
|
return qs.stringify(params, { arrayFormat: 'comma' });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
67
orion-ops-ui/src/api/asset/host-group.ts
Normal file
67
orion-ops-ui/src/api/asset/host-group.ts
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 主机分组创建请求
|
||||||
|
*/
|
||||||
|
export interface HostGroupCreateRequest {
|
||||||
|
parentId?: number;
|
||||||
|
name?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 主机分组更新请求
|
||||||
|
*/
|
||||||
|
export interface HostGroupUpdateRequest extends HostGroupCreateRequest {
|
||||||
|
id?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 主机分组查询请求
|
||||||
|
*/
|
||||||
|
export interface HostGroupQueryRequest {
|
||||||
|
searchValue?: string;
|
||||||
|
id?: number;
|
||||||
|
parentId?: number;
|
||||||
|
name?: string;
|
||||||
|
type?: string;
|
||||||
|
sort?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 主机分组查询响应
|
||||||
|
*/
|
||||||
|
export interface HostGroupQueryResponse {
|
||||||
|
id: number;
|
||||||
|
parentId: number;
|
||||||
|
name: string;
|
||||||
|
sort: number;
|
||||||
|
children: Array<HostGroupQueryResponse>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 创建主机分组
|
||||||
|
*/
|
||||||
|
export function createHostGroup(request: HostGroupCreateRequest) {
|
||||||
|
return axios.post('/asset/host-group/create', request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 更新主机分组
|
||||||
|
*/
|
||||||
|
export function updateHostGroup(request: HostGroupUpdateRequest) {
|
||||||
|
return axios.put('/asset/host-group/update', request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 查询全部主机分组树
|
||||||
|
*/
|
||||||
|
export function getHostGroupTree() {
|
||||||
|
return axios.get<Array<HostGroupQueryResponse>>('/asset/host-group/tree');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除主机分组
|
||||||
|
*/
|
||||||
|
export function deleteHostGroup(id: number) {
|
||||||
|
return axios.delete('/asset/host-group/delete', { params: { id } });
|
||||||
|
}
|
||||||
@@ -13,9 +13,10 @@ body {
|
|||||||
-moz-osx-font-smoothing: grayscale;
|
-moz-osx-font-smoothing: grayscale;
|
||||||
-webkit-font-smoothing: antialiased;
|
-webkit-font-smoothing: antialiased;
|
||||||
--color-scrollbar-track: var(--color-neutral-1);
|
--color-scrollbar-track: var(--color-neutral-1);
|
||||||
--color-scrollbar-thumb: #959FAB;
|
--color-scrollbar-thumb: var(--color-neutral-5);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// -- echarts
|
||||||
.echarts-tooltip-diy {
|
.echarts-tooltip-diy {
|
||||||
background: linear-gradient(304.17deg,
|
background: linear-gradient(304.17deg,
|
||||||
rgba(253, 254, 255, 0.6) -6.04%,
|
rgba(253, 254, 255, 0.6) -6.04%,
|
||||||
@@ -67,31 +68,10 @@ body {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.general-card {
|
// -- arco
|
||||||
border-radius: 4px;
|
.arco-table-td-content {
|
||||||
border: none;
|
color: rgba(var(--gray-9), .95);
|
||||||
|
font-size: 13px;
|
||||||
& > .arco-card-header {
|
|
||||||
height: auto;
|
|
||||||
padding: 16px;
|
|
||||||
border: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
& > .arco-card-body {
|
|
||||||
padding: 0 16px 16px 16px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.a-query-header-grid {
|
|
||||||
.arco-grid-item {
|
|
||||||
&:last-child {
|
|
||||||
display: flex !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.split-line {
|
|
||||||
border-color: rgb(var(--gray-2));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.arco-table-cell {
|
.arco-table-cell {
|
||||||
@@ -120,6 +100,31 @@ body {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.a-query-header-grid {
|
||||||
|
.arco-grid-item {
|
||||||
|
&:last-child {
|
||||||
|
display: flex !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -- card
|
||||||
|
.general-card {
|
||||||
|
border-radius: 4px;
|
||||||
|
border: none;
|
||||||
|
|
||||||
|
& > .arco-card-header {
|
||||||
|
height: auto;
|
||||||
|
padding: 16px;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
& > .arco-card-body {
|
||||||
|
padding: 0 16px 16px 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -- click-icon
|
||||||
.click-icon-wrapper {
|
.click-icon-wrapper {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
@@ -136,6 +141,113 @@ body {
|
|||||||
background: var(--color-fill-3);
|
background: var(--color-fill-3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// -- element
|
||||||
|
.split-line {
|
||||||
|
border-color: rgb(var(--gray-2));
|
||||||
|
}
|
||||||
|
|
||||||
|
.usn {
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pointer {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx0 {
|
||||||
|
margin: 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx2 {
|
||||||
|
margin: 0 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx4 {
|
||||||
|
margin: 0 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ml4 {
|
||||||
|
margin-left: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mr4 {
|
||||||
|
margin-right: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mt4 {
|
||||||
|
margin-top: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mb4 {
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx4 {
|
||||||
|
margin-left: 4px;
|
||||||
|
margin-right: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.my4 {
|
||||||
|
margin-top: 4px;
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ml8 {
|
||||||
|
margin-left: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mr8 {
|
||||||
|
margin-right: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mt8 {
|
||||||
|
margin-top: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mb8 {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx8 {
|
||||||
|
margin-left: 8px;
|
||||||
|
margin-right: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.my8 {
|
||||||
|
margin-top: 8px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.ml16 {
|
||||||
|
margin-left: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mr16 {
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mt16 {
|
||||||
|
margin-top: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mb16 {
|
||||||
|
margin-bottom: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mx16 {
|
||||||
|
margin-left: 16px;
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.my16 {
|
||||||
|
margin-top: 16px;
|
||||||
|
margin-bottom: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
.copy-left {
|
.copy-left {
|
||||||
color: rgb(var(--arcoblue-6));
|
color: rgb(var(--arcoblue-6));
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -150,6 +262,7 @@ body {
|
|||||||
color: rgb(var(--red-6));
|
color: rgb(var(--red-6));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// -- 滚动条
|
||||||
#app {
|
#app {
|
||||||
-webkit-font-smoothing: antialiased;
|
-webkit-font-smoothing: antialiased;
|
||||||
-moz-osx-font-smoothing: grayscale;
|
-moz-osx-font-smoothing: grayscale;
|
||||||
|
|||||||
@@ -1,10 +1,13 @@
|
|||||||
|
// -- container
|
||||||
.layout-container {
|
.layout-container {
|
||||||
background-color: var(--color-fill-2);
|
background-color: var(--color-fill-2);
|
||||||
padding: 16px;
|
padding: 16px;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// -- table-view
|
||||||
.table-search-card {
|
.table-search-card {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding-top: 16px;
|
padding-top: 16px;
|
||||||
@@ -64,10 +67,12 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// -- modal
|
||||||
.modal-form {
|
.modal-form {
|
||||||
padding: 24px 20px 4px 20px;
|
padding: 24px 20px 4px 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// -- card-view
|
||||||
.card-list-item {
|
.card-list-item {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -105,109 +110,8 @@
|
|||||||
padding: 18px 24px 14px 24px;
|
padding: 18px 24px 14px 24px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.arco-table-td-content {
|
// -- card
|
||||||
color: rgba(var(--gray-9), .95);
|
.simple-card {
|
||||||
font-size: 13px;
|
background: var(--color-bg-2);
|
||||||
}
|
border-radius: 4px;
|
||||||
|
|
||||||
.usn {
|
|
||||||
user-select: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.hide {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pointer {
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx0 {
|
|
||||||
margin: 0 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx2 {
|
|
||||||
margin: 0 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx4 {
|
|
||||||
margin: 0 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.ml4 {
|
|
||||||
margin-left: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mr4 {
|
|
||||||
margin-right: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt4 {
|
|
||||||
margin-top: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mb4 {
|
|
||||||
margin-bottom: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx4 {
|
|
||||||
margin-left: 4px;
|
|
||||||
margin-right: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.my4 {
|
|
||||||
margin-top: 4px;
|
|
||||||
margin-bottom: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.ml8 {
|
|
||||||
margin-left: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mr8 {
|
|
||||||
margin-right: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt8 {
|
|
||||||
margin-top: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mb8 {
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx8 {
|
|
||||||
margin-left: 8px;
|
|
||||||
margin-right: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.my8 {
|
|
||||||
margin-top: 8px;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.ml16 {
|
|
||||||
margin-left: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mr16 {
|
|
||||||
margin-right: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mt16 {
|
|
||||||
margin-top: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mb16 {
|
|
||||||
margin-bottom: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx16 {
|
|
||||||
margin-left: 16px;
|
|
||||||
margin-right: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.my16 {
|
|
||||||
margin-top: 16px;
|
|
||||||
margin-bottom: 16px;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,17 @@
|
|||||||
<template>
|
<template>
|
||||||
<a-layout-footer class="footer">
|
<a-layout-footer class="footer">
|
||||||
<div class="footer-text">
|
<a-space direction="vertical" size="medium">
|
||||||
项目地址 <a target="_blank" href="https://github.com/lijiahangmax/orion-ops-pro">github</a> - <a
|
<a-space size="large">
|
||||||
target="_blank" href="https://gitee.com/lijiahangmax/orion-ops-pro">gitee</a>
|
<a target="_blank" href="https://github.com/lijiahangmax/orion-ops-pro">官网</a>
|
||||||
<a class="license" target="_blank" href="https://github.com/lijiahangmax/orion-ops-pro/blob/main/LICENSE">License - Apache 2.0</a>
|
<a target="_blank" href="https://github.com/lijiahangmax/orion-ops-pro">文档</a>
|
||||||
</div>
|
<a target="_blank" href="https://github.com/lijiahangmax/orion-ops-pro">教程</a>
|
||||||
|
<a target="_blank" href="https://github.com/lijiahangmax/orion-ops-pro">github</a>
|
||||||
|
<a target="_blank" href="https://gitee.com/lijiahangmax/orion-ops-pro">gitee</a>
|
||||||
|
</a-space>
|
||||||
|
<span class="copyright">
|
||||||
|
Copyright<icon-copyright /> 2023 By OrionOpsPro
|
||||||
|
</span>
|
||||||
|
</a-space>
|
||||||
</a-layout-footer>
|
</a-layout-footer>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -16,21 +23,17 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
height: 40px;
|
height: 80px;
|
||||||
color: var(--color-text-2);
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
|
||||||
&-text {
|
a {
|
||||||
text-wrap: none;
|
text-decoration: none;
|
||||||
|
color: rgb(var(--primary-6));
|
||||||
a {
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.license {
|
|
||||||
display: inline-block;
|
|
||||||
margin-left: 16px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.copyright {
|
||||||
|
color: var(--color-text-3)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -30,10 +30,10 @@ use([
|
|||||||
|
|
||||||
export default {
|
export default {
|
||||||
install(Vue: App) {
|
install(Vue: App) {
|
||||||
Vue.component('Chart', Chart);
|
Vue.component('chart', Chart);
|
||||||
Vue.component('Breadcrumb', Breadcrumb);
|
Vue.component('breadcrumb', Breadcrumb);
|
||||||
Vue.component('a-query-header', AQueryHeader);
|
Vue.component('a-query-header', AQueryHeader);
|
||||||
Vue.component('card-list', CardList);
|
Vue.component('card-list', CardList);
|
||||||
Vue.component('Editor', Editor);
|
Vue.component('editor', Editor);
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -136,6 +136,7 @@
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// fixme 提成公共方法
|
||||||
// 寻找当前节点
|
// 寻找当前节点
|
||||||
const findNode = (id: number, arr: Array<MenuQueryResponse>): MenuQueryResponse | undefined => {
|
const findNode = (id: number, arr: Array<MenuQueryResponse>): MenuQueryResponse | undefined => {
|
||||||
for (let node of arr) {
|
for (let node of arr) {
|
||||||
|
|||||||
@@ -6,16 +6,16 @@
|
|||||||
<!-- 信息部分 -->
|
<!-- 信息部分 -->
|
||||||
<div class="card-list-info">
|
<div class="card-list-info">
|
||||||
<!-- 路由面包屑 -->
|
<!-- 路由面包屑 -->
|
||||||
<Breadcrumb />
|
<breadcrumb />
|
||||||
<!-- 分页部分 -->
|
<!-- 分页部分 -->
|
||||||
<div class="pagination-wrapper">
|
<div class="pagination-wrapper">
|
||||||
<a-pagination v-if="pagination"
|
<a-pagination v-if="pagination"
|
||||||
size="mini"
|
size="mini"
|
||||||
v-model:current="pagination.current"
|
v-model:current="(pagination as PaginationProps).current"
|
||||||
v-model:page-size="pagination.pageSize"
|
v-model:page-size="(pagination as PaginationProps).pageSize"
|
||||||
v-bind="pagination"
|
v-bind="pagination"
|
||||||
:auto-adjust="false"
|
:auto-adjust="false"
|
||||||
@change="page => emits('pageChange', page, pagination.pageSize)"
|
@change="page => emits('pageChange', page, (pagination as PaginationProps).pageSize)"
|
||||||
@page-size-change="limit => emits('pageChange', 1, limit)" />
|
@page-size-change="limit => emits('pageChange', 1, limit)" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
<div v-if="navbar" class="layout-navbar">
|
<div v-if="navbar" class="layout-navbar">
|
||||||
<NavBar />
|
<NavBar />
|
||||||
</div>
|
</div>
|
||||||
<a-layout>
|
<a-layout style="flex-direction: row;">
|
||||||
<!-- 左侧菜单栏 -->
|
<!-- 左侧菜单栏 -->
|
||||||
<a-layout-sider v-if="renderMenu"
|
<a-layout-sider v-if="renderMenu"
|
||||||
v-show="!hideMenu"
|
v-show="!hideMenu"
|
||||||
@@ -33,10 +33,13 @@
|
|||||||
</a-drawer>
|
</a-drawer>
|
||||||
<!-- body -->
|
<!-- body -->
|
||||||
<a-layout class="layout-content" :style="paddingStyle">
|
<a-layout class="layout-content" :style="paddingStyle">
|
||||||
|
<!-- 页签 -->
|
||||||
<TabBar v-if="appStore.tabBar" />
|
<TabBar v-if="appStore.tabBar" />
|
||||||
|
<!-- 页面 -->
|
||||||
<a-layout-content>
|
<a-layout-content>
|
||||||
<PageLayout />
|
<PageLayout />
|
||||||
</a-layout-content>
|
</a-layout-content>
|
||||||
|
<!-- 页脚 -->
|
||||||
<Footer v-if="footer" />
|
<Footer v-if="footer" />
|
||||||
</a-layout>
|
</a-layout>
|
||||||
</a-layout>
|
</a-layout>
|
||||||
@@ -81,9 +84,7 @@
|
|||||||
return { ...paddingLeft, ...paddingTop };
|
return { ...paddingLeft, ...paddingTop };
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
// 设置菜单展开状态
|
||||||
* 设置菜单展开状态
|
|
||||||
*/
|
|
||||||
const setCollapsed = (val: boolean) => {
|
const setCollapsed = (val: boolean) => {
|
||||||
if (!isInit.value) return;
|
if (!isInit.value) return;
|
||||||
appStore.updateSettings({ menuCollapse: val });
|
appStore.updateSettings({ menuCollapse: val });
|
||||||
@@ -117,7 +118,7 @@
|
|||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
z-index: 100;
|
z-index: 120;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: @nav-size-height;
|
height: @nav-size-height;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,6 +7,10 @@ const ASSET: AppRouteRecordRaw = {
|
|||||||
component: DEFAULT_LAYOUT,
|
component: DEFAULT_LAYOUT,
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
|
name: 'assetHostGroup',
|
||||||
|
path: '/asset/host-group',
|
||||||
|
component: () => import('@/views/asset/host-group/index.vue'),
|
||||||
|
}, {
|
||||||
name: 'assetHost',
|
name: 'assetHost',
|
||||||
path: '/asset/host',
|
path: '/asset/host',
|
||||||
component: () => import('@/views/asset/host/index.vue'),
|
component: () => import('@/views/asset/host/index.vue'),
|
||||||
|
|||||||
@@ -0,0 +1,278 @@
|
|||||||
|
<template>
|
||||||
|
<div class="tree-container">
|
||||||
|
<a-tree
|
||||||
|
:blockNode="true"
|
||||||
|
:draggable="props.editMode"
|
||||||
|
:data="treeData">
|
||||||
|
|
||||||
|
<template #title="node">
|
||||||
|
<template v-if="node.editable">
|
||||||
|
<a-input size="mini"
|
||||||
|
v-model="currName"
|
||||||
|
:max-length="32"
|
||||||
|
:disabled="node.loading"
|
||||||
|
autofocus
|
||||||
|
@change="() => saveNode(node.key)">
|
||||||
|
<template #suffix>
|
||||||
|
<!-- 加载中 -->
|
||||||
|
<icon-loading v-if="node.loading" />
|
||||||
|
<!-- 保存 -->
|
||||||
|
<icon-check v-else
|
||||||
|
class="pointer"
|
||||||
|
title="保存"
|
||||||
|
@click="saveNode(node.key)" />
|
||||||
|
</template>
|
||||||
|
</a-input>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<span class="node-title" v-else>
|
||||||
|
{{ node.title }}
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
<!-- 操作图标 -->
|
||||||
|
<template #drag-icon="{ node }">
|
||||||
|
<a-space v-if="!node.editable">
|
||||||
|
<icon-edit class="tree-icon"
|
||||||
|
title="重命名"
|
||||||
|
@click="rename(node.title, node.key)" />
|
||||||
|
<icon-delete class="tree-icon"
|
||||||
|
title="删除"
|
||||||
|
@click="rename(node.title, node.key)" />
|
||||||
|
<icon-plus class="tree-icon"
|
||||||
|
title="新增"
|
||||||
|
@click="rename(node.title, node.key)" />
|
||||||
|
</a-space>
|
||||||
|
</template>
|
||||||
|
</a-tree>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
export default {
|
||||||
|
name: 'host-group-tree'
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { nextTick, ref } from 'vue';
|
||||||
|
import { TagProps } from '@/store/modules/tab-bar/types';
|
||||||
|
import { TreeNodeData } from '@arco-design/web-vue';
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
editMode: Boolean
|
||||||
|
});
|
||||||
|
|
||||||
|
const currName = ref();
|
||||||
|
|
||||||
|
|
||||||
|
function sleep(ms) {
|
||||||
|
return new Promise(resolve => setTimeout(resolve, ms));
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存节点
|
||||||
|
const saveNode = async (key: string) => {
|
||||||
|
// 寻找节点
|
||||||
|
const node = findNode(key, treeData.value);
|
||||||
|
if (currName.value) {
|
||||||
|
node.loading = true;
|
||||||
|
try {
|
||||||
|
if (key.startsWith('create')) {
|
||||||
|
// 调用创建 api
|
||||||
|
await sleep(340);
|
||||||
|
node.key = 'await id';
|
||||||
|
} else {
|
||||||
|
// 调用重命名 api
|
||||||
|
await sleep(340);
|
||||||
|
}
|
||||||
|
node.title = currName.value;
|
||||||
|
} catch (e) {
|
||||||
|
} finally {
|
||||||
|
node.loading = false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (key.startsWith('create')) {
|
||||||
|
// 寻找父节点
|
||||||
|
// 移除子节点
|
||||||
|
}
|
||||||
|
}
|
||||||
|
node.editable = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
// 重命名
|
||||||
|
const rename = (title: string, key: string) => {
|
||||||
|
const node = findNode(key, treeData.value);
|
||||||
|
currName.value = title;
|
||||||
|
node.editable = true;
|
||||||
|
};
|
||||||
|
|
||||||
|
// 寻找当前节点
|
||||||
|
const findNode = (id: string, arr: Array<TreeNodeData>): TreeNodeData | undefined => {
|
||||||
|
for (let node of arr) {
|
||||||
|
if (node.key === id) {
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 寻找子级
|
||||||
|
for (let node of arr) {
|
||||||
|
if (node?.children?.length) {
|
||||||
|
const inChildNode = findNode(id, node.children);
|
||||||
|
if (inChildNode) {
|
||||||
|
return inChildNode;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return undefined;
|
||||||
|
};
|
||||||
|
|
||||||
|
function onIconClick(node: any) {
|
||||||
|
const children = node.children || [];
|
||||||
|
children.push({
|
||||||
|
title: 'new tree node',
|
||||||
|
key: node.key + '-' + (children.length + 1)
|
||||||
|
});
|
||||||
|
node.children = children;
|
||||||
|
|
||||||
|
treeData.value = [...treeData.value];
|
||||||
|
}
|
||||||
|
|
||||||
|
const treeData = ref(
|
||||||
|
[
|
||||||
|
{
|
||||||
|
title: 'Trunk',
|
||||||
|
key: '0-0',
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-0-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Branch',
|
||||||
|
key: '0-0-2',
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-0-2-1'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Trunk',
|
||||||
|
key: '0-1',
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
title: 'Branch',
|
||||||
|
key: '0-1-1',
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-1-11',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-1-12',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-1-13',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-1-41',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-1-51',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-1-61',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-17-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-81-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-19-1-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-10-1-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-111-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-21-1-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-31-1-1',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '40-1-1-2',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
title: 'Leaf',
|
||||||
|
key: '0-1-2',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
]
|
||||||
|
);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="less" scoped>
|
||||||
|
.tree-container {
|
||||||
|
min-width: 100%;
|
||||||
|
width: max-content;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(.arco-tree-node-title) {
|
||||||
|
padding-right: 48px;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--color-fill-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.arco-tree-node-title-text {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-title {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-handler {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(.arco-tree-node-selected) {
|
||||||
|
background-color: var(--color-fill-2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tree-icon {
|
||||||
|
font-size: 12px;
|
||||||
|
color: rgb(var(--primary-6));
|
||||||
|
}
|
||||||
|
|
||||||
|
.drag-icon {
|
||||||
|
padding-left: -8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,141 @@
|
|||||||
|
<template>
|
||||||
|
<div class="view-container">
|
||||||
|
<!-- 左侧菜单 -->
|
||||||
|
<div class="simple-card tree-card">
|
||||||
|
<!-- 主机分组标题 -->
|
||||||
|
<div class="tree-card-header">
|
||||||
|
<!-- 标题 -->
|
||||||
|
<div class="tree-card-title">
|
||||||
|
主机菜单
|
||||||
|
</div>
|
||||||
|
<div class="tree-card-switch">
|
||||||
|
<a-switch type="round"
|
||||||
|
v-model="editMode"
|
||||||
|
checked-text="编辑模式"
|
||||||
|
unchecked-text="授权模式" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 主机分组树 -->
|
||||||
|
<div class="tree-card-main">
|
||||||
|
<host-group-tree ref="tree" :editMode="editMode" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 身体部分 -->
|
||||||
|
<div class="view-body">
|
||||||
|
<div class="simple-card fixed-header">
|
||||||
|
<!-- 头部左侧 -->
|
||||||
|
<a-space>
|
||||||
|
<a-select placeholder="选择角色" />
|
||||||
|
<a-select placeholder="选择用户" />
|
||||||
|
</a-space>
|
||||||
|
<!-- 头部右侧 -->
|
||||||
|
<a-space>
|
||||||
|
<!-- 配置 -->
|
||||||
|
<a-button type="primary">
|
||||||
|
配置
|
||||||
|
<template #icon>
|
||||||
|
<icon-layers />
|
||||||
|
</template>
|
||||||
|
</a-button>
|
||||||
|
<!-- 授权 -->
|
||||||
|
<a-button type="primary">
|
||||||
|
授权
|
||||||
|
<template #icon>
|
||||||
|
<icon-safe />
|
||||||
|
</template>
|
||||||
|
</a-button>
|
||||||
|
</a-space>
|
||||||
|
</div>
|
||||||
|
<!-- 右侧数据 -->
|
||||||
|
<div class="simple-card data-content">
|
||||||
|
右侧数据
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
export default {
|
||||||
|
name: 'host-group-view'
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import HostGroupTree from './host-group-tree.vue';
|
||||||
|
import { ref } from 'vue';
|
||||||
|
|
||||||
|
const editMode = ref(true);
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="less" scoped>
|
||||||
|
@tree-width: 50%;
|
||||||
|
|
||||||
|
.view-container {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tree-card {
|
||||||
|
margin-right: 16px;
|
||||||
|
width: @tree-width;
|
||||||
|
height: 100%;
|
||||||
|
position: absolute;
|
||||||
|
|
||||||
|
&-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 16px;
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 44px;
|
||||||
|
border-bottom: 1px var(--color-border-2) solid;
|
||||||
|
}
|
||||||
|
|
||||||
|
&-title {
|
||||||
|
color: rgba(var(--gray-10), .95);
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: 600;
|
||||||
|
overflow: hidden;
|
||||||
|
white-space: nowrap;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
&-main {
|
||||||
|
padding: 8px 8px 8px 16px;
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: calc(100% - 48px);
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.view-body {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
width: calc(100% - @tree-width - 16px);
|
||||||
|
position: absolute;
|
||||||
|
left: calc(@tree-width + 16px);
|
||||||
|
|
||||||
|
.fixed-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
|
||||||
|
margin-bottom: 16px;
|
||||||
|
padding: 8px 16px;
|
||||||
|
height: 48px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.data-content {
|
||||||
|
display: flex;
|
||||||
|
padding: 16px;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
</style>
|
||||||
72
orion-ops-ui/src/views/asset/host-group/index.vue
Normal file
72
orion-ops-ui/src/views/asset/host-group/index.vue
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
<template>
|
||||||
|
<div class="index-container" v-if="render">
|
||||||
|
<host-group-view />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
export default {
|
||||||
|
name: 'assetHostGroup'
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { Message } from '@arco-design/web-vue';
|
||||||
|
|
||||||
|
import { computed, ref, onBeforeMount, onUnmounted } from 'vue';
|
||||||
|
import { useAppStore, useCacheStore, useDictStore } from '@/store';
|
||||||
|
import HostGroupView from './components/host-group-view.vue';
|
||||||
|
|
||||||
|
const render = ref(false);
|
||||||
|
const table = ref();
|
||||||
|
const card = ref();
|
||||||
|
const modal = ref();
|
||||||
|
const config = ref();
|
||||||
|
const appStore = useAppStore();
|
||||||
|
const cacheStore = useCacheStore();
|
||||||
|
|
||||||
|
// 添加回调
|
||||||
|
const modalAddCallback = () => {
|
||||||
|
table.value.addedCallback();
|
||||||
|
};
|
||||||
|
|
||||||
|
// 修改回调
|
||||||
|
const modalUpdateCallback = () => {
|
||||||
|
table.value.updatedCallback();
|
||||||
|
};
|
||||||
|
|
||||||
|
// 加载 tags
|
||||||
|
const loadTags = async () => {
|
||||||
|
try {
|
||||||
|
// 设置到缓存
|
||||||
|
// cacheStore.set('hostTags', data);
|
||||||
|
} catch {
|
||||||
|
Message.error('tag加载失败');
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
onBeforeMount(async () => {
|
||||||
|
// 加载角色列表
|
||||||
|
|
||||||
|
// 加载用户列表
|
||||||
|
|
||||||
|
// 加载主机列表
|
||||||
|
|
||||||
|
render.value = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 卸载时清除 cache
|
||||||
|
onUnmounted(() => {
|
||||||
|
cacheStore.reset('user', 'roles', 'hosts');
|
||||||
|
});
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="less" scoped>
|
||||||
|
.index-container {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
padding: 16px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -50,8 +50,8 @@
|
|||||||
<!-- 右侧操作 -->
|
<!-- 右侧操作 -->
|
||||||
<div class="table-right-bar-handle">
|
<div class="table-right-bar-handle">
|
||||||
<a-space>
|
<a-space>
|
||||||
<!-- 仅看收藏 -->
|
<!-- 仅看收藏 fixme 去掉 -->
|
||||||
<a-checkbox v-model="formModel.favorite" @change="fetchTableData()">
|
<a-checkbox v-if="false" v-model="formModel.favorite" @change="fetchTableData()">
|
||||||
<template #checkbox="{ checked }">
|
<template #checkbox="{ checked }">
|
||||||
<a-tag :checked="checked"
|
<a-tag :checked="checked"
|
||||||
class="only-favorite"
|
class="only-favorite"
|
||||||
|
|||||||
@@ -6,30 +6,48 @@ export const RoleStatus = {
|
|||||||
ENABLED: 1,
|
ENABLED: 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// 查询操作
|
||||||
|
const queryType = ['query', 'view'];
|
||||||
|
const addType = ['add', 'create'];
|
||||||
|
const updateType = ['update', 'modify'];
|
||||||
|
const deleteType = ['delete', 'remove'];
|
||||||
|
const standardRead = [...queryType];
|
||||||
|
const standardWrite = [...addType, ...updateType, ...deleteType];
|
||||||
|
|
||||||
// 快速分配菜单操作
|
// 快速分配菜单操作
|
||||||
export const quickGrantMenuOperator = [
|
export const quickGrantMenuOperator = [
|
||||||
{
|
{
|
||||||
name: '',
|
name: '',
|
||||||
rule: undefined
|
rule: undefined
|
||||||
|
}, {
|
||||||
|
name: '常规读操作',
|
||||||
|
rule: (perm: string) => {
|
||||||
|
return !!standardRead.find(s => perm.includes(s));
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
name: '常规写操作',
|
||||||
|
rule: (perm: string) => {
|
||||||
|
return !!standardWrite.find(s => perm.includes(s));
|
||||||
|
}
|
||||||
}, {
|
}, {
|
||||||
name: '查询',
|
name: '查询',
|
||||||
rule: (perm: string) => {
|
rule: (perm: string) => {
|
||||||
return perm.includes('query') || perm.includes('view');
|
return !!queryType.find(s => perm.includes(s));
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
name: '新增',
|
name: '新增',
|
||||||
rule: (perm: string) => {
|
rule: (perm: string) => {
|
||||||
return perm.includes('add') || perm.includes('create');
|
return !!addType.find(s => perm.includes(s));
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
name: '修改',
|
name: '修改',
|
||||||
rule: (perm: string) => {
|
rule: (perm: string) => {
|
||||||
return perm.includes('update') || perm.includes('modify');
|
return !!updateType.find(s => perm.includes(s));
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
name: '删除',
|
name: '删除',
|
||||||
rule: (perm: string) => {
|
rule: (perm: string) => {
|
||||||
return perm.includes('delete') || perm.includes('remove');
|
return !!deleteType.find(s => perm.includes(s));
|
||||||
}
|
}
|
||||||
}, {
|
}, {
|
||||||
name: '导入',
|
name: '导入',
|
||||||
|
|||||||
Reference in New Issue
Block a user