WPF报表自定义通用可筛选列头-WPF特工队内部资料

 

 由于项目需要制作一个可通用的报表多行标题,且可实现各种类型的内容显示,包括文本、输入框、下拉框、多选框等(自定的显示内容可自行扩展),并支持参数绑定转换,效果如下: 

  WPF报表自定义通用可筛选列头-WPF特工队内部资料_第1张图片

  WPF报表自定义通用可筛选列头-WPF特工队内部资料_第2张图片

  

  源码结构WPF报表自定义通用可筛选列头-WPF特工队内部资料_第3张图片

 

ColumnItem类:对列的宽度,对象方式,显示类型,绑定名称,converter资源进行设置

    /// 
    /// 描述  动态列项目,用于在后台构建多列标题绑定项
    /// 
    [Serializable]
    public sealed class ColumnItem : NotifyPropertyChangedBase, ICloneable
    {
        /// 
        /// 实例化  类新实例,一般用于顶级标题栏
        /// 
        /// 默认列名
        public ColumnItem(String name)
            : this(name, "")
        {
        }

        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 绑定名称
        public ColumnItem(String name, String bindingName)
            : this(name, bindingName, "", "", HorizontalAlignment.Left, 80)
        {
        }

        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 绑定名称
        /// 绑定的资源键(传递此资源时,要在资源中声明)
        /// 字符串格式
        /// 对齐方式
        /// 列宽
        /// 是否显示
        public ColumnItem(String name, String bindingName, String converterResourceKey, String stringFormat, HorizontalAlignment alignment, int width, Visibility visibility)
            : this(name, "", bindingName, converterResourceKey, stringFormat, alignment, width, visibility, ColumnType.TextBlock)
        {

        }

        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 绑定名称
        /// 绑定的资源键(传递此资源时,要在资源中声明)
        /// 字符串格式
        /// 对齐方式
        /// 列宽
        public ColumnItem(String name, String bindingName, String converterResourceKey, String stringFormat, HorizontalAlignment alignment, int width)
            : this(name, "", bindingName, converterResourceKey, stringFormat, alignment, width, Visibility.Visible, ColumnType.TextBlock)
        {

        }

        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 扩展列名
        /// 绑定名称
        /// 绑定的资源键(传递此资源时,要在资源中声明)
        /// 字符串格式
        /// 对齐方式
        /// 列宽
        public ColumnItem(String name, String extendName, String bindingName, String converterResourceKey, String stringFormat, HorizontalAlignment alignment, int width)
       : this(name, extendName, bindingName, converterResourceKey, stringFormat, alignment, width, Visibility.Visible, ColumnType.TextBlock)
        {

        }

        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 绑定名称
        /// 绑定的资源键(传递此资源时,要在资源中声明)
        /// 字符串格式
        /// 对齐方式
        /// 列宽
        /// 数据模板内容类型
        public ColumnItem(String name, String bindingName, String converterResourceKey, String stringFormat, HorizontalAlignment alignment, int width, ColumnType columnType)
            : this(name, "", bindingName, converterResourceKey, stringFormat, alignment, width, Visibility.Visible, columnType)
        {

        }
        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 扩展列名
        /// 绑定名称
        /// 绑定的资源键(传递此资源时,要在资源中声明)
        /// 字符串格式
        /// 对齐方式
        /// 列宽
        /// 数据模板内容类型
        public ColumnItem(String name, String extendName, String bindingName, String converterResourceKey, String stringFormat, HorizontalAlignment alignment, int width, Visibility visibility)
            : this(name, extendName, bindingName, converterResourceKey, stringFormat, alignment, width, visibility, ColumnType.TextBlock)
        {

        }

        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 扩展列名
        /// 绑定名称
        /// 绑定的资源键(传递此资源时,要在资源中声明)
        /// 字符串格式
        /// 对齐方式
        /// 列宽
        /// 数据模板内容类型
        public ColumnItem(String name, String extendName, String bindingName, String converterResourceKey, String stringFormat, HorizontalAlignment alignment, int width, ColumnType columnType)
            : this(name, extendName, bindingName, converterResourceKey, stringFormat, alignment, width, Visibility.Visible, columnType)
        {

        }

        /// 
        /// 实例化  类新实例
        /// 
        /// 默认列名
        /// 扩展列名
        /// 绑定名称
        /// 绑定的资源键(传递此资源时,要在资源中声明)
        /// 字符串格式
        /// 对齐方式
        /// 列宽
        /// 是否显示
        /// 数据模板内容类型
        public ColumnItem(String name, String extendName, String bindingName, String converterResourceKey, String stringFormat, HorizontalAlignment alignment, int width, Visibility visibility, ColumnType columnType)
        {
            this.ID = Guid.NewGuid();
            if (String.IsNullOrWhiteSpace(name))
            {
                throw new ArgumentNullException("titleName", "标题不能为 null 或 空白字符串。");
            }
            this.Name = name;
            this.ExtendName = extendName;
            this.BindingName = (String.IsNullOrWhiteSpace(bindingName)) ? "simple" : bindingName;
            this.ConverterResourceKey = converterResourceKey;
            this.StringFormat = stringFormat;
            this.Alignment = alignment;
            this.Width = width;
            this.Visibility = visibility;
            this.Type = columnType;
            this.Columns = new ColumnItemCollection(this);  //初始化
            this.Level = 0; //默认
            this.TextWrapping = System.Windows.TextWrapping.NoWrap;
        }



        private Int32 mWidth;
        private Visibility mVisibility;
        private HorizontalAlignment mAlignment;

        public Guid ID { get; private set; }


        /// 
        /// 列宽
        /// 
        public Int32 Width
        {
            get { return mWidth; }
            set { mWidth = value; OnPropertyChanged("Width"); }
        }


        /// 
        /// 列显示
        /// 
        public Visibility Visibility
        {
            get { return mVisibility; }
            set { mVisibility = value; OnPropertyChanged("Visibility"); }
        }

        /// 
        /// 获取或设置水平对齐方式
        /// 
        public HorizontalAlignment Alignment
        {
            get { return mAlignment; }
            set { mAlignment = value; OnPropertyChanged("Alignment"); }
        }

        /// 
        /// 默认列名
        /// 
        public String Name { get; private set; }

        /// 
        /// 扩展列名
        /// 
        public String ExtendName { get; set; }

        /// 
        /// 绑定名称
        /// 
        public String BindingName { get; set; }

        /// 
        /// 获取或设置转换资源键
        /// 
        public String ConverterResourceKey { get; set; }

        /// 
        /// 获取或设置字符串格式
        /// 
        public String StringFormat { get; set; }

        /// 
        /// 控件类型
        /// 
        public ColumnType Type { get; set; }

        /// 
        /// 获取或设置是否自动换行(默认为 NoWrap)
        /// 
        public TextWrapping TextWrapping { get; set; }

        public ColumnComboBox ColumnComboBox { get; set; }


        /// 
        /// 获取列集合
        /// 
        public ColumnItemCollection Columns { get; private set; }

        /// 
        /// 获取级别
        /// 
        public int Level { get; internal set; }

        /// 
        /// 获取父级
        /// 
        public ColumnItem Parent { get; internal set; }

        /// 
        /// 获取子级深度
        /// 
        /// 
        public int ChildLevelDepth()
        {
            if (this.Columns.Count > 0)
            {
                return this.CalcLevelDepth(true) - this.Level;
            }
            else
            {
                return 0;
            }
        }

        /// 
        /// 计算获取级别深度
        /// 
        /// 计算子级
        /// 
        private int CalcLevelDepth(bool child)
        {
            if (this.Columns.Count > 0)
            {
                int level = this.Columns.Max(c => c.CalcLevelDepth(child));
                if (child)
                {
                    return level;
                }
                else
                {
                    return level - this.Level;
                }
            }
            else
            {
                return this.Level;
            }
        }

        /// 
        /// 获取当前列的所有最后子集的总数
        /// 
        /// 
        public int LastLevelColumnCount()
        {
            int value = 0;
            if (this.Columns.Count > 0)
            {
                value += this.Columns.Sum(c => c.LastLevelColumnCount());
            }
            else
            {
                value = 1;
            }
            return value;
        }

        /// 
        /// 合计总宽度
        /// 
        /// 
        public int TotalGroupWidth()
        {
            int value;
            if (this.Columns.Count > 0)
            {
                value = this.Columns.Sum(c => c.TotalGroupWidth());
            }
            else
            {
                value = this.Width;
            }
            return value;
        }

        /// 
        /// 验证(必须设置绑定值)
        /// 
        internal void CreateVerify()
        {
            if (this.Columns.Count == 0)
            {
                if (String.IsNullOrWhiteSpace(this.BindingName))
                {
                    throw new ArgumentNullException(String.Format("{0} 为末级时,绑定路径 BindingPath 为 null 或空白字符串。", this.Name));
                }
            }
        }

        public void SetColumnComboBox(ColumnComboBox columnComboBox)
        {
            this.ColumnComboBox = columnComboBox;
        }

        #region ICloneable ColumnItem 对象深度复制  
        public object Clone()
        {
            ColumnItem item = (ColumnItem)this.MemberwiseClone();
            item.Parent = null;
            item.Name = this.Name;
            item.ExtendName = this.ExtendName;
            item.Columns = new ColumnItemCollection(item);
            foreach (var o in this.Columns)
            {
                var _o = (ColumnItem)o.Clone();
                _o.Parent = null;
                item.Columns.Add(_o);
            }
            return item;
        }
        private void OnClone(ColumnItemCollection collection, ColumnItemCollection collection2)
        {
            foreach (var item in collection2)
            {
                var _o = (ColumnItem)item.Clone();
                if (item.Columns.Count > 0)
                {
                    OnClone(item.Columns, item.Columns);
                }
                collection.Add(_o);
            }
        }


        #endregion
    }
    [Serializable]
    public class ColumnComboBox : NotifyPropertyChangedBase
    {
        public ColumnComboBox(String comboBoxBindName)
            : this(comboBoxBindName, "", "", "")
        {

        }
        public ColumnComboBox(String comboBoxBindName, String selectedItemBindName)
            : this(comboBoxBindName, selectedItemBindName, "", "")
        {

        }
        public ColumnComboBox(String comboBoxBindName, String selectedItemBindName, String selectedValuePath, String displayMemberPath)
        {
            this.mComboBoxBindName = comboBoxBindName;
            this.mSelectedItemBindName = selectedItemBindName;
            this.mSelectedValuePath = selectedValuePath;
            this.mDisplayMemberPath = displayMemberPath;
        }


        private String mSelectedItemBindName;
        private String mComboBoxBindName;
        private String mSelectedValuePath = "";
        private String mDisplayMemberPath = "";
        private ObservableCollection<string> itemSource;  //

        /// 
        /// 绑定资源
        /// 
        public String ComboBoxBindName
        {
            get
            {
                return mComboBoxBindName;
            }

            set
            {
                mComboBoxBindName = value;
            }
        }

        /// 
        /// 选中值
        /// 
        public string SelectedValuePath
        {
            get
            {
                return mSelectedValuePath;
            }

            set
            {
                mSelectedValuePath = value;
            }
        }

        /// 
        /// 显示值
        /// 
        public string DisplayMemberPath
        {
            get
            {
                return mDisplayMemberPath;
            }

            set
            {
                mDisplayMemberPath = value;
            }
        }

        public string SelectedItemBindName
        {
            get
            {
                return mSelectedItemBindName;
            }

            set
            {
                mSelectedItemBindName = value;
            }
        }
    }

报表内容中需要其他类型时,扩展IDataGridDataTemplateService接口

    /// 
    /// 控件数据模板构建服务,
    /// 根据的内容多样性构建不同的显示控件,主要针对单一控件显示
    /// 
    public interface IDataGridDataTemplateService
    {
        /// 
        /// 动态构内容绑定模板
        /// 
        /// 自定义列信息
        /// 是否参数转换
        /// 绑定的子列索引 
        /// 
        String CreateCellXaml(ColumnItem column, bool bindingParameter, int? gridColum);
    }

DataGridTemplateColumnHelper核心类,组织动态的列标题以及绑定列,可在下载源码参考。

    /// 
    /// 自定义模板列构建帮助类
    /// 
    internal static class DataGridTemplateColumnHelper
    {
        /// 
        /// 设置标题Grid列宽度,组织出一个复杂的标题格式
        /// 
        /// 标题字符串
        /// 
        /// 列索引,用于控制标题内容显示的边线
        private static void SetColumnDefinition(StringBuilder sbStr, ColumnItem column, ref int index)
        {
            if (column.Columns.Count > 0)
            {
                foreach (var item in column.Columns)
                {
                    SetColumnDefinition(sbStr, item, ref index);
                }
            }
            else  //默认包含一列
            {
                if (index > 0) //当index>0时,添加一个右侧的Rectangle矩形图形边线位置,默认是左侧和右侧是不绘制边线
                {
                    sbStr.AppendLine("");
                }
                //添加一个标题的显示内容列,并设置固定宽度
                sbStr.AppendLine(String.Format("", column.Width));
                index++;
            }
        }

        /// 
        /// 设置标题内容
        /// 
        /// 
        /// 当前列
        /// 总列数
        /// 内容索引列
        private static void SetContentPresenter(StringBuilder sbStr, ColumnItem column, int totalcolumnCount, ref int colIndex)
        {
            //计算出当前列所占的Grid的ColumnDefinition数目
            int columnOffset = column.LastLevelColumnCount() * 2 - 1;
            //计算当前列在整个标题中的标题行深度
            int LevelDepth = column.Parent.ChildLevelDepth();
            //用于控制绘制标题内容下面显示下线,以及当前显示标题的Grid.RowSpan
            int rowOffset;  //一般情况默认为1
            if (column.Columns.Count == 0)//
            {
                rowOffset = LevelDepth * 2 - 1;
            }
            else
            {
                rowOffset = 1; //
            }
            //计算出当前标题在第几个RowDefinition画内容下边线
            int lineRow = (column.Level * 2 + 1) + (rowOffset - 1);  
            //计算出当前标题在第几个ColumnDefinition画内容右边线
            int lineColumn = (colIndex + 1) + (columnOffset - 1);
            //画标题,并设置当前标题在Grid中的定位
            sbStr.AppendLine(
                CreateDataGridTemplateColumnHeaderContent(
                    String.IsNullOrWhiteSpace(column.ExtendName) ? column.Name : column.ExtendName,  //标题显示内容
                    column.Level * 2,  //所属行,标题内容显示的Grid.Row
                    rowOffset,  //标题内容显示的Grid.RowSpan
                    colIndex,  //标题内容显示的Grid.Column列
                    columnOffset //标题内容显示的Grid.ColumnSpan
                ));
            //存在子级,时添加下线
            if (column.Columns.Count > 0)
            {
                sbStr.AppendLine(String.Format("", lineRow, colIndex, 1, columnOffset));
            }
            //标题右侧下线
            if (lineColumn < (totalcolumnCount * 2 - 1))
            {
                sbStr.AppendLine(String.Format("", column.Level * 2, lineColumn, rowOffset, 1));
            }
            //存在子级,先从子级起画
            if (column.Columns.Count > 0)
            {
                foreach (var item in column.Columns)
                {
                    SetContentPresenter(sbStr, item, totalcolumnCount, ref colIndex);
                }
            }
            else
            {
                colIndex += 2; //含分隔线
            }
        }

        /// 
        /// 设置单元格绑定
        /// 
        /// 
        /// 
        /// 
        /// 
        private static void SetCellBinding(StringBuilder sbStr, ColumnItem column, bool bindingParameter, ref int index)
        {
            if (column.Columns.Count > 0)
            {
                foreach (var item in column.Columns)
                {
                    SetCellBinding(sbStr, item, bindingParameter, ref index);
                }
            }
            else
            {
                if (index > 0)
                {
                    sbStr.AppendLine(String.Format("", index));
                    index++;
                }
                //构建指定类型的项绑定
                IDataGridDataTemplateFactory templateFactory = ServiceFactory.GetService();
                IDataGridDataTemplateService templateService = templateFactory.GetService(column.Type);
                sbStr.AppendLine(templateService.CreateCellXaml(column, bindingParameter, index));
                index++;
            }
        }

        /// 
        /// 设置单元格列宽
        /// 
        /// 
        /// 
        /// 
        private static void SetCellColumnDefinition(StringBuilder sbStr, ColumnItem column, ref int index)
        {
            if (column.Columns.Count > 0)
            {
                foreach (var item in column.Columns)
                {
                    SetCellColumnDefinition(sbStr, item, ref index);
                }
            }
            else
            {
                if (index > 0)
                {
                    sbStr.AppendLine("");
                }
                sbStr.AppendLine(String.Format("", column.Width));
                index++;
            }
        }

        /// 
        /// 创建组列
        /// 
        /// 
        /// 是否是参数
        /// 
        public static DataGridTemplateColumn CreateTemplateGroupColumn(ColumnItem column, bool bindingParameter)
        {
            var templateColumn = new DataGridTemplateColumn();
            //当前列包含的子级深度
            int LevelDepth = column.ChildLevelDepth();
            //获取当前列子级的总列数
            int totalcolumnCount = column.LastLevelColumnCount();    //
            //设置当前列的宽度
            templateColumn.Width = new DataGridLength(column.TotalGroupWidth() + LevelDepth);
            //构建HeaderStyle以及CellTemplate模板
            //动态构建标题样式 DataGridTemplateColumn.HeaderStyle
            #region 构建多行标题
            var sbHeaderStr = new StringBuilder();
            sbHeaderStr.AppendLine("");
            for (int i = 0; i <= LevelDepth; i++)
            {
                if (i > 0)
                {
                    sbHeaderStr.AppendLine("");  //构建分割线,
                }
                if (i < LevelDepth)
                {
                    sbHeaderStr.AppendLine("");  //内容区域
                }
                else
                {
                    sbHeaderStr.AppendLine(""); //内容区域
                }
            }
            sbHeaderStr.AppendLine("");
            sbHeaderStr.AppendLine("");
            int index = 0;
            foreach (var item in column.Columns)
            {
                SetColumnDefinition(sbHeaderStr, item, ref index);
            }
            sbHeaderStr.AppendLine("");

            int columnOffset = totalcolumnCount * 2 - 1;

            sbHeaderStr.AppendLine(CreateDataGridTemplateColumnHeaderContent(String.IsNullOrWhiteSpace(column.ExtendName) ? column.Name : column.ExtendName, -1, -1, -1, columnOffset));

            sbHeaderStr.AppendLine(String.Format("", columnOffset));

            index = 0;
            foreach (var item in column.Columns)
            {
                SetContentPresenter(sbHeaderStr, item, totalcolumnCount, ref index);
            }
            var headerStyleStr = HeaderStyleString();
            headerStyleStr = headerStyleStr.Replace("{#content#}", sbHeaderStr.ToString());
            templateColumn.HeaderStyle = (Style)XamlReader.Parse(headerStyleStr);
            #endregion
            //动态构建绑定DataTemplate DataGridTemplateColumn.CellTemplate
            #region 构建多行标题数据绑定模板
            var sbCellTempStr = new StringBuilder();
            sbCellTempStr.AppendLine("");
            index = 0;
            foreach (var item in column.Columns)
            {
                SetCellColumnDefinition(sbCellTempStr, item, ref index);
            }
            sbCellTempStr.AppendLine("");
            index = 0;
            foreach (var item in column.Columns)
            {
                SetCellBinding(sbCellTempStr, item, bindingParameter, ref index);
            }
            var cellTemplateStr = CellDataTemplateString();
            cellTemplateStr = cellTemplateStr.Replace("{#content#}", sbCellTempStr.ToString());
            templateColumn.CellTemplate = (DataTemplate)XamlReader.Parse(cellTemplateStr);
            #endregion
            return templateColumn;
        }

        /// 
        /// 动态构建标题模板
        /// 
        /// 目标结果
        /// 标题名称
        /// 所属Grid行Grid.Row
        /// 所属合并行Grid.RowSpan
        /// 所属Grid列Grid.Column
        /// 所属合并列Grid.ColumnSpan
        private static String CreateDataGridTemplateColumnHeaderContent(String title, int row, int rowSpan, int column, int columnSpan)
        {
            StringBuilder sbStr = new StringBuilder();
            sbStr.AppendLine("");
            if (row > 0)
            {
                sbStr.AppendFormat(" Grid.Row=\"{0}\"", row);
            }
            if (rowSpan > 0)
            {
                sbStr.AppendFormat(" Grid.RowSpan=\"{0}\"", rowSpan);
            }
            if (column > 0)
            {
                sbStr.AppendFormat(" Grid.Column=\"{0}\"", column);
            }
            if (columnSpan > 0)
            {
                sbStr.AppendFormat(" Grid.ColumnSpan=\"{0}\"", columnSpan);
            }
            sbStr.Append(">");
            sbStr.AppendLine("");
            sbStr.AppendLine(String.Format("", title));
            sbStr.AppendLine("");
            sbStr.AppendLine("");
            return sbStr.ToString();
        }


        /// 
        /// 创建单列
        /// 
        /// 
        /// 
        /// 
        public static DataGridTemplateColumn CreateTemplateSingleColumn(ColumnItem column, bool bindingParameter)
        {
            if (column == null)
            {
                throw new ArgumentNullException("column");
            }
            if (String.IsNullOrWhiteSpace(column.BindingName))
            {
                throw new ArgumentNullException("column.BindingName", "末级列的绑定名称不能为 null 或空白字符串。");
            }
            //创建DataGrid的列
            var templateColumn = new DataGridTemplateColumn();
            //设置列的宽度
            templateColumn.Width = new DataGridLength(column.Width);
            //构建模板字符串
            var sbStr = new StringBuilder();
            //根据模板创建标题
            sbStr.AppendLine(CreateDataGridTemplateColumnHeaderContent(String.IsNullOrWhiteSpace(column.ExtendName) ? column.Name : column.ExtendName, -1, -1, -1, -1));
            //动态构建标题样式 DataGridTemplateColumn.HeaderStyle
            #region DataGridTemplateColumn.HeaderStyle
            var headerStyleStr = HeaderStyleString();
            headerStyleStr = headerStyleStr.Replace("{#content#}", sbStr.ToString());
            templateColumn.HeaderStyle = (Style)XamlReader.Parse(headerStyleStr);
            sbStr.Clear();
            #endregion
            //动态构建绑定DataTemplate DataGridTemplateColumn.CellTemplate
            #region DataGridTemplateColumn.CellTemplate
            //构建绑定模板
            IDataGridDataTemplateFactory templateFactory = ServiceFactory.GetService();
            IDataGridDataTemplateService templateService = templateFactory.GetService(column.Type);
            sbStr.AppendLine(templateService.CreateCellXaml(column, bindingParameter, null));
            String cellTemplateStr = CellDataTemplateString();
            cellTemplateStr = cellTemplateStr.Replace("{#content#}", sbStr.ToString());
            templateColumn.CellTemplate = (DataTemplate)XamlReader.Parse(cellTemplateStr);
            #endregion
            return templateColumn;
        }

        #region 本地资源模板处理

        private static object lockObj = new object();
        private static String _HeaderStyleString = null;  //
        private static String _CellDataTemplateString = null;//

        /// 
        /// 获取标题样式
        /// 
        /// 
        public static String HeaderStyleString()
        {
            lock (lockObj)
            {
                if (_HeaderStyleString == null)
                {
                    _HeaderStyleString = GetResourceXamlString("HeaderStyle.xaml");
                }
                return _HeaderStyleString;
            }
        }

        /// 
        /// 获取单元格模板字符串
        /// 
        /// 
        public static String CellDataTemplateString()
        {
            lock (lockObj)
            {
                if (_CellDataTemplateString == null)
                {
                    _CellDataTemplateString = GetResourceXamlString("CellDataTemplate.xaml");
                }
                return _CellDataTemplateString;
            }
        }

        /// 
        /// 获取资源Xaml字符串
        /// 
        /// 文件名称
        /// 
        private static String GetResourceXamlString(String fileName)
        {
            var res = Application.GetResourceStream(new Uri(String.Format("pack://application:,,,/TGP.DataGridDemo;component/Template/{0}", fileName), UriKind.RelativeOrAbsolute));
            using (var sr = new StreamReader(res.Stream))
            {
                return sr.ReadToEnd();
            }
        }

        #endregion
    }

 

你可能感兴趣的:(WPF报表自定义通用可筛选列头-WPF特工队内部资料)