Skip to content
This repository has been archived by the owner on Dec 31, 2024. It is now read-only.

Commit

Permalink
Merge pull request oliverschwendener#250 from tkohlmeier/control-pane…
Browse files Browse the repository at this point in the history
…l-plugin

created control panel plugin
  • Loading branch information
oliverschwendener authored Dec 29, 2019
2 parents 72cdeab + 2c93549 commit 5426027
Show file tree
Hide file tree
Showing 17 changed files with 337 additions and 0 deletions.
7 changes: 7 additions & 0 deletions src/common/config/control-panel-options.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
export interface ControlPanelOptions {
isEnabled: boolean;
}

export const defaultControlPanelOptions: ControlPanelOptions = {
isEnabled: false,
};
3 changes: 3 additions & 0 deletions src/common/config/user-config-options.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import { ColorConverterOptions, defaultColorConverterOptions } from "./color-con
import { ApplicationSearchOptions, defaultApplicationSearchOptions } from "./application-search-options";
import { defaultDictionaryOptions, DictionaryOptions } from "./dictionary-options";
import { BrowserBookmarksOptions, defaultBrowserBookmarksOptions } from "./browser-bookmarks-options";
import { ControlPanelOptions, defaultControlPanelOptions } from "./control-panel-options";

export interface UserConfigOptions {
appearanceOptions: AppearanceOptions;
Expand All @@ -48,6 +49,7 @@ export interface UserConfigOptions {
colorConverterOptions: ColorConverterOptions;
uwpSearchOptions: UwpSearchOptions;
browserBookmarksOptions: BrowserBookmarksOptions;
controlPanelOptions: ControlPanelOptions;
}

export const defaultUserConfigOptions: UserConfigOptions = {
Expand All @@ -58,6 +60,7 @@ export const defaultUserConfigOptions: UserConfigOptions = {
colorConverterOptions: defaultColorConverterOptions,
colorThemeOptions: defaultColorThemeOptions,
commandlineOptions: defaultCommandlineOptions,
controlPanelOptions: defaultControlPanelOptions,
currencyConverterOptions: defaultCurrencyConverterOptions,
dictionaryOptions: defaultDictionaryOptions,
emailOptions: defaultEmailOptions,
Expand Down
2 changes: 2 additions & 0 deletions src/common/icon/default-icons.ts
Original file line number Diff line number Diff line change
Expand Up @@ -93,3 +93,5 @@ export const defaultBookmarkIcon: Icon = {
parameter: `<svg xmlns="http://www.w3.org/2000/svg" version="1.0" viewBox="0 0 48 48"><path fill="#F44336" d="M37,43l-13-6l-13,6V9c0-2.2,1.8-4,4-4h18c2.2,0,4,1.8,4,4V43z"></path></svg>`,
type: IconType.SVG,
};

export const defaultControlPanelIcon = "";
3 changes: 3 additions & 0 deletions src/common/translation/english-translation-set.ts
Original file line number Diff line number Diff line change
Expand Up @@ -339,4 +339,7 @@ export const englishTranslationSet: TranslationSet = {
chooseFile: "Choose file",
chooseFolder: "Choose folder",
restartRequired: "Restart required",

controlPanel: "Control Panel",
controlPanelSettingsDescription: "This plugin enables you to quickly find control panel items.",
};
3 changes: 3 additions & 0 deletions src/common/translation/german-translation-set.ts
Original file line number Diff line number Diff line change
Expand Up @@ -339,4 +339,7 @@ export const germanTranslationSet: TranslationSet = {
chooseFile: "Datei auswählen",
chooseFolder: "Ordner auswählen",
restartRequired: "Neustart erforderlich",

controlPanel: "Systemsteuerung",
controlPanelSettingsDescription: "Dieses Plugin erlaubt es dir, Elemente der Systemsteuerung schnell zu finden.",
};
3 changes: 3 additions & 0 deletions src/common/translation/russian-translation-set.ts
Original file line number Diff line number Diff line change
Expand Up @@ -339,4 +339,7 @@ export const russianTranslationSet: TranslationSet = {
chooseFile: "Выбрать файл",
chooseFolder: "Выбрать папку",
restartRequired: "Необходима перезагрузка",

controlPanel: "панель управления",
controlPanelSettingsDescription: "Этот плагин позволяет быстро находить элементы панели управления.",
};
3 changes: 3 additions & 0 deletions src/common/translation/translation-set.ts
Original file line number Diff line number Diff line change
Expand Up @@ -336,4 +336,7 @@ export interface TranslationSet {
chooseFile: string;
chooseFolder: string;
restartRequired: string;

controlPanel: string;
controlPanelSettingsDescription: string;
}
1 change: 1 addition & 0 deletions src/main/plugin-type.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,4 +22,5 @@ export enum PluginType {
Dictionary = "dictionary",
BrowserBookmarks = "browser-bookmarks",
Test = "test-plugin",
ControlPanel = "control-panel-plugin",
}
6 changes: 6 additions & 0 deletions src/main/plugins/control-panel-plugin/control-panel-item.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
export class ControlPanelItem {
public Name: string;
public CanonicalName: string;
public Description: string;
public IconBase64: string;
}
138 changes: 138 additions & 0 deletions src/main/plugins/control-panel-plugin/control-panel-items-retriever.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
import { executeCommandWithOutput } from "../../executors/command-executor";
import { ControlPanelItem } from "./control-panel-item";
import * as Powershell from "node-powershell";

export class ControlPanelItemsRetriever {
public static RetrieveControlPanelItems(alreadyKnownItems: ControlPanelItem[]): Promise<ControlPanelItem[]> {
return new Promise((resolve, reject) => {
this.executeCommandWithUtf8Output('powershell -Command "Get-ControlPanelItem | ConvertTo-Json"')
.then((controlPanelItemsJson) => {
const controlPanelItems: ControlPanelItem[] = JSON.parse(controlPanelItemsJson);

const alreadyKnownItemsStillPresent = alreadyKnownItems.filter((item) => controlPanelItems.some((i) => i.Name === item.Name));
const newControlPanelItems = controlPanelItems.filter((item) => !alreadyKnownItems.some((i) => i.Name === item.Name));

const iconSize = 128;
const getIconsCommand = `
$iconExtractorCode = '${this.iconExtractorCode}';
$iconExtractorType = Add-Type -TypeDefinition $iconExtractorCode -PassThru -ReferencedAssemblies 'System.Drawing.dll';
$ErrorActionPreference = "SilentlyContinue";
Get-Item -Path "Registry::HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Explorer\\ControlPanel\\NameSpace\\*" |
Select-Object -ExpandProperty Name |
ForEach-Object {
if ($_.substring($_.lastindexof("\\") + 1) -match "{.+}") { return $Matches[0] }
else { return $null }
} |
Where-Object { $_ -ne $null } |
ForEach-Object {
$defaultIconValue = Get-ItemPropertyValue -Path "Registry::HKEY_CLASSES_ROOT\\CLSID\\$_\\DefaultIcon" -Name "(default)";
$defaultIconValueSplit = $defaultIconValue.Split(',');
$iconPath = $defaultIconValueSplit[0];
$iconIndex = if ($defaultIconValueSplit.Length -gt 1) { $defaultIconValueSplit[1] } else { $null };
$iconBase64 = $iconExtractorType[0]::GetIconAsBase64($iconPath, ${iconSize}, $iconIndex);
@{
applicationName = Get-ItemPropertyValue -Path "Registry::HKEY_CLASSES_ROOT\\CLSID\\$_" -Name "System.ApplicationName";
iconBase64 = $iconBase64;
};
} |
ConvertTo-Json
`;
const shell = new Powershell({});
shell.addCommand(getIconsCommand)
.then(() => shell.invoke())
.then(
(controlPanelItemIconsJson) => {
const controlPanelItemIcons: Array<{ applicationName: string, iconBase64: string }> = JSON.parse(controlPanelItemIconsJson);
for (const icon of controlPanelItemIcons) {
const item = newControlPanelItems.find((i) => i.CanonicalName === icon.applicationName);
if (item != null && icon.iconBase64 != null) {
item.IconBase64 = icon.iconBase64;
}
}
resolve(alreadyKnownItemsStillPresent.concat(newControlPanelItems));
})
.finally(() => shell.dispose())
.catch((reason) => reject(reason));
}).catch((reason) => reject(reason));
});
}

private static readonly iconExtractorCode = `
using System;
using System.Drawing;
using System.Drawing.Imaging;
using System.IO;
using System.Runtime.InteropServices;
namespace IconExtractor
{
public class IconExtractor
{
private const uint GroupIcon = 14;
private const uint LoadLibraryAsDatafile = 0x00000002;
private delegate bool EnumResNameDelegate(IntPtr hModule, IntPtr lpszType, IntPtr lpszName, IntPtr lParam);
[DllImport("kernel32.dll", SetLastError = true)]
static extern IntPtr LoadLibraryEx(string lpFileName, IntPtr hFile, uint dwFlags);
[DllImport("kernel32.dll", EntryPoint = "EnumResourceNamesW", CharSet = CharSet.Unicode, SetLastError = true)]
static extern bool EnumResourceNamesWithId(IntPtr hModule, uint lpszType, EnumResNameDelegate lpEnumFunc, IntPtr lParam);
[DllImport("kernel32.dll", SetLastError = true)]
static extern bool FreeLibrary(IntPtr hModule);
[DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
static extern IntPtr LoadImage(IntPtr hinst, IntPtr lpszName, uint uType, int cxDesired, int cyDesired, uint fuLoad);
[DllImport("user32.dll", CharSet = CharSet.Auto)]
static extern bool DestroyIcon(IntPtr handle);
public static string GetIconAsBase64(string filePath, int iconSize, string iconIndex = null)
{
var iconPointer = GetIconPointer(filePath, iconSize, iconIndex);
if (iconPointer != IntPtr.Zero)
{
var icon = Icon.FromHandle(iconPointer);
var bitmap = icon.ToBitmap();
DestroyIcon(iconPointer);
using (var stream = new MemoryStream())
{
bitmap.Save(stream, ImageFormat.Png);
var base64String = Convert.ToBase64String(stream.ToArray());
return base64String;
}
}
return null;
}
private static IntPtr GetIconPointer(string filePath, int iconSize, string iconIndex = null)
{
var dataFilePointer = LoadLibraryEx(filePath, IntPtr.Zero, LoadLibraryAsDatafile);
if (dataFilePointer == IntPtr.Zero)
return IntPtr.Zero;
var iconIndexPointer = iconIndex != null
? new IntPtr(Math.Abs(Convert.ToInt32(iconIndex)))
: IntPtr.Zero;
var iconPointer = LoadImage(dataFilePointer, iconIndexPointer, 1, iconSize, iconSize, 0);
if (iconPointer == IntPtr.Zero)
{
EnumResourceNamesWithId(dataFilePointer, GroupIcon, (hModule, lpszType, lpszName, lParam) =>
{
iconPointer = lpszName;
return false;
}, IntPtr.Zero);
iconPointer = LoadImage(dataFilePointer, iconPointer, 1, iconSize, iconSize, 0);
}
FreeLibrary(dataFilePointer);
return iconPointer;
}
}
}
`;

private static executeCommandWithUtf8Output(command: string): Promise<string> {
return executeCommandWithOutput(`cmd /c chcp 65001>nul && ${command}`);
}
}
Loading

0 comments on commit 5426027

Please sign in to comment.