Как правильно реализовать BackgroundWorker с обновлениями ProgressBar?

-Обновленный - 14/10 также спросил этовопрос

Чтобы дать четкое представление о том, что происходит и с учетом комментариев и из этой статьиВот

Что я действительно хочу сейчас сделать, так это вызвать новую форму с индикатором выполнения и запустить ее и анимировать, пока мой фоновый поток запускает мой длинный процесс в базе данных, а затем вызывать событие закрытия формы

Фоновый рабочий настроен здесь

 public partial class MainWindow : Window
{
    //Declare background workers
    BackgroundWorker bw = new BackgroundWorker();
    BackgroundWorker bwLoadCSV = new BackgroundWorker();
    BackgroundWorker bwProgressBar = new BackgroundWorker();

Затем делегаты добавили сюда

  public MainWindow()
    {
        bwLoadCSV.WorkerReportsProgress = true;
        bwLoadCSV.WorkerSupportsCancellation = true;
        bwLoadCSV.DoWork += new DoWorkEventHandler(bwLoadCSV_DoWork);
        bwLoadCSV.ProgressChanged += new ProgressChangedEventHandler(bwLoadCSV_ProgressChanged);
        bwLoadCSV.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bwLoadCSV_RunWorkerCompleted);

Звонок производится здесь из основного окна класса

  private void CSV_Load_Click(object sender, RoutedEventArgs e)
    ///Function to read csv into datagrid
    ///
    {
        //Turn Cursor to wait
        System.Windows.Forms.Cursor.Current = System.Windows.Forms.Cursors.WaitCursor;

        //Test connection to sql server
        if (CHHoursDataProvider.IsDatabaseOnline() == false)
        {
            System.Windows.Forms.MessageBox.Show("Can not establish contact with sql server" + "\n" + "Contact IT", "Connection Error");
            //Set UI picture
            return;
        }
        //Set a control to update the user here
        tbLoadDgStat.Visibility = Visibility.Visible;

        tbLoadDgStat.Text = "Getting data templete from Database...";
        string FilePath = txFilePath.Text;
        if (bwLoadCSV.IsBusy != true)
        {
            //load the context object with parameters for Background worker
            bwCSVLoadContext Context = new bwCSVLoadContext();
            Context.Site = cBChSite.Text;
            Context.FilePath = txFilePath.Text;
            Context.FileName = fileTest;
            Context.Wageyear = cbWageYear.Text;
            Context.Startdate = ((DateTime)dpStartDate.SelectedDate);
            Context.Enddate = ((DateTime)dpEndDate.SelectedDate);

            bwLoadCSV.RunWorkerAsync(Context);                

        }

Фоновый работник делает работу

private void bwLoadCSV_DoWork(object sender, DoWorkEventArgs e)
    {
        BackgroundWorker worker = sender as BackgroundWorker;

        bwCSVLoadContext Context = e.Argument as bwCSVLoadContext;


        worker.ReportProgress((10));
        if ((worker.CancellationPending == true))
        {
            e.Cancel = true;

        }
        else
        {
            // Perform a time consuming operation and report progress load csv into datagrid.

Чтобы сообщить о фоновой работе, я делаю это. Именно здесь я пытаюсь загрузить новый вызов формы ProgressDialog, на котором есть индикатор выполнения, который я пытаюсь установить на Indeterminable, чтобы он просто «пролистывал» мою форму ProgressDialoge, чтобы показать пользователю, что что-то еще происходит. Я использовал репортерную часть фоновой работы, потому что считаю, что он имеет доступ к потоку главного окна, и я надеюсь, что метод invoke затем вызывается из потока основного окна, но я не совсем уверен

Вот репортер

 private void bwLoadCSV_ProgressChanged(object sender, ProgressChangedEventArgs e)

    {


        this.Dispatcher.Invoke((MethodInvoker)delegate { tbLoadDgStat.Visibility = Visibility.Visible; });
        //tbLoadDgStat.Visibility = Visibility.Visible;


        //this.progressBar1.Value = e.ProgressPercentage;//This works but pauses on long steps
        if (e.ProgressPercentage == 10)
        {
            //Try to open a new form with a class ProgressDialog and set the progressbar
            // on the frm to IsIndeterminate=true
            //THIS IS NOT WORKING
            this.Dispatcher.BeginInvoke (new Action(() =>
              {  ProgressDialog progressDialog = new ProgressDialog();
                progressDialog.SetIndeterminate(true);
            }));

             //this updates the main form OK
             this.Dispatcher.Invoke((MethodInvoker)delegate { tbLoadDgStat.Text = "Getting data templete from Database..."; });

        }
        else if (e.ProgressPercentage == 20)
        {

            this.Dispatcher.Invoke((MethodInvoker)delegate { tbLoadDgStat.Text = "Data template retrieved..."; });

        }
        else
        {

            if (e.ProgressPercentage % 10 == 0)
            {
                this.Dispatcher.Invoke((MethodInvoker)delegate { tbLoadDgStat.Text = "Adding Data..." + e.ProgressPercentage.ToString() + "%"; });
            }
        }

Наконец, xaml для формы ProgressDialog и ее класса

<Window x:Class="Test_Read_CSV.ProgressDialog"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="Progress Dialog" Height="115" Width="306" Name="ProgressPopup">
<Grid>
    <ProgressBar Height="31" HorizontalAlignment="Left" Margin="12,33,0,0" Name="progressBar1" VerticalAlignment="Top" Width="250" />
    <TextBox Height="23" HorizontalAlignment="Left" Margin="7,4,0,0" Name="tbEvent" VerticalAlignment="Top" Width="254" IsReadOnly="True" IsEnabled="False" />
</Grid>

класс

/// <summary>
/// Interaction logic for ProgressDialog.xaml
/// </summary>
public partial class ProgressDialog : Window
{
    public ProgressDialog()
    {
        WindowStartupLocation = WindowStartupLocation.CenterScreen;
        InitializeComponent();
    }

    public ProgressDialog(String Purpose)
    {
        InitializeComponent();
        tbEvent.Text = Purpose;
        WindowStartupLocation = WindowStartupLocation.CenterScreen;

    }
    public void UpdateProgress(int progress)
    {
        progressBar1.Dispatcher.BeginInvoke(
            new Action(() =>
            {
                progressBar1.Value = progress;
            }
        ));
    }

    public void SetIndeterminate(bool isIndeterminate)
    {
        progressBar1.Dispatcher.BeginInvoke(
            new Action(() =>
            {
                if (isIndeterminate)
                {
                    progressBar1.IsIndeterminate = true;
                }
                else
                {
                    progressBar1.IsIndeterminate = false;
                }
            }
        ));
    }
}

}

Я прочитал и сделал несколько учебников по фоновому рабочему и даже по темам, но не могу получить желаемый результат

Идея состоит в том, что у меня есть два длинных процесса, в которых я либо получаю клон данных с моего удаленного диска, либо обновляю базу данных из моего приложения wpf (.net 4). Пока процесс запущен, я хочу, чтобы индикатор выполнения и его обновление обновлялись, потому что я ясно дал понять, что какая-то работа продолжается. Так что я сделал обычные процедуры выполнения отчетов в фоновом режиме, и это работает .... Однако в моей ветке dowork у меня есть эта команда

CHHoursDataProvider CH = new CHHoursDataProvider();
oTable = CH.CloneCHHours();

это где связь с БД и эта команда занимает хорошие 60 - 90 секунд на удаленном соединении VPN, так что даже если я делаю это

CHHoursDataProvider CH = new CHHoursDataProvider();
worker.ReportProgress((10));
oTable = CH.CloneCHHours();
worker.ReportProgress((20));

Главное окно все еще выглядит замороженным и как будто оно разбилось!

Поэтому все, что я хочу сделать, - это в начале вызова фоновой работы установить индикатор выполнения и оставить его включенным до конца задачи. Это все, что мне нужно сделать, чтобы закончить свой первый в истории проект, и через три дня я все еще не могу разобраться с ним!

Итак, я попробовал следующее

В bw прогресс изменился и в главном окне класс

 this.progressBar2.IsIndeterminate = true;

Однако анимация не начинается, пока не закончится поток Dowork.

Затем я создал еще один фоновый рабочий для обновления progressbar2, с которым была связана ссылка на кнопку в главном окне, но как только я попытался использовать его из другого фонового рабочего или из класса основного окна, он не запускался до тех пор, пока не был запущен dowork. поток завершен на первом фоне рабочего

Затем я попытался следовать методу вызова, но ДЕЙСТВИТЕЛЬНО заблудился на этом!

Так что кто-нибудь может помочь, я могу догадаться, что это связано с тем, что они работают с неправильным потоком и т. Д., Но что я делаю, я понятия не имею.

Я могу разместить больше кода по мере необходимости

Ян

Ответы на вопрос(1)

Ваш ответ на вопрос