运行时错误 3061 预期参数太少 1 DAO 声明

Runtime Error 3061 too few parameters expected 1 DAO declarations

我一直在寻找一些代码来创建日历年并将从 table 中获取数据并将其放在日历上的相应日期。我在网上找到了一些代码(来自旧版本的访问),非常符合要求,经过一些修改,它完全可以满足我的需要。最初,代码从一个 table 中提取数据,并将其设置为当年的 运行。我使用两个查询 qr_SafetyCal 和 qr_SafetyCal2 来优化 table 中的数据。第一个查询确定数据的优先级并消除任何给定日期的多个事件。第二个查询使用第一个的结果并在查询条件中指定年份。

只要我在 qr_SafetyCal2 中设置年份标准并指定第一天,代码就可以完美运行,例如。 1/1/2017 (datStart) 在我想要显示的日历年的基础代码中。

得到代码后,我创建了一个弹出窗体,用户 select 报告的年份但是当我 运行 报告时,我收到以下错误,运行时错误 3061预期参数很少 1.

根据我的研究,我相信当我在 DAO 记录集使用的查询条件中引用表单时,我改变了代码的动态。 据我了解,查询中的条件不会传递给 rs,因此需要在代码中声明。我想不通的是如何通过引用表单在代码中声明变量。我希望这对某些人有意义,解释很长,但很难描述你不理解的东西。

下面是所有代码,您会看到一些我已经尝试过但没有奏效的内容。任何帮助将不胜感激。如果代码格式不正确,我提前道歉。

Option Compare Database
Option Explicit

Private m_strCTLLabel  As String
Private m_strCTLLabelHeader As String
Private colCalendarDates As Collection

Function getCalendarData() As Boolean
Dim rs As DAO.Recordset
Dim strDate As String
Dim strCode As String

Dim i As Integer

    'Dim qdf As DAO.QueryDef
    'Set qdf = CurrentDb.QueryDef("qr_SafetyCal2")
    'qdf.Parameters("[Forms]![fr_SafetyCal]![cboYear]") = [Forms]![fr_SafetyCal]![cboYear]
    'Set rs = qdf.OpenRecordset("qr_SafetyCal2", dbOpenDynaset)

    Set rs = CurrentDb.OpenRecordset("qr_SafetyCal2", dbOpenDynaset)

    Set colCalendarDates = New Collection
    With rs
        If (Not .BOF) Or (Not .EOF) Then
            .MoveLast
            .MoveFirst
        End If
        If .RecordCount > 0 Then
            For i = 1 To .RecordCount
                strDate = .Fields("Date")
                strCode = .Fields("ShortName")
                colCalendarDates.Add strCode, strDate
                .MoveNext
            Next i
        End If
        .Close
    End With
    'Return of dates and data collection form qr_SafetyCal2
    Set rs = Nothing

End Function

Public Sub loadReportYearCalendar(theReport As Report)
Dim i As Integer
Dim datStart As Date
Dim rptControl As Report

    m_strCTLLabel = "labelCELL"
    m_strCTLLabelHeader = "labelDAY"

    'Load calendar data for the specified year into the collection
    Call getCalendarData

    With theReport
        'Get the first month of the specified year
        datStart = "1/1/2017" '"1/1/" & Year(Date), "1/1/" & Forms!
        [fr_SafetyCal]![cboYear], Forms![fr_SafetyCal]![txtCalYear]
        'Add the specified year to the report's label
        .Controls("labelCalendarHeaderLine2").Caption = Year(datStart) & " 
        iCalendar"
        For i = 1 To 12
            'Set pointer to subreport control hosting the mini-calendar
            Set rptControl = .Controls("childCalendarMonth" & i).Report
            'Run procedure to populate control with it's respective year
            Call loadReportCalendar(rptControl, datStart)
            'Reset and obtain first day of the following month
            datStart = DateAdd("m", 1, datStart)
        Next i
     End With
    'Clean up
    Set colCalendarDates = Nothing
    Set rptControl = Nothing
End Sub


Public Sub loadReportCalendar(theReport As Report, Optional StartDate As 
Date, Optional theHeaderColor As Variant)
Dim i As Integer
Dim intCalDay As Integer
Dim datStartDate As Date
Dim intWeekDay As Integer

    datStartDate = StartDate
    intWeekDay = Weekday(datStartDate)

    With theReport

        .Controls("labelMONTH").Caption = Format(StartDate, "mmmm")

        'Change the day label's backcolor if necessary
        If Not (IsMissing(theHeaderColor)) Then
             For i = 1 To 7
                 .Controls("labelDayHeader" & i).BackColor = theHeaderColor
            Next
        End If

        For i = 1 To 42
            With .Controls(m_strCTLLabel & i)
                If (i >= intWeekDay) And (Month(StartDate) = 
                Month(datStartDate)) Then
                    If (datStartDate = Date) Then
                        .BackColor = 14277081
                    End If

                    On Error Resume Next

                    Dim strCaption As String
                    Dim strKey As String

                    strKey = datStartDate
                    strCaption = ""
                    strCaption = colCalendarDates.Item(strKey)
                    colCalendarDates.Remove strKey
                    'Set back color to grean on days in the past that have 
                    no corresponding event
                    If (datStartDate < Date) And (strCaption = vbNullString) Then
                    .Caption = Day(datStartDate)
                    .Bold = False
                    .BackColor = vbGreen
                    .ForeColor = vbWhite
                    .Heavy = True
                'Do not set a back color for days in the future
                ElseIf (datStartDate > Date) And (strCaption = vbNullString) Then
                    .Caption = Day(datStartDate)
                    .Bold = False
                'Set the corresponding labels and formats for each specified event
                Else
                    .Caption = strCaption
                    .Bold = True
                    Select Case strCaption
                        Case "FA"
                            .BackColor = vbYellow
                            .ForeColor = 0
                            .LeftMargin = 0
                            .TextAlign = 2
                        Case "FAM"
                            .BackColor = vbYellow
                            .ForeColor = 0
                            .LeftMargin = 0
                            .TextAlign = 2
                            .Heavy = True
                        Case "LTA"
                            .BackColor = vbRed
                            .ForeColor = vbWhite
                            .LeftMargin = 0
                            .TextAlign = 2
                        Case "MED"
                            .BackColor = vbRed
                            .ForeColor = vbWhite
                            .LeftMargin = 0
                            .TextAlign = 2
                    End Select

                End If

               datStartDate = DateAdd("d", 1, datStartDate)
            Else
                .Caption = ""
            End If
        End With
    Next i
End With

End Sub

这是两个查询的SQL,第一个是qr_SafetyCal,第二个是qr_SafetyCal2:

SELECT tb_CaseLog.Date, Max(tb_Treatment.Priority) AS MaxOfPriority, 
Count(tb_Treatment.TreatmentID) AS CountOfTreatmentID
FROM tb_Treatment INNER JOIN tb_CaseLog ON tb_Treatment.TreatmentID = 
tb_CaseLog.Treatment
GROUP BY tb_CaseLog.Date;

SELECT qr_SafetyCal.Date, tb_Treatment.ShortName, 
qr_SafetyCal.CountOfTreatmentID AS [Count], Year([Date]) AS CalYear
FROM qr_SafetyCal INNER JOIN tb_Treatment ON qr_SafetyCal.MaxOfPriority = 
tb_Treatment.Priority;

无需引用 QueryDef。

通过引用组合框打开包含筛选数据集的记录集对象,如:

Set rs = CurrentDb.OpenRecordset("SELECT * FROM qr_SafetyCal2 WHERE CalYear=" & [Forms]![fr_SafetyCal]![cboYear], dbOpenDynaset)

或者如果代码在表格后面:

Set rs = CurrentDb.OpenRecordset("SELECT * FROM qr_SafetyCal2 WHERE CalYear=" & Me.[cboYear], dbOpenDynaset)

两个示例都假定字段是数字类型。

如果查询中没有带有年份值的字段,可以从VBA构造中的日期值字段中提取:

Set rs = CurrentDb.OpenRecordset("SELECT * FROM qr_SafetyCal2 WHERE Year([YourFieldnameHere])=" & [Forms]![fr_SafetyCal]![cboYear], dbOpenDynaset)

设置datStart变量的代码:

'Get the first month of the specified year
datStart = "1/1/" & Forms![fr_SafetyCal].[cboYear]