c# - 如何在运行时连接到 Office fluent 功能区并在 C# 中添加/操作功能区选项卡?
问题描述
我想连接到正在运行的 excel,或启动 excel,并在运行时在根/主功能区/菜单栏上添加一个选项卡。当我通过互操作连接到 excel 应用程序对象时,我看到了一个动态对象菜单栏。它在运行时解析为我无法解析的 .com 对象。
我可以遍历每个菜单并查看 menu.name、ID 和菜单项。它看起来像我正在运行的 excel 中的功能区项目,但我无法在运行时添加、删除或影响项目。菜单、菜单、菜单项都是微软私有的。我缺少什么来解决和添加/操作/删除我自己的运行时菜单项?我不想编写和编译静态或运行时 xml(还)。我看到其他供应商这样做。我缺少什么程序集或包括什么?这是我从纯粹的黑客攻击中得到的。
using System;
using System . Collections . Generic;
using System . Linq;
using System . Text;
using System . Threading . Tasks;
using Microsoft.Office.Core;
using System . Linq . Expressions;
using Microsoft . Office . Interop . Outlook;
using Microsoft . Office . Interop . Excel;
using System .Diagnostics;
using System . Runtime . InteropServices;
using System . Runtime . InteropServices . ComTypes;
using System . Diagnostics . Contracts;
using System . Windows . Controls . Ribbon;
using Microsoft . Office . Tools . Excel;
using Microsoft . Office . Tools . Ribbon;
using System . ComponentModel . Design;
using System . Reflection;
using Microsoft . Office . Interop . Access;
private static void ExcelChops ( )
{
Process [ ] Running = Process . GetProcessesByName ( "Excel" );
if ( Running . Count()==0 )
{
return;
}
Microsoft . Office . Interop . Excel . Application ExcelApplication = ( Microsoft . Office . Interop . Excel . Application ) Marshal . GetActiveObject ( "Excel.Application" );
if ( ExcelApplication == null )
{
return;
}
string ActiveExcelApplicationCaption = ExcelApplication . Caption;
Windows ExcelWindows = ExcelApplication . Windows;
int ExcelWindowCount = ExcelWindows . Count;
XlWindowState WindowState = ExcelApplication . WindowState;
Window ExcelWindow = ExcelApplication . Windows [ 1 ];
String ExcelWindoWindowCaption = ExcelWindow . Caption;
System . Diagnostics . Debug . WriteLine ( String . Format ( "\nExcel Application Caption {0} " , ActiveExcelApplicationCaption ) );
System . Diagnostics . Debug . WriteLine ( String . Format ( "\nExcel Window Caption {0} " , ExcelWindoWindowCaption ) );
System . Diagnostics . Debug . WriteLine ( String . Format ( "Excel Window Count {0} " , ExcelWindowCount ) );
System . Diagnostics . Debug . WriteLine ( String . Format ( "Excel Window State {0} " , WindowState ) );
//Microsoft.Office.Interop.Excel.Panes panes = ExcelWindow . Panes;
//IteratePanes ( panes );
Microsoft.Office.Interop.Excel.MenuBar aMB = ExcelApplication . ActiveMenuBar;
IterateMenus ( aMB , 0 );
System . Diagnostics . Debug . WriteLine ( String . Format ( "{0} {1} " , "Completed" , ( ( ( System . Environment . StackTrace ) . Split ( '\n' ) ) [ 2 ] . Trim ( ) ) ) );
}
private static void IterateMenus ( MenuBar aMB , int v )
{
string caption = aMB . Caption;
int ndx = aMB . Index;
dynamic parent = aMB . Parent;
Menus menus = aMB . Menus;
int menusCount = aMB . Menus . Count;
for ( int i = 1 ; i <= menusCount ; i++ )
{
Menu a = menus [ i ];
int b = a . Index;
string c = a . Caption;
System . Diagnostics . Debug . WriteLine ( String . Format ( "{0} {1} " , b , c ) );
IterateMenus ( a , v + 1 );
}
}
private static void IterateMenus ( Menu A , int v )
{
string caption = A . Caption;
int ndx = A . Index;
MenuItems items = A . MenuItems;
int itemsCount = items . Count;
for ( int i = 1 ; i <= itemsCount ; i++ )
{
dynamic a = items [ i ];
Type t = a.GetType ( );
object o = a as object;
Type to = o . GetType ( );
String oo = to . ToString ( );
var occ = to . Name;
var ooc = to . TypeHandle;
System . Diagnostics . Debug . WriteLine ( String . Format ( "menu item {0} of {1} {2} {3} " , i , itemsCount, occ, caption) );
}
}
解决方案
功能区不能像您希望的那样简单地以编程方式控制。
在功能区之前,Office 应用程序可以访问MenuBar
(Excel 未记录)和CommandBar
对象。AMenuBar
是经典的菜单类型(文件、编辑、视图、窗口、帮助等)。ACommandBar
是经典类型的工具栏(菜单下方的按钮行)。使用这些对象,您可以直接操作这些遗留 UI 功能。
丝带完全不同。使用功能区,您无法任意操作它。这是为了保护一个加载项免受另一个加载项的影响。为了使用功能区,您必须有一个加载项来提供描述您要应用的功能区配置的 XML。有几种方法可以创建 Excel 加载项:
- 使用适用于 Office 的 Visual Studio 工具(VSTO)
- 创建一个 COM 库(现在找不到很好的在线资源)
- 这是我从未尝试过的使用 Visual Studio Code 的选项(不确定这是否适用于您的用例)
除此之外,我建议您在 Google 上搜索“excel 插件功能区”,以获取有关如何在插件中使用功能区的各种文档。
和对象MenuBar
和CommandBar
属性仍然存在于遗留和非功能区目的(例如显示右键单击菜单)。如果您在具有功能区的 Excel 版本中创建新的应用程序级命令栏,新的命令栏会被毫不客气地转储到一个通用选项卡中,所有加载项的命令栏都会在该选项卡中结束。调整内置 CommandBars 不会影响内置功能区。
推荐阅读
- python - 将 CRC16 CCITT 代码从 C 转换为 Python
- python - 如何使用 KNN 结果快速创建图(ImageNet)
- powershell - 在 PowerShell 中正确拆分所选数据和格式
- php - Codeigniter 4 验证可选值
- spring-boot - 设置供消费者和生产者微服务使用的 ActiveMQ 映像
- javascript - 有没有办法使用索引访问对象属性?
- java - 循环提供误报的条件
- python-3.x - 按距离零最远的值对元组列表进行排序
- performance - 用于检测内存碎片的性能计数器
- r - Rmarkdown 文本在代码块内换行注释