Любой лучший способ установить ширину ListBox, равную ширине столбца - PullRequest
0 голосов
/ 07 апреля 2020

я написал следующий код. как я новичок в VBA. Могу ли я получить небольшую версию или альтернативный код для достижения того же результата. я хочу, чтобы автоматически размер ширины столбца с максимальной длиной столбца таблицы используются ячейки. Я использовал метод словаря для хранения максимальной длины ячеек в столбце и l oop всех столбцов и сохранения максимальной длины в словаре.

'============Refresh ListBox===================
Public Function RefreshData()

Dim max As Integer
Dim Letter As String
Dim RngAddress As String
Dim C As Integer

'enable microsoft scripting runtime from tools ---> reference

Dim mydictionery As Scripting.Dictionary
Set mydictionery = New Scripting.Dictionary

C = 1
max = 0
Letter = "A"

For i = 1 To 26

RngAddress = Letter & 2 & ":" & Letter & 3

For Each cell In Range(RngAddress)
If Len(cell) > max Then max = Len(cell)
Next

mydictionery.Add Letter & max, 0 'add Letter & max to prevent duplicate key

max = 0 'reset max for next range
C = C + 1 'increase counter for dictionery
Letter = Chr(Asc(Letter) + 1) 'next column for range

Next i

Dim iRow As Long
Dim sh As Worksheet

Set sh = ThisWorkbook.Sheets("Database")
Last_Row = Application.WorksheetFunction.CountA(sh.Range("AB:AB"))

        With DataEntryFormDynamic.ListBox1


        .ColumnCount = 28
        .ColumnHeads = True

        .ColumnWidths = Mid(mydictionery.Keys(0), 2, 15) * 6 & "," & Mid(mydictionery.Keys(1), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(2), 2, 15) * 6 & "," & Mid(mydictionery.Keys(3), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(4), 2, 15) * 6 & "," & Mid(mydictionery.Keys(5), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(6), 2, 15) * 6 & "," & Mid(mydictionery.Keys(7), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(8), 2, 15) * 6 & "," & Mid(mydictionery.Keys(9), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(10), 2, 15) * 6 & "," & Mid(mydictionery.Keys(11), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(12), 2, 15) * 6 & "," & Mid(mydictionery.Keys(12), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(14), 2, 15) * 6 & "," & Mid(mydictionery.Keys(15), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(16), 2, 15) * 6 & "," & Mid(mydictionery.Keys(17), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(18), 2, 15) * 6 & "," & Mid(mydictionery.Keys(19), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(20), 2, 15) * 6 & "," & Mid(mydictionery.Keys(21), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(22), 2, 15) * 6 & "," & Mid(mydictionery.Keys(23), 2, 15) * 6 & "," _
                      & Mid(mydictionery.Keys(24), 2, 15) * 6 & "," & Mid(mydictionery.Keys(25), 2, 15) * 6 & ","

        If Last_Row = 1 Then
        .RowSource = "Database!A2:AB2"
        Else
        .RowSource = "Database!A2:AB" & Last_Row
        End If
        End With

Set mydictionery = Nothing
End Function


'in userform

Private Sub userform activate ()
Call RefreshData
end sub

Заранее благодарен за вашу помощь и поддержку в этом.

Ответы [ 2 ]

1 голос
/ 07 апреля 2020

Ширина каждого столбца помещается в массив и превращается в один символ и применяется к списку.

   Private Sub UserForm_Initialize()
        Dim rngDB As Range, rng As Range
        Dim sWidth As String
        Dim vR() As Variant
        Dim n As Integer

        Set rngDB = Range("a1:ab1")
        For Each rng In rngDB
            n = n + 1
            ReDim Preserve vR(1 To n)
            vR(n) = rng.EntireColumn.Width
        Next rng
        sWidth = Join(vR, ";")
        Debug.Print sWidth
        With ListBox1
            .ColumnCount = 28
            .ColumnWidths = sWidth '<~~ 24;24;24;24;24;39;39;39;39;45;45;45;45;45;45;45;45;45;45;45;51;51;51;51;57;57;57;63
            .RowSource = "A1:AB2"
            .BorderStyle = fmBorderStyleSingle
        End With
    End Sub

- что необходимо изменить до максимального значения

Private Sub UserForm_Initialize()
    Dim rngDB As Range, rng As Range
    Dim sWidth As String
    Dim vR() As Variant
    Dim n As Integer
    Dim myMax As Single

    Set rngDB = Range("a1:ab1")
    For Each rng In rngDB
        n = n + 1
        ReDim Preserve vR(1 To n)
        vR(n) = rng.EntireColumn.Width
    Next rng
    myMax = WorksheetFunction.Max(vR)
    For i = 1 To n
        vR(i) = myMax
    Next i

    sWidth = Join(vR, ";")
    Debug.Print sWidth
    With ListBox1
        .ColumnCount = 28
        .ColumnWidths = sWidth '<~~ 63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63;63
        .RowSource = "A1:AB2"
        .BorderStyle = fmBorderStyleSingle
    End With
End Sub

Изображение

enter image description here

1 голос
/ 07 апреля 2020

Вот одна альтернатива:

Private Sub UserForm_Activate()
    RefreshData
End Sub

Public Function RefreshData()
    Dim sh As Worksheet, lRow As Long, rng As Range

    Set sh = ThisWorkbook.Sheets("Database")
    lRow = Application.max(sh.Cells(Rows.Count, "AB").End(xlUp).Row, 2)
    Set rng = sh.Range("A2:AB" & lRow)

    With Me.ListBox1
        .ColumnCount = 28
        .ColumnHeads = True
        .ColumnWidths = ColWidths(rng.Resize(2, rng.Columns.Count)) 'just using first 2 rows?
        .RowSource = "'" & rng.Parent.Name & "'!" & rng.Address()
    End With

End Function

Public Function ColWidths(rng As Range) As String
    Dim col As Range, arr(), i As Long, m
    ReDim arr(0 To rng.Columns.Count - 1)
    Debug.Print UBound(arr) + 1 & " columns"
    For i = 1 To rng.Columns.Count
        m = rng.Parent.Evaluate("=Max(Len(" & rng.Columns(i).Address() & "))")
        If IsError(m) Then m = 1 'or some other suitable value on an error
        arr(i - 1) = m * 6
    Next i
    Debug.Print Join(arr, ",")
    ColWidths = Join(arr, ",")
End Function
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...