Форум программистов, компьютерный форум, киберфорум
C# .NET
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.91/11: Рейтинг темы: голосов - 11, средняя оценка - 4.91
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93

Работа с плагинами программы

24.11.2013, 14:34. Показов 2252. Ответов 15
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Пишу программу с поддержкой плагинов. Нужно как-то в метод Run(объект_формы) плагину передать объект главной формы приложения, чтобы при выполнении метода плагина Load() можно было обновлять ProgressBar в главной форме. Делать ссылку на MainForm в плагине считаю ненужным, так как планируется написание плагинов сторонними разработчиками и соответственно при написании другого плагина человеком, у которого нет исходника основной программы, могут возникнуть трудности. Может я неправильно понимаю смысл работы плагинов. Объясните кому нетрудно.
0
Programming
Эксперт
39485 / 9562 / 3019
Регистрация: 12.04.2006
Сообщений: 41,671
Блог
24.11.2013, 14:34
Ответы с готовыми решениями:

Работас плагинами и их реализация
Всем доброго времени суток. Есть идея и начальные наработки программы-загрузчика контента. Суть такая: Программа предназначена для...

Обмен между плагинами
Добрый вечер. Создаю плагинное приложение. На входе-битовый файл, из файла 1-й плагин извлекает по очереди байтовые буферы. Ети буферы...

При запуске программы выдается сообщение "Прекращена работа программы... "
у меня есть готовая программа в Visual Studio, пытаюсь открыть .exe файл и сразу идет вылет программы, со словами "Прекращена работа...

15
432 / 433 / 93
Регистрация: 16.07.2012
Сообщений: 886
24.11.2013, 16:40
Сделайте в классе плагина событие, а в основной форме подписывайтесь на него
0
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93
24.11.2013, 17:02  [ТС]
canopen, как понять "подписывайтесь"?
0
432 / 433 / 93
Регистрация: 16.07.2012
Сообщений: 886
24.11.2013, 17:30
Предположим класс плагина выглядит как-то так:
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
public class Plugin
{
    public event EventHandler<ProgressChangedEventArgs> ProgressChanged;
 
    public void Load()
    {
        for (int i = 1; i <= 100; i++)
        {
            OnProgressChanged(i);
        }
    }
 
    private void OnProgressChanged(int progressPercentage)
    {
        if (ProgressChanged != null)
        {
            ProgressChanged(this, new ProgressChangedEventArgs(progressPercentage, null));
        }
    }
}
Тогда в основной форме мы можем подписаться на событие ProgressChanged и в зависимости от ProgressPercentage менять состояние какого-то progressBar расположенного на этой форме:
C#
1
2
3
var plugin = new Plugin();
plugin.ProgressChanged += (s, e) => progressBar.Value = e.ProgressPercentage;
plugin.Load();
Таким образом классу Plugin совершенно ничего не нужно знать о том, кто его использует. Он просто в нужные моменты генерирует нужные события (возможно, с данными).
0
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93
24.11.2013, 18:40  [ТС]
canopen, а можно вам проект скинуть на правку? Просто я плагины использую через интерфейс. Да и вообще недавно начал C# учить и много не понятно пока.
0
432 / 433 / 93
Регистрация: 16.07.2012
Сообщений: 886
24.11.2013, 18:51
Лучше просто почитайте про события. Кстати, то что вы используете интерфейсы ничего не меняет - интерфейс так же как и класс может содержать события.
0
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93
24.11.2013, 19:00  [ТС]
canopen, я обязательно прочитаю про события, просто мне легче понять, когда есть живой, так сказать, пример. Тем более если код мой. Так гораздо понятнее будет. Один примерчик, я много не прошу

Вот файл плагина ссылка
Вот файл интерфейса ссылка

Покажите, плиз, как его изменить, чтобы использовать события.
0
438 / 362 / 100
Регистрация: 29.06.2010
Сообщений: 981
Записей в блоге: 1
24.11.2013, 19:49
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
public interface IPlugin
        {
            string Name { get; }
            string Description { get; }
            void Run(string param);
            event EventHandler<ProgressChangedEventArgs> ProgressChanged;
        }
 
        [Export(typeof(IPlugin))]
        public class Class1 : IPlugin
        {
            public string Name { get { return "My name is PluginTest"; } }
            public string Description { get { return "Descriptions of PluginTest"; } }
            public Form PlugForm;
 
            public void Run(string param)
            {
                PluginForm PlugForm = new PluginForm();
                PlugForm.Text = param;
                PlugForm.Show();
            }
 
            public event EventHandler<ProgressChangedEventArgs> ProgressChanged;
        }
0
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93
24.11.2013, 23:03  [ТС]
Написал так в MainForm
C#
1
2
3
4
5
6
        void MainMenu_Plugins_ComboBox_SelectedIndexChanged(object sender, EventArgs e)
        {
            int index = MainMenu_Plugins_ComboBox.ComboBox.SelectedIndex;
            pl.Plugins[index].Run(this.Text);
            pl.Plugins[index].ProgressChanged += (s, o) => this.LoadingProgressBar.Value = o.ProgressPercentage;
        }
Метод Run вызывается нормально. Метод Load выглядит так

C#
1
2
3
4
5
6
7
8
9
        public void Load()
        {
            for (int i = 0; i <= 5; i++)
            {
                OnProgressChanged(i);
                MessageBox.Show("Load Done " + i + "!", "", MessageBoxButtons.OK, MessageBoxIcon.None);
            }
            
        }
Окошко показывает, а прогрессбар не увеличивается.
0
438 / 362 / 100
Регистрация: 29.06.2010
Сообщений: 981
Записей в блоге: 1
24.11.2013, 23:54
Сначала подписаться потом запустить

C#
1
2
pl.Plugins[index].ProgressChanged += (s, o) => this.LoadingProgressBar.Value = o.ProgressPercentage;
pl.Plugins[index].Run(this.Text);
0
Эксперт .NET
 Аватар для Casper-SC
4434 / 2094 / 404
Регистрация: 27.03.2010
Сообщений: 5,657
Записей в блоге: 1
25.11.2013, 00:26
Вот пример, как из плагина работая в другом потоке инициировать события так, чтобы в основном потоке не нужно было бы вызывать Invoke для контролов.

Библиотека: PluginLibrary
IPlugin.cs
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
using System;
using System.ComponentModel;
using System.Threading;
 
namespace PluginLibrary
{
    public interface IPlugin
    {
        string Name { get; }
        string Description { get; }
        void Run(SynchronizationContext sycnContext);
        event EventHandler<ProgressChangedEventArgs> ProgressChanged;
    }
}
Библиотека: PluginExample
PluginTest.cs
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
using System;
using System.ComponentModel;
using System.ComponentModel.Composition;
using System.Threading;
using PluginLibrary;
 
namespace PluginExample
{
    [Export(typeof(IPlugin))]
    public class PluginTest : IPlugin
    {
        private SynchronizationContext _sycnContext;
 
        public string Description
        {
            get { return "Описание плагина из библиотеки PluginExample.dll"; }
        }
 
        public string Name
        {
            get { return "Имя плагина из библиотеки PluginExample.dll"; }
        }
 
        public event EventHandler<ProgressChangedEventArgs> ProgressChanged;
 
        public void Run(SynchronizationContext sycnContext)
        {
            _sycnContext = sycnContext;
            ThreadPool.QueueUserWorkItem(unused =>
                {
                    for (int i = 0; i < 100; i++)
                    {
                        Thread.Sleep(100);
                        var args = new ProgressChangedEventArgs(i, null);
                        _sycnContext.Post(OnProgressChanged, args);
                    }
                });
        }
 
        protected void OnProgressChanged(object args)
        {
            ProgressChangedEventArgs eventArgs = (ProgressChangedEventArgs)args;
            var handler = ProgressChanged;
            if (handler != null)
            {
                handler(this, eventArgs);
            }
        }
    }
}

Проект: WinForms_Plugin
MainForm.cs
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
using System;
using System.ComponentModel;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Threading;
using System.Windows.Forms;
using PluginLibrary;
 
namespace WinForms_Plugin
{
    public partial class MainForm : Form
    {
        private IPlugin _currentPlugin;
 
        public MainForm()
        {
            InitializeComponent();
        }
 
        private void MainForm_Load(object sender, EventArgs e)
        {
            string path = Path.Combine(Path.GetDirectoryName(Application.ExecutablePath), "PluginExample.dll");
            var asm = Assembly.LoadFile(path);
            _currentPlugin = Activator.CreateInstance(asm.GetTypes().First(t => t.GetInterfaces().Contains(typeof(IPlugin)))) as IPlugin;
            _currentPlugin.ProgressChanged += new EventHandler<ProgressChangedEventArgs>(CurrentPlugin_ProgressChanged);
        }
 
        void CurrentPlugin_ProgressChanged(object sender, ProgressChangedEventArgs e) //!!!
        {
            //Плагин инициирует события в другом потоке, через контекст синхронизации.
            //Вызывать Invoke не нужно
            progressBar1.Value = e.ProgressPercentage; 
        }
 
        private void TestPluginButton_Click(object sender, EventArgs e)
        {
            nameLabel.Text = _currentPlugin.Name;
            descriptionLabel.Text = _currentPlugin.Description;
            SynchronizationContext context = new SynchronizationContext();
            _currentPlugin.Run(SynchronizationContext.Current);
        }
    }
}
Вложения
Тип файла: zip WinForms_Plugin.zip (106.1 Кб, 14 просмотров)
0
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93
25.11.2013, 00:38  [ТС]
Блин, мужики, сложно это пока для меня. Да и поздно уже и башка закипела. Скину файлы для примера. Исправьте кому не сложно.
Program.cs
MainForm.cs
IPlugin.cs
PluginTest.cs
PluginForm.cs
0
Эксперт .NET
 Аватар для Casper-SC
4434 / 2094 / 404
Регистрация: 27.03.2010
Сообщений: 5,657
Записей в блоге: 1
25.11.2013, 02:00
Не проще проект готовый выложить, чтобы из этих кусков не сидеть не гадать, не складывать проект. Вообще, я увидел в методе Main ты вызываешь создание экземпляра класса после того, как закроется главное окно. И классы не файлами выкладывать надо, а просто текстом, их только скачать все ещё тот гемор, телодвижений много, хотя бы в архив можно было скинуть.
0
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93
25.11.2013, 04:37  [ТС]
Мужики, вот архив и в нем полность вся папка рабочая, включая IPlugin и PluginTest. Помогите, плиз, реально в это все уперлось и не двигается.
Архив
0
438 / 362 / 100
Регистрация: 29.06.2010
Сообщений: 981
Записей в блоге: 1
25.11.2013, 12:22
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
public partial class PluginForm : Form
    {
        private IPlugin _plugin;
        public PluginForm(IPlugin plugin)
        {
            InitializeComponent();
            _plugin = plugin;
        }
        
        
        void Button1_Click(object sender, EventArgs e)
        {
            _plugin.Load();
        }
    }
 
 
[Export(typeof(IPlugin))]
    public class Class1 : IPlugin
    {
        public string Name { get { return "My name is PluginTest"; } }
        public string Description { get { return "Descriptions of PluginTest"; } }
        public Form PlugForm;
        
        public event EventHandler<ProgressChangedEventArgs> ProgressChanged;
 
        public void Run(string param)
        {
            PluginForm PlugForm = new PluginForm(this);
            PlugForm.Text = param;
            PlugForm.Show();
        }
        
        public void Load()
        {
            for (int i = 0; i <= 300; i++)
            {
                OnProgressChanged(i);
                //MessageBox.Show("Load Done " + i + "!", "", MessageBoxButtons.OK, MessageBoxIcon.None);
            }
            
        }
        
        private void OnProgressChanged(int progressPercentage)
        {
            if (ProgressChanged != null)
            {
                ProgressChanged(this, new ProgressChangedEventArgs(progressPercentage, null));
            }
        }
    }
0
0 / 0 / 3
Регистрация: 24.05.2012
Сообщений: 93
25.11.2013, 12:46  [ТС]
Grishaco, куда на пиво кинуть? Спасибище тебе, дружище!!! Вот от всей души - СПАСИБО! Вот именно то что нужно!
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
inter-admin
Эксперт
29715 / 6470 / 2152
Регистрация: 06.03.2009
Сообщений: 28,500
Блог
25.11.2013, 12:46
Помогаю со студенческими работами здесь

Исключение при работе с плагинами
Вернулся к старому. Когда пересоздал проект с плагинами, не подгружаются плагины из папки. Ошибка string pluginFiles =...

Нужно создать плагин, посоветуйте программу для работы с звуковыми файлами c плагинами на C#
Посоветуйте, пожалуйста, программу для работы со звуковыми файлами, которая: 1. Позволяет создавать плагины. 2. Стабильно работает при...

Автозапуск программы: прекращена работа программы
Ребята, написал не сложную программу для управления сервисом, и задал в автозапуске, но при старте пишет типичную ошибку виндовс7 что не...

При запуске программы ошибка "Прекращена работа программы"
Делал простенькое приложение на С#, в котором создается два потока, один наблюдает за запущенными процессами, второй представляет собой...

Работа с плагинами через интерфейс
В общем я сподвигся на такое Есть интерфейс public interface IPlugin { string PluginName { get; } // имя плагина...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
16
Ответ Создать тему
Новые блоги и статьи
Использование SDL3-callbacks вместо функции main() на Android, Desktop и WebAssembly
8Observer8 24.01.2026
Если вы откроете примеры для начинающих на официальном репозитории SDL3 в папке: examples, то вы увидите, что все примеры используют следующие четыре обязательные функции, а привычная функция main(). . .
моя боль
iceja 24.01.2026
Выложила интерполяцию кубическими сплайнами www. iceja. net REST сервисы временно не работают, только через Web. Написала за 56 рабочих часов этот сайт с нуля. При помощи perplexity. ai PRO , при. . .
Модель сукцессии микоризы
anaschu 24.01.2026
Решили писать научную статью с неким РОманом
http://iceja.net/ математические сервисы
iceja 20.01.2026
Обновила свой сайт http:/ / iceja. net/ , приделала Fast Fourier Transform экстраполяцию сигналов. Однако предсказывает далеко не каждый сигнал (см ограничения http:/ / iceja. net/ fourier/ docs ). Также. . .
http://iceja.net/ сервер решения полиномов
iceja 18.01.2026
Выкатила http:/ / iceja. net/ сервер решения полиномов (находит действительные корни полиномов методом Штурма). На сайте документация по API, но скажу прямо VPS слабенький и 200 000 полиномов. . .
Расчёт переходных процессов в цепи постоянного тока
igorrr37 16.01.2026
/ * Дана цепь(не выше 3-го порядка) постоянного тока с элементами R, L, C, k(ключ), U, E, J. Программа находит переходные токи и напряжения на элементах схемы классическим методом(1 и 2 з-ны. . .
Восстановить юзерскрипты Greasemonkey из бэкапа браузера
damix 15.01.2026
Если восстановить из бэкапа профиль Firefox после переустановки винды, то список юзерскриптов в Greasemonkey будет пустым. Но восстановить их можно так. Для этого понадобится консольная утилита. . .
Сукцессия микоризы: основная теория в виде двух уравнений.
anaschu 11.01.2026
https:/ / rutube. ru/ video/ 7a537f578d808e67a3c6fd818a44a5c4/
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru