我正在尝试将 Excel 2013 嵌入到 WPF 应用程序中。问题是,当我在以下代码中调用 SetWindowLongPtr 时,Excel 2013 会立即崩溃。我挖了一下,发现如果我注释掉WS.CHILD 样式,它工作正常,但是Excel 工作表变成只读的,这不是我想要的。相同的代码适用于 Excel 2010。
Excel.Application _excelApp;
IntPtr _wrappedApplicationHandle;
Int64 lngStyle;
Int64 lExStyle;
private void Button_Click_1(object sender, RoutedEventArgs e)
{
_excelApp = new Excel.Application()
{
Visible = true,
DisplayFormulaBar = true,
};
_wrappedApplicationHandle = new IntPtr( _excelApp.Hwnd);
lngStyle = GetWindowLongPtr(_wrappedApplicationHandle, (int)GWL.STYLE).ToInt64();
lngStyle &= ~(int)WS.CAPTION;
lngStyle &= ~(int)WS.SIZEBOX;
lngStyle |= (int)WS.MAXIMIZE;
lngStyle |= (int)WS.CHILD; //<< crashes with this line
lngStyle |= (int)WS.CLIPSIBLINGS;
lngStyle |= (int)WS.CLIPCHILDREN;
SetWindowLongPtr(new HandleRef(_excelApp, _wrappedApplicationHandle),
(int)GWL.STYLE,
new IntPtr(lngStyle));
...
}
编辑
我正在挖掘更多信息。我尝试将上面的代码包装在 try/catch block 中以查看会发生什么。它永远不会到达 catch block 。 Excel 2013 崩溃并出现臭名昭著的“应用程序已停止工作。向 MS 发送报告”错误。我已经在 Visual Studio 中打开了所有 Win32/COM/C++ 异常(通过 Debug 菜单 > Exceptions 对话框),但这也无济于事。错误对话框中有一个调试 按钮。如果我单击它并打开调试器,我看到的错误消息是“0xC0000005:访问冲突读取位置 0x0000000000000000。”
我还发现,在上面的代码中注释掉 WS.CHILD 行并不能严格地使工作表只读。它只是阻止常见的键盘/鼠标输入到达工作表。但是键盘的上下文菜单键等一些键仍然会到达那里并显示上下文菜单(虽然通过鼠标右键单击不起作用)。同样,我可以通过鼠标与 Office Ribbon 进行交互。似乎只有工作表区域(白色背景的网格)没有接收键盘/鼠标输入。
编辑 2
我只是记得(显然)与 Hans Passant 在他下面的帖子中解释的内容相反,当您创建 Excel VSTO 工作簿项目时,VS2010 本身托管了一个 Excel 实例。虽然我没有 VS2013(完整版)并且无法确认,但我怀疑 VS2013 会对 Excel 2013 VSTO 工作簿项目做同样的事情。考虑到 VS2010 及更高版本本身都是 WPF 应用程序,这如何适应图片?
编辑 3
@acelent 提出的私有(private)接口(interface)理论(参见下面的评论)似乎是正确的。我窥探了 VS2010 托管的 Excel 实例,发现有一个类名 = EXCELI 的新窗口,当我们正常打开 Excel 时它不存在(窗口的正常层次结构是 XLMAIN(应用程序)> XLDESK(工作区区域)> EXCEL7(工作簿))。此外,该工作簿不再作为 ActiveX 对象提供,这曾经是 Office Web Components 库可用时的情况(最后随 Office 2003 一起提供)。所以总而言之,我们似乎走到了死胡同,我将向我的客户建议@Hans 的回答,除非有人在接下来的几个小时内提出实际的工作方法。
最佳答案
你需要放弃这个,你不能让它可靠地工作。
WS_CHILD 不能按设计工作,Windows 有严格的要求,即父窗口和它的子窗口属于同一个进程。它们在彼此之间传递消息,当子进程试图取消引用属于父进程的指针时,它会崩溃。
Windows 确实有一些对 SetParent() 的 appcompat 支持,这是必需的,因为这通常在 Windows 3.x 程序中完成。具有独立地址空间的进程的概念在该 Windows 版本中完全不存在,因此这在当时不是问题。这在 20 年后仍然正常工作的几率与该过程的行为与 Windows 3.x 应用程序的行为程度成正比。它适用于控制台窗口或像记事本这样的简单应用程序。 Office 2013 应用远远超出简单和 3.x。输入有问题当然是一个标志。你可以修改 AttachThreadInput()尝试解决它,但您很可能会迎面遇到下一个问题,包括它在调试时导致死锁的恶习。
嵌入 Office 应用程序曾经是 Office 中受到强烈支持的功能,Microsoft 有意将它们设计为可嵌入的。底层技术称为 OLE 链接和嵌入。然而,这项技术已被强烈反对。 .NET Framework 有意排除了对它的支持。 Office 2003 是最后一个仍然很好地支持它的版本。麻烦从2007年开始,到2010年彻底报废,再也回不来了。
这里的前进方向恰恰相反。不要嵌入 Office 应用,让 Office 应用嵌入你。强烈支持为 Office 程序编写加载项。
关于c# - Excel 2013 崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20344321/
为了将Cucumber用于命令行脚本,我按照提供的说明安装了arubagem。它在我的Gemfile中,我可以验证是否安装了正确的版本并且我已经包含了require'aruba/cucumber'在'features/env.rb'中为了确保它能正常工作,我写了以下场景:@announceScenario:Testingcucumber/arubaGivenablankslateThentheoutputfrom"ls-la"shouldcontain"drw"假设事情应该失败。它确实失败了,但失败的原因是错误的:@announceScenario:Testingcucumber/ar
当我在Rails控制台中按向上或向左箭头时,出现此错误:irb(main):001:0>/Users/me/.rvm/gems/ruby-2.0.0-p247/gems/rb-readline-0.4.2/lib/rbreadline.rb:4269:in`blockin_rl_dispatch_subseq':invalidbytesequenceinUTF-8(ArgumentError)我使用rvm来管理我的ruby安装。我正在使用=>ruby-2.0.0-p247[x86_64]我使用bundle来管理我的gem,并且我有rb-readline(0.4.2)(人们推荐的最少
如何在ruby中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL
我正在尝试在Ruby中复制Convert.ToBase64String()行为。这是我的C#代码:varsha1=newSHA1CryptoServiceProvider();varpasswordBytes=Encoding.UTF8.GetBytes("password");varpasswordHash=sha1.ComputeHash(passwordBytes);returnConvert.ToBase64String(passwordHash);//returns"W6ph5Mm5Pz8GgiULbPgzG37mj9g="当我在Ruby中尝试同样的事情时,我得到了相同sha
C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.
我如何做Ruby方法"Flatten"RubyMethod在C#中。此方法将锯齿状数组展平为一维数组。例如:s=[1,2,3]#=>[1,2,3]t=[4,5,6,[7,8]]#=>[4,5,6,[7,8]]a=[s,t,9,10]#=>[[1,2,3],[4,5,6,[7,8]],9,10]a.flatten#=>[1,2,3,4,5,6,7,8,9,10 最佳答案 递归解决方案:IEnumerableFlatten(IEnumerablearray){foreach(variteminarray){if(itemisIEnume
我最近从C#转向了Ruby,我发现自己无法制作可折叠的标记代码区域。我只是想到做这种事情应该没问题:classExamplebegin#agroupofmethodsdefmethod1..enddefmethod2..endenddefmethod3..endend...但是这样做真的可以吗?method1和method2最终与method3是同一种东西吗?还是有一些我还没有见过的用于执行此操作的Ruby惯用语? 最佳答案 正如其他人所说,这不会改变方法定义。但是,如果要标记方法组,为什么不使用Ruby语义来标记它们呢?您可以使用
什么是Linq聚合方法的ruby等价物。它的工作原理是这样的varfactorial=new[]{1,2,3,4,5}.Aggregate((acc,i)=>acc*i);每次将数组序列中的值传递给lambda时,变量acc都会累积。 最佳答案 这在数学以及几乎所有编程语言中通常称为折叠。它是更普遍的变形概念的一个实例。Ruby从Smalltalk中继承了这个特性的名称,它被称为inject:into:(像aCollectioninject:aStartValueinto:aBlock一样使用。)所以,在Ruby中,它称为inj
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭8年前。Improvethisquestion几年前我去学校学习编程,毕业后我找到了一份系统管理方面的工作,这就是我职业生涯的方向。我想重新开始某种开发,并且一直在“玩”C#和ASP.NET,但我已经听到很多关于其他"new"语言的讨论(新的意思是它们是新的)我)喜欢Ruby和F#。我想我想知道我是否在浪费时间学习主要的MS语言,而不是成为一名通才。很长一段时间没有离开开发社区(如果我曾经离开过的话)让我在潮流中挣扎,我不想落在时代的
我有一个简单的Ruby脚本,我用它在某些HTTPheader上执行private_encrypt以签署要发送到rubyRESTAPI的Web请求,该API会根据Base64编码字符串测试Base64编码字符串生成而不是解码Base64和解密数据然后测试原始字符串。我使用的脚本是require"openssl"require"base64"path_to_cert=ARGV[0].dupplain_text=Base64.decode64(ARGV[1].dup)private_key=OpenSSL::PKey::RSA.new(File.read(path_to_cert))pu