miércoles, 31 de julio de 2024

84 - Cómo llamar a otros libros para compartir información.

En esta entrada veremos cómo llamar a los libros cuando necesitamos invocar su nombre desde una macro, ya sea para capturar  información o enviarla a otros libros.


La macro se encuentra en el libro Ventas. Desde el libro Pedido se trae información a éste libro activo y también al libro de Taller. Como segundo paso también moveremos datos entre diferentes hojas del libro Ventas.

En el código siguiente solo dejaré las instrucciones de declaración de variables y un pase como ejemplo. El pegado puede ser como en el ejemplo, 'Paste Special', solo 'Paste' o cualquier otro tipo de pegado como los vistos en el video Nº 71 de mi canal.

Macro ubicada en un módulo del libro VENTAS (libro activo).

NOTA: En el video, la ruta de los libros se toma a partir del libro Ventas, el que contiene la macro. Pero en caso de que se encuentre, por ejemplo, en otro disco (Ej.1) o en un servidor (Ej.2), las instrucciones serían de este modo:

Ejemplo 1:
          ruta = "D:\AL_TRABAJO_2023\Javi_Prueba_IMAGENES\Datos\"
          nbreLibro = "Consultas Julio.xlsm"
          Workbooks.Open (ruta & nbreLibro)

Ejemplo 2: 
          ruta = "\\EMPRESA\Sucursal1\Depto.Pedidos\Marketing y ventas\"
          nbreLibro = "Consultas Julio.xlsm"
          Workbooks.Open (ruta & nbreLibro)

Sub Registra_Pedido()
Dim libV As Workbook, libP As Workbook, libT As Workbook
Dim hoV As Worksheet, hoP As Worksheet
Dim hoE As Worksheet
Dim ini As Integer    'filas en hoja Ventas
Dim sFila As Integer  'fila en hoja Entregas
Dim ruta As String
Dim nbreLibro As String
Application.ScreenUpdating = False
'libro y hoja de Ventas
Set libV = ActiveWorkbook
Set hoV = libV.Sheets("Control de Ventas")
    'se deja la hoja desprotegida y sin filtros
    With hoV                      
        .Select
        .Unprotect
        If .FilterMode = True Then .ShowAllData
        
'se establece primera fila libre para el registro
        ini = .Range("B" & Rows.Count).End(xlUp).Row + 1 
    End With              
'libro y hoja de origen (Pedido)
ruta = ThisWorkbook.Path & "/"

'*** CONTROLAR QUE EXISTA LA CARPETA Y EL LIBRO
    On Error Resume Next
    midire = ThisWorkbook.Path & "/PEDIDOS REGISTRADOS"
    'si la carpeta existe la guarda como ruta
    If Dir(midire, vbDirectory) = "" Then
        MsgBox "No se encuentra la subcarpeta de Pedidos. El proceso se cancela."
        Exit Sub
    End If
    'se solicita el nombre del Pedido a registrar 
    '(ver en el módulo 1 otras maneras de obtener ese nbre.)
    nbreLibro = InputBox("Ingresa el nro de Pedido para registrar en el libro de Ventas.")
    If Dir(midire & "/" & nbreLibro) = "" Then
        MsgBox "No se encuentra el libro de Pedidos. El proceso se cancela."
        Exit Sub
    End If
    On Error GoTo 0
'*******

Workbooks.Open (ruta & "PEDIDOS REGISTRADOS/" & nbreLibro)    
Set libP = ActiveWorkbook
Set hoP = libP.Sheets("Pedido")

'libro y hoja de destino (Taller)
On Error GoTo sinLibroTaller
Workbooks.Open (ruta & "Libro TALLER.xlsm")
Set libT = ActiveWorkbook
On Error GoTo 0
MsgBox "A continuación se procederá a pasar la información del Pedido al Sistema de Ventas.", , "INFORMACIÓN"

'-------------- PASA de LIBRO PEDIDOS a LIBRO VENTAS ------------
libP.Activate
hoP.Select
'pase de campos
hoV.Range("H" & ini) = Range("C7")
hoV.Range("I" & ini) = Range("E7")

'------INSERTA HOJA EN TALLER y se registran campos del PEDIDO--
'rango de la hoja Pedido que será copiado en la nueva hoja de Taller
Range("A66:J160").Copy
libT.Activate
'se agrega una nueva hoja en el libro Taller que se acaba de activar
Sheets.Add After:=Sheets(Sheets.Count)
'el rango se pega a partir de A3 con un pegado especial.
Range("A3").Select
Selection.PasteSpecial Paste:=xlPasteColumnWidths, Operation:=xlNone, _
    SkipBlanks:=False, Transpose:=False
ActiveSheet.Paste
'se copìan otros rangos. AHORA LA HOJA ACTIVA ES LA NUEVA DE TALLER
'se indica el origen anteponiendo el objeto 'hoP'
hoP.Range("A24:C32,F24:J32").Copy Destination:=ActiveSheet.[L2]
'se asigna un nombre a la nueva hoja de Taller
ActiveSheet.Name = hoP.Range("C7")
ActiveWorkbook.Save     'guarda y cierra libro Taller
ActiveWorkbook.Close

'------------- Se cierra libro Pedidos quedando activo el libro Ventas    ---------
libP.Close True   

'------------- PASA DATOS DESDE VENTAS A HOJA auxiliar del mismo libro ---
libV.Activate
Set hoE = libV.Sheets("Entregas")
With hoE
    .Select
    .Unprotect
End With
    'se busca el fin de la tabla, agregando otra fila y redimensionando la misma.
    Set tablax = hoE.ListObjects(1)
    sFila = tablax.Range.Columns(1).Cells.Find("*", SearchOrder:=xlByRows, SearchDirection:=xlPrevious).Row + 1
    sFila = sFila + 1
    miRango = .Range(Cells(6, 2), Cells(sFila, 10)).Address
    tablax.Resize Range(miRango)
    'pase de datos a la nueva fila
    Range("B" & sFila) = hoV.Range("A" & ini)
    Range("D" & sFila) = hoV.Range("E" & ini)
    Range("D" & sFila).NumberFormat = "m/d/yyyy"
    Range("B" & sFila).Select
ActiveSheet.Protect , DrawingObjects:=True, Contents:=True, Scenarios:=True _
        , AllowFormattingCells:=True, AllowFormattingColumns:=True, _
        AllowFormattingRows:=True, AllowSorting:=True, AllowFiltering:=True
'-----------------------------------------------------------------------------------------
'se selecciona la 1ra.hoja de Ventas dejando seleccionada la fila del registro
hoV.Select 
ActiveSheet.Range("A" & ini).Select
ActiveSheet.Protect , DrawingObjects:=True, Contents:=True, Scenarios:=True _
        , AllowFormattingCells:=True, AllowFormattingColumns:=True, _
        AllowFormattingRows:=True, AllowSorting:=True, AllowFiltering:=True
'ActiveWorkbook.Save            'opcional
MsgBox "Fin del proceso de captura del Pedido.", , "Información"
Exit Sub

sinLibroTaller:
    MsgBox "No se encontró el libro Taller en la carpeta activa. El proceso se cancela."
End Sub


Ver video Nº 84 desde aquí.

Descargar el libro de ejemplo desde aquí


miércoles, 17 de julio de 2024

83 - El Administrador de Nombres y sus múltiples usos.

Es una de las herramientas más útiles a la hora de trabajar con validación de datos, listas desplegables, Tablas y Macros. Veremos a continuación sus principales usos.

CASO 1: cuando una misma lista (con nro. fijo de elementos) será utilizada en 1 o varias hojas con celdas con validación de datos. O en formularios, en algún control desplegable. El típico caso es el de los meses del año o días de semana.
Muchos usuarios colocan la lista en la primera hoja donde la van a utilizar. Y en las celdas con validación, en el campo Origen colocan ese rango. O a veces colocan allí la lista de elementos separados con punto y coma.
El problema surge cuando necesitamos utilizar esa lista de meses o días en otras hojas y tenemos que repetir el rango o recordar en qué hoja se encuentra ya creada.



SOLUCIÓN: seleccionar el rango (en este ejemplo desde Enero a Diciembre) y desde el menú Fórmulas, Administrador de nombres, asignar un Nombre (por ej: Meses), en el campo Ámbito' optar por 'Libro' y en el campo 'Se refiere a' introducir o seleccionar la hoja y el rango ocupado por esa lista.

Luego en cada celda con validación de datos ( o en los desplegables de algún Userform) donde se requiere mostrar la lista para seleccionar algún elemento, la opción será colocar ese nombre como se muestra en la imagen siguiente:



CASO 2: una lista dinámica que será utilizada en varias hojas con celdas con validación de datos (o controles desplegables en un Userform). El caso típico son las listas de Conceptos que pueden irse incrementando.  


SOLUCIÓN: Seleccionar una celda de la lista y desde el Administrador de nombres, Nuevo. Asignar un nombre, Ambito = Libro y en el campo 'Se refiere a' colocar la siguiente fórmula: 

=DESREF(Listas!$B$16;0;0;CONTARA(Listas!$B$16:$B$200;1);-1)

Argumentos del ejemplo: nombre de la hoja que contiene a la lista (Listas!), la celda del primer elemento ($B$16) y la celda hasta donde puede llegar el total de elementos ($B$200)




CASO 3: cuando una columna de una hoja se utilice para mostrar en desplegables. Aquí la lista también será dinámica, es decir que a medida que se agreguen datos a la hoja se irá incrementando el nro. de elementos que se mostrará en los desplegables. El caso típico son las hojas 'Base' como lista de Productos, Clientes u otras bases del sistema.

SOLUCIÓN: Si se trata de una hoja en formato de rango, seleccionar la columna y utilizar la función DESREF tal como vimos en el Caso 2, permitiendo así el agregado de elementos en esa columna indicando un máximo de filas (en exceso) que se considere que puede llegar a tener la base. 

=DESREF(Clientes!$B$4;0;0;CONTARA(Clientes!$B$4:$B$20000;1);-1)

Si se trata de una hoja en formato Tabla, en el campo 'Se refiere a' colocar solamente el nombre de la Tabla y la columna. Aquí no es necesario utilizar la función DESREF.
=nbre_tabla[nbre_columna]



CASO 4: Uso de listas dependientes. Es decir, cuando según la selección en una lista será la lista que se mostrará en otro desplegable. Y así tantas listas como desplegables se encuentre en la hoja o en un formulario.
Algunos usos: 
  1. Categoría de conceptos, Subcategoría.
  2. Lista de países, provincias, departamentos, localidades.
  3. Listado de carpetas, subcarpetas, archivos (se puede considerar varios niveles de subcarpetas)
  4. Lista de Personal: Planta o Sucursal, Área de trabajo, Nivel o categoría, Nombre.
SOLUCIONES: habrá que crear listas y a cada una de ellas asignarles un nombre y un rango de valores. Ese rango puede ser fijo o dinámico (en este caso con el uso de la función DESREF.). Es necesario que cada lista tenga por nombre el elemento de donde proviene.

Para el pto.1: En la siguiente imagen vemos que tenemos varias listas de Conceptos. Al momento de registrar un pago vamos a elegir una categoría y según esa selección se nos mostrará la lista de subcategorías.



NOTA: Como en este caso el nombre creado como 'Conceptos' corresponde a celdas de una fila y no de columna, no es posible asignarlo en la propiedad RowSource del primer control ComboBox6. Sino que se tendrá que rellenar mediante programación al momento de Inicializar el formulario. 

Private Sub UserForm_Initialize()

For Each ct In Range("Conceptos")      'nombre del rango entre comillas

    ComboBox6.AddItem ct.Text

Next ct

'el resto de las instrucciones para este evento


Y para el segundo control desplegable, de Subcategorías, estas serán las instrucciones, donde la propiedad RowSource toma como rango el texto del control anterior.

Private Sub ComboBox6_AfterUpdate()

If ComboBox6 = "" Then

       ComboBox7.RowSource = "": ComboBox7.Clear    

Else

       ComboBox7.RowSource = "=" & Trim(ComboBox6.Text)

End If

End Sub


Para el pto.2:  Para la siguiente imagen se crearon las siguientes listas: 
  • PAISES
  • Argentina, Bolivia y el resto de países. 
  • Cordoba, Misiones y el resto de provincias por cada país.
  • Punilla, Calamuchita y el resto de departamentos por cada provincia.
  • Cosquín, La Falda y el resto de localidades por cada departamento.

NOTA: Como el rango 'Paises' se encuentra en una columna (y no en fila como en el ejemplo anterior), aquí sí es posible establecer, desde el modo diseño, la propiedad RowSource del primer desplegable (PAIS)


Y para el resto de los controles seguimos el ejemplo del punto anterior, con los eventos Change o AfterUpdate. O como en el libro que se puede descargar (ver enlace al pie) donde utilicé el evento Enter.

IMPORTANTE: si se van a utilizar varias listas en el libro, recomiendo utilizar una hoja para contenerlas y así encontrarlas a todas en un mismo lugar.


Descargar libro de ejemplo desde aquí.

Ver video Nº 83 desde aquí.