wpf C# иконка в трее закрывается через 30 минут
Следующее приложение отображает активность диска в значке системного окна. Обычно он работает около 30-40 минут, а затем завершается, оставляя значок на рабочем столе. Это как если бы оно было убито системой как ненужная фоновая задача. Почему это происходит, и как я могу предотвратить это?
public partial class MainWindow : Window
{
public System.Windows.Forms.NotifyIcon ni = new System.Windows.Forms.NotifyIcon();
public MainWindow()
{
InitializeComponent();
ni.Visible = true;
ni.Text = "disktray"; // tooltip text show over tray icon
CreateTextIcon("0");
ni.DoubleClick +=
delegate (object sender, EventArgs args)
{
//this.Show();
//this.WindowState = WindowState.Normal;
ni.Visible = false;
ni.Dispose();
System.Windows.Application.Current.Shutdown();
};
}
private void Window_Loaded(object sender, RoutedEventArgs e)
{
CreateTextIcon("0");
DispatcherTimer timer = new DispatcherTimer()
{
Interval = TimeSpan.FromMilliseconds(1024)
};
timer.Tick += Timer_Tick;
timer.Start();
this.Hide();
}
private void Timer_Tick(object sender, EventArgs e)
{
Diskpercent();
string iii = diskpercentvalue.ToString();
CreateTextIcon(iii);
}
public PerformanceCounter myCounter =
new PerformanceCounter("PhysicalDisk", "% Disk Time", "_Total");
public int diskpercentvalue = 0;
public void Diskpercent()
{
var d = Convert.ToInt32(myCounter.NextValue());
if (d > 99) d = 99; // can go over 100%
diskpercentvalue = d;
}
public System.Drawing.Font fontToUse =
new System.Drawing.Font("Microsoft Sans Serif", 16, System.Drawing.FontStyle.Regular, GraphicsUnit.Pixel);
public System.Drawing.Brush brushToUse = new SolidBrush(System.Drawing.Color.White);
public Bitmap bitmapText = new Bitmap(16, 16);
public IntPtr hIcon;
public void CreateTextIcon(string str)
{
//System.Drawing.Font fontToUse = new System.Drawing.Font("Microsoft Sans Serif", 16, System.Drawing.FontStyle.Regular, GraphicsUnit.Pixel);
//System.Drawing.Brush brushToUse = new SolidBrush(System.Drawing.Color.White);
//Bitmap bitmapText = new Bitmap(16, 16);
Graphics g = System.Drawing.Graphics.FromImage(bitmapText);
//IntPtr hIcon;
g.Clear(System.Drawing.Color.Transparent);
g.TextRenderingHint = System.Drawing.Text.TextRenderingHint.SingleBitPerPixelGridFit;
g.DrawString(str, fontToUse, brushToUse, -4, -2);
hIcon = (bitmapText.GetHicon());
ni.Icon = System.Drawing.Icon.FromHandle(hIcon);
}
}
2 ответа
Эта строка вызывает утечку памяти:
hIcon = (bitmapText.GetHicon());
Икон должен быть уничтожен:
hIcon = (bitmapText.GetHicon());
ni.Icon = System.Drawing.Icon.FromHandle(hIcon);
DestroyIcon(hIcon);
Добавьте этот код в класс, чтобы определить DestroyIcon:
[DllImport("user32.dll", SetLastError = true)]
static extern bool DestroyIcon(IntPtr hIcon);
увидеть:
https://msdn.microsoft.com/en-us/library/system.drawing.icon.fromhandle(v=vs.110).aspx
Перейдите в App.xaml.cs и реализуйте его, как показано ниже. Хитрость заключается в том, чтобы никогда не закрывать главное окно, так как закрытое окно не может быть показано снова. Вместо этого отмените закрытие и просто скройте его.
using System.ComponentModel;
using System.Windows;
namespace BackgroundApplication
{
public partial class App : Application
{
private System.Windows.Forms.NotifyIcon _notifyIcon;
private bool _isExit;
protected override void OnStartup(StartupEventArgs e)
{
base.OnStartup(e);
MainWindow = new MainWindow();
MainWindow.Closing += MainWindow_Closing;
_notifyIcon = new System.Windows.Forms.NotifyIcon();
_notifyIcon.DoubleClick += (s, args) => ShowMainWindow();
_notifyIcon.Icon = BackgroundApplication.Properties.Resources.MyIcon;
_notifyIcon.Visible = true;
CreateContextMenu();
}
private void CreateContextMenu()
{
_notifyIcon.ContextMenuStrip =
new System.Windows.Forms.ContextMenuStrip();
_notifyIcon.ContextMenuStrip.Items.Add("MainWindow...").Click += (s, e) => ShowMainWindow();
_notifyIcon.ContextMenuStrip.Items.Add("Exit").Click += (s, e) => ExitApplication();
}
private void ExitApplication()
{
_isExit = true;
MainWindow.Close();
_notifyIcon.Dispose();
_notifyIcon = null;
}
private void ShowMainWindow()
{
if (MainWindow.IsVisible)
{
if (MainWindow.WindowState == WindowState.Minimized)
{
MainWindow.WindowState = WindowState.Normal;
}
MainWindow.Activate();
}
else
{
MainWindow.Show();
}
}
private void MainWindow_Closing(object sender, CancelEventArgs e)
{
if (!_isExit)
{
e.Cancel = true;
MainWindow.Hide(); // A hidden window can be shown again, a closed one not
}
}
}
}
Перейдите в дополнение к App.xaml и удалите Startup-Uri, чтобы при запуске приложения в область уведомлений добавлялся только NotifyIcon.
<Application x:Class="BackgroundApplication.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:BackgroundApplication">
<Application.Resources>
</Application.Resources>
</Application>
Взято из:
https://www.thomasclaudiushuber.com/2015/08/22/creating-a-background-application-with-wpf/
Примечание. Если вы хотите получить доступ к переменной, вы должны создать общедоступное свойство только для чтения в App.xaml.cs:
public System.Windows.Forms.NotifyIcon NotifyIcon { get { return _notifyIcon; } }
Затем вы можете использовать его в главном окне следующим образом:
((App)App.Current).NotifyIcon