2 * PROJECT: ReactOS system libraries
3 * LICENSE: GPL - See COPYING in the top level directory
4 * FILE: dll/shellext/stobject/volume.cpp
5 * PURPOSE: Volume notification icon handler
6 * PROGRAMMERS: David Quintana <gigaherz@gmail.com>
14 #define GET_X_LPARAM(lp) ((int)(short)LOWORD(lp))
15 #define GET_Y_LPARAM(lp) ((int)(short)HIWORD(lp))
17 WINE_DEFAULT_DEBUG_CHANNEL(stobject
);
25 DWORD g_muteControlID
;
27 UINT g_mmDeviceChange
;
29 static BOOL g_IsMute
= FALSE
;
30 static BOOL g_IsRunning
= FALSE
;
32 static HRESULT __stdcall
Volume_FindMixerControl(CSysTray
* pSysTray
)
39 TRACE("Volume_FindDefaultMixerID\n");
41 result
= waveOutMessage((HWAVEOUT
)WAVE_MAPPER
, DRVM_MAPPER_PREFERRED_GET
, (DWORD_PTR
)&waveOutId
, (DWORD_PTR
)¶m2
);
45 if (waveOutId
== (DWORD
)-1)
47 TRACE("WARNING: waveOut has no default device, trying with first available device...\n", waveOutId
);
53 TRACE("waveOut default device is %d\n", waveOutId
);
55 result
= mixerGetID((HMIXEROBJ
)waveOutId
, &mixerId
, MIXER_OBJECTF_WAVEOUT
);
59 TRACE("mixerId for waveOut default device is %d\n", mixerId
);
67 MIXERCONTROL mixerControl
;
68 MIXERLINECONTROLS mixerLineControls
;
73 if (mixerGetDevCapsW(g_mixerId
, &mixerCaps
, sizeof(mixerCaps
)))
76 if (mixerCaps
.cDestinations
== 0)
79 TRACE("mixerCaps.cDestinations %d\n", mixerCaps
.cDestinations
);
82 for (idx
= 0; idx
< mixerCaps
.cDestinations
; idx
++)
84 mixerLine
.cbStruct
= sizeof(mixerLine
);
85 mixerLine
.dwDestination
= idx
;
86 if (!mixerGetLineInfoW((HMIXEROBJ
)g_mixerId
, &mixerLine
, 0))
88 if (mixerLine
.dwComponentType
>= MIXERLINE_COMPONENTTYPE_DST_SPEAKERS
&&
89 mixerLine
.dwComponentType
<= MIXERLINE_COMPONENTTYPE_DST_HEADPHONES
)
91 TRACE("Destination %d was not speakers or headphones.\n");
95 if (idx
>= mixerCaps
.cDestinations
)
98 TRACE("Valid destination %d found.\n");
100 g_mixerLineID
= mixerLine
.dwLineID
;
102 mixerLineControls
.cbStruct
= sizeof(mixerLineControls
);
103 mixerLineControls
.dwLineID
= mixerLine
.dwLineID
;
104 mixerLineControls
.cControls
= 1;
105 mixerLineControls
.dwControlType
= MIXERCONTROL_CONTROLTYPE_MUTE
;
106 mixerLineControls
.pamxctrl
= &mixerControl
;
107 mixerLineControls
.cbmxctrl
= sizeof(mixerControl
);
109 if (mixerGetLineControlsW((HMIXEROBJ
)g_mixerId
, &mixerLineControls
, MIXER_GETLINECONTROLSF_ONEBYTYPE
))
112 TRACE("Found control id %d for mute: %d\n", mixerControl
.dwControlID
);
114 g_muteControlID
= mixerControl
.dwControlID
;
119 HRESULT
Volume_IsMute()
122 MIXERCONTROLDETAILS mixerControlDetails
;
124 if (g_mixerId
!= (UINT
)-1 && g_muteControlID
!= (DWORD
)-1)
126 BOOL detailsResult
= 0;
127 mixerControlDetails
.cbStruct
= sizeof(mixerControlDetails
);
128 mixerControlDetails
.hwndOwner
= 0;
129 mixerControlDetails
.dwControlID
= g_muteControlID
;
130 mixerControlDetails
.cChannels
= 1;
131 mixerControlDetails
.paDetails
= &detailsResult
;
132 mixerControlDetails
.cbDetails
= sizeof(detailsResult
);
133 if (mixerGetControlDetailsW((HMIXEROBJ
) g_mixerId
, &mixerControlDetails
, 0))
136 TRACE("Obtained mute status %d\n", detailsResult
);
138 g_IsMute
= detailsResult
!= 0;
144 HRESULT STDMETHODCALLTYPE
Volume_Init(_In_ CSysTray
* pSysTray
)
147 WCHAR strTooltip
[128];
149 TRACE("Volume_Init\n");
153 hr
= Volume_FindMixerControl(pSysTray
);
157 g_mmDeviceChange
= RegisterWindowMessageW(L
"winmm_devicechange");
160 g_hIconVolume
= LoadIcon(g_hInstance
, MAKEINTRESOURCE(IDI_VOLUME
));
161 g_hIconMute
= LoadIcon(g_hInstance
, MAKEINTRESOURCE(IDI_VOLMUTE
));
171 icon
= g_hIconVolume
;
173 LoadStringW(g_hInstance
, IDS_VOL_VOLUME
, strTooltip
, _countof(strTooltip
));
174 return pSysTray
->NotifyIcon(NIM_ADD
, ID_ICON_VOLUME
, icon
, strTooltip
);
177 HRESULT STDMETHODCALLTYPE
Volume_Update(_In_ CSysTray
* pSysTray
)
181 TRACE("Volume_Update\n");
183 PrevState
= g_IsMute
;
186 if (PrevState
!= g_IsMute
)
188 WCHAR strTooltip
[128];
193 LoadStringW(g_hInstance
, IDS_VOL_MUTED
, strTooltip
, _countof(strTooltip
));
197 icon
= g_hIconVolume
;
198 LoadStringW(g_hInstance
, IDS_VOL_VOLUME
, strTooltip
, _countof(strTooltip
));
201 return pSysTray
->NotifyIcon(NIM_MODIFY
, ID_ICON_VOLUME
, icon
, strTooltip
);
209 HRESULT STDMETHODCALLTYPE
Volume_Shutdown(_In_ CSysTray
* pSysTray
)
211 TRACE("Volume_Shutdown\n");
215 return pSysTray
->NotifyIcon(NIM_DELETE
, ID_ICON_VOLUME
, NULL
, NULL
);
218 HRESULT
Volume_OnDeviceChange(_In_ CSysTray
* pSysTray
, WPARAM wParam
, LPARAM lParam
)
220 return Volume_FindMixerControl(pSysTray
);
223 static void _RunVolume()
225 // FIXME: ensure we are loading the right one
226 ShellExecuteW(NULL
, NULL
, L
"sndvol32.exe", NULL
, NULL
, SW_SHOWNORMAL
);
229 static void _RunMMCpl()
231 ShellExecuteW(NULL
, NULL
, L
"mmsys.cpl", NULL
, NULL
, SW_NORMAL
);
234 static void _ShowContextMenu(CSysTray
* pSysTray
)
236 WCHAR strAdjust
[128];
238 LoadStringW(g_hInstance
, IDS_VOL_OPEN
, strOpen
, _countof(strOpen
));
239 LoadStringW(g_hInstance
, IDS_VOL_ADJUST
, strAdjust
, _countof(strAdjust
));
241 HMENU hPopup
= CreatePopupMenu();
242 AppendMenuW(hPopup
, MF_STRING
, IDS_VOL_OPEN
, strOpen
);
243 AppendMenuW(hPopup
, MF_STRING
, IDS_VOL_ADJUST
, strAdjust
);
245 DWORD flags
= TPM_RETURNCMD
| TPM_NONOTIFY
| TPM_RIGHTALIGN
| TPM_BOTTOMALIGN
;
246 DWORD msgPos
= GetMessagePos();
248 SetForegroundWindow(pSysTray
->GetHWnd());
249 DWORD id
= TrackPopupMenuEx(hPopup
, flags
,
250 GET_X_LPARAM(msgPos
), GET_Y_LPARAM(msgPos
),
251 pSysTray
->GetHWnd(), NULL
);
266 HRESULT STDMETHODCALLTYPE
Volume_Message(_In_ CSysTray
* pSysTray
, UINT uMsg
, WPARAM wParam
, LPARAM lParam
, LRESULT
&lResult
)
268 if (uMsg
== g_mmDeviceChange
)
269 return Volume_OnDeviceChange(pSysTray
, wParam
, lParam
);
274 TRACE("Volume_Message: WM_USER+220\n");
278 return Volume_Init(pSysTray
);
280 return Volume_Shutdown(pSysTray
);
285 TRACE("Volume_Message: WM_USER+221\n");
288 lResult
= (LRESULT
)g_IsRunning
;
294 TRACE("Volume_Message uMsg=%d, w=%x, l=%x\n", uMsg
, wParam
, lParam
);
296 Volume_Update(pSysTray
);
304 TRACE("TODO: display volume slider\n");
307 case WM_LBUTTONDBLCLK
:
315 _ShowContextMenu(pSysTray
);
318 case WM_RBUTTONDBLCLK
:
327 TRACE("Volume_Message received for unknown ID %d, ignoring.\n");