• 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

    • Kids game on consoles or phones. Maintaining a Windows PC or updating a driver is way beyond a typical young person's skill set.
    • Here are some of the ways you can use Safari's new look on Apple iOS 26 by Aditya Tiwari iPhone's latest software update brings a ton of new features this year and is currently available in beta on supported iPhone models. Its fresh coat of the controversial Liquid Glass is also coming to the Safari web browser, alongside new iPhone wallpapers and a revamped Photos app. In this article, let's talk about the new way you can customize the Safari browser on your iPhone. A major highlight among new Safari features for iOS 26 is the new Compact mode for the Tab Bar. It serves as the default layout on the new iPhone update, available alongside the two other layouts from previous years. Use the new Compact Tab Bar layout in Safari You can follow these steps to customize Tab Bar layouts in Safari: Go to the Settings app on iOS 26. Open the Apps section from the bottom of the screen and scroll down or search to find Safari. You will find the three layout options under the Tabs section: Compact, Bottom, and Top. Tap on Compact to make it default, if it's not already selected. Each Safari layout offers its own set of perks and downsides. The new Compact Mode on Safari does what its name says. Its translucent interface uses much less screen real estate compared to the Bottom and Top layouts. You can find the Back button on the left side of the pill-shaped address bar that doubles up as a search bar. Next to it is a three-dot menu button to access features such as Share, All Tabs, Bookmarks, New Tab, and more. Use Bottom and Top layouts for quicker access to Safari options Bottom and Top layouts are almost the same, except that the latter places the address bar at the top of the screen. While the Top layout might be inconvenient when using an iPhone with one hand, some users may prefer to have their address bar in this position, perhaps for a more symmetrical view. After all, it's original and the only Safari layout available until iOS 15. Apple has given the Liquid Design treatment to both the Bottom and Top layouts, featuring a translucent, floating Tab Bar interface that offers a sneak peek of what's behind. Alongside the options for Bookmarks, Share, and All Tabs, you can find the Forward button to go to the next page in the browser. This is not the case with the Compact mode, but iOS already lets you swipe from the right edge of the screen to go forward. It appears that Apple has made it difficult to use Safari with one hand. It has shifted the New Tab and Tab Groups menu buttons to the top-left corner of the All Tabs page on iOS 26. This is where the Compact mode stands out, as these options are easily accessible. Alternatively, when on the All Tabs page, you can press the tabs pill at the bottom of the screen to access the Tab Groups menu. One thing you'll notice is that the Compact and Bottom tab layouts shrink into a small pill when you scroll down a web page. The pill is translucent, like the Tab Bar, and changes colors according to its background, providing more screen real estate on your iPhone. However, in the Top layout, the address bar collapses into a pill at the top of the screen. You can scroll up on the screen or tap the pill to get back to the whole UI. Apart from that, the Safari 26 beta version for iOS 26 and iPad 26 brings several under-the-hood features to the table. For instance, Apple has added Screen Time support to WKWebView, which powers the browsing experience for in-app browsers. Safari users on iOS, iPadOS, and macOS can also report issues when they have trouble with a web page, accessible via the Page menu on the left of the address bar in all three layouts. The new Safari features and changes are currently being tested through the Apple Developer Program. These features are expected to be available to the public when Apple releases the iOS 26 update later this year. We'll continue to update this article as more beta releases arrive. If you plan to install iOS 26 developer beta, keep in mind that these releases can be unstable, and make sure to back up your data beforehand.
    • I experienced this in the 80s and 90s and I believe it ultimately set me back. My “computer lab” in grade school and then again in high school in the 90’s had all macs. Then I hit the workforce. Windows and Office were foreign to me. I can see Apple excelling in the creative space but not in the mainstream office. This isn’t a knock on Apple, but more a reality that’s concerning in the context of this article. ….just my experience and opinion
    • Is this the non-subscription version? Can I move the license to a a new PC later on?
  • Recent Achievements

    • Week One Done
      DXB APPS earned a badge
      Week One Done
    • One Month Later
      DecaffKnight94 earned a badge
      One Month Later
    • Dedicated
      S.P earned a badge
      Dedicated
    • One Month Later
      adxnksd42031 earned a badge
      One Month Later
    • Rising Star
      aphanic went up a rank
      Rising Star
  • Popular Contributors

    1. 1
      +primortal
      663
    2. 2
      ATLien_0
      259
    3. 3
      Michael Scrip
      234
    4. 4
      Steven P.
      157
    5. 5
      +FloatingFatMan
      151
  • Tell a friend

    Love Neowin? Tell a friend!