Update (30 Dec 2024): Updated form control box and sizing for issues with new tB versions and updated WinDevLib version.
Update (03 Mar 2024): .twinproj has been updated to use a more recent version of WinDevLib (formerly tbShellLib) due to errors in the package tB did not raise at the time this project was released.
This app exposes undocumented radio management functionality in Windows. I've always been a big fan of making your own settings apps, so don't like being told the official one is the only way to do something. The IMediaRadioManager
, IRadioInstance
, and IRadioCollection
interfaces are documented, but the coclasses representing the system objects actually implementing them are not; they're just documented for hardware providers to provide. There's three of these I could find: the WWAN manager, which includes cellular radios, the WLAN manager, which includes WiFi radios, and the Bluetooth manager (self explanatory). These provide control over individual radios. I tested this app on my Surface tablet, where I had one of each. The app lets you control them all individually; you choose one on the list then can query, enable, or disable:
In the above picture, I first selected the radio then click Query to confirm it was on, then clicked disable. You can see in the system tray the WiFi has been disabled.
In addition, there's the IRadioManager
interface and RadioManagementAPI
coclass-- the master switch for all system radios, popularly known as Airplane Mode. Both of these are undocumented, and have only been partially reverse engineered. I found an example of using this interface in the wintouchg repo here on GitHub. You can query the state and enable and disable. Below, you can see that after clicking Enable, the Airplane Mode indicator replaced the WiFi connection in the system tray, indicating we've successfully entered the mode:
-The IMediaRadioManager
individual interfaces require Windows 8 or later. I'm unsure of when support for the IRadioManager
airplane mode interface started, but can say it's on Windows 10 and later.
This project was written exclusively in twinBASIC (GitHub), the currently in-development succesor to VB6 with full backwards compatibility. Written to be compiled for both 32bit and 64bit targets.
-All of the interfaces and APIs are part of my WinDevLib (Windows Development Library, formerly tbShellLib) package for twinBASIC; this is entirely downloaded into the project file (for now), which is why the size is so large. But the compiler takes only what it needs; the exe doesn't contain the whole thing. You can add it to your own projects from the Package Manager or from the project repository.
-Any recent version of tB should compile it.
The individual radios are a pretty straightforward object which provides a collection, which provides an individual interface:
[ InterfaceId ("70AA1C9E-F2B4-4C61-86D3-6B9FB75FD1A2") ]
[ OleAutomation (False) ]
Interface IRadioInstance Extends IUnknown
Sub GetRadioManagerSignature(pguidSignature As UUID)
Sub GetInstanceSignature(pbstrId As String)
Sub GetFriendlyName(ByVal lcid As Long, pbstrName As String)
Sub GetRadioState(pRadioState As DEVICE_RADIO_STATE)
Sub SetRadioState(ByVal radioState As DEVICE_RADIO_STATE, ByVal uTimeoutSec As Long)
[ PreserveSig ] Function IsMultiComm() As BOOL
[ PreserveSig ] Function IsAssociatingDevice() As BOOL
End Interface
We just enumerate those:
Set pWW = New WwanRadioManager
If pWW IsNot Nothing Then
pWW.GetRadioInstances(pColWW)
If (pColWW IsNot Nothing) Then
pColWW.GetCount(nWW)
If nWW Then
For i = 0 To nWW - 1
pColWW.GetAt(i, pInstWW)
Debug.Print "Add WW radio 0"
pInstWW.GetFriendlyName(GetUserDefaultLCID(), sBuff)
pInstWW.GetInstanceSignature(sBuffId)
pInstWW.GetRadioManagerSignature(priid)
List1.AddItem(sBuff & " (InstId=" & sBuffId & ", RmSig=" & GUIDToString(priid))
Set pInstWW = Nothing
Next
End If
Text1.Text = CStr(nWW) & " WWAN " & IIf(nWW = 1, "radio", "radios") & " found."
and repeat for the other two. Once we have the IRadioInstance
, GetRadioState
and SetRadioState
are very simple methods to use.
IRadioManager
for airplane mode is even simpler:
[ InterfaceId ("db3afbfb-08e6-46c6-aa70-bf9a34c30ab7") ]
Interface IRadioManager Extends IUnknown
Sub IsRMSupported(pdwState As Long) 'Always 1
Sub GetUIRadioInstances(ppInstances As IUnknown) 'IUIRadioInstanceCollection
Sub GetSystemRadioState(pbEnabled As BOOL, param2 As Long, param3 As RADIO_CHANGE_REASON)
Sub SetSystemRadioState(ByVal bEnabled As BOOL)
Sub Refresh()
Sub OnHardwareSliderChange(ByVal param1 As Long, ByVal param2 As Long)
End Interface
The arguments besides enabled aren't well understood; but they don't seem critical to functionality. All we have to do is Set pRM = New RadioManagementAPI
then we have the IRadioManager
interface and the dead simple GetSystemRadioState
and SetSystemRadioState
.
Overall, this is a very simple program. The hard part was done by the people who reverse engineered this stuff to make it easy to consume by people like me. Simple, but very useful if like me you want alternative to the crappy Settings app Microsoft refuses to improve. Enjoy!