AutoColumnSizeGrid

using System;
using System.Collections;
using System.ComponentModel;
using System.Diagnostics;
using System.Drawing;
using System.Runtime.InteropServices;
using System.Windows.Forms;

namespace AutoColumnSize {

 /// <summary>
 /// Windows Forms derived DataGrid that will keep the grid height
 /// to row-height increments so resizing vertically will snap between
 /// row-heights & never display a partial row. Rows will either be
 /// fully displayed or not shown. This grid will also resize column
 /// widths so that the grid will not display a horizontal scrollbar.
 /// This grid will take into account whether or not the vertical
 /// scrollbar is visible when it makes the column widths change.
 /// </summary>
 public class Grid : DataGrid {

  private const int MINIMUM_COLUMN_WIDTH_DEFAULT = 10;

  private static readonly object EventMinimumColumnWidthChanged = new object();

  private int  minimumColumnWidth = MINIMUM_COLUMN_WIDTH_DEFAULT;
  private bool inEnsureTableStyleExists;

  /// <summary>
  /// Initializes a new instance of the AutoGrid.
  /// </summary>
  public Grid() {

   // we have to add an event handler for the OnNavigate event
   // because the DataGrid didn't make its own OnNavigate routine
   // that raises the event virtual like most property event
   // raising routines...
   //
   this.Navigate += new NavigateEventHandler(OnNavigate);
  }

  /// <summary>
  /// tells how many pixels the border of this grid will be
  /// </summary>
  private int BorderWidth {
   get {
    if (BorderStyle == BorderStyle.Fixed3D) {
     return SystemInformation.Border3DSize.Width;
    }
    else if (BorderStyle == BorderStyle.FixedSingle) {
     return 2;
    }
    else {
     return 0;
    }
   }
  }

  /// <summary>
  /// gets or sets the data member that the grid is displaying data for.
  /// </summary>
  public new string DataMember {
   get {
    return base.DataMember;
   }
   set {
    EnsureTableStyleExists(DataSource, value);
    base.DataMember = value;
   }
  }

  /// <summary>
  /// gets or sets the data source that the grid is displaying data for.
  /// </summary>
  public new object DataSource {
   get {
    return base.DataSource;
   }
   set {
    EnsureTableStyleExists(value, DataMember);
    base.DataSource = value;
   }
  }

  /// <summary>
  /// width at which columns stop shrinking as the grid is resized. if
  /// the grid's width is not sufficient enough to display all columns
  /// with a width of MinimumColumnWidth, the grid will introduce a
  /// horizontal scrollbar.
  /// </summary>
  [DefaultValue(MINIMUM_COLUMN_WIDTH_DEFAULT)]
  public int MinimumColumnWidth {
   get {
    return minimumColumnWidth;
   }
   set {
    if (minimumColumnWidth != value) {
     minimumColumnWidth = value;
     PerformLayout(this, "MinimumColumnWidth");
     OnMinimumColumnWidthChanged(EventArgs.Empty);
    }
   }
  }

  /// <summary>
  /// event handler for when the MinimumColumnWidth changes
  /// </summary>
  public event EventHandler MinimumColumnWidthChanged {
   add {
    Events.AddHandler(EventMinimumColumnWidthChanged, value);
   }
   remove {
    Events.RemoveHandler(EventMinimumColumnWidthChanged, value);
   }

  }

  /// <summary>
  /// overridden to give this grid a chance to remove its event handler
  /// from its own OnNavigate event.
  /// </summary>
  protected override void Dispose(bool disposing) {
   if (disposing) {
    this.Navigate -= new NavigateEventHandler(OnNavigate);
   }
   base.Dispose(disposing);
  }


  /// <summary>
  /// checks to see if there is an existing tableStyle for the
  /// given CurrencyManager, creating one if it does not exist yet.
  /// </summary>
  private void EnsureTableStyleExists(object dataSource, string dataMember) {

   // requesting the BindingContext below will cause us to reenter
   // this function, and we only want to do this once.
   //
   if (inEnsureTableStyleExists) {
    return;
   }

   inEnsureTableStyleExists = true;
   try {
    if ((dataSource != null) && (BindingContext != null) && !(dataSource == Convert.DBNull)) {

     CurrencyManager  listManager = (CurrencyManager)BindingContext[dataSource, dataMember];
     Debug.Assert(listManager != null, "BindingContext didn't give us a CurrencyManager?");

     if (listManager != null) {
      string  listName = GetListName(dataSource, listManager);

      // if we don't have a tableStyle for this list, then create one
      // and add it to our collection.
      //
      if (!TableStyles.Contains(listName)) {

       DataGridTableStyle  tableStyle = new DataGridTableStyle(listManager);
       TableStyles.Add(tableStyle);
      }
     }
    }
   }
   finally {
    inEnsureTableStyleExists = false;
   }
  }

  /// <summary>
  /// method that will use the current DataSource and DataMember to determine
  /// what tableStyle the grid is currently displaying.
  /// </summary>
  private DataGridTableStyle GetCurrentTableStyle() {

   DataGridTableStyle currentTableStyle = null;

   if (DataSource != null && BindingContext != null) {
    string    listName = GetListName(DataSource, (CurrencyManager)BindingContext[DataSource, DataMember]);

    if (TableStyles.Contains(listName)) {
     currentTableStyle = TableStyles[listName];
    }
   }

   return currentTableStyle;
  }

  /// <summary>
  /// returns the width that columns can take up on this
  /// grid with the given tableStyle
  /// </summary>
  private int GetFullWidth(DataGridTableStyle tableStyle) {

   return Width
    - 2*BorderWidth
    - (VertScrollBar.Visible ? VertScrollBar.Width : 0)
    - (tableStyle.RowHeadersVisible ? tableStyle.RowHeaderWidth: 0);  
  }

  /// <summary>
  /// method that duplicates internal code in WinForms.CurrencyManager because
  /// almost nothing that DataBinding does is publicly exposed...
  /// </summary>
  private string GetListName(object dataSource, CurrencyManager listManager) {

   string  name = null;
   IList  list = listManager.List;

   // if it's a typed-list, then just return the typed-list's name
   //
   if (list != null && list is ITypedList) {
    name = ((ITypedList)listManager.List).GetListName(null);
   }
   else {
    Type  finalType = null;
    object  tempList = dataSource;

    // otherwise we have to track down what the CurrencyManager
    // stores as its "finalType" -- another internal to WinForms
    // concept that we can't access directly.
    //
    if (tempList is Array) {
     finalType = dataSource.GetType();
     tempList = (Array)tempList;
    }

    if (tempList is IListSource) {
     tempList = ((IListSource)tempList).GetList();
    }

    if (tempList is IList) {
     if (finalType == null) {
      finalType = tempList.GetType();
     }
    }

    if (finalType != null) {
     name = finalType.Name;
    }
   }

   if (name == null) {
    name = String.Empty;
   }

   return name;
  }

  /// <summary>
  /// </summary>
  protected override void OnBindingContextChanged(EventArgs e) {

   // if our binding-context changed, then we need to ensure
   // that we have a TableStyle for the new context
   //
   EnsureTableStyleExists(DataSource, DataMember);
   base.OnBindingContextChanged(e);
  }

  /// <summary>
  /// overridden to
  /// </summary>
  protected override void OnLayout(LayoutEventArgs levent) {

   // first, give the DataGrid a chance to lay things out
   // so we can tell if a vertical scrollbar is present
   //
   base.OnLayout(levent);

   // now that the DataGrid is done, we need to figure out if
   // there is any column resize work to be done
   //
   DataGridTableStyle tableStyle = GetCurrentTableStyle();
   if (tableStyle != null) {

    int  fullWidth = GetFullWidth(tableStyle);

    if (fullWidth > 0) {

     bool vertScrollVisible;
     int  sizableColumnCount;

     // it's possible that we will begin with both a vertical & horizontal scrollbar
     // after the initial base.OnLayout call and that resizing the columns to make
     // the horizontal scrollbar disappear will also force the vertical scrollbar
     // to disappear too, and we wouldn't want to leave the area where the
     // vertical scrollbar was as blank client area, so we set this up as a
     // loop in case we need to lay columns out twice.
     //
     do {
      vertScrollVisible = VertScrollBar.Visible;
      sizableColumnCount = 0;

      foreach (DataGridColumnStyle column in tableStyle.GridColumnStyles) {

       // checkbox columns should not change size so we simply
       // remove the size of that column from the overall width
       // we're splitting the columns up across
       //
       if (column is DataGridBoolColumn) {
        fullWidth -= column.Width;
       }
       else {
        sizableColumnCount++;
       }
      }

      if (sizableColumnCount > 0) {

       int  columnWidth = fullWidth / sizableColumnCount;
       int  remainder = fullWidth % sizableColumnCount;
       int  index = 0;
       bool resized = false;

       foreach (DataGridColumnStyle column in tableStyle.GridColumnStyles) {

        if (!(column is DataGridBoolColumn)) {

         int  width = (index < remainder
          ? Math.Max(columnWidth + 1, MinimumColumnWidth)
          : Math.Max(columnWidth, MinimumColumnWidth)); 

         if (column.Width != width) {
          column.Width = width;
          resized = true;
         }

         index++;
        }
       }

       // if we changed any column widths, we need to give the DataGrid
       // a chance to lay out again so we can see if scrollbar visiblity
       // changed before returning
       //
       if (resized) {
        base.OnLayout(levent);
       }
      }

      fullWidth = GetFullWidth(tableStyle);

     } while (vertScrollVisible != VertScrollBar.Visible);
    }
   }
  }

  /// <summary>
  /// called when MinimumColumnWidth changes for this grid.
  /// </summary>
  protected virtual void OnMinimumColumnWidthChanged(EventArgs e) {
   EventHandler eh = Events[EventMinimumColumnWidthChanged] as EventHandler;
   if (eh != null) {
    eh(this, e);
   }
  }


  /// <summary>
  /// called when user navigates within the grid to a sub-table
  /// </summary>
  private void OnNavigate(object sender, NavigateEventArgs ne) {

   // first, make sure a tableStyle exists for this new
   // DataSource/DataMember
   //
   EnsureTableStyleExists(DataSource, DataMember);

   // then simply force a layout so we can rearrange column widths
   //
   OnLayout(new LayoutEventArgs(null, null));
  }

  /// <summary>
  /// obscures WinForms DataGrid method so we can ensure that
  /// a TableStyle exists before calling the base class version.
  /// </summary>
  public new void SetDataBinding(object dataSource, string dataMember) {

   EnsureTableStyleExists(dataSource, dataMember);
   base.SetDataBinding(dataSource, dataMember);
  }
 }
}

 

USAGE:

private AutoColumnSize.Grid dataGrid1;

            SqlConnection nwindConn = new SqlConnection("Data Source=localhost;Integrated Security=SSPI;Initial Catalog=northwind");

            SqlCommand selectCMD = new SqlCommand("SELECT CustomerID, CompanyName FROM Customers", nwindConn);
            selectCMD.CommandTimeout = 30;

            SqlDataAdapter custDA = new SqlDataAdapter();
            custDA.SelectCommand = selectCMD;

            nwindConn.Open();

            DataSet custDS = new DataSet();
            custDA.Fill(custDS, "Customers");

            nwindConn.Close();
            if (custDS.Tables.Count > 0)
            {
                dataGrid1.DataSource = custDS.Tables[0];
            }

源码来自:https://pan.quark.cn/s/a4b39357ea24 ### 操作指南:洗衣机使用方法详解#### 1. 启动与水量设定- **使用方法**:使用者必须首先按下洗衣设备上的“启动”按键,同时依据衣物数量设定相应的“水量选择”旋钮(高、中或低水量)。这一步骤是洗衣机运行程序的开端。- **运作机制**:一旦“启动”按键被触发,洗衣设备内部的控制系统便会启动,通过感应器识别水量选择旋钮的位置,进而确定所需的水量高度。- **技术执行**:在当代洗衣设备中,这一流程一般由微处理器掌管,借助电磁阀调控进水量,直至达到指定的高度。#### 2. 进水过程- **使用说明**:启动后,洗衣设备开始进水,直至达到所选的水位(高、中或低)。- **技术参数**:水量的监测通常采用浮子式水量控制器或压力感应器来实现。当水位达到预定值时,进水阀会自动关闭,停止进水。- **使用提醒**:务必确保水龙头已开启,并检查水管连接是否牢固,以防止漏水。#### 3. 清洗过程- **使用步骤**:2秒后,洗衣设备进入清洗环节。在此期间,滚筒会执行一系列正转和反转的动作: - 正转25秒 - 暂停3秒 - 反转25秒 - 再次暂停3秒- **重复次数**:这一系列动作将重复执行5次,总耗时为280秒。- **技术关键**:清洗环节通过电机驱动滚筒旋转,利用水流冲击力和洗衣液的化学效果,清除衣物上的污垢。#### 4. 排水与甩干- **使用步骤**:清洗结束后,洗衣设备会自动进行排水,将污水排出,然后进入甩干阶段,甩干时间为30秒。- **技术应用**:排水是通过泵将水抽出洗衣设备;甩干则是通过高速旋转滚筒,利用离心力去除衣物上的水分。- **使用提醒**:...
代码下载地址: https://pan.quark.cn/s/c289368a8f5c 在安卓应用开发领域,构建一个高效且用户友好的聊天系统是一项核心任务。 为了协助开发者们迅速达成这一目标,本文将分析几种常见的安卓聊天框架,并深入说明它们的功能特性、应用方法及主要优势。 1. **环信(Easemob)** 环信是一个专为移动应用打造的即时通讯软件开发套件,涵盖了文本、图片、语音、视频等多种消息形式。 通过整合环信SDK,开发者能够迅速构建自身的聊天平台。 环信支持消息内容的个性化定制,能够应对各种复杂的应用场景,并提供多样的API接口供开发者使用。 2. **融云(RongCloud)** 融云作为国内领先的IM云服务企业,提供了全面的聊天解决方案,包括一对一交流、多人群聊、聊天空间等。 融云的突出之处在于其稳定运行和高并发处理性能,以及功能完备的后台管理工具,便于开发者执行用户管理、消息发布等操作。 再者,融云支持多种消息格式,如位置信息、文件传输、表情符号等,显著增强了用户聊天体验。 3. **Firebase Cloud Messaging(FCM)** FCM由Google提供的云端消息传递服务,可达成安卓设备与服务器之间的即时数据交换。 虽然FCM主要应用于消息推送,但配合Firebase Realtime Database或Firestore数据库,开发者可以开发基础的聊天软件。 FCM的显著优势在于其全球性的推送网络,保障了消息能够及时且精确地传输至用户。 4. **JMessage(极光推送)** 极光推送是一款提供消息发布服务的软件开发工具包,同时具备基础的即时通讯能力。 除了常规的文字、图片信息外,极光推送还支持个性化消息,使得开发者能够实现更为复杂的聊天功能。 此...
【轴承故障诊断】基于融合鱼鹰和柯西变异的麻雀优化算法OCSSA-VMD-CNN-BILSTM轴承诊断研究【西储大学数据】(Matlab代码实现)内容概要:本文提出了一种基于融合鱼鹰和柯西变异的麻雀优化算法(OCSSA)优化变分模态分解(VMD)参数,并结合卷积神经网络(CNN)与双向长短期记忆网络(BiLSTM)的OCSSA-VMD-CNN-BiLSTM模型,用于轴承故障诊断。该方法利用西储大学公开的轴承故障数据集进行实验验证,通过OCSSA优化VMD的分解层数和惩罚因子,有效提升了信号去噪与特征提取能力;随后将分解得到的模态分量输入CNN-BiLSTM网络进行自动特征学习与分类识别,实现了高精度的故障状态判别。研究展示了智能优化算法与深度学习在机械故障诊断领域的深度融合应用。; 适合人群:具备一定信号处理、机器学习及MATLAB编程基础的研究生、科研人员及从事工业设备故障诊断的工程技术人员。; 使用场景及目标:①应用于旋转机械的轴承早期故障诊断与状态监测;②为复杂工况下的振动信号分析提供有效的去噪与特征提取方案;③推动智能优化算法与深度学习模型在工业大数据分析中的集成应用研究。; 阅读建议:建议读者结合提供的Matlab代码实现,复现实验流程,重点关注OCSSA算法的改进机制、VMD参数优化过程以及CNN-BiLSTM网络结构设计,深入理解各模块协同工作的原理,便于进一步拓展至其他设备的故障诊断任务中。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值