Package com.sk89q.jinglenote

Source Code of com.sk89q.jinglenote.MidiJingleSequencer

/*
* CommandBook
* Copyright (C) 2011 sk89q <http://www.sk89q.com>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/

package com.sk89q.jinglenote;

import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;

import javax.sound.midi.InvalidMidiDataException;
import javax.sound.midi.MidiMessage;
import javax.sound.midi.MidiSystem;
import javax.sound.midi.MidiUnavailableException;
import javax.sound.midi.Receiver;
import javax.sound.midi.Sequence;
import javax.sound.midi.Sequencer;
import javax.sound.midi.ShortMessage;

/**
* A sequencer that reads MIDI files.
*
* @author sk89q
*/
public class MidiJingleSequencer implements JingleSequencer {

    private static final int[] instruments = {
        0, 0, 0, 0, 0, 0, 0, 5, // 8
        6, 0, 0, 0, 0, 0, 0, 0, // 16
        0, 0, 0, 0, 0, 0, 0, 5, // 24
        5, 5, 5, 5, 5, 5, 5, 5, // 32
        6, 6, 6, 6, 6, 6, 6, 6, // 40
        5, 5, 5, 5, 5, 5, 5, 2, // 48
        5, 5, 5, 5, 0, 0, 0, 0, // 56
        0, 0, 0, 0, 0, 0, 0, 0, // 64
        0, 0, 0, 0, 0, 0, 0, 0, // 72
        0, 0, 0, 0, 0, 0, 0, 0, // 80
        0, 0, 0, 0, 0, 0, 0, 0, // 88
        0, 0, 0, 0, 0, 0, 0, 0, // 96
        0, 0, 0, 0, 0, 0, 0, 0, // 104
        0, 0, 0, 0, 0, 0, 0, 0, // 112
        1, 1, 1, 3, 1, 1, 1, 5, // 120
        1, 1, 1, 1, 1, 2, 4, 3, // 128
    };

    private static final int[] percussion = {
        3, 3, 4, 4, 3, 2, 3, 2, //8 - Electric Snare
        2, 2, 2, 2, 2, 2, 2, 2, //16 - Hi Mid Tom
        3, 2, 3, 3, 3, 0, 3, 3, //24 - Cowbell
        3, 3, 3, 2, 2, 3, 3, 3, //32 - Low Conga
        2, 2, 0, 0, 2, 2, 0, 0, //40 - Long Whistle
        3, 3, 3, 3, 3, 3, 5, 5, //48 - Open Cuica
        3, 3,                   //50 - Open Triangle
    };

    protected final File midiFile;
    private Sequencer sequencer = null;

    public MidiJingleSequencer(File midiFile, boolean loop) throws MidiUnavailableException, InvalidMidiDataException, IOException {

        this.midiFile = midiFile;

        try {
            sequencer = MidiSystem.getSequencer(false);
            sequencer.open();
            Sequence seq = MidiSystem.getSequence(midiFile);
            sequencer.setSequence(seq);
            if (loop) {
                sequencer.setLoopCount(Sequencer.LOOP_CONTINUOUSLY);
            }
        } catch (MidiUnavailableException e) {
            if (sequencer != null && sequencer.isOpen()) {
                sequencer.close();
            }
            throw e;
        } catch (InvalidMidiDataException e) {
            if (sequencer != null && sequencer.isOpen()) {
                sequencer.close();
            }
            throw e;
        } catch (IOException e) {
            if (sequencer != null && sequencer.isOpen()) {
                sequencer.close();
            }
            throw e;
        }
    }

    @Override
    public void run(final JingleNotePlayer notePlayer) throws InterruptedException {

        final Map<Integer, Integer> patches = new HashMap<Integer, Integer>();

        try {
            if (sequencer.getSequence() == null) {
                return;
            }
            if (!sequencer.isOpen()) {
                sequencer.open();
            }
            sequencer.getTransmitter().setReceiver(new Receiver() {

                @Override
                public void send(MidiMessage message, long timeStamp) {
                    if ((message.getStatus() & 0xF0) == ShortMessage.PROGRAM_CHANGE) {
                        ShortMessage msg = (ShortMessage) message;
                        int chan = msg.getChannel();
                        int patch = msg.getData1();
                        patches.put(chan, patch);
                    } else if ((message.getStatus() & 0xF0) == ShortMessage.NOTE_ON) {
                        ShortMessage msg = (ShortMessage) message;
                        int chan = msg.getChannel();
                        int n = msg.getData1();
                        if (chan == 9) { // Percussion
                            // Sounds like utter crap
                            //notePlayer.play(new Note(toMCSound(toMCPercussion(patches.get(chan))), toMCNote(n),  10 * (msg.getData2() / 127f)));
                        } else {
                            notePlayer.play(new Note(toMCSound(toMCInstrument(patches.get(chan))), toMCNote(n), 10 * (msg.getData2() / 127f)));
                        }
                    }
                }

                @Override
                public void close() {
                }
            });

            try {
                if (sequencer.isOpen()) {
                    sequencer.start();
                }
            }
            catch(Exception ignored) {}

            while (sequencer.isRunning()) {
                Thread.sleep(1000);
            }

            if (sequencer.isRunning()) {
                sequencer.stop();
            }
        } catch (MidiUnavailableException e) {
            e.printStackTrace();
        } finally {
            if (sequencer.isOpen()) {
                sequencer.close();
            }
        }
    }

    @Override
    public void stop() {
        if (sequencer != null && sequencer.isOpen()) {
            sequencer.close();
        }
    }

    protected static byte toMCNote(int n) {
        if (n < 54) {
            return (byte) ((n - 6) % (18 - 6));
        } else if (n > 78) {
            return (byte) ((n - 6) % (18 - 6) + 12);
        } else {
            return (byte) (n - 54);
        }
    }

    protected static byte toMCInstrument(Integer patch) {
        if (patch == null) {
            return 0;
        }
        if (patch < 0 || patch >= instruments.length) {
            return 0;
        }

        return (byte) instruments[patch];
    }

    protected Instrument toMCSound(byte instrument) {
        switch (instrument) {
            case 1:
                return Instrument.BASS_GUITAR;
            case 2:
                return Instrument.SNARE_DRUM;
            case 3:
                return Instrument.STICKS;
            case 4:
                return Instrument.BASS_DRUM;
            case 5:
                return Instrument.GUITAR;
            case 6:
                return Instrument.BASS;
            default:
                return Instrument.PIANO;
        }
    }

    protected static byte toMCPercussion(Integer patch) {
        if (patch == null) {
            return 0;
        }

        int i = patch - 33;
        if (i < 0 || i >= percussion.length) {
            return 1;
        }

        return (byte) percussion[i];
    }


    public boolean isSongPlaying() {
        return sequencer.isRunning();
    }

    public Sequencer getSequencer() {
        return sequencer;
    }
}
TOP

Related Classes of com.sk89q.jinglenote.MidiJingleSequencer

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.