summaryrefslogtreecommitdiff
path: root/libs/ardour/ardour/smf_source.h
blob: 4855cb6e319c69399273f37f61982265d0128dc4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
/*
    Copyright (C) 2006 Paul Davis
	Written by Dave Robillard, 2006

    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 2 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, write to the Free Software
    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.

*/

#ifndef __ardour_smf_filesource_h__ 
#define __ardour_smf_filesource_h__

#include <cstdio>
#include <time.h>

#include <ardour/midi_source.h>
#include <evoral/SMF.hpp>

namespace Evoral { template<typename T> class Event; }

namespace ARDOUR {

template<typename T> class MidiRingBuffer;

/** Standard Midi File (Type 0) Source */
class SMFSource : public MidiSource, public Evoral::SMF {
  public:
	/** Constructor for existing external-to-session files */
	SMFSource (Session& session, std::string path, Source::Flag flags = Source::Flag(0));

	/* Constructor for existing in-session files */
	SMFSource (Session& session, const XMLNode&);

	virtual ~SMFSource ();

	bool set_name (const std::string& newname) { return (set_source_name(newname, false) == 0); }
	int set_source_name (string newname, bool destructive);
	
	static bool safe_file_extension (const Glib::ustring& path);
	
	Glib::ustring path() const { return _path; }

	void set_allow_remove_if_empty (bool yn);
	void mark_for_remove();
	
	void append_event_unlocked_beats(const Evoral::Event<double>& ev);
	void append_event_unlocked_frames(const Evoral::Event<nframes_t>& ev);

	int move_to_trash (const string trash_dir_name);

	void mark_streaming_midi_write_started (NoteMode mode, nframes_t start_time);
	void mark_streaming_write_completed ();

	void   mark_take (string);
	string take_id() const { return _take_id; }

	static void set_search_path (string);
	static void set_header_position_offset (nframes_t offset, bool negative);

	XMLNode& get_state ();
	int set_state (const XMLNode&);

	void load_model(bool lock=true, bool force_reload=false);
	void destroy_model();

	void flush_midi();

  private:
	int init (string idstr, bool must_exist);

	nframes_t read_unlocked (
			MidiRingBuffer<nframes_t>& dst,
			nframes_t start,
			nframes_t cn,
			nframes_t stamp_offset,
			nframes_t negative_stamp_offset) const;

	nframes_t write_unlocked (
			MidiRingBuffer<nframes_t>& src,
			nframes_t cnt);

	bool find (std::string path, bool must_exist, bool& is_new);
	bool removable() const;
	bool writable() const { return _flags & Writable; }
	
	void set_default_controls_interpolation();

	Glib::ustring  _path;
	string         _take_id;
	bool           _allow_remove_if_empty;
	double         _last_ev_time_beats;
	nframes_t      _last_ev_time_frames;

	static string _search_path;
};

}; /* namespace ARDOUR */

#endif /* __ardour_smf_filesource_h__ */