mac

122
Macros personalizadas en Excel Visual Basic para Aplicaciones VBA V B A 03/08/2010 Adolfo Aparicio 1

Upload: faroviejo

Post on 30-May-2015

3.242 views

Category:

Documents


8 download

TRANSCRIPT

Page 1: Mac

Macros personalizadas en

Excel

Visual Basic para AplicacionesV B AV B A

03/08/2010 Adolfo Aparicio 1

Page 2: Mac

i i Mmi primera MacroEditor de Visual BasicEditor de Visual Basic

Alt+F11Barra de Herramientas: Visual BasicH i t M Edit d V BHerramientas, Macro, Editor de V.B.

Insertar MóduloPrimera macroPrimera macroSub Hola_Mundo()

ActiveCell.Value = "Hola Mundo"End Sub

03/08/2010 Adolfo Aparicio 2

Page 3: Mac

Obj t RObjeto RangoPrograma que deja un valor en una celda yPrograma que deja un valor en una celda y modifica su formato

Sub Saludo()Worksheets("Hoja2") ActivateWorksheets( Hoja2 ).ActivateActiveSheet.Range("C5").Value = "¿Cómo esta usted?"ActiveSheet.Range("C5").Font.Bold = TrueActiveSheet.Range("C5").Font.Color = RGB(255, 0, 0)

End Sub

03/08/2010 Adolfo Aparicio 3

Page 4: Mac

Range y OffsetSub primero()

'Queremos asignar un valor al objeto RangeRange("B10").Value = "Hola"' Otra forma de trabajar es poniendo el objeto WorkSheets que está por encima de RangeWorksheets(1) Range("B11") Value = "¿Qué tal?"Worksheets(1).Range( B11 ).Value ¿Qué tal?' Y aún podemos poner los objetos superiores' Application que hace referencia a la aplicación Excel' Y WorkBooks que se refiere al libro de trabajoApplication.Workbooks(1).Worksheets(1).Range("B12").Value = "Felicidades"Application.Workbooks("Mac01.xls").Worksheets("Hoja1").Range("B13").Value = "América"' Application normalmente no se pone porque todo cuelga de Excel' WorkBooks conviene ponerlo cuando se trabaja con varios libros' WorkSheet conviene si se trabaja con varias hojas, aunque muchas veces no se poneRange("B14") Value = 8 6 'Los decimales con puntoRange( B14 ).Value = 8.6 Los decimales con puntoRange("B15").SelectApplication.ActiveWindow.ActiveCell.Value = "Adios"' Señale con el ratón ActiveWindow y pulse F1 que es la ayudaActiveCell.Offset(1, 0).Value = "Bye"ActiveCell.Offset(2, 0).ActivateActiveCell.Value = "Hasta la vista"ActiveSheet.Range("A1").Offset(17, 1).Value = "100%"

End Sub

03/08/2010 Adolfo Aparicio 4

Page 5: Mac

L E t t With E d WithLa Estructura With - End WithSirve para ejecutar una serie de acciones sobre un mismoSirve para ejecutar una serie de acciones sobre un mismo Objeto, sin tener que repetir toda su jerarquía

Ej.: Propiedades del objeto Range

Sub Escribe()ActiveSheet.Range("C7").Value = "Cta. Resultados"ActiveSheet.Range("C7").Font.Bold = TrueActiveSheet.Range("C7").Font.Color = RGB(0, 255, 0)ct eS eet a ge( C ) o t Co o G (0, 55, 0)

End Sub

Sub Escribe_bis()With ActiveSheet.Range("C7")

.Value = "Cta. Resultados"

.Font.Bold = True

.Font.Color = RGB(0, 255, 0)End With

03/08/2010 Adolfo Aparicio 5

End Sub

Page 6: Mac

Di I tBDim e InputBoxOption Explicit sirve para que nos obliguemos ha definir todas las variablesDim permite definir el tipo de variableSi no se definen las variables se toman como VARIAN que son las que más ocupanInputBox permite capturar datos del usuarioInputBox devuelve siempre datos tipo StringChr(13) es para cambiar de línea

Option ExplicitOption Explicit

Sub Entrar_Valor()Dim Texto As StringT I B ("I d i " & Ch (13) & "P l C ill D10" "E d dTexto = InputBox("Introducir un texto" & Chr(13) & "Para la Casilla D10", "Entrada de

Datos")ActiveSheet.Range("D10").Value = Texto

End Sub

03/08/2010 Adolfo Aparicio 6

Page 7: Mac

Di I tBDim e InputBoxInputBox(Mensaje, Título, Valor por defecto, Posición horizontal, Posición Vertical,

Archivoayuda, Número de contexto para la ayuda)

Sub Entrar_Valor_Bis()'Este procedimiento es igual que el anterior pero no utiliza variablesActiveSheet.Range("D11").Value _= InputBox("Introducir un texto " & Chr(10) & "Para la casilla D11", "Entrada de datos")'El guión bajo permite partir una línea de código demasiado larga. Ver Chr(10)

End Sub

Sub Entrar Valor Tris()Sub Entrar_Valor_Tris()'En este caso se pide al usuario que entre la casilla donde se introducirá el textoDim Casilla As String 'Casilla puede ser por ejemplo D12Dim Texto As StringCasilla = InputBox("En que casilla quiere entrar el valor" "Entrar Casilla")Casilla InputBox( En que casilla quiere entrar el valor , Entrar Casilla )Texto = InputBox("Introducir un texto" & Chr(13) _& "Para la casilla " & Casilla, "Entrada de datos") ‘ Operador de concatenación &ActiveSheet.Range(Casilla).Value = Texto

End Sub

03/08/2010 Adolfo Aparicio 7

Page 8: Mac

Tipos de variablesTipo de datos Tamaño de almacenamiento Intervalo

Byte 1 byte 0 a 255Byte 1 byte 0 a 255

Boolean 2 bytes True o False

Integer 2 bytes -32.768 a 32.767

Long (entero largo) 4 bytes -2.147.483.648 a 2.147.483.647

Si l ( fl t t / i ió -3,402823E38 a -1,401298E-45 para valores negativos;Single (coma flotante/precisión simple) 4 bytes

3,402823E38 a 1,401298E 45 para valores negativos; 1,401298E-45 a 3,402823E38 para valores positivos

Double (coma flotante/precisión doble) 8 bytes

-1,79769313486232E308 a -4,94065645841247E-324 para valores negativos; 4,94065645841247E-324 a 1,79769313486232E308 para valores positivos

Currency (entero a escala) 8 bytes -922.337.203.685.477,5808 a 922.337.203.685.477,5807

Decimal 14 bytes

+/-79.228.162.514.264.337.593.543.950.335 sin punto decimal; +/-7,9228162514264337593543950335 con 28 posiciones a la derecha del signo decimal; el número más pequeño distinto de cero es +/-el número más pequeño distinto de cero es /0,0000000000000000000000000001

Date 8 bytes 1 de enero de 100 a 31 de diciembre de 9999

Object 4 bytes Cualquier referencia a tipo Object

String (longitud variable) 10 bytes + longitud de la cadena Desde 0 a 2.000 millones

String (longitud fija) Longitud de la cadena Desde 1 a 65.400 aproximadamente

Variant (con números) 16 bytes Cualquier valor numérico hasta el intervalo de un tipo Double

Variant (con caracteres) 22 bytes + longitud de cadena El mismo intervalo que para un tipo String de longitud i bl

03/08/2010 Adolfo Aparicio 8

Variant (con caracteres) 22 bytes longitud de cadena variable

Definido por el usuario (utilizando Type)

Número requerido por los elementos

El intervalo de cada elemento es el mismo que el intervalo de su tipo de datos.

Page 9: Mac

Ejercicio 1Ejercicio 1

Crear un libro llamado “Rellena.xls”Programar un procedimiento que nos pregunte en que hoja queremos situarnosNos pregunte en que celda queremos situarnosp g q qNos pregunte lo que queremos escribirEl programa pone lo que hemos dicho y lo pone de color verde y cursiva, sobre fondo rojode co o e de y cu s a, sob e o do ojo

Pista: ActiveCell.Interior.Color=RGB(x,y,z)Ejecute el programa

Primero, dando una sola celdaPrimero, dando una sola celdaSegundo. Cuando pida la celda introduzca un rango para ver como funciona Range

¿Ha usado With – End With?

03/08/2010 Adolfo Aparicio 9

Page 10: Mac

Suma dos númerosVal(Cadena) Convierte la cadena a un valor numérico

Sub Sumar()

Val(Cadena). Convierte la cadena a un valor numéricoStr(Número). Convierte el número a una expresión cadena

CBool, CByte, CCur, CCur, CDate, CDec, CInt, CLng, CSng, CStr, CVar

()'Pide dos números y pone en una celda su suma'Dim Numero1 As Integer'Dim Numero2 As IntegerNumero1 = InputBox("Entrar el primer valor", "Entrada de datos")Numero2 = InputBox("Entrar el segundo valor" "Entrada de datos")Numero2 InputBox( Entrar el segundo valor , Entrada de datos )Worksheets("Hoja1").Activate 'Esto se pone por si estamos en una hoja distinta de la Hoja1ActiveSheet.Range("E10").Value = Numero1 + Numero2

End Sub

Sub Sumar Bis()Sub Sumar_Bis()'Este procedimiento es similar al anterior'En el procedimiento anterior si se mete como variable una palabra, da error.'Por eso en este procedimiento añadimos la función ValDim Numero1 As IntegerDi N 2 A I tDim Numero2 As IntegerNumero1 = Val(InputBox("Entrar el primer valor", "Entrada de datos"))Numero2 = Val(InputBox("Entrar el segundo valor", "Entrada de datos"))ActiveSheet.Range("E11").Value = Numero1 + Numero2

End Sub

03/08/2010 Adolfo Aparicio 10

Page 11: Mac

Ejercicio 2Ejercicio 2El siguiente programa no funciona bienEl á d l tá l d b 4 5 lt 5 5 24 75El área del un rectángulo de base 4,5 y altura 5,5 es 24,75Pero este programa da 24. El problema es que no da ERRORModifique el código del procedimiento para solucionarlo

Sub area()Dim base As IntegerDim altura As IntegerDim superficie As Integer'Los decimales se introducen con coma en un inputbox, y con punto en el códigoaltura = InputBox("Introduzca la altura del rectángulo")base = InputBox("Introduzca la base del rectángulo")superficie = base * alturaMsgBox ("El área del rectángulo es " & superficie)

03/08/2010 Adolfo Aparicio 11

End Sub

Page 12: Mac

P bli P i t C llPublic – Private. CellsPublic. Indica que el procedimiento Sub es accesible para todos los demás procedimientos de todos los módulosPrivate. Indica que el procedimiento Sub es accesible sólo para otros procedimientos del módulo en el que se declaraPor defecto los procedimientos son PublicCells comienza a contar filas y columnas a partir del rangoCells comienza a contar filas y columnas a partir del rango especificado en el objeto Range

Cells(fila,columna)

Private Sub Celda()Cells(12, 3).Value = "Solo " & 2ActiveSheet.Cells(10, 6).Value = "Paris"'La Celda 10,6 es la F10La Celda 10,6 es la F10Range("C13:D14").Value = "Cuadrado"Range(Cells(15, 3), Cells(16, 4)).Value = "Cubo"Range("C17:F20").Cells(2, 1).Value = "Elipse" 'Esto solo pone una elipse

S

03/08/2010 Adolfo Aparicio 12

End Sub

Page 13: Mac

Variables de objetosVariables de objetosVariables de objetosVariables de objetosUna variable objeto sirve para hacer referencia a un objeto esto significaUna variable objeto sirve para hacer referencia a un objeto esto significaUna variable objeto sirve para hacer referencia a un objeto, esto significa Una variable objeto sirve para hacer referencia a un objeto, esto significa que podremos acceder a las propiedades de un objeto e invocar sus que podremos acceder a las propiedades de un objeto e invocar sus métodos a través de la variable en lugar de hacerlo directamente a través métodos a través de la variable en lugar de hacerlo directamente a través del objeto.del objeto.Para declarar una variable objeto se utiliza también la palabra Para declarar una variable objeto se utiliza también la palabra DimDimj pj p

DimDim Var_Objeto Var_Objeto AsAs ObjetoObjetoPor EjemploPor Ejemplo

DimDim R R AsAs RangeRangeDimDim HojaHoja AsAs WorkSheetWorkSheetDimDim Hoja Hoja AsAs WorkSheetWorkSheet

Para asignar un objeto a una variable debe utilizar la instrucción Para asignar un objeto a una variable debe utilizar la instrucción SetSet..Set Set Variable_Objeto = ObjetoVariable_Objeto = Objeto

Por EjemploPor EjemploSetSet R= ActiveSheet.Range("A1:B10")R= ActiveSheet.Range("A1:B10")Set Set Hoja = ActiveSheetHoja = ActiveSheetSetSet Hoja = WorkSheets(1)Hoja = WorkSheets(1)

A veces pude ser interesante desasignar una variable objetoA veces pude ser interesante desasignar una variable objeto

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 1313

A veces pude ser interesante desasignar una variable objetoA veces pude ser interesante desasignar una variable objetoDimDim Var_Objeto = Var_Objeto = NothingNothing

Page 14: Mac

Variables de objetosVariables de objetosVariables de objetosVariables de objetosPosiblemente no se utilice demasiado esta clase de variables (dependerá de las preferencias del programador), pero hay casos en los que no hay más remedio que utilizarlas, por ejemplo en estructuras For Each ... Next como veremos, o cuando sea necesario construir funciones que devuelvan rangos referencias a hojas etc

Sub objeto()Di R A R

construir funciones que devuelvan rangos, referencias a hojas, etc.

Dim R As RangeSet R = ActiveSheet.Range("H21:I22")R.Value = "Roma"R.Font.Bold = TrueR.Font.Color = RGB(0, 255, 100)

End Sub

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 1414

End Sub

Page 15: Mac

E t t C di i lEstructuras CondicionalesIf Condición Then

S té i 1Senténcia1Senténcia2

.

.SenténciaNEnd If

Sub Condicional()ActiveSheet.Range("E14").Value = 0ActiveSheet.Range("E15").Value = 0ct eS eet a ge( 5 ) a ue 0ActiveSheet.Range("E16").Value = 0ActiveSheet.Range("E14").Value = Val(InputBox("Entrar el precio", "Entrar"))'Si el valor de la casilla E14 es mayor que 1000, entonces pedir descuentoIf ActiveSheet.Range("E14").Value > 1000 Then

ActiveSheet.Range("E15").Value = Val(InputBox("Entrar Descuento", "Entrar"))End IfActiveSheet.Range("E16").Value = _ActiveSheet.Range("E14").Value - ActiveSheet.Range("E15")

End Sub

03/08/2010 Adolfo Aparicio 15

End Sub

Page 16: Mac

E t t C di i lEstructuras CondicionalesSub Condicional_Bis()

'Igual que el procedimiento anterior pero ahora usando variablesDim Precio As IntegerDi D t A I tDim Descuento As IntegerPrecio = 0Descuento = 0Precio = Val(InputBox("Entrar el precio", "Entrar"))' Si el valor de la variable precio es mayor que 1000, entonces, pedir descuentoIf Precio > 1000 Then

Descuento = Val(InputBox("Entrar descuento", "Entrar"))End IfActiveSheet.Range("F14").Value = PrecioActiveSheet.Range("F15").Value = DescuentoActiveSheet.Range("F16").Value = Precio - Descuento

End Sub

Sub Condicional2()If ActiveSheet.Range("F14").Value = ActiveSheet.Range("F16").Value Then

ActiveSheet.Range("F14").Font.Color = RGB(0, 0, 255)ActiveSheet.Range("F16").Font.Color = RGB(0, 0, 255)

End IfEnd Sub

03/08/2010 Adolfo Aparicio 16

Page 17: Mac

Estructuras Condicionales. ElElse

Sub Condicional_Else()Dim Precio As SingleDim Descuento As SingleDim Descuento As SinglePrecio = 0Descuento = 0Precio = Val(InputBox("Entrar el precio", "Entrar"))' Si el valor de la variable precio es mayor que 1000 entonces aplicar descuento del 10% Si el valor de la variable precio es mayor que 1000, entonces, aplicar descuento del 10%If Precio > 1000 Then

Descuento = Precio * (10 / 100)ActiveSheet.Range("G13").Value = 0.1

Else ' Sino Aplicar descuento del 5%Else Sino Aplicar descuento del 5%Descuento = Precio * (5 / 100)ActiveSheet.Range("G13").Value = 0.05

End IfActiveSheet.Range("G14").Value = Preciog ( )ActiveSheet.Range("G15").Value = DescuentoActiveSheet.Range("G16").Value = Precio - Descuento

End Sub

03/08/2010 Adolfo Aparicio 17

Page 18: Mac

Estructuras Condicionales.Estructuras Condicionales. Else

Sub Condicional_Else2()'Ponga valores en G10 y en G11.'La macro calcula la diferencia la pone en G12 y asigna colorActiveSheet.Range("G12").Value = ActiveSheet.Range("G10").Value -

ActiveSheet.Range("G11").ValueIf Range("G12").Value < 0 Then

'Si la diferencia es negativa pone color rojoActiveSheet.Range("G12").Font.Color = RGB(255, 0, 0)

Else'En caso contrario pone color azul

ActiveSheet Range("G12") Font Color = RGB(0 0 255)ActiveSheet.Range( G12 ).Font.Color = RGB(0, 0, 255)End If

End Sub

03/08/2010 Adolfo Aparicio 18

Page 19: Mac

El valor NothingAlgunas veces puede que sea necesario desasignar una variable del objeto al cual haceAlgunas veces puede que sea necesario desasignar una variable del objeto al cual hace referencia, en este caso debe igualar la variable al valor Nothing de la forma siguiente.

Set Variable_Objeto = NothingHabitualmente se utiliza Nothing en una estructura condicional para comprobar si la variable objeto está asignada. Observe que si se utiliza una variable objeto a la cual todavía no se le ha

S b bj Bi ()

hecho ninguna asignación el programa dará error y detendrá su ejecución. Es buena práctica hacer este tipo de comprobaciones antes de trabajar con variables objeto.

Sub objeto_Bis()Dim R As RangeSet R = ActiveSheet.Range("E12:F13")R.Value = "Milan"R.Font.Bold = TrueR.Font.Bold TrueSet R = Nothing 'Nothing permite asigna a la variable objeto un valor nulo.

' Esto es útil junto con un IF para verificar si la variable esta asignadaIf R Is Nothing Then

MsgBox Prompt:="La variable Objeto no ha sido asignada", Buttons:=vbOK, _Title:="Error"

ElseR.Value = "Hola"

End IfEnd Sub

03/08/2010 Adolfo Aparicio 19

End Sub

Page 20: Mac

C di i l id dCondicionales anidadasSub Condicional_doble()

Dim a As IntegergDim b As IntegerDim C As Stringa = ActiveSheet.Range("G10").Valueb = ActiveSheet.Range("G11").ValueIf a = b Then

C = "Los valores de G10 y G11 son iguales"Else

If a > b ThenC = "G10 es mayor que G11"

ElseC = "G10 es menor que G11"

End IfEnd IfActiveSheet.Range("G9").Value = C

End Sub

03/08/2010 Adolfo Aparicio 20

Page 21: Mac

El IfElseIfIf condición 1 Then Sub Condicional doble 2()

El procedimiento anterior se puede abreviar con un EsleIf

If condición 1 ThenSentencia 1Sentencia 2

ElseIf condición 2 ThenSentencia 3

Sub Condicional_doble_2()Dim a As IntegerDim b As IntegerDim C As Stringa = ActiveSheet Range("G10") ValueSentencia 3

Sentencia 4ElseIf condición 3 Then

Sentencia 5Sentencia 6

a = ActiveSheet.Range( G10 ).Valueb = ActiveSheet.Range("G11").ValueIf a = b Then

C = "Los valores de G10 y G11 son iguales“'ElseIf abrevia dos condicuonales anidadosSentencia 6

ElseIf condición 4 ThenSentencia 7Sentencia 8

Else

ElseIf abrevia dos condicuonales anidadosElseIf a > b Then

C = "G10 es mayor que G11"Else

C = "G10 es menor que G11"Sentencia 9Sentencia 10

EndIf

qEnd IfActiveSheet.Range("G9").Value = C

End Sub

03/08/2010 Adolfo Aparicio 21

Page 22: Mac

Operador Lógico ANDOperador Lógico ANDp gp gSub YAcero() 'Uso del condicional AND

Dim Producto As String, Cantidad As String, Precio As SingleDim Total As Single, Descuento As Single, Total_Descuento As SinglePrecio = 0 'UCase convierte a mayúsculasProducto = UCase(InputBox("Entrar nombre del Producto", "Entrar"))Precio = Val(InputBox("Entrar Precio", "Entrar"))Cantidad = Val(InputBox("Entrar Cantidad", "Entrar"))Total = Precio * CantidadActiveSheet Range("H10") Value = ProductoActiveSheet.Range( H10 ).Value = ProductoActiveSheet.Range("H11").Value = PrecioActiveSheet.Range("H12").Value = CantidadActiveSheet.Range("H13").Value = Total'Si el Total es mayor que 10000 y el producto es Acero, aplicar descuentoIf Total > 10000 And Producto = "ACERO" Then

Descuento = Val(InputBox("Entrar Descuento", "Entrar"))Total_Descuento = Total * (Descuento / 100)Total = Total - Total_DescuentoActi eSheet Range("H14") Val e Total Desc entoActiveSheet.Range("H14").Value = Total_DescuentoActiveSheet.Range("H15").Value = Total

End IfRange("H12").NumberFormat = "#,##0“'Formato de Celdas Range("H11,H13,H14,H15").NumberFormat = "#,##0.00 $"

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 2222

g ( , , , ) , $End Sub

Page 23: Mac

Operador Lógico OROperador Lógico ORp gp gSub OAcero() ' Condicional OR

Dim Producto As String, Cantidad As Integer, Precio As SingleDim Total As Single, Descuento As Single, Total_Descuento As SingleP i 0Precio = 0'LCase convierte a minúsculasProducto = LCase(InputBox("Entrar Nombre del Producto", "Entrar"))Precio = Val(InputBox("Entrar el Precio", "Entrar"))Cantidad = Val(InputBox("Entrar la Cantidad", "Entrar"))( p ( , ))Total = Precio * CantidadActiveSheet.Range("I10").Value = ProductoActiveSheet.Range("I11").Value = PrecioActiveSheet.Range("I12").Value = Cantidad

i Sh (" 3") l lActiveSheet.Range("I13").Value = Total'si Total es mayor de 10.000 o el producto es Acero, aplicar descuentoIf Total > 10000 Or Producto = “acero" Then

Descuento = Val(InputBox("Entrad Descuento", "Entrar"))Total Descuento = Total * (Descuento / 100)Total_Descuento Total (Descuento / 100)Total = Total - Total_DescuentoActiveSheet.Range("I14").Value = Total_DescuentoActiveSheet.Range("I15").Value = Total

End If

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 2323

End Sub

Page 24: Mac

Operador Lógico NOTOperador Lógico NOTp gp gSub operadorNO()

Dim Precio As IntegerDim Descuento As IntegerDim Descuento As IntegerPrecio = 0Descuento = 0Precio = Val(InputBox("Entrar el Precio", "Entrar"))( p ( , ))' Si el valor de la variable precio NO es menor o igual que

1000,' entonces pedir descuentoIf Not P ecio < 1000 ThenIf Not Precio <= 1000 Then

Descuento = Val(InputBox("Entrar Descuento", "Entrar"))End IfActiveSheet.Range("B19").Value = PrecioActiveSheet.Range( B19 ).Value PrecioActiveSheet.Range("B20").Value = DescuentoActiveSheet.Range("B21").Value = Precio - Descuento

End Sub

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 2424

Page 25: Mac

Tablas de VerdadTablas de VerdadTablas de VerdadTablas de Verdad

A B C NO(A) Y(A;B;C) O(A;B;C)

VERDADERO VERDADERO VERDADERO FALSO VERDADERO VERDADERO

VERDADERO VERDADERO FALSO FALSO FALSO VERDADERO

VERDADERO FALSO VERDADERO FALSO FALSO VERDADERO

VERDADERO FALSO FALSO FALSO FALSO VERDADERO

FALSO VERDADERO VERDADERO VERDADERO FALSO VERDADERO

FALSO VERDADERO FALSO VERDADERO FALSO VERDADERO

FALSO FALSO VERDADERO VERDADERO FALSO VERDADERO

FALSO FALSO FALSO VERDADERO FALSO FALSO

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 2525

FALSO FALSO FALSO VERDADERO FALSO FALSO

Page 26: Mac

CalculadoraMacro que suma resta multiplica o divide los valores de las casillas C19 y C20 dependiendo de si C21 contiene

Sub Calculadora()Dim Signo As String * 1 'Un solo carácter alfanumérico

Macro que suma, resta, multiplica o divide los valores de las casillas C19 y C20 dependiendo de si C21 contiene el signo +, -, x, : El resultado lo deja en C22. Si en C21 no hay ninguno de los signos anteriores en C22 debe dejarse un 0

g gDim Valor1 As Integer, Valor2 As Integer, Total As IntegerValor1 = ActiveSheet.Range("C19").ValueValor2 = ActiveSheet.Range("C20").ValueSigno = ActiveSheet.Range("C21").ValueTotal = 0Total 0If Signo = "+" Then

Total = Valor1 + Valor2End IfIf Signo = "-" Then

Total = Valor1 - Valor2Total = Valor1 - Valor2End IfIf Signo = "x" Then

Total = Valor1 * Valor2End IfIf Signo = ":" ThenIf Signo = : Then

Total = Valor1 / Valor2End IfActiveSheet.Range("C22").Value = Total

End Sub

03/08/2010 Adolfo Aparicio 26

Page 27: Mac

La estructura Select Case

Sub calcula case()

La estructura Select Case da mayor legibilidad al programa anterior

Sub calcula_case()Dim Signo As String * 1Dim Valor1 As Integer, Valor2 As Integer, Total As IntegerValor1 = ActiveSheet.Range("D19").ValueValor2 = ActiveSheet.Range("D20").ValueSigno = ActiveSheet Range("D21") ValueSigno = ActiveSheet.Range("D21").ValueSelect Case Signo

Case "+"Total = Valor1 + Valor2

Case "-"T t l V l 1 V l 2Total = Valor1 - Valor2

Case "x"Total = Valor1 * Valor2

Case “:"Total = Valor1 / Valor2

Case ElseTotal = 0

End SelectActiveSheet.Range("D22").Value = Total

End Sub

03/08/2010 Adolfo Aparicio 27

Page 28: Mac

EjercicioEjercicio

Cree un programa que pregunte laCree un programa que pregunte la fecha de nacimiento, calcule cuantos días han transcurrido hasta eldías han transcurrido hasta el momento actual y diga en qué día de la semana nacióla semana nació.

03/08/2010 Adolfo Aparicio 28

Page 29: Mac

Solución EjercicioSub nacimiento()Sub nacimiento()

Dim dias As Integer, Dsemana As Integer, Factual As Date, d As String'Dsemana es una variable que da un número que indica el día de la semana'dado por la función WEEKDAY, que en Excel es =DIASEM(fecha)Static Fnacimiento As DateStatic Fnacimiento As DateFactual = Date 'Date es la función de VBA equivalente a =HOY()Fnacimiento = FactualFnacimiento = InputBox(Prompt:="Introduzca su fecha de nacimiento", Title:="Formato DD-MM-AAAA",

Default:=Fnacimiento))dias = Factual - FnacimientoDsemana = Application.WorksheetFunction.Weekday(Fnacimiento)Select Case Dsemana

Case 1: d = "Domingo"Case 2: d = "Lunes"Case 3: d = "Martes"Case 4: d = "Miercoles"Case 5: d = "Jueves"Case 6: d = "Viernes"Case 7: d = "Sabado"

End SelectMsgBox Prompt:="Usted nació un " & d & " hace " & dias & " días" & Chr(10), Title:="Esta información es

correcta siempre que hoy sea " & Factual

03/08/2010 Adolfo Aparicio 29

correcta siempre que hoy sea " & FactualEnd Sub

Page 30: Mac

Cada sentencia Case evalúa un rango de valoresCada sentencia Case evalúa un rango de valoresSub Media()

Dim Nota1 As Single, Nota2 As Single, Nota3 As SingleDim califica As String, Media As SingleNota1 = CSng(InputBox("Entrar Nota primera evaluación" "Nota"))Nota1 CSng(InputBox( Entrar Nota primera evaluación , Nota ))Nota2 = CSng(InputBox("Entrar Nota Segunda evaluación", "Nota"))Nota3 = CSng(InputBox("Entrar Nota Tercera evaluación", "Nota"))Media = (Nota1 + Nota2 + Nota3) / 3ActiveSheet.Range("C17").Value = Nota1ActiveSheet Range("D17") Value = Nota2ActiveSheet.Range( D17 ).Value = Nota2ActiveSheet.Range("E17").Value = Nota3ActiveSheet.Range("D18").Value = MediaSelect Case Media

Case Is < 5lifi "S "califica = "Suspenso"

Case 5 To 6.99califica = "Aprobado"

Case 6.99 To 8.99califica = "Notable"

Case Is > 8, 99califica = "Sobresaliente"

End SelectActiveSheet.Range("E18").Value = califica

End Sub

03/08/2010 Adolfo Aparicio 30

Page 31: Mac

Select Case y FiltrosSelect Case y FiltrosSub con_case_y_filtro()

Dim Signo As StringDim Valor1 As Variant, Valor2 As Variant, Total As

SingleDim Continuar As Boolean

If IsEmpty(ActiveSheet.Range("E21")) ThenMsgBox Prompt:="la casilla E21 está vacía",

Title:="ERROR"Continuar = False

End IfDim Continuar As BooleanValor1 = ActiveSheet.Range("E19").ValueValor2 = ActiveSheet.Range("E20").ValueSigno = ActiveSheet.Range("E21").ValueContinuar = True' Si en la casilla E19 no ha n alor n mérico

End IfIf Continuar Then

Select Case SignoCase "+"

Total = Valor1 + Valor2Case " "' Si en la casilla E19 no hay un valor numérico

If Not IsNumeric(ActiveSheet.Range("E19")) ThenMsgBox Prompt:="En E19 no hay ningún valor

numérico", Title:="ERROR"Continuar = False

End If

Case "-"Total = Valor1 - Valor2

Case "x"Total = Valor1 * Valor2

Case "/"1 / 2End If

' Si en la casilla E20 no hay un valor numéricoIf Not IsNumeric(ActiveSheet.Range("E20")) Then

MsgBox Prompt:="En E20 no hay ningún valor numérico", Title:="ERROR"

Continuar = False

Total = Valor1 / Valor2Case Else

Total = 0End SelectActiveSheet.Range("E22").Value = Total

Continuar = FalseEnd If

End IfEnd Sub

03/08/2010 Adolfo Aparicio 31

Page 32: Mac

Lista de Funciones de C b ióComprobación

IsNuméric(Expresión)Comprueba si expresión tiene un valor que se puede interpretar como 'numérico.

IsDate(Expresión)Comprueba si expresión tiene un valor que se puede interpretar como tipo fecha.

IsEmpty(Expresión)Comprueba que expresión tenga algún valor, que se haya inicializado.

I E (E ió )IsError(Expresión)Comprueba si expresión devuelve algún valor de error.

IsArray(Expresión)Comprueba si expresión (una variable) es un array o no.

IsObject(Expresión)IsObject(Expresión)Comprueba si expresión (una variable) representa una variable tipo objeto.

IsNull(Expresión)Comprueba si expresión contiene un valor nulo debido a datos no válidos.

NothinggNo es propiamente una función, sirve para comprobar si una variable objeto esta 'asociada a un objeto antes de hacer cualquier operación con ella. Recuerde que para trabajar con 'una variable objeto antes debe asignarse a uno (mediante la instrucción Set), en caso contrario se producirá un error en el programa cuando utilice el objeto y se detendrá su ejecución.

03/08/2010 Adolfo Aparicio 32

Page 33: Mac

Select Case y FiltroySub con_case_y_filtro_Bis()' En lugar de los tres If de comprobación se puede utilizar el operador OR de la manera siguiente

Dim Signo As StringDim Valor1 As Variant, Valor2 As Variant, Total As SingleDi C ti A B lDim Continuar As BooleanValor1 = ActiveSheet.Range("F19").ValueValor2 = ActiveSheet.Range("F20").ValueSigno = ActiveSheet.Range("F21").ValueContinuar = True' Si en la casilla F19 no hay un valor numéricoIf Not IsNumeric(ActiveSheet.Range("F19")) Or Not IsNumeric(ActiveSheet.Range("F20")) Or

IsEmpty(ActiveSheet.Range("F21")) ThenMsgBox Prompt:="Debe entrar número en F19, F20 y un signo (+,-,x,/) en F21", Title:="ERROR"

ElseS C SSelect Case Signo

Case "+“: Total = Valor1 + Valor2Case "-“: Total = Valor1 - Valor2Case "x“: Total = Valor1 * Valor2Case "/“: Total = Valor1 / Valor2Case Else: Total = 0

End SelectActiveSheet.Range("F22").Value = Total

End IfEnd Sub

03/08/2010 Adolfo Aparicio 33

Page 34: Mac

La función MsgBox (F1)

Muestra un mensaje en un cuadro de diálogo hasta que el usuario pulse un botón. La función devuelve un dato tipo Integer en función del botón pulsado por el usuario. A la hora de invocar está función, se permiten diferentes tipos de botones.botones.

MsgBox( Mensaje, Botones, Título, Archivo de ayuda, contexto)Mensaje: Obligatorio, es el mensaje que se muestra dentro del cuadro de diálogo.Botones: Opcional. Es un número o una suma de números o constantes, que sirve para mostrar determinados botones e iconos dentro del cuadro de diálogosirve para mostrar determinados botones e iconos dentro del cuadro de diálogo. Si se omite este argumento asume valor 0 que corresponde a un único Botón OK.Título : Opcional. Es el texto que se mostrará en la barra del título del cuadro de diálogo.

MsgBox Prompt:="En la casilla A1 no hay ningún valor numérico" Title:="ERROR"MsgBox Prompt:="En la casilla A1 no hay ningún valor numérico", Title:="ERROR"MsgBox Prompt := "La variable Objeto no ha sido asignada", Buttons:=vbOk, Title := "Error"X= MsgBox ("Hola usuario, Ha acabado el proceso", VbOkOnly, "Mensaje")X=MsgBox("Desea Continuar", vbYesNo + vbQuestion, "Opción",,)Mas datos = MsgBox("Otro registro ?" vbYesNo+vbQuestion "Entrada de datos")Mas_datos = MsgBox( Otro registro ? , vbYesNo+vbQuestion, Entrada de datos )MsgBox Prompt:=Texto, Buttons:=vbOKOnly + vbInformation, Title:=TituloMsgBox ("Debe introducir valores numéricos")

03/08/2010 Adolfo Aparicio 34

Page 35: Mac

MsgBoxSub MesajeCaja()

Dim nom As String, Respuesta As Integernom = "Antonio"MsgBox ("Hola " & nom) 'Se pueden poner paréntesis o noMsgBox "Hola " & nomMsgBox "Mire el Título", , "Aqui se puede poner el título que se desee"MsgBox "Observe este texto" & vbCrLf & "que ocupa" & vbCrLf & "tres líneas",, "Título"MsgBox "Mire el icono de" & vbCrLf & "Interrogación", vbQuestion, _"Icono de Interrogación"MsgBox "Otro icono", vbCritical, "Icono Crítico" 'SonidoMsgBox "Otro", vbExclamation, "Icono Exclamación" 'SonidoMsgBox "Otro más", vbInformation, "Icono Información" 'SonidoRespuesta = MsgBox("Observe que al incluir más" & vbCrLf & _"de un botón, en el MsgBox" & vbCrLf & "pongo paréntesis y utilizo" _& vbCrLf & "una variable que recogerá" & vbCrLf & "el botón que hemos pulsado", q g q p , _vbYesNo + vbQuestion, "Dos Botones")MsgBox "La Respuesta ha sido " & Respuesta, , "Respuesta"Respuesta = MsgBox("Tres Botones", vbYesNoCancel + vbInformation, _"Con icono de Información") 'Con paréntesis necesariamenteMsgBox "La Respuesta ha sido " & Respuesta "Respuesta"MsgBox La Respuesta ha sido & Respuesta, , RespuestaRespuesta = MsgBox("Tres Botones pero" & vbCrLf & "el activo es el segundo", _vbAbortRetryIgnore + vbCritical + vbDefaultButton2, "Icono Crítico")MsgBox "La Respuesta ha sido " & Respuesta, , "Respuesta"

End Sub

03/08/2010 Adolfo Aparicio 35

Page 36: Mac

I tBInputBoxVariable = InputBox (mensaje, Titulo, Defecto, Coordenada Horizontal, Coordenada Vertical))Las coordenadas se miden en Twips desde el extremo superior izquierdo de la ventana

1 cm = 566 Twips1 pixel = 15 Twips

Sub InputCaja()Dim Respuesta As StringRespuesta = InputBox("Primera Línea" & vbCrLf & Chr(9) _& "Segunda Línea con Tabulador Chr(9)", "Aquí el Título") 'Chr(10) equivale a vbCrLfRespuesta = InputBox("Haz clic en [Cancel]", "A ver que pasa si se cacela")MsgBox "Al pulsar Calcelar el resultado es = " & Respuesta 'Respuesta nula ""Respuesta = InputBox("Aparece un valor por defecto", "Título", "Aparece esto por defecto")Respuesta = InputBox("Situo la ventana", "1200 Twips a la derecha y 1400 hacia abajo", "coordenadas

1200x1400", 1200, 1400)Respuesta = InputBox("Otra posición", , "1 cm = 566 Twips y 1 pixel = 15 Twips", 50, 75)

E d S b

03/08/2010 Adolfo Aparicio 36

End Sub

Page 37: Mac

La instrucción With (repaso)

Sub OAcero_with()Dim Producto As StringDim Cantidad As IntegerDim Precio As SingleDi T t l A Si l

' Si total mayor que 10.000 o el producto es Acero, aplicar descuento.

If Total > 10000 Or Producto = "Acero" ThenDescuento = Val(InputBox("Entrar Descuento",

"Entrar"))Dim Total As SingleDim Descuento As SingleDim Total_Descuento As SinglePrecio = 0Producto = LCase(InputBox("Entrar Nombre del

Entrar ))Total_Descuento = Total * (Descuento / 100)Total = Total - Total_DescuentoWith ActiveSheet

.Range("J14").Value = Total_Descuento( p (Producto", "Entrar"))

Precio = Val(InputBox("Entrar el precio", "Entrar"))Cantidad = Val(InputBox("Entrar la cantidad", "Entrar"))Total = Precio * CantidadWith ActiveSheet

.Range("J15").Value = TotalEnd With

End IfEnd Sub

With ActiveSheet.Range("J10").Value = Producto.Range("J11").Value = Precio.Range("J12").Value = Cantidad.Range("J13").Value = Totalg ( )

End With

03/08/2010 Adolfo Aparicio 37

Page 38: Mac

E t t titiEstructuras repetitivasSub Media_notas()

Dim Nota As IntegerNota = Val(InputBox("Entrar la Nota 3: ", "Entrar

Nota"))A ti Sh t R ("G19") V l N tDim Media As Single

Media = 0

'Ob t it l i i t

ActiveSheet.Range("G19").Value = NotaMedia = Media + Nota

Nota = Val(InputBox("Entrar la Nota 4: ", "Entrar Nota"))'Observe que este programa repite el siguiente

bloque de sentencias, 5 veces

Nota = Val(InputBox("Entrar la Nota 1: ", "Entrar Nota"))

Nota ))ActiveSheet.Range("G20").Value = NotaMedia = Media + Nota

Nota = Val(InputBox("Entrar la Nota 5: ", "EntrarNota ))ActiveSheet.Range("G17").Value = NotaMedia = Media + Nota

Nota = Val(InputBox("Entrar la Nota 2: ", "Entrar

Nota Val(InputBox( Entrar la Nota 5: , Entrar Nota"))

ActiveSheet.Range("g21").Value = NotaMedia = Media + Nota

( p (Nota"))

ActiveSheet.Range("G18").Value = NotaMedia = Media + Nota

Media = Media / 5ActiveSheet.Range("G22").Value = Media

End Sub

03/08/2010 Adolfo Aparicio 38

Page 39: Mac

B l F N tBucle For … NextSub Totalizar()

Dim i As IntegerDim Total As IntegerDim Valor As IntegerDim Valor As Integer

For i = 1 To 10Valor = Val(InputBox("Entrar el valor " & i, "Entrada"))a o a ( put o ( t a e a o & , t ada ))Total = Total + Valor

Next i

ActiveSheet.Range("C11").Value = TotalEnd Sub

03/08/2010 Adolfo Aparicio 39

Page 40: Mac

Recorrer casillas de una HojaRecorrer casillas de una HojaPropiedad Cells

sirve para referenciar una celda o un rango de celdas p gsegún coordenadas de fila y columna

Sub rellenar() 'Rellena de H16 a H20 con los pares del 2 al 10Dim Fila As Integer, i As Integera s tege , s tegeFila = 16For i = 2 To 10 Step 2

ActiveSheet.Cells(Fila, 8).Value = iFila = Fila + 1 'Esto es un contador

Next iEnd Sub

Sub rellenar_Bis() 'Rellena de H16 a H20 con los pares del 2 al 10, sin contador FilaDim i As IntegergFor i = 16 To 20

ActiveSheet.Cells(i, 9).Value = i * 2 - 30Next i

End Sub

03/08/2010 Adolfo Aparicio 40

Page 41: Mac

Rellenar una serieRellenar una serieLlenar un rango de filas, empezando por una celda, que se debe especificar desde teclado, con una serie de 10 valores correlativos (comenzando por el 1).

Sub serie()Dim Casilla_Inicial As StringDim i As IntegerDi Fil A I C l A IDim Fila As Integer, Columna As IntegerCasilla_Inicial = InputBox("Introducir la casilla Inicial : “ & chr(10) & “Por ejemplo la K10”,

"Casilla Inicial")ActiveSheet.Range(Casilla_Inicial).ActivateFila = ActiveCell RowFila = ActiveCell.RowColumna = ActiveCell.Column'ROW y COLUMN devuelven la fila y la columna de un objeto range.'en este caso se utilizan para obtener la fila y la columna de la casilla activa.For i = 1 To 10For i 1 To 10

ActiveSheet.Cells(Fila, Columna).Value = iFila = Fila + 1

Next iEnd Sub

03/08/2010 Adolfo Aparicio 41

Page 42: Mac

Rellenar una serieRellenar una serieRecuerde que cuando utilizamos Cells como propiedad de un rango (Objeto Range), Cells empieza a contar a partir de la casilla referenciada por Range

Sub serie_Bis()Dim Casilla_Inicial As StringDim i As IntegergDim Fila As Integer, Columna As IntegerCasilla_Inicial = InputBox("Introducir la casilla Inicial : " & chr(10) & “Por ejemplo

la L10”, "Casilla Inicial")ActiveSheet Range(Casilla Inicial) ActivateActiveSheet.Range(Casilla_Inicial).Activate Fila = 1For i = 1 To 10

ActiveSheet.Range(Casilla_Inicial).Cells(Fila, 1).Value = iFila = Fila + 1

Next iEnd Sub

03/08/2010 Adolfo Aparicio 42

Page 43: Mac

Rellenar una serieRellenar una serieUna variante del programa anterior.No se usa Fila, se usa la variable del For

Sub serie_Tris()Dim Casilla_Inicial As StringDim i As IntegerDim i As IntegerDim Fila As Integer, Columna As IntegerCasilla_Inicial = InputBox("Introducir la casilla Inicial : " & chr(10) &

“Por ejemplo la M10”, "Casilla Inicial")ActiveSheet.Range(Casilla_Inicial).Activate‘ Activate (con Range) activa una sola celda. Range("B2").Activate

‘ Para seleccionar un rango de celdas, use el método Select. Range("A1:C3").Select

For i = 1 To 10ActiveSheet.Range(Casilla_Inicial).Cells(i, 1).Value = i

Next iE d S b

03/08/2010 Adolfo Aparicio 43

End Sub

Page 44: Mac

For-Next y CellsFor Next y CellsVolvemos a calcular las notas medias, pero usando la estructura For_Next y la propiedad Cells

Sub Media_notas_Bis()Dim Nota As IntegerDim Media As SingleDim Fila As IntegerMedia = 0For Fila = 1 To 5

Nota = Val(InputBox("Entrar la " & " Nota " & Fila, "Entrar Nota"))( p ( ))ActiveSheet.Range(“N10").Cells(Fila, 1) = Nota'lo de Range(“N10") se pone para marcar la celda de inicio,'si no se pone comienza en A1Media = Media + Nota 'esto es un acumuladoMedia Media Nota esto es un acumulado

Next FilaMedia = Media / 5ActiveSheet.Range(“N10").Cells(6, 1).Value = Media

End Sub

03/08/2010 Adolfo Aparicio 44

End Sub

Page 45: Mac

P i d d Off tPropiedad OffsetEsta propiedad es también muy útil a la hora deEsta propiedad es también muy útil a la hora de recorrer rango.Offset, que significa desplazamiento, es una propiedad del objeto Range y se utiliza parapropiedad del objeto Range y se utiliza para referenciar una casilla situada a n Filas y n Columnas de una casilla dada.Ejemplos:Ejemplos:

ActiveSheet.Range("A1").Offset(2, 2).Value = "Hola“Casilla C3 = Hola, 2 filas y 2 columnas desde A1.

ActiveCell Offset(5 1) Value = "Hola“ActiveCell.Offset(5,1).Value = Hola5 Filas por debajo de la casilla Activa = Hola

ActiveCell.Offset(2,2).ActivateActivar la casilla que está 2 filas y 2 columnas de la activa

03/08/2010 Adolfo Aparicio 45

y

Page 46: Mac

For-Next y Offset.Sin cambiar celda activaSin cambiar celda activa

Recorrer rangos con la propiedad OffSet (desplazamiento)

Sub Media_notas_Tris()Dim Nota As IntegerDim Media As SingleDim Fila As IntegerDim Fila As IntegerMedia = 0ActiveSheet.Range("O10").Activate 'la casilla activa siempre es la mismaFor Fila = 0 To 4

Nota = Val(InputBox("Entrar la " & " Nota " & Fila + 1, "Entrar Nota"))ActiveCell.Offset(Fila, 0).Value = NotaMedia = Media + Nota

Next FilaMedia = Media / 5ActiveCell.Offset(5, 0).Value = Media

End Sub

03/08/2010 Adolfo Aparicio 46

Page 47: Mac

For-Next y Offset.Cambia Celda ActivaCambia Celda Activa

Sub Media_notas_Tetra()Dim Nota As IntegerDim Nota As IntegerDim Media As SingleDim i As IntegerMedia = 0ActiveSheet Range("P10") ActivateActiveSheet.Range( P10 ).ActivateFor i = 1 To 5

Nota = Val(InputBox("Entrar la " & " Nota " & i, "Entrar Nota"))ActiveCell.Value = NotaMedia Media + NotaMedia = Media + Nota'Hacer activa la casilla situada una fila por debajo de la actualActiveCell.Offset(1, 0).Activate

Next iMedia = Media / 5ActiveCell.Value = Media

End Sub

03/08/2010 Adolfo Aparicio 47

Page 48: Mac

D Whil LDo While..LoopEstructura Repetitiva (Hacer Mientras)

La estructura repetitiva FOR se adapta perfectamente a aquellasLa estructura repetitiva FOR se adapta perfectamente a aquellas situaciones en que se sabe previamente el número de veces que se ha de repetir un procesoDo While..Loop es una estructura repetitiva que se repite p p q pmientras se cumpla el criterio

Do While Condición En las sentencias interiores seSentencia1Sentencia2

.

En las sentencias interiores se tiene que producir en algún momento un cambio que haga que la condición deje de .

.Sentencia NLoop

q jcumplirse para así poder salir del bucle.

03/08/2010 Adolfo Aparicio 48

Loop

Page 49: Mac

Sub Registros() Do While Nombre <> ""

Rellenar una Base de Datos'Rellenar los registros de una Base de Datos. Hoja3Dim Nombre As String, Ciudad As StringDim Edad As Integer, Fecha As Date'Activar Hoja3Worksheets("Hoja3").Activate

Ciudad = InputBox("Entre la Ciudad : ", "Ciudad")Edad = Val(InputBox("Entre la Edad : ", "Edad"))Fecha = CDate(InputBox("Entra la Fecha : ", "Fecha"))'Copiar los datos en las casillas correspondientesWith ActiveCell( j )

With ActiveSheet.Range("B4").Value = "Nombre".Range("C4").Value = "Ciudad".Range("D4").Value = "Edad"Range("E4") Value = "Fecha"

.Value = Nombre

.Offset(0, 1).Value = Ciudad

.Offset(0, 2).Value = Edad

.Offset(0, 3).Value = FechaEnd With.Range( E4 ).Value Fecha

End With'Para poner negrita y centrar la cabeceraRange("B4:E4").SelectWith Selection

Font Bold = True

End With'Hacer activa la celda de la fila siguiente a la actualActiveCell.Offset(1, 0).ActivateNombre = InputBox("Entre el Nombre (Return para

Terminar) : ", "Nombre")Loop 'pide nuevos datos mientras nombre no este vacío.Font.Bold = True

.HorizontalAlignment = xlCenterEnd With'Activar casilla B5ActiveSheet.Range("B5").ActivateNombre Inp tBo ("Entre el Nombre (Ret rn para Terminar)

Loop pide nuevos datos mientras nombre no este vacío'Seleccionamos la Base de Datos y la ponemos amarillaApplication.Goto Reference:="R4C2"Selection.CurrentRegion.SelectWith Selection.Interior

ColorIndex = 6Nombre = InputBox("Entre el Nombre (Return para Terminar) : ", "Nombre")

'Mientras la variable Nombre sea diferente a cadena vacía

.ColorIndex = 6

.Pattern = xlSolidEnd With

End Sub

03/08/2010 Adolfo Aparicio 49

Page 50: Mac

Detecta donde nos hemos quedadoDetecta donde nos hemos quedadoSub Registros_Bis()

Dim Nombre As StringDim Ciudad As StringDim Edad As Integer

Do While Nombre <> ""Ciudad = InputBox("Entre la Ciudad : ",

"Ciudad")Edad = Val(InputBox("Entre la Edad : "Dim Edad As Integer

Dim Fecha As DateWorksheets("Hoja3").ActivateActiveSheet.Range("B4").Activate'Buscar la primera celda vacía de la columna

Edad = Val(InputBox( Entre la Edad : , "Edad"))

Fecha = CDate(InputBox("Entra la Fecha : ", "Fecha"))

With ActiveCellBuscar la primera celda vacía de la columna

B y convertirla en activaDo While Not IsEmpty(ActiveCell)

ActiveCell.Offset(1, 0).ActivateLoop

.Value = Nombre

.Offset(0, 1).Value = Ciudad

.Offset(0, 2).Value = Edad

.Offset(0, 3).Value = FechapNombre = InputBox("Entre el Nombre

(Return para Terminar) : ", "Nombre")' Mientras la variable Nombre sea diferente a

cadena vacía

End WithActiveCell.Offset(1, 0).ActivateNombre = InputBox("Entre el Nombre

(Return para Terminar) : ", "Nombre")LoopLoop

End Sub

03/08/2010 Adolfo Aparicio 50

Page 51: Mac

¿Desea introducir más datos ?¿Desea introducir más datos ?Sub Registros_Tris()

Dim Nombre As StringDim Ciudad As StringDim Edad As Integer

Do While Mas_datos = vbYesNombre = InputBox("Entre el Nombre: ",

"Nombre")Ciudad = InputBox("Entre la Ciudad : ", "Ciudad")Dim Edad As Integer

Dim Fecha As DateDim Mas_datos As Integer 'Mas_datos es

una variable de tipo IntegerWorksheets("Hoja3").ActivateA ti Sh t R ("B4") A ti t

p ( , )Edad = Val(InputBox("Entre la Edad : ", "Edad"))Fecha = CDate(InputBox("Entra la Fecha : ",

"Fecha"))With ActiveCell

Value = NombreActiveSheet.Range("B4").Activate'Buscar la primera celda vacía de la

columna B y convertirla en activaDo While Not IsEmpty(ActiveCell)

ActiveCell.Offset(1, 0).Activate

.Value = Nombre

.Offset(0, 1).Value = Ciudad

.Offset(0, 2).Value = Edad

.Offset(0, 3).Value = FechaEnd With

LoopMas_datos = vbYes'es necesaria la línea anterior al bucle

Mas_datos = vbYes, para que cuando se evalúe la

ActiveCell.Offset(1, 0).Activate'Preguntar al usuario si desea entrar otro registroMas_datos = MsgBox("Otro registro ?", vbYesNo +

vbQuestion, "Entrada de datos")Loopevalúe la

'condición por vez primera esta se cumpla y se ejecuten las sentencias de dentro del bucle

LoopEnd Sub

03/08/2010 Adolfo Aparicio 51

Page 52: Mac

E t t D L WhilEstructura Do..Loop WhileEl funcionamiento de esta estructura repetitiva es similar a laEl funcionamiento de esta estructura repetitiva es similar a la anterior salvo que la condición se evalúa al final, la inmediata consecuencia de esto es que las instrucciones del cuerpo del bucle se ejecutaran al menos una vez.Esta estructura es más adecuada para casos como el anterior. Si vamos a entrar datos, al menos uno entraremos, por tanto las instrucciones del cuerpo del bucle se deben ejecutar al menos

l d idi i ituna vez, luego ya decidiremos si se repiten o no.En este caso no es necesario la línea Mas_Datos = vbYes antes de Do para forzar la entrada en el bucle ya que la condición va al finalfinal.

03/08/2010 Adolfo Aparicio 52

Page 53: Mac

D L WhilDo..Loop WhileSub Registros_Tetra()

Dim Nombre As StringDo

Nombre = InputBox("Entre el Nombre: ", "Nombre")Dim Ciudad As StringDim Edad As IntegerDim Fecha As DateDim Mas_datos As Integer'M d t i bl d ti

Ciudad = InputBox("Entre la Ciudad : ", "Ciudad")Edad = Val(InputBox("Entre la Edad : ", "Edad"))Fecha = CDate(InputBox("Entra la Fecha : ", "Fecha"))With ActiveCell

V l N b'Mas_datos es una variable de tipo Integer

Worksheets("Hoja3").ActivateActiveSheet.Range("B4").Activate'Buscar la primera celda vacía de la

.Value = Nombre

.Offset(0, 1).Value = Ciudad

.Offset(0, 2).Value = Edad

.Offset(0, 3).Value = FechaEnd WithBuscar la primera celda vacía de la

columna B y convertirla en activaDo While Not IsEmpty(ActiveCell)

ActiveCell.Offset(1, 0).ActivateLoop

End WithActiveCell.Offset(1, 0).ActivateMas_datos = MsgBox("Otro registro ?", vbYesNo +

vbQuestion, "Entrada de datos")'Mientras Mas datos = vbYesp Mientras Mas_datos vbYes

Loop While Mas_datos = vbYesEnd Sub

03/08/2010 Adolfo Aparicio 53

Page 54: Mac

E t t D L U tilEstructura Do..Loop Until

Hacer Hasta que se cumpla la condiciónHacer.. Hasta que se cumpla la condiciónEs otra estructura que evalúa la condición al final.La interpretación es distinta ya que el bucle se vaLa interpretación es distinta, ya que el bucle se va repitiendo HASTA que se cumple la condición,no MIENTRAS se cumple la condición.pDe las dos estructura use la que más le guste

03/08/2010 Adolfo Aparicio 54

Page 55: Mac

D L U tilDo..Loop UntilSub Registros_Penta()

Di N b A St iDo

N b I tB ("E t l N b " "N b ")Dim Nombre As StringDim Ciudad As StringDim Edad As IntegerDim Fecha As DateDim Mas datos As Integer

Nombre = InputBox("Entre el Nombre: ", "Nombre")Ciudad = InputBox("Entre la Ciudad: ", "Ciudad")Edad = Val(InputBox("Entre la Edad: ", "Edad"))Fecha = CDate(InputBox("Entre la Fecha: ", "Fecha"))With Acti eCellDim Mas_datos As Integer

'Mas_datos es una variable de tipo Integer

Worksheets("Hoja3").ActivateActiveSheet Range("B4") Activate

With ActiveCell.Value = Nombre.Offset(0, 1).Value = Ciudad.Offset(0, 2).Value = EdadOffset(0 3) Value = FechaActiveSheet.Range( B4 ).Activate

'Buscar la primera celda vacía de la columna B y convertirla en activa

Do While Not IsEmpty(ActiveCell)ActiveCell.Offset(1, 0).Activate

.Offset(0, 3).Value = FechaEnd WithActiveCell.Offset(1, 0).ActivateMas_datos = MsgBox("Otro registro ?", vbYesNo +

vbQuestion, "Entrada de datos")Loop

Q , )'Hasta que Mas_Datos sea igual a vbNo

Loop Until Mas_datos = vbNoEnd Sub

03/08/2010 Adolfo Aparicio 55

Page 56: Mac

E t t F E hEstructura For EachEste bucle se utiliza básicamente para ejecutar un grupo de sentencias con los elementos de una colección o una matriz.Rec erde q e na colección es n conj nto de objetos hojas rangosRecuerde que una colección es un conjunto de objetos, hojas, rangos, etc.

‘Para cambiar los nombres de las hojas de un libro de trabajoSub NombraHojas()j ()

'Programa que pregunta el nombre para cada hoja de un libro de trabajo,'si no se pone nombre a la hoja, queda el que tiene.Dim Nuevo_Nombre As StringDim hoja As Worksheet' P d h j d l j t W kSh t' Para cada hoja del conjunto WorkSheetsFor Each hoja In Worksheets

Nuevo_Nombre = InputBox("Nombre de la Hoja : " & hoja.Name, "Nombrar Hojas")If Nuevo_Nombre <> "" Then

hoja.Name = Nuevo Nombrej _End If

Next'** Hoja va referenciando cada una de las hojas del conjunto WorkSheets a cada paso de bucle

End Sub

03/08/2010 Adolfo Aparicio 56

Page 57: Mac

EXIT FOREXIT FOREsta macro es una variante de la anteriorSi se pulsa CANCEL o el nombre de hoja esta vacío “” se sale del bucle con un EXIT FOREXIT FOR.EXIT FOR permite salir de un bucle FOR o FOR EACH, mientras que EXIT DOabandona directamente un bucle DOAdemás nos hemos ahorrado el END IF

Sub NombraHojas2()'Si se pulsa cancelar o no se pone nada en el nombre se sale con el EXIT FORDim Nuevo_Nombre As StringDi h j A W k hDim hoja As WorksheetFor Each hoja In Worksheets

Nuevo_Nombre = InputBox("Nombre de la Hoja : " & hoja.Name, "Nombrar Hojas", hoja.Name)

If Nuevo_Nombre = "" Then Exit For 'EXIT FOR sale del buclehoja.Name = Nuevo_Nombre

NextEnd Sub

03/08/2010 Adolfo Aparicio 57

End Sub

Page 58: Mac

Ll RLlenar un RangoSub Llena_Rango()

Dim R As RangeWorksheets("Hoja1").Activate' Para cada celda del rango N16:P19 de la Hoja1For Each R In ActiveSheet.Range("N16:P19")

R.Value = InputBox("Entrar valor para la celda " & R.Address, "Entrada de valores")

NextE d S b

Se ha declarado una variable tipo Range, este tipo de datos sirve para guardar Rangos de una o más casillas, estas variables pueden luego utilizar todas las propiedades y métodos propios de los Objetos Range

End Sub

propiedades y métodos propios de los Objetos Range.La asignación de las variables que sirven para guardar o referenciar objetos (Range, WorkSheet, etc.) deben inicializarse muchas veces a través de la instrucción SET

03/08/2010 Adolfo Aparicio 58

Page 59: Mac

P di i tProcedimientosEn los programas largos conviene dividir el trabajo en varios procedimientos.Inconvenientes de los procedimientos largos:

grandes bloques de código implican mayor complicación del mismorepetición de sentenciasmayores problemas de seguimiento a la hora de:

depurar erroresampliar funcionalidadesincluir modificaciones

Fil fí d “di id á ”Filosofía de “divide y vencerás”tratar cada problema o tarea de forma más o menos aislada

Para llamar un procedimiento desde otro se utiliza la instrucciónCall Nombre_Procedimiento

Sub P Uno()Sub P_Uno()Sentencias

.Call P_Dos()

.Sentencias

.End Sub

03/08/2010 Adolfo Aparicio 59

Page 60: Mac

CallSub Registros_Hexa()'el mismo procedimiento que Registros_Bis() pero usando una llamada CALL a otro procedimiento'el código que salta casilla hasta que se encuentra

DoNombre = InputBox("Entre el Nombre: ",

"Nombre")Ciudad = InputBox("Entre la Ciudad : " "Ciudad")el código que salta casilla hasta que se encuentra

una vacía se implementa en un procedimiento'llamado, Saltar_Celdas_Llenas.'Para entrar valores se ha sustituido Do While..Loop por Do.. Loop While.

Ciudad = InputBox( Entre la Ciudad : , Ciudad )Edad = Val(InputBox("Entre la Edad : ", "Edad"))fecha = CDate(InputBox("Entra la Fecha : ",

"Fecha"))With ActiveCell

Dim Nombre As StringDim Ciudad As StringDim Edad As IntegerDim fecha As Date

.Value = Nombre

.Offset(0, 1).Value = Ciudad

.Offset(0, 2).Value = Edad

.Offset(0, 3).Value = fechaDim Mas_datos As Integer' Llamada a la función Saltar_Celdas_Llenas, el

programa salta aquí a ejecutar las' instrucciones de este procedimiento y luego

vuelve para continuar la ejecución

End WithActiveCell.Offset(1, 0).ActivateMas_datos = MsgBox("Otro registro ?", vbYesNo

+ vbQuestion, "Entrada de datos")vuelve para continuar la ejecución

' a partir de la instrucción DoCall Saltar_Celdas_Llenas

Loop While Mas_datos = vbYesEnd Sub

03/08/2010 Adolfo Aparicio 60

Page 61: Mac

F ió ll dFunción llamadaSub Saltar_Celdas_Llenas()

Worksheets("Hoja3") ActivateWorksheets( Hoja3 ).ActivateActiveSheet.Range("B4").ActivateDo While Not IsEmpty(ActiveCell)

ActiveCell Offset(1 0) ActivateActiveCell.Offset(1, 0).ActivateLoop

End Sub

Función que salta celdas de una misma columna.Sirve para encontrar la primera celda vacía de la columna

03/08/2010 Adolfo Aparicio 61

Page 62: Mac

P á tPasar parámetrosLos parámetros son el mecanismo por el cual un procedimiento puede pasarle valores a otro y de esta forma condicionar, moldear, etc. las

i j tacciones que ejecuta.El procedimiento llamado gana entonces en flexibilidad. La sintaxis de llamada de un procedimiento es la siguiente:

Call Procedimiento(Parámetro1 Parámetro2 ParámetroN)Call Procedimiento(Parámetro1, Parámetro2,..., ParámetroN)

Los parámetros pueden ser valores o variables.La sintaxis para el procedimiento llamado es la siguiente:

Sub Procedimiento(Parámetro1 as Tipo,..., ParámetroN As Tipo)

Observe que aquí los parámetros son variables que recibirán los valores y evidentemente debe haber coincidencia de tipo.

Por ejemplo, si el primer parámetro es una variable tipo Integer, el primer valor que se le debe pasar al procedimiento cuando se llama también ha de ser de tipo Integer (recuerde que puede ser un valor directamente o una variable).

03/08/2010 Adolfo Aparicio 62

Page 63: Mac

Call Procedimiento(Parámetro1, Parámetro2,..., ParámetroN)

Sub Registros_Septa() DoDim Nombre As StringDim Ciudad As StringDim Edad As IntegerDim fecha As Date

Nombre = InputBox("Entre el Nombre : ", "Nombre")Ciudad = InputBox("Entre la Ciudad : ", "Ciudad")Edad = Val(InputBox("Entre la Edad : ", "Edad"))fecha = CDate(InputBox("Entre la Fecha : ", "Fecha"))

Dim Mas_datos As Integer' Llamada a la función

Saltar_Celdas_Llenas_Bis' Mediante dos parámetros se

comunica al procedimiento llamado en

With ActiveCell.Value = Nombre.Offset(0, 1).Value = Ciudad.Offset(0, 2).Value = Edad

comunica al procedimiento llamado en que hoja y celda comenzar

CallSaltar_Celdas_Llenas_Bis("Hoja3", "B4")

'Los parámetros pueden ser valores o

.Offset(0, 3).Value = fechaEnd WithActiveCell.Offset(1, 0).ActivateMas_datos = MsgBox("Otro registro ?", vbYesNo +

vbQuestion "Entrada de datos")variables vbQuestion, Entrada de datos )Loop While Mas_datos = vbYes

End Sub

03/08/2010 Adolfo Aparicio 63

Page 64: Mac

P di i t á tProcedimiento con parámetrosSub Saltar_Celdas_Llenas_Bis(hoja As String, Casilla_Inicial As String)

'los parámetros son variables que recibirán los valores'debe haber coincidencia de tipos.Worksheets(hoja).ActivateActiveSheet.Range(Casilla_Inicial).ActivateDo While Not IsEmpty(ActiveCell)

ActiveCell.Offset(1, 0).ActivateLoop

End Sub

Sirve para Saltar celdas llenas de una columna hasta encontrar una vacía que se convierte en activaParámetros :

Hoja : Hoja donde está el rango a saltar.Casilla_Inicial : Casilla Inicial de la columna

Gracias a los parámetros, sirve para recorrer cualquier rango en cualquier hoja.

03/08/2010 Adolfo Aparicio 64

Page 65: Mac

Los parámetros pueden ser valores o variablesSub Registros Octa()Sub Registros_Octa()

Dim Nombre As String, Ciudad As String, Edad As Integer, fecha As Date, Mas_datos As Integer' Al procedimiento Saltar_Celdas_Llenas_Bis se le pueden pasar valores como en el caso anterior, o variables como en este.

'******************* novedad **********************Dim hoja As StringDim Casilla Inicial As StringDim Casilla_Inicial As Stringhoja = InputBox("En que hoja está la base de datos : ", "Entrar Nombre de Hoja")Casilla_Inicial = InputBox("En que casilla comienza la base de datos", "Casilla Inicial")' Observe que los parámetros son dos variables cuyo valor se ha entrado desde teclado en' las dos instrucciones InputBox anteriores.C ll S lt C ld Ll Bi (h j C ill I i i l)Call Saltar_Celdas_Llenas_Bis(hoja, Casilla_Inicial)

'******************* novedad **********************Do

Nombre = InputBox("Entre el Nombre : ", "Nombre")Ciudad = InputBox("Entre la Ciudad : ", "Ciudad")Edad = Val(InputBox("Entre la Edad : ", "Edad"))fecha = CDate(InputBox("Entre la Fecha : ", "Fecha"))With ActiveCell

.Value = Nombre

.Offset(0, 1).Value = Ciudad

.Offset(0, 2).Value = Edad

.Offset(0, 3).Value = fechaEnd WithActiveCell.Offset(1, 0).ActivateMas_datos = MsgBox("Otro registro ?", vbYesNo + vbQuestion, "Entrada de datos")

03/08/2010 Adolfo Aparicio 65

g ( g )Loop While Mas_datos = vbYes

End Sub

Page 66: Mac

Variables Locales y variables GlobalesEl ámbito de una variable declarada dentro de una función es la propia funciónEl ámbito de una variable declarada dentro de una función es la propia función.Es decir, no podrá utilizarse fuera de dicha función.Así, el siguiente programa que debería sumar las cinco filas siguientes a partir de la casilla activa y guardar el resultado en la sexta es incorrecto.

Sub Hacer()..

Call Sumar Cinco Siguientes

Sub Sumar_Cinco_Siguientes()Dim i As IntegerDim Suma As SingleSuma=0Call Sumar_Cinco_Siguientes

ActiveCell.Offset(6,0).Value = Suma..

End Sub

Suma=0For i=1 To 5

Suma = Suma+ActiveCell.Offset(i,0).ValueNext i

End SubEnd Sub End Sub

Es incorrecto porque tanto las variable i como la variable Suma están declaradas dentro del procedimiento Sumar_Cinco_Siguientes consecuentemente, su ámbito de acción es este procedimiento.Por tanto, la instrucción ActiveCell.Offset(6,0).Value = Suma del procedimiento Hacer, generaría un error ( ) p g(con Option Explicit activado) ya que la variable Suma no está declarada dentro de él.Si piensa en declarar la variable Suma dentro del procedimiento Hacer, no solucionará nada porque esta será local a dicho procedimiento, en este caso tendría dos variables llamadas Suma pero cada una de ellas local a su propio procedimiento y consecuentemente con el ámbito de acción restringido a ellos.

03/08/2010 Adolfo Aparicio 66

Page 67: Mac

Variables GlobalesVariables GlobalesUna solución seria declarar “suma” como variable global.Una variable global se declara fuera de todos los procedimientos y es reconocida por todos los procedimientos del mód loreconocida por todos los procedimientos del módulo.

Option ExplicitDi S A Si l ‘S i bl l b l id t d l di i t d l ód lDim Suma As Single ‘Suma es una variable global reconocida por todos los procedimientos del módulo

Sub Hacer_Bis().

Call Sumar_Cinco_Siguientes_BisActiveCell.Offset(6,0).Value = SumaActiveCell.Offset(6,0).Value Suma

.End Sub

Sub Sumar_Cinco_Siguientes_Bis()Dim i As IntegerDim i As IntegerSuma=0For i=1 To 5

Suma = Suma+ActiveCell.Offset(i,0).ValueNext i

E d S b

03/08/2010 Adolfo Aparicio 67

End Sub

Page 68: Mac

Pasar variables como á tparámetros

La variable parámetro S (a la que se ha cambiado el nombre adrede) de Sumar Cinco Siguientes Tris es la variable Suma declarada enSumar_Cinco_Siguientes_Tris es la variable Suma declarada en Hacer_Tris.Funcionará porque en Visual Basic, a menos que se indique lo contrario, el paso de parámetros es por referencia.

Sub Hacer_Tris()Dim Suma As Single

Sub Sumar_Cinco_Siguientes_Tris(S As Single)Dim i As IntegerSuma=0.

.‘ Llamada a la función Sumar_Cinco_Siguientes

pasándole la variable SumaCall Sumar Cinco Siguientes Tris(Suma)

Suma=0For i=1 To 5

S = S+ActiveCell.Offset(i,0).ValueNext i

End Sub_ _ g _ ( )ActiveCell.Offset(6,0).Value = Suma

.

.End Sub

End Sub

03/08/2010 Adolfo Aparicio 68

Page 69: Mac

Paso por referencia y paso por valorPaso por referencia y paso por valorEl paso por valor significa que la variable parámetro del procedimiento recibe el valor de la variable (o directamente el valor) de su parámetro correspondiente de la instrucción de llamada y en el paso por referencia, la variable parámetrode la instrucción de llamada y en el paso por referencia, la variable parámetro del procedimiento es la misma que su parámetro correspondiente de la instrucción de llamada, es decir, la declarada en el procedimiento desde el que se hace la llamada.Por defecto, y siempre que en la instrucción de llamada se utilicen variables, lasPor defecto, y siempre que en la instrucción de llamada se utilicen variables, las llamadas son por referencia.Si desea que el paso de parámetros sea por valor, debe anteponer a la variable parámetro la palabra reservada ByVal

Sub Saltar_Celdas_Llenas(ByVal Hoja As String, ByVal Casilla_Inicial As String)

Aunque lo elegante y efectivo por razones de memoria seria pasar siempre que sea posible por valor, es poco habitual que así se haga en Visual Basic, seguramente por comodidadseguramente por comodidad.Como suponemos que hará como la mayoría, es decir, pasar por referencia, tenga cuidado con los (indeseables) efectos laterales.

03/08/2010 Adolfo Aparicio 69

Page 70: Mac

Ef t L t lEfecto LateralEste programa no funciona bienp gEn la Hoja4 disponemos de 5 valores en cada una de las tres columnas B,C,D, y deseamos sumarlosDebería sumar los cinco valores de cada columna y poner su suma justo bajo ellosEl mal funcionamiento se debe a que la variable Fila pasa al procedimiento llamado, como variable y no como valor, pese a que se cambia el nombre por F, sigue siendo la misma

Sub Efecto_Lateral()Dim Fila As Integer

Sub Recorrer_Sumar(F As Integer, C As Integer, Q As Integer)Dim i As IntegerDim Fila As Integer

Worksheets("Hoja4").ActivateFila = 5Call Recorrer_Sumar(Fila, 2, 5) ' Columna BCall Recorrer_Sumar(Fila, 3, 5) ' Columna CCall Recorrer Sumar(Fila 4 5) ' Columna D

Dim i As IntegerDim Total As IntegerTotal = 0For i = 1 To Q

Total = Total + ActiveSheet.Cells(F, C).ValueF = F + 1 ' OJO con esta asignación recuerde que F es la variable FilaCall Recorrer_Sumar(Fila, 4, 5) ' Columna D

End SubF = F + 1 ' OJO con esta asignación, recuerde que F es la variable Fila

declarada en el procedimiento Efecto_LateralNext iActiveSheet.Cells(F, C) = Total

End Sub

03/08/2010 Adolfo Aparicio 70

Page 71: Mac

B V lByValSe corrige añadiendo ByVal a la variable, lo que hace que pase como valor.g y q q p

Sub Efecto_Lateral_bis() 'Este procedimiento es igual al Efecto_Lateral

Sub Recorrer_Sumar_bis(ByVal F As Integer, C As Integer, Q As Integer)

'con la salvedad de que en este se llama a Recorrer_Sumar_bis

Dim Fila As IntegerWorksheets("Hoja4").ActivateFila = 5

'Este sub es idéntico al anterior salvo porque en la variable F hemos añadido ByVal,'que transfiere el parámetro como valor y no como variable

Dim i As IntegerDim Total As IntegerFila = 5

Call Recorrer_Sumar_bis(Fila, 2, 5) ' Columna B

Call Recorrer_Sumar_bis(Fila, 3, 5) ' Columna C

Dim Total As IntegerTotal = 0For i = 1 To Q

Total = Total + ActiveSheet.Cells(F, C).ValueF = F + 1

Call Recorrer_Sumar_bis(Fila, 4, 5) ' Columna DEnd Sub

F F + 1Next iActiveSheet.Cells(F, C) = Total

End Sub

03/08/2010 Adolfo Aparicio 71

Page 72: Mac

F iFuncionesLas funciones no ejecutan acciones simplemente dan como resultadoLas funciones no ejecutan acciones, simplemente dan como resultado un valorLas variables de la función se introducen como argumentos

Function Area Cuadrado(x y)Function Area_Cuadrado(x, y)Area_Cuadrado = x * y

End FunctionEnd Function

En la categoría de Funciones “Definidas por el usuario” encontrará esta función que podrá aplicar normalmente a la hoja de cálculo.También se puede usar esta función llamándola desde un procedimiento o desde otra función.

03/08/2010 Adolfo Aparicio 72

Page 73: Mac

F ió ll d S bFunción llamada por un SubUna función puede ser llamada por un procedimiento u otra función.Las funciones tienen tipo (esta es de tipo integer) ya que devuelven un valor

Sub Llama_suma() 'Procedimiento que llama a una función de varias formas. Ver distintas formas.Dim x As IntegerDim n1 As Integer, n2 As Integerx = Sumardos(5, 5)( )n1 = Val(InputBox("Entrar un número : ", "Entrada"))n2 = Val(InputBox("Entrar otro número : ", "Entrada"))x = Sumardos(n1, n2)ActiveCell.Value = Sumardos(ActiveSheet.Range("K10").Value, ActiveSheet.Range("K11").Value)x = Sumardos(5, 4) + Sumardos(n1, n2)

End SubFunction Sumardos(V1 As Integer, V2 As Integer) As Integer

Dim Total As IntegerTotal = V1 + V2Sumardos = Total

End Function

03/08/2010 Adolfo Aparicio 73

Page 74: Mac

EjercicioEjercicio

Cree una función que calcule elCree una función que calcule el factorial de un númeroPor ejemploPor ejemplo. Factorial(5)=5x4x3x2x1=120A i t f ióAunque ya existe una función en Excel que calcula el factorial:=FACT(numero)

03/08/2010 Adolfo Aparicio 74

Page 75: Mac

Función FactorialFunción FactorialFunction factori(n As Long) 'FUNCIÓN que calcula el factorial de un número

Dim F As LongDim i As LonggF = 1For i = n To 1 Step -1

F = F * iNextNextfactori = F

End Function

Function Factorial(ByVal n As Integer)' Un buen ejemplo del uso de ByVal para transferir variables' Si no se pusiera en este caso no calcularía bienn = n - 1If n = 0 Then

Factorial = 1Exit Function

End IfFactorial = Factorial(n) * (n + 1)

03/08/2010 Adolfo Aparicio 75

( ) ( )End Function

Page 76: Mac

Función que detecta Celda V íVacía

Sub Detecta_Vacia()Dim Casilla As StringWorksheets("Hoja4").ActivateCasilla = Casilla_Vacia("B5") 'Llama a la función Casilla_VaciaMsgBox Prompt:=Casilla, Title:="La primera celda vacía"

End Sub

Función Casilla_Vacia de Tipo StringSirve para Recorrer las filas de una columna hasta encontrar una vacía.Parámetros :Casilla_Inicio : Casilla donde debe empezar a buscar.D l t i ti l f i d l i illDevuelve un string que contiene la referencia de la primera casilla

Function Casilla_Vacia(Casilla_Inicio As String) As StringActiveSheet.Range(Casilla_Inicio).ActivateDo While Not IsEmpty(ActiveCell)

ActiveCell.Offset(1, 0).ActivateLoopCasilla_Vacia = ActiveCell.Address

03/08/2010 Adolfo Aparicio 76

End Function

Page 77: Mac

F ió B V lFunción que Busca un ValorSub Busca()

Dim Casilla As String, Valor As Integerg gWorksheets("Hoja4").ActivateValor = CInt(InputBox("Valor buscado: ", "Entrar Datos"))Casilla = Buscar_Valor("C5", Valor) ‘Llama a la función Buscar_ValorIf Casilla = "" Then ' Si valor no encontrado

MsgBox ("NO se ha encontrado el valor buscado")Else 'Valor encontrado

MsgBox ("El primer " & Valor & " esta en la celda: " & Casilla)End If

End Sub

Función que devuelve la dirección de la primera celda vacía de un rango.La f nción es de tipo String a q e de el e la casilla en la forma "FilaCol mna " porLa función es de tipo String ya que devuelve la casilla en la forma "FilaColumna ", por ejemplo "A10".Utilizaremos la propiedad Address del objeto range, esta propiedad devuelve un string que contiene la referencia "FilaColumna" de una casilla o rango de casillas.En el caso de un rango devuelve, "FilaColumna_Inicial:FilaColumna_Final", por ejemplo "A1 C10"

03/08/2010 Adolfo Aparicio 77

"A1:C10"

Page 78: Mac

Función que Busca un ValorqFunción Buscar de Tipo StringSirve para: Recorrer las filas de una columna hasta encontrar el valor buscado o una de vacía.Parámetros :

Casilla_Inicial: Casilla donde debe empezar a buscarValor Buscado: Valor que se debe encontrarValor_Buscado: Valor que se debe encontrar

Devuelve: Un string que contiene la referencia de la casilla donde se ha encontrado el valorTambién puede devolver "" en caso que el valor buscado no esté

Function Buscar_Valor(Casilla_Inicial As String, Valor_Buscado As Integer) As StringActiveSheet.Range(Casilla_Inicial).Activate' Mientras casilla no vacía Y valor de casilla diferente al buscadoDo While Not IsEmpty(ActiveCell) And ActiveCell.Value <> Valor_Buscado

ActiveCell.Offset(1, 0).ActivateLLoop' Si la casilla donde se ha detenido la búsqueda NO ESTÁ VACÍA es que se ha encontrado' el valorIf Not IsEmpty(ActiveCell) Then

B scar Valor Acti eCell Address ' De ol er la casilla donde se ha encontrado el alorBuscar_Valor = ActiveCell.Address ' Devolver la casilla donde se ha encontrado el valorElse ' La casilla está vacía, NO se ha encontrado el valor buscado

Buscar_Valor = "" ' Devolver una cadena vacíaEnd If

End Function

03/08/2010 Adolfo Aparicio 78

End Function

Page 79: Mac

Busca Valor por filas y lcolumnasProcedimiento idéntico a Buscar() pero que llama a la función Buscar Valor Bis que busca por filas y columnasBuscar_Valor_Bis que busca por filas y columnas

Sub Busca_Bis()Dim Casilla As StringDim Valor As IntegerWorksheets("Hoja4").ActivateValor = CInt(InputBox("Valor buscado: ", "Entrar Datos"))C ill B V l Bi ("B5" V l ) 'V l f ió B V l BiCasilla = Buscar_Valor_Bis("B5", Valor) 'Ver la función Buscar_Valor_Bis' Si valor no encontradoIf Casilla = "" Then

MsgBox ("NO se ha encontrado el valor buscado")Else 'Valor encontradoElse 'Valor encontrado

MsgBox ("El primer " & Valor & " esta en la celda: " & Casilla)End If

End Sub

03/08/2010 Adolfo Aparicio 79

Page 80: Mac

Busca Valor por filas y columnasFunction Buscar_Valor_Bis(Casilla_Inicial As String, Valor_Buscado As Integer) As String

Dim Incremento_Columna As IntegerDim Continuar As Boolean

If Not IsEmpty(ActiveCell.Offset(0, Incremento_Columna)) ThenContinuar = False

Else ' La casilla está vacía, no se ha encontrado el valorActiveCell.Offset(1, 0).Activate ' Saltar a una nueva fila

ActiveSheet.Range(Casilla_Inicial).ActivateContinuar = TrueDo While Continuar

Incremento_Columna = 0' Buscar el valor por las columnas hasta

If IsEmpty(ActiveCell) Then ' Si la casilla de la nueva fila está vacía

Continuar = False ' Parar la búsqueda, no hay más casilla a recorrer

End If Buscar el valor por las columnas hasta encontrarlo o encontrar una celda vacía.

Do While Not IsEmpty(ActiveCell.Offset(0, Incremento_Columna)) And _

ActiveCell.Offset(0, Incremento Columna) Value <> Valor Buscado

End IfLoop' Si la casilla donde se ha detenido la búsqueda NO ESTÁ

VACÍA es que se ha encontrado el valor.If Not IsEmpty(ActiveCell) ThenIncremento_Columna).Value <> Valor_Buscado

' Siguiente columnaIncremento_Columna =

Incremento_Columna + 1Loop

If Not IsEmpty(ActiveCell) ThenBuscar_Valor_Bis = ActiveCell(0,

Incremento_Columna).Address ' Devolver la casilla donde ' se ha encontrado el valor

Else ' La casilla está vacía, NO se ha encontrado el valor b scado' Si no está vacía la casilla entonces parar la

búsqueda, se ha encontrado el valorbuscado

Buscar_Valor_Bis = "" ' Devolver una cadema vacíaEnd If

End Function

03/08/2010 Adolfo Aparicio 80

Page 81: Mac

L lá l P i tLa cláusula PrivatePuede anteponer la cláusula private a todos los procedimientos y funciones que sean llamados sólo desde el mismo módulollamados sólo desde el mismo módulo.Es una forma de ahorrar memoria y hacer que el programa corra un poco más rápido.Si necesita llamar un procedimiento o función desde otro módulo, nunca debe precederlo por la cláusula private

' Módulo 1Sub General

' Módulo 2Sub Procedimiento de modulo2Sub General

....End Sub

P i S b P i d

Sub Procedimiento_de_modulo2‘ Esto es correcto. Llama al procedimiento General definido en Módulo1

Call General' Esto no es correcto. Llama al procedimiento Privado definido en Módulo 1, este' procedimiento va precedido pro la cláusula Private, por tanto sólo puede ser llamado

Private Sub Privado....

End Sub

' desde procedimientos de su propio módulo

Call PrivadoEnd Sub

03/08/2010 Adolfo Aparicio 81

Page 82: Mac

EjercicioEjercicio

Programe una macro que proporcionePrograme una macro que proporcione las 4!=24 combinaciones de las cuatro letras ABCDcuatro letras ABCD

03/08/2010 Adolfo Aparicio 82

Page 83: Mac

Permutaciones de ABCDPermutaciones de ABCDSub permuta()Dim i As Byte, j As Byte, k As Byte, l As ByteDim a() As Byte, mensaje As String

Function palabra(i As Byte, j As Byte, k As Byte, l As Byte) As String

Di l t A St i * 1For i = 1 To 4For j = 1 To 4

For k = 1 To 4For l = 1 To 4

Dim letra As String * 1Dim n As Byte, x As ByteDim a(1 To 4) As Bytea(1) = i: a(2) = j: a(3) = k: a(4) = lFor l = 1 To 4

If i = j Or i = k Or i = l Or j = k Or j = l Or k = l ThenElse

mensaje = mensaje & palabra(i, j, k, l) & vbCrLf

( ) ( ) j ( ) ( )For n = 1 To 4

x = a(n)Select Case x

Case 1: letra = "A"Exit ForEnd If

Next lNext k

Case 1: letra = "A"Case 2: letra = "B"Case 3: letra = "C"Case 4: letra = "D"Next k

Next jNext iMsgBox mensaje

End Selectpalabra = palabra & letra

Next nEnd Function

03/08/2010 Adolfo Aparicio 83

End Sub End Function

Page 84: Mac

Permutaciones de ABCD con RNDSub permuta_bis()'Permutaciones de ABCD 4!=24

For j = 1 To i - 1If a(i) = a(j) Then i = i - 1: Exit For

Dim i As Byte, j As Byte, n As ByteDim a(1 To 4) As StringDim b(1 To 24) As StringDim frase As String

Next jNext ib(n) = ""For i = 1 To 4Dim frase As String

Dim x As SingleRandomizeFor n = 1 To 24

F i 1 T 4

For i 1 To 4b(n) = b(n) & a(i)

Next iFor j = 1 To n - 1

If b(j) b( ) Th 1 E it FFor i = 1 To 4x = RndSelect Case x

Case Is < 0.25: a(i) = "A"

If b(j) = b(n) Then n = n - 1: Exit ForNext j

Next nFor n = 1 To 24( )

Case 0.25 To 0.5: a(i) = "B"Case 0.5 To 0.75: a(i) = "C"Case Is > 0.75: a(i) = "D"

End Select

frase = frase & b(n) & vbCrLfNext nMsgBox fraseEnd Sub

03/08/2010 Adolfo Aparicio 84

End Select End Sub

Page 85: Mac

I t E t ód lImportar y Exportar módulosCiertos procedimientos que pueden ser utilizados en multitud de p q pocasiones, seria interesante tenerlos disponibles en cualquiera de las hojas que confeccionemos.Podría pensar en utilizar las opciones de copiar y pegar para pasarprocedimientos de una hoja a otra es un método totalmente válido yprocedimientos de una hoja a otra, es un método totalmente válido y efectivo, pero existe otro método más "profesional“.Consiste en guardar los procedimientos de un módulo aparte y exportarlo a un archivo .BAS que es independiente de cualquier hoja de cálculode cálculo.Luego, cuando en una nueva hoja necesite estas funciones, solo deberá importar este archivo para incorporarlo.Consejo: Aproveche las ventajas que proporciona la programación modularmodular.Consejo: agrupe todas las funciones que usted considere de utilización general en uno o dos módulos y luego utilice las opciones de importación y exportación para incorporarlos a sus programas.

03/08/2010 Adolfo Aparicio 85

Page 86: Mac

I t E t ód lImportar y Exportar módulosExportar un módulo. Guardar un módulo en un archivop

Abra la hoja donde tiene los procedimientos que desea exportar1. Pase al editor de Visual Basic y active el módulo a exportar.2. Active la opción de la barra de menús Archivo/ Exportar archivo. Aparece un cuadro de diálogo.3 En cuadro de edición Nombre de Archivo teclee el nombre para el archivo donde se3. En cuadro de edición Nombre de Archivo, teclee el nombre para el archivo donde se guardará el módulo, por ejemplo "General.Bas", observe que .BAS es la extensión de estos archivos.4. Pulse sobre el botón Guardar.

Importar un móduloImportar un móduloCierre todos los archivos de Excel y abra uno nuevo.1. Active el editor Visual Basic.2. Active opción de la barra de menús Archivo/ Importar Archivo. Aparece un cuadro de diálogo.3 Seleccione en la lista Buscar en: la carpeta donde tiene ubicado el archivo a importar3. Seleccione en la lista Buscar en: la carpeta donde tiene ubicado el archivo a importar4. Una vez localizada la carpeta, seleccione el archivo a importar (General.Bas en el ejemplo) y pulse sobre Abrir.Observe como en la ventana de proyecto se ha incorporado un nuevo módulo que contiene todos los procedimientos y funciones del archivo importado.

03/08/2010 Adolfo Aparicio 86

Page 87: Mac

L b d dLa grabadora de macrosMicrosoft Excel lleva incluida una utilidad que sirve para registrar acciones que se llevan a

b lib d t b j i t l f dcabo en un libro de trabajo y registrarlas en forma de macro.Podemos aprovechar esta utilidad para generar código engorroso por su sintaxis un tanto complicada de recordar, además de ahorrar tiempo.Casi siempre después deberemos modificarlo para adaptarlo a nuestros programas

03/08/2010 Adolfo Aparicio 87

Page 88: Mac

Macro realizada con Grabadora y alguna modificación

Sub Poner_Bordes()W k h t ("H j 4") A ti t

With Selection.Borders(xlEdgeRight) ' Borde derechoLi St l lC tiWorksheets("Hoja4").Activate

' Seleccionar el rango B5:D10Range("B5:D10").Select' No hay borde diagonal hacia abajoSelection.Borders(xlDiagonalDown).LineStyle = xlNone

.LineStyle = xlContinuous

.Weight = xlMedium

.ColorIndex = xlAutomaticEnd With

' Bordes verticales interiores de la selección' No hay borde diagonal hacia arribaSelection.Borders(xlDiagonalUp).LineStyle = xlNone' Borde izquierdo de la seleccónWith Selection.Borders(xlEdgeLeft)

.LineStyle = xlContinuous 'Estilo de línea continuo

With Selection.Borders(xlInsideVertical).LineStyle = xlContinuous.Weight = xlThin ' Ancho Simple..ColorIndex = xlAutomatic

End With.Weight = xlMedium ' Ancho de línea Medio.ColorIndex = xlAutomatic ' Color de línea automático

(negro)End With' Borde superior de la selección

' No hay bordes horiontales interiores en la selecciónSelection.Borders(xlInsideHorizontal).LineStyle = xlNone' Seleccionar rango B9:D9Range("B9:D9").Select' No hay borde diagonal hacia arriba

With Selection.Borders(xlEdgeTop).LineStyle = xlContinuous.Weight = xlMedium.ColorIndex = xlAutomatic

End With

y gSelection.Borders(xlDiagonalDown).LineStyle = xlNone' No hay borde diagonal hacia arribaSelection.Borders(xlDiagonalUp).LineStyle = xlNone' Borde inferior de la selecciónWith Selection.Borders(xlEdgeBottom) ' Doble línea

' Borde inferior de la selecciónWith Selection.Borders(xlEdgeBottom)

.LineStyle = xlContinuous

.Weight = xlMedium

.ColorIndex = xlAutomatic

With Selection.Borders(xlEdgeBottom) Doble línea.LineStyle = xlDouble.Weight = xlThick.ColorIndex = xlAutomatic

End WithRange("A1") Select

03/08/2010 Adolfo Aparicio 88

End WithRange( A1 ).Select

End Sub

Page 89: Mac

Macro con Grabadora y li dgeneralizada

Fichero decora.basPodemos crear una Macro con la Grabadora y luego perfeccionarla y darle carácter más general introduciendo parámetros.Esta macro pone bordes a una tabla, en la primera fila pone doble línea.El primer parámetro es el número de hoja (no el nombre), y el segundo la casilla p p j ( ), y ginicialLa macro se encargará de seleccionar todas las casillas adyacentes y de buscar la primera fila.En esta macro además se han incluido funcionalidades como borrar los f t t d li l lí j t l h d l l tformatos antes de aplicar las líneas, ajustar el ancho de las columnas, etc.Observe la propiedad CurrentRegion del objeto Range, esta propiedad devuelve el rango de las casillas llenas adyacentes a una dada.Por ejemplo imagine una hoja con el rango A1:B10 lleno de valores, la instruccióninstrucción

ActiveSheet.Range("A1").CurrentRegion.SelectSeleccionaria el rango correspondiente a A1:B10

03/08/2010 Adolfo Aparicio 89

Page 90: Mac

Insertar funciones en una Hojaj'va pidiendo números y los va colocando en las celdas de la columna B partir de B12'al final coloca la función =SUMA para sumar los valores introducidos y la función =PROMEDIO

' Establecer la casilla inicial del rango a sumarCasilla_Inicial = "B12"' Establecer la casilla final del rango a sumar.' C l di ió d l ill ti l últiintroducidos y la función PROMEDIO

Sub Sumar()Worksheets("Hoja4").ActivateDim Valor As IntegerDim Casilla_Inicial As StringDim Casilla FInal As String

' Coger la dirección de la casilla activa, la últimaCasilla_FInal = ActiveCell.AddressActiveCell.Offset(1, 0).Activate' Poner en la casilla activa la función SUMAActiveCell.Formula = "=Sum(" & Casilla_Inicial & ":" &

C ill FI l & ")"Dim Casilla_FInal As String' Hacer activa la casilla B12 de la hoja activaActiveSheet.Range("B12").ActivateDo

' Entrar un valor y convertirlo a numéricoV l V l(I tB ("E t l " "E t d "))

Casilla_FInal & ")"ActiveCell.Offset(1, 0).Activate' Poner en la casilla activa la función promedioActiveCell.Formula = "=Average(" & Casilla_Inicial & ":" &

Casilla_FInal & ")"'Ob l f i i lé lValor = Val(InputBox("Entrar un valor", "Entrada"))

' Si el valor es distinto de 0If Valor <> 0 Then

' Guardar el valor en la casilla activaActiveCell.Value = Valor

'Observar que las funciones se ponen en inglés y que al ejecutarse se traducen automáticamente'Si no se conoce el nombre de una función puede usarse la grabadoraEnd Sub

' Hacer activa la casilla de la fila siguienteActiveCell.Offset(1, 0).Activate

End IfLoop Until Valor = 0

03/08/2010 Adolfo Aparicio 90

Page 91: Mac

Operar con Funciones de E lExcel

Podemos operar con Funciones de Excel sin necesidad de insertarlas en una celdaVea la siguiente Calculadora Préstamos

Sub prestamo()Static Principal ‘ Variable estática No cambiaStatic Principal Variable estática. No cambiaStatic TasaStatic TerminosDim Pago As DoublePrincipal = Application.InputBox(Prompt:="Principal (100000 por ejemplo)", _

Default:=Principal)Tasa = Application.InputBox(Prompt:="Tipo de interés nominal anual (4,75 por ejemplo)", _

Default:=Tasa)Terminos = Application.InputBox(Prompt:="Número de años (30 por ejemplo)", _

Default:=Terminos)Default:=Terminos)‘ Vea como se usa la función de Excel Pmt (Pago) sin necesidad de calcularla en una celdaPago = Application.WorksheetFunction.Pmt(Tasa / 1200, Terminos * 12, Principal)MsgBox Prompt:="La Mensualidad es " & Format(-Pago, "Currency"), Title:="Calculadora de Préstamos"

End Sub

03/08/2010 Adolfo Aparicio 91

Page 92: Mac

EjercicioEjercicioEl siguiente programa calcula la longitud de una circunferenciaC t d f tú i ál l dCree otros dos programas que efectúen ese mismo cálculo usando la función de Excel =PI()

Uno de ellos insertando la función en una celdaEl otro sin necesidad de usar la hoja de Excel

Sub perimetro()Dim radio As Double, longitud As DoublegConst pi = 3.141592radio = InputBox("Introduzca el radio de la circunferencia")longitud = 2 * pi * radiolongitud = 2 pi radioActiveCell.Value = longitud

End Sub

03/08/2010 Adolfo Aparicio 92

Page 93: Mac

Solución EjercicioSolución EjercicioSub perimetro2()

Dim radio As Double, longitud As DoubleDim pi As Double Worksheets("Hoja1").Range("B3").Formula = "=pi()"pi = Range("B3").Valueradio = InputBox("Introduzca el radio de la circunferencia")longitud = 2 * pi * radiolongitud 2 pi radioActiveCell.Value = longitud

End Sub

Sub perimetro3()Sub perimetro3()Dim radio As Double, longitud As DoubleDim pi As Doublepi = Application.WorksheetFunction.pi()radio = InputBox("Introduzca el radio de la circunferencia")longitud = 2 * pi * radioMsgBox = longitud

End Sub

03/08/2010 Adolfo Aparicio 93

End Sub

Page 94: Mac

Array = MatrizArray = MatrizSub Array1()

'Declarar una matriz de tamaño 10Dim x(1 To 10) As Double'C l l l l t i

Sub Array2() 'Para una matriz verticar usando dos dimensiones'Declarar una matriz de 10 filas y una columnaDim x(1 To 10 1 To 1) As Double'Calcular valores aleatorios

For j = 1 To 10x(j) = Round(Rnd() * 100, 0)'Los valores aleatorios se calculan usando formulas VBA

Next j

Dim x(1 To 10, 1 To 1) As Double'Calcular valores aleatoriosFor j = 1 To 10

x(j, 1) = Round(Rnd() * 100, 0)Next j'T f i l t id d l t i l'Transferir el contenido de la matriz a una fila

Range(Cells(2, 2), Cells(2, 11)).FormulaArray = xEnd Sub

Sub Array3()

'Transferir el contenido de la matriz a una columnaRange(Cells(4, 2), Cells(13, 2)).FormulaArray = x

End Sub

'Calcular valores aleatorios'En este caso, los valores aleatorios se calculan usando formulas

ExcelRange(Cells(4, 3), Cells(13, 3)).FormulaArray = "=Round(Rand() *

100, 0)"'Vea la diferencia entre Rnd (del caso anterior) y Rand.'Una es una fórmula VBA la otra es una fórmula Excel en inglés

End Sub

03/08/2010 Adolfo Aparicio 94

Page 95: Mac

Horas Semanales TrabajadasHoras Semanales TrabajadasSub HorasSemanales()'Calcula las horas semanales t b j d

For j = 1 To 2If j = 1 Then

trabajadasDim a(1 To 5, 1 To 2) As SingleDim dia As String, jornada As StringDim i As Byte, j As Byte

jornada = "INICIO"Else

jornada = "FINAL"End Ify j y

'Tipo byte entre 0 y 255Dim horas As SingleFor i = 1 To 5

Select Case i

End Ifa(i, j) = InputBox("Introduzca la hora de " &

jornada _& " de la jornada de " & dia & "," & vbCrLf & _"en formato decimal Por ejemplo 17:30 son lasSelect Case i

Case 1: dia = "Lunes"Case 2: dia = "Martes"Case 3: dia = "Miercoles"

en formato decimal. Por ejemplo 17:30 son las 17,5")

If j = 2 Then horas = horas + a(i, 2) - a(i, 1)Next j

Case 4: dia = "Jueves"Case 5: dia = "Viernes"

End Select

Next iMsgBox "Horas semanales = " & horasEnd Sub

03/08/2010 Adolfo Aparicio 95

Page 96: Mac

Beneficio Medio de un Grupo de Empresasp pSub BeneficioMedio()'Calcula el beneficio medio de un grupo de empresas

Di () A D bl 'D fi t i i d i ú l di ióDim a() As Double 'Define una matriz sin decir aún la dimensiónDim n As ByteDim i As ByteDim media As DoubleDim media As Doublen = InputBox("Número de empresas del Grupo =")ReDim a(n) 'Redimensiona una matriz dinámicaFor i = 1 To nFor i 1 To n

a(i) = InputBox("Beneficio de la Empresa " & i & " = ", "La Media hasta ahora es " & media)

media = (media * (i - 1) + a(i)) / iNextMsgBox "Beneficio Medio del Grupo de Empresas= " & media

End Sub

03/08/2010 Adolfo Aparicio 96

Page 97: Mac

D t ió d EDetección de ErroresErrores en tiempo de compilación. Son los típicos errores que impiden hacer funcionar el programa debido, por ejemplo, a errores de sintaxis en las instrucciones, llamadas a funciones p g , p j p , ,que no existen o llamadas con el tipo o el número de parámetros incorrectos, etc. Este tipo de errores no dan demasiados problemas, primero porque el compilador avisa de donde se han producido y luego porque simplemente revisando la sintaxis se solucionan rápidamente. Errores en tiempo de ejecución. Estos errores se producen por una mala programación del código al no haber previsto determinados casos concretos o especiales, como por ejemplo intentar abrir un archivo que no existe imprimir sin comprobar que la impresora está conectadaintentar abrir un archivo que no existe, imprimir sin comprobar que la impresora está conectada, definir mal la dimensión de un array e intentar acceder a miembros que no existen, etc. Cuando se produce este tipo de errores se detiene la ejecución del programa y normalmente se informa del tipo de error que se ha producido. Muchos de estos errores se pueden solucionar mediante rutinas o funciones de tratamiento de errores.Errores de función. Son los más complicados de detectar ya que ni se detectan en la fase de p y qejecución, ni provocan la detención del programa, son debidos a la incorrecta programación de algún proceso y como resultado se obtienen datos erróneos. Errores de este tipo son cálculos mal hechos, bucles infinitos, devolución de valores incorrectos, etc. Como ni los detecta el compilador, ni provocan la interrupción del programa deben revisarse a mano usando las herramientas de depuración.

Consejo: modularice su programa utilizando procedimientos cortos que realicen trabajos concretos y precisos, de esta forma conseguirá, además de que el programa quede más elegante y en un futuro sea más sencillo modificarlo y depurarlo.

03/08/2010 Adolfo Aparicio 97

Page 98: Mac

EjercicioEjercicio

Cree una Macro con un juego queCree una Macro con un juego que consiste en que la máquina piensa un número entre 0 y 100 y el jugadornúmero entre 0 y 100, y el jugador debe adivinarlo. Para ello, dispone de 10 tiradas y el programa le indica si10 tiradas, y el programa le indica si el número secreto es mayor o menor al introducidoal introducido.

03/08/2010 Adolfo Aparicio 98

Page 99: Mac

Solución Ejercicio: AdivinaSub adivina()

Dim zona As StringDim x As Byte, n As ByteDim tirada As ByteRandomizeRandomizex = Fix(Rnd * 101) : tirada = 1 'FIX=INT=parte enteraDo

If zona = "" Thenn = InputBox("Introduzca un número entero del 0 al 100" & vbCrLf _& "Dispone de 10 tiradas para lograrlo" "Tirada número " & tirada)& Dispone de 10 tiradas para lograrlo , Tirada número & tirada)

Elsen = InputBox("El número secreto es " & zona & vbCrLf & "Introduzca otro", "Tirada número " & tirada)

End IfIf n = x Then

MsgBox "Felicidades!!!" & vbCrLf & "Ha adivinado el número secreto " & x & " en " & tirada & " tiradas"MsgBox Felicidades!!! & vbCrLf & Ha adivinado el número secreto & x & , en & tirada & tiradasExit Sub

End IfIf x < n Then

zona = "Inferior"ElElse

zona = "Superior"End Iftirada = tirada + 1

Loop Until tirada > 10

03/08/2010 Adolfo Aparicio 99

MsgBox "Ha agotado las 10 tiradas disponibles" & vbCrLf & "El número secreto es " & xEnd Sub

Page 100: Mac

D ió dDepuración de programasEstas herramientas son muy útiles a la hora de comprobar paso a paso el funcionamiento del programa y detectar los procesos que provocan un mal funcionamiento del mismo.programa y detectar los procesos que provocan un mal funcionamiento del mismo.Importe Módulo5.basActive la barra de depuración

(Ver/ Barras de Herramientas/ Depuración)Modo Ejecución paso a paso

Paso a Paso por Instrucciones. F8Paso a Paso por Procedimientos. MAY+F8

Sirve para ejecutar todo un procedimiento. Cuando en la ejecución de un procedimiento, se llega a una línea que llama a otro procedimiento o función, pulsando este botón se puede provocar la ejecución de todo el código de esta función para luego continuar con el modo paso a paso.

Modo InterrupciónModo InterrupciónEn programas largos resulta fastidioso tener que ejecutarlos paso a paso, sobretodo si sabemos que el error se produce en una parte avanzada del programa. El modo interrupción, permite la ejecución del programa hasta una instrucción determinada para, a partir de esta, ejecutar paso a paso y así poder detectar el error.Definir puntos de interrupciónp p

1. Sitúe el cursor sobre la instrucción en la cual debe detenerse el programa para continuar paso a paso.2. Pulse sobre el botón . También puede activar la opción Depuración/ Alternar punto de interrupción, pulsar la tecla F9 o bien hacer un clic en la parte izquierda de la ventana del módulo (la franja vertical en color gris).Para desactivar un punto de interrupción siga los mismos pasos

03/08/2010 Adolfo Aparicio 100

Para desactivar un punto de interrupción siga los mismos pasos

Page 101: Mac

L V t d I ióLa Ventana de InspecciónInspecciones rápidas de variables

Estas opciones sirven para revisar el valor de las variables a medida que se va ejecutando el programa.p gPara ver los valores que van tomando las variables es conveniente tener visible la Ventana de inspección, para activarla Ver/ Ventana de Inspección

Añadir una variable a la ventana de inspección1. Seleccione la variable que desee añadir a la ventana haciendo un clic sobre ella.2 Activar Depuración/ Inspección rápida o MaY+F92. Activar Depuración/ Inspección rápida o MaY+F9.

Aparece un cuadro de diálogo donde se muestra el valor actual de la variable. Si no está ejecutando el programa paso a paso, aparecerá el valor Fuera de Contexto.

3. Pulse sobre el botón Agregar para añadir la variable a la ventana de inspección.Debe tener en cuenta que para revisar las variables las expresiones que les asignan valores deben de ejecutarse al menos una vezvalores deben de ejecutarse al menos una vez.

Cuando ejecuta el programa paso a paso, si sitúa el puntero de ratón sobre una variable, se muestra el valor de la mismaBorrar una variable de la ventana de Inspección

Sólo debe seleccionarla en la ventana de inspección y pulsar sobre la tecla SUPRSólo debe seleccionarla en la ventana de inspección y pulsar sobre la tecla SUPR.Modificar el valor de una variable en tiempo de ejecución

A veces resulta interesante cambiar el valor de alguna variable cuando se está ejecutando el programa, para ver que ocurre si coge determinados valores, para terminar un bucle,…

03/08/2010 Adolfo Aparicio 101

Page 102: Mac

E i d R i ióExpresiones de RevisiónAdemás de permitir añadir una variable o

ió d t d l V t I di texpresión dentro de la Ventana Inmediato, una Expresión de Revisión permite interrumpir la ejecución del programa cuando una variable coge determinado valor.Piense que muchas veces un programa dejaPiense que muchas veces un programa deja de funcionar, o funciona mal cuando una variable coge determinados valores.Con una expresión de revisión, podremos detener la ejecución del programa cuando una variable contiene determinado valor ( a

ti d d t i d l ) lpartir de determinado valor), luego, podremos continuar con la ejecución paso a paso para ver que ocurre a partir de este punto.Sitúe el cursor sobre una variable y seleccione Agregar Inspección Interrupciónseleccione Agregar Inspección, Interrupción cuando el valor sea verdadero, y luego en la ventana de inspecciones, editar la variable y añadirla una condición lógica que al cumplirse parará el procedimiento.

03/08/2010 Adolfo Aparicio 102

Page 103: Mac

L V t I di tLa Ventana InmediatoEs otra forma de inspeccionar variables cuando el programa está en modo interrupción (ejecutándoseen modo interrupción (ejecutándose paso a paso)Además, ofrece la posibilidad de cambiar valores de las variablesE incluso ejecutar o evaluar jexpresiones. Para ver el valor de una variable en la ventana inmediato debe anteponerle un ? y luego pulsar Enter.Para activar la ventana InmediatoPara activar la ventana Inmediato, active opción Ver/Inmediato, o pulse la combinación CONTROL+G.Pruebe

?2+3?2^3?exp(1)?Suma_Columna

03/08/2010 Adolfo Aparicio 103

Page 104: Mac

L i t ió D b P i tLa instrucción Debug.PrintEsta instrucción se utiliza directamente sobre el código del programaPermite ver todos los valores que ha ido tomando una variable o expresión durante la ejecución delexpresión durante la ejecución del programa.Los valores se mostrarán en la ventana Inmediato una vez finalizado el programa.p gEsta expresión resulta útil en una fase avanzada de depuración ya que permite ir viendo la evolución de una variable o expresión sin necesidad de poner puntos denecesidad de poner puntos de interrupción.Cuando el programa esté listo deben eliminarse.Ejecute Dos a la diez()

03/08/2010 Adolfo Aparicio 104

Ejecute Dos_a_la_diez()

Page 105: Mac

F l iFormularios

Mostrar la barra de herramientas para cuadros de controlCuadro de TextoCuadro de TextoEtiquetaBotón de ComandoModo DiseñoPropiedades

E l i d d C ti bi l t t L b l1 D tEn la propiedad Caption, cambien el texto Label1 por Datos a Buscar

03/08/2010 Adolfo Aparicio 105

Page 106: Mac

L tLos eventosCuando se programan controles bien sea directamente en laCuando se programan controles bien sea directamente en la hoja o desde un formulario, debe tener en cuenta los eventos.Un evento se da cuando ocurre algo sobre un objetoEn entornos Windows constantemente se están produciendoEn entornos Windows constantemente se están produciendo eventos que son recogidos por el sistema.

Clicks con el ratón sobre un controlTeclear sobre un cuadro de texto etcTeclear sobre un cuadro de texto, etc.

Programar un evento significa hacer que se ejecuten determinadas instrucciones cuando ocurra dicho evento.En general todos los controles son capaces de capturarEn general, todos los controles son capaces de capturar diferentes eventos.

03/08/2010 Adolfo Aparicio 106

Page 107: Mac

C d C bi d (C b B )Cuadros Combinados (ComboBox)

Con un ComboBox podremos escoger el campo esCon un ComboBox podremos escoger el campo, es decir, podremos extraer coincidencias de Nombre, Apellidos, la Ciudad, etc.pPara ello incluiremos un cuadro combinado que permita escoger en que campo o columna tiene que b l i id ibuscarse la coincidencia.La lista, por supuesto, mostrará los nombres de las columnascolumnas.

03/08/2010 Adolfo Aparicio 107

Page 108: Mac

Formularios y Formularios y yyControlesControles

Cree un nuevo libro (mundo.xls)Cree un nuevo libro (mundo.xls)Acceda al Editor de Visual BasicAcceda al Editor de Visual BasicMenú, Insertar, UserFormMenú, Insertar, UserForm

ááEn el formulario que aparece UserForm1 se insertarán los En el formulario que aparece UserForm1 se insertarán los controles del Cuadro de Herramientas que también ha controles del Cuadro de Herramientas que también ha aparecido. Si no aparece haga clic en el icono “Cuadro de aparecido. Si no aparece haga clic en el icono “Cuadro de Herramientas”.Herramientas”.Insertemos ControlesInsertemos ControlesClic en el “Cuadro de Texto” del cuadro de herramientasClic en el “Cuadro de Texto” del cuadro de herramientasMarcar un recuadro en el UserForm1Marcar un recuadro en el UserForm1l l “ ó d d ”l l “ ó d d ”Clic en el “Botón de Comando”Clic en el “Botón de Comando”

Marcar un pequeño recuadro en el UserForm1Marcar un pequeño recuadro en el UserForm1Inserte un segundo Botón de ComandoInserte un segundo Botón de Comando

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 108108

Page 109: Mac

Hola MundoHola MundoHola MundoHola Mundo

Seleccione el CommandButton1 y vea las propiedades. Si no Seleccione el CommandButton1 y vea las propiedades. Si no aparecen pulse el icono Ventana de Propiedadesaparecen pulse el icono Ventana de PropiedadesEn la propiedad Caption escriba SaludoEn la propiedad Caption escriba SaludoEn la propiedad Caption del CommandButton2 escriba BorrarEn la propiedad Caption del CommandButton2 escriba BorrarHaga clic en el icono Ver Código de la Ventana de ProyectoHaga clic en el icono Ver Código de la Ventana de ProyectoObserve que aparece un área para introducir e código de los Observe que aparece un área para introducir e código de los ggeventos asociados al formularioeventos asociados al formularioArriba aparecen dos desplegables. El de la izquierda contiene Arriba aparecen dos desplegables. El de la izquierda contiene los Objetos y el de la derecha los Procedimientoslos Objetos y el de la derecha los Procedimientos

l d l d l bl l b dl d l d l bl l b dSeleccione del desplegable el objeto CommandButton1.Seleccione del desplegable el objeto CommandButton1.En el otro desplegable aparece automáticamente ClickEn el otro desplegable aparece automáticamente ClickEscribe el siguiente códigoEscribe el siguiente código

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 109109

Page 110: Mac

Código para el UserFormCódigo para el UserFormCódigo para el UserFormCódigo para el UserForm

Private Sub CommandButton1_Click()TextBox1.Text = "Hola Mundo"

E d S bEnd SubPrivate Sub CommandButton2_Click()

TextBox1.Text = ""End Sub

Vuelva al formulario. Basta cerrar la ventna, o mejor hacer clic Vuelva al formulario. Basta cerrar la ventna, o mejor hacer clic en el icono Ver Objetoen el icono Ver Objetoen el icono Ver Objetoen el icono Ver ObjetoGrabarGrabarEjecutar el programaEjecutar el programaSon procedimientos de eventoSon procedimientos de evento

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 110110

Son procedimientos de eventoSon procedimientos de evento

Page 111: Mac

Programar User FormsProgramar User FormsProgramar User FormsProgramar User Forms

El procedimiento a seguir es:El procedimiento a seguir es:1.1. Menú Insertar UserFormMenú Insertar UserForm1.1. Menú Insertar UserFormMenú Insertar UserForm

–– Cuadro de HerramientasCuadro de Herramientas

2.2. Establecer Propiedades de los objetosEstablecer Propiedades de los objetosp jp j3.3. Escribir el códigoEscribir el código

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 111111

Page 112: Mac

ContraseñaContraseñaContraseñaContraseñaInsertar un nuevo formulario. UserForm2Insertar un nuevo formulario. UserForm2Propiedades del UserForm2Propiedades del UserForm2

–– Name = frmClaveName = frmClave–– Caption = ContraseñaCaption = Contraseña

Insertar una Etiqueta (Insertar una Etiqueta (labellabel))Insertar una Etiqueta (Insertar una Etiqueta (labellabel))–– Name =lblClaveName =lblClave–– Caption = Introduzca la clave secretaCaption = Introduzca la clave secreta

Inserte un Cuadro de Texto (TextBox)Inserte un Cuadro de Texto (TextBox)–– Name = txtContraseñaName = txtContraseñaName txtContraseñaName txtContraseña–– MaxLength = 6MaxLength = 6–– PasswordChar = *PasswordChar = *

Inserte el CommandBooton1Inserte el CommandBooton1–– Name = cdmAceptarName = cdmAceptar

D f lt TD f lt T

Si un botón se pone como Si un botón se pone como Dafault = True, automáticamente Dafault = True, automáticamente los demás se ponen Default = los demás se ponen Default =

–– Default = TrueDefault = True–– Caption = AceptarCaption = Aceptar

Inserte el CommandBooton2Inserte el CommandBooton2–– Name = cdmCancelarName = cdmCancelar–– Caption = CancelarCaption = Cancelar

ppFalseFalse

En un formulario solo puede En un formulario solo puede haber un botón con la propiedad haber un botón con la propiedad Cancel = TrueCancel = True

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 112112

Caption CancelarCaption Cancelar–– Cancel = TrueCancel = True

Cancel = TrueCancel = True

Page 113: Mac

Código del UserForm Código del UserForm ggContraseñaContraseña

Escriba los siguientes procedimientos de eventoEscriba los siguientes procedimientos de eventoY ejecute el formulario para ver el funcionamientoY ejecute el formulario para ver el funcionamiento

Private Sub cmdAceptar_Click()If UCase(txtContraseña.Text) <> "MACROS" Then

MsgBox "Contraseña Incorrecta", vbCriticalEnd

ElseMsgBox "Contraseña Aceptada", vbExclamationEnd

End IfEnd Sub

Private Sub cmdCancelar_Click()End

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 113113

EndEnd Sub

Page 114: Mac

Formulario Formulario FechasFechas

Nuevo Libro Fechas.xlsNuevo Libro Fechas.xlsAbrir el Editor de Visual BasicAbrir el Editor de Visual BasicInsertar un UserFormInsertar un UserFormInsertar un UserFormInsertar un UserFormIncrustar los controles de la imagenIncrustar los controles de la imagen

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 114114

Page 115: Mac

PropiedadesPropiedadesPropiedadesPropiedades

–– TextBox1TextBox1Name = Name = txtPrimeroMesVienetxtPrimeroMesViene

–– TextBox2TextBox2

–– UserForm1UserForm1Name = frmFechasName = frmFechasCaption = FechasCaption = Fechas

–– Label1Label1Name = txtFechaName = txtFecha

–– Label5Label5Name = lblEtiq4Name = lblEtiq4Caption = 1º MesCaption = 1º Mes

Label1Label1Name = lblEtiq1Name = lblEtiq1Caption = Hoy esCaption = Hoy es

–– Label2Label2Name = lblHoyName = lblHoy Caption 1 Mes Caption 1 Mes

SiguienteSiguiente–– TextBox3TextBox3

Name = txtSiguienteName = txtSiguiente–– CommanButton1CommanButton1

Name = lblHoyName = lblHoy–– Label3Label3

Name = lblEtiq2Name = lblEtiq2Caption = Primero del mes Caption = Primero del mes que vieneque viene CommanButton1CommanButton1

Name = cmdCalcularName = cmdCalcularCaption = CalcularCaption = Calcular

–– CommandButton2CommandButton2Name = cmdOtraName = cmdOtra

que vieneque viene

–– Label4Label4Name = lblEtiq3Name = lblEtiq3Caption = Escribe una fechaCaption = Escribe una fecha

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 115115

Name = cmdOtraName = cmdOtraCaption = OtraCaption = Otra

pp

Page 116: Mac

Procedimientos de EventoProcedimientos de EventoPrivate SubUserForm_Activate()

lblHoy Caption Date

Private Sub cdmCalcula_Click()txtSiguiente.Text =

PrimeroMesCualquiera(txtFecha Text)lblHoy.Caption = DatetxtPrimeroMesViene.Text =

PrimeroMestxtFecha Text = Date

PrimeroMesCualquiera(txtFecha.Text)End SubPrivate Sub cmdOtra_Click()

txtFecha Text ""txtFecha.Text = DatetxtFecha.SetFocus

End Sub

txtFecha.Text = ""txtSiguiente.Text = ""txtFecha.SetFocus

E d S bEnd Sub

Function PrimeroMes()PrimeroMes = DateSerial(Year(Now), Month(Now) + 1, 1)

End FunctionFunction PrimeroMesCualquiera(Cual As Date) As Date

PrimeroMesCualquiera = DateSerial(Year(Cual), Month(Cual) + 1, 1)

03/08/201003/08/2010 Adolfo AparicioAdolfo Aparicio 116116

End Function

Page 117: Mac

Pasar una Matriz a una FunciónSub SumaCien()

Dim i As ByteDim a(100) As Byte

Function calcula(a() As Byte) As Single

Dim a(100) As Byte'Genera una matriz de números aleatorios

enteros entre 0 y 100Randomize

Dim i As ByteDim s As Single'Calcula la suma de los 100

'Si no se pone Randomize los valores aleatorios siempre son los mismos

'Pruebelo. Abra el libro y vuelva a lanzar la macro. Los resultado serán los mismos.

números que contiene la matrizFor i = 1 To 100

s = s + a(i)macro. Los resultado serán los mismos.For i = 1 To 100

a(i) = Int(Rnd * 101)Next i

s s a(i)Next icalcula = s

End FunctionMsgBox "Suma de 100 números aleatorios" & vbCrLf & _

"enteros entre 0 y 100" & vbCrLf & vbCrLf & Chr(9) & calcula(a())

End Function

03/08/2010 Adolfo Aparicio 117

Chr(9) & calcula(a())End Sub

Page 118: Mac

C lt NConsulta News

Asunto: Consulta como buscar ultimoAsunto: Consulta como buscar ultimo Quiero averiguar como se hace una bú d d lti i t d t dbúsqueda de un ultimo registro dentro de un rango determinado. Ej. Tengo un rango de A1:A10 en donde hay datos desde A1 hastaA1:A10 en donde hay datos desde A1 hasta A6. En la celda A15 quiero obtener el valor o dato que haya en la última celda ocupada deldato que haya en la última celda ocupada del rango A1:A10, es decir que me escriba en este caso lo que hay en A6.

03/08/2010 Adolfo Aparicio 118

este caso lo que hay en A6.

Page 119: Mac

S l ióSoluciónSub UltimoValor()

Worksheets("Hoja1").Activate 'Aqui le dices la hojaActiveSheet.Range("A10").Activate 'Aqui le dices el final del

rangorangoDo While IsEmpty(ActiveCell)

ActiveCell.Offset(-1, 0).ActivateLoopRange("A15").Value = ActiveCell.Value

End SubEnd Sub

Se pude hacer con una formula, pero si tiene celdas vacías intermedias no funciona=INDIRECTO(CONCATENAR("A";CONTAR(A1:A10)))

03/08/2010 Adolfo Aparicio 119

Page 120: Mac

P t C lPregunta Color

Asunto: codigo para contar celdas de XAsunto: codigo para contar celdas de Xalguien me puede pasar el codigo - formula

t l d ld tipara contar el numero de celdas que tienen X's color (trama) saludos!!

03/08/2010 Adolfo Aparicio 120

Page 121: Mac

R t C l 1Respuesta Color 1Primero vamos a colorear unas cuantas celdas del rangoPrimero vamos a colorear unas cuantas celdas del rango D1:D30. Para ello ejecuta el siguiente código:

Sub colorea()Dim CeldaDim CeldaDim R As RangeSet R = Range("D1.D30")R SelectR.SelectFor Each Celda In R

Celda.Interior.ColorIndex = Int(Rnd * 10) + 1NextNext

End Sub

En el código anterior hemos pedido que nos coloree con un máximo

03/08/2010 Adolfo Aparicio 121

En el código anterior hemos pedido que nos coloree con un máximo de 10 colores, aunque sabemos que existen 56 colores distintos.

Page 122: Mac

R t C l 2Respuesta Color 2En segundo lugar ejecuta esta macro:

Sub pru()MsgBox "Celdas de color Rojo (3): " & CuentaColor(Range("D6.D30"), 3)

End Sub

Se basa en la función CuentaColor, que cuenta el color rojo (# 3)Function CuentaColor(R As Range, tono As Byte) As Byte

Dim num As LonggDim CeldaFor Each Celda In R

If Celda.Interior.ColorIndex = 3 Then num = num + 1Next

V fi h C ld C l l

NextCuentaColor = num

End Function

03/08/2010 Adolfo Aparicio 122

Ver fichero CeldaColor.xls