• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2008, The Android Open Source Project
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  *  * Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  *  * Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24  */
25 
26 #include "AudioPlugin.h"
27 
28 #include <fcntl.h>
29 #include <sys/stat.h>
30 #include <sys/time.h>
31 #include <time.h>
32 #include <math.h>
33 #include <string.h>
34 
35 extern NPNetscapeFuncs*         browser;
36 extern ANPLogInterfaceV0        gLogI;
37 extern ANPCanvasInterfaceV0     gCanvasI;
38 extern ANPPaintInterfaceV0      gPaintI;
39 extern ANPAudioTrackInterfaceV0 gSoundI;
40 extern ANPTypefaceInterfaceV0   gTypefaceI;
41 
42 
inval(NPP instance)43 static void inval(NPP instance) {
44     browser->invalidaterect(instance, NULL);
45 }
46 
rnd16(float x,int inset)47 static uint16_t rnd16(float x, int inset) {
48     int ix = (int)roundf(x) + inset;
49     if (ix < 0) {
50         ix = 0;
51     }
52     return static_cast<uint16_t>(ix);
53 }
54 
inval(NPP instance,const ANPRectF & r,bool doAA)55 static void inval(NPP instance, const ANPRectF& r, bool doAA) {
56     const int inset = doAA ? -1 : 0;
57 
58     PluginObject *obj = reinterpret_cast<PluginObject*>(instance->pdata);
59     NPRect inval;
60     inval.left = rnd16(r.left, inset);
61     inval.top = rnd16(r.top, inset);
62     inval.right = rnd16(r.right, -inset);
63     inval.bottom = rnd16(r.bottom, -inset);
64     browser->invalidaterect(instance, &inval);
65 }
66 
audioCallback(ANPAudioEvent evt,void * user,ANPAudioBuffer * buffer)67 static void audioCallback(ANPAudioEvent evt, void* user, ANPAudioBuffer* buffer) {
68     switch (evt) {
69         case kMoreData_ANPAudioEvent: {
70             SoundPlay* play = reinterpret_cast<SoundPlay*>(user);
71             size_t amount = fread(buffer->bufferData, 1, buffer->size, play->file);
72             buffer->size = amount;
73             if (amount == 0) {
74                 gSoundI.stop(play->track);
75                 fclose(play->file);
76                 play->file = NULL;
77                 // TODO need to notify our main thread to delete the track now
78             }
79 
80             if (play->fileSize > 0) {
81                 // TODO we need to properly update the progress value
82                 play->progress = 1;
83                 inval(play->instance);
84             }
85 
86 
87             break;
88         }
89         default:
90             break;
91     }
92 }
93 
94 ///////////////////////////////////////////////////////////////////////////////
95 
AudioPlugin(NPP inst)96 AudioPlugin::AudioPlugin(NPP inst) : SubPlugin(inst) {
97 
98     const char path[] = "/sdcard/sample.raw";
99 
100     // open a file stream
101     FILE* f = fopen(path, "r");
102     gLogI.log(kDebug_ANPLogType, "--- path %s FILE %p", path, f);
103 
104     // setup our private audio struct's default values
105     m_soundPlay = new SoundPlay;
106     m_soundPlay->instance = inst;
107     m_soundPlay->progress = 0;
108     m_soundPlay->fileSize = 0;
109     m_soundPlay->file = f;
110     m_soundPlay->track = NULL;
111 
112     // create the audio track
113     if (f) {
114         m_soundPlay->track = gSoundI.newTrack(44100, kPCM16Bit_ANPSampleFormat, 2, audioCallback, m_soundPlay);
115         if (!m_soundPlay->track) {
116             fclose(f);
117             m_soundPlay->file = NULL;
118         }
119     }
120 
121     // get the audio file's size
122     int fileDescriptor = open(path, O_RDONLY);
123     struct stat fileStatus;
124 
125     if(fileDescriptor <= 0) {
126         gLogI.log(kError_ANPLogType, "fopen error");
127     }
128     else if (fstat(fileDescriptor, &fileStatus) != 0) {
129         gLogI.log(kDebug_ANPLogType, "File Size: %d", fileStatus.st_size);
130         m_soundPlay->fileSize = fileStatus.st_size;
131     } else {
132         gLogI.log(kError_ANPLogType, "fstat error");
133     }
134 
135     // configure the UI elements
136     m_activeTouch = false;
137 
138     memset(&m_trackRect, 0, sizeof(m_trackRect));
139     memset(&m_playRect,  0, sizeof(m_playRect));
140     memset(&m_pauseRect, 0, sizeof(m_pauseRect));
141     memset(&m_stopRect,  0, sizeof(m_stopRect));
142 
143     m_paintTrack = gPaintI.newPaint();
144     gPaintI.setFlags(m_paintTrack, gPaintI.getFlags(m_paintTrack) | kAntiAlias_ANPPaintFlag);
145     gPaintI.setColor(m_paintTrack, 0xFFC0C0C0);
146 
147     m_paintRect = gPaintI.newPaint();
148     gPaintI.setFlags(m_paintRect, gPaintI.getFlags(m_paintRect) | kAntiAlias_ANPPaintFlag);
149     gPaintI.setColor(m_paintRect, 0xFFA8A8A8);
150 
151     m_paintText = gPaintI.newPaint();
152     gPaintI.setFlags(m_paintText, gPaintI.getFlags(m_paintText) | kAntiAlias_ANPPaintFlag);
153     gPaintI.setColor(m_paintText, 0xFF2F4F4F);
154     gPaintI.setTextSize(m_paintText, 18);
155 
156     m_paintTrackProgress = gPaintI.newPaint();
157     gPaintI.setFlags(m_paintTrackProgress, gPaintI.getFlags(m_paintTrackProgress) | kAntiAlias_ANPPaintFlag);
158     gPaintI.setColor(m_paintTrackProgress, 0xFF545454);
159 
160     m_paintActiveRect = gPaintI.newPaint();
161     gPaintI.setFlags(m_paintActiveRect, gPaintI.getFlags(m_paintActiveRect) | kAntiAlias_ANPPaintFlag);
162     gPaintI.setColor(m_paintActiveRect, 0xFF545454);
163 
164     ANPTypeface* tf = gTypefaceI.createFromName("serif", kItalic_ANPTypefaceStyle);
165     gPaintI.setTypeface(m_paintText, tf);
166     gTypefaceI.unref(tf);
167 
168     //register for touch events
169     ANPEventFlags flags = kTouch_ANPEventFlag;
170     NPError err = browser->setvalue(inst, kAcceptEvents_ANPSetValue, &flags);
171     if (err != NPERR_NO_ERROR) {
172         gLogI.log(kError_ANPLogType, "Error selecting input events.");
173     }
174 }
175 
~AudioPlugin()176 AudioPlugin::~AudioPlugin() {
177     gPaintI.deletePaint(m_paintTrack);
178     gPaintI.deletePaint(m_paintRect);
179     gPaintI.deletePaint(m_paintText);
180     gPaintI.deletePaint(m_paintTrackProgress);
181     gPaintI.deletePaint(m_paintActiveRect);
182     if(m_soundPlay->track)
183         gSoundI.deleteTrack(m_soundPlay->track);
184     delete m_soundPlay;
185 }
186 
supportsDrawingModel(ANPDrawingModel model)187 bool AudioPlugin::supportsDrawingModel(ANPDrawingModel model) {
188     return (model == kBitmap_ANPDrawingModel);
189 }
190 
drawPlugin(const ANPBitmap & bitmap,const ANPRectI & clip)191 void AudioPlugin::drawPlugin(const ANPBitmap& bitmap, const ANPRectI& clip) {
192     ANPCanvas* canvas = gCanvasI.newCanvas(&bitmap);
193 
194     ANPRectF clipR;
195     clipR.left = clip.left;
196     clipR.top = clip.top;
197     clipR.right = clip.right;
198     clipR.bottom = clip.bottom;
199     gCanvasI.clipRect(canvas, &clipR);
200 
201     draw(canvas);
202     gCanvasI.deleteCanvas(canvas);
203 }
204 
draw(ANPCanvas * canvas)205 void AudioPlugin::draw(ANPCanvas* canvas) {
206 
207     PluginObject *obj = (PluginObject*) this->inst()->pdata;
208 
209     gLogI.log(kError_ANPLogType, "Drawing");
210 
211     const float trackHeight = 30;
212     const float buttonWidth = 60;
213     const float buttonHeight = 30;
214     const int W = obj->window->width;
215     const int H = obj->window->height;
216 
217     // color the plugin canvas
218     gCanvasI.drawColor(canvas, 0xFFCDCDCD);
219 
220     // get font metrics
221     ANPFontMetrics fontMetrics;
222     gPaintI.getFontMetrics(m_paintText, &fontMetrics);
223 
224     // draw the track box (1 px from the edge)
225     m_trackRect.left = 1;
226     m_trackRect.top = 1;
227     m_trackRect.right = W - 2;
228     m_trackRect.bottom = 1 + trackHeight;
229     gCanvasI.drawRect(canvas, &m_trackRect, m_paintTrack);
230 
231     // draw the progress bar
232     if (m_soundPlay->progress > 0) {
233         // TODO need to draw progress bar to cover the proper percentage of the track bar
234         gCanvasI.drawRect(canvas, &m_trackRect, m_paintTrackProgress);
235     }
236 
237     // draw the play box (under track box)
238     m_playRect.left = m_trackRect.left + 5;
239     m_playRect.top = m_trackRect.bottom + 10;
240     m_playRect.right = m_playRect.left + buttonWidth;
241     m_playRect.bottom = m_playRect.top + buttonHeight;
242     gCanvasI.drawRect(canvas, &m_playRect, getPaint(&m_playRect));
243     // draw the play box (under track box)
244     const char playText[] = "Play";
245     gCanvasI.drawText(canvas, playText, sizeof(playText)-1, m_playRect.left + 5,
246                       m_playRect.top - fontMetrics.fTop, m_paintText);
247 
248     // draw the pause box (under track box)
249     m_pauseRect.left = m_playRect.right + 20;
250     m_pauseRect.top = m_trackRect.bottom + 10;
251     m_pauseRect.right = m_pauseRect.left + buttonWidth;
252     m_pauseRect.bottom = m_pauseRect.top + buttonHeight;
253     gCanvasI.drawRect(canvas, &m_pauseRect, getPaint(&m_pauseRect));
254     // draw the text in the pause box
255     const char pauseText[] = "Pause";
256     gCanvasI.drawText(canvas, pauseText, sizeof(pauseText)-1, m_pauseRect.left + 5,
257                       m_pauseRect.top - fontMetrics.fTop, m_paintText);
258 
259     // draw the stop box (under track box)
260     m_stopRect.left = m_pauseRect.right + 20;
261     m_stopRect.top = m_trackRect.bottom + 10;
262     m_stopRect.right = m_stopRect.left + buttonWidth;
263     m_stopRect.bottom = m_stopRect.top + buttonHeight;
264     gCanvasI.drawRect(canvas, &m_stopRect, getPaint(&m_stopRect));
265     // draw the text in the pause box
266     const char stopText[] = "Stop";
267     gCanvasI.drawText(canvas, stopText, sizeof(stopText)-1, m_stopRect.left + 5,
268                       m_stopRect.top - fontMetrics.fTop, m_paintText);
269 }
270 
getPaint(ANPRectF * input)271 ANPPaint* AudioPlugin::getPaint(ANPRectF* input) {
272     return (input == m_activeRect) ? m_paintActiveRect : m_paintRect;
273 }
274 
handleEvent(const ANPEvent * evt)275 int16_t AudioPlugin::handleEvent(const ANPEvent* evt) {
276     NPP instance = this->inst();
277 
278     switch (evt->eventType) {
279         case kDraw_ANPEventType:
280             switch (evt->data.draw.model) {
281                 case kBitmap_ANPDrawingModel:
282                     drawPlugin(evt->data.draw.data.bitmap, evt->data.draw.clip);
283                     return 1;
284                 default:
285                     break;   // unknown drawing model
286             }
287 
288         case kTouch_ANPEventType: {
289             int x = evt->data.touch.x;
290             int y = evt->data.touch.y;
291             if (kDown_ANPTouchAction == evt->data.touch.action) {
292 
293                 m_activeTouchRect = validTouch(x,y);
294                 if(m_activeTouchRect) {
295                     m_activeTouch = true;
296                     return 1;
297                 }
298 
299             } else if (kUp_ANPTouchAction == evt->data.touch.action && m_activeTouch) {
300                 handleTouch(x, y);
301                 m_activeTouch = false;
302                 return 1;
303             } else if (kCancel_ANPTouchAction == evt->data.touch.action) {
304                 m_activeTouch = false;
305             }
306             break;
307         }
308         default:
309             break;
310     }
311     return 0;   // unknown or unhandled event
312 }
313 
invalActiveRect()314 void AudioPlugin::invalActiveRect() {
315 
316 }
317 
validTouch(int x,int y)318 ANPRectF* AudioPlugin::validTouch(int x, int y) {
319 
320     if (m_playRect.left && x < m_playRect.right && y > m_playRect.top && y < m_playRect.bottom)
321         return &m_playRect;
322     else if (m_pauseRect.left && x < m_pauseRect.right && y > m_pauseRect.top && y < m_pauseRect.bottom)
323         return &m_pauseRect;
324     else if (x > m_stopRect.left && x < m_stopRect.right && y > m_stopRect.top && y < m_stopRect.bottom)
325         return &m_stopRect;
326     else
327         return NULL;
328 }
329 
handleTouch(int x,int y)330 void AudioPlugin::handleTouch(int x, int y) {
331     NPP instance = this->inst();
332 
333     // if the track is null then return
334     if (NULL == m_soundPlay->track) {
335         gLogI.log(kError_ANPLogType, "---- %p unable to create track",
336                   instance);
337         return;
338     }
339 
340     // check to make sure the currentRect matches the activeRect
341     ANPRectF* currentRect = validTouch(x,y);
342     if (m_activeTouchRect != currentRect)
343         return;
344 
345     if (currentRect == &m_playRect) {
346 
347         gLogI.log(kDebug_ANPLogType, "---- %p starting track (%d)",
348                   m_soundPlay->track, gSoundI.isStopped(m_soundPlay->track));
349 
350         if (gSoundI.isStopped(m_soundPlay->track)) {
351             gSoundI.start(m_soundPlay->track);
352         }
353     }
354     else if (currentRect == &m_pauseRect) {
355 
356         gLogI.log(kDebug_ANPLogType, "---- %p pausing track (%d)",
357                   m_soundPlay->track, gSoundI.isStopped(m_soundPlay->track));
358 
359         if (!gSoundI.isStopped(m_soundPlay->track)) {
360             gSoundI.pause(m_soundPlay->track);
361         }
362     }
363     else if (currentRect == &m_stopRect) {
364 
365         gLogI.log(kDebug_ANPLogType, "---- %p stopping track (%d)",
366                   m_soundPlay->track, gSoundI.isStopped(m_soundPlay->track));
367 
368         if (!gSoundI.isStopped(m_soundPlay->track)) {
369             gSoundI.stop(m_soundPlay->track);
370         }
371         if (m_soundPlay->file) {
372             fseek(m_soundPlay->file, 0, SEEK_SET);
373         }
374     }
375     else {
376         return;
377     }
378 
379     // set the currentRect to be the activeRect
380     m_activeRect = currentRect;
381     inval(instance);
382 }
383