KeyListener Plugin v1.0.0
Inspired by KeyCodeInfo by nstechbytes
Inspired by KeyCodeInfo by nstechbytes
KeyListener tracks your keyboard in real time and instantly outputs the keys you press. Carefully crafted to let users enter Key Combinations to be used with the HotKey plugin with ease.
HotKeySetup.gif
- Hotkey Support: Crafted to work with HotKey Plugin out of the box.
- Multiple Output Formats: Output key names, hex codes, numbers, virtual keys or enum names.
- Automatic Smart Update: Only update when needed.
- Control: Make it yours, all options are... optional...
KeyFormatPreview.gif
The package includes the plugin plus 3 example skins, those seen on the gifs above.
HotKey plugin is required to use the HotKey Setup Example skin.
Install it if you haven't already.
KeyListener_1.0.rmskin
- To nstechbytes who created the base this plugin has been built upon.
- To Yincognito who helped me with ideas and testing.
- And to anyone who tests it and shares their experience
v1.0.0.0
- Initial release!
Documentation
1. Add the Plugin to Your SkinThe plugin is set up to enter hotkeys for the HotKey plugin by default, so the measure above will be everything you need if that's what you aim for.
Set up the measure with the desired parameters if you want to do anything else.
Any missing option will use its default value.
2. Configure Options
Here it includes all options and their default values (read "
Options" below to know their possible values):
You will need to set variables for your hotkeys. Key names will be displayed in the user's keyboard layout language, that's why we can't use Name on the HotKey plugin, use any other format instead. I like using Hex format for the HotKey plugin measures, and Name format to display the keys to the user. HotKey plugin only accepts keys separated by a space, luckily the HexSeparator option is set to a blank space " " by default. To simplify the task, use the built-in variables $format$ on the OnStopAction to set the hotkeys. This way we don't need multiple measures, although the plugin supports child measures to output the same keys in different formats at the same time.
I like adding UpdateDivider=-1 because the measure doesn't really need to be updated by the skin at all, since it will update itself automatically when pressing the keys.
4. Add Conditions
With a simple conditional we can make our meter display the keys in real time when the plugin is active:Basically when the plugin is not active (-1), the meter's text will default to the #KeyName# variable, when it is active but is value is 0 (that is, no key has been entered), the meter's text will change to "Listening to Keyboard...". Finally, on any other value the meter's text will default to %1, which is the string value of the measure which will change as we press keys.
5. Add the meter
We will default our meter's text to the #KeyName# variable and we will also use MeasureName=KeyListener6. Add Commands
Start the plugin with this bang:5. Add final touches and you're done
After adding a simple shape meter and of course, the hotkey plugin measure, this is how it looks: Now you can click the box to start the plugin, then enter any key or key combination. Finally test it by pressing it. You should see "Key up" and "Key Down" on the log.
For more information about HotKey setup, check the HotKeySetup.ini file in the package.
For more information about Key Formats and Child Measures, check the KeyFormat.ini file in the package.
For more information about ClearOnRelease and ClearDelay, check the KeyPreview.ini file in the package.
Code:
[KeyListener]Measure=PluginPlugin=KeyListenerLockInput=1OnStopAction=[]OnDismissAction=[]OnUpdateAction=[]
Set up the measure with the desired parameters if you want to do anything else.
Any missing option will use its default value.
2. Configure Options
Here it includes all options and their default values (read "
Code:
[KeyListener]Measure=PluginPlugin=KeyListener;ParentName=AutoUpdate=1KeyFormat=1LockInput=0AllowCombo=1RequireModifier=1NameSeparator=" + "HexSeparator=" "NumberSeparator=" "VKSeparator=" "EnumSeparator=" "StopOnRelease=1ClearOnRelease=1ClearDelay=0OnStopAction=[]OnDismissAction=[]
- It supports DynamicVariables, Substitutions, ifConditions and ifMatch as any other measure.
- All Arguments can be changed dynamically either by setting Variables or SetOption.
You will need to set variables for your hotkeys. Key names will be displayed in the user's keyboard layout language, that's why we can't use Name on the HotKey plugin, use any other format instead. I like using Hex format for the HotKey plugin measures, and Name format to display the keys to the user. HotKey plugin only accepts keys separated by a space, luckily the HexSeparator option is set to a blank space " " by default. To simplify the task, use the built-in variables $format$ on the OnStopAction to set the hotkeys. This way we don't need multiple measures, although the plugin supports child measures to output the same keys in different formats at the same time.
Code:
[Variables]KeyName=KeyHex=[KeyListener]......OnStopAction=[!SetVariable KeyName "$name$"][!SetVariable KeyHex "$hex$"][!UpdateMeasure "HotKey"]OnUpdateAction=[!UpdateMeter KeyName][!Redraw]UpdateDivider=-1
4. Add Conditions
With a simple conditional we can make our meter display the keys in real time when the plugin is active:
Code:
[KeyListener]......IfCondition=KeyListener = -1IfTrueAction=[!SetOption KeyName Text "#*KeyName*#"]ifCondition2=KeyListener = 0IfTrueAction2=[!SetOption KeyName Text "Listening to Keyboard..."] IfFalseAction2=[!SetOption KeyName Text "%1"]
5. Add the meter
We will default our meter's text to the #KeyName# variable and we will also use MeasureName=KeyListener
Code:
[KeyName]Meter=String......MeasureName=KeyListenerText=#KeyName#
Start the plugin with this bang:
Code:
LeftMouseUpAction=[!CommandMeasure "KeyListener" "Start"];The plugin will stop automatically after we release the keys, so we only need to start it.
After adding a simple shape meter and of course, the hotkey plugin measure, this is how it looks:
Code:
[Rainmeter]Update=1000AccurateText=1[Variables]KeyName=KeyHex=;KeyListener plugin Measure:[KeyListener]Measure=PluginPlugin=KeyListenerLockInput=1OnDismissAction=[]OnStopAction=[!SetVariable KeyName "$name$"][!SetVariable KeyHex "$hex$"][!UpdateMeasure "HotKey"]OnUpdateAction=[!UpdateMeter KeyName][!Redraw];ConditionalsIfCondition=KeyListener = -1IfTrueAction=[!SetOption KeyName Text "#*KeyName*#"]IfFalseAction=[!SetOption KeyName Text "%1"]UpdateDivider=-1;HotKey plugin Measures:[HotKey]Measure=PluginPlugin=HotKeyHotKey=#KeyHex#KeyDownAction=[!log "HotKey Down"]KeyUpAction=[!log "HotKey Up"]DynamicVariables=1;Background[KeyBox]Group=Key1Meter=ShapeShape=Rectangle 0,0,250,30,5 | Fill Color 25,25,25 | StrokeWidth 0LeftMouseUpAction=[!CommandMeasure KeyListener "Start"];Hotkey display[KeyName]Meter=StringMeasureName=KeyListenerStringAlign=LeftCenterAntiAlias=1FontFace=ConsolasFontColor=255,255,255FontSize=11Text=#KeyName#X=5ry=15
example.gif
For more information about Key Formats and Child Measures, check the KeyFormat.ini file in the package.
For more information about ClearOnRelease and ClearDelay, check the KeyPreview.ini file in the package.
The following applies to both parent and child measures.
- Number Value
Returns a different value according to the situation.- -1 Plugin is inactive.
- 0 No pressed keys.
- 1-255 Key Number Value.
- 2255255-5255255255255255 Number of keys on a combination, followed by their key number value on groups of 3 digits.
While the first 3 values are pretty straight forward, the last one can be somewhat.. impressive?
Well, it is actually pretty simple, something to keep in mind is that a combo is limited to a total of 5 keys.
The number value for 2 keys, for example CTRL + C, would be 2162067. This means 2 keys: 162 (CTRL) and 67 (C) or 2,162,067.
2-digit key numbers will be returned with a leading 0 so each key has a total of 3 digits.
So, CTRL + ALT + SHIFT + T would be 4162164160084, or 4 Keys: 162 (CTRL), 164 (ALT), 160 (SHIFT), and 84 (T), or 4,162,164,160,084.
While I personally don't know why or how would anyone need this information, I thought it could be useful to know the exact number of keys that are inside a combo (that's the first value) at some point, and all the other key numbers.. well, who knows.
Anyways, to decode this number and extract the keys, Yincognito has been really kind to prepare an example skin on how it would go for us, it's actually quite simple:here are two ways to approach it, one uncommented and the other commented:The uncommented formulas use Log() to do stuff based on the Code's number of digits, and that allows knowing exactly what part a value corresponds to. Normally, (Trunc(#Code#/10**Trunc(Log(#Code#)-N))%1000) would have been enough, if not for Log(#Code#)-N messing things when below 0, so I had to use (NMax+Trunc(Log(#Code#)-Code:
[Variables]Code=2162067;Code=4162164160084[Rainmeter]Update=1000AccurateText=1DynamicWindowSize=1---Measures---[KeyCount]Measure=CalcFormula=(Trunc(#Code#/10**((15+Trunc(Log(#Code#)-0))%15))%1000);Formula=Trunc((#Code#%1000**Max(0,4))/1000**Max(0,3))DynamicVariables=1[KeyNumb1]Measure=CalcFormula=(Trunc(#Code#/10**((15+Trunc(Log(#Code#)-3))%15))%1000);Formula=Trunc((#Code#%1000**Max(0,3))/1000**Max(0,2))DynamicVariables=1[KeyNumb2]Measure=CalcFormula=(Trunc(#Code#/10**((15+Trunc(Log(#Code#)-6))%15))%1000);Formula=Trunc((#Code#%1000**Max(0,2))/1000**Max(0,1))DynamicVariables=1[KeyNumb3]Measure=CalcFormula=(Trunc(#Code#/10**((15+Trunc(Log(#Code#)-9))%15))%1000);Formula=Trunc((#Code#%1000**Max(0,1))/1000**Max(0,0))DynamicVariables=1[KeyNumb4]Measure=CalcFormula=(Trunc(#Code#/10**((15+Trunc(Log(#Code#)-12))%15))%1000);Formula=Trunc((#Code#%1000**Max(0,0))/1000**Max(0,-1))DynamicVariables=1[KeyNumb5]Measure=CalcFormula=(Trunc(#Code#/10**((15+Trunc(Log(#Code#)-15))%15))%1000);Formula=Trunc((#Code#%1000**Max(0,-1))/1000**Max(0,-2))DynamicVariables=1---Meters---[Result]Meter=StringSolidColor=0,0,0,128FontColor=255,255,255,255FontFace=ConsolasFontSize=16Padding=5,5,5,5AntiAlias=1MeasureName =KeyCountMeasureName2=KeyNumb1MeasureName3=KeyNumb2MeasureName4=KeyNumb3MeasureName5=KeyNumb4MeasureName6=KeyNumb5Text=Key Count = %1#CRLF#Key No. 1 = %2#CRLF#Key No. 2 = %3#CRLF#Key No. 3 = %4#CRLF#Key No. 4 = %5#CRLF#Key No. 5 = %6UpdateDivider=-1
N))%NMax to keep it afloat and still provide the suited value to have the "empty" [KeyNumbX] return 0. I could have used 3*n instead of N, with n ranging from 0 to 5 and matching the measure indexes, but I didn't to keep the formula shorter. The idea is simple, get the integer part of the division of Code by powers of 10 that are subsequently offset by 3 based on the Code's number of digits, followed by getting the remainder of that divided by 1000.
The commented formulas are a simpler and reverse version of the uncommented ones, but without any reliance on the Code's number of digits, so values will "migrate" between measures based on the latter. Again, it could have been even the simpler Trunc((#Code#%1000**(N))/1000**(N-1)), but I had to use Max(0,X) for the same reason, to keep things positive. It's the reverse of the previous idea, as we now first get the remainder of Code divided by powers of 1000, followed by getting the integer part of that divided by the next lower power of 1000. - String Value
Returns the pressed key or key combinations on different formats depending on the KeyFormat and NameSeparator,HexSeparator,NumberSeparator,VKSeparator and EnumSeparator options.- Alt + H when KeyFormat = 1 and NameSeparator=" + ".
- 0xA4 + 0x48 when KeyFormat = 2 and HexSeparator=" + ".
- 164 + 72 when KeyFormat = 3 and NumberSeparator=" + ".
- VK_LMENU + H when KeyFormat = 4 and VKSeparator=" + ".
- LMenu + H when KeyFormat = 5 and EnumSeparator=" + ".
Parent Measure Options
Modifiers are: ALT, CTRL, SHIFT, etc.
For example, NumberSeparator=" * "
Will output: 162 * 67
These options also format the built-in variables (see built-in variables on the Actions section).
In Combo mode, also clears the parent measure and its children values after the ClearDelay timer stops, however only after the key that started the combination is released.
It also clears the measure values when the measure is commanded to Stop (or Dismiss), however, this time the ClearDelay timer is stopped immediately.
Sets a timer in milliseconds to clear the values of the parent measure and its children. The timer starts after the key is released.
If ClearOnRelease = 1, then it also clears the measure values before stopping, however the ClearDelay timer will be ignored and stopped immediately.Child Measure Options
Child Measures support the following options, they also support all the actions described on the next section.
This option works exactly the same as in the Parent Measures.
This option works exactly the same as in the Parent Measures.
Actions Other Options
- AutoUpdate
Default: 1
Values:- 0 Disabled
- 1 Enabled
*Meaningful events are those that produce an actual change on the value of the measure, this means it won't update if you enter duplicated keys, you reach the maximum amount of keys on a key combination, or you execute a clear command when there are no keys to be cleared. Not even if you use a clear command on a OnStopAction or OnDismissAction, in that case it will only update once, after the actions have been executed and the plugin has been stopped.
You can take advantage of this option by using OnUpdateAction to also update any meter that needs to use the measure's values.
You can take advantage of this option by using OnUpdateAction to also update any meter that needs to use the measure's values.
- LockInput: Default: 0
Values:- 0 Disabled
- 1 Enabled
This is a powerful option, however needs to be used wisely.
When it is enabled, only the current active measure will be able to "listen" to key events. This means that you won't be able to type on any other application, including the system itself until the plugin stops.
This option is helpful when using the plugin to set HotKeys, preventing the system from executing them while you're setting them up. For example, if you were to enter any hotkey using the windows key, as soon as you release the key the Start Menu would pop up. Enabling this option will let you use the key without windows knowing.
So remember to always stop the plugin when using this option. You can use the StopOnRelease option to do it automatically after releasing the key, or manually stop it using the measure commands.
A notice will be logged when the plugin is enabled, once per session.
Ctrl + Alt + Delete will bypass this option at any time.
When it is enabled, only the current active measure will be able to "listen" to key events. This means that you won't be able to type on any other application, including the system itself until the plugin stops.
This option is helpful when using the plugin to set HotKeys, preventing the system from executing them while you're setting them up. For example, if you were to enter any hotkey using the windows key, as soon as you release the key the Start Menu would pop up. Enabling this option will let you use the key without windows knowing.
So remember to always stop the plugin when using this option. You can use the StopOnRelease option to do it automatically after releasing the key, or manually stop it using the measure commands.
A notice will be logged when the plugin is enabled, once per session.
Ctrl + Alt + Delete will bypass this option at any time.
- KeyFormat
Default: 1
Values:- 1 Name
- 2 Hex
- 3 Number
- 4 VK
- 5 Enum
- Any other value will be treated as 1.
Some keys don't have a name registered by windows, when that happens, they will return their Enum name instead.
Names are dependent of keyboard layout language.
Here are examples of these formats outputs:
Name: Ctrl
Hex: 0xA2
Number: 162
VK: VK_LCONTROL
Enum: LControlKey
Names are dependent of keyboard layout language.
Here are examples of these formats outputs:
Name: Ctrl
Hex: 0xA2
Number: 162
VK: VK_LCONTROL
Enum: LControlKey
- AllowCombo
Default: 1
Values:- 0 Disabled
- 1 Enabled
To start a combination, simply hold a key, then press other keys to add them to the combination. Only the first key is needed to be held. The combination will end the moment the key that started the combo is released.
Combinations can be a maximum of 5 keys, for example:
CTRL + ALT + SHIFT + A + B
Combinations can only start with a Modifier unless RequireModifier is disabled, for example:
A + B + C + D + E (When RequireModifier = 0)
Modifiers can't be added to a combination when a non-modifier key precedes it, for example:
CTRL + B![✅]()
B + CTRL![❌]()
After reaching the maximum of 5 keys, any other pressed key will be ignored until a new combo is started (that is, when the key that started the combo is released).
Combinations can be a maximum of 5 keys, for example:
CTRL + ALT + SHIFT + A + B
Combinations can only start with a Modifier unless RequireModifier is disabled, for example:
A + B + C + D + E (When RequireModifier = 0)
Modifiers can't be added to a combination when a non-modifier key precedes it, for example:
CTRL + B
B + CTRL
After reaching the maximum of 5 keys, any other pressed key will be ignored until a new combo is started (that is, when the key that started the combo is released).
- RequireModifier
Default: 1
Values:- 0 Disabled
- 1 Enabled
Modifiers are: ALT, CTRL, SHIFT, etc.
- NameSeparator Default: " + "
- HexSeparator Default: " "
- NumberSeparator Default: " "
- VKSeparator Default: " "
- EnumSeparator Default: " "
Values:- Any string.
For example, NumberSeparator=" * "
Will output: 162 * 67
These options also format the built-in variables (see built-in variables on the Actions section).
- ClearOnRelease
Default: 1
Values:- 0 Disabled
- 1 Enabled
In Combo mode, also clears the parent measure and its children values after the ClearDelay timer stops, however only after the key that started the combination is released.
It also clears the measure values when the measure is commanded to Stop (or Dismiss), however, this time the ClearDelay timer is stopped immediately.
- ClearDelay: Default: 0
Value:- Any time in milliseconds, lower limit is 0.
Sets a timer in milliseconds to clear the values of the parent measure and its children. The timer starts after the key is released.
- StopOnRelease
Default: 1
Values:- 0 Disabled
- 1 Enabled
If ClearOnRelease = 1, then it also clears the measure values before stopping, however the ClearDelay timer will be ignored and stopped immediately.
Child Measures support the following options, they also support all the actions described on the next section.
- ParentName
- Represents the name of the Parent measure. Do not include brackets.
- KeyFormat
Default: 1
Values:- 1 Name
- 2 Hex
- 3 Number
- 4 VK
- 5 Enum
- Any other value will be treated as 1.
This option works exactly the same as in the Parent Measures.
- NameSeparator Default: " + "
- HexSeparator Default: " "
- NumberSeparator Default: " "
- VKSeparator Default: " "
- EnumSeparator Default: " "
Values:- Any string.
This option works exactly the same as in the Parent Measures.
The following actions are supported by both the Parent and Child Measures.
The child's actions are always executed first, and the parent's at the end.
Built-In Variables
The following built-in variables are only to be used on the actions described in this section, they return the measure value on any format. When the return is a key combination they use the separator given on the Separator option of the same name, for example, $hex$ will use the HexSeparator.
These variables allow you to export keys on different formats from a single measure.
They are not case sensitive, so $hex$, $Hex$ and $hEx$ will work.
Example: OnStopAction= [!Log "The hex value of the $name$ key is: $hex$"]
Logs: The hex value of the W key is: 0x57
Example: OnDismissAction= [!Log "The $name$ key has been dismissed."]
Logs: The W key has been dismiss.
The child's actions are always executed first, and the parent's at the end.
Built-In Variables
The following built-in variables are only to be used on the actions described in this section, they return the measure value on any format. When the return is a key combination they use the separator given on the Separator option of the same name, for example, $hex$ will use the HexSeparator.
These variables allow you to export keys on different formats from a single measure.
They are not case sensitive, so $hex$, $Hex$ and $hEx$ will work.
- $name$
Returns the keys using the Name format, e.g. Caps Lock. - $hex$
Returns the keys using the Hex format, e.g. 0x14. - $number$
Returns the keys using the Number format, e.g. 20. - $vk$
Returns the keys using the VK format, e.g. VK_CAPITAL. - $enum$
Returns the keys using the Enum format, e.g. CapsLock.
- OnStopAction
Example: OnStopAction= [!Log "The hex value of the $name$ key is: $hex$"]
Logs: The hex value of the W key is: 0x57
- OnDismissAction
Example: OnDismissAction= [!Log "The $name$ key has been dismissed."]
Logs: The W key has been dismiss.
Since I find warnings really annoying, I only included warnings for when trying to stop a measure that is not running, or trying to start a measure that is already running. Also added a notice when LockInput is enabled due to its nature, however this notice will only be logged once, after the plugin has been started for the first time in a session. However, they can be disabled using the following options.
These are Parent-only options.
Requires Debug Mode to be enabled to see the logs (Right Click on Rainmeter's tray icon --> Logging --> Debug Mode)
Be aware that logging can have an impact on CPU usage while typing.
These are Parent-only options.
- Debug
Default: 0
Values:- 0 Disabled
- 1 Enabled
Requires Debug Mode to be enabled to see the logs (Right Click on Rainmeter's tray icon --> Logging --> Debug Mode)
Be aware that logging can have an impact on CPU usage while typing.
- Warnings
Default: 1
Values:- 0 Disabled
- 1 Enabled
The following commands are used with the [!CommandMeasure MeasureName "Command"] bang.
Only Parent Measures can be commanded, however, children actions will be executed as well.
- Start
- Starts the plugin, then updates the measure.
- Stop
- Stops the plugin and executes the OnStopAction, then updates the measure.
- Toggle
- Toggles the plugin and executes the OnStopAction (on Stop), then updates the measure.
- Dismiss
- Stops the plugin and executes the OnDismissAction, then updates the measure.
- Clear
- Clears the plugin's measure values if they're not cleared yet, then updates the measure (if necessary).
- Note: This command works even if the measure is not active
Statistics: Posted by RicardoTM — Today, 2:33 am — Replies 3 — Views 92