Reemplazo a Tone() y Beep()

Reemplazo a Tone() y Beep()

Postby Cgallegoa » Tue Nov 24, 2009 11:52 pm

Hola amigos,

Estoy haciendo una pequeña aplicación en la que es indispensable el uso de las funciones Tone() o Beep() en virtud de que se necesitan diferentes frecuencias y duraciones.

Hemos probado en 13 equipos nuevos, incluídos portátiles, y ninguno trae Speaker Interno. Parece que la tendencia en los nuevos equipos es hacia parlantes externos vía tarjeta de audio/video. Estas funciones sólo emiten sonidos hacia el Speaker Interno. Hemos probado de todo: Con FWH, sólo con xHarbour, Clipper puro y duro, Pascal, etc., y nada. Hemos seguido todas las sugerencias encontradas en este maravilloso foro, y nada. Definitivamente es la falta del condenado Speaker interno.

Hay alguna forma de reemplazar estas funciones por alguna que permita Frecuencia y Duración pero que emita el sonido hacia parlantes externos si no encuenta el Speaker interno ? Los archivos wav no sirve. Son estáticos. Alguna pista ? No tengo la más remota idea de cómo hacerlo. (midi, media, etc.)

Por supuesto, tampoco funciona MsgBeep(). Temporalmente reemplazamos MsgBeep() por SndPlaySound("ERROR.WAV",1).

Pero, necesitamos manejar la Frecuencia y Duración (Tonos y Tiempos) de los sonidos.

Creo que es muy importante encontrar una alternativa dada la característica de los nuevos equipos.

Saludos y gracias por su ayuda,

Carlos Gallego
Saludos,

Carlos Gallego

*** FWH-24.07, xHarbour 1.3.1 Build 20240624, Borland C++7.70, PellesC ***
Cgallegoa
 
Posts: 492
Joined: Sun Oct 16, 2005 3:32 am
Location: Quito - Ecuador

Re: Reemplazo a Tone() y Beep()

Postby Alfredo Arteaga » Wed Nov 25, 2009 1:54 am

Carlos. Busqué en el foro por 'speaker' y encontré esto:

Code: Select all  Expand view
// The following procs provide various sounds which can be
// requested from anywhere in the app...

// Here are 4 "songs", each of which can be requested...

PROCEDURE Correct()
tunes({ {392,2}, {523,2}, {880,2}, {1046,8} } )
RETURN

PROCEDURE Wrong()
tunes( { {164,10} } )
RETURN

PROCEDURE TheFifth()
tunes( { {392,2}, {392,2}, {392,2}, {311,8} } )
RETURN

PROCEDURE NannyBoo()
tunes( { {196,4}, {196,4}, {164,4}, {220,4}, {196,8}, {164,8} } )
RETURN

// Now here is where a song gets initially processed...

PROCEDURE tunes(aSong)
aeval(aSong, { | a | MYTONE(a[1], a[2]) } )
RETURN

// And here is where the C code takes over to finish the job...

#pragma BEGINDUMP

#include <hbapi.h>
#include <windows.h>

HB_FUNC( MYTONE )
{
   Beep( hb_parnl( 1 ), hb_parnl( 2 ) * 70 );
}

#pragma ENDDUMP
 


Según comentario de Roger Seiler "Confirmed: this "PRAGMA" solution works here on both internal and external speakers, and on XP Pro and Vista Home Premium."

Lo probé con inetrnal speakers y se oye bien.

Esperamos tus comentarios.
User avatar
Alfredo Arteaga
 
Posts: 326
Joined: Sun Oct 09, 2005 5:22 pm
Location: Mexico

Re: Reemplazo a Tone() y Beep()

Postby Cgallegoa » Wed Nov 25, 2009 6:23 pm

Alfredo, gracias por tu respuesta.

Ya había probado el código suegerido por Roger Seiler. Pero nada. Parece que si el equipo no tiene parlante interno no hay forma de que Beep() suene. Ya hemos probado de todo sin suerte. Nos queda conseguir un equipo viejo que tenga el bendito parlante.

Gracias por tu interés. Si se te ocurre aglo, lánzame una cuerda por favor.

Maestro Antonio, alguna sugerencia ?

Saludos,

Carlos Gallego
Saludos,

Carlos Gallego

*** FWH-24.07, xHarbour 1.3.1 Build 20240624, Borland C++7.70, PellesC ***
Cgallegoa
 
Posts: 492
Joined: Sun Oct 16, 2005 3:32 am
Location: Quito - Ecuador

Re: Reemplazo a Tone() y Beep()

Postby Antonio Linares » Wed Nov 25, 2009 6:46 pm

Carlos,

En Windows 7 Beep() usa los altavoces externos:

On Windows 7 Beep does not make sound on a motherboard speaker, but on a sound card


http://msdn.microsoft.com/en-us/library/ms679277(VS.85).aspx

Una solución podría ser que useis Windows 7 :-)
regards, saludos

Antonio Linares
www.fivetechsoft.com
User avatar
Antonio Linares
Site Admin
 
Posts: 42080
Joined: Thu Oct 06, 2005 5:47 pm
Location: Spain

Re: Reemplazo a Tone() y Beep()

Postby Antonio Linares » Wed Nov 25, 2009 6:59 pm

Carlos,

Aqui teneis un ejemplo de como simular tone en la tarjeta externa usando DirectSound:

http://www.planet-source-code.com/vb/scripts/ShowCode.asp?txtCodeId=39362&lngWId=1

Prueba el ejemplo y si te funciona, habría que portar el código a C para usarlo desde [x]Harbour
regards, saludos

Antonio Linares
www.fivetechsoft.com
User avatar
Antonio Linares
Site Admin
 
Posts: 42080
Joined: Thu Oct 06, 2005 5:47 pm
Location: Spain

Re: Reemplazo a Tone() y Beep()

Postby Cgallegoa » Thu Nov 26, 2009 4:55 am

Antonio, :D

Estás de muy buen humor.

Windows 7 : El médico me prohibió que lo usara antes de 7 meses hasta ver si ese nuevo virus-S.O. muta y gira hacia AH1N1-Vista, o AH1N1-XP ó es un nuevo bicho.

El ejemplo que me indicas está en VB 6.0 y no tengo la herramienta (ni la he usado) :?

Tal vez alguna alma caritativa que tenga VB6, quiera compilar el ejemplo y envierme el exe.

Saludos,

Carlos Gallego
Saludos,

Carlos Gallego

*** FWH-24.07, xHarbour 1.3.1 Build 20240624, Borland C++7.70, PellesC ***
Cgallegoa
 
Posts: 492
Joined: Sun Oct 16, 2005 3:32 am
Location: Quito - Ecuador

Re: Reemplazo a Tone() y Beep()

Postby Cgallegoa » Thu Nov 26, 2009 7:26 am

Maestro Antonio,

Por aquello de la dignidad, ya intalé VB y lo compilé. Funciona perfecto. Ahora voy a ver si puedo simular las notas musicales.

Como se portaría a C ? Hasta allá no llegan mis conocimientos. :oops:

Este es el código en VB:
Code: Select all  Expand view
VERSION 5.00
Begin VB.Form frmMain
   BorderStyle     =   1  'Fixed Single
   Caption         =   "Oscillator 1.0 - Carlos Gallego VB 6.0"
   ClientHeight    =   5775
   ClientLeft      =   45
   ClientTop       =   330
   ClientWidth     =   5655
   BeginProperty Font
      Name            =   "Tahoma"
      Size            =   8.25
      Charset         =   238
      Weight          =   400
      Underline       =   0   '
False
      Italic          =   0   'False
      Strikethrough   =   0   '
False
   EndProperty
   LockControls    =   -1  'True
   MaxButton       =   0   '
False
   ScaleHeight     =   385
   ScaleMode       =   3  'Pixel
   ScaleWidth      =   377
   StartUpPosition =   2  '
CenterScreen
   Begin VB.Frame fraMod
      Caption         =   "Amplitude Modulation"
      ForeColor       =   &H00800080&
      Height          =   615
      Left            =   2280
      TabIndex        =   3
      Top             =   120
      Width           =   3255
      Begin VB.HScrollBar hscMod
         Height          =   200
         LargeChange     =   10
         Left            =   120
         Max             =   100
         TabIndex        =   4
         Top             =   280
         Width           =   3015
      End
   End
   Begin VB.Frame fraInfo
      Caption         =   "Details"
      ForeColor       =   &H00800080&
      Height          =   1335
      Left            =   120
      TabIndex        =   14
      Top             =   4320
      Width           =   5415
      Begin VB.Label lblDetails
         ForeColor       =   &H00800000&
         Height          =   975
         Left            =   120
         TabIndex        =   15
         Top             =   240
         Width           =   5175
      End
   End
   Begin VB.Frame fraOsc
      Caption         =   "Oscilloscope"
      ForeColor       =   &H00800080&
      Height          =   1935
      Left            =   120
      TabIndex        =   11
      Top             =   2280
      Width           =   5415
      Begin VB.PictureBox picOsc
         AutoRedraw      =   -1  'True
         BackColor       =   &H00004000&
         Enabled         =   0   '
False
         Height          =   1215
         Left            =   120
         ScaleHeight     =   77
         ScaleMode       =   3  'Pixel
         ScaleWidth      =   341
         TabIndex        =   12
         TabStop         =   0   '
False
         Top             =   240
         Width           =   5175
      End
      Begin VB.Line lin2
         BorderColor     =   &H000000FF&
         X1              =   5280
         X2              =   5280
         Y1              =   1440
         Y2              =   1800
      End
      Begin VB.Line lin1
         BorderColor     =   &H000000FF&
         X1              =   120
         X2              =   120
         Y1              =   1440
         Y2              =   1800
      End
      Begin VB.Line linGraph
         BorderColor     =   &H00FF0000&
         X1              =   120
         X2              =   5280
         Y1              =   1680
         Y2              =   1680
      End
      Begin VB.Label lblGraph
         Alignment       =   2  'Center
         ForeColor       =   &H00004000&
         Height          =   255
         Left            =   120
         TabIndex        =   13
         Top             =   1440
         Width           =   5175
      End
   End
   Begin VB.Frame fraOpt
      Caption         =   "Oscillator"
      ForeColor       =   &H00800080&
      Height          =   615
      Left            =   120
      TabIndex        =   0
      Top             =   120
      Width           =   2055
      Begin VB.OptionButton optOsc
         Appearance      =   0  '
Flat
         Caption         =   "Square"
         ForeColor       =   &H00800000&
         Height          =   255
         Index           =   1
         Left            =   1080
         TabIndex        =   2
         Top             =   240
         Width           =   855
      End
      Begin VB.OptionButton optOsc
         Appearance      =   0  'Flat
         Caption         =   "Sine"
         ForeColor       =   &H00800000&
         Height          =   255
         Index           =   0
         Left            =   120
         TabIndex        =   1
         Top             =   240
         Value           =   -1  '
True
         Width           =   855
      End
   End
   Begin VB.Frame fraVolume
      Caption         =   "Volume"
      ForeColor       =   &H00800080&
      Height          =   615
      Left            =   120
      TabIndex        =   8
      Top             =   1560
      Width           =   5415
      Begin VB.HScrollBar hscVol
         Height          =   200
         LargeChange     =   50
         Left            =   120
         Max             =   1000
         TabIndex        =   9
         Top             =   280
         Value           =   100
         Width           =   4335
      End
      Begin VB.Label lblVol
         Alignment       =   1  'Right Justify
         ForeColor       =   &H00800000&
         Height          =   255
         Left            =   4560
         TabIndex        =   10
         Top             =   255
         Width           =   735
      End
   End
   Begin VB.Frame fraFreq
      Caption         =   "Frequency"
      ForeColor       =   &H00800080&
      Height          =   615
      Left            =   120
      TabIndex        =   5
      Top             =   840
      Width           =   5415
      Begin VB.HScrollBar hscFreq
         Height          =   200
         LargeChange     =   50
         Left            =   120
         Max             =   1000
         Min             =   1
         TabIndex        =   6
         Top             =   280
         Value           =   185
         Width           =   4335
      End
      Begin VB.Label lblFreq
         Alignment       =   1  '
Right Justify
         ForeColor       =   &H00800000&
         Height          =   255
         Left            =   4560
         TabIndex        =   7
         Top             =   255
         Width           =   735
      End
   End
   Begin VB.Timer tmrMod
      Left            =   2160
      Top             =   0
   End
End
Attribute VB_Name = "frmMain"
Attribute VB_GlobalNameSpace = False
Attribute VB_Creatable = False
Attribute VB_PredeclaredId = True
Attribute VB_Exposed = False
Option Explicit

Const nSamples = 44100
Const nBasicBufferSize = 4096
Const pi = 3.14159265358979

Dim DX7 As New DirectX7, DS As DirectSound, DSB As DirectSoundBuffer
Dim PCM As WAVEFORMATEX, DSBD As DSBUFFERDESC
Dim nFreq&, nMod!, nModDir%

Private Sub SinBuffer(ByVal nFrequency&, ByVal nVolume!, Optional ByVal bSquare As Boolean)
Dim lpBuffer() As Byte, I&, C!, nBuffer&
lblFreq = FormatNumber(nFreq, 0) & " Hz"
lblVol = FormatPercent(nVolume, 0)
C = nSamples / nFrequency
nBuffer = (nBasicBufferSize \ C) * C
If nBuffer = 0 Then nBuffer = C
ReDim lpBuffer(nBuffer - 1)
For I = 0 To nBuffer - 1
C = Sin(I * 2 * pi / nSamples * nFrequency)
If bSquare Then
C = Sgn(C)
If C = 0 Then C = 1
End If
lpBuffer(I) = (C * nMod * nVolume + 1) * 127.5!
Next
If DSBD.lBufferBytes <> nBuffer Then
DSBD.lBufferBytes = nBuffer
Set DSB = DS.CreateSoundBuffer(DSBD, PCM)
End If
DSB.WriteBuffer 0, 0, lpBuffer(0), DSBLOCK_ENTIREBUFFER
DSB.Play DSBPLAY_LOOPING
lblDetails = "Channels: 1 (Mono)" & vbCrLf _
& "Bits per sample: 8" & vbCrLf _
& "Samples per second: " & FormatNumber(44100, 0) & vbCrLf _
& "DirectSound buffer size: " & FormatNumber(nBuffer, 0) & " bytes" & vbCrLf _
& "Period: " & FormatNumber(1000 / nFrequency, 3) & " ms"
C = 1000
Do While nFrequency * 20 > picOsc.ScaleWidth
nFrequency = nFrequency \ 2
C = C / 2
Loop
lblGraph = FormatNumber(C, 1) & " ms"
picOsc.Cls
picOsc.Line (0, picOsc.ScaleHeight \ 2)-(picOsc.ScaleWidth, picOsc.ScaleHeight \ 2), &H8000&
picOsc.Line (0, (picOsc.ScaleHeight \ 2) * (1 - nVolume))-(picOsc.ScaleWidth, (picOsc.ScaleHeight \ 2) * (1 - nVolume)), &H6000&
picOsc.Line (0, (picOsc.ScaleHeight \ 2) * (1 + nVolume))-(picOsc.ScaleWidth, (picOsc.ScaleHeight \ 2) * (1 + nVolume)), &H6000&
If optOsc(1).Value Then
For I = 0 To picOsc.ScaleWidth
C = Sgn(Sin(I / picOsc.ScaleWidth * pi * 2 * nFrequency))
If C = 0 Then C = 1
picOsc.PSet (I, ((picOsc.ScaleHeight - 1) \ 2) * (1 - C * nMod * nVolume)), vbGreen
Next
Else
picOsc.Line (0, picOsc.ScaleHeight \ 2)-(0, picOsc.ScaleHeight \ 2)
For I = 0 To picOsc.ScaleWidth
picOsc.Line -(I, (picOsc.ScaleHeight \ 2) * (1 - Sin(I / picOsc.ScaleWidth * pi * 2 * nFrequency) * nMod * nVolume)), vbGreen
Next
End If
Refresh
End Sub

Private Sub Form_Load()
nMod = 1
Set DS = DX7.DirectSoundCreate(vbNullString)
DS.SetCooperativeLevel hWnd, DSSCL_NORMAL
PCM.nFormatTag = WAVE_FORMAT_PCM
PCM.nChannels = 1
PCM.lSamplesPerSec = nSamples
PCM.nBitsPerSample = 8
PCM.nBlockAlign = 1
PCM.lAvgBytesPerSec = PCM.lSamplesPerSec * PCM.nBlockAlign
DSBD.lFlags = DSBCAPS_STATIC
hscFreq_Scroll
End Sub

Private Sub hscFreq_Change()
hscFreq_Scroll
End Sub

Private Sub hscFreq_Scroll()
nFreq = 1 + hscFreq.Value * 22.049! * Log(1 + hscFreq.Value / 1000) / Log(2)
SinBuffer nFreq, hscVol.Value / 1000, optOsc(1).Value
End Sub

Private Sub hscMod_Change()
hscMod_Scroll
End Sub

Private Sub hscMod_Scroll()
If hscMod.Value = 0 Then
tmrMod.Interval = 0
nMod = 1
Else
tmrMod.Interval = 1
End If
End Sub

Private Sub hscVol_Change()
hscVol_Scroll
End Sub

Private Sub hscVol_Scroll()
SinBuffer nFreq, hscVol.Value / 1000, optOsc(1).Value
End Sub

Private Sub optOsc_Click(Index As Integer)
SinBuffer nFreq, hscVol.Value / 1000, optOsc(1).Value
End Sub

Private Sub tmrMod_Timer()
If nModDir >= 0 Then
nMod = nMod + 0.2! / (101 - hscMod.Value)
If nMod > 1 Then nMod = 1: nModDir = -1
Else
nMod = nMod - 0.2! / (101 - hscMod.Value)
If nMod < -1 Then nMod = -1: nModDir = 1
End If
SinBuffer nFreq, hscVol.Value / 1000, optOsc(1).Value
End Sub

Gracias por tu ayuda y saludos,

Carlos Gallego
Saludos,

Carlos Gallego

*** FWH-24.07, xHarbour 1.3.1 Build 20240624, Borland C++7.70, PellesC ***
Cgallegoa
 
Posts: 492
Joined: Sun Oct 16, 2005 3:32 am
Location: Quito - Ecuador

Re: Reemplazo a Tone() y Beep()

Postby Antonio Linares » Thu Nov 26, 2009 2:03 pm

Carlos,

El código tiene su nivel de complejidad porque usa DirectSound, etc y son objetos OLE que hay que investigar, probar, etc.

No te valdría modificar el ejemplo en VB para que le pases los parámetros que necesites y los ejecute, y que llamases a ese EXE desde tu aplicación ? :-)

En relación al Windows 7, como decimos por aqui, para gustos los colores :-) Nosotros llevamos usándolo desde las versiones beta y funciona realmente bien, rápido y muy estable, pero es comprensible que determinadas empresas no puedan ó no quieran plantearse la actualización a él
regards, saludos

Antonio Linares
www.fivetechsoft.com
User avatar
Antonio Linares
Site Admin
 
Posts: 42080
Joined: Thu Oct 06, 2005 5:47 pm
Location: Spain


Return to FiveWin para Harbour/xHarbour

Who is online

Users browsing this forum: No registered users and 50 guests