From d4eab7e7b8be9148a7821ae03b9d55484874d425 Mon Sep 17 00:00:00 2001 From: Rui Guerra Date: Thu, 20 May 2021 23:08:37 +0100 Subject: [PATCH] added feature to support single and multiple fileds filtering based on string split (cherry picked from commit 6d91333cf28a56526d0e9d3c9efad7b3efe01f79) --- .../FieldSelectionDialog.Designer.cs | 69 +++++++++++----- src/ParquetFileViewer/FieldSelectionDialog.cs | 78 +++++++++++++++++-- 2 files changed, 122 insertions(+), 25 deletions(-) diff --git a/src/ParquetFileViewer/FieldSelectionDialog.Designer.cs b/src/ParquetFileViewer/FieldSelectionDialog.Designer.cs index a5897a5..5018512 100644 --- a/src/ParquetFileViewer/FieldSelectionDialog.Designer.cs +++ b/src/ParquetFileViewer/FieldSelectionDialog.Designer.cs @@ -32,21 +32,26 @@ private void InitializeComponent() this.allFieldsRadioButton = new System.Windows.Forms.RadioButton(); this.showSelectedFieldsRadioButton = new System.Windows.Forms.RadioButton(); this.fieldsPanel = new System.Windows.Forms.Panel(); - this.doneButton = new System.Windows.Forms.Button(); this.allFieldsRememberRadioButton = new System.Windows.Forms.RadioButton(); + this.doneButton = new System.Windows.Forms.Button(); + this.filterColumnsTextbox = new System.Windows.Forms.TextBox(); + this.clearfilterColumnsButton = new System.Windows.Forms.Button(); this.mainTableLayoutPanel.SuspendLayout(); this.SuspendLayout(); // // mainTableLayoutPanel // - this.mainTableLayoutPanel.ColumnCount = 2; + this.mainTableLayoutPanel.ColumnCount = 3; this.mainTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Absolute, 20F)); this.mainTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Percent, 100F)); + this.mainTableLayoutPanel.ColumnStyles.Add(new System.Windows.Forms.ColumnStyle(System.Windows.Forms.SizeType.Absolute, 30F)); this.mainTableLayoutPanel.Controls.Add(this.allFieldsRadioButton, 0, 0); this.mainTableLayoutPanel.Controls.Add(this.showSelectedFieldsRadioButton, 0, 2); - this.mainTableLayoutPanel.Controls.Add(this.fieldsPanel, 1, 3); - this.mainTableLayoutPanel.Controls.Add(this.doneButton, 1, 4); + this.mainTableLayoutPanel.Controls.Add(this.fieldsPanel, 1, 4); this.mainTableLayoutPanel.Controls.Add(this.allFieldsRememberRadioButton, 0, 1); + this.mainTableLayoutPanel.Controls.Add(this.doneButton, 1, 5); + this.mainTableLayoutPanel.Controls.Add(this.filterColumnsTextbox, 1, 3); + this.mainTableLayoutPanel.Controls.Add(this.clearfilterColumnsButton, 2, 3); this.mainTableLayoutPanel.Dock = System.Windows.Forms.DockStyle.Fill; this.mainTableLayoutPanel.Location = new System.Drawing.Point(0, 0); this.mainTableLayoutPanel.Name = "mainTableLayoutPanel"; @@ -54,6 +59,7 @@ private void InitializeComponent() this.mainTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 30F)); this.mainTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 30F)); this.mainTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 30F)); + this.mainTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 28F)); this.mainTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Percent, 100F)); this.mainTableLayoutPanel.RowStyles.Add(new System.Windows.Forms.RowStyle(System.Windows.Forms.SizeType.Absolute, 30F)); this.mainTableLayoutPanel.Size = new System.Drawing.Size(429, 346); @@ -92,25 +98,14 @@ private void InitializeComponent() // fieldsPanel // this.fieldsPanel.AutoScroll = true; + this.mainTableLayoutPanel.SetColumnSpan(this.fieldsPanel, 2); this.fieldsPanel.Dock = System.Windows.Forms.DockStyle.Fill; this.fieldsPanel.Enabled = false; - this.fieldsPanel.Location = new System.Drawing.Point(23, 93); + this.fieldsPanel.Location = new System.Drawing.Point(23, 121); this.fieldsPanel.Name = "fieldsPanel"; - this.fieldsPanel.Size = new System.Drawing.Size(403, 220); + this.fieldsPanel.Size = new System.Drawing.Size(403, 192); this.fieldsPanel.TabIndex = 2; // - // doneButton - // - this.doneButton.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Bottom) - | System.Windows.Forms.AnchorStyles.Right))); - this.doneButton.Location = new System.Drawing.Point(329, 319); - this.doneButton.Name = "doneButton"; - this.doneButton.Size = new System.Drawing.Size(97, 24); - this.doneButton.TabIndex = 3; - this.doneButton.Text = "Done"; - this.doneButton.UseVisualStyleBackColor = true; - this.doneButton.Click += new System.EventHandler(this.doneButton_Click); - // // allFieldsRememberRadioButton // this.allFieldsRememberRadioButton.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Bottom) @@ -125,6 +120,42 @@ private void InitializeComponent() this.allFieldsRememberRadioButton.UseVisualStyleBackColor = true; this.allFieldsRememberRadioButton.CheckedChanged += new System.EventHandler(this.AllFieldsRememberRadioButton_CheckedChanged); // + // doneButton + // + this.doneButton.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Bottom) + | System.Windows.Forms.AnchorStyles.Right))); + this.mainTableLayoutPanel.SetColumnSpan(this.doneButton, 2); + this.doneButton.Location = new System.Drawing.Point(329, 319); + this.doneButton.Name = "doneButton"; + this.doneButton.Size = new System.Drawing.Size(97, 24); + this.doneButton.TabIndex = 3; + this.doneButton.Text = "Done"; + this.doneButton.UseVisualStyleBackColor = true; + this.doneButton.Click += new System.EventHandler(this.doneButton_Click); + // + // filterColumnsTextbox + // + this.filterColumnsTextbox.Enabled = false; + this.filterColumnsTextbox.Location = new System.Drawing.Point(23, 93); + this.filterColumnsTextbox.Name = "filterColumnsTextbox"; + this.filterColumnsTextbox.Size = new System.Drawing.Size(373, 20); + this.filterColumnsTextbox.TabIndex = 0; + this.filterColumnsTextbox.TextChanged += new System.EventHandler(this.filterColumnsTextbox_TextChanged); + // + // clearfilterColumnsButton + // + this.clearfilterColumnsButton.Anchor = ((System.Windows.Forms.AnchorStyles)(((System.Windows.Forms.AnchorStyles.Top | System.Windows.Forms.AnchorStyles.Bottom) + | System.Windows.Forms.AnchorStyles.Right))); + this.clearfilterColumnsButton.Enabled = false; + this.clearfilterColumnsButton.Location = new System.Drawing.Point(402, 93); + this.clearfilterColumnsButton.Name = "clearfilterColumnsButton"; + this.clearfilterColumnsButton.Size = new System.Drawing.Size(24, 22); + this.clearfilterColumnsButton.TabIndex = 4; + this.clearfilterColumnsButton.Text = "X"; + this.clearfilterColumnsButton.TextAlign = System.Drawing.ContentAlignment.TopCenter; + this.clearfilterColumnsButton.UseVisualStyleBackColor = true; + this.clearfilterColumnsButton.Click += new System.EventHandler(this.clearfilterColumnsButton_Click); + // // FieldsToLoadForm // this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); @@ -150,5 +181,7 @@ private void InitializeComponent() private System.Windows.Forms.Panel fieldsPanel; private System.Windows.Forms.Button doneButton; private System.Windows.Forms.RadioButton allFieldsRememberRadioButton; + private System.Windows.Forms.TextBox filterColumnsTextbox; + private System.Windows.Forms.Button clearfilterColumnsButton; } } \ No newline at end of file diff --git a/src/ParquetFileViewer/FieldSelectionDialog.cs b/src/ParquetFileViewer/FieldSelectionDialog.cs index a4443b3..580e268 100644 --- a/src/ParquetFileViewer/FieldSelectionDialog.cs +++ b/src/ParquetFileViewer/FieldSelectionDialog.cs @@ -13,9 +13,10 @@ public partial class FieldsToLoadForm : Form private const int DynamicFieldCheckboxYIncrement = 30; public static readonly List UnsupportedSchemaTypes = new List() { SchemaType.List, SchemaType.Map, SchemaType.Struct }; - public IEnumerable PreSelectedFields { get; set; } + public List PreSelectedFields { get; set; } public IEnumerable AvailableFields { get; set; } public List NewSelectedFields { get; set; } + public List PreserveSelectedFields { get; set; } public FieldsToLoadForm() { @@ -29,19 +30,24 @@ public FieldsToLoadForm(IEnumerable availableFields, IEnumerable { InitializeComponent(); this.AvailableFields = availableFields; - this.PreSelectedFields = preSelectedFields ?? new List(); + this.PreSelectedFields = preSelectedFields.ToList() ?? new List(); this.NewSelectedFields = new List(); } private void FieldsToLoadForm_Load(object sender, EventArgs e) { this.CenterToParent(); + this.RenderFieldsCheckboxes(this.AvailableFields, this.PreSelectedFields); + } + + private void RenderFieldsCheckboxes(IEnumerable availableFields, IEnumerable preSelectedFields) + { this.fieldsPanel.SuspendLayout(); //Suspending the layout while dynamically adding controls adds significant performance improvement this.fieldsPanel.Controls.Clear(); try { - if (this.AvailableFields != null) + if (availableFields != null) { int locationX = 0; int locationY = 5; @@ -49,15 +55,15 @@ private void FieldsToLoadForm_Load(object sender, EventArgs e) HashSet fieldNames = new HashSet(); bool isClearingSelectAllCheckbox = false; - foreach (Field field in this.AvailableFields) + foreach (Field field in availableFields) { if (isFirst) //Add toggle all checkbox and some other setting changes { isFirst = false; - if (this.PreSelectedFields != null) + if (preSelectedFields != null) { - foreach (string preSelectedField in this.PreSelectedFields) + foreach (string preSelectedField in preSelectedFields) { this.showSelectedFieldsRadioButton.Checked = true; break; @@ -77,6 +83,10 @@ private void FieldsToLoadForm_Load(object sender, EventArgs e) selectAllCheckbox.CheckedChanged += (object checkboxSender, EventArgs checkboxEventArgs) => { var selectAllCheckBox = (CheckBox)checkboxSender; + var showFilterControls = !(selectAllCheckBox.Enabled && selectAllCheckBox.Checked && string.IsNullOrWhiteSpace(this.filterColumnsTextbox.Text)); + this.filterColumnsTextbox.Enabled = showFilterControls; + this.clearfilterColumnsButton.Enabled = showFilterControls; + if (!isClearingSelectAllCheckbox) { foreach (Control control in this.fieldsPanel.Controls) @@ -84,7 +94,10 @@ private void FieldsToLoadForm_Load(object sender, EventArgs e) if (!control.Tag.Equals(SelectAllCheckboxName) && control is CheckBox checkbox) { if (checkbox.Enabled) + { checkbox.Checked = selectAllCheckBox.Checked; + //this.PreSelectedFields.Remove((string)checkbox.Tag); + } } } } @@ -102,7 +115,7 @@ private void FieldsToLoadForm_Load(object sender, EventArgs e) Name = string.Concat("checkbox_", field.Name), Text = string.Concat(field.Name, isUnsupportedFieldType ? "(Unsupported)" : string.Empty), Tag = field.Name, - Checked = this.PreSelectedFields.Contains(field.Name), + Checked = preSelectedFields.Contains(field.Name), Location = new Point(locationX, locationY), AutoSize = true, Enabled = !isUnsupportedFieldType @@ -110,6 +123,17 @@ private void FieldsToLoadForm_Load(object sender, EventArgs e) fieldCheckbox.CheckedChanged += (object checkboxSender, EventArgs checkboxEventArgs) => { var fieldCheckBox = (CheckBox)checkboxSender; + + if (fieldCheckBox.Checked) + { + this.PreSelectedFields.Add((string)fieldCheckBox.Tag); + } + else + { + this.PreSelectedFields.Remove((string)fieldCheckBox.Tag); + } + + if (!fieldCheckBox.Checked) { foreach (Control control in this.fieldsPanel.Controls) @@ -120,6 +144,7 @@ private void FieldsToLoadForm_Load(object sender, EventArgs e) { isClearingSelectAllCheckbox = true; checkbox.Checked = false; + this.PreSelectedFields.Remove((string)fieldCheckBox.Tag); isClearingSelectAllCheckbox = false; break; } @@ -150,6 +175,8 @@ private void allFieldsRadioButton_CheckedChanged(object sender, EventArgs e) if (((RadioButton)sender).Checked) { this.fieldsPanel.Enabled = false; + this.filterColumnsTextbox.Enabled = false; + this.clearfilterColumnsButton.Enabled = false; this.allFieldsRememberRadioButton.Checked = false; this.showSelectedFieldsRadioButton.Checked = false; } @@ -160,6 +187,8 @@ private void AllFieldsRememberRadioButton_CheckedChanged(object sender, EventArg if (((RadioButton)sender).Checked) { this.fieldsPanel.Enabled = false; + this.filterColumnsTextbox.Enabled = false; + this.clearfilterColumnsButton.Enabled = false; this.allFieldsRadioButton.Checked = false; this.showSelectedFieldsRadioButton.Checked = false; } @@ -170,6 +199,8 @@ private void showSelectedFieldsRadioButton_CheckedChanged(object sender, EventAr if (((RadioButton)sender).Checked) { this.fieldsPanel.Enabled = true; + this.filterColumnsTextbox.Enabled = true; + this.clearfilterColumnsButton.Enabled = true; this.allFieldsRadioButton.Checked = false; this.allFieldsRememberRadioButton.Checked = false; } @@ -225,5 +256,38 @@ private void ShowError(Exception ex, string customMessage = null, bool showStack { MessageBox.Show(string.Concat(customMessage ?? "Something went wrong:", Environment.NewLine, showStackTrace ? ex.ToString() : ex.Message), ex.Message, MessageBoxButtons.OK, MessageBoxIcon.Error); } + + + private void filterColumnsTextbox_TextChanged(object sender, EventArgs e) + { + if (!string.IsNullOrWhiteSpace(this.filterColumnsTextbox.Text)) + { + IEnumerable filteredFields; + var filteredColumnsNames = this.filterColumnsTextbox.Text.Split(',').ToList(); + + if (filteredColumnsNames.Count == 1) + { + var filter = filteredColumnsNames[0]; + filteredFields = this.AvailableFields.Where(w => w.Name.Contains(filter)); + } + else + { + char[] charsToTrim = { '"', ' ', '\'' }; + filteredColumnsNames = filteredColumnsNames.Select(s => s.Trim(charsToTrim)).ToList(); + filteredFields = this.AvailableFields.Where(w => filteredColumnsNames.Contains(w.Name)); + } + + this.RenderFieldsCheckboxes(filteredFields, this.PreSelectedFields); + } + else + { + this.RenderFieldsCheckboxes(this.AvailableFields, this.PreSelectedFields); + } + } + + private void clearfilterColumnsButton_Click(object sender, EventArgs e) + { + this.filterColumnsTextbox.Text = string.Empty; + } } }