/** * Copyright 2010-2011 Voxeo Corporation * * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this * file except in compliance with the License. * * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software distributed * under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS * OF ANY KIND, either express or implied. See the License for the specific language * governing permissions and limitations under the License. */ package com.voxeo.moho.media; import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; import java.util.concurrent.locks.Condition; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; import javax.media.mscontrol.Parameters; import javax.media.mscontrol.mediagroup.MediaGroup; import javax.media.mscontrol.mediagroup.Player; import org.apache.log4j.Logger; import com.voxeo.moho.common.util.SettableResultFuture; import com.voxeo.moho.event.EventSource; import com.voxeo.moho.event.OutputCompleteEvent; public class OutputImpl<T extends EventSource> implements Output<T> { private static final Logger LOG = Logger.getLogger(OutputImpl.class); protected MediaGroup _group; protected SettableResultFuture<OutputCompleteEvent<T>> _future = new SettableResultFuture<OutputCompleteEvent<T>>(); final Lock lock = new ReentrantLock(); private Condition speedActionResult = lock.newCondition(); private Condition volumeActionResult = lock.newCondition(); private Condition pauseActionResult = lock.newCondition(); private Condition resumeActionResult = lock.newCondition(); private boolean _normalDisconnected = false; protected OutputImpl(final MediaGroup group) { _group = group; } protected void done(final OutputCompleteEvent<T> outputCompleteEvent) { _future.setResult(outputCompleteEvent); } @Override public synchronized void jump(final int index) { if (!_future.isDone()) { final Parameters params = _group.getParameters(null); final int oldValue = (Integer) params.get(Player.JUMP_PLAYLIST_INCREMENT); try { if (index > 0) { params.put(Player.JUMP_PLAYLIST_INCREMENT, index); _group.setParameters(params); _group.triggerAction(Player.JUMP_FORWARD_IN_PLAYLIST); } else if (index < 0) { params.put(Player.JUMP_PLAYLIST_INCREMENT, -index); _group.setParameters(params); _group.triggerAction(Player.JUMP_BACKWARD_IN_PLAYLIST); } } finally { params.put(Player.JUMP_PLAYLIST_INCREMENT, oldValue); _group.setParameters(params); } } } @Override public synchronized void move(final boolean direction, final long time) { if (!_future.isDone()) { final Parameters params = _group.getParameters(null); final int oldValue = (Integer) params.get(Player.JUMP_TIME); params.put(Player.JUMP_TIME, time); _group.setParameters(params); try { if (direction) { _group.triggerAction(Player.JUMP_FORWARD); } else { _group.triggerAction(Player.JUMP_BACKWARD); } } finally { params.put(Player.JUMP_TIME, oldValue); _group.setParameters(params); } } } @Override public void speed(final boolean upOrDown) { lock.lock(); try { if (!_future.isDone()) { if (upOrDown) { _group.triggerAction(Player.SPEED_UP); } else { _group.triggerAction(Player.SPEED_DOWN); } while (!speedResult && !_future.isDone()) { try { speedActionResult.await(5, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } } speedResult = false; } } finally { lock.unlock(); } } protected boolean speedResult = false; protected void speedActionDone() { lock.lock(); speedResult = true; try { speedActionResult.signalAll(); } finally { lock.unlock(); } } protected boolean volumeResult = false; @Override public void volume(final boolean upOrDown) { lock.lock(); try { if (!_future.isDone()) { if (upOrDown) { _group.triggerAction(Player.VOLUME_UP); } else { _group.triggerAction(Player.VOLUME_DOWN); } while (!volumeResult && !_future.isDone()) { try { volumeActionResult.await(5, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } } volumeResult = false; } } finally { lock.unlock(); } } protected void volumeActionDone() { lock.lock(); volumeResult = true; try { volumeActionResult.signalAll(); } finally { lock.unlock(); } } protected boolean paused = false; protected boolean pauseResult = false; @Override public void pause() { lock.lock(); try { if (!_future.isDone() && !paused) { _group.triggerAction(Player.PAUSE); while (!pauseResult && !_future.isDone()) { try { pauseActionResult.await(5, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } } pauseResult = false; } } finally { lock.unlock(); } } protected void pauseActionDone() { lock.lock(); pauseResult = true; paused = true; try { pauseActionResult.signalAll(); } finally { lock.unlock(); } } @Override public void resume() { lock.lock(); try { if (!_future.isDone() && paused) { _group.triggerAction(Player.RESUME); while (!resumeResult && !_future.isDone()) { try { resumeActionResult.await(5, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } } resumeResult = false; } } finally { lock.unlock(); } } protected boolean resumeResult = false; protected void resumeActionDone() { lock.lock(); resumeResult = true; paused = false; try { resumeActionResult.signalAll(); } finally { lock.unlock(); } } @Override public void stop() { if (!_future.isDone()) { _group.triggerAction(Player.STOP); try { _future.get(10, TimeUnit.SECONDS); } catch (InterruptedException e) { // ignore } catch (ExecutionException e) { // ignore } catch (TimeoutException e) { LOG.error("Timeout when waiting output complete for MediaGroup " + _group, e); } } } @Override public boolean cancel(final boolean mayInterruptIfRunning) { return _future.cancel(mayInterruptIfRunning); } @Override public OutputCompleteEvent<T> get() throws InterruptedException, ExecutionException { return _future.get(); } @Override public OutputCompleteEvent<T> get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { return _future.get(timeout, unit); } @Override public boolean isCancelled() { return _future.isCancelled(); } @Override public boolean isDone() { return _future.isDone(); } public synchronized boolean isPending() { return !_future.isDone(); } public void normalDisconnect(boolean normal) { _normalDisconnected = normal; } public boolean isNormalDisconnect() { return _normalDisconnected; } }