- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
This文章说:
If an application must block a potential system shutdown, it can call the ShutdownBlockReasonCreate function. The caller provides a reason string that will be displayed to the user.
并且在 ShutdownBlockReasonCreate documentation 它清楚地表明在尝试关闭时将向用户显示一个带有原因字符串的对话框窗口:
Indicates that the system cannot be shut down and sets a reason string to be displayed to the user if system shutdown is initiated
并且在 this 中确认了该对话窗口的外观讨论:
The user could click "Shut down anyway". Besides, the system assumes "Shut down anyway" if the user takes no action within some number of seconds.
但是,在我调用 ShutdownBlockReasonCreate 并传递当前应用程序的主窗口句柄后,确保该函数成功并通过调用 ShutdownBlockReasonQuery 双重确保它。函数来检索原因字符串,它不会阻止用户关闭系统并且不会显示对话窗口。
为什么它对我的系统没有影响?我该如何解决这个问题?。
我使用管理员(内置)帐户在 Windows 10 x64 上运行,我使用的代码来自 this GitHub 存储库:
using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;
using Vanara.PInvoke;
using static Vanara.PInvoke.User32;
namespace Vanara.Windows.Forms.Forms
{
/// <summary>Used to define a set of operations within which any shutdown request will be met with a reason why this application is blocking it.</summary>
/// <remarks>This is to be used in either a 'using' statement or for the life of the application.
/// <para>To use for the life of the form, define a class field:
public class PreventShutdownContext : IDisposable
{
private HandleRef href;
/// <summary>Initializes a new instance of the <see cref="PreventShutdownContext"/> class.</summary>
/// <param name="window">The <see cref="Form"/> or <see cref="Control"/> that contains a valid window handle.</param>
/// <param name="reason">The reason the application must block system shutdown. Because users are typically in a hurry when shutting down the system, they may spend only a few seconds looking at the shutdown reasons that are displayed by the system. Therefore, it is important that your reason strings are short and clear.</param>
public PreventShutdownContext(Control window, string reason)
{
href = new HandleRef(window, window.Handle);
Reason = reason;
}
/// <summary>The reason the application must block system shutdown. Because users are typically in a hurry when shutting down the system, they may spend only a few seconds looking at the shutdown reasons that are displayed by the system. Therefore, it is important that your reason strings are short and clear.</summary>
/// <value>The reason string.</value>
public string Reason
{
get
{
if (!ShutdownBlockReasonQuery(href.Handle, out var reason))
Win32Error.ThrowLastError();
return reason;
}
set
{
if (value == null) value = string.Empty;
if (ShutdownBlockReasonQuery(href.Handle, out var _))
ShutdownBlockReasonDestroy(href.Handle);
if (!ShutdownBlockReasonCreate(href.Handle, value))
Win32Error.ThrowLastError();
}
}
/// <summary>Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.</summary>
public void Dispose()
{
ShutdownBlockReasonDestroy(href.Handle);
}
}
}
...
[DllImport(Lib.User32, SetLastError = true, ExactSpelling = true)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool ShutdownBlockReasonCreate(HWND hWnd, [MarshalAs(UnmanagedType.LPWStr)] string reason);
[DllImport(Lib.User32, SetLastError = true, ExactSpelling = true)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool ShutdownBlockReasonQuery(HWND hWnd, [MarshalAs(UnmanagedType.LPWStr)] StringBuilder pwszBuff, ref uint pcchBuff);
[DllImport(Lib.User32, SetLastError = true, ExactSpelling = true)]
[return: MarshalAs(UnmanagedType.Bool)]
public static extern bool ShutdownBlockReasonDestroy(HWND hWnd);
像这样的用法:
using (new PreventShutdownContext(this, "This app is super busy right now."))
{
// Do something that can't be interrupted...
}
我按原样尝试了代码,使用它的 P/Invoke 定义,还对我使用 IntPtr 结构而不是自定义的窗口句柄的代码做了一些修改 HWND结构,并将我在上面的评论中指定的应用程序的主窗口句柄传递给它。
最佳答案
这是设计使然。
documentation (以及您引用的主题)可能会产生轻微的误导。
Indicates that the system cannot be shut down and sets a reason string to be displayed to the user if system shutdown is initiated.
If an application must block a potential system shutdown, it can call the
ShutdownBlockReasonCreate
function.
这个函数实际上只是为您的应用程序设置消息字符串。此函数不会阻止您的应用程序被关闭。
要实现关闭 block ,只需按照您引用的文章中描述的步骤操作即可。您需要对 WM_QUERYENDSESSION
消息使用react并返回 FALSE
(0)。作为引用,另请参阅 WM_QUERYENDSESSION
文档。
您可能还会发现有趣的 this topic - 它描述了 Windows Vista 引入的变化,并包含如何实现关闭逻辑的最佳实践。
顺便说一句,关于您的应用程序不会有特殊的“对话窗口”。将显示标准的 Windows 关机 UI(它因操作系统版本而异)。您的应用程序将出现在“阻止关闭的应用程序”列表中,其中包含您使用 ShutdownBlockReasonCreate
函数注册的消息 - 但前提是它返回 FALSE
用于 WM_QUERYENDSESSION
消息。
如果上述解决方案(WM_QUERYENDSESSION
)没有解决问题,可能是系统设置忽略了这个机制。
正如@ElektroStudios 在他们的研究中发现的那样:
AutoEndTasks
注册表值(在 HKCU\Control Panel\Desktop
注册表项中找到),则关机不会显示任何 UI 让用户取消关机。所以在这些情况下创建“取消关闭原因”是没有用的,因为应用程序无论如何都会被强制立即关闭(以继续关闭)。作为引用,请阅读 this MS Docs topic .AutoEndTasks
注册表值必须为 0(零);否则,任何试图阻止关机的应用程序都将被终止,并且在关机时不会显示任何用户界面。AutoEndTasks
值可以添加到 HKEY_USERS\.DEFAULT\Control Panel\Desktop
键,该键覆盖 HKCU
中定义的值配置单元和 HKU\{SID}
。这意味着,如果 AutoEndTasks
在 HKCU
中为 false
(0) 但在 中为 true
(1) HKU\.DEFAULT
,则应用不会阻止系统关机,也不会显示关机界面。如果 AutoEndTasks
在 HKU\.DEFAULT
中为 false
但在 HKCU
中为 true,则应用将阻止系统从关闭和关闭 UI 将显示。AutoEndTasks
值不需要重新启动/注销系统即可生效。因此,一旦在适当的键中将其设置为 false
(例如 HKEY_USERS\.DEFAULT\Control Panel\Desktop
),应用程序将阻止系统关闭,我们当我们完成使用该功能时,可以将该值恢复到之前的状态。关于c# - 调用 'ShutdownBlockReasonCreate'函数不能阻止用户关闭系统,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54842853/
我是 javascript 的新手(今天开始弄乱它)。 我正在尝试更改名为“bar”的元素(div)的高度。条形图将成为图表的一部分。 我可以毫无问题地将按钮连接到更改栏高度的函数。一切正常,除了条形
错误 -> “UIVIew”没有名为“addSubView”的成员 override func viewDidLoad() { super.viewDidLoad() // Do an
我在命令行工具项目中复制并粘贴了 main.swift 下面链接中的代码。 How do you use CGEventTapCreate in Swift? 它构建没有错误,但是当我运行时, gua
我在尝试编译我的代码时遇到以下错误。 ERROR! ..\myCode\CPOI.cpp:68:41: error: cannot dynamic_cast 'screenType' (of type
我正在尝试将多个字符串连接到一个我已为其分配内存的字符串指针。这是一个例子: char *finalNumString = malloc(sizeof(char)*1024); finalNumStr
我在使用 dup2() 和 pipe() 时遇到问题。 当我尝试将管道的写入端 dup2 到 STDOUT_FILENO 时,我收到了 EBADF。 我用 gdb 在 dup2(pout[1], ST
首先,我应该说我运行的是 Windows 7。 因此,今天早上我尝试像往常一样从我的存储库中提取数据,但我做不到。我得到了错误: The authenticity of host 'github.co
刚开始在虚拟环境中运行Python,乱用Django,无法激活虚拟环境。 花了最后 4 个小时尝试在本地终端/VS 代码上激活虚拟环境 (venv),但没有成功。 避免使用“sudo pip inst
Tidyverse 的粉丝经常给出使用小标题而不是数据框的几个优点。它们中的大多数似乎旨在保护用户免于犯错误。例如,与数据框不同,小标题: 不需要 ,drop=FALSE不从数据中删除维度的论据。 不
我一直在对 Elm 应用程序进行 docker 化时遇到问题。据我所知,我已经创建了一个完整且有效的 Docker 文件……但它不起作用。 我会解释的。 所以我的脚本在 3 个文件中运行。 首先是启动
我可以在 Controller 中使用@Autowired,例如 @RestController public class Index { @Autowired HttpServlet
我定义了一个方法和一个函数: def print(str:String) = println val intToString = (n:Int) => n.toString 现在我想创作它们。 我的问
当我控制台单独记录变量“pokemons”时,它确实返回一个数组。但是当我尝试映射它时,出现错误: TypeError: pokemons.map is not a function 我的代码: im
每当我尝试在 Python 解释器中导入 smtplib 时,都会收到此错误: ImportError: cannot import name fix_eols 我该如何解决这个问题? 编辑:这是完整
我正在使用 Meteor.js 开发一个项目,但在使用 Handlebar 时遇到了一些问题:我想检索集合的最后一项,并显示字段:其中包含 html 的文本: 这是我的javascript代码: Te
你好,我想使用 Service 实现 GestureDetector 但是我有这个错误The method onTouchEvent(MotionEvent) of type GestureServi
我正在尝试在 Controller bean 中 Autowiring 接口(interface) 在我放置的上下文配置文件中 和 我的 Controller 类是 @Controller pub
我试图在 mainwindow.cpp 中包含 QtSvg,但是当我编译时它说无法打开包含文件:QtSvg。我已经在我的 *.pro 文件中添加了这个(QT += svg)。我可以知道可能是什么问题吗
鉴于以下 PostgreSQL 代码,我认为这段代码不容易受到 SQL 注入(inject)攻击: _filter 'day' _start 1 _end 10 _sort 'article_name
我想执行以下操作。这在 MySQL 中是非法的。 PostGRESQL 中关联的 CTE(“with”子句)有效。这里的假设是 MySQL 中的子查询不是完全限定的 CTE。 请注意:这个查询显然非常
我是一名优秀的程序员,十分优秀!