前言

用.net6开发一个Winform程序,处理Excel文件,并把结果导出Excel文件。

要用到两个算法,一是turf.js库的booleanPointInPolygon方法,判断经纬度坐标是否在区域内;二是经纬度纠偏算法,因为对方给的区域坐标集合有偏移,需要纠偏。

这两个算法,网上找C#的实现,一是不好找;二是找来的不信任,我还要测试以确保没有问题。我之前做电子地图使用过turf.js库和js版本的纠偏算法,比较信任,确定没有问题。

所以我就打算通过C#调用js库的方法,来实现数据处理。

安装ClearScript

ClearScript是微软开源的js引擎,支持windows、linux、mac。

NuGet搜索安装:

Microsoft.ClearScript.Core

Microsoft.ClearScript.V8

Microsoft.ClearScript.V8.Native.win-x64

引入js文件



把leaflet.mapCorrection.js、turf.v6.5.0.min.js和自己写的calc.js放入工程中,右击属性设置复制到输出目录:如果较新则复制。

calc.js通过调用leaflet.mapCorrection.js和turf.v6.5.0.min.js中的方法实现功能,文件内容如下:

function calc(lng, lat, polygonStr) {
var point = turf.point([lng, lat]);
var polygonPoints = JSON.parse(polygonStr);
var polygon = turf.polygon(polygonPoints); var bl = turf.booleanPointInPolygon(point, polygon);
return bl;
} function correct(lng, lat) {
var newPoint = new CoordConvertor().gcj02_To_gps84(lng, lat);
return newPoint;
}

创建V8ScriptEngine对象

private V8ScriptEngine _engine = new V8ScriptEngine();

通过js引擎加载js文件

在Form1_Load方法中添加如下代码:

_engine.AddHostType("Console", typeof(Console));
string fileName = AppDomain.CurrentDomain.BaseDirectory + "turf.v6.5.0.min.js";
string js;
using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
byte[] bArr = new byte[fs.Length];
await fs.ReadAsync(bArr, 0, bArr.Length);
js = ASCIIEncoding.UTF8.GetString(bArr);
}
_engine.Execute(js);
fileName = AppDomain.CurrentDomain.BaseDirectory + "calc.js";
using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
byte[] bArr = new byte[fs.Length];
await fs.ReadAsync(bArr, 0, bArr.Length);
js = ASCIIEncoding.UTF8.GetString(bArr);
}
_engine.Execute(js);
fileName = AppDomain.CurrentDomain.BaseDirectory + "leaflet.mapCorrection.js";
using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
byte[] bArr = new byte[fs.Length];
await fs.ReadAsync(bArr, 0, bArr.Length);
js = ASCIIEncoding.UTF8.GetString(bArr);
}
_engine.Execute(js);

C#调用js方法实现经纬度坐标纠偏

double lng = Convert.ToDouble(lnglat[0]);
double lat = Convert.ToDouble(lnglat[1]); //坐标纠偏
dynamic newPoint = _engine.Invoke("correct", new object[] { lng, lat });
lng = newPoint.lng;
lat = newPoint.lat;

C#调用js方法判断经纬度点位是否在多边形内

//_selectedRegionPoints是多边形坐标点位集合json字符串
bool bl = (bool)_engine.Invoke("calc", new object[] { lng, lat, _selectedRegionPoints });

程序开发完成后发布



发布后文件夹拷贝到用户的win10系统中可以直接使用,不需要安装.net6环境。我自己的很老的win7 sp1虚拟机上跑不起来,ClearScriptV8.win-x64.dll无法加载成功,暂不知道为什么。

Form1.cs完整代码如下:

当时程序写的急,当然,程序还可以优化,不过没必要,要处理的数据量不大,功能没问题就行。

using Models;
using Newtonsoft.Json;
using System.Drawing;
using System.Text;
using System.Text.RegularExpressions;
using Microsoft.ClearScript.JavaScript;
using Microsoft.ClearScript.V8;
using NPOI.HSSF.UserModel;
using NPOI.XSSF.UserModel;
using NPOI.SS.UserModel;
using System.Reflection;
using System.Windows.Forms;
using NPOI.Util; namespace 点位
{
public partial class Form1 : Form
{
private Regions _regions;
private List<CameraInfo> _cameraList = new List<CameraInfo>();
private V8ScriptEngine _engine = new V8ScriptEngine();
private string _selectedRegionPoints; public Form1()
{
InitializeComponent();
} private async void Form1_Load(object sender, EventArgs e)
{
//通过js引擎加载js文件
_engine.AddHostType("Console", typeof(Console));
string fileName = AppDomain.CurrentDomain.BaseDirectory + "turf.v6.5.0.min.js";
string js;
using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
byte[] bArr = new byte[fs.Length];
await fs.ReadAsync(bArr, 0, bArr.Length);
js = ASCIIEncoding.UTF8.GetString(bArr);
}
_engine.Execute(js);
fileName = AppDomain.CurrentDomain.BaseDirectory + "calc.js";
using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
byte[] bArr = new byte[fs.Length];
await fs.ReadAsync(bArr, 0, bArr.Length);
js = ASCIIEncoding.UTF8.GetString(bArr);
}
_engine.Execute(js);
fileName = AppDomain.CurrentDomain.BaseDirectory + "leaflet.mapCorrection.js";
using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
byte[] bArr = new byte[fs.Length];
await fs.ReadAsync(bArr, 0, bArr.Length);
js = ASCIIEncoding.UTF8.GetString(bArr);
}
_engine.Execute(js); //行政区划下拉列表初始化
fileName = AppDomain.CurrentDomain.BaseDirectory + "安徽.json";
using (FileStream fs = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
byte[] bArr = new byte[fs.Length];
await fs.ReadAsync(bArr, 0, bArr.Length);
string json = ASCIIEncoding.UTF8.GetString(bArr);
_regions = JsonConvert.DeserializeObject<Regions>(json);
} List<Records> citys = _regions.RECORDS.ToList().FindAll(a => a.civilcode.Length == 4);
cbxCity.DataSource = citys;
cbxCity.DisplayMember = "civilname";
cbxCity.ValueMember = "civilcode";
} private void button1_Click(object sender, EventArgs e)
{
openFileDialog1.Title = "选择要处理的Excel文件";
openFileDialog1.Filter = "Excel文件(*.xlsx)|*.xlsx";
if (openFileDialog1.ShowDialog() == DialogResult.OK)
{
using (FileStream fs = new FileStream(openFileDialog1.FileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{ }
}
} private void cbxCity_SelectedIndexChanged(object sender, EventArgs e)
{
Records record = cbxCity.SelectedItem as Records;
List<Records> citys = _regions.RECORDS.ToList().FindAll(a => a.civilcode.Length > 4 && a.civilcode.Substring(0, 4) == record.civilcode);
citys.Insert(0, new Records() { civilcode = null, civilname = "==请选择==" });
cbxCounty.DataSource = citys;
cbxCounty.DisplayMember = "civilname";
cbxCounty.ValueMember = "civilcode";
} private void cbxCounty_SelectedIndexChanged(object sender, EventArgs e)
{
Records record = cbxCounty.SelectedItem as Records;
if (record.civilcode == null)
{
record = cbxCity.SelectedItem as Records;
}
Regex regex = new Regex(@"^POLYGON\((\(.*\),?)*\)$");
var mc = regex.Matches(record.polygongeo);
StringBuilder sb = new StringBuilder();
foreach (Match m in mc)
{
string value = m.Groups[1].Value.TrimStart('(').TrimEnd(')');
string[] lnglatArr = value.Split(',');
bool first = true;
if (sb.Length > 0)
{
sb.Append(",");
}
sb.Append("[[");
foreach (string lnglatStr in lnglatArr)
{
string[] lnglat = lnglatStr.Trim().Split(' ');
double lng = Convert.ToDouble(lnglat[0]);
double lat = Convert.ToDouble(lnglat[1]); //坐标纠偏
dynamic newPoint = _engine.Invoke("correct", new object[] { lng, lat });
lng = newPoint.lng;
lat = newPoint.lat; if (first)
{
first = false;
sb.AppendFormat($"[{lng}, {lat}]");
}
else
{
sb.AppendFormat($",[{lng}, {lat}]");
}
}
sb.Append("]]");
}
_selectedRegionPoints = sb.ToString();
} private async void openFileDialog1_FileOk(object sender, System.ComponentModel.CancelEventArgs e)
{
await Task.Delay(10); //读取Excel
_cameraList = new List<CameraInfo>();
using (FileStream fs = new FileStream(openFileDialog1.FileName, FileMode.Open, FileAccess.Read, FileShare.Read))
{
XSSFWorkbook workbook = new XSSFWorkbook(fs);
ISheet sheet = workbook.GetSheetAt(0);
for (int i = 1; i <= sheet.LastRowNum; i++)
{
IRow row = sheet.GetRow(i);
CameraInfo cameraInfo = new CameraInfo();
cameraInfo.CameraNo = row.GetCell(1).StringCellValue.Trim();
cameraInfo.City = row.GetCell(2).StringCellValue.Trim();
cameraInfo.County = row.GetCell(3).StringCellValue.Trim();
cameraInfo.CameraName = row.GetCell(4).StringCellValue.Trim();
cameraInfo.Lng = row.GetCell(5).StringCellValue.Trim();
cameraInfo.Lat = row.GetCell(6).StringCellValue.Trim();
cameraInfo.CameraFunType = row.GetCell(7).StringCellValue.Trim();
cameraInfo.Region = row.GetCell(8).StringCellValue.Trim();
cameraInfo.Type = row.GetCell(9).StringCellValue.Trim();
cameraInfo.Status = row.GetCell(10).StringCellValue.Trim();
cameraInfo.Mac = row.GetCell(11).StringCellValue.Trim();
cameraInfo.Ip = row.GetCell(12).StringCellValue.Trim();
_cameraList.Add(cameraInfo);
}
} //过滤数据
_cameraList = _cameraList.FindAll(cameraInfo =>
{
if (!string.IsNullOrWhiteSpace(cameraInfo.Lng) && !string.IsNullOrWhiteSpace(cameraInfo.Lat))
{
double lng = Convert.ToDouble(cameraInfo.Lng);
double lat = Convert.ToDouble(cameraInfo.Lat);
bool bl = (bool)_engine.Invoke("calc", new object[] { lng, lat, _selectedRegionPoints });
if (bl) //区域内
{
return false;
}
else //区域外
{
return true;
}
}
else
{
return false;
}
}); saveFileDialog1.Title = "选择处理结果要保存的位置及文件名";
saveFileDialog1.Filter = "Excel文件(*.xlsx)|*.xlsx";
if (saveFileDialog1.ShowDialog() == DialogResult.OK)
{
if (File.Exists(saveFileDialog1.FileName))
{
File.Delete(saveFileDialog1.FileName);
}
string template = AppDomain.CurrentDomain.BaseDirectory + "点位模板.xlsx";
XSSFWorkbook workbook;
using (FileStream fs = new FileStream(template, FileMode.Open, FileAccess.Read, FileShare.Read))
{
workbook = new XSSFWorkbook(fs); using (FileStream fs2 = new FileStream(saveFileDialog1.FileName, FileMode.OpenOrCreate, FileAccess.ReadWrite))
{
ISheet sheet = workbook.GetSheetAt(0);
sheet.RemoveRow(sheet.GetRow(1));
sheet.RemoveRow(sheet.GetRow(2));
Dictionary<int, ICellStyle> cellStyles = GetCellStyles(sheet);
int i = 1;
foreach (CameraInfo cameraInfo in _cameraList)
{
IRow row = sheet.CreateRow(i);
ICell cell1 = row.CreateCell(1, CellType.String);
ICell cell2 = row.CreateCell(2, CellType.String);
ICell cell3 = row.CreateCell(3, CellType.String);
ICell cell4 = row.CreateCell(4, CellType.String);
ICell cell5 = row.CreateCell(5, CellType.String);
ICell cell6 = row.CreateCell(6, CellType.String);
ICell cell7 = row.CreateCell(7, CellType.String);
ICell cell8 = row.CreateCell(8, CellType.String);
ICell cell9 = row.CreateCell(9, CellType.String);
ICell cell10 = row.CreateCell(10, CellType.String);
ICell cell11 = row.CreateCell(11, CellType.String);
ICell cell12 = row.CreateCell(12, CellType.String);
SetCellStyles(row, cellStyles);
cell1.SetCellValue(cameraInfo.CameraNo);
cell2.SetCellValue(cameraInfo.City);
cell3.SetCellValue(cameraInfo.County);
cell4.SetCellValue(cameraInfo.CameraName);
cell5.SetCellValue(cameraInfo.Lng);
cell6.SetCellValue(cameraInfo.Lat);
cell7.SetCellValue(cameraInfo.CameraFunType);
cell8.SetCellValue(cameraInfo.Region);
cell9.SetCellValue(cameraInfo.Type);
cell10.SetCellValue(cameraInfo.Status);
cell11.SetCellValue(cameraInfo.Mac);
cell12.SetCellValue(cameraInfo.Ip);
i++;
}
workbook.Write(fs2);
}
MessageBox.Show("完成");
}
}
} private Dictionary<int, ICellStyle> GetCellStyles(ISheet sheet)
{
var styleRow = sheet.GetRow(5);
Dictionary<int, ICellStyle> result = new Dictionary<int, ICellStyle>();
for (int i = 1; i <= 12; i++)
{
result.Add(i, styleRow.GetCell(i).CellStyle);
}
return result;
} private void SetCellStyles(IRow row, Dictionary<int, ICellStyle> styles)
{
for (int i = 1; i <= 12; i++)
{
row.GetCell(i).CellStyle = styles[i];
}
} private void Form1_FormClosing(object sender, FormClosingEventArgs e)
{
_engine.Dispose();
} private void button2_Click(object sender, EventArgs e)
{
folderBrowserDialog1.Description = "选择模板文件保存位置";
if (folderBrowserDialog1.ShowDialog() == DialogResult.OK)
{
string template = AppDomain.CurrentDomain.BaseDirectory + "点位模板.xlsx";
string filePath = Path.Combine(folderBrowserDialog1.SelectedPath, "点位模板.xlsx");
if (File.Exists(filePath))
{
if (MessageBox.Show("模板文件已存在,是否覆盖?", "提示", MessageBoxButtons.OKCancel) == DialogResult.OK)
{
File.Copy(template, filePath, true);
MessageBox.Show("下载完成");
}
}
else
{
File.Copy(template, filePath, true);
MessageBox.Show("下载完成");
}
}
}
}
}

C#调用js库的方法的更多相关文章

  1. 解决jQuery多个版本,与其他js库冲突方法

    jQuery多个版本或和其他js库冲突主要是常用的$符号的问题,这个问题 jquery早早就有给我们预留处理方法了,下面一起来看看解决办法. 1.同一页面jQuery多个版本或冲突解决方法. < ...

  2. 解决同一页面jQuery多个版本或和其他js库冲突方法

    <!DOCTYPE html> <html lang="en" xmlns="http://www.w3.org/1999/xhtml"> ...

  3. [转]JS调用Android里面的方法,Android调用JS里面的方法

    FROM : http://blog.csdn.net/hj563308597/article/details/45197709 Android WebView 在公司Android的开发过程中遇到一 ...

  4. qs.js库 使用方法

    1.qs.js库说明 qs是一个url参数转化(parse和stringify)的js库. https://www.npmjs.com/package/qs 2.使用(以vue文件做示例) (1)基本 ...

  5. 一个简单的C共享库的创建及Python调用此库的方法

    /********************************************************************* * Author  : Samson * Date    ...

  6. HTML5的兼容问题以及调用js文件的方法

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  7. js调用php和php调用js的方法举例

    js调用php和php调用js的方法举例1 JS方式调用PHP文件并取得php中的值 举一个简单的例子来说明: 如在页面a.html中用下面这句调用: <script type="te ...

  8. JS调用PHP 和 PHP调用JS的方法举例

    http://my.oschina.net/jiangchike/blog/220988 1.JS方式调用PHP文件并取得PHP中的值举一个简单的例子来说明:如在页面test_json1中用下面这句调 ...

  9. Delphi 中调用JS文件中的方法

    unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...

  10. WebView调用js方法获取返回值的完美解决方案

    在Android项目中我们或多或少会涉及到与js交互的问题,这其中WebView是必须掌握的控件,今天主要说说我们通过WebView调用js方法,然后如何很好的获取返回值.这里我总结了三种方式,大家可 ...

随机推荐

  1. 一天十道Java面试题----第五天(spring的事务传播机制------>mybatis的优缺点)

    这里是参考B站上的大佬做的面试题笔记.大家也可以去看视频讲解!!! 文章目录 41.spring的事务传播机制 42 .spring事务什么时候会失效 43 .什么的是bean的自动装配.有哪些方式? ...

  2. JUC中的AQS底层详细超详解

    摘要:当你使用java实现一个线程同步的对象时,一定会包含一个问题:你该如何保证多个线程访问该对象时,正确地进行阻塞等待,正确地被唤醒? 本文分享自华为云社区<JUC中的AQS底层详细超详解,剖 ...

  3. Linux---配置新服务器的常见操作(CentOS7)

    修改 Centos7为字符界面 Centos7取消了之前的 inittab文件的作用,改用命令实现, systemctl get-default : 获得默认启动方式 systemctl set-de ...

  4. Containerd 如何配置 Proxy?

    前言 在某些 air gap 场景中,往往需要离线或使用代理 (Proxy), 例如: 需要通过 Proxy pull 容器镜像: Docker Hub: docker.io Quay: quay.i ...

  5. 关于Wegame页面空白的问题解决

    前言 前几天帮亲戚家装电脑系统,装好后发现 wegame 所有页面都不能正确加载(全部是空白页面),很神奇,在网上找了很多种解决办法都没有效果,后来不过细心的我发现360浏览器一直提示我证书不安全过期 ...

  6. Selenium4+Python3系列(十) - Page Object设计模式

    前言 Page Object(PO)模式,是Selenium实战中最为流行,并且被自动化测试同学所熟悉和推崇的一种设计模式之一.在设计测试时,把页面元素定位和元素操作方法按照页面抽象出来,分离成一定的 ...

  7. Windows server 2008 tomcat间歇性掉线关闭

    1.代码没有问题,已经正常运行一年. 2.近期无更新代码. 3.tomcat 无运行报错. 今天突然间歇性掉线次数很多,客户不停反应情况,这边一时从代码开始找,我问了下在场运维其他服务器系统有无此现象 ...

  8. Day29 派生, 封装 , 多态, 反射

    Day29 派生, 封装 , 多态, 反射 内容概要 派生方法的实践 面向对象之封装 面向对象之多态 面向对象之反射 反射的实践案例 内容详细 1.派生方法的实践 #需求展示 import json ...

  9. Pod控制器详解

    Pod控制器详解 7.1 Pod控制器介绍 Pod是kubernetes的最小管理单元,在kubernetes中,按照pod的创建方式可以将其分为两类: 自主式pod:kubernetes直接创建出来 ...

  10. JavaEE Day07 HTML

    今日内容 Web概念概述 HTML 一.Web概念概述 1. JavaWeb:使用Java语言开发的基于互联网的项目 2.软件架构 C/S架构:Client/Server--- 客户端/服务器端(安卓 ...