Фоновый рабочий все еще замораживает пользовательский интерфейс - PullRequest
0 голосов
/ 10 мая 2019

Даже если я настрою фонового работника, я не могу заставить его не заморозить интерфейс.

Я посмотрел в Интернете, но не смог найти решение.Я попытался создать базовые потоки, и это сработало, но мне нужно также обновлять пользовательский интерфейс во время его работы, поэтому я переключился на фоновых рабочих, но если бы вы могли помочь мне разобраться, как использовать базовый поток и получить эту работу, которая будет в порядке.

using MediaToolkit;
using MediaToolkit.Model;
using System;
using System.Collections;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.IO;
using System.Linq;
using System.Net;
using System.Net.Sockets;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;
using VideoLibrary;

namespace ConsoleApp1
{
public partial class Form1 : Form
{
    private List<VideoInfo> vids = new List<VideoInfo>();
    private List<VideoInfo> failedVids = new List<VideoInfo>();
    private int threads = 0;
    BackgroundWorker worker;
    private delegate void DELEGATE();
    static void Main(string[] args)
    {
        Form1 form = new Form1();
        form.ShowDialog();
    }

    public Form1()
    {
        InitializeComponent();
        worker = new BackgroundWorker();
    }

    private void button1_Click(object sender, EventArgs e)
    {
        if(threads == 0)
        {
            progressBar1.Maximum = vids.Count;
            progressBar1.Value = 0;
            failedVids.Clear();
            worker.DoWork += doWork;
            worker.RunWorkerAsync();
            //downloadVid();
        }else
        {
            Console.WriteLine("Waiting for threads" + threads);
        }
        /*Thread thread1 = new Thread(new ThreadStart(downloadVid));
        thread1.Start();*/
    }

    private void button1_Click_1(object sender, EventArgs e)
    {
        VideoInfo vid = new VideoInfo(tbLink.Text, tbTitle.Text, tbArtist.Text);
        vids.Add(vid);
        tbLink.Clear();
        tbTitle.Clear();
        listView1.Items.Add(vid.getLink());
    }

    private int downloadThread(int i)
    {
        Console.WriteLine(i);
        var source = @"C:\Users\derri\Downloads\DownloadTest\";
        var youtube = YouTube.Default;
        VideoInfo vidInfo = vids[(int) i];
        var vid = youtube.GetVideo(vidInfo.getLink());
        Console.WriteLine(vidInfo.getLink());
        try
        {
            File.WriteAllBytes(source + vid.FullName, vid.GetBytes());
        }
        catch (Exception e)
        {
            failedVids.Add(vids[(int)i]);
            Console.WriteLine(e);
            goto End;
        }

        var inputFile = new MediaFile { Filename = source + vid.FullName };
        var outputFile = new MediaFile { Filename = $"{source + vidInfo.getArtist() + " - " + vidInfo.getTitle()}.mp3" };

        using (var engine = new Engine())
        {
            engine.GetMetadata(inputFile);

            engine.Convert(inputFile, outputFile);
        }
        File.Exists(outputFile.Filename);
        File.Delete(inputFile.Filename);
        setTags(vidInfo.getArtist(), vidInfo.getTitle());
    End:
        threads--;
        return 1;
    }

    private void doWork(object sender, DoWorkEventArgs e)
    {
        Delegate del = new DELEGATE(downloadVid);
        this.Invoke(del);
    }

    private void downloadVid()
    {
        int prog = 0;
        for (int i = 0; i < vids.Count; i++)
        {
            Console.WriteLine(i);
            while ((threads > 5)) { }
            Thread thread = new Thread(() => { prog += downloadThread(i); });
            thread.Start();
            threads++;
            System.Threading.Thread.Sleep(1000);
            //thread.Join();
            /*ParameterizedThreadStart start = new ParameterizedThreadStart(downloadThread);
            Thread t = new Thread(start);
            t.Start(i);
            progressBar1.Value++;
            threads++;*/

        }

        while (threads > 0){}
        foreach (VideoInfo failedVid in failedVids)
        {
            listView2.Items.Add(failedVid.getLink());
        }
        listView1.Clear();
        vids.Clear();

    }

    private void setTags(string artist, string title)
    {
        TagLib.File file = TagLib.File.Create("C:\\Users\\derri\\Downloads\\DownloadTest\\" + artist + " - " + title + ".mp3");
        file.Tag.Artists = (new String[] { artist });
        file.Tag.Title = (title);
        file.Save();
    }
}

}

1 Ответ

0 голосов
/ 10 мая 2019

this.Invoke будет выполняться в потоке пользовательского интерфейса.

Узнайте, как использовать события ReportProgress или RunWorkerCompleted класса BackgroundWorker. Оба позволяют передавать данные из фонового потока в поток пользовательского интерфейса.

...