diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 4a37d799..02ae859b 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -75,6 +75,9 @@ jobs: - name: Build Sandboxie-Plus 64 bit run: SandboxiePlus\qmake_plus.cmd x64 + - name: Build SbieShell 64 bit + run: msbuild /t:restore,build -p:RestorePackagesConfig=true SandboxiePlus\SbieShell\SbieShell.sln /p:Configuration="Release" /p:Platform=x64 + - name: Build Sandboxie-Plus 32 bit run: SandboxiePlus\qmake_plus.cmd Win32 diff --git a/CHANGELOG.md b/CHANGELOG.md index 0b224670..157a5c68 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -29,7 +29,10 @@ This project adheres to [Semantic Versioning](http://semver.org/). ## [1.0.22 / 5.55.22] - 2022-05-xx ### Added -- added auto update download and silent install option to sandman.exe +- added auto update download and silent install option to sandman.exe [#917](https://github.com/sandboxie-plus/Sandboxie/issues/917) +- trace monitor mode can now also save to file [#1851](https://github.com/sandboxie-plus/Sandboxie/issues/1851) +- trace log now shows ipc object type information +- added support for windows 11 context menus ### Fixed - fixed sandman crash issue [#1846](https://github.com/sandboxie-plus/Sandboxie/issues/1846) diff --git a/Installer/copy_build.cmd b/Installer/copy_build.cmd index 852509c1..1477a103 100644 --- a/Installer/copy_build.cmd +++ b/Installer/copy_build.cmd @@ -118,6 +118,9 @@ IF %archPath% == x64 ( mkdir %instPath%\32\ copy /y %~dp0..\Sandboxie\Bin\Win32\SbieRelease\SbieSvc.exe %instPath%\32\ copy /y %~dp0..\Sandboxie\Bin\Win32\SbieRelease\SbieDll.dll %instPath%\32\ + + copy /y %~dp0..\SandboxiePlus\x64\Release\SbieShellExt.dll %instPath%\ + copy /y %~dp0..\SandboxiePlus\x64\Release\SbieShellPkg.msix %instPath%\ ) copy /y %~dp0..\Sandboxie\install\Templates.ini %instPath%\ diff --git a/Sandboxie/core/drv/ipc.c b/Sandboxie/core/drv/ipc.c index c77aeacd..88152a9b 100644 --- a/Sandboxie/core/drv/ipc.c +++ b/Sandboxie/core/drv/ipc.c @@ -1062,7 +1062,15 @@ _FX NTSTATUS Ipc_CheckGenericObject( } RtlStringCbPrintfW(access_str, sizeof(access_str), L"(I%c) %08X", letter, GrantedAccess); - Log_Debug_Msg(mon_type, access_str, Name->Buffer); + //Log_Debug_Msg(mon_type, access_str, Name->Buffer); + + if (Session_MonitorCount) { + + POBJECT_TYPE ObjectType = pObGetObjectType(Object); + + const WCHAR* strings[4] = { Name->Buffer, access_str, ObjectType ? ObjectType->Name.Buffer : NULL, NULL }; + Session_MonitorPutEx(mon_type, strings, NULL, PsGetCurrentProcessId(), PsGetCurrentThreadId()); + } } } diff --git a/Sandboxie/core/drv/session.c b/Sandboxie/core/drv/session.c index 0962396c..8a275f3d 100644 --- a/Sandboxie/core/drv/session.c +++ b/Sandboxie/core/drv/session.c @@ -706,6 +706,9 @@ _FX NTSTATUS Session_Api_MonitorPut2(PROCESS *proc, ULONG64 *parms) ULONG log_type; WCHAR *log_data; WCHAR *name; + const WCHAR *type_pipe = L"Pipe"; + const WCHAR *type_file = L"File"; + const WCHAR *type_name = NULL; NTSTATUS status; ULONG log_len; @@ -783,8 +786,10 @@ _FX NTSTATUS Session_Api_MonitorPut2(PROCESS *proc, ULONG64 *parms) Obj_ObjectTypes[i], KernelMode, NULL, &object); - if (status != STATUS_OBJECT_TYPE_MISMATCH) + if (status != STATUS_OBJECT_TYPE_MISMATCH) { + type_name = Obj_ObjectTypes[i]->Name.Buffer; break; + } } // DbgPrint("IPC Status = %08X Object = %08X for Open <%S>\n", status, object, name); @@ -795,7 +800,7 @@ _FX NTSTATUS Session_Api_MonitorPut2(PROCESS *proc, ULONG64 *parms) // to get the name assigned to it at time of creation // - if ((log_type & MONITOR_TYPE_MASK) == MONITOR_PIPE) { + else if ((log_type & MONITOR_TYPE_MASK) == MONITOR_PIPE) { OBJECT_ATTRIBUTES objattrs; IO_STATUS_BLOCK IoStatusBlock; @@ -834,6 +839,8 @@ _FX NTSTATUS Session_Api_MonitorPut2(PROCESS *proc, ULONG64 *parms) status = STATUS_OBJECT_NAME_NOT_FOUND; } + type_name = type_pipe; + //DbgPrint("PIPE Status3 = %08X Object = %08X for Open <%S>\n", status, object, name); } @@ -885,7 +892,7 @@ _FX NTSTATUS Session_Api_MonitorPut2(PROCESS *proc, ULONG64 *parms) name[1] = L'\0'; }*/ - const WCHAR* strings[2] = { name, NULL }; + const WCHAR* strings[4] = { name, L"", type_name, NULL }; Session_MonitorPutEx(log_type | MONITOR_USER, strings, NULL, proc->pid, PsGetCurrentThreadId()); } diff --git a/SandboxiePlus/MiscHelpers/Common/PanelView.cpp b/SandboxiePlus/MiscHelpers/Common/PanelView.cpp index 18989b56..f1ca2705 100644 --- a/SandboxiePlus/MiscHelpers/Common/PanelView.cpp +++ b/SandboxiePlus/MiscHelpers/Common/PanelView.cpp @@ -143,7 +143,7 @@ void CPanelView::RecursiveCopyPanel(const QModelIndex& ModelIndex, QList CPanelView::DumpPanel() { QAbstractItemModel* pModel = GetModel(); @@ -153,7 +153,13 @@ void CPanelView::OnCopyPanel() QModelIndex ModelIndex = pModel->index(i, 0); RecursiveCopyPanel(ModelIndex, Rows); } - FormatAndCopy(Rows); + + return Rows; +} + +void CPanelView::OnCopyPanel() +{ + FormatAndCopy(DumpPanel()); } void CPanelView::FormatAndCopy(QList Rows, bool Headder) diff --git a/SandboxiePlus/MiscHelpers/Common/PanelView.h b/SandboxiePlus/MiscHelpers/Common/PanelView.h index bbbe3cfc..5716a417 100644 --- a/SandboxiePlus/MiscHelpers/Common/PanelView.h +++ b/SandboxiePlus/MiscHelpers/Common/PanelView.h @@ -14,6 +14,8 @@ public: static void SetMaxCellWidth(int iMaxWidth) { m_MaxCellWidth = iMaxWidth; } static void SetCellSeparator(const QString& Sep) { m_CellSeparator = Sep; } + virtual QList DumpPanel(); + static QString m_CopyCell; static QString m_CopyRow; static QString m_CopyPanel; diff --git a/SandboxiePlus/QSbieAPI/SbieTrace.cpp b/SandboxiePlus/QSbieAPI/SbieTrace.cpp index 326102a2..9b582b2e 100644 --- a/SandboxiePlus/QSbieAPI/SbieTrace.cpp +++ b/SandboxiePlus/QSbieAPI/SbieTrace.cpp @@ -65,6 +65,7 @@ CTraceEntry::CTraceEntry(quint32 ProcessId, quint32 ThreadId, quint32 Type, cons m_ThreadId = ThreadId; m_Name = LogData.length() > 0 ? LogData.at(0) : QString("(empty)"); m_Message = LogData.length() > 1 ? LogData.at(1) : QString(); + m_SubType = LogData.length() > 2 ? LogData.at(2) : QString(); m_Type.Flags = Type; m_TimeStamp = QDateTime::currentDateTime(); // ms resolution @@ -136,6 +137,9 @@ QString CTraceEntry::GetTypeStr() const if(Type.isEmpty()) Type = "Unknown: " + QString::number(m_Type.Type); + if(!m_SubType.isEmpty()) + Type.append(" / " + m_SubType); + if (m_Type.User) Type.append(" (U)"); else diff --git a/SandboxiePlus/QSbieAPI/SbieTrace.h b/SandboxiePlus/QSbieAPI/SbieTrace.h index ed1f2321..cbf06465 100644 --- a/SandboxiePlus/QSbieAPI/SbieTrace.h +++ b/SandboxiePlus/QSbieAPI/SbieTrace.h @@ -67,6 +67,7 @@ public: protected: QString m_Name; QString m_Message; + QString m_SubType; quint32 m_ProcessId; quint32 m_ThreadId; QDateTime m_TimeStamp; diff --git a/SandboxiePlus/QSbieAPI/SbieUtils.cpp b/SandboxiePlus/QSbieAPI/SbieUtils.cpp index 13247639..bc4ff266 100644 --- a/SandboxiePlus/QSbieAPI/SbieUtils.cpp +++ b/SandboxiePlus/QSbieAPI/SbieUtils.cpp @@ -361,9 +361,15 @@ bool CSbieUtils::CreateShortcut(CSbieAPI* pApi, QString LinkPath, const QString QString StartArgs; if (bRunElevated) StartArgs += "/elevated "; - StartArgs += "/box:" + boxname; - if (!arguments.isEmpty()) - StartArgs += " \"" + arguments + "\""; + if (!boxname.isEmpty()) + StartArgs += "/box:" + boxname; + if (!arguments.isEmpty()) { + if (!StartArgs.isEmpty()) StartArgs += " "; + if(arguments.contains(" ")) + StartArgs += "\"" + arguments + "\""; + else + StartArgs += arguments; + } IUnknown *pUnknown; HRESULT hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC, IID_IUnknown, (void **)&pUnknown); @@ -381,7 +387,7 @@ bool CSbieUtils::CreateShortcut(CSbieAPI* pApi, QString LinkPath, const QString if (!workdir.isEmpty()) pShellLink->SetWorkingDirectory(workdir.toStdWString().c_str()); if (!LinkName.isEmpty()) { - QString desc = QString("%1 [%2]").arg(LinkName).arg(boxname); + QString desc = QString("%1 [%2]").arg(LinkName).arg(boxname.isEmpty() ? "DefaultBox" : boxname); pShellLink->SetDescription(desc.toStdWString().c_str()); } diff --git a/SandboxiePlus/SandMan/SandMan.cpp b/SandboxiePlus/SandMan/SandMan.cpp index 95a78e0e..6ea8c508 100644 --- a/SandboxiePlus/SandMan/SandMan.cpp +++ b/SandboxiePlus/SandMan/SandMan.cpp @@ -21,7 +21,6 @@ #include "Views/TraceView.h" #include "Windows/SelectBoxWindow.h" #include "../UGlobalHotkey/uglobalhotkeys.h" -#include "Wizards/SetupWizard.h" #include "Helpers/WinAdmin.h" CSbiePlusAPI* theAPI = NULL; @@ -356,7 +355,6 @@ void CSandMan::CreateMenus() m_pStopSvc = m_pMaintenanceItems->addAction(tr("Stop Service"), this, SLOT(OnMaintenance())); m_pUninstallSvc = m_pMaintenanceItems->addAction(tr("Uninstall Service"), this, SLOT(OnMaintenance())); m_pMaintenance->addSeparator(); - m_pSetupWizard = m_pMaintenance->addAction(CSandMan::GetIcon("Software"), tr("Setup Wizard"), this, SLOT(OnMaintenance())); if(IsFullyPortable()) m_pUninstallAll = m_pMaintenance->addAction(CSandMan::GetIcon("Uninstall"), tr("Uninstall All"), this, SLOT(OnMaintenance())); @@ -1175,18 +1173,6 @@ void CSandMan::OnStatusChanged() OnLogMessage(tr("Default sandbox not found; creating: %1").arg("DefaultBox")); theAPI->CreateBox("DefaultBox"); } - - int BusinessUse = theConf->GetInt("Options/BusinessUse", 2); - if (g_CertInfo.business && BusinessUse == 0) // if we have a Business cert switch to that use case - theConf->SetValue("Options/BusinessUse", 1); - - int WizardLevel = theConf->GetBool("Options/WizardLevel", 0); - if (WizardLevel == 0) { - if (CSetupWizard::ShowWizard()) - UpdateSettings(); - else // if user canceled mark that and not show again - theConf->SetValue("Options/WizardLevel", -1); - } } else { @@ -1737,16 +1723,10 @@ void CSandMan::OnMaintenance() AutorunEnable(false); - CSbieUtils::RemoveContextMenu(); + CSettingsWindow__RemoveContextMenu(); CSbieUtils::RemoveContextMenu2(); } - - else if (sender() == m_pSetupWizard) { - if (CSetupWizard::ShowWizard()) - UpdateSettings(); - return; - } - + HandleMaintenance(Status); } diff --git a/SandboxiePlus/SandMan/Views/TraceView.cpp b/SandboxiePlus/SandMan/Views/TraceView.cpp index 3bd144a5..8678b869 100644 --- a/SandboxiePlus/SandMan/Views/TraceView.cpp +++ b/SandboxiePlus/SandMan/Views/TraceView.cpp @@ -455,7 +455,6 @@ void CTraceView::OnSetMode() m_pTraceTree->setEnabled(!m_pMonitorMode->isChecked()); m_pTraceStatus->setEnabled(!m_pMonitorMode->isChecked()); - m_pSaveToFile->setEnabled(!m_pMonitorMode->isChecked()); m_FullRefresh = true; @@ -556,27 +555,36 @@ void CTraceView::SaveToFile() return; } - QVector ResourceLog = theAPI->GetTrace(); - for (int i = 0; i < ResourceLog.count(); i++) + if (m_pMonitorMode->isChecked()) { - CTraceEntryPtr pEntry = ResourceLog.at(i); + QList Rows = m_pMonitor->DumpPanel(); + foreach(const QStringList& Row, Rows) + File.write(Row.join("\t").toLatin1() + "\n"); + } + else + { + QVector ResourceLog = theAPI->GetTrace(); + for (int i = 0; i < ResourceLog.count(); i++) + { + CTraceEntryPtr pEntry = ResourceLog.at(i); - //int iFilter = CTraceView__Filter(pEntry, this); - //if (!iFilter) - // continue; + //int iFilter = CTraceView__Filter(pEntry, this); + //if (!iFilter) + // continue; - QStringList Line; - Line.append(pEntry->GetTimeStamp().toString("hh:mm:ss.zzz")); - QString Name = pEntry->GetProcessName(); - Line.append(Name.isEmpty() ? tr("Unknown") : Name); - Line.append(QString("%1").arg(pEntry->GetProcessId())); - Line.append(QString("%1").arg(pEntry->GetThreadId())); - Line.append(pEntry->GetTypeStr()); - Line.append(pEntry->GetStautsStr()); - Line.append(pEntry->GetName()); - Line.append(pEntry->GetMessage()); + QStringList Line; + Line.append(pEntry->GetTimeStamp().toString("hh:mm:ss.zzz")); + QString Name = pEntry->GetProcessName(); + Line.append(Name.isEmpty() ? tr("Unknown") : Name); + Line.append(QString("%1").arg(pEntry->GetProcessId())); + Line.append(QString("%1").arg(pEntry->GetThreadId())); + Line.append(pEntry->GetTypeStr()); + Line.append(pEntry->GetStautsStr()); + Line.append(pEntry->GetName()); + Line.append(pEntry->GetMessage()); - File.write(Line.join("\t").toLatin1() + "\n"); + File.write(Line.join("\t").toLatin1() + "\n"); + } } File.close(); diff --git a/SandboxiePlus/SandMan/Windows/SettingsWindow.cpp b/SandboxiePlus/SandMan/Windows/SettingsWindow.cpp index f82d636f..27d66e1c 100644 --- a/SandboxiePlus/SandMan/Windows/SettingsWindow.cpp +++ b/SandboxiePlus/SandMan/Windows/SettingsWindow.cpp @@ -215,6 +215,12 @@ void CSettingsWindow::closeEvent(QCloseEvent *e) Qt::CheckState CSettingsWindow__IsContextMenu() { + QSettings settings("HKEY_LOCAL_MACHINE\\SOFTWARE\\Classes\\PackagedCom\\Package", QSettings::NativeFormat); + foreach(const QString & Key, settings.childGroups()) { + if (Key.indexOf("SandboxieShell") == 0) + return Qt::Checked; + } + QString cmd = CSbieUtils::GetContextMenuStartCmd(); if (cmd.contains("SandMan.exe", Qt::CaseInsensitive)) return Qt::Checked; // set up and sandman @@ -225,11 +231,33 @@ Qt::CheckState CSettingsWindow__IsContextMenu() void CSettingsWindow__AddContextMenu() { + QSettings settings("HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion", QSettings::NativeFormat); + if (settings.value("CurrentBuild") >= 22000) // Windows 11 + { + QProcess Proc; + Proc.execute("rundll32.exe", QStringList() << QCoreApplication::applicationDirPath().replace("/", "\\") + "\\SbieShellExt.dll,RegisterPackage"); + Proc.waitForFinished(); + return; + } + CSbieUtils::AddContextMenu(QApplication::applicationDirPath().replace("/", "\\") + "\\SandMan.exe", CSettingsWindow::tr("Run &Sandboxed"), //CSettingsWindow::tr("Explore &Sandboxed"), QApplication::applicationDirPath().replace("/", "\\") + "\\Start.exe"); } +void CSettingsWindow__RemoveContextMenu() +{ + QSettings settings("HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion", QSettings::NativeFormat); + if (settings.value("CurrentBuild") >= 22000) // Windows 11 + { + QProcess Proc; + Proc.execute("rundll32.exe", QStringList() << QCoreApplication::applicationDirPath().replace("/", "\\") + "\\SbieShellExt.dll,RemovePackage"); + Proc.waitForFinished(); + } + + CSbieUtils::RemoveContextMenu(); +} + void CSettingsWindow__AddBrowserIcon() { QString Path = QStandardPaths::writableLocation(QStandardPaths::DesktopLocation).replace("/", "\\"); @@ -405,7 +433,7 @@ void CSettingsWindow::SaveSettings() if (ui.chkShellMenu->isChecked()) CSettingsWindow__AddContextMenu(); else - CSbieUtils::RemoveContextMenu(); + CSettingsWindow__RemoveContextMenu(); } if (ui.chkShellMenu2->isChecked() != CSbieUtils::HasContextMenu2()) { @@ -542,26 +570,80 @@ void CSettingsWindow::SaveSettings() { QByteArray Certificate = ui.txtCertificate->toPlainText().toUtf8(); if (g_Certificate != Certificate) { - + QPalette palette = QApplication::palette(); + QString CertPath = theAPI->GetSbiePath() + "\\Certificate.dat"; + if (!Certificate.isEmpty()) { + + auto Args = GetArguments(Certificate, L'\n', L':'); + + bool bLooksOk = true; + if (Args.value("NAME").isEmpty()) // mandatory + bLooksOk = false; + //if (Args.value("UPDATEKEY").isEmpty()) + // bLooksOk = false; + if (Args.value("SIGNATURE").isEmpty()) // absolutely mandatory + bLooksOk = false; + + if (bLooksOk) { + QString TempPath = QDir::tempPath() + "/Sbie+Certificate.dat"; + QFile CertFile(TempPath); + if (CertFile.open(QFile::WriteOnly)) { + CertFile.write(Certificate); + CertFile.close(); + } + + WindowsMoveFile(TempPath.replace("/", "\\"), CertPath.replace("/", "\\")); + } + else { + Certificate.clear(); + QMessageBox::critical(this, "Sandboxie-Plus", tr("This does not look like a certificate, please enter the entire certificate not just a portion of it.")); + } + } + else if(!g_Certificate.isEmpty()){ + WindowsMoveFile(CertPath.replace("/", "\\"), ""); + } + if (theGUI->m_DarkTheme) palette.setColor(QPalette::Text, Qt::black); ui.lblCertExp->setVisible(false); - bool bRet = ApplyCertificate(Certificate, this); - if (Certificate.isEmpty()) + { palette.setColor(QPalette::Base, Qt::white); - else if (!bRet) - palette.setColor(QPalette::Base, QColor(255, 192, 192)); - else if (g_CertInfo.expired || g_CertInfo.outdated) { - palette.setColor(QPalette::Base, QColor(255, 255, 192)); - ui.lblCertExp->setVisible(true); + } + else if (!theAPI->ReloadCert().IsError()) + { + g_FeatureFlags = theAPI->GetFeatureFlags(); + theGUI->UpdateCertState(); + + if (g_CertInfo.expired || g_CertInfo.outdated) { + if(g_CertInfo.expired) + QMessageBox::information(this, "Sandboxie-Plus", tr("This certificate is unfortunately expired.")); + else + QMessageBox::information(this, "Sandboxie-Plus", tr("This certificate is unfortunately outdated.")); + + palette.setColor(QPalette::Base, QColor(255, 255, 192)); + ui.lblCertExp->setVisible(true); + } + else { + QMessageBox::information(this, "Sandboxie-Plus", tr("Thank you for supporting the development of Sandboxie-Plus.")); + + palette.setColor(QPalette::Base, QColor(192, 255, 192)); + } } else - palette.setColor(QPalette::Base, QColor(192, 255, 192)); + { + QMessageBox::critical(this, "Sandboxie-Plus", tr("This support certificate is not valid.")); + + palette.setColor(QPalette::Base, QColor(255, 192, 192)); + Certificate.clear(); + g_CertInfo.State = 0; + } + + g_Certificate = Certificate; ui.txtCertificate->setPalette(palette); } @@ -578,71 +660,6 @@ void CSettingsWindow::SaveSettings() emit OptionsChanged(); } -bool CSettingsWindow::ApplyCertificate(const QByteArray &Certificate, QWidget* widget) -{ - QString CertPath = theAPI->GetSbiePath() + "\\Certificate.dat"; - if (!Certificate.isEmpty()) { - - auto Args = GetArguments(Certificate, L'\n', L':'); - - bool bLooksOk = true; - if (Args.value("NAME").isEmpty()) // mandatory - bLooksOk = false; - //if (Args.value("UPDATEKEY").isEmpty()) - // bLooksOk = false; - if (Args.value("SIGNATURE").isEmpty()) // absolutely mandatory - bLooksOk = false; - - if (bLooksOk) { - QString TempPath = QDir::tempPath() + "/Sbie+Certificate.dat"; - QFile CertFile(TempPath); - if (CertFile.open(QFile::WriteOnly)) { - CertFile.write(Certificate); - CertFile.close(); - } - - WindowsMoveFile(TempPath.replace("/", "\\"), CertPath.replace("/", "\\")); - } - else { - QMessageBox::critical(widget, "Sandboxie-Plus", tr("This does not look like a certificate, please enter the entire certificate not just a portion of it.")); - return false; - } - } - else if(!g_Certificate.isEmpty()){ - WindowsMoveFile(CertPath.replace("/", "\\"), ""); - } - - if (Certificate.isEmpty()) - return false; - - if (!theAPI->ReloadCert().IsError()) - { - g_FeatureFlags = theAPI->GetFeatureFlags(); - theGUI->UpdateCertState(); - - if (g_CertInfo.expired || g_CertInfo.outdated) { - if(g_CertInfo.expired) - QMessageBox::information(widget, "Sandboxie-Plus", tr("This certificate is unfortunately expired.")); - else - QMessageBox::information(widget, "Sandboxie-Plus", tr("This certificate is unfortunately outdated.")); - } - else { - QMessageBox::information(widget, "Sandboxie-Plus", tr("Thank you for supporting the development of Sandboxie-Plus.")); - } - - g_Certificate = Certificate; - return true; - } - else - { - QMessageBox::critical(widget, "Sandboxie-Plus", tr("This support certificate is not valid.")); - - g_CertInfo.State = 0; - g_Certificate.clear(); - return false; - } -} - void CSettingsWindow::apply() { if (!ui.btnEditIni->isEnabled()) diff --git a/SandboxiePlus/SandMan/Windows/SettingsWindow.h b/SandboxiePlus/SandMan/Windows/SettingsWindow.h index db22691f..bca2b341 100644 --- a/SandboxiePlus/SandMan/Windows/SettingsWindow.h +++ b/SandboxiePlus/SandMan/Windows/SettingsWindow.h @@ -24,8 +24,6 @@ public: virtual void accept() {} virtual void reject(); - static bool ApplyCertificate(const QByteArray &Certificate, QWidget* widget); - static void LoadCertificate(); signals: @@ -90,6 +88,7 @@ private: }; void CSettingsWindow__AddContextMenu(); +void CSettingsWindow__RemoveContextMenu(); void CSettingsWindow__AddBrowserIcon(); void WindowsMoveFile(const QString& from, const QString& to); diff --git a/SandboxiePlus/SandMan/sandman_zh_CN.ts b/SandboxiePlus/SandMan/sandman_zh_CN.ts index 2fa8c327..b8ce70e5 100644 --- a/SandboxiePlus/SandMan/sandman_zh_CN.ts +++ b/SandboxiePlus/SandMan/sandman_zh_CN.ts @@ -253,12 +253,12 @@ 浏览文件夹 - + This sandbox has been deleted hence configuration can not be saved. 该沙盒已被删除,因此配置无法保存 - + Some changes haven't been saved yet, do you really want to close this options window? 部分变更未保存,确定要关闭这个选项窗口吗? @@ -302,10 +302,10 @@ - - - - + + + + Group: %1 组: %1 @@ -315,7 +315,7 @@ 请输入新组的名称 - + Enter program: 请输入程序: @@ -432,9 +432,9 @@ - - - + + + All Programs 所有程序 @@ -452,7 +452,7 @@ - + Template values can not be edited. 模板值无法被编辑 @@ -559,25 +559,25 @@ - + Any 任意 - + TCP TCP - + UDP UDP - + ICMP ICMP @@ -598,7 +598,7 @@ - + Allow 允许 @@ -613,7 +613,7 @@ 阻止 (网络设备) - + Block 阻止 @@ -860,39 +860,39 @@ Full path: %4 删除所有内容,包括所有快照 - + Original location 原始位置 - + Browse for location 浏览位置 - + Clear folder list 清除文件夹列表 - - - + + + Select Directory 选择目录 - + Close until all programs stop in this box 关闭,在沙盒内全部程序停止后再显示 - + There are %1 new files available to recover. 有 %1 个新文件可供恢复 - + There are %1 files and %2 folders in the sandbox, occupying %3 of disk space. There are %1 files and %2 folders in the sandbox, occupying %3 bytes of disk space. 此沙盒中共有 %1 个文件和 %2 个文件夹,占用了 %3 磁盘空间 @@ -901,22 +901,22 @@ Full path: %4 CSandBox - + Waiting for folder: %1 正在等待文件夹: %1 - + Deleting folder: %1 正在删除文件夹: %1 - + Merging folders: %1 &gt;&gt; %2 正在合并文件夹: %1 &gt;&gt; %2 - + Finishing Snapshot Merge... 正在完成快照合并... @@ -987,294 +987,313 @@ Full path: %4 CSandMan - - + + Sandboxie-Plus v%1 Sandboxie-Plus v%1 - + + Reset Columns 重置列 - + + Copy Cell 复制此格 - + + Copy Row 复制此行 - + + Copy Panel 复制此表 - + Time|Message 时间|消息 - + Sbie Messages 沙盒消息 - + Trace Log 跟踪日志 - + Show/Hide 显示/隐藏 - + &Sandbox 沙盒(&S) - + Create New Box 新建沙盒 - + Create Box Group 新建沙盒组 - + Terminate All Processes 终止所有进程 - + Window Finder 检查窗口是否在沙盒中运行 - + &Maintenance 维护(&M) - + Connect 连接 - + Disconnect 断开 - + Stop All 停止所有 - + &Advanced 高级(&A) - + Install Driver 安装驱动 - + Start Driver 启动驱动 - + Stop Driver 停止驱动 - + Uninstall Driver 卸载驱动 - + Install Service 安装服务 - + Start Service 启动服务 - + Stop Service 停止服务 - + Uninstall Service 卸载服务 - + Uninstall All 全部卸载 - + Exit 退出 - + &View 视图(&V) - + Simple View 简易视图 - + Advanced View 高级视图 - + Always on Top 窗口置顶 - + Show Hidden Boxes 显示隐藏沙盒 - + Show All Sessions 显示所有会话的进程 - + Clean Up 清理 - + Cleanup Processes 清理所有记录 - + Cleanup Message Log 清理消息日志 - + Cleanup Trace Log 清理跟踪日志 - + Keep terminated 保持终止 - + &Options 选项(&O) - + Global Settings 全局设置 - + Reset all hidden messages 重置所有已隐藏消息 - + Reset all GUI options 重置所有图形界面设置选项 - + Edit ini file 编辑配置文件 - + Reload ini file 重载配置文件 - + Trace Logging 启用跟踪日志 - + &Help 帮助(&H) - + Support Sandboxie-Plus with a Donation 捐赠支持 Sandboxie-Plus - + Visit Support Forum 访问用户支持论坛 - + Online Documentation 在线文档 - + Check for Updates 检查更新 - + About the Qt Framework 关于 Qt 框架 - - + + About Sandboxie-Plus 关于 Sandboxie-Plus - + Cleanup 清理 - + + <a href="sbie://update/package" style="color: red;">There is a new build of Sandboxie-Plus available</a> + <a href="sbie://update/package" style="color: red;">Sandboxie-Plus 存在可供选择的新构建更新</a> + + + + Click to install update + 应用更新 + + + <a href="https://sandboxie-plus.com/go.php?to=patreon">Support Sandboxie-Plus on Patreon</a> <a href="https://sandboxie-plus.com/go.php?to=patreon">在 Patreon 上捐赠 Sandboxie-Plus</a> - + + Click to open web browser + 在浏览器打开捐赠页面 + + + Do you want to close Sandboxie Manager? 确定要关闭 Sandboxie 管理器? - + Sandboxie-Plus was running in portable mode, now it has to clean up the created services. This will prompt for administrative privileges. Do you want to do the clean up? @@ -1283,32 +1302,32 @@ Do you want to do the clean up? 是否确认清理? - - - - - - + + + + + + Don't show this message again. 不再显示此消息 - + This box provides enhanced security isolation, it is suitable to test untrusted software. 此类沙盒提供增强的安全隔离,它适用于测试不受信任的软件 - + This box provides standard isolation, it is suitable to run your software to enhance security. 此类沙盒提供标准的隔离,它适用于以安全的方式来运行你的软件 - + This box does not enforce isolation, it is intended to be used as an application compartment for software virtualization only. 此类沙盒不执行隔离,它的目的是将一个应用程序虚拟化 - + This box prevents access to all user data locations, except explicitly granted in the Resource Access options. @@ -1317,124 +1336,129 @@ This box prevents access to all user data locations, except explicitly granted i 此类沙盒将限制沙盒内程序对沙盒外数据的访问,除非在资源访问选项中明确授权 - + Unknown operation '%1' requested via command line 来自命令行的未知操作请求 '%1' - + - Driver/Service NOT Running! - 驱动程序/服务未运行! - + - Deleting Sandbox Content - 正在删除沙盒内容 - + Executing OnBoxDelete: %1 在删除沙盒时执行: %1 - + Auto Deleting %1 Content 自动删除 %1 的内容 - + Auto deleting content of %1 自动删除 %1 的内容 - - - + + + Sandboxie-Plus - Error Sandboxie-Plus - 错误 - + Failed to stop all Sandboxie components 停止全部的 Sandboxie 组件失败 - + Failed to start required Sandboxie components 启动所需的 Sandboxie 组件失败 - + Maintenance operation Successful 维护操作成功 - + + <p>A Sandboxie-Plus update has been downloaded to the following location:</p><p><a href="%2">%1</a></p><p>Do you want to begin the installation? If any programs are running sandboxed, they will be terminated.</p> + <p>Sandboxie-Plus 更新包已下载到:</p><p><a href="%2">%1</a></p><p>是否立即安装?(任何在沙盒中运行的程序都将被自动终止)</p> + + + The supporter certificate is not valid for this build, please get an updated certificate 此赞助者凭据对该版本沙盒无效,请获取可用的新凭据 - + The supporter certificate has expired%1, please get an updated certificate 此赞助者凭据已过期%1,请获取可用的新凭据 - + , but it remains valid for the current build ,但它对当前构建的沙盒版本仍然有效 - + The supporter certificate will expire in %1 days, please get an updated certificate 此赞助者凭据将在 %1 天后过期,请获取可用的新凭据 - + Checking for certificate... 检索凭据... - + No certificate found on server! 未在服务器检索到凭据! - + There is no updated certificate available. 目前没有可用的凭据更新 - + The selected feature set is only available to project supporters. Processes started in a box with this feature set enabled without a supporter certificate will be terminated after 5 minutes.<br /><a href="https://sandboxie-plus.com/go.php?to=sbie-get-cert">Become a project supporter</a>, and receive a <a href="https://sandboxie-plus.com/go.php?to=sbie-cert">supporter certificate</a> 选定的特性只对项目赞助者可用。如果没有赞助者凭据,在启用该特性的沙盒里启动的进程,将在 5 分钟后将被终止。<br /><a href="https://sandboxie-plus.com/go.php?to=sbie-get-cert">成为项目赞助者</a>,以获得<a href="https://sandboxie-plus.com/go.php?to=sbie-cert">赞助者凭据</a> - + Error Status: 0x%1 (%2) 错误状态: 0x%1 (%2) - + Unknown 未知 - + Failed to remove old box data files 无法删除旧沙盒中的数据文件 - + The operation was canceled by the user 该操作已被用户取消 - + Unknown Error Status: 0x%1 未知错误状态: 0x%1 - + No new updates found, your Sandboxie-Plus is up-to-date. Note: The update check is often behind the latest GitHub release to ensure that only tested updates are offered. @@ -1443,63 +1467,63 @@ Note: The update check is often behind the latest GitHub release to ensure that 注意: 更新检查通常落后于 GitHub 释出的版本,以确保只提供经过测试的更新 - + <h3>About Sandboxie-Plus</h3><p>Version %1</p><p>Copyright (c) 2020-2022 by DavidXanatos</p> <h3>关于 Sandboxie-Plus</h3><p>版本 %1</p><p>Copyright (c) 2020-2022 by DavidXanatos</p> - + This copy of Sandboxie+ is certified for: %1 此 Sandboxie+ 副本已授权给: %1 - + Sandboxie+ is free for personal and non-commercial use. Sandboxie+ 可免费用于个人和非商业用途 - + Sandboxie-Plus is an open source continuation of Sandboxie.<br />Visit <a href="https://sandboxie-plus.com">sandboxie-plus.com</a> for more information.<br /><br />%3<br /><br />Driver version: %1<br />Features: %2<br /><br />Icons from <a href="https://icons8.com">icons8.com</a> Sandboxie-Plus 是著名程序 Sandboxie 自开源以来的一个延续<br />访问 <a href="https://sandboxie-plus.com">sandboxie-plus.com</a> 来了解更多信息<br /><br />%3<br /><br />驱动版本: %1<br />特性: %2<br /><br />图标来源: <a href="https://icons8.com">icons8.com</a> - + Default sandbox not found; creating: %1 未找到默认沙盒,正在创建:%1 - + Do you want to check if there is a new version of Sandboxie-Plus? 您是否想检查有无 Sandboxie-Plus 新版本发布? - + WARNING: Sandboxie-Plus.ini in %1 cannot be written to, settings will not be saved. 警告: %1 中的 Sandboxie-Plus.ini 不能被写入,设置将不会被保存 - - + + Pause Forcing Programs 停用必沙程序规则 - + Some compatibility templates (%1) are missing, probably deleted, do you want to remove them from all boxes? 部分兼容性模板(%1)丢失,可能已被删除,是否要在所有沙盒中移除? - + Cleaned up removed templates... 已清理缺失的模板... - + - Portable - 便携版 - + Sandboxie-Plus was started in portable mode, do you want to put the Sandbox folder into its parent directory? Yes will choose: %1 No will choose: %2 @@ -1508,385 +1532,384 @@ No will choose: %2 “否”将选择目录: %2 - + - NOT connected - 未连接 - + The program %1 started in box %2 will be terminated in 5 minutes because the box was configured to use features exclusively available to project supporters. 在沙盒 %2 中启动的程序 %1 将在 5 分钟之后自动终止,因为此沙盒被配置为使用项目赞助者的特供功能 - + The box %1 is configured to use features exclusively available to project supporters, these presets will be ignored. 沙盒 %1 被配置为使用项目赞助者专有的沙盒类型,这些预设选项将被忽略 - + <br /><a href="https://sandboxie-plus.com/go.php?to=sbie-get-cert">Become a project supporter</a>, and receive a <a href="https://sandboxie-plus.com/go.php?to=sbie-cert">supporter certificate</a> <br /><a href="https://sandboxie-plus.com/go.php?to=sbie-get-cert">成为项目赞助者</a>,以获得<a href="https://sandboxie-plus.com/go.php?to=sbie-cert">赞助者凭据</a> - + PID %1: 进程 PID %1: - + %1 (%2): %1 (%2): - + Recovering file %1 to %2 恢复文件 %1 到 %2 - + The file %1 already exists, do you want to overwrite it? 文件 %1 已存在,要覆盖它吗? - + Do this for all files! 对所有文件都执行此操作! - + Failed to recover some files: 部分文件恢复失败: - + Only Administrators can change the config. 仅管理员可更改该配置 - + Please enter the configuration password. 请输入配置保护密码 - + Login Failed: %1 登录失败:%1 - + Do you want to terminate all processes in all sandboxes? 确定要终止所有沙盒中的所有进程吗? - + Terminate all without asking 终止所有且不再询问 - + Sandboxie-Plus was started in portable mode and it needs to create necessary services. This will prompt for administrative privileges. Sandboxie-Plus 正以便携模式启动,需要创建所需的服务,这将会寻求管理员权限 - + CAUTION: Another agent (probably SbieCtrl.exe) is already managing this Sandboxie session, please close it first and reconnect to take over. 警告:另一代理程序 (可能是 SbieCtrl.exe) 已接管当前 Sandboxie 会话,请将其关闭,然后尝试重新连接以接管控制 - + Executing maintenance operation, please wait... 正在执行操作维护,请稍候... - + Do you also want to reset hidden message boxes (yes), or only all log messages (no)? 请确认是否要重置已隐藏的消息框(选“是”),或者仅重置所有日志消息(选“否”)? - + The changes will be applied automatically whenever the file gets saved. 每次该文件被保存时,更改将自动应用 - + The changes will be applied automatically as soon as the editor is closed. 编辑器被关闭后,更改将很快自动应用 - + Administrator rights are required for this operation. 此操作需要管理员权限 - + Failed to execute: %1 执行失败:%1 - + Failed to connect to the driver 连接驱动程序失败 - + Failed to communicate with Sandboxie Service: %1 无法与 Sandboxie 服务通信:%1 - + An incompatible Sandboxie %1 was found. Compatible versions: %2 发现不兼容的 Sandboxie %1,其它兼容的版本:%2 - + Can't find Sandboxie installation path. 无法找到 Sandboxie 的安装路径 - + Failed to copy configuration from sandbox %1: %2 复制沙盒配置 %1: %2 失败 - + A sandbox of the name %1 already exists 名为 %1 的沙盒已存在 - + Failed to delete sandbox %1: %2 删除沙盒 %1: %2 失败 - + The sandbox name can not be longer than 32 characters. 沙盒名称不能超过 32 个字符 - + The sandbox name can not be a device name. 沙盒名称不能为设备名称 - + The sandbox name can contain only letters, digits and underscores which are displayed as spaces. 沙盒名称只能包含字母、数字和下划线(显示为空格) - + Failed to terminate all processes 终止所有进程失败 - + Delete protection is enabled for the sandbox 该沙盒已启用删除保护 - + All sandbox processes must be stopped before the box content can be deleted 在删除沙盒内容之前,必须先停止沙盒内的所有进程 - + Error deleting sandbox folder: %1 删除沙盒文件夹出错:%1 - + A sandbox must be emptied before it can be deleted. 沙盒被删除前必须清空 - + Failed to move directory '%1' to '%2' 移动目录 '%1' 到 '%2' 失败 - + This Snapshot operation can not be performed while processes are still running in the box. 因有进程正在沙盒中运行,此快照操作无法完成 - + Failed to create directory for new snapshot 创建新快照的目录失败 - + Installation Directory: %1 安装目录: %1 - + Sandboxie-Plus Version: %1 (%2) Sandboxie-Plus 版本号: %1 (%2) - + Current Config: %1 当前配置: %1 - + Data Directory: %1 数据存放目录: %1 - + Please enter the duration, in seconds, for disabling Forced Programs rules. 请输入「停用必沙程序规则」的持续时间 (单位: 秒) - + Maintenance operation failed (%1) 维护作业执行失败 (%1) - + Failed to copy box data files 复制沙盒数据文件失败 - + Snapshot not found 没有找到快照 - + Error merging snapshot directories '%1' with '%2', the snapshot has not been fully merged. 合并快照目录 '%1' 和 '%2' 出错,快照没有被完全合并 - + Failed to remove old snapshot directory '%1' 移除旧快照的目录 '%1' 失败 - + Can't remove a snapshot that is shared by multiple later snapshots 无法移除被多个后续快照所共享的快照 - + You are not authorized to update configuration in section '%1' 您未被授权在 '%1' 更新配置 - + Failed to set configuration setting %1 in section %2: %3 在 %2: %3 中设定配置设置 %1 失败 - + Can not create snapshot of an empty sandbox 无法为空的沙盒创建快照 - + A sandbox with that name already exists 已存在同名沙盒 - + The config password must not be longer than 64 characters 配置保护密码长度不能超过 64 个字符 - + Operation failed for %1 item(s). %1 项操作失败 - + Do you want to open %1 in a sandboxed (yes) or unsandboxed (no) Web browser? 是否在沙盒中的浏览器打开链接 %1 ? - + Remember choice for later. 记住选择供之后使用 - + Checking for updates... 正在检查更新... - + server not reachable 无法连接到服务器 - - + + Failed to check for updates, error: %1 检查更新失败,错误:%1 - + <p>Do you want to go to the <a href="%1">info page</a>?</p> <p>您是否要前往< "%1">信息页</a>?</p> - + Don't show this announcement in the future. 今后不再显示此公告 - + <p>There is a new version of Sandboxie-Plus available.<br /><font color='red'>New version:</font> <b>%1</b></p> <p>有新版 Sandboxie-Plus 可用,<br /><font color='red'>新版本:</font><b>%1</b></p> - + <p>Do you want to download the latest version?</p> <p>是否下载最新版本?</p> - + <p>Do you want to go to the <a href="%1">download page</a>?</p> <p>是否要打开<a href="%1">下载页面</a>?</p> - + Don't show this message anymore. 不再显示此消息 - + Downloading new version... 正在下载新版本... - + Failed to download update from: %1 从 %1 下载更新文件时失败了 - <p>New Sandboxie-Plus has been downloaded to the following location:</p><p><a href="%2">%1</a></p><p>Do you want to begin the installation? If any programs are running sandboxed, they will be terminated.</p> - <p>新版 Sandboxie-Plus 已下载到下列位置:</p><p><a href="%2">%1</a></p><p>您想开始安装吗?开始安装时,任何在沙盒运行中的程序都将被终止.</p> + <p>新版 Sandboxie-Plus 已下载到下列位置:</p><p><a href="%2">%1</a></p><p>您想开始安装吗?开始安装时,任何在沙盒运行中的程序都将被终止.</p> - + The selected window is running as part of program %1 in sandbox %2 选择的窗口正作为程序 %1 的一部分,并运行在沙盒 %2 中 - + The selected window is not running as part of any sandboxed program. 选择的窗口并未作为任何沙盒化程序的一部分而运行 - + Drag the Finder Tool over a window to select it, then release the mouse to check if the window is sandboxed. 拖拽准星到被选窗口上,松开鼠标检查窗口是否来自沙盒化的程序 - + Sandboxie-Plus - Window Finder Sandboxie-Plus - 窗口探查器 @@ -2570,17 +2593,17 @@ No will choose: %2 Don't show any icon - 不显示图标 + 不显示 Show Plus icon - 显示 Plus 版图标 + Plus 版 Show Classic icon - 显示经典版图标 + 经典版 @@ -2613,83 +2636,83 @@ No will choose: %2 关闭程序 - + Run &Sandboxed 在沙盒中运行(&S) - + This supporter certificate has expired, please <a href="sbie://update/cert">get an updated certificate</a>. 此赞助者凭据已过期,请<a href="sbie://update/cert">获取新凭据</a> - + This supporter certificate will <font color='red'>expire in %1 days</font>, please <a href="sbie://update/cert">get an updated certificate</a>. 此赞助者凭据将<font color='red'>在 %1 天后过期</font>,请<a href="sbie://update/cert">获取新凭据</a> - + Run &Un-Sandboxed 在沙盒外运行(&U) - + This does not look like a certificate, please enter the entire certificate not just a portion of it. 这看起来不像是一份凭据,请输入完整的凭据,而不仅仅是其中的一部分 - + This certificate is unfortunately expired. 很不幸此凭据已过期 - + This certificate is unfortunately outdated. 很不幸此凭据已过时 - + Thank you for supporting the development of Sandboxie-Plus. 感谢您对 Sandboxie-Plus 开发工作的支持 - + This support certificate is not valid. 此赞助者凭据无效 - - + + Select Directory 选择目录 - + Please enter the new configuration password. 请输入新的配置保护密码 - + Please re-enter the new configuration password. 请再次输入新的配置保护密码 - + Passwords did not match, please retry. 输入的密码不一致,请重新输入 - + Process 进程 - + Folder 文件夹 - + Please enter a program file name 请输入一个程序文件名 @@ -3722,7 +3745,7 @@ The process match level has a higher priority than the specificity and describes Open access to COM infrastructure (not recommended) - 开放 COM infrastructure 的访问权限 (不推荐) + 开放“COM 基础结构”的访问权限 (不推荐) @@ -4068,27 +4091,27 @@ Please note that this values are currently user specific and saved globally for QPlatformTheme - + OK 确定 - + Apply 应用 - + Cancel 取消 - + &Yes 是(&Y) - + &No 否(&N) @@ -4188,349 +4211,358 @@ Please note that this values are currently user specific and saved globally for 常规选项 - + Show Notifications for relevant log Messages 显示有关日志消息的通知 - + Open urls from this ui sandboxed - 此用户界面上的链接在沙盒中打开 + 总是在沙盒中打开设置页面的链接 - + Run box operations asynchronously whenever possible (like content deletion) 尽可能以异步方式执行沙盒的各类操作 (如内容删除) - - Show boxes in tray list: - 显示沙盒列表: + + General Options + 常规选项 - + + Show boxes in tray list: + 沙盒列表显示: + + + Add 'Run Un-Sandboxed' to the context menu 添加“在沙盒外运行”到右键菜单 - + Show a tray notification when automatic box operations are started 当沙盒自动化作业事件开始执行时,弹出托盘通知 - + Activate Kernel Mode Object Filtering 激活内核模式的对象过滤器 - + Hook selected Win32k system calls to enable GPU acceleration (experimental) Hook 选定的 Win32k 系统调用钩子以启用 GPU 加速 (实验性) - + Compatibility 软件兼容 - + Edit ini Section 配置文本 - + Save 保存 - + Edit ini 编辑配置 - + Cancel 取消 - + Support 捐赠支持 - + Install updates automatically 自动安装更新 - + This supporter certificate has expired, please <a href="sbie://update/cert">get an updated certificate</a>. 此赞助者凭据已过期,请<a href="sbie://update/cert">获取新凭据</a> - + Check periodically for updates of Sandboxie-Plus 定期检查有无 Sandboxie-Plus 更新 - + + Download Updates automatically + 仅下载更新包 + + + In the future, don't notify about certificate expiration 不再通知凭据过期的情况 - Restart required (!) - 需要重启本软件 (!) + 需要重启本软件 (!) - + Start UI with Windows - 系统启动时启动界面 + 随系统启动沙盒管理器 - + Add 'Run Sandboxed' to the explorer context menu 向文件资源管理器添加“在沙盒中运行”右键菜单 - + On main window close: - 关闭主窗口: + 关闭主窗口时: - + Start UI when a sandboxed process is started - 进程在沙盒中启动时启动界面 + 随沙盒化应用启动沙盒管理器 - + Show first recovery window when emptying sandboxes 清空沙盒时先显示恢复窗口 - + Use Dark Theme (fully applied after a restart) 使用暗色主题 (重启程序后完全应用) - + Config protection 保护配置 - + Sandbox <a href="sbie://docs/ipcrootpath">ipc root</a>: 沙盒 <a href="sbie://docs/ipcrootpath">IPC 根目录</a>: - + Sandbox <a href="sbie://docs/keyrootpath">registry root</a>: 沙盒 <a href="sbie://docs/keyrootpath">注册表根目录</a>: - + Clear password when main window becomes hidden 主窗口隐藏时清除密码 - + Only Administrator user accounts can use Pause Forcing Programs command Only Administrator user accounts can use Pause Forced Programs Rules command 仅管理员用户可「停用必沙程序规则」 - + Sandbox <a href="sbie://docs/filerootpath">file system root</a>: 沙盒 <a href="sbie://docs/filerootpath">文件系统根目录</a>: - + Hotkey for terminating all boxed processes: 终止所有沙盒内进程的快捷键: - + Systray options 任务栏托盘区域选项 - + Show recoverable files as notifications 将可恢复的文件以通知形式显示 - + UI Language: 界面语言: - + Show Icon in Systray: - 显示托盘图标: + 托盘图标显示: - + Shell Integration 系统集成 - + Run Sandboxed - Actions “在沙盒中运行”选项 - + Always use DefaultBox 总是使用 DefaultBox - + Start Sandbox Manager 沙盒管理器启动选项 - + Advanced Config 高级选项 - + Use Windows Filtering Platform to restrict network access 使用 Windows 筛选平台 (WFP) 限制网络访问 - + Separate user folders 独立的用户文件夹 - + Sandboxing features 沙盒功能 - + Supporters of the Sandboxie-Plus project receive a <a href="https://sandboxie-plus.com/go.php?to=sbie-cert">supporter certificate</a>. It's like a license key but for awesome people using open source software. :-) Sandboxie-Plus 项目的赞助者将收到<a href="https://sandboxie-plus.com/go.php?to=sbie-cert">赞助者凭据</a>,这类似于许可密钥,是为拥抱开源软件的优秀人士准备的 :-) - + Program Control 程序控制 - + Config Protection 保护配置 - + Only Administrator user accounts can make changes 仅管理员用户可更改 - + Password must be entered in order to make changes 更改必须输入密码 - + Change Password 更改密码 - + Keeping Sandboxie up to date with the rolling releases of Windows and compatible with all web browsers is a never-ending endeavor. Please consider supporting this work with a donation.<br />You can support the development with a <a href="https://sandboxie-plus.com/go.php?to=donate">PayPal donation</a>, working also with credit cards.<br />Or you can provide continuous support with a <a href="https://sandboxie-plus.com/go.php?to=patreon">Patreon subscription</a>. 使 Sandboxie 与 Windows 的滚动更新保持同步,并和主流浏览器保持兼容性,是一项永无止境的努力,请考虑捐赠以支持这项工作<br />您可以通过 <a href="https://sandboxie-plus.com/go.php?to=donate">PayPal 捐赠</a> (支持使用信用卡付款)来支持项目的开发<br />您也可以通过 <a href="https://sandboxie-plus.com/go.php?to=patreon">Patreon 订阅</a> 来提供持续的捐赠支持 - + Enter the support certificate here 在此输入赞助者凭据 - + Support Settings 支持设置 - + Portable root folder 便携化根目录 - + ... ... - + Sandbox default 沙盒预设 - + Watch Sandboxie.ini for changes 监控 Sandboxie.ini 变更 - - + + Name 名称 - + Path 路径 - + Remove Program 删除程序 - + Add Program 添加程序 - + When any of the following programs is launched outside any sandbox, Sandboxie will issue message SBIE1301. 下列程序在沙盒之外启动时,Sandboxie 将发出 SBIE1301 警告 - + Add Folder 添加文件夹 - + Prevent the listed programs from starting on this system 阻止下列程序在此系统中启动 - + Issue message 1308 when a program fails to start 程序启动失败时发出问题代码 1308 - + In the future, don't check software compatibility 之后不再检查软件兼容性 - + Enable 启用 - + Disable 禁用 - + Sandboxie has detected the following software applications in your system. Click OK to apply configuration settings, which will improve compatibility with these applications. These configuration settings will have effect in all existing sandboxes and in any new sandboxes. 沙盒已检测到系统中安装了以下软件,点击“确定”应用配置,将改进与这些软件的兼容性,这些配置将作用于所有沙盒,包括现存和未来新增的沙盒 diff --git a/SandboxiePlus/SbieShell/SbieShell.sln b/SandboxiePlus/SbieShell/SbieShell.sln new file mode 100644 index 00000000..5f7fb7ff --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShell.sln @@ -0,0 +1,75 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 16 +VisualStudioVersion = 16.0.29021.104 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{49D9D73C-85D3-4EE8-901F-5FEC9C17D7EE}" + ProjectSection(SolutionItems) = preProject + SbieShellPkg\AppxManifest.xml = SbieShellPkg\AppxManifest.xml + EndProjectSection +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "SbieShellExt", "SbieShellExt\SbieShellExt.vcxproj", "{16746B0C-5840-4ED4-AEDF-1ABB617827FD}" +EndProject +Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "SbieShell", "SbieShell\SbieShell.vcxproj", "{454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}" + ProjectSection(ProjectDependencies) = postProject + {16746B0C-5840-4ED4-AEDF-1ABB617827FD} = {16746B0C-5840-4ED4-AEDF-1ABB617827FD} + EndProjectSection +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Debug|ARM = Debug|ARM + Debug|ARM64 = Debug|ARM64 + Debug|x64 = Debug|x64 + Debug|x86 = Debug|x86 + Release|Any CPU = Release|Any CPU + Release|ARM = Release|ARM + Release|ARM64 = Release|ARM64 + Release|x64 = Release|x64 + Release|x86 = Release|x86 + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|Any CPU.ActiveCfg = Debug|ARM + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|Any CPU.Build.0 = Debug|ARM + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|ARM.ActiveCfg = Debug|ARM + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|ARM.Build.0 = Debug|ARM + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|ARM64.ActiveCfg = Debug|ARM64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|ARM64.Build.0 = Debug|ARM64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|x64.ActiveCfg = Debug|x64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|x64.Build.0 = Debug|x64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|x86.ActiveCfg = Debug|Win32 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Debug|x86.Build.0 = Debug|Win32 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|Any CPU.ActiveCfg = Release|Win32 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|ARM.ActiveCfg = Release|ARM + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|ARM.Build.0 = Release|ARM + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|ARM64.ActiveCfg = Release|ARM64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|ARM64.Build.0 = Release|ARM64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|x64.ActiveCfg = Release|x64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|x64.Build.0 = Release|x64 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|x86.ActiveCfg = Release|Win32 + {16746B0C-5840-4ED4-AEDF-1ABB617827FD}.Release|x86.Build.0 = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|Any CPU.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|Any CPU.Build.0 = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|ARM.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|ARM.Build.0 = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|ARM64.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|ARM64.Build.0 = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|x64.ActiveCfg = Debug|x64 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|x64.Build.0 = Debug|x64 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|x86.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Debug|x86.Build.0 = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Release|Any CPU.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Release|ARM.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Release|ARM64.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Release|x64.ActiveCfg = Release|x64 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Release|x64.Build.0 = Release|x64 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Release|x86.ActiveCfg = Release|Win32 + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181}.Release|x86.Build.0 = Release|Win32 + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {EAC535F2-5E51-4F68-AA53-B42E305FDCF8} + EndGlobalSection +EndGlobal diff --git a/SandboxiePlus/SbieShell/SbieShell/SbieShell.manifest b/SandboxiePlus/SbieShell/SbieShell/SbieShell.manifest new file mode 100644 index 00000000..2fa9b854 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShell/SbieShell.manifest @@ -0,0 +1,9 @@ + + + + + diff --git a/SandboxiePlus/SbieShell/SbieShell/SbieShell.vcxproj b/SandboxiePlus/SbieShell/SbieShell/SbieShell.vcxproj new file mode 100644 index 00000000..20f1fa89 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShell/SbieShell.vcxproj @@ -0,0 +1,139 @@ + + + + + Debug + Win32 + + + Debug + x64 + + + Release + Win32 + + + Release + x64 + + + + {454D5FCC-E25A-4B45-9CA2-01ABB0FA5181} + CustomActions + Win32Proj + 10.0.19041.0 + SbieShell + + + + Application + Unicode + v142 + + + Application + Unicode + v142 + + + Application + Unicode + v142 + + + Application + Unicode + v142 + + + + + + + + + + + + + + + + + + + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + + + _USRDLL;CUSTOMACTIONS_EXPORTS;%(PreprocessorDefinitions) + MultiThreaded + NotUsing + stdcpp17 + + + + + %(AdditionalDependencies) + + + + + _USRDLL;CUSTOMACTIONS_EXPORTS;%(PreprocessorDefinitions) + MultiThreaded + NotUsing + stdcpp17 + Disabled + + + + + %(AdditionalDependencies) + + + + + _USRDLL;CUSTOMACTIONS_EXPORTS;%(PreprocessorDefinitions) + MultiThreaded + NotUsing + stdcpp17 + + + + + %(AdditionalDependencies) + + + + + _USRDLL;CUSTOMACTIONS_EXPORTS;%(PreprocessorDefinitions) + MultiThreaded + NotUsing + stdcpp17 + Disabled + + + + + %(AdditionalDependencies) + + + + + + + + + + + + \ No newline at end of file diff --git a/SandboxiePlus/SbieShell/SbieShell/SbieShell.vcxproj.filters b/SandboxiePlus/SbieShell/SbieShell/SbieShell.vcxproj.filters new file mode 100644 index 00000000..9af0df18 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShell/SbieShell.vcxproj.filters @@ -0,0 +1,17 @@ + + + + + {4FC737F1-C7A5-4376-A066-2A32D752A2FF} + cpp;c;cxx;def;odl;idl;hpj;bat;asm;asmx + + + + + Source Files + + + + + + \ No newline at end of file diff --git a/SandboxiePlus/SbieShell/SbieShell/main.cpp b/SandboxiePlus/SbieShell/SbieShell/main.cpp new file mode 100644 index 00000000..b3c0fd02 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShell/main.cpp @@ -0,0 +1,194 @@ +#include +//#include + +#include +#include +#include + +//#pragma comment(lib, "windowsapp.lib") + +HMODULE g_combase = NULL; + +extern "C" +{ + int32_t __stdcall WINRT_GetRestrictedErrorInfo(void** info) noexcept { + typedef int32_t(__stdcall* FUNC)(void** info); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "GetRestrictedErrorInfo"); + return Func(info); + } + int32_t __stdcall WINRT_RoGetActivationFactory(void* classId, winrt::guid const& iid, void** factory) noexcept { + typedef int32_t(__stdcall* FUNC)(void* classId, winrt::guid const& iid, void** factory); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "RoGetActivationFactory"); + return Func(classId, iid, factory); + } + int32_t __stdcall WINRT_RoOriginateLanguageException(int32_t error, void* message, void* exception) noexcept { + typedef int32_t(__stdcall* FUNC)(int32_t error, void* message, void* exception); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "RoOriginateLanguageException"); + return Func(error, message, exception); + } + int32_t __stdcall WINRT_SetRestrictedErrorInfo(void* info) noexcept { + typedef int32_t(__stdcall* FUNC)(void* info); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "SetRestrictedErrorInfo"); + return Func(info); + } + + int32_t __stdcall WINRT_WindowsCreateString(wchar_t const* sourceString, uint32_t length, void** string) noexcept { + typedef int32_t(__stdcall* FUNC)(wchar_t const* sourceString, uint32_t length, void** string); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "WindowsCreateString"); + return Func(sourceString, length, string); + } + int32_t __stdcall WINRT_WindowsCreateStringReference(wchar_t const* sourceString, uint32_t length, void* hstringHeader, void** string) noexcept { + typedef int32_t(__stdcall* FUNC)(wchar_t const* sourceString, uint32_t length, void* hstringHeader, void** string); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "WindowsCreateStringReference"); + return Func(sourceString, length, hstringHeader, string); + } + int32_t __stdcall WINRT_WindowsDeleteString(void* string) noexcept { + typedef int32_t(__stdcall* FUNC)(void* string); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "WindowsDeleteString"); + return Func(string); + } + int32_t __stdcall WINRT_WindowsPreallocateStringBuffer(uint32_t length, wchar_t** charBuffer, void** bufferHandle) noexcept { + typedef int32_t(__stdcall* FUNC)(uint32_t length, wchar_t** charBuffer, void** bufferHandle); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "WindowsPreallocateStringBuffer"); + return Func(length, charBuffer, bufferHandle); + } + int32_t __stdcall WINRT_WindowsDeleteStringBuffer(void* bufferHandle) noexcept { + typedef int32_t(__stdcall* FUNC)(void* bufferHandle); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "WindowsDeleteStringBuffer"); + return Func(bufferHandle); + } + int32_t __stdcall WINRT_WindowsPromoteStringBuffer(void* bufferHandle, void** string) noexcept { + typedef int32_t(__stdcall* FUNC)(void* bufferHandle, void** string); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "WindowsPromoteStringBuffer"); + return Func(bufferHandle, string); + } + wchar_t const* __stdcall WINRT_WindowsGetStringRawBuffer(void* string, uint32_t* length) noexcept { + typedef wchar_t const* (__stdcall* FUNC)(void* string, uint32_t* length); + static FUNC Func = NULL; + if (Func == NULL) Func = (FUNC)GetProcAddress(g_combase, "WindowsGetStringRawBuffer"); + return Func(string, length); + } +} + +int RegisterSparsePackage(const std::wstring& sparseExtPath, const std::wstring& sparsePackagePath) +{ + winrt::Windows::Management::Deployment::PackageManager manager; + winrt::Windows::Management::Deployment::AddPackageOptions options; + winrt::Windows::Foundation::Uri externalUri(sparseExtPath.c_str()); + winrt::Windows::Foundation::Uri packageUri(sparsePackagePath.c_str()); + options.ExternalLocationUri(externalUri); + auto deploymentOperation = manager.AddPackageByUriAsync(packageUri, options); + + auto deployResult = deploymentOperation.get(); + + if (!SUCCEEDED(deployResult.ExtendedErrorCode())) + { + // Deployment failed + std::wstring error = L"AddPackageByUriAsync failed (Errorcode: "; + error += std::to_wstring(deployResult.ExtendedErrorCode()); + error += L"):\n"; + error += deployResult.ErrorText(); + + return -1; + } + return 0; +} + +int UnregisterSparsePackage(const std::wstring & sparsePackageName) +{ + + winrt::Windows::Management::Deployment::PackageManager manager; + winrt::Windows::Foundation::Collections::IIterable packages; + try + { + packages = manager.FindPackagesForUser(L""); + } + catch (winrt::hresult_error const& ex) + { + std::wstring error = L"FindPackagesForUser failed (Errorcode: "; + error += std::to_wstring(ex.code().value); + error += L"):\n"; + error += ex.message(); + return -1; + } + + for (const auto& package : packages) + { + if (package.Id().Name() != sparsePackageName) + continue; + + winrt::hstring fullName = package.Id().FullName(); + auto deploymentOperation = manager.RemovePackageAsync(fullName, winrt::Windows::Management::Deployment::RemovalOptions::None); + auto deployResult = deploymentOperation.get(); + if (SUCCEEDED(deployResult.ExtendedErrorCode())) + break; + + // Undeployment failed + std::wstring error = L"RemovePackageAsync failed (Errorcode: "; + error += std::to_wstring(deployResult.ExtendedErrorCode()); + error += L"):\n"; + error += deployResult.ErrorText(); + + return -1; + } + return 0; +} + +int wmain(int argc, wchar_t* argv[]) +{ + /*g_combase = LoadLibraryW(L"combase.dll"); + + if (argc < 2) + { + //wchar_t PackageName[MAX_PATH]; + //UINT32 Length = MAX_PATH; + //NTSTATUS status = GetCurrentPackageFullName(&Length, PackageName); + //if (status >= 0) + // printf("%S", PackageName); + return 0; + } + + if (wcsicmp(argv[1], L"-install") == 0) + { + wchar_t path[MAX_PATH]; + GetModuleFileName(NULL, path, MAX_PATH); + wchar_t* ptr = wcsrchr(path, L'\\'); + *ptr = L'\0'; + std::wstring sparseExtPath = path; + wcscat(path, L"\\SbieShellPkg.msix"); + std::wstring sparsePackagePath = path; + return RegisterSparsePackage(sparseExtPath, sparsePackagePath); + } + + else if (wcsicmp(argv[1], L"-uninstall") == 0) + { + std::wstring sparsePackageName = L"SandboxieShell"; + return UnregisterSparsePackage(sparsePackageName); + }*/ + + HMODULE ext = LoadLibraryW(L"SbieShellExt.dll"); + + typedef int (*FUNC)(); + if (wcsicmp(argv[1], L"-install") == 0) + { + FUNC func = (FUNC)GetProcAddress(ext, "RegisterPackage"); + func(); + } + else if (wcsicmp(argv[1], L"-uninstall") == 0) + { + FUNC func = (FUNC)GetProcAddress(ext, "RemovePackage"); + func(); + } + + return 0; +} \ No newline at end of file diff --git a/SandboxiePlus/SbieShell/SbieShellExt/SbieShellExt.vcxproj b/SandboxiePlus/SbieShell/SbieShellExt/SbieShellExt.vcxproj new file mode 100644 index 00000000..65ffbee6 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/SbieShellExt.vcxproj @@ -0,0 +1,348 @@ + + + + + Debug + ARM + + + Debug + ARM64 + + + Debug + Win32 + + + Release + ARM + + + Release + ARM64 + + + Release + Win32 + + + Debug + x64 + + + Release + x64 + + + + 16.0 + Win32Proj + {16746b0c-5840-4ed4-aedf-1abb617827fd} + PhotoStoreContextMenu + 10.0.19041.0 + + + + DynamicLibrary + true + v142 + Unicode + + + DynamicLibrary + true + v142 + Unicode + + + DynamicLibrary + false + v142 + true + Unicode + + + DynamicLibrary + false + v142 + true + Unicode + + + DynamicLibrary + true + v142 + Unicode + + + DynamicLibrary + true + v142 + Unicode + + + DynamicLibrary + false + v142 + true + Unicode + + + DynamicLibrary + false + v142 + true + Unicode + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + true + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + true + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + false + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + false + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + true + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + Link + + + true + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + false + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + Link + + + false + $(ProjectDir)\..\..\$(Platform)\$(Configuration) + + + + Level3 + true + WIN32;_DEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + Use + pch.h + stdcpp17 + + + Windows + true + false + runtimeobject.lib;%(AdditionalDependencies) + Source.def + + + + + Level3 + true + WIN32;_DEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + Use + pch.h + + + Windows + true + false + runtimeobject.lib;%(AdditionalDependencies) + Source.def + + + + + Level3 + true + true + true + WIN32;NDEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + Use + pch.h + stdcpp17 + + + Windows + true + true + true + false + runtimeobject.lib;%(AdditionalDependencies) + Source.def + + + + + Level3 + true + true + true + WIN32;NDEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + Use + pch.h + + + Windows + true + true + true + false + runtimeobject.lib;%(AdditionalDependencies) + Source.def + + + + + Level3 + true + _DEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + Use + pch.h + stdcpp17 + + + Windows + true + false + Source.def + runtimeobject.lib;%(AdditionalDependencies) + + + "$(WindowsSdkDir)bin\$(TargetPlatformVersion)\x64\makeappx.exe" pack /d $(SolutionDir)\SbieShellPkg /p $(ProjectDir)\..\..\$(Platform)\$(Configuration)\SbieShellPkg.msix /nv + $(ProjectDir)\..\..\$(Platform)\$(Configuration)\SbieShellPkg.msix + + + + + Level3 + true + _DEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + Use + pch.h + + + Windows + true + false + runtimeobject.lib;%(AdditionalDependencies) + Source.def + + + + + Level3 + true + true + true + NDEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + Use + pch.h + stdcpp17 + + + Windows + true + true + true + false + runtimeobject.lib;%(AdditionalDependencies) + Source.def + + + "$(WindowsSdkDir)bin\$(TargetPlatformVersion)\x64\makeappx.exe" pack /d $(SolutionDir)\SbieShellPkg /p $(ProjectDir)\..\..\$(Platform)\$(Configuration)\SbieShellPkg.msix /nv + $(ProjectDir)\..\..\$(Platform)\$(Configuration)\SbieShellPkg.msix + + + + + Level3 + true + true + true + NDEBUG;PHOTOSTORECONTEXTMENU_EXPORTS;_WINDOWS;_USRDLL;%(PreprocessorDefinitions) + true + Use + pch.h + + + Windows + true + true + true + false + runtimeobject.lib;%(AdditionalDependencies) + Source.def + + + + + + + + + + Create + Create + Create + Create + Create + Create + Create + Create + + + + + + + + + This project references NuGet package(s) that are missing on this computer. Use NuGet Package Restore to download them. For more information, see http://go.microsoft.com/fwlink/?LinkID=322105. The missing file is {0}. + + + + \ No newline at end of file diff --git a/SandboxiePlus/SbieShell/SbieShellExt/SbieShellExt.vcxproj.filters b/SandboxiePlus/SbieShell/SbieShellExt/SbieShellExt.vcxproj.filters new file mode 100644 index 00000000..1e57c7b1 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/SbieShellExt.vcxproj.filters @@ -0,0 +1,33 @@ + + + + + {4FC737F1-C7A5-4376-A066-2A32D752A2FF} + cpp;c;cc;cxx;c++;cppm;ixx;def;odl;idl;hpj;bat;asm;asmx + + + {93995380-89BD-4b04-88EB-625FBE52EBFB} + h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd + + + {67DA6AB6-F800-4c08-8B7A-83BB121AAD01} + rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms + + + + + Header Files + + + Header Files + + + + + Source Files + + + Source Files + + + \ No newline at end of file diff --git a/SandboxiePlus/SbieShell/SbieShellExt/Source.def b/SandboxiePlus/SbieShell/SbieShellExt/Source.def new file mode 100644 index 00000000..1ffdadd7 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/Source.def @@ -0,0 +1,5 @@ +LIBRARY +EXPORTS +DllCanUnloadNow PRIVATE +DllGetClassObject PRIVATE +DllGetActivationFactory PRIVATE \ No newline at end of file diff --git a/SandboxiePlus/SbieShell/SbieShellExt/dllmain.cpp b/SandboxiePlus/SbieShell/SbieShellExt/dllmain.cpp new file mode 100644 index 00000000..4791c7ec --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/dllmain.cpp @@ -0,0 +1,462 @@ +// dllmain.cpp : Defines the entry point for the DLL application. +#include "pch.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include + +using namespace Microsoft::WRL; + +std::wstring g_path; + +BOOL APIENTRY DllMain( HMODULE hModule, + DWORD ul_reason_for_call, + LPVOID lpReserved + ) +{ + switch (ul_reason_for_call) + { + case DLL_PROCESS_ATTACH: + { + wchar_t path[MAX_PATH]; + GetModuleFileName(hModule, path, MAX_PATH); + wchar_t* ptr = wcsrchr(path, L'\\'); + *ptr = L'\0'; + g_path = std::wstring(path); + + break; + } + case DLL_THREAD_ATTACH: + case DLL_THREAD_DETACH: + case DLL_PROCESS_DETACH: + break; + } + return TRUE; +} + +class TestExplorerCommandBase : public RuntimeClass, IExplorerCommand, IObjectWithSite> +{ +public: + virtual const wchar_t* Title() = 0; + virtual const EXPCMDFLAGS Flags() { return ECF_DEFAULT; } + virtual const EXPCMDSTATE State(_In_opt_ IShellItemArray* selection) { return ECS_ENABLED; } + + // IExplorerCommand + IFACEMETHODIMP GetTitle(_In_opt_ IShellItemArray* items, _Outptr_result_nullonfailure_ PWSTR* name) + { + *name = nullptr; + auto title = wil::make_cotaskmem_string_nothrow(Title()); + RETURN_IF_NULL_ALLOC(title); + *name = title.release(); + return S_OK; + } + IFACEMETHODIMP GetIcon(_In_opt_ IShellItemArray*, _Outptr_result_nullonfailure_ PWSTR* icon)// { *icon = nullptr; return E_NOTIMPL; } + { + std::wstring bpPath = g_path + L"\\SandMan.exe,-0"; + auto iconPath = wil::make_cotaskmem_string_nothrow(bpPath.c_str()); + RETURN_IF_NULL_ALLOC(iconPath); + *icon = iconPath.release(); + return S_OK; + } + IFACEMETHODIMP GetToolTip(_In_opt_ IShellItemArray*, _Outptr_result_nullonfailure_ PWSTR* infoTip) { *infoTip = nullptr; return E_NOTIMPL; } + IFACEMETHODIMP GetCanonicalName(_Out_ GUID* guidCommandName) { *guidCommandName = GUID_NULL; return S_OK; } + IFACEMETHODIMP GetState(_In_opt_ IShellItemArray* selection, _In_ BOOL okToBeSlow, _Out_ EXPCMDSTATE* cmdState) + { + *cmdState = State(selection); + return S_OK; + } + enum ECommand { + eExplore, + eOpen, + }; + virtual ECommand GetCommand() = 0; + IFACEMETHODIMP Invoke(_In_opt_ IShellItemArray* selection, _In_opt_ IBindCtx*) noexcept try + { + /*HWND parent = nullptr; + if (m_site) + { + ComPtr oleWindow; + RETURN_IF_FAILED(m_site.As(&oleWindow)); + RETURN_IF_FAILED(oleWindow->GetWindow(&parent)); + } + + std::wostringstream title; + title << Title(); + + if (selection) + { + DWORD count; + RETURN_IF_FAILED(selection->GetCount(&count)); + title << L" (" << count << L" selected items)"; + } + else + { + title << L"(no selected items)"; + } + + MessageBox(parent, title.str().c_str(), L"TestCommand", MB_OK);*/ + + + if (selection) + { + DWORD fileCount = 0; + RETURN_IF_FAILED(selection->GetCount(&fileCount)); + for (DWORD i = 0; i < fileCount; i++) + { + IShellItem* shellItem = nullptr; + selection->GetItemAt(i, &shellItem); + LPWSTR itemName = nullptr; + shellItem->GetDisplayName(SIGDN_FILESYSPATH, &itemName); + if (itemName) + { + std::wstring file = g_path + L"\\SandMan.exe"; + + std::wstring params = L"/box:__ask__"; + if(GetCommand() == eExplore) + params += L" C:\\WINDOWS\\explorer.exe"; + params += L" \""; + params += itemName; + params += L"\""; + + SHELLEXECUTEINFO shExecInfo = { sizeof(SHELLEXECUTEINFO) }; + shExecInfo.hwnd = nullptr; + shExecInfo.lpVerb = L"open"; + shExecInfo.lpFile = file.c_str(); + shExecInfo.lpParameters = params.c_str(); + shExecInfo.nShow = SW_NORMAL; + ShellExecuteEx(&shExecInfo); + + CoTaskMemFree(itemName); + } + } + } + + return S_OK; + } + CATCH_RETURN(); + + IFACEMETHODIMP GetFlags(_Out_ EXPCMDFLAGS* flags) { *flags = Flags(); return S_OK; } + IFACEMETHODIMP EnumSubCommands(_COM_Outptr_ IEnumExplorerCommand** enumCommands) { *enumCommands = nullptr; return E_NOTIMPL; } + + // IObjectWithSite + IFACEMETHODIMP SetSite(_In_ IUnknown* site) noexcept { m_site = site; return S_OK; } + IFACEMETHODIMP GetSite(_In_ REFIID riid, _COM_Outptr_ void** site) noexcept { return m_site.CopyTo(riid, site); } + +protected: + ComPtr m_site; +}; + +class __declspec(uuid("EA3E972D-62C7-4309-8F15-883263041E99")) ExploreCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"Explore Sandboxed"; } + ECommand GetCommand() { return eExplore; } +}; + +class __declspec(uuid("3FD2D9EE-DAF9-404A-9B7E-13B2DCD63950")) OpenCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"Open Sandboxed"; } + ECommand GetCommand() { return eOpen; } +}; + +CoCreatableClass(ExploreCommandHandler) +CoCreatableClass(OpenCommandHandler) + +CoCreatableClassWrlCreatorMapInclude(ExploreCommandHandler) +CoCreatableClassWrlCreatorMapInclude(OpenCommandHandler) + +/* +class TestExplorerCommandBase : public RuntimeClass, IExplorerCommand, IObjectWithSite> +{ +public: + virtual const wchar_t* Title() = 0; + virtual const EXPCMDFLAGS Flags() { return ECF_DEFAULT; } + virtual const EXPCMDSTATE State(_In_opt_ IShellItemArray* selection) { return ECS_ENABLED; } + + // IExplorerCommand + IFACEMETHODIMP GetTitle(_In_opt_ IShellItemArray* items, _Outptr_result_nullonfailure_ PWSTR* name) + { + *name = nullptr; + auto title = wil::make_cotaskmem_string_nothrow(Title()); + RETURN_IF_NULL_ALLOC(title); + *name = title.release(); + return S_OK; + } + IFACEMETHODIMP GetIcon(_In_opt_ IShellItemArray*, _Outptr_result_nullonfailure_ PWSTR* icon) { *icon = nullptr; return E_NOTIMPL; } + IFACEMETHODIMP GetToolTip(_In_opt_ IShellItemArray*, _Outptr_result_nullonfailure_ PWSTR* infoTip) { *infoTip = nullptr; return E_NOTIMPL; } + IFACEMETHODIMP GetCanonicalName(_Out_ GUID* guidCommandName) { *guidCommandName = GUID_NULL; return S_OK; } + IFACEMETHODIMP GetState(_In_opt_ IShellItemArray* selection, _In_ BOOL okToBeSlow, _Out_ EXPCMDSTATE* cmdState) + { + *cmdState = State(selection); + return S_OK; + } + IFACEMETHODIMP Invoke(_In_opt_ IShellItemArray* selection, _In_opt_ IBindCtx*) noexcept try + { + HWND parent = nullptr; + if (m_site) + { + ComPtr oleWindow; + RETURN_IF_FAILED(m_site.As(&oleWindow)); + RETURN_IF_FAILED(oleWindow->GetWindow(&parent)); + } + + std::wostringstream title; + title << Title(); + + if (selection) + { + DWORD count; + RETURN_IF_FAILED(selection->GetCount(&count)); + title << L" (" << count << L" selected items)"; + } + else + { + title << L"(no selected items)"; + } + + MessageBox(parent, title.str().c_str(), L"TestCommand", MB_OK); + return S_OK; + } + CATCH_RETURN(); + + IFACEMETHODIMP GetFlags(_Out_ EXPCMDFLAGS* flags) { *flags = Flags(); return S_OK; } + IFACEMETHODIMP EnumSubCommands(_COM_Outptr_ IEnumExplorerCommand** enumCommands) { *enumCommands = nullptr; return E_NOTIMPL; } + + // IObjectWithSite + IFACEMETHODIMP SetSite(_In_ IUnknown* site) noexcept { m_site = site; return S_OK; } + IFACEMETHODIMP GetSite(_In_ REFIID riid, _COM_Outptr_ void** site) noexcept { return m_site.CopyTo(riid, site); } + +protected: + ComPtr m_site; +}; + +class __declspec(uuid("3282E233-C5D3-4533-9B25-44B8AAAFACFA")) TestExplorerCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"ShellDemo Command1"; } + const EXPCMDSTATE State(_In_opt_ IShellItemArray* selection) override { return ECS_DISABLED; } +}; + +class __declspec(uuid("817CF159-A4B5-41C8-8E8D-0E23A6605395")) TestExplorerCommand2Handler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"ShellDemo ExplorerCommand2"; } +}; + +class SubExplorerCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"SubCommand"; } +}; + +class CheckedSubExplorerCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"CheckedSubCommand"; } + const EXPCMDSTATE State(_In_opt_ IShellItemArray* selection) override { return ECS_CHECKBOX | ECS_CHECKED; } +}; + +class RadioCheckedSubExplorerCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"RadioCheckedSubCommand"; } + const EXPCMDSTATE State(_In_opt_ IShellItemArray* selection) override { return ECS_CHECKBOX | ECS_RADIOCHECK; } +}; + +class HiddenSubExplorerCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"HiddenSubCommand"; } + const EXPCMDSTATE State(_In_opt_ IShellItemArray* selection) override { return ECS_HIDDEN; } +}; + +class EnumCommands : public RuntimeClass, IEnumExplorerCommand> +{ +public: + EnumCommands() + { + m_commands.push_back(Make()); + m_commands.push_back(Make()); + m_commands.push_back(Make()); + m_commands.push_back(Make()); + m_current = m_commands.cbegin(); + } + + // IEnumExplorerCommand + IFACEMETHODIMP Next(ULONG celt, __out_ecount_part(celt, *pceltFetched) IExplorerCommand** apUICommand, __out_opt ULONG* pceltFetched) + { + ULONG fetched{ 0 }; + wil::assign_to_opt_param(pceltFetched, 0ul); + + for (ULONG i = 0; (i < celt) && (m_current != m_commands.cend()); i++) + { + m_current->CopyTo(&apUICommand[0]); + m_current++; + fetched++; + } + + wil::assign_to_opt_param(pceltFetched, fetched); + return (fetched == celt) ? S_OK : S_FALSE; + } + + IFACEMETHODIMP Skip(ULONG celt) { return E_NOTIMPL; } + IFACEMETHODIMP Reset() + { + m_current = m_commands.cbegin(); + return S_OK; + } + IFACEMETHODIMP Clone(__deref_out IEnumExplorerCommand** ppenum) { *ppenum = nullptr; return E_NOTIMPL; } + +private: + std::vector> m_commands; + std::vector>::const_iterator m_current; +}; + +class __declspec(uuid("1476525B-BBC2-4D04-B175-7E7D72F3DFF8")) TestExplorerCommand3Handler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"ShellDemo CommandWithSubCommands"; } + const EXPCMDFLAGS Flags() override { return ECF_HASSUBCOMMANDS; } + + IFACEMETHODIMP EnumSubCommands(_COM_Outptr_ IEnumExplorerCommand** enumCommands) + { + *enumCommands = nullptr; + auto e = Make(); + return e->QueryInterface(IID_PPV_ARGS(enumCommands)); + } +}; + +class __declspec(uuid("30DEEDF6-63EA-4042-A7D8-0A9E1B17BB99")) TestExplorerCommand4Handler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"ShellDemo Command4"; } +}; + +class __declspec(uuid("50419A05-F966-47BA-B22B-299A95492348")) TestExplorerHiddenCommandHandler final : public TestExplorerCommandBase +{ +public: + const wchar_t* Title() override { return L"ShellDemo HiddenCommand"; } + const EXPCMDSTATE State(_In_opt_ IShellItemArray* selection) override { return ECS_HIDDEN; } +}; + +CoCreatableClass(TestExplorerCommandHandler) +CoCreatableClass(TestExplorerCommand2Handler) +CoCreatableClass(TestExplorerCommand3Handler) +CoCreatableClass(TestExplorerCommand4Handler) +CoCreatableClass(TestExplorerHiddenCommandHandler) + +CoCreatableClassWrlCreatorMapInclude(TestExplorerCommandHandler) +CoCreatableClassWrlCreatorMapInclude(TestExplorerCommand2Handler) +CoCreatableClassWrlCreatorMapInclude(TestExplorerCommand3Handler) +CoCreatableClassWrlCreatorMapInclude(TestExplorerCommand4Handler) +CoCreatableClassWrlCreatorMapInclude(TestExplorerHiddenCommandHandler) +*/ + + +STDAPI DllGetActivationFactory(_In_ HSTRING activatableClassId, _COM_Outptr_ IActivationFactory** factory) +{ + return Module::GetModule().GetActivationFactory(activatableClassId, factory); +} + +STDAPI DllCanUnloadNow() +{ + return Module::GetModule().GetObjectCount() == 0 ? S_OK : S_FALSE; +} + +STDAPI DllGetClassObject(_In_ REFCLSID rclsid, _In_ REFIID riid, _COM_Outptr_ void** instance) +{ + return Module::GetModule().GetClassObject(rclsid, riid, instance); +} + + +/////////////////////////////////////////////////////////////////////////////////////////////////////////7 +// + +#include +#include +#include + +#pragma comment(lib, "windowsapp.lib") + +int RegisterSparsePackage(const std::wstring& sparseExtPath, const std::wstring& sparsePackagePath) +{ + winrt::Windows::Management::Deployment::PackageManager manager; + winrt::Windows::Management::Deployment::AddPackageOptions options; + winrt::Windows::Foundation::Uri externalUri(sparseExtPath.c_str()); + winrt::Windows::Foundation::Uri packageUri(sparsePackagePath.c_str()); + options.ExternalLocationUri(externalUri); + auto deploymentOperation = manager.AddPackageByUriAsync(packageUri, options); + + auto deployResult = deploymentOperation.get(); + + if (!SUCCEEDED(deployResult.ExtendedErrorCode())) + { + // Deployment failed + std::wstring error = L"AddPackageByUriAsync failed (Errorcode: "; + error += std::to_wstring(deployResult.ExtendedErrorCode()); + error += L"):\n"; + error += deployResult.ErrorText(); + + return -1; + } + return 0; +} + +int UnregisterSparsePackage(const std::wstring& sparsePackageName) +{ + + winrt::Windows::Management::Deployment::PackageManager manager; + winrt::Windows::Foundation::Collections::IIterable packages; + try + { + packages = manager.FindPackagesForUser(L""); + } + catch (winrt::hresult_error const& ex) + { + std::wstring error = L"FindPackagesForUser failed (Errorcode: "; + error += std::to_wstring(ex.code().value); + error += L"):\n"; + error += ex.message(); + return -1; + } + + for (const auto& package : packages) + { + if (package.Id().Name() != sparsePackageName) + continue; + + winrt::hstring fullName = package.Id().FullName(); + auto deploymentOperation = manager.RemovePackageAsync(fullName, winrt::Windows::Management::Deployment::RemovalOptions::None); + auto deployResult = deploymentOperation.get(); + if (SUCCEEDED(deployResult.ExtendedErrorCode())) + break; + + // Undeployment failed + std::wstring error = L"RemovePackageAsync failed (Errorcode: "; + error += std::to_wstring(deployResult.ExtendedErrorCode()); + error += L"):\n"; + error += deployResult.ErrorText(); + + return -1; + } + return 0; +} + +extern "C" __declspec(dllexport) int RegisterPackage() +{ + std::wstring sparseExtPath = g_path; + std::wstring sparsePackagePath = g_path + L"\\SbieShellPkg.msix"; + return RegisterSparsePackage(sparseExtPath, sparsePackagePath); +} + +extern "C" __declspec(dllexport) int RemovePackage() +{ + std::wstring sparsePackageName = L"SandboxieShell"; + return UnregisterSparsePackage(sparsePackageName); +} diff --git a/SandboxiePlus/SbieShell/SbieShellExt/framework.h b/SandboxiePlus/SbieShell/SbieShellExt/framework.h new file mode 100644 index 00000000..54b83e94 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/framework.h @@ -0,0 +1,5 @@ +#pragma once + +#define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers +// Windows Header Files +#include diff --git a/SandboxiePlus/SbieShell/SbieShellExt/packages.config b/SandboxiePlus/SbieShell/SbieShellExt/packages.config new file mode 100644 index 00000000..50c8ee2a --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/packages.config @@ -0,0 +1,4 @@ + + + + \ No newline at end of file diff --git a/SandboxiePlus/SbieShell/SbieShellExt/pch.cpp b/SandboxiePlus/SbieShell/SbieShellExt/pch.cpp new file mode 100644 index 00000000..64b7eef6 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/pch.cpp @@ -0,0 +1,5 @@ +// pch.cpp: source file corresponding to the pre-compiled header + +#include "pch.h" + +// When you are using pre-compiled headers, this source file is necessary for compilation to succeed. diff --git a/SandboxiePlus/SbieShell/SbieShellExt/pch.h b/SandboxiePlus/SbieShell/SbieShellExt/pch.h new file mode 100644 index 00000000..885d5d62 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellExt/pch.h @@ -0,0 +1,13 @@ +// pch.h: This is a precompiled header file. +// Files listed below are compiled only once, improving build performance for future builds. +// This also affects IntelliSense performance, including code completion and many code browsing features. +// However, files listed here are ALL re-compiled if any one of them is updated between builds. +// Do not add files here that you will be updating frequently as this negates the performance advantage. + +#ifndef PCH_H +#define PCH_H + +// add headers that you want to pre-compile here +#include "framework.h" + +#endif //PCH_H diff --git a/SandboxiePlus/SbieShell/SbieShellPkg/AppxManifest.xml b/SandboxiePlus/SbieShell/SbieShellPkg/AppxManifest.xml new file mode 100644 index 00000000..c21026c1 --- /dev/null +++ b/SandboxiePlus/SbieShell/SbieShellPkg/AppxManifest.xml @@ -0,0 +1,110 @@ + + + + + SandboxieShell + Sparse Package + Assets\storelogo.png + true + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +