区域选择
区域选择是通过插件CellRangeSelector实现的,默认不支持跨冻结列进行选择,修正了选择,支持跨冻结列,代码如下,通过判断选择的起点和终点所落在的冻结范围进行计算,如从左往右进行复制,此时需要判拷贝的列数是否超过了冻结列,如果超过冻结了,此时需要在右侧(canvas)中进行绘制,否则在原canvas中绘制。从右往左逻辑同。
function show(range, cfg) {
var canvas = cfg.canvas;
var start = cfg.start;
var end = cfg.end;
var from = grid.getCellNodeBox(range.fromRow, range.fromCell);
var to = grid.getCellNodeBox(range.toRow, range.toCell);
// range left to right
var directLR = start <= end;
// frozenColumn temp var
var frozenColumn = grid.getOptions().frozenColumn;
var lR = false;
// lR true condition
if (directLR) {
lR = start <= frozenColumn && end > frozenColumn;
} else {
lR = start > frozenColumn && end <= frozenColumn;
}
if (!_elem) {
_elem = $("<div></div>", { css: options.selectionCss })
.addClass(options.selectionCssClass)
.css("position", "absolute")
.appendTo(grid.getActiveCanvasNode());
_borderWidth = _elem.css("border-left-width");
}
// lr ,create new element .
if (lR && !_rElem) {
_rElem = $("<div></div>", { css: options.selectionCss })
.addClass(options.selectionCssClass)
.css("position", "absolute")
.appendTo(canvas);
}
// ! lr ,remove _rElem
if (!lR && _rElem) {
_rElem.remove();
_rElem = null;
}
var temp = lR && directLR ? _rElem : _elem;
var lWidth = lR ? directLR ? canvas.width() : to.right - 2 : (to.right - from.left - 2);
var lLeft = lR && !directLR ? 0 : (from.left - 1);
//console.log("DIRECT_LR:" + directLR + " FROM_L:" + from.left + " TO_L:" + to.left + " FROM_R:" + from.right + " TO_R" + to.right + " START:" + start + " END:" + end + " LWIDTH:" + lWidth)
if (_elem) {
_elem.css("border-right-width", lR ? directLR ? "0px" : _borderWidth : _borderWidth);
_elem.css("border-left-width", lR ? directLR ? _borderWidth : "0px" : _borderWidth);
_elem.css({
top: from.top - 1,
left: lLeft,
height: to.bottom - from.top - 2,
width: lWidth
});
}
if (_rElem) {
var rLeft = directLR ? 0 : from.left;
var rWidth = !directLR ? canvas.width() : (to.right - 2);
//console.log("RWIDTH:" + rWidth + " RLEFT:" + rLeft);
_rElem.css("border-left-width", lR ? directLR ? "0px" : _borderWidth : _borderWidth);
_rElem.css("border-right-width", lR ? directLR ? _borderWidth : "0px" : _borderWidth);
_rElem.css({
top: from.top - 1,
left: rLeft,
height: to.bottom - from.top - 2,
width: rWidth
});
}
return temp;
}
拷贝粘贴的实现:
获取选中的数据,监听ctrl + c 和 ctrl + v 事件进行拦截,是通过CellExternalFullCopyManager插件实现的(支持跨冻结列拷贝、粘贴)。
当监听到ctrl + c 的时候,先拦截事件,并将选中区域的数据转换成数据保存在临时的text中,并且选中text的值,之后将text删除,此时粘贴板上已经有复制的值。
当监听到ctrl + v 的时候,首先判断是否有自定义的拦截,如果有自定义拦截,执行自定义拦截,否则执行默认粘贴方法。
- 默认粘贴支持两种模式,一种是选择了区域,此时粘贴会在选择区域内进行粘贴;
- 另一种模式是仅指定了起点,按照复制的区域,从顶点开始粘贴后续区域。
function _decodeTabularDataExt(_grid, ta) {
var columns = _grid.getColumns();
var clippeds = toRow(_grid, ta);
document.body.removeChild(ta);
var ranges = _grid.getSelectionModel().getSelectedRanges();
if (ranges && ranges[0] && !((ranges[0].fromCell - ranges[0].toCell == 0) && (ranges[0].fromRow - ranges[0].toRow == 0))) {
var to = ranges[0];
var startCell = to.fromCell;
var startRow = to.fromRow;
// to.toRow - to.fromRow
// 通过数据反向操作
for (var y = 0; y < clippeds.length; y++) {
var desty = startRow + y;
if (desty <= to.toRow) {
var dt = _grid.getDataItem(desty);
if (dt) {
for (var x = 0; x < clippeds[y].length; x++) {
var destx = startCell + x;
if (destx < grid.getColumns().length && destx <= to.toCell) {
if (!_grid.onBeforeEditCellReadonly(desty, destx, dt, columns[destx])) {
setDataItemValueForColumn(dt, columns[destx], clippeds[y][x]);
// dt[columns[destx].field] = clippeds[y][x];
}
}
}
}
}
}
} else {
// if startPoint
// else if range
var selectedCell = _grid.getActiveCell();
if (selectedCell) {
var startCell = selectedCell.cell;
var startRow = selectedCell.row;
// 通过数据反向操作
for (var y = 0; y < clippeds.length; y++) {
var desty = startRow + y;
var dt = _grid.getDataItem(desty);
if (dt) {
for (var x = 0; x < clippeds[y].length; x++) {
var destx = startCell + x;
if (destx < grid.getColumns().length) {
if (!_grid.onBeforeEditCellReadonly(desty, destx, dt, columns[destx])) {
setDataItemValueForColumn(dt, columns[destx], clippeds[y][x]);
//dt[columns[destx].field] = clippeds[y][x];
}
}
}
}
}
}
}
_grid.invalidate();
_grid.render();
clearCopySelection();
return;
}
setTimeout(function () {
if (_grid.trigger(_self.onPasteCells, toRow(_grid, ta), e) === true) {
return;
}
_decodeTabularData(_grid, ta);
}, 50);
自定义粘贴示例:
copyExtManager.onPasteCells.subscribe(function (e, args) {
if ($(e.target).hasClass("bom-past")) {
var selectedCell = grid.getActiveCell();
if (selectedCell) {
var startRow = selectedCell.row;
var rowData = grid.getDataItem(startRow);
var children = [];
rowData.___path = "";
var treeViewStack = [rowData];
var bomValidator = {
parentPath: function (row, rows) {
var tempRow = row;
var parentPath = [];
while (tempRow) {
parentPath.push(tempRow.f3000);
if (tempRow.parent) {
tempRow = findItemById(tempRow.parent);
} else {
tempRow = null;
}
}
return parentPath;
},
cycleCheck: function (row, parentPath) {
}
}
var parentPath = bomValidator.parentPath(rowData, getSgAllData());
for (var x = 0; x < args.length; x++) {
var nRow = $.extend({}, rowData);
nRow.id = sbz.uuid();
for (var y = 1; y < args[x].length; y++) {
if (y + 1 < grid.getColumns().length) {
nRow[grid.getColumns()[y + 1].field] = args[x][y];
}
}
nRow.f1000 = nRow.f2001 = nRow.f2000;
nRow.f3000 = nRow.f3001 = nRow.f2000;
nRow.__state = 0;
nRow.___path = args[x][0];
if (treeViewStack.length > 0) {
if (treeViewStack[treeViewStack.length - 1].___path.length < nRow.___path.length) {
var parent = treeViewStack[treeViewStack.length - 1];
// 找到父
var nParentPath = treeViewStack.filter((v) => {
return v.f2000 == nRow.f2000;
});
if (nParentPath.length > 0 || parentPath.indexOf(nRow.f2000) != -1) {
alert("死循环:" + parentPath);
return;
}
nRow.parentNumber = parent.f2000;
treeViewStack.push(nRow);
// 确定层级
treeViewPath(parent, nRow, x);
} else {
// 找到最近的兄弟或直接的父
var tempPaths = [];
tempPaths.push(parentPath);
for (var i = treeViewStack.length - 1; i >= 0; i--) {
if (treeViewStack[i].___path.length >= nRow.___path.length) {
treeViewStack.pop();
} else {
tempPaths.push(treeViewStack[i].f2000);
}
}
if (tempPaths.indexOf(nRow.f2000) != -1) {
alert("死循环");
return;
}
var parent = treeViewStack[treeViewStack.length - 1];
nRow.parentNumber = parent.f2000;
treeViewStack.push(nRow);
// 确定层级
treeViewPath(parent, nRow, x);
}
}
// 提示是否增加 行 ,假设增加
// 直接更新和刷新状态...
children[x] = nRow;
}
if (children && children.length > 0) {
var parents = treeViewSameNodes(rowData);
for (var i = 0; i < parents.length; i++) {
treeViewCopyData(children, parents[i]);
}
treeViewInsert(sgBomData, rowData, children);
refresh();
}
}
return true;
}
return false;
});
- slickgrid ( nsunleo-slickgrid ) 4 解决区域选择和列选择冲突
slickgrid ( nsunleo-slickgrid ) 3 解决区域选择和列选择冲突 之前启用区域选择的时候,又启用了列选择(CheckboxSelectColumn),此时发现选择状态与区域 ...
- slickgrid ( nsunleo-slickgrid ) 3 修正区域选择不能跨冻结列的问题
slickgrid ( nsunleo-slickgrid ) 3 修正区域选择不能跨冻结列的问题 上次解决区域选择不能跨冻结列问题的时候,剩了个尾巴,从右往左选择的时候,会出现选择不正常的情况,后 ...
- slickgrid ( nsunleo-slickgrid ) 2 修正区域选择不能跨冻结列的问题
slickgrid( nsunleo-slickgrid ) 2 修正区域选择不能跨冻结列的问题 周六的时候,留了个小小的尾巴,区域选择的问题进做到了定位: 问题原因,在slickgrid启动冻结之 ...
- 【opencv学习笔记六】图像的ROI区域选择与复制
图像的数据量还是比较大的,对整张图片进行处理会影响我们的处理效率,因此常常只对图像中我们需要的部分进行处理,也就是感兴趣区域ROI.今天我们来看一下如何设置图像的感兴趣区域ROI.以及对ROI区域图像 ...
- 图像的ROI区域选择与复制
[opencv学习笔记六]图像的ROI区域选择与复制 孜然 7 人赞同了该文章 图像的数据量还是比较大的,对整张图片进行处理会影响我们的处理效率,因此常常只对图像中我们需要的部分进行处理,也就是感 ...
- js实现选择及复制粘贴
最近项目有个点击复制到粘贴板的需求,在这里做一个简单的例子分享给大家,没考虑兼容性,需要兼容的大家去查找下文档 //html<p id="element">测试测试&l ...
- Vim的使用 区域选择
1.Visual Block 区域选择,这里的Visual表示视觉,图像,可视化. 2. 小写v:字符选择 shift+v(大写V):行选择 ctrl+v:矩 ...
- 个人永久性免费-Excel催化剂功能第82波-复制粘贴按源区域大小自动扩展收缩目标区域
日常工作中,复制粘贴的操作,永远是最高频的操作,没有之一,在最高频的操作上,进行优化,让过程更智能,比一天到晚鼓吹人工智能替换人的骇人听闻的新闻来得更实际.此篇带来一点点的小小的改进,让日后无数的复制 ...
- 个人永久性免费-Excel催化剂功能第44波-可见区域复制粘贴不覆盖隐藏内容
Excel的复制粘贴操作,每天都在进行,若其中稍能提升一点效率,长久来说,实在是很可观的效率提升. Excel自带的复制粘贴功能,若复制的数据源或粘贴的目标位置中有隐藏的行列内容,简单一个复制粘贴充满 ...
随机推荐
- 【死磕JVM】给同事讲了一遍GC后,他要去面试,年轻人,就是容易冲动!
前言 在一个风和日丽的中午,和同事小勇一起走在公司楼下的小公园里面,看到很多的小姐姐,心想什么时候能够和这些小姐姐一起讨论人生呀,美滋滋,嘿嘿嘿. 收起你的哈喇子好不好,小勇总是在这个时候发出声音,挺 ...
- 打造云原生大型分布式监控系统系列文章-腾讯工程师roc
附上本系列文章链接 打造云原生大型分布式监控系统(一): 大规模场景下 Prometheus 的优化手段 打造云原生大型分布式监控系统(二): Thanos 架构详解 打造云原生大型分布式监控系统(二 ...
- 如何快速编写一个微信Api?
概述 Magicodes.Wx.Sdk致力于打造最简洁最易于使用的微信Sdk,逐步包括公众号Sdk.小程序Sdk.企业微信Sdk等,以及Abp VNext集成. 本篇将侧重于讲述如何向Magicode ...
- Chrome最新0day RCE(2021/4/13)
关于Chrome Chrome就是Google浏览器... POC Git链接 https://github.com/r4j0x00/exploits/tree/master/chrome-0day ...
- (十五)struts2的文件上传和下载
文件上传的原理 我们以前学习上传的时候知道需要将表单的enctype属性设置为multipart/form-data. 表单的enctype属性指定的是表单数据的编码方式,有三个值: -applica ...
- Kafka和Stream架构的使用
Kafka的单节点运行 启动服务 Kafka 使用 ZooKeeper 如果你还没有 ZooKeeper 服务器,你需要先启动一个 ZooKeeper 服务器. 您可以通过与 kafka 打包在一起的 ...
- Centos下搭建LAMP+PHP
Centos下搭建LAMP+PHP LAMP是指一组通常一起使用来运行动态网站或者服务器的自由软件名称首字母缩写: Linux,操作系统 Apache,网页服务器 MariaDB或MySQL,数据库管 ...
- 一次死锁导致CPU异常飘高的整个故障排查过程
目录 一.问题详情 top 命令截图 联系腾讯云排查 检查系统日志发现异常 二. 问题解析 三.问题原因 最终结论 四.扩展 进程的几种状态 马后炮 如何快速清理僵尸进程(Z) 内核参数相关 如何查看 ...
- Socket简介 与 UDP
1. Socket 简介 2. UDP 1. Socket 简介 网络编程,是指让在不同的电脑上的软件能够进行数据传递,即进程之间的通信. 本地的进程间通信(IPC) 例如有:队列.同步(互斥锁.条件 ...
- k8s新手必看
转载自https://juejin.im/post/6844903840139968520 Kubernetes零基础快速入门!初学者必看! 起源 Kubernetes 源自于 google 内部的服 ...