• 0

[C#] Dynamic Event Handling


Question

I have an arbitrary list of elements that I'm adding to a tableLayoutPanel (it was the first layout I came across that I could add elements to and it would be scrollable). For each element, I add the name of the element, with a TrackBar and a TextBox. (so three controls per row).

I want to link the TrackBar with the TextBox in a way where when I move the TrackBar, it changes the numeric value in the textBox to the value of the TrackBar (and ViceVersa).

I know how to add the Events for the two (+= new EventHandler(Scrolled)). The problem is all my elements are being added with the same EventName, so no matter which control I manipulate, i want the corresponding controls to change as well... I think this is OK, because I don't know how many of these elements I will have until runtime.

My question is this... inside the Scrolled method, how can I make elementScrollBar1 change elementTextBox1, and 2 -> 2, 3 -> 3, etc? The only things passed in are object sender, EventArgs events. Is what I'm trying to do possible? If so please help.

Link to comment
https://www.neowin.net/forum/topic/610910-c-dynamic-event-handling/
Share on other sites

21 answers to this question

Recommended Posts

  • 0

You need something like this:

		private void AddControls_Click(object sender, EventArgs e)
		{
			TextBox textbox = new TextBox();
			TrackBar trackbar = new TrackBar();
			trackbar.Maximum = 10;
			trackbar.Tag = textbox;

			trackbar.Scroll +=  delegate(object sender2, EventArgs e2)
								{
									TrackBar currentTrackBar = sender2 as TrackBar;
									TextBox currentTextBox = currentTrackBar.Tag as TextBox;
									currentTextBox.Text = currentTrackBar.Value.ToString();
								};
			tableLayoutPanel1.Controls.AddRange(new Control[] { textbox, trackbar });

		}

Add that into a button click event.

Hope this helps.

  • 0

My controls are added dynamically when the program is run, not by a click event... I'm guessing replacing my code with this for those two controls will have the same effect? Looking at this it seems like I should do this for other trackbar/textbox combos I have... this links the two I'm guessing?

Does this also allow someone to edit the textbox and it affect the trackbar?

  • 0

I figured out that it doesn't and modified my code to make it work.. I did something similar to what you showed, using the textBox.Tag = trackBar to put a reference.

Is this common practice? If I want to relate two controls, should I just reference each within the Tag variable? Is that common practice? It seems a little odd to me, even though it works as a hack.

  • 0

Why don't you create a custom control that includes the elements you need, and put the logic there? That's probably the most common practice in this case, since it's exactly why custom controls exist...

Just add a new UserControl to your project, name it something useful, and add a TextBox and Trackbar to it with the layout you want. Then set up the event handlers as you normally would inside the control, you can even use the Forms Designer for all of this.

Then in your main app, instead of dynamically generating TextBoxes and Trackbars, you add MyCustomControls that encapsulate the functionality you need. If you need to you can then expose a method or property on your custom control like GetValue() that returns the value (as indicated by both the textbox and trackbar).

  • 0

I didn't know about this custom control stuff, this seems to be much easier to work with... but I have a new problem because of it

So I have a control with a bunch of controls, one of them again linking a trackbar to a textBox... when I manipulate the trackBar, it updates the textBox correctly.. I have all the event stuff inside the Control Class... but the problem is, whenever I change the trackBar I want to call my update() method from the main class (Form1)... How would one do this?

Edited by zackiv31
  • 0
  zackiv31 said:
I didn't know about this custom control stuff, this seems to be much easier to work with... but I have a new problem because of it

So I have a control with a bunch of controls, one of them again linking a trackbar to a textBox... when I manipulate the trackBar, it updates the textBox correctly.. I have all the event stuff inside the Control Class... but the problem is, whenever I change the trackBar I want to call my update() method from the main class (Form1)... How would one do this?

Ideally, your control would expose an event that would notify any listeners that the trackbar has changed.

  • 0

You would create a delegate and an event that are public in your user control, and when you handle the trackbar change event in your user control, you fire another event indicating that it changed to the 'outside' world.

using System;
using System.ComponentModel;
using System.Windows.Forms;

namespace cs3
{
	public delegate void ValueChangedHandler(object sender, int newValue);

	public partial class UserControl1 : UserControl
	{
		public UserControl1()
		{
			InitializeComponent();
			textBox1.Text = trackBar1.Value.ToString();
		}

		[Category("Action")]
		public event ValueChangedHandler ValueChanged;

		private void trackBar1_ValueChanged(object sender, EventArgs e)
		{
			textBox1.Text = trackBar1.Value.ToString();
			OnValueChanged(trackBar1.Value);
		}

		protected void OnValueChanged(int newValue)
		{
			if (ValueChanged != null)
				ValueChanged(this, newValue);
		}
	}
}

  • 0
  zackiv31 said:
		[Category("Action")]
		  public event ValueChangedHandler ValueChanged;

This is new to me.. but seems interesting... what exactly are the [] for? What does it represent or tell he compiler?

It's an attribute for the designer. Add the control to your form, and then select the control. In the Properties pane of the designer, you see properties categorized. If you click on the Events button(the lightning bolt), it lists the events that your control exposes. If you didn't put the CategoryAttribute above your event, it would show up at the bottom of the list in the Misc category.

Attributes are one of the cool aspects of .NET that can be leveraged in many interesting ways. You could effectively describe your class with attributes and discover the metadata at runtime, and perform actions based on it. I've built a framework that utilized attributes to describe how properties should be validated. It was only a proof of concept, but it worked quite well. I'm now seeing that same idea show up in other frameworks, such as the Castle Project's MVC framework.

Edited by azcodemonkey
  • 0

here's my userControl:

	public delegate void ValueChangedHandler(object sender);

	public partial class ListBag : UserControl
	{
		[Category("Action")]
		public event ValueChangedHandler ValueChanged;

here's the declaration in Form1:

ListBag lb = new ListBag();
					lb.ValueChanged += new ValueChangedHandler;

It doesn't recognize this, it tries to make it " += ValueChangedHandler(someNameHere)"...

  • 0

But you said you wanted your Form1 to know when something changed inside the user control. That's why you need to have a method inside Form1 that gets called when the event is fired. It can be named anything you want (like OnListBagValueChanged or something).

  • 0
  zackiv31 said:
wo wo wo... OnValueChanged is a part of my custom UserControl... therefore Form1 has no idea abou the inner workings of it...

someone confirm this? (and help still)

Your form needs to have a method that matches the signature of the delegate of your control.

protected void UserCtrlValueChanged(object sender)
{
	...
}

Then you can use the += syntax to add a new handler in your form's InitializeComponent method.

userCtrlInstance.ValueChanged += new ValueChangedHandler(UserCtrlValueChanged);

You should be able to do this completely through the designer. Add the control to your form, select the control, go to the Properties panel in the IDE and select the Event(lightning bolt), and double-click the ValueChanged event in the UI. This will do all of the above correctly for you.

Edit: Brandon got it. :)

  • 0

Not that it matters much, but if that's what you delegate looks like - you don't actually need a custom one. Just use the built-in EventHandler delegate. Other than that little nit, it looks like you've got things figured out.

  • 0

Well it depends. Are you using:

public delegate void ValueChangedHandler(object sender);

or

public delegate void ValueChangedHandler(object sender, int newValue);

If the latter, then just keep it as is. If the former, then you don't need to declare that delegate at all.

protected void UserCtrlValueChanged(object sender)
{
...
}

in your main form, becomes

protected void UserCtrlValueChanged(object sender, EventArgs e)
{
...
}

And everywhere you used "ValueChangedHandler" before, you could now use just the built-in EventHandler delegate.

For example:

userCtrlInstance.ValueChanged += new ValueChangedHandler(UserCtrlValueChanged);

becomes

userCtrlInstance.ValueChanged += new EventHandler(UserCtrlValueChanged);

But if you are using a custom delegate to pass an integer with the new value, then you'll want to keep doing it the way you are. And when the event is raised, you just pass EventArgs.Empty for the second parameter.

This topic is now closed to further replies.
  • Recently Browsing   0 members

    • No registered users viewing this page.
  • Posts

    • Minor Problems over the years with Windows 11 Pro on my original Intel 10700 Desktop i was using and not many problems on my Newer AMD Ryzen 7 7700X. Overall Windows 11 works extremely well, if don't do too many modifications, and do the regular security updates, and keep 3rd party apps up to date, and users shouldn't have any issues, and if a user does, should be a minor little issue that is easily fixable My goal is to have the 2 remaining household systems replaced for Windows 11 Compatible machines, hopefully by early September if not sooner.
    • Bill Gates says he'll donate 99 percent of his wealth to Africa by Hamid Ganji Microsoft co-founder and tech billionaire Bill Gates has pledged to donate a significant chunk of his personal wealth to African countries. As reported by the BBC, Gates's funding will be spent on improving health and education infrastructure in Africa over the next twenty years. The Gates Foundation has played an active role in improving public health and education in Africa over the past decades. Bill Gates aims to double down on that effort by donating most of his fortune to Africa. Last month, he also said that 99% of his fortune, which could exceed $200 billion, will go to African countries by 2045. Speaking at the African Union (AU) headquarters in Ethiopia's capital, Addis Ababa, Gates said, "By unleashing human potential through health and education, every country in Africa should be on a path to prosperity." The tech billionaire also told young African innovators to start relying on AI to improve health and education in their countries and use the technology to benefit the entire continent. He introduced Rwanda as a successful example of using AI in healthcare to identify high-risk pregnancies. "Africa largely skipped traditional banking and now you have a chance, as you build your next generation healthcare systems, to think about how AI is built into that," Gates added. By donating 99 percent of his personal fortune by 2045, Bill Gates can still maintain his position as one of the richest people on earth. According to Bloomberg, Gates's current net worth is around $175 billion. Gates's decision to donate 99 percent of his wealth to Africa came after the US administration cut USAID funding to African countries following the DOGE investigations. In a recent interview with the New York Times, Bill Gates called out Elon Musk for cutting the USAID budget, saying Musk has a role in the death of the poorest children on earth.
    • Very cool next level tech (for Witcher 4, CybePunk 2, etc.), but I'd be more impressed with seeing this at 4k than through a too low-bitrate YouTube ~1080p filter.
    • I was thinking, and was going to post, almost exactly what @Brandon Hjust posted!
  • Recent Achievements

    • Week One Done
      Adam Todd earned a badge
      Week One Done
    • Contributor
      Ed B went up a rank
      Contributor
    • One Month Later
      moporcho earned a badge
      One Month Later
    • One Month Later
      Parotel earned a badge
      One Month Later
    • Reacting Well
      Cryptecks earned a badge
      Reacting Well
  • Popular Contributors

    1. 1
      +primortal
      202
    2. 2
      snowy owl
      146
    3. 3
      ATLien_0
      133
    4. 4
      Xenon
      120
    5. 5
      +FloatingFatMan
      110
  • Tell a friend

    Love Neowin? Tell a friend!