* @version 1
*/
-public class MooTrackView extends JPanel implements ActionListener {
+public class MooTrackView extends JPanel {
private Track track;
- private NoteArea notes;
+ private MooTrackTitle title;
private Rectangle box;
+
private JPopupMenu popup;
- private JPopupMenu notePopup;
- private JMenuItem menuItem;
+ private JMenuItem popupAdd;
+ private ArrayList rects;
+ private Insets insets;
+ private int ticksPerSixteenth, popupY = 0;
protected static int viewLength = 0;
+ protected static int extraHeight = 0;
public static final int NOTE_HEIGHT = 10, NOTE_WIDTH = 40, VIEW_WIDTH = 200;
- public MooTrackView (Track track) {
+ public MooTrackView (Track track, MooTrackTitle title) {
super(true);
this.track = track;
- setLayout(new BorderLayout());
+ this.title = title;
+ insets = getInsets();
+
+ // Configures panel
+ setBackground(Color.white);
+ setBorder(BorderFactory.createLineBorder(Color.black));
+ setLayout(null);
+ setPreferredSize(new Dimension(VIEW_WIDTH, 140 * NOTE_HEIGHT));
+
+ // Creates temporary variables
+ MidiEvent note;
+ MooNoteElement elem;
+ rects = new ArrayList(track.size() / 2);
+ extraHeight = Toolkit.getDefaultToolkit().getScreenSize().height - 150;
+
+ // Places note elements
+ for (int i = 0; i < track.size(); i++) {
+ note = track.get(i);
+ if (note instanceof MooNote) {
+ // Adds the note element to the note area and moves it to the appropriate place.
+ MooNote mn = (MooNote)note;
+ elem = new MooNoteElement(this, mn);
+ add(elem);
+ layoutElement(elem, false);
+ }
+ setPreferredSize(new Dimension(VIEW_WIDTH, viewLength + extraHeight));
- notes = new NoteArea(track);
- notes.setBackground(Color.white);
- notes.setBorder(BorderFactory.createLineBorder(Color.black));
+ }
+ // Creates pop-up menu.
popup = new JPopupMenu();
- menuItem = new JMenuItem("Add...");
- menuItem.addActionListener(this);
- popup.add(menuItem);
+ PopupListener pList = new PopupListener();
+ popupAdd = new JMenuItem("Add note...");
+ popupAdd.addActionListener(pList);
+ popup.add(popupAdd);
+
+ // Adds listeners for popup menu and keyboard synthesizer.
+ addMouseListener(new MAdapter());
+ addKeyListener(new MooKeyboard());
+ }
- notePopup = new JPopupMenu();
- menuItem = new JMenuItem("Preferences...");
- menuItem.addActionListener(this);
- notePopup.add(menuItem);
+ public void layoutElement(MooNoteElement elem, boolean old){
+ // If the element is currently in the view, removes its coordinates from the list.
+ Rectangle r = new Rectangle();
+ if (old){
+ r = elem.getBounds(r);
+ for (Iterator i = rects.iterator(); i.hasNext();){
+ Object ob = i.next();
+ if (r.equals(ob)){
+ rects.remove(ob);
+ break;
+ }
+ }
+ }
-// notes.addMouseListener(new PopupListener());
- add(notes, BorderLayout.CENTER);
+ // Creates temporary variables.
+ ticksPerSixteenth = Moosique.getSequence().getResolution() / 4;
+ MooNote mn = elem.getNote();
+ int x, y, height;
+
+ // Calculates coordinates.
+ x = insets.left;
+ y = insets.top + (int)(mn.getTick() / ticksPerSixteenth) * NOTE_HEIGHT;
+ height = (mn.getDuration() / ticksPerSixteenth) * NOTE_HEIGHT;
+ if (height == 0) height = NOTE_HEIGHT;
+ r = new Rectangle(x, y, NOTE_WIDTH, height);
+
+ // Places the element in the appropriate place.
+ while(isOccupied(r)) r.translate(NOTE_WIDTH, 0);
+ elem.setBounds(r);
+ rects.add(r);
+ if (viewLength < (y + height)){
+ viewLength = y + height;
+ if(old)setPreferredSize(new Dimension(VIEW_WIDTH, viewLength + extraHeight));
+ }
}
- public void actionPerformed(ActionEvent e) {}
-
+ /**
+ * Returns the track of this view.
+ * @return the track of this view
+ */
public Track getTrack() {
return track;
}
+ /**
+ * Returns the title of this view.
+ * @return the title of this view
+ */
+ public MooTrackTitle getTitle() {
+ return title;
+ }
+ private boolean isOccupied(Rectangle r) {
+ Iterator it = rects.iterator();
+ while (it.hasNext()) {
+ if(r.intersects((Rectangle)it.next())) return true;
+ }
+ return false;
+ }
+
/**
- * Updates the track view.
+ * Adds the given note to the current track, and visualises it.
+ * @param mn the note to add
*/
- public void update(long tickPosition) {
+ public void addNote(MooNote mn) {
+ mn.addTo(track);
+ MooNoteElement elem = new MooNoteElement(this, mn);
+ add(elem);
+ layoutElement(elem, false);
+ setPreferredSize(new Dimension(VIEW_WIDTH, viewLength + extraHeight));
repaint();
}
- class NoteArea extends JPanel {
- public static final int NOTE_HEIGHT = 10, NOTE_WIDTH = 40, VIEW_WIDTH = 200;
- private ArrayList rects;
-
- public NoteArea(Track track) {
- // Configuring panel
- super(true);
- setLayout(null);
- setPreferredSize(new Dimension(VIEW_WIDTH, 140 * NOTE_HEIGHT));
-
- // Creating temporary variables
- MidiEvent note;
- MooNoteElement elem;
- int x, y, height;
- int beatsPerSixteenth = Moosique.getSequence().getResolution() / 4;
- rects = new ArrayList(track.size() / 2);
-
- // Placing note elements
- Insets insets = getInsets();
- for (int i = 0; i < track.size(); i++) {
- note = track.get(i);
- if (note instanceof MooNote) {
- // Adds the note element to the note area.
- MooNote mn = (MooNote)note;
- elem = new MooNoteElement(mn);
- add(elem);
-
- // Moves the note element to the appropriate place.
- x = insets.left;
- y = insets.top + (int)(mn.getTick() / beatsPerSixteenth) * NOTE_HEIGHT;
- height = (mn.getDuration() / beatsPerSixteenth) * NOTE_HEIGHT;
- if (height == 0) height = NOTE_HEIGHT;
- Rectangle r = new Rectangle(x, y, NOTE_WIDTH, height);
- while(isOccupied(r)) r.translate(NOTE_WIDTH, 0);
- elem.setBounds(r);
- rects.add(r);
- if (viewLength < (y + height)) viewLength = y + height;
- }
- setPreferredSize(new Dimension(VIEW_WIDTH, viewLength));
- }
- validate();
- }
-
- private boolean isOccupied(Rectangle r) {
- Iterator it = rects.iterator();
- while (it.hasNext()) {
- if(r.intersects((Rectangle)it.next())) return true;
- }
- return false;
- }
+ /**
+ * Removes the given note element from the view and its note from the current track.
+ * @param elem the note element to remove
+ */
+ public void removeNote(MooNoteElement elem) {
+ elem.getNote().removeFrom(track);
+ remove(elem);
+ elem.getNote().removeFrom(track);
+ repaint();
+ }
- public void paintComponent(Graphics g) {
- super.paintComponent(g);
- Graphics2D g2 = (Graphics2D)g;
- for (int c = 0; c < viewLength || c < getHeight(); c += NOTE_HEIGHT) {
- for (int r = 0; r < (10 * NOTE_WIDTH); r += NOTE_WIDTH) {
- box = new Rectangle(r, c, NOTE_WIDTH, NOTE_HEIGHT);
- g2.setColor(Color.gray);
- g2.draw(box);
- }
+ private void addStandardNote() {
+ long timestamp = (long)(ticksPerSixteenth * (popupY - insets.top) / NOTE_HEIGHT);
+ System.out.println(ticksPerSixteenth + ", " + popupY + ", " + insets.top + ", " + timestamp);
+ addNote(new MooNote(title.getChannel(), 60, 100, timestamp, Moosique.getSequence().getResolution() / 4));
+ }
+
+ public void paintComponent(Graphics g) {
+ super.paintComponent(g);
+ Graphics2D g2 = (Graphics2D)g;
+ for (int c = 0; c < viewLength || c < getHeight(); c += NOTE_HEIGHT) {
+ for (int r = 0; r < (10 * NOTE_WIDTH); r += NOTE_WIDTH) {
+ box = new Rectangle(r, c, NOTE_WIDTH, NOTE_HEIGHT);
+ g2.setColor(Color.gray);
+ g2.draw(box);
}
}
}
- class PopupListener extends MouseAdapter {
+ class MAdapter extends MouseAdapter {
+ public void mouseClicked(MouseEvent e) {
+ if (e.getClickCount() == 2) {
+ popupY = e.getY();
+ addStandardNote();
+ }
+ }
+
public void mousePressed(MouseEvent e) {
- maybeShowPopup(e);
+ if (e.isPopupTrigger()) {
+ popupY = e.getY();
+ popup.show(e.getComponent(), e.getX(), e.getY());
+ }
}
- public void mouseReleased(MouseEvent e) {
- maybeShowPopup(e);
+ public void mouseEntered(MouseEvent e) {
+ // Moosique.setActiveChannel(track.getChannel());
+ grabFocus();
}
+ }
- private void maybeShowPopup(MouseEvent e) {
- if (e.isPopupTrigger()) {
- if (findComponentAt(e.getX(), e.getY()) instanceof MooNoteElement)
- notePopup.show(e.getComponent(), e.getX(), e.getY());
- else
- popup.show(e.getComponent(), e.getX(), e.getY());
+ class PopupListener implements ActionListener {
+ public void actionPerformed(ActionEvent e) {
+ Object source = e.getSource();
+ if (source == popupAdd) {
+ addStandardNote();
}
+ // new MooNote(int channel, int pitch, int velocity, long timestamp, int duration)
}
}
-}
+}
\ No newline at end of file