我不确定为什么其他人之前没有问过这个问题。但是你注意到asp:Calendar在结尾显示了额外的一周吗?
例如,如果VisibleMonth设置为2010-03-01,FirstDayOfWeek设置为Sunday: 它将显示6周。
我想知道微软为什么要显示完全在四月份的最后一行。我试图在网上搜索一个属性,但它似乎不存在。
我能想到的唯一解决方案是覆盖Pre_Render并检查所有个别日期,如果它们仍在VisibleDate的一周内。但当然这是极端的检查,因为控件的每次渲染都会显示出来。
这是我的工作。
protected void Calendar1_DayRender(object sender, DayRenderEventArgs e)
{
int dayOfWeek = Convert.ToInt16(e.Day.Date.DayOfWeek);
int compensate = dayOfWeek - Convert.ToInt16(DayOfWeek.Sunday);
DateTime WeekStart = e.Day.Date.AddDays(-1 * compensate);
DateTime WeekEnd = WeekStart.AddDays(6);
// If the start and end of the week does not have relevance to the current month
if (WeekStart.Month != Calendar1.VisibleDate.Month &&
WeekEnd .Month != Calendar1.VisibleDate.Month)
{
e.Cell.Text = "";
e.Cell.Height = 0;
e.Cell.Visible = false;
}
}
答案 0 :(得分:7)
几个月内,日历控件会在月初显示额外的一周。 (当月的第一天是一周的第一天时。)
设置e.cell.Visible = false时,它不会渲染元素。所以在chrome中你最后得到一行<tr></tr>
。 Chrome将其渲染为空白行。而且由于我认为没有办法通过日历控件设置TR元素的高度/样式,所以最终会看到一个丑陋的日历,它缺少它在某些月份的第一行。
当您设置Visible = false时,将高度设置为0也不会执行任何操作。如果您没有设置Visible = false并且只是将height设置为0,则它仍然无法在chrome中正确呈现。所以解决方案是将高度设置为 1
这是我修改后的解决方案。
onrowrender
protected void Calendar1_DayRender(object sender, DayRenderEventArgs e){
hideExtraWeek(sender, e, (DayOfWeek)Calendar1.FirstDayOfWeek);
}
功能
protected void hideExtraWeek(object sender, DayRenderEventArgs e, DayOfWeek dw){
if (dw == (DayOfWeek)7) dw = (DayOfWeek)0; // FirstDayOfweek returns 7 when set to default, But it's zero based so valid values are 0 to 6
Boolean blnBrowserDoesntSupportsEmptyRow= Request.Browser.Browser=="Chrome" ||
Request.Browser.Browser=="Safari";
int dayOfWeek = Convert.ToInt16(e.Day.Date.DayOfWeek);
int compensate = dayOfWeek - Convert.ToInt16(dw);
DateTime WeekStart = e.Day.Date.AddDays(-1 * compensate);
DateTime WeekEnd = WeekStart.AddDays(6);
// If the start and end of the week does not have relevance to the current month
if (WeekStart.Month==WeekEnd.Month && e.Day.IsOtherMonth){
e.Cell.Text = "";
e.Cell.Height = 1; // fix for chrome. Visible=false leaves a blank row even when there are no <td>s in the <tr>
e.Cell.Visible = blnBrowserDoesntSupportsEmptyRow;
}
}
答案 1 :(得分:2)
如果您有SelectWeekText和SelectionMode =“DayWeek”或“DayWeekMonth”,那么您将遇到仍然显示隐藏周的周选择标记的问题。这是我用一点JQuery做的。
Sub cal_DayRender(ByVal sender As Object, ByVal e As DayRenderEventArgs)
If HideExtraWeek(e, If(Cal.FirstDayOfWeek = WebControls.FirstDayOfWeek.Default, Threading.Thread.CurrentThread.CurrentUICulture.DateTimeFormat.FirstDayOfWeek, Cal.FirstDayOfWeek)) Then
e.Cell.Style("display") = "none"
e.Cell.CssClass = "hiddenWeek"
Exit Sub
End If
'do other render stuff here
End Sub
Private Function HideExtraWeek(ByVal e As DayRenderEventArgs, ByVal startOfWeekDay As Integer) As Boolean
If e.Day.IsOtherMonth Then
'hide empty weeks, logic credited to Robert
Dim currDay As Integer = e.Day.Date.DayOfWeek
Dim weekStart As DateTime = e.Day.Date.AddDays(startOfWeekDay - currDay) 'first day of the week
Dim weekEnd As DateTime = weekStart.AddDays(6)
Return (weekStart.Month = weekEnd.Month) 'the entire week is part of the other month
End If
Return False
End Function
<script type="text/javascript">
$(document).ready(function() {
$("td.hiddenWeek").parent().hide();
}
</script>
答案 2 :(得分:1)
好的,我有另一个解决方案。经过一番调整。希望它有所帮助。它有更多的功能,但如果你的日历得到很多命中,它可能只节省1%的CPU :)。 确保将Calendar1_VisibleMonthChanged
事件附加到OnVisibleMonthChanged
private DateTime fromDate;
private DateTime toDate;
private Boolean blnBrowserDoesntSupportsEmptyRow = Request.Browser.Browser=="Chrome" ||
Request.Browser.Browser=="Safari";
protected void Page_Load(object sender, EventArgs e){
if (!Page.IsPostBack){
setFromToDates(new DateTime(DateTime.Today.Year, DateTime.Today.Month, 1));
}
}
protected void Calendar1_VisibleMonthChanged(object sender, MonthChangedEventArgs e){
setFromToDates(e.NewDate);
}
protected void setFromToDates(DateTime monthStart){
DayOfWeek dw = (DayOfWeek)Calendar1.FirstDayOfWeek;
if (dw == (DayOfWeek)7) dw = (DayOfWeek)0;
if (monthStart.DayOfWeek == dw){
this.fromDate = monthStart;// if 1st day of calendar is also 1st of the month. just set as is
}else{
int dayOfWeek = Convert.ToInt16(monthStart.DayOfWeek);
dayOfWeek = dayOfWeek == 0 ? 7 : dayOfWeek;
int compensate = dayOfWeek - Convert.ToInt16(dw);
this.fromDate = monthStart.AddDays(-1 * compensate);// set FromDate to the beggning day of the calendar. I.e may start from e.g. 25th of the previous month
}
this.toDate = monthStart.AddMonths(1);
if (this.toDate.DayOfWeek != dw)
{
int dayOfWeek = Convert.ToInt16(this.toDate.DayOfWeek);
dayOfWeek = dayOfWeek == 0 ? 7 : dayOfWeek;
int compensate = dayOfWeek - Convert.ToInt16(dw);
this.toDate=this.toDate.AddDays(7-compensate);
}
}
protected void Calendar1_DayRender(object sender, DayRenderEventArgs e){
// hide extra week
hideExtraWeek(sender, e);
}
// returns weather or not the given day is hidden
protected Boolean hideExtraWeek(object sender, DayRenderEventArgs e){
Boolean isVisibleDay = e.Day.Date >= this.fromDate && e.Day.Date < this.toDate;
// If the start and end of the week does not have relevance to the current month
if (!isVisibleDay){
e.Cell.Text = "";
e.Cell.Height = 1; // fix for chrome. Visible=false leaves a blank row even when there are no <td>s in the <tr>
e.Cell.Visible = blnBrowserDoesntSupportsEmptyRow;
}
return !isVisibleDay;
}
答案 3 :(得分:1)
<asp:Calendar ID="Calendar1" runat="server" CellPadding="1" CellSpacing="0"
Width="600px" FirstDayOfWeek="Monday" BorderColor="#a1a1a1"
BorderWidth="1" BorderStyle="None" ShowGridLines="True" ShowDescriptionAsToolTip="True"
NextPrevStyle-CssClass=""
Height="500px" OnDayRender="Calendar1_DayRender"
SelectionMode="None" NextMonthText="&gt;&gt;" PrevMonthText="&lt;&lt;">
<OtherMonthDayStyle BorderStyle="None" />
</asp:Calendar>
bool weekstart = false;
int[] longmonths = new int[] { 1, 3, 5, 7, 8, 10, 12 };// 31 days in a Month
int[] shortmonths = new int[] { 4, 6, 9, 11 };// 30 days in a Month
protected void Calendar1_DayRender(object sender, DayRenderEventArgs e)
{
if (e.Day.IsOtherMonth)
{
e.Cell.Text = String.Empty;
e.Cell.Height = 0;
if (e.Day.Date.DayOfWeek == DayOfWeek.Monday )//Monday is FirstDayOfWeek
{
if (shortmonths.Contains(e.Day.Date.Month) && e.Day.Date.Day == 24)
{
weekstart = true;
}
else if (longmonths.Contains(e.Day.Date.Month) && e.Day.Date.Day == 25)
{
weekstart = true;
}
}
if (weekstart)
{
e.Cell.Visible = false;
}
}
else if (!e.Day.IsOtherMonth)
{
weekstart = false;
}
}
答案 4 :(得分:1)
这是我使用的VB代码。必要时,它将删除顶部和底部的空行。它使用私有变量。此方法在Chrome或Safari中似乎没有任何问题。
Private _hideEmptyWeek As Boolean
Protected Sub Calendar1_DayRender(ByVal sender As Object, ByVal e As System.Web.UI.WebControls.DayRenderEventArgs) Handles Calendar1.DayRender
If e.Day.IsOtherMonth Then
'' Hide first week if empty
If e.Day.Date = e.Day.Date.AddDays(-e.Day.Date.Day + 1).AddMonths(1).AddDays(-7) Then ' If this date is a full week before next month
_hideEmptyWeek = True
End If
'' Hide last week if empty
If e.Day.Date.DayOfWeek = DayOfWeek.Sunday And e.Day.Date.Day < 7 Then ' If this is the first Sunday of next month
_hideEmptyWeek = True
End If
'' Hide cell if we are in an empty week
If _hideEmptyWeek = True Then
e.Cell.Visible = False
End If
Else
_hideEmptyWeek = False
End If
End Sub