I’m trying to create a custom download app. Its all working except for the download all button that cant pick up the “percent1” variable from the “DownloadProgressChangedEventArgs”. I have instantiated it prior to the mainForm constructor but it wont read the changed value.
Here’s the code, partially stripped since most of it isnt relevant to the question:
public partial class Main : Form
{
//Variables (not all, just the one im having issues with)
private double percentage1;
//Main form constructor
public Main(){...}
//Download File Async custom method
public void DldFile(string url, string fileName, string localPath, AsyncCompletedEventHandler completedName, DownloadProgressChangedEventHandler progressName)
{
WebClient webClient = new WebClient();
webClient.DownloadFileAsync(new Uri(url), localPath + "\\" + fileName);
webClient.DownloadFileCompleted += new AsyncCompletedEventHandler(completedName);
webClient.DownloadProgressChanged += new DownloadProgressChangedEventHandler(progressName);
}
//Button 1 click event to start download
private void btnDld1_Click(object sender, EventArgs e)
{
if (url1 != "" && Directory.Exists(localPath1))
{
_startDate1 = DateTime.Now;
DldFile(url1, fileName1, localPath1, completed1, progress1);
}
//took out the try/catch, other ifs to try and cut it down
}
//Download Progress Changed event for Download 1
public void progress1(object sender, DownloadProgressChangedEventArgs e)
{
percentage1 = e.ProgressPercentage; //THIS IS WHERE I WAS EXPECTING TO UPDATE "percentage1"
progressBar1.Value = int.Parse(Math.Truncate(percentage1).ToString());
}
//Button that starts all downloads click event where all my problems are at the moment
private void btnDldAll_Click(object sender, EventArgs e)
{
//The progress bar that should let me know the global status for all webClients
progressBarAll.Value = (
int.Parse(Math.Truncate(percentage1).ToString()) + //HERE IS MY PROBLEM
int.Parse(Math.Truncate(percentage2).ToString()) + //HERE IS MY PROBLEM
int.Parse(Math.Truncate(percentage3).ToString()) + //HERE IS MY PROBLEM
int.Parse(Math.Truncate(percentage4).ToString()) + //HERE IS MY PROBLEM
int.Parse(Math.Truncate(percentage5).ToString())) / 5; //HERE IS MY PROBLEM
//Checks if the link exists and starts it from the download button click event
if (url1 != "")
{
btnDld1.PerformClick();
}
//Continues for url2, 3, 4, 5 and else
}
}
So this is the shortest way i found of letting you know what im trying to pull off, if there’s something missing please let me know, i’ll try to add any info as fast as possible.
I have tried to instantiate “progress1” to try and acess its percentage1 variable, but it didnt work. I’ve tried doing the same thing with the webClient but didnt work either. I have used google and stackflow search to no avail. So im not sure if the question is too dumb, or there’s a diferent way to look at the issue thats completely out of my mindset.
So main problem is updating the “percentage1” variable and using it.
There are other problems regarding the “progressBarAll.Value” calculation that will be solved when i can get my hands on the right value. So no need to worry about that if you see it.
Try not to think about ‘using the event arguments outside the event’. Think about updating the state of your form.
Use properties to simplify the update logic:
I would refactor the code even further, but I think it will be easier for you to understand if the code is closer to the original.
The main idea is to create a set of linked properties which work like mathematical functions. When writing the
PercentageXproperties I’m kind of saying ‘letPercentageAllbe the average of all percentages’. Then I have each download update it’s own progress. Once any progress is updated it updates the average, and I don’t have to rememver that inside the progress changed event handler.And the last point is updating progress bars from percentage properties. It’s quite straightforward: once a percentage is changed, I need to update a bar. If so, why bother writing something like
In this case I have to remember everywhere that once I change the
Percentage1I have to update the bar. And in my example I just create a strict rule for that which is only in one place and works everytime. So I just cannot forget it. And if I need to change the rule, I need to change only one place, so again I cannot make a mistake.The technique I demonstrate can be expressed as a well-known rule: ‘one rule – one place’, which means that you should try to have only single place in code that expresses each logical rule that exists in your program. It is a very important idea, I suggest you learn and use it.