Quantcast
Channel: Rainmeter Forums
Viewing all articles
Browse latest Browse all 723

KeyListener Plugin v1.0.0

$
0
0
KeyListener Plugin v1.0.0
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
✨ Features
  • 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


📥 Download & Installation

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

👏 Special Thanks
  • 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 :)
📝 Changelog

v1.0.0.0
  • Initial release!


Documentation


📌 How to Use
1. Add the Plugin to Your Skin

Code:

[KeyListener]Measure=PluginPlugin=KeyListenerLockInput=1OnStopAction=[]OnDismissAction=[]OnUpdateAction=[]
The 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):

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.
3. Configure Actions
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
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:

Code:

[KeyListener]......IfCondition=KeyListener = -1IfTrueAction=[!SetOption KeyName Text "#*KeyName*#"]ifCondition2=KeyListener = 0IfTrueAction2=[!SetOption KeyName Text "Listening to Keyboard..."] IfFalseAction2=[!SetOption KeyName Text "%1"] 
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=KeyListener

Code:

[KeyName]Meter=String......MeasureName=KeyListenerText=#KeyName#
6. Add Commands
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.
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:

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
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.
example.gif
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.
🏷️ Measure Values
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? :Whistle
    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 :rofl:.

    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:

    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
    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#)-
    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=" + ".
⚙️ Options
Parent Measure Options
  • AutoUpdate
    Default: 1
    Values:
    • 0 Disabled
    • 1 Enabled
The plugin measure will update itself and then its children automatically on meaningful* key events, or after any command.
*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.


  • LockInput: Default: 0
    Values:
    • 0 Disabled
    • 1 Enabled
Prevents other applications from receiving the keyboard input.
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.


  • KeyFormat
    Default: 1
    Values:
    • 1 Name
    • 2 Hex
    • 3 Number
    • 4 VK
    • 5 Enum
    • Any other value will be treated as 1.
Determines on which format the string measure value will be returned.
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


  • AllowCombo
    Default: 1
    Values:
    • 0 Disabled
    • 1 Enabled
Allows to enter key combinations.
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).


  • RequireModifier
    Default: 1
    Values:
    • 0 Disabled
    • 1 Enabled
Forces combinations to start with a Modifier key.

Modifiers are: ALT, CTRL, SHIFT, etc.



  • NameSeparator Default: " + "
  • HexSeparator Default: " "
  • NumberSeparator Default: " "
  • VKSeparator Default: " "
  • EnumSeparator Default: " "
    Values:
    • Any string.
Lets you choose any separator string for key combinations, each format can have a different separator.

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
Clears the parent measure and its children values when the key is released, after the ClearDelay timer stops.

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.
*Requires ClearOnRelease = 1

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
Stops the plugin immediately after releasing the key, and any OnStopAction on the parent and its children has been executed.
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.



  • 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.
Determines on which format the string measure value will be returned.
This option works exactly the same as in the Parent Measures.



  • NameSeparator Default: " + "
  • HexSeparator Default: " "
  • NumberSeparator Default: " "
  • VKSeparator Default: " "
  • EnumSeparator Default: " "
    Values:
    • Any string.
Lets you choose any separator string for key combinations, each format can have a different separator.

This option works exactly the same as in the Parent Measures.
Actions
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.
  • $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
Executes given bangs when the plugin stops.

Example: OnStopAction= [!Log "The hex value of the $name$ key is: $hex$"]

Logs: The hex value of the W key is: 0x57



  • OnDismissAction
Executes given bangs when the plugin is dismissed.

Example: OnDismissAction= [!Log "The $name$ key has been dismissed."]

Logs: The W key has been dismiss.

Other Options
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.


  • Debug
    Default: 0
    Values:
    • 0 Disabled
    • 1 Enabled
Enables debug logging.
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
Enables warning and notice logging.
🛠️ Commands



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.
Example: [!CommandMeasure MeasureName "Start"]



  • Stop
    • Stops the plugin and executes the OnStopAction, then updates the measure.
Example: [!CommandMeasure MeasureName "Stop"]



  • Toggle
    • Toggles the plugin and executes the OnStopAction (on Stop), then updates the measure.
Example: [!CommandMeasure MeasureName "Toggle"]



  • Dismiss
    • Stops the plugin and executes the OnDismissAction, then updates the measure.
Example: [!CommandMeasure MeasureName "Dismiss"]



  • Clear
    • Clears the plugin's measure values if they're not cleared yet, then updates the measure (if necessary).
Example: [!CommandMeasure MeasureName "Clear"]
  • Note: This command works even if the measure is not active

Statistics: Posted by RicardoTM — Today, 2:33 am — Replies 3 — Views 92



Viewing all articles
Browse latest Browse all 723

Trending Articles