You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

233 lines
6.9KB

  1. /* SpiralSound
  2. * Copyleft (C) 2001 David Griffiths <dave@pawfal.org>
  3. *
  4. * This program is free software; you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation; either version 2 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program; if not, write to the Free Software
  16. * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  17. */
  18. #include "StreamPlugin.h"
  19. #include "StreamPluginGUI.h"
  20. #include <FL/Fl_Button.h>
  21. #include "SpiralIcon.xpm"
  22. #include "../../RiffWav.h"
  23. #include "../../NoteTable.h"
  24. using namespace std;
  25. static const float TRIG_THRESH = 0.1;
  26. static const float BUFSECONDS = 1.0f;
  27. extern "C" {
  28. SpiralPlugin* SpiralPlugin_CreateInstance()
  29. {
  30. return new StreamPlugin;
  31. }
  32. char** SpiralPlugin_GetIcon()
  33. {
  34. return SpiralIcon_xpm;
  35. }
  36. int SpiralPlugin_GetID()
  37. {
  38. return 0x0119;
  39. }
  40. string SpiralPlugin_GetGroupName()
  41. {
  42. return "Delay/Sampling";
  43. }
  44. }
  45. ///////////////////////////////////////////////////////
  46. StreamPlugin::StreamPlugin() :
  47. m_SampleRate (44100),
  48. m_SampleSize (256),
  49. m_StreamPos (0),
  50. m_GlobalPos (0),
  51. m_Pitch (1.0f),
  52. m_SamplePos (-1),
  53. m_Pos (0),
  54. m_Mode(STOPM)
  55. {
  56. m_PluginInfo.Name = "Stream";
  57. m_PluginInfo.Width = 245;
  58. m_PluginInfo.Height = 165;
  59. m_PluginInfo.NumInputs = 3;
  60. m_PluginInfo.NumOutputs = 3;
  61. m_PluginInfo.PortTips.push_back ("Pitch CV");
  62. m_PluginInfo.PortTips.push_back ("Play Trigger");
  63. m_PluginInfo.PortTips.push_back ("Stop Trigger");
  64. m_PluginInfo.PortTips.push_back ("Left Out");
  65. m_PluginInfo.PortTips.push_back ("Right Out");
  66. m_PluginInfo.PortTips.push_back ("Finish Trigger");
  67. m_GUIArgs.Volume = 1.0f;
  68. m_GUIArgs.PitchMod = 1.0f;
  69. m_GUIArgs.PlayOut = false;
  70. m_AudioCH->Register ("Volume", &m_GUIArgs.Volume);
  71. m_AudioCH->Register ("Pitch", &m_GUIArgs.PitchMod, ChannelHandler::INPUT);
  72. m_AudioCH->RegisterData ("FileName", ChannelHandler::INPUT,
  73. &m_GUIArgs.FileName, sizeof (m_GUIArgs.FileName));
  74. m_AudioCH->Register ("Time", &m_GUIArgs.Time);
  75. m_AudioCH->Register ("TimeOut", &m_GUIArgs.TimeOut, ChannelHandler::OUTPUT);
  76. m_AudioCH->Register ("MaxTime", &m_GUIArgs.MaxTime, ChannelHandler::OUTPUT);
  77. m_AudioCH->Register ("Playing", &m_GUIArgs.PlayOut, ChannelHandler::OUTPUT);
  78. }
  79. StreamPlugin::~StreamPlugin()
  80. {
  81. }
  82. PluginInfo &StreamPlugin::Initialise(const HostInfo *Host)
  83. {
  84. PluginInfo& Info = SpiralPlugin::Initialise(Host);
  85. return Info;
  86. }
  87. SpiralGUIType *StreamPlugin::CreateGUI() {
  88. return new StreamPluginGUI(m_PluginInfo.Width, m_PluginInfo.Height, this, m_AudioCH, m_HostInfo);
  89. }
  90. void StreamPlugin::Execute() {
  91. if (m_File.IsOpen()) {
  92. for (int n=0; n<m_HostInfo->BUFSIZE; n++) {
  93. bool FinTrig = false;
  94. float CVPitch = GetInput(0, n)*10.0f;
  95. if (GetInput (1, n) > 0) m_Mode = PLAYM;
  96. if (GetInput (2, n) > 0) {
  97. m_Mode = STOPM;
  98. m_Pos = 0;
  99. m_GlobalPos = 0;
  100. m_StreamPos = 0;
  101. }
  102. if (m_Pos<0) {
  103. m_Pos = m_SampleSize - 1;
  104. m_StreamPos -= m_SampleSize;
  105. FinTrig = m_StreamPos < 0;
  106. if (FinTrig) {
  107. m_StreamPos = m_File.GetSize() - m_SampleSize;
  108. m_GlobalPos = m_StreamPos;
  109. }
  110. m_File.SeekToChunk (m_StreamPos);
  111. m_File.LoadChunk (m_SampleSize, m_SampleL, m_SampleR);
  112. }
  113. else if (m_Pos >= m_SampleSize) {
  114. m_Pos = 0;
  115. m_StreamPos += m_SampleSize;
  116. FinTrig = m_StreamPos >= m_File.GetSize();
  117. if (FinTrig) {
  118. m_StreamPos = 0;
  119. m_GlobalPos = 0;
  120. }
  121. m_File.SeekToChunk (m_StreamPos);
  122. m_File.LoadChunk (m_SampleSize, m_SampleL, m_SampleR);
  123. }
  124. if (FinTrig) SetOutput (2, n, 1);
  125. else SetOutput (2, n, 0);
  126. if (m_Mode==PLAYM) {
  127. SetOutput (0, n, m_SampleL[m_Pos] * m_GUIArgs.Volume);
  128. SetOutput (1, n, m_SampleR[m_Pos] * m_GUIArgs.Volume);
  129. m_Pos += m_GUIArgs.PitchMod + CVPitch;
  130. m_GlobalPos += m_GUIArgs.PitchMod + CVPitch;
  131. }
  132. else {
  133. SetOutput (0, n, 0);
  134. SetOutput (1, n, 0);
  135. }
  136. }
  137. m_GUIArgs.TimeOut = m_GlobalPos / (float)m_SampleRate;
  138. m_GUIArgs.PlayOut = m_Mode==PLAYM;
  139. }
  140. }
  141. void StreamPlugin::ExecuteCommands() {
  142. if (m_AudioCH->IsCommandWaiting()) {
  143. switch (m_AudioCH->GetCommand()) {
  144. case SET_TIME:
  145. SetTime();
  146. break;
  147. case LOAD:
  148. OpenStream();
  149. break;
  150. case RESTART:
  151. m_StreamPos = 0;
  152. m_GlobalPos = 0;
  153. break;
  154. case STOP:
  155. m_Mode = STOPM;
  156. break;
  157. case PLAY:
  158. m_Mode = PLAYM;
  159. break;
  160. }
  161. }
  162. }
  163. void StreamPlugin::SetTime (void) {
  164. m_GlobalPos = m_SampleRate * m_GUIArgs.Time;
  165. m_StreamPos = (int)(m_SampleRate * m_GUIArgs.Time);
  166. m_Pos = m_SampleSize;
  167. }
  168. void StreamPlugin::OpenStream (void) {
  169. m_StreamPos = 0;
  170. m_GlobalPos = 0;
  171. if (m_File.IsOpen ()) m_File.Close ();
  172. m_File.Open (m_GUIArgs.FileName, WavFile::READ);
  173. m_SampleL.Allocate (m_SampleSize);
  174. m_SampleR.Allocate (m_SampleSize);
  175. m_Pitch = m_SampleRate / (float)m_HostInfo->SAMPLERATE;
  176. if (m_File.IsStereo ()) {
  177. m_Pitch *= 2;
  178. m_GUIArgs.MaxTime = GetLength();
  179. }
  180. else m_GUIArgs.MaxTime = GetLength() / 2;
  181. }
  182. float StreamPlugin::GetLength (void) {
  183. if (m_File.IsStereo()) return m_File.GetSize() / (float)m_File.GetSamplerate ();
  184. else return m_File.GetSize() / (float)m_File.GetSamplerate () * 2;
  185. }
  186. void StreamPlugin::StreamOut (ostream &s) {
  187. s << m_Version << " "
  188. << m_GUIArgs.Volume << " "
  189. << m_GUIArgs.PitchMod << " "
  190. << strlen (m_GUIArgs.FileName) << " "
  191. << m_GUIArgs.FileName << " "
  192. // is it really necessary to save this lot??
  193. << m_Pos << " "
  194. << m_StreamPos << " "
  195. << m_GlobalPos << " "
  196. << m_Pitch << " " << endl;
  197. }
  198. void StreamPlugin::StreamIn(istream &s) {
  199. int version;
  200. s >> version;
  201. s >> m_GUIArgs.Volume >> m_GUIArgs.PitchMod;
  202. int size;
  203. s >> size;
  204. if (size > 255) size = 255;
  205. s.ignore (1);
  206. s.get (m_GUIArgs.FileName, size+1);
  207. if (m_GUIArgs.FileName != "None") OpenStream ();
  208. // is it really necessary to load this lot??
  209. s >> m_Pos;
  210. s >> m_StreamPos;
  211. s >> m_GlobalPos;
  212. s >> m_Pitch;
  213. }