MyORM的使用(四) 实现Winform下的分页

其实这篇跟MyOrm的关系并不太大,只是分页时使用了MyOrm的查询,不过Winform中分页也是比较实用的,也作为一个例子吧。

不知道为什么MS并没有考虑在Winform中提供分页的功能,虽然Winform不像Webform需要考虑减少数据量的交互,但是在数据量很大时或者网络条件不好时还是需要分页的。

在Winform中实现分页的麻烦主要在于MS没有给分页定义事件或设置,所以想要做到Webform那样易用比较困难。实现分页的一个简单的方法是,写一个控制页码的用户控件,在控件触发的页码更改事件里绑定DataSource。这样的缺点在于,实际使用的时候还是需要手动写一些代码,包括获取总记录数、分页查询。另外,Winform里有多种需要绑定数据的控件(虽然最常用的就是DataGridView),还有第三方控件或者自定义的控件,通用性不够好。

为了使用的方便,自己定义一个数据源的类比较好,继承BindingSource是个不错的选择。BindingSource具备了基本的功能,我们只需要再增加分页功能。

定义PagedBindingSource:

 

 

  1.     public class PagedBindingSource : BindingSource, INotifyPropertyChanged
  2.     {
  3.         protected const int DefaultPageSize = 20;
  4.         private bool _autoRefresh = true;
  5.         private int _totalCount;
  6.         private int _startIndex;
  7.         private int _pageSize = DefaultPageSize;
  8.         //private PropertyDescriptor _sortProperty;
  9.         //private ListSortDirection _sortDirection;
  10.         //public override bool SupportsSorting
  11.         //{
  12.         //    get { return true; }
  13.         //}
  14.         //public override void ApplySort(PropertyDescriptor property, ListSortDirection sort)
  15.         //{
  16.         //    if (property == _sortProperty) sort = _sortDirection ^ ListSortDirection.Descending;
  17.         //    _sortProperty = property;
  18.         //    _sortDirection = sort;
  19.         //    RefreshCurrentPage();
  20.         //}
  21.         //public override void RemoveSort()
  22.         //{
  23.         //    _sortProperty = null;
  24.         //    RefreshCurrentPage();
  25.         //}
  26.         //public override bool IsSorted
  27.         //{
  28.         //    get { return _sortProperty != null; }
  29.         //}
  30.         //public override PropertyDescriptor SortProperty
  31.         //{
  32.         //    get { return _sortProperty; }
  33.         //}
  34.         //public override ListSortDirection SortDirection
  35.         //{
  36.         //    get { return _sortDirection; }
  37.         //}
  38.         protected virtual object GetDataSource(int startIndex, int pageSize, PropertyDescriptor orderby, ListSortDirection direction)
  39.         {
  40.             if (PageChanged != null)
  41.             {
  42.                 PageChangedEventArgs arg = new PageChangedEventArgs(startIndex, pageSize, orderby, direction);
  43.                 PageChanged(this, arg);
  44.                 return arg.ReturnSource;
  45.             }
  46.             return null;
  47.         }
  48.         protected virtual int GetTotalCount()
  49.         {
  50.             if (CountNeeded != null)
  51.             {
  52.                 CountEventArgs arg = new CountEventArgs();
  53.                 CountNeeded(this, arg);
  54.                 return arg.TotalCount;
  55.             }
  56.             return 0;
  57.         }
  58.         [DefaultValue(true)]
  59.         public bool AutoRefresh
  60.         {
  61.             get { return _autoRefresh; }
  62.             set { _autoRefresh = value; }
  63.         }
  64.         [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
  65.         public int TotalCount
  66.         {
  67.             get { return _totalCount; }
  68.         }
  69.         [DefaultValue(DefaultPageSize)]
  70.         public int PageSize
  71.         {
  72.             get { return _pageSize; }
  73.             set
  74.             {
  75.                 if (_pageSize != value)
  76.                 {
  77.                     _pageSize = value;
  78.                     OnPropertyChanged("PageSize");
  79.                     if (AutoRefresh) RefreshCurrentPage();
  80.                 }
  81.             }
  82.         }
  83.         [DefaultValue(0)]
  84.         public int StartIndex
  85.         {
  86.             get { return _startIndex; }
  87.             set
  88.             {
  89.                 if (_startIndex != value)
  90.                 {
  91.                     _startIndex = value;
  92.                     OnPropertyChanged("StartIndex");
  93.                     if (AutoRefresh) RefreshCurrentPage();
  94.                 }
  95.             }
  96.         }
  97.         public override void Clear()
  98.         {
  99.             _totalCount = 0;
  100.             _startIndex = 0;
  101.             OnPropertyChanged(null);
  102.             DataSource = null;
  103.         }
  104.         public virtual void RefreshSource()
  105.         {
  106.             _totalCount = GetTotalCount();
  107.             _startIndex = 0;
  108.             OnPropertyChanged(null);
  109.             RefreshCurrentPage();
  110.         }
  111.         public virtual void RefreshCurrentPage()
  112.         {
  113.             DataSource = GetDataSource(StartIndex, PageSize, SortProperty, SortDirection);
  114.         }
  115.         protected void OnPropertyChanged(string propertyName)
  116.         {
  117.             if (PropertyChanged != null) PropertyChanged(thisnew PropertyChangedEventArgs(propertyName));
  118.         }
  119.         #region INotifyPropertyChanged Members
  120.         public event PropertyChangedEventHandler PropertyChanged;
  121.         public event PageChangedEventHandler PageChanged;
  122.         public event GetCountEventHandler CountNeeded;
  123.         #endregion
  124.     }
  125.     public delegate void PageChangedEventHandler(object sender, PageChangedEventArgs e);
  126.     public delegate void GetCountEventHandler(object sender, CountEventArgs e);
  127.     public class CountEventArgs : EventArgs
  128.     {
  129.         private int totalCount;
  130.         public int TotalCount
  131.         {
  132.             get { return totalCount; }
  133.             set { totalCount = value; }
  134.         }
  135.     }
  136.     public class PageChangedEventArgs : EventArgs
  137.     {
  138.         internal PageChangedEventArgs(int startIndex, int pageSize, PropertyDescriptor orderby, ListSortDirection direction)
  139.         {
  140.             this.startIndex = startIndex;
  141.             this.pageSize = pageSize;
  142.             this.orderby = orderby;
  143.             this.direction = direction;
  144.         }
  145.         private int startIndex;
  146.         public int StartIndex
  147.         {
  148.             get { return startIndex; }
  149.         }
  150.         private int pageSize;
  151.         public int PageSize
  152.         {
  153.             get { return pageSize; }
  154.         }
  155.         private PropertyDescriptor orderby;
  156.         public PropertyDescriptor Orderby
  157.         {
  158.             get { return orderby; }
  159.         }
  160.         private ListSortDirection direction;
  161.         public ListSortDirection Direction
  162.         {
  163.             get { return direction; }
  164.         }
  165.         private object returnSource;
  166.         public object ReturnSource
  167.         {
  168.             get { return returnSource; }
  169.             set { returnSource = value; }
  170.         }
  171.     }

注意到其中有2个事件PageChanged和CountNeeded,一个是分页的查询,一个是获取记录总数。底层的实现不同,这2个事件的实现也不同。比如在MyOrm里提供了分页查询的SearchSection方法,对ProductsView的分页查询可能像这样:

  1.         private void pagedBindingSource1_CountNeeded(object sender, CountEventArgs e)
  2.         {
  3.             e.TotalCount = new ProductsViewDAO().Count(SearchCondition);
  4.         }
  5.         private void pagedBindingSource1_PageChanged(object sender, PageChangedEventArgs e)
  6.         {
  7.             e.ReturnSource = new ProductsViewDAO().SearchSection(SearchCondition, e.StartIndex, e.PageSize, e.Orderby == null ? null : e.Orderby.Name, e.Direction);
  8.         }

另外还可以继承PagedBindingSource,自己实现GetTotalCount方法和GetDataSource方法,可以不需要在事件中添加代码。例如可以为MyOrm定制一个ConditionPagedSource:

  1.    public class ConditionPagedSource : PagedBindingSource
  2.     {
  3.         private IObjectViewDAO _objectViewDAO;
  4.         private Condition _condition;
  5.         private Type _objectType;
  6.         protected override int GetTotalCount()
  7.         {
  8.             return ObjectViewDAO.Count(Condition);
  9.         }
  10.         protected override object GetDataSource(int startIndex, int pageSize, PropertyDescriptor orderby, ListSortDirection direction)
  11.         {
  12.             return ObjectViewDAO.SearchSection(Condition, startIndex, pageSize, orderby == null ? null : orderby.Name, direction);
  13.         }
  14.         [DefaultValue(null), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
  15.         public Condition Condition
  16.         {
  17.             get
  18.             {
  19.                 return _condition;
  20.             }
  21.             set
  22.             {
  23.                 _condition = value;
  24.                 RefreshSource();
  25.             }
  26.         }
  27.         [DefaultValue(null), TypeConverter(typeof(TypeTypeConverter)), RefreshProperties(RefreshProperties.Repaint)]
  28.         public Type ObjectType
  29.         {
  30.             get { return _objectType; }
  31.             set
  32.             {
  33.                 if (_objectType != value)
  34.                 {
  35.                     _objectType = value;
  36.                     DataSource = value;
  37.                 }
  38.             }
  39.         }
  40.         [Browsable(false), DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
  41.         public IObjectViewDAO ObjectViewDAO
  42.         {
  43.             get { return _objectViewDAO == null ? DAOFactoryUtil.GetObjectViewDAO(NorthwindFactory.DAOFactory, ObjectType) : _objectViewDAO; }
  44.             set { _objectViewDAO = value; }
  45.         }
  46.     }

这样就完成了分页查询的数据源。ObjectType使用了TypeTypeConverter来实现设计时支持,其实就是可以把输入的类型名称转化为Type。再在ObjectType的内部DataSource = value也是为了在设计时能自动添加数据列,Winform中添加Object类型的DataSource其实就是BindingSource绑定到对象的Type,这里也照样。

TypeTypeConverter定义:

  1. public class TypeTypeConverter : TypeConverter
  2.     {
  3.         public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType)
  4.         {
  5.             return sourceType == typeof(string) || base.CanConvertFrom(context, sourceType);
  6.         }
  7.         public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType)
  8.         {
  9.             return destinationType == typeof(string) || base.CanConvertTo(context, destinationType);
  10.         }
  11.         public override object ConvertFrom(ITypeDescriptorContext context, System.Globalization.CultureInfo culture, object value)
  12.         {
  13.             if (value is stringreturn Type.GetType((string)value);
  14.             return base.ConvertFrom(context, culture, value);
  15.         }
  16.         public override object ConvertTo(ITypeDescriptorContext context, System.Globalization.CultureInfo culture, object value, Type destinationType)
  17.         {
  18.             if (destinationType == typeof(string)) return ((Type)value).FullName;
  19.             return base.ConvertTo(context, culture, value, destinationType);
  20.         }
  21.     }

分页数据源有了,但是在界面上还需要提供一个控件来控制翻页。也许有现成的控件可以用,我这里仿造BindingNavigator提供了一个PageNavigator:

  1. public class PageNavigator : BindingNavigator
  2.     {
  3.         private ToolStripItem _currentPageItem;
  4.         private ToolStripItem _moveFirstPageItem;
  5.         private ToolStripItem _moveLastPageItem;
  6.         private ToolStripItem _moveNextPageItem;
  7.         private ToolStripItem _movePreviousPageItem;
  8.         private string _pageCountFormat;
  9.         private ToolStripItem _pageCountItem;
  10.         private PagedBindingSource _pageSource;
  11.         public override void AddStandardItems()
  12.         {
  13.             MoveFirstPageItem = new ToolStripButton();
  14.             MovePreviousPageItem = new ToolStripButton();
  15.             MoveNextPageItem = new ToolStripButton();
  16.             MoveLastPageItem = new ToolStripButton();
  17.             CurrentPageItem = new ToolStripTextBox();
  18.             PageCountItem = new ToolStripLabel();
  19.             ToolStripSeparator separator = new ToolStripSeparator();
  20.             ToolStripSeparator separator2 = new ToolStripSeparator();
  21.             char ch = string.IsNullOrEmpty(base.Name) || char.IsLower(base.Name[0]) ? 'p' : 'P';
  22.             MoveFirstPageItem.Name = ch + "ageNavigatorMoveFirstPageItem";
  23.             MovePreviousPageItem.Name = ch + "ageNavigatorMovePreviousPageItem";
  24.             MoveNextPageItem.Name = ch + "ageNavigatorMoveNextPageItem";
  25.             MoveLastPageItem.Name = ch + "ageNavigatorMoveLastPageItem";
  26.             CurrentPageItem.Name = ch + "ageNavigatorCurrentPageItem";
  27.             PageCountItem.Name = ch + "ageNavigatorPageCountItem";
  28.             separator.Name = ch + "ageNavigatorSeparator";
  29.             separator2.Name = ch + "ageNavigatorSeparator";
  30.             MoveFirstPageItem.Text = "PageNavigatorMoveFirstPageItemText";
  31.             MovePreviousPageItem.Text = "PageNavigatorMovePreviousPageItemText";
  32.             MoveNextPageItem.Text = "PageNavigatorMoveNextPageItemText";
  33.             MoveLastPageItem.Text = "PageNavigatorMoveLastPageItemText";
  34.             PageCountItem.ToolTipText = "PageNavigatorPageCountItemTip";
  35.             CurrentPageItem.ToolTipText = "PageNavigatorCurrentPageItemTip";
  36.             PageCountItem.AutoToolTip = false;
  37.             PageCountItem.Text = "/0";
  38.             CurrentPageItem.AutoToolTip = false;
  39.             CurrentPageItem.Text = "0";
  40.             CurrentPageItem.AccessibleName = "PageNavigatorPositionAccessibleName";
  41.             PageCountFormat = "/{0}";
  42.             Bitmap bitmap1 = new Bitmap(typeof(BindingNavigator), "BindingNavigator.MoveFirst.bmp");
  43.             Bitmap bitmap2 = new Bitmap(typeof(BindingNavigator), "BindingNavigator.MovePrevious.bmp");
  44.             Bitmap bitmap3 = new Bitmap(typeof(BindingNavigator), "BindingNavigator.MoveNext.bmp");
  45.             Bitmap bitmap4 = new Bitmap(typeof(BindingNavigator), "BindingNavigator.MoveLast.bmp");
  46.             bitmap1.MakeTransparent(Color.Magenta);
  47.             bitmap2.MakeTransparent(Color.Magenta);
  48.             bitmap3.MakeTransparent(Color.Magenta);
  49.             bitmap4.MakeTransparent(Color.Magenta);
  50.             MoveFirstPageItem.Image = bitmap1;
  51.             MovePreviousPageItem.Image = bitmap2;
  52.             MoveNextPageItem.Image = bitmap3;
  53.             MoveLastPageItem.Image = bitmap4;
  54.             MoveFirstPageItem.RightToLeftAutoMirrorImage = true;
  55.             MovePreviousPageItem.RightToLeftAutoMirrorImage = true;
  56.             MoveNextPageItem.RightToLeftAutoMirrorImage = true;
  57.             MoveLastPageItem.RightToLeftAutoMirrorImage = true;
  58.             MoveFirstPageItem.DisplayStyle = ToolStripItemDisplayStyle.Image;
  59.             MovePreviousPageItem.DisplayStyle = ToolStripItemDisplayStyle.Image;
  60.             MoveNextPageItem.DisplayStyle = ToolStripItemDisplayStyle.Image;
  61.             MoveLastPageItem.DisplayStyle = ToolStripItemDisplayStyle.Image;
  62.             CurrentPageItem.AutoSize = false;
  63.             CurrentPageItem.Width = 50;
  64.             Items.AddRange(new ToolStripItem[] { MoveFirstPageItem, MovePreviousPageItem, separator, CurrentPageItem, PageCountItem, separator2, MoveNextPageItem, MoveLastPageItem });
  65.         }
  66.         private void AcceptNewPage()
  67.         {
  68.             if (CurrentPageItem != null && _pageSource != null)
  69.             {
  70.                 int currentPage;
  71.                 int.TryParse(CurrentPageItem.Text, out currentPage);
  72.                 currentPage--;
  73.                 if (currentPage >= 0)
  74.                 {
  75.                     int startIndex = currentPage * _pageSource.PageSize;
  76.                     if (((((startIndex != _pageSource.StartIndex) && (startIndex >= 0)) && (startIndex < _pageSource.TotalCount)) ? 1 : 0) != 0)
  77.                     {
  78.                         _pageSource.StartIndex = startIndex;
  79.                         if (!_pageSource.AutoRefresh) _pageSource.RefreshCurrentPage();
  80.                         return;
  81.                     }
  82.                 }
  83.                 CancelNewPage();
  84.             }
  85.         }
  86.         private void CancelNewPage()
  87.         {
  88.             RefreshPageProperty();
  89.         }
  90.         private void OnCurrentPageKey(object sender, KeyEventArgs e)
  91.         {
  92.             Keys keyCode = e.KeyCode;
  93.             if (keyCode != Keys.Return)
  94.             {
  95.                 if (keyCode != Keys.Escape)
  96.                 {
  97.                     return;
  98.                 }
  99.             }
  100.             else
  101.             {
  102.                 AcceptNewPage();
  103.                 return;
  104.             }
  105.             CancelNewPage();
  106.         }
  107.         private void OnCurrentPageLostFocus(object sender, EventArgs e)
  108.         {
  109.             AcceptNewPage();
  110.         }
  111.         private void OnMoveFirstPage(object sender, EventArgs e)
  112.         {
  113.             if (Validate() && _pageSource != null)
  114.             {
  115.                 _pageSource.StartIndex = 0;
  116.                 if (!_pageSource.AutoRefresh) _pageSource.RefreshCurrentPage();
  117.             }
  118.         }
  119.         private void OnMoveLastPage(object sender, EventArgs e)
  120.         {
  121.             if (Validate() && _pageSource != null)
  122.             {
  123.                 _pageSource.StartIndex = (_pageSource.TotalCount / _pageSource.PageSize) * _pageSource.PageSize;
  124.                 if (!_pageSource.AutoRefresh) _pageSource.RefreshCurrentPage();
  125.             }
  126.         }
  127.         private void OnMoveNextPage(object sender, EventArgs e)
  128.         {
  129.             if (Validate() && _pageSource != null)
  130.             {
  131.                 _pageSource.StartIndex += _pageSource.PageSize;
  132.                 if (!_pageSource.AutoRefresh) _pageSource.RefreshCurrentPage();
  133.             }
  134.         }
  135.         private void OnMovePreviousPage(object sender, EventArgs e)
  136.         {
  137.             if (Validate() && _pageSource != null)
  138.             {
  139.                 _pageSource.StartIndex -= _pageSource.PageSize;
  140.                 if (!_pageSource.AutoRefresh) _pageSource.RefreshCurrentPage();
  141.             }
  142.         }
  143.         public void RefreshPageProperty()
  144.         {
  145.             if (_pageSource != null)
  146.             {
  147.                 int pageSize = _pageSource.PageSize;
  148.                 if (pageSize > 0)
  149.                 {
  150.                     int currentPage = _pageSource.StartIndex / pageSize;
  151.                     int totalPage = ((_pageSource.TotalCount + pageSize) - 1) / pageSize;
  152.                     PageCountItem.Text = string.Format(PageCountFormat, totalPage);
  153.                     if (totalPage == 0)
  154.                     {
  155.                         CurrentPageItem.Text = "0";
  156.                     }
  157.                     else
  158.                     {
  159.                         CurrentPageItem.Text = Convert.ToString(currentPage + 1);
  160.                     }
  161.                     if (currentPage <= 0)
  162.                     {
  163.                         MoveFirstPageItem.Enabled = false;
  164.                         MovePreviousPageItem.Enabled = false;
  165.                     }
  166.                     else
  167.                     {
  168.                         MoveFirstPageItem.Enabled = true;
  169.                         MovePreviousPageItem.Enabled = true;
  170.                     }
  171.                     if (currentPage >= (totalPage - 1))
  172.                     {
  173.                         MoveLastPageItem.Enabled = false;
  174.                         MoveNextPageItem.Enabled = false;
  175.                     }
  176.                     else
  177.                     {
  178.                         MoveLastPageItem.Enabled = true;
  179.                         MoveNextPageItem.Enabled = true;
  180.                     }
  181.                 }
  182.             }
  183.         }
  184.         private void WireUpButton(ref ToolStripItem oldButton, ToolStripItem newButton, EventHandler clickHandler)
  185.         {
  186.             if (oldButton != newButton)
  187.             {
  188.                 if (oldButton != null)
  189.                 {
  190.                     oldButton.Click -= clickHandler;
  191.                 }
  192.                 if (newButton != null)
  193.                 {
  194.                     newButton.Click += clickHandler;
  195.                 }
  196.                 oldButton = newButton;
  197.             }
  198.         }
  199.         private void WireUpTextBox(ref ToolStripItem oldTextBox, ToolStripItem newTextBox, KeyEventHandler keyUpHandler, EventHandler lostFocusHandler)
  200.         {
  201.             if (oldTextBox != newTextBox)
  202.             {
  203.                 ToolStripControlHost host = oldTextBox as ToolStripControlHost;
  204.                 ToolStripControlHost host2 = newTextBox as ToolStripControlHost;
  205.                 if (host != null)
  206.                 {
  207.                     host.KeyUp -= keyUpHandler;
  208.                     host.LostFocus -= lostFocusHandler;
  209.                 }
  210.                 if (host2 != null)
  211.                 {
  212.                     host2.KeyUp += keyUpHandler;
  213.                     host2.LostFocus += lostFocusHandler;
  214.                 }
  215.                 oldTextBox = newTextBox;
  216.             }
  217.         }
  218.         public ToolStripItem CurrentPageItem
  219.         {
  220.             get
  221.             {
  222.                 return _currentPageItem;
  223.             }
  224.             set
  225.             {
  226.                 WireUpTextBox(ref _currentPageItem, value, new KeyEventHandler(OnCurrentPageKey), new EventHandler(OnCurrentPageLostFocus));
  227.             }
  228.         }
  229.         public ToolStripItem MoveFirstPageItem
  230.         {
  231.             get
  232.             {
  233.                 return _moveFirstPageItem;
  234.             }
  235.             set
  236.             {
  237.                 WireUpButton(ref _moveFirstPageItem, value, new EventHandler(OnMoveFirstPage));
  238.             }
  239.         }
  240.         public ToolStripItem MoveLastPageItem
  241.         {
  242.             get
  243.             {
  244.                 return _moveLastPageItem;
  245.             }
  246.             set
  247.             {
  248.                 WireUpButton(ref _moveLastPageItem, value, new EventHandler(OnMoveLastPage));
  249.             }
  250.         }
  251.         public ToolStripItem MoveNextPageItem
  252.         {
  253.             get
  254.             {
  255.                 return _moveNextPageItem;
  256.             }
  257.             set
  258.             {
  259.                 WireUpButton(ref _moveNextPageItem, value, new EventHandler(OnMoveNextPage));
  260.             }
  261.         }
  262.         public ToolStripItem MovePreviousPageItem
  263.         {
  264.             get
  265.             {
  266.                 return _movePreviousPageItem;
  267.             }
  268.             set
  269.             {
  270.                 WireUpButton(ref _movePreviousPageItem, value, new EventHandler(OnMovePreviousPage));
  271.             }
  272.         }
  273.         public string PageCountFormat
  274.         {
  275.             get
  276.             {
  277.                 return _pageCountFormat;
  278.             }
  279.             set
  280.             {
  281.                 if (!(_pageCountFormat == value))
  282.                 {
  283.                     _pageCountFormat = value;
  284.                 }
  285.             }
  286.         }
  287.         public ToolStripItem PageCountItem
  288.         {
  289.             get
  290.             {
  291.                 return _pageCountItem;
  292.             }
  293.             set
  294.             {
  295.                 _pageCountItem = value;
  296.             }
  297.         }
  298.         public PagedBindingSource PageSource
  299.         {
  300.             get
  301.             {
  302.                 return _pageSource;
  303.             }
  304.             set
  305.             {
  306.                 if (value != _pageSource)
  307.                 {
  308.                     if (_pageSource != null)
  309.                     {
  310.                         _pageSource.PropertyChanged -= new PropertyChangedEventHandler(PageSource_PropertyChanged);
  311.                     }
  312.                     _pageSource = value;
  313.                     RefreshPageProperty();
  314.                     if (_pageSource != null)
  315.                     {
  316.                         _pageSource.PropertyChanged += new PropertyChangedEventHandler(PageSource_PropertyChanged);
  317.                     }
  318.                 }
  319.             }
  320.         }
  321.         void PageSource_PropertyChanged(object sender, PropertyChangedEventArgs e)
  322.         {
  323.             RefreshPageProperty();
  324.         }
  325.     }

定义比较繁琐,继承BindingNavigator是为了获得良好的设计时支持,自己写个设计控件实在不是个小工程,所以虽然别扭就凑合一下吧。

这是PagedBindingSource、DataGridView、PageNavigator之间的关系图:

 

因为DataGridView不提供分页的功能,只能通过PageNavigator控制PagedBindingSource的翻页。

在设计界面中设定PageNavigator的PageSource和DataGridView的DataSource为创建好的ConditionPagedSource后,分页查询就完成了,很简单。(DataGridView在设置DataSource后为自动把AutoGenerateColumns设为false,有可能造成没有列显示出来,需要把InitializeComponent方法中的dataGridView1.AutoGenerateColumns = false;删掉再编辑界面就可以,一个比较烦人的问题)

最后的效果:

示例代码可以到CodePlex/MyOrm下载。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值