Undocumented Matlab
  • SERVICES
    • Consulting
    • Development
    • Training
    • Gallery
    • Testimonials
  • PRODUCTS
    • IQML: IQFeed-Matlab connector
    • IB-Matlab: InteractiveBrokers-Matlab connector
    • EODML: EODHistoricalData-Matlab connector
    • Webinars
  • BOOKS
    • Secrets of MATLAB-Java Programming
    • Accelerating MATLAB Performance
    • MATLAB Succinctly
  • ARTICLES
  • ABOUT
    • Policies
  • CONTACT
  • SERVICES
    • Consulting
    • Development
    • Training
    • Gallery
    • Testimonials
  • PRODUCTS
    • IQML: IQFeed-Matlab connector
    • IB-Matlab: InteractiveBrokers-Matlab connector
    • EODML: EODHistoricalData-Matlab connector
    • Webinars
  • BOOKS
    • Secrets of MATLAB-Java Programming
    • Accelerating MATLAB Performance
    • MATLAB Succinctly
  • ARTICLES
  • ABOUT
    • Policies
  • CONTACT

Updating speaker sound volume

October 15, 2009 9 Comments

In a recent CSSM forum post, a reader asked whether it is possible to get and/or set the system speaker sound volume programmatically. The purpose would be to connect this programming snippet to a GUI control that would allow users to increase/decrease playback volume in the same GUI window as the other playback controls.
This is apparently not possible with documented Matlab functions: we can modify specific playback files, but not the speaker in general. However, it is possible using standard Java calls. Note that I am specifically not mentioning the word undocumented here. The reason is that these Java calls can all be achieved using Matlab’s documented, although mostly unknown, Java interface.
Today, I have submitted a new utility to the Matlab File Exchange, called SoundVolume. SoundVolume enables getting and setting the speaker volume programmatically. For the benefit of users, here is the core of the program (the actual program contains extra sanity checks and exception handling):

function volume = SoundVolume(volume)
   % Loop over the system's mixers to find the speaker port
   import javax.sound.sampled.*
   mixerInfos = AudioSystem.getMixerInfo;
   foundFlag = 0;
   for mixerIdx = 1 : length(mixerInfos)
      mixer = AudioSystem.getMixer(mixerInfos(mixerIdx));
      ports = getTargetLineInfo(mixer);
      for portIdx = 1 : length(ports)
         port = ports(portIdx);
         try
            portName = port.getName;  % better
         catch   %#ok
            portName = port.toString; % sub-optimal
         end
         if ~isempty(strfind(lower(char(portName)),'speaker'))
            foundFlag = 1;  break;
         end
      end
   end
   if ~foundFlag
      error('Speaker port not found');
   end
   % Get and open the speaker port's Line object
   line = AudioSystem.getLine(port);
   line.open();
   % Loop over the Line's controls to find the Volume control
   ctrls = line.getControls;
   foundFlag = 0;
   for ctrlIdx = 1 : length(ctrls)
      ctrl = ctrls(ctrlIdx);
      ctrlName = char(ctrls(ctrlIdx).getType);
      if ~isempty(strfind(lower(ctrlName),'volume'))
         foundFlag = 1;  break;
      end
   end
   if ~foundFlag
      error('Volume control not found');
   end
   % Get or set the volume value according to the user request
   oldValue = ctrl.getValue;
   if nargin
      ctrl.setValue(volume);
   end
   if nargout
      volume = oldValue;
   end

function volume = SoundVolume(volume) % Loop over the system's mixers to find the speaker port import javax.sound.sampled.* mixerInfos = AudioSystem.getMixerInfo; foundFlag = 0; for mixerIdx = 1 : length(mixerInfos) mixer = AudioSystem.getMixer(mixerInfos(mixerIdx)); ports = getTargetLineInfo(mixer); for portIdx = 1 : length(ports) port = ports(portIdx); try portName = port.getName; % better catch %#ok portName = port.toString; % sub-optimal end if ~isempty(strfind(lower(char(portName)),'speaker')) foundFlag = 1; break; end end end if ~foundFlag error('Speaker port not found'); end % Get and open the speaker port's Line object line = AudioSystem.getLine(port); line.open(); % Loop over the Line's controls to find the Volume control ctrls = line.getControls; foundFlag = 0; for ctrlIdx = 1 : length(ctrls) ctrl = ctrls(ctrlIdx); ctrlName = char(ctrls(ctrlIdx).getType); if ~isempty(strfind(lower(ctrlName),'volume')) foundFlag = 1; break; end end if ~foundFlag error('Volume control not found'); end % Get or set the volume value according to the user request oldValue = ctrl.getValue; if nargin ctrl.setValue(volume); end if nargout volume = oldValue; end

Note that our Matlab code could have been much simpler, following the examples provided here and here. Unfortunately, Matlab prevents using Java classnames containing a period (e.g., Port.Info) or static Interface values, so we cannot directly access Port.Info.SPEAKER or FloatControl.Type.VOLUME. I’ve reported this bug to Mathworks earlier this year, and I do not know in which Matlab release (if at all) they intend to fix it. Until then we need to use workarounds, as in the code above. I posted a similar issue (and its workaround) earlier in this blog, when setting system-tray popup message.
For more information about accessing Java’s extensive sound-related functionality, read Sun’s official Java sound documentation trail.

Related posts:

  1. Waiting for asynchronous events – The Matlab waitfor function can be used to wait for asynchronous Java/ActiveX events, as well as with timeouts. ...
  2. Extending a Java class with UDD – Java classes can easily be extended in Matlab, using pure Matlab code. ...
  3. Customizing figure toolbar background – Setting the figure toolbar's background color can easily be done using just a tiny bit of Java magic powder. This article explains how. ...
  4. Setting system tray popup messages – System-tray icons and messages can be programmatically set and controlled from within Matlab, using new functionality available since R2007b....
  5. Uicontrol callbacks – This post details undocumented callbacks exposed by the underlying Java object of Matlab uicontrols, that can be used to modify the control's behavior in a multitude of different events...
  6. Tab panels – uitab and relatives – This article describes several undocumented Matlab functions that support tab-panels...
Java
Print Print
« Previous
Next »
9 Responses
  1. Andrey October 18, 2009 at 10:34 Reply

    Hi Yair!
    As always, your posts are very interesting, thanks for this one.
    What do you think about wrapping this kind of object in Java and loading it with ‘javaobject’?

    • Yair Altman October 22, 2009 at 16:34 Reply

      @Andrey – Since all of the SoundVolume functionality is Java-based, you can of course place it in a Java class, compile it, and load it into Matlab using javaObject or even directly.

      This is a Matlab-centric blog, so I focus on how things can be done in Matlab. But of course they can also be done using other means. For example, calling an external executable utility or ActiveX that does the same thing…

  2. Kostas April 19, 2010 at 15:24 Reply

    Has anyone tried this code with Vista? The code runs, the input argument is stored, but the speaker volume is not affected.

    • Yair Altman May 9, 2014 at 08:52 Reply

      The latest update should hopefully solve the issue with Windows Vista, 7 & 8

  3. Sebastian Pape July 1, 2010 at 07:17 Reply

    Hi!
    I used that code but I got always the message:

    ??? Error using ==> SoundVolume2
    Volume control not found

    (with Win XP)

    Do you have an idea what could be wrong?

    Best Regards

    • Yair Altman July 1, 2010 at 08:07 Reply

      @Sebastian – yes: the program searched all the available controls in your computer’s speaker port’s Line object, and could not find any control that affects the volume. I suggest that you place breakpoints in the relevant loops (lines 53-65 and 81-92) to see why this is. For example, it is possible that you have more than one speaker ports in your system and the program searched the wrong port, or maybe you really don’t have programmatic access to the Volume control…

  4. Niklas Rönnberg November 16, 2010 at 06:49 Reply

    Hi,

    Thanks for this interesting blog post! I’m currently developing different sound related tests in Matlab and for calibration purposes with external equipment it is important that the computer output isn’t changed between the experiments. So, the above code example is really useful. However, since I’m using Windows XP I have another mixer fader to worry about, and that is the Wave fader in the Windows mixer…

    If I use ctrlIdx = 4 I’m able to find the Wave fader, but this is not looking the same as for ctrlIdx = 1, where Matlab states “Volume with current value: 0.0 (range: 0.0 – 1.0)”, instead Matlab states “Wave Control containing Volume, Balance, and Mute Controls.”.

    Do you have any suggestions how I can manage to be able to control the Wave fader as well?

    Many thanks!

    • Yair Altman November 16, 2010 at 10:25 Reply

      @Niklas – different computer systems have different audio devices and drivers. My article merely showed how to solve the problem for the particular configuration that exists on my system. You can use the methodology that I explained above (for exactly this purpose), in order to solve the problem for your particular configuration. If you would like my assistance in your particular case, please email me offline.

      -Yair

    • Eric November 19, 2010 at 07:57 Reply

      @Niklas –
      The Wave control is a compound control when it shows that message, which means that the wave volume control is a sub-control of the wave control. To access it (building off of Yair’s code – by the way, thanks, Yair. This helped me a lot), use something like this (tested with WinXP, Matlab R2009a):

      % Loop over the Line's controls to find the Wave control (a compound
      % control)
      ctrls = line.getControls;
      foundFlag = 0;
      for ctrlIdx = 1 : length(ctrls)
          waveCtrl = ctrls(ctrlIdx);
          ctrlName = char(waveCtrl.getType);
          if ~isempty(strfind(lower(ctrlName),'wave'))
              foundFlag = 1;  break;
          end
      end
      if ~foundFlag
          error('Wave control not found');
      end
      % now find the volume control under the wave control
      ctrls = waveCtrl.getMemberControls;
      foundFlag = 0;
      for ctrlIdx = 1:length(ctrls)
          waveVolCtrl = ctrls(ctrlIdx);
          ctrlName = char(waveVolCtrl.getType);
          if(~isempty(strfind(lower(ctrlName),'volume')))
              foundFlag = 1;
              break;
          end
      end
      if ~foundFlag
          error('Wave volume control not found');
      end

      % Loop over the Line's controls to find the Wave control (a compound % control) ctrls = line.getControls; foundFlag = 0; for ctrlIdx = 1 : length(ctrls) waveCtrl = ctrls(ctrlIdx); ctrlName = char(waveCtrl.getType); if ~isempty(strfind(lower(ctrlName),'wave')) foundFlag = 1; break; end end if ~foundFlag error('Wave control not found'); end % now find the volume control under the wave control ctrls = waveCtrl.getMemberControls; foundFlag = 0; for ctrlIdx = 1:length(ctrls) waveVolCtrl = ctrls(ctrlIdx); ctrlName = char(waveVolCtrl.getType); if(~isempty(strfind(lower(ctrlName),'volume'))) foundFlag = 1; break; end end if ~foundFlag error('Wave volume control not found'); end

Leave a Reply
HTML tags such as <b> or <i> are accepted.
Wrap code fragments inside <pre lang="matlab"> tags, like this:
<pre lang="matlab">
a = magic(3);
disp(sum(a))
</pre>
I reserve the right to edit/delete comments (read the site policies).
Not all comments will be answered. You can always email me (altmany at gmail) for private consulting.

Click here to cancel reply.

Useful links
  •  Email Yair Altman
  •  Subscribe to new posts (feed)
  •  Subscribe to new posts (reader)
  •  Subscribe to comments (feed)
 
Accelerating MATLAB Performance book
Recent Posts

Speeding-up builtin Matlab functions – part 3

Improving graphics interactivity

Interesting Matlab puzzle – analysis

Interesting Matlab puzzle

Undocumented plot marker types

Matlab toolstrip – part 9 (popup figures)

Matlab toolstrip – part 8 (galleries)

Matlab toolstrip – part 7 (selection controls)

Matlab toolstrip – part 6 (complex controls)

Matlab toolstrip – part 5 (icons)

Matlab toolstrip – part 4 (control customization)

Reverting axes controls in figure toolbar

Matlab toolstrip – part 3 (basic customization)

Matlab toolstrip – part 2 (ToolGroup App)

Matlab toolstrip – part 1

Categories
  • Desktop (45)
  • Figure window (59)
  • Guest bloggers (65)
  • GUI (165)
  • Handle graphics (84)
  • Hidden property (42)
  • Icons (15)
  • Java (174)
  • Listeners (22)
  • Memory (16)
  • Mex (13)
  • Presumed future risk (394)
    • High risk of breaking in future versions (100)
    • Low risk of breaking in future versions (160)
    • Medium risk of breaking in future versions (136)
  • Public presentation (6)
  • Semi-documented feature (10)
  • Semi-documented function (35)
  • Stock Matlab function (140)
  • Toolbox (10)
  • UI controls (52)
  • Uncategorized (13)
  • Undocumented feature (217)
  • Undocumented function (37)
Tags
AppDesigner (9) Callbacks (31) Compiler (10) Desktop (38) Donn Shull (10) Editor (8) Figure (19) FindJObj (27) GUI (141) GUIDE (8) Handle graphics (78) HG2 (34) Hidden property (51) HTML (26) Icons (9) Internal component (39) Java (178) JavaFrame (20) JIDE (19) JMI (8) Listener (17) Malcolm Lidierth (8) MCOS (11) Memory (13) Menubar (9) Mex (14) Optical illusion (11) Performance (78) Profiler (9) Pure Matlab (187) schema (7) schema.class (8) schema.prop (18) Semi-documented feature (6) Semi-documented function (33) Toolbar (14) Toolstrip (13) uicontrol (37) uifigure (8) UIInspect (12) uitable (6) uitools (20) Undocumented feature (187) Undocumented function (37) Undocumented property (20)
Recent Comments
Contact us
Captcha image for Custom Contact Forms plugin. You must type the numbers shown in the image
Undocumented Matlab © 2009 - Yair Altman
This website and Octahedron Ltd. are not affiliated with The MathWorks Inc.; MATLAB® is a registered trademark of The MathWorks Inc.
Scroll to top