mirror of
https://github.com/greenshot/greenshot
synced 2025-08-19 21:13:23 -07:00
Changed the update check, this should be more stable and flexible.
This commit is contained in:
parent
9c31b90021
commit
700fb07e40
7 changed files with 304 additions and 459 deletions
|
@ -239,7 +239,7 @@ namespace Greenshot.Drawing
|
||||||
|
|
||||||
_counterStart = value;
|
_counterStart = value;
|
||||||
Invalidate();
|
Invalidate();
|
||||||
_propertyChanged?.Invoke(this, new PropertyChangedEventArgs("CounterStart"));
|
_propertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(CounterStart)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
22
Greenshot/Forms/MainForm.Designer.cs
generated
22
Greenshot/Forms/MainForm.Designer.cs
generated
|
@ -73,7 +73,6 @@ namespace Greenshot {
|
||||||
this.toolStripCloseSeparator = new System.Windows.Forms.ToolStripSeparator();
|
this.toolStripCloseSeparator = new System.Windows.Forms.ToolStripSeparator();
|
||||||
this.contextmenu_exit = new GreenshotPlugin.Controls.GreenshotToolStripMenuItem();
|
this.contextmenu_exit = new GreenshotPlugin.Controls.GreenshotToolStripMenuItem();
|
||||||
this.notifyIcon = new System.Windows.Forms.NotifyIcon(this.components);
|
this.notifyIcon = new System.Windows.Forms.NotifyIcon(this.components);
|
||||||
this.backgroundWorkerTimer = new System.Windows.Forms.Timer(this.components);
|
|
||||||
this.contextMenu.SuspendLayout();
|
this.contextMenu.SuspendLayout();
|
||||||
this.SuspendLayout();
|
this.SuspendLayout();
|
||||||
//
|
//
|
||||||
|
@ -122,7 +121,7 @@ namespace Greenshot {
|
||||||
this.contextmenu_capturelastregion.Name = "contextmenu_capturelastregion";
|
this.contextmenu_capturelastregion.Name = "contextmenu_capturelastregion";
|
||||||
this.contextmenu_capturelastregion.ShortcutKeyDisplayString = "Shift + Print";
|
this.contextmenu_capturelastregion.ShortcutKeyDisplayString = "Shift + Print";
|
||||||
this.contextmenu_capturelastregion.Size = new System.Drawing.Size(170, 22);
|
this.contextmenu_capturelastregion.Size = new System.Drawing.Size(170, 22);
|
||||||
this.contextmenu_capturelastregion.Click += new System.EventHandler(this.Contextmenu_capturelastregionClick);
|
this.contextmenu_capturelastregion.Click += new System.EventHandler(this.Contextmenu_CaptureLastRegionClick);
|
||||||
//
|
//
|
||||||
// contextmenu_capturewindow
|
// contextmenu_capturewindow
|
||||||
//
|
//
|
||||||
|
@ -130,7 +129,7 @@ namespace Greenshot {
|
||||||
this.contextmenu_capturewindow.Name = "contextmenu_capturewindow";
|
this.contextmenu_capturewindow.Name = "contextmenu_capturewindow";
|
||||||
this.contextmenu_capturewindow.ShortcutKeyDisplayString = "Alt + Print";
|
this.contextmenu_capturewindow.ShortcutKeyDisplayString = "Alt + Print";
|
||||||
this.contextmenu_capturewindow.Size = new System.Drawing.Size(170, 22);
|
this.contextmenu_capturewindow.Size = new System.Drawing.Size(170, 22);
|
||||||
this.contextmenu_capturewindow.Click += new System.EventHandler(this.Contextmenu_capturewindow_Click);
|
this.contextmenu_capturewindow.Click += new System.EventHandler(this.Contextmenu_CaptureWindow_Click);
|
||||||
//
|
//
|
||||||
// contextmenu_capturefullscreen
|
// contextmenu_capturefullscreen
|
||||||
//
|
//
|
||||||
|
@ -144,7 +143,7 @@ namespace Greenshot {
|
||||||
this.contextmenu_captureie.Name = "contextmenu_captureie";
|
this.contextmenu_captureie.Name = "contextmenu_captureie";
|
||||||
this.contextmenu_captureie.ShortcutKeyDisplayString = "Ctrl + Shift + Print";
|
this.contextmenu_captureie.ShortcutKeyDisplayString = "Ctrl + Shift + Print";
|
||||||
this.contextmenu_captureie.Size = new System.Drawing.Size(170, 22);
|
this.contextmenu_captureie.Size = new System.Drawing.Size(170, 22);
|
||||||
this.contextmenu_captureie.Click += new System.EventHandler(this.Contextmenu_captureie_Click);
|
this.contextmenu_captureie.Click += new System.EventHandler(this.Contextmenu_CaptureIe_Click);
|
||||||
//
|
//
|
||||||
// toolStripListCaptureSeparator
|
// toolStripListCaptureSeparator
|
||||||
//
|
//
|
||||||
|
@ -210,7 +209,7 @@ namespace Greenshot {
|
||||||
this.contextmenu_settings.Image = ((System.Drawing.Image)(resources.GetObject("contextmenu_settings.Image")));
|
this.contextmenu_settings.Image = ((System.Drawing.Image)(resources.GetObject("contextmenu_settings.Image")));
|
||||||
this.contextmenu_settings.Name = "contextmenu_settings";
|
this.contextmenu_settings.Name = "contextmenu_settings";
|
||||||
this.contextmenu_settings.Size = new System.Drawing.Size(170, 22);
|
this.contextmenu_settings.Size = new System.Drawing.Size(170, 22);
|
||||||
this.contextmenu_settings.Click += new System.EventHandler(this.Contextmenu_settingsClick);
|
this.contextmenu_settings.Click += new System.EventHandler(this.Contextmenu_SettingsClick);
|
||||||
//
|
//
|
||||||
// toolStripMiscSeparator
|
// toolStripMiscSeparator
|
||||||
//
|
//
|
||||||
|
@ -229,13 +228,13 @@ namespace Greenshot {
|
||||||
this.contextmenu_donate.Image = ((System.Drawing.Image)(resources.GetObject("contextmenu_donate.Image")));
|
this.contextmenu_donate.Image = ((System.Drawing.Image)(resources.GetObject("contextmenu_donate.Image")));
|
||||||
this.contextmenu_donate.Name = "contextmenu_donate";
|
this.contextmenu_donate.Name = "contextmenu_donate";
|
||||||
this.contextmenu_donate.Size = new System.Drawing.Size(170, 22);
|
this.contextmenu_donate.Size = new System.Drawing.Size(170, 22);
|
||||||
this.contextmenu_donate.Click += new System.EventHandler(this.Contextmenu_donateClick);
|
this.contextmenu_donate.Click += new System.EventHandler(this.Contextmenu_DonateClick);
|
||||||
//
|
//
|
||||||
// contextmenu_about
|
// contextmenu_about
|
||||||
//
|
//
|
||||||
this.contextmenu_about.Name = "contextmenu_about";
|
this.contextmenu_about.Name = "contextmenu_about";
|
||||||
this.contextmenu_about.Size = new System.Drawing.Size(170, 22);
|
this.contextmenu_about.Size = new System.Drawing.Size(170, 22);
|
||||||
this.contextmenu_about.Click += new System.EventHandler(this.Contextmenu_aboutClick);
|
this.contextmenu_about.Click += new System.EventHandler(this.Contextmenu_AboutClick);
|
||||||
//
|
//
|
||||||
// toolStripCloseSeparator
|
// toolStripCloseSeparator
|
||||||
//
|
//
|
||||||
|
@ -255,12 +254,6 @@ namespace Greenshot {
|
||||||
this.notifyIcon.Text = "Greenshot";
|
this.notifyIcon.Text = "Greenshot";
|
||||||
this.notifyIcon.MouseUp += new System.Windows.Forms.MouseEventHandler(this.NotifyIconClickTest);
|
this.notifyIcon.MouseUp += new System.Windows.Forms.MouseEventHandler(this.NotifyIconClickTest);
|
||||||
//
|
//
|
||||||
// backgroundWorkerTimer
|
|
||||||
//
|
|
||||||
this.backgroundWorkerTimer.Enabled = true;
|
|
||||||
this.backgroundWorkerTimer.Interval = 300000;
|
|
||||||
this.backgroundWorkerTimer.Tick += new System.EventHandler(this.BackgroundWorkerTimerTick);
|
|
||||||
//
|
|
||||||
// MainForm
|
// MainForm
|
||||||
//
|
//
|
||||||
this.AutoScaleDimensions = new System.Drawing.SizeF(96F, 96F);
|
this.AutoScaleDimensions = new System.Drawing.SizeF(96F, 96F);
|
||||||
|
@ -282,8 +275,7 @@ namespace Greenshot {
|
||||||
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_capturewindowfromlist;
|
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_capturewindowfromlist;
|
||||||
private System.Windows.Forms.ToolStripSeparator toolStripListCaptureSeparator;
|
private System.Windows.Forms.ToolStripSeparator toolStripListCaptureSeparator;
|
||||||
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_openrecentcapture;
|
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_openrecentcapture;
|
||||||
private System.Windows.Forms.Timer backgroundWorkerTimer;
|
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_captureie;
|
||||||
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_captureie;
|
|
||||||
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_donate;
|
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_donate;
|
||||||
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_openfile;
|
private GreenshotPlugin.Controls.GreenshotToolStripMenuItem contextmenu_openfile;
|
||||||
private System.Windows.Forms.ToolStripSeparator toolStripPluginSeparator;
|
private System.Windows.Forms.ToolStripSeparator toolStripPluginSeparator;
|
||||||
|
|
|
@ -418,6 +418,11 @@ namespace Greenshot {
|
||||||
HandleDataTransport(dataTransport);
|
HandleDataTransport(dataTransport);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Start the update check in the background
|
||||||
|
var updateService = new UpdateService();
|
||||||
|
updateService.Startup();
|
||||||
|
SimpleServiceProvider.Current.AddService(updateService);
|
||||||
|
|
||||||
// Make Greenshot use less memory after startup
|
// Make Greenshot use less memory after startup
|
||||||
if (_conf.MinimizeWorkingSetSize) {
|
if (_conf.MinimizeWorkingSetSize) {
|
||||||
PsAPI.EmptyWorkingSet();
|
PsAPI.EmptyWorkingSet();
|
||||||
|
@ -444,11 +449,9 @@ namespace Greenshot {
|
||||||
Exit();
|
Exit();
|
||||||
break;
|
break;
|
||||||
case CommandEnum.FirstLaunch:
|
case CommandEnum.FirstLaunch:
|
||||||
Invoke((MethodInvoker)delegate {
|
|
||||||
LOG.Info("FirstLaunch: Created new configuration, showing balloon.");
|
LOG.Info("FirstLaunch: Created new configuration, showing balloon.");
|
||||||
var notifyIconClassicMessageHandler = SimpleServiceProvider.Current.GetInstance<INotificationService>();
|
var notifyIconClassicMessageHandler = SimpleServiceProvider.Current.GetInstance<INotificationService>();
|
||||||
notifyIconClassicMessageHandler.ShowInfoMessage(Language.GetFormattedString(LangKey.tooltip_firststart, HotkeyControl.GetLocalizedHotkeyStringFromString(_conf.RegionHotkey)), 2000, ShowSetting);
|
notifyIconClassicMessageHandler.ShowInfoMessage(Language.GetFormattedString(LangKey.tooltip_firststart, HotkeyControl.GetLocalizedHotkeyStringFromString(_conf.RegionHotkey)), 2000, ShowSetting);
|
||||||
});
|
|
||||||
break;
|
break;
|
||||||
case CommandEnum.ReloadConfig:
|
case CommandEnum.ReloadConfig:
|
||||||
LOG.Info("Reload requested");
|
LOG.Info("Reload requested");
|
||||||
|
@ -554,15 +557,16 @@ namespace Greenshot {
|
||||||
/// <param name="sender">object</param>
|
/// <param name="sender">object</param>
|
||||||
/// <param name="e">PropertyChangedEventArgs</param>
|
/// <param name="e">PropertyChangedEventArgs</param>
|
||||||
private void OnIconSizeChanged(object sender, PropertyChangedEventArgs e) {
|
private void OnIconSizeChanged(object sender, PropertyChangedEventArgs e) {
|
||||||
if (e.PropertyName == "IconSize")
|
if (e.PropertyName != "IconSize")
|
||||||
{
|
{
|
||||||
ApplyDpiScaling();
|
return;
|
||||||
string ieExePath = PluginUtils.GetExePath("iexplore.exe");
|
}
|
||||||
if (!string.IsNullOrEmpty(ieExePath)) {
|
ApplyDpiScaling();
|
||||||
contextmenu_captureie.Image = PluginUtils.GetCachedExeIcon(ieExePath, 0);
|
string ieExePath = PluginUtils.GetExePath("iexplore.exe");
|
||||||
}
|
if (!string.IsNullOrEmpty(ieExePath)) {
|
||||||
}
|
contextmenu_captureie.Image = PluginUtils.GetCachedExeIcon(ieExePath, 0);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Modify the DPI settings depending in the current value
|
/// Modify the DPI settings depending in the current value
|
||||||
|
@ -627,7 +631,7 @@ namespace Greenshot {
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Check if OneDrive is blocking hotkeys
|
/// Check if OneDrive is blocking hotkeys
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <returns>true if onedrive has hotkeys turned on</returns>
|
/// <returns>true if one-drive has hotkeys turned on</returns>
|
||||||
private static bool IsOneDriveBlockingHotkey()
|
private static bool IsOneDriveBlockingHotkey()
|
||||||
{
|
{
|
||||||
if (!WindowsVersion.IsWindows10OrLater)
|
if (!WindowsVersion.IsWindows10OrLater)
|
||||||
|
@ -706,14 +710,16 @@ namespace Greenshot {
|
||||||
private void CaptureFile() {
|
private void CaptureFile() {
|
||||||
var openFileDialog = new OpenFileDialog
|
var openFileDialog = new OpenFileDialog
|
||||||
{
|
{
|
||||||
Filter = "Image files (*.greenshot, *.png, *.jpg, *.gif, *.bmp, *.ico, *.tiff, *.wmf)|*.greenshot; *.png; *.jpg; *.jpeg; *.gif; *.bmp; *.ico; *.tiff; *.tif; *.wmf"
|
Filter = @"Image files (*.greenshot, *.png, *.jpg, *.gif, *.bmp, *.ico, *.tiff, *.wmf)|*.greenshot; *.png; *.jpg; *.jpeg; *.gif; *.bmp; *.ico; *.tiff; *.tif; *.wmf"
|
||||||
};
|
};
|
||||||
if (openFileDialog.ShowDialog() == DialogResult.OK) {
|
if (openFileDialog.ShowDialog() != DialogResult.OK)
|
||||||
if (File.Exists(openFileDialog.FileName)) {
|
{
|
||||||
CaptureHelper.CaptureFile(openFileDialog.FileName);
|
return;
|
||||||
}
|
}
|
||||||
}
|
if (File.Exists(openFileDialog.FileName)) {
|
||||||
}
|
CaptureHelper.CaptureFile(openFileDialog.FileName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private void CaptureFullScreen() {
|
private void CaptureFullScreen() {
|
||||||
CaptureHelper.CaptureFullscreen(true, _conf.ScreenCaptureMode);
|
CaptureHelper.CaptureFullscreen(true, _conf.ScreenCaptureMode);
|
||||||
|
@ -807,7 +813,7 @@ namespace Greenshot {
|
||||||
int index = counter.ContainsKey(tabData.Key) ? counter[tabData.Key] : 0;
|
int index = counter.ContainsKey(tabData.Key) ? counter[tabData.Key] : 0;
|
||||||
captureIeTabItem.Image = tabData.Key.DisplayIcon;
|
captureIeTabItem.Image = tabData.Key.DisplayIcon;
|
||||||
captureIeTabItem.Tag = new KeyValuePair<WindowDetails, int>(tabData.Key, index++);
|
captureIeTabItem.Tag = new KeyValuePair<WindowDetails, int>(tabData.Key, index++);
|
||||||
captureIeTabItem.Click += Contextmenu_captureiefromlist_Click;
|
captureIeTabItem.Click += Contextmenu_CaptureIeFromList_Click;
|
||||||
contextmenu_captureiefromlist.DropDownItems.Add(captureIeTabItem);
|
contextmenu_captureiefromlist.DropDownItems.Add(captureIeTabItem);
|
||||||
if (counter.ContainsKey(tabData.Key)) {
|
if (counter.ContainsKey(tabData.Key)) {
|
||||||
counter[tabData.Key] = index;
|
counter[tabData.Key] = index;
|
||||||
|
@ -884,7 +890,7 @@ namespace Greenshot {
|
||||||
// captureForm.MakeCapture(CaptureMode.Window, false);
|
// captureForm.MakeCapture(CaptureMode.Window, false);
|
||||||
// Now we check which windows are there to capture
|
// Now we check which windows are there to capture
|
||||||
ToolStripMenuItem captureWindowFromListMenuItem = (ToolStripMenuItem)sender;
|
ToolStripMenuItem captureWindowFromListMenuItem = (ToolStripMenuItem)sender;
|
||||||
AddCaptureWindowMenuItems(captureWindowFromListMenuItem, Contextmenu_capturewindowfromlist_Click);
|
AddCaptureWindowMenuItems(captureWindowFromListMenuItem, Contextmenu_CaptureWindowFromList_Click);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void CaptureWindowFromListMenuDropDownClosed(object sender, EventArgs e) {
|
private void CaptureWindowFromListMenuDropDownClosed(object sender, EventArgs e) {
|
||||||
|
@ -959,19 +965,19 @@ namespace Greenshot {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Contextmenu_capturelastregionClick(object sender, EventArgs e) {
|
private void Contextmenu_CaptureLastRegionClick(object sender, EventArgs e) {
|
||||||
BeginInvoke((MethodInvoker)delegate {
|
BeginInvoke((MethodInvoker)delegate {
|
||||||
CaptureHelper.CaptureLastRegion(false);
|
CaptureHelper.CaptureLastRegion(false);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Contextmenu_capturewindow_Click(object sender,EventArgs e) {
|
private void Contextmenu_CaptureWindow_Click(object sender,EventArgs e) {
|
||||||
BeginInvoke((MethodInvoker)delegate {
|
BeginInvoke((MethodInvoker)delegate {
|
||||||
CaptureHelper.CaptureWindowInteractive(false);
|
CaptureHelper.CaptureWindowInteractive(false);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Contextmenu_capturewindowfromlist_Click(object sender,EventArgs e) {
|
private void Contextmenu_CaptureWindowFromList_Click(object sender,EventArgs e) {
|
||||||
ToolStripMenuItem clickedItem = (ToolStripMenuItem)sender;
|
ToolStripMenuItem clickedItem = (ToolStripMenuItem)sender;
|
||||||
BeginInvoke((MethodInvoker)delegate {
|
BeginInvoke((MethodInvoker)delegate {
|
||||||
try {
|
try {
|
||||||
|
@ -983,11 +989,11 @@ namespace Greenshot {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Contextmenu_captureie_Click(object sender, EventArgs e) {
|
private void Contextmenu_CaptureIe_Click(object sender, EventArgs e) {
|
||||||
CaptureIE();
|
CaptureIE();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Contextmenu_captureiefromlist_Click(object sender, EventArgs e) {
|
private void Contextmenu_CaptureIeFromList_Click(object sender, EventArgs e) {
|
||||||
if (!_conf.IECapture) {
|
if (!_conf.IECapture) {
|
||||||
LOG.InfoFormat("IE Capture is disabled.");
|
LOG.InfoFormat("IE Capture is disabled.");
|
||||||
return;
|
return;
|
||||||
|
@ -1015,9 +1021,9 @@ namespace Greenshot {
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Context menu entry "Support Greenshot"
|
/// Context menu entry "Support Greenshot"
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="sender"></param>
|
/// <param name="sender">object</param>
|
||||||
/// <param name="e"></param>
|
/// <param name="e">EventArgs</param>
|
||||||
private void Contextmenu_donateClick(object sender, EventArgs e) {
|
private void Contextmenu_DonateClick(object sender, EventArgs e) {
|
||||||
BeginInvoke((MethodInvoker)delegate {
|
BeginInvoke((MethodInvoker)delegate {
|
||||||
Process.Start("http://getgreenshot.org/support/?version=" + Assembly.GetEntryAssembly().GetName().Version);
|
Process.Start("http://getgreenshot.org/support/?version=" + Assembly.GetEntryAssembly().GetName().Version);
|
||||||
});
|
});
|
||||||
|
@ -1028,7 +1034,7 @@ namespace Greenshot {
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="sender"></param>
|
/// <param name="sender"></param>
|
||||||
/// <param name="e"></param>
|
/// <param name="e"></param>
|
||||||
private void Contextmenu_settingsClick(object sender, EventArgs e) {
|
private void Contextmenu_SettingsClick(object sender, EventArgs e) {
|
||||||
BeginInvoke((MethodInvoker)ShowSetting);
|
BeginInvoke((MethodInvoker)ShowSetting);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1056,7 +1062,7 @@ namespace Greenshot {
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <param name="sender"></param>
|
/// <param name="sender"></param>
|
||||||
/// <param name="e"></param>
|
/// <param name="e"></param>
|
||||||
private void Contextmenu_aboutClick(object sender, EventArgs e) {
|
private void Contextmenu_AboutClick(object sender, EventArgs e) {
|
||||||
ShowAbout();
|
ShowAbout();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1110,7 +1116,7 @@ namespace Greenshot {
|
||||||
|
|
||||||
// Only add if the value is not fixed
|
// Only add if the value is not fixed
|
||||||
if (!_conf.Values["CaptureMousepointer"].IsFixed) {
|
if (!_conf.Values["CaptureMousepointer"].IsFixed) {
|
||||||
// For the capture mousecursor option
|
// For the capture mouse-cursor option
|
||||||
ToolStripMenuSelectListItem captureMouseItem = new ToolStripMenuSelectListItem
|
ToolStripMenuSelectListItem captureMouseItem = new ToolStripMenuSelectListItem
|
||||||
{
|
{
|
||||||
Text = Language.GetString("settings_capture_mousepointer"),
|
Text = Language.GetString("settings_capture_mousepointer"),
|
||||||
|
@ -1463,27 +1469,5 @@ namespace Greenshot {
|
||||||
notifyIcon = null;
|
notifyIcon = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Do work in the background
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="sender"></param>
|
|
||||||
/// <param name="e"></param>
|
|
||||||
private void BackgroundWorkerTimerTick(object sender, EventArgs e) {
|
|
||||||
if (_conf.MinimizeWorkingSetSize) {
|
|
||||||
PsAPI.EmptyWorkingSet();
|
|
||||||
}
|
|
||||||
if (UpdateHelper.IsUpdateCheckNeeded()) {
|
|
||||||
LOG.Debug("BackgroundWorkerTimerTick checking for update");
|
|
||||||
// Start update check in the background
|
|
||||||
var backgroundTask = new Thread(UpdateHelper.CheckAndAskForUpdate)
|
|
||||||
{
|
|
||||||
Name = "Update check",
|
|
||||||
IsBackground = true
|
|
||||||
};
|
|
||||||
backgroundTask.Start();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
35
Greenshot/Helpers/Entities/UpdateFeed.cs
Normal file
35
Greenshot/Helpers/Entities/UpdateFeed.cs
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
/*
|
||||||
|
* Greenshot - a free and open source screenshot tool
|
||||||
|
* Copyright (C) 2007-2020 Thomas Braun, Jens Klingen, Robin Krom
|
||||||
|
*
|
||||||
|
* For more information see: http://getgreenshot.org/
|
||||||
|
* The Greenshot project is hosted on GitHub https://github.com/greenshot/greenshot
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 1 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
using Newtonsoft.Json;
|
||||||
|
|
||||||
|
namespace Greenshot.Helpers.Entities
|
||||||
|
{
|
||||||
|
[JsonObject]
|
||||||
|
public class UpdateFeed
|
||||||
|
{
|
||||||
|
[JsonProperty("release")]
|
||||||
|
public string CurrentReleaseVersion { get; set; }
|
||||||
|
|
||||||
|
[JsonProperty("beta")]
|
||||||
|
public string CurrentBetaVersion { get; set; }
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,184 +0,0 @@
|
||||||
/*
|
|
||||||
* Greenshot - a free and open source screenshot tool
|
|
||||||
* Copyright (C) 2007-2020 Thomas Braun, Jens Klingen, Robin Krom
|
|
||||||
*
|
|
||||||
* For more information see: http://getgreenshot.org/
|
|
||||||
* The Greenshot project is hosted on GitHub https://github.com/greenshot/greenshot
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 1 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Reflection;
|
|
||||||
using System.Windows.Forms;
|
|
||||||
using Greenshot.Configuration;
|
|
||||||
using GreenshotPlugin.Core;
|
|
||||||
using GreenshotPlugin.IniFile;
|
|
||||||
using log4net;
|
|
||||||
|
|
||||||
namespace Greenshot.Helpers {
|
|
||||||
/// <summary>
|
|
||||||
/// Description of RssFeedHelper.
|
|
||||||
/// </summary>
|
|
||||||
public static class UpdateHelper {
|
|
||||||
private static readonly ILog Log = LogManager.GetLogger(typeof(UpdateHelper));
|
|
||||||
private static readonly CoreConfiguration CoreConfig = IniConfig.GetIniSection<CoreConfiguration>();
|
|
||||||
private const string StableDownloadLink = "https://getgreenshot.org/downloads/";
|
|
||||||
private const string VersionHistoryLink = "https://getgreenshot.org/version-history/";
|
|
||||||
private static readonly object LockObject = new object();
|
|
||||||
private static RssFile _latestGreenshot;
|
|
||||||
private static string _downloadLink = StableDownloadLink;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Is an update check needed?
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>bool true if yes</returns>
|
|
||||||
public static bool IsUpdateCheckNeeded() {
|
|
||||||
lock (LockObject)
|
|
||||||
{
|
|
||||||
if (CoreConfig.UpdateCheckInterval == 0)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
DateTime checkTime = CoreConfig.LastUpdateCheck;
|
|
||||||
checkTime = checkTime.AddDays(CoreConfig.UpdateCheckInterval);
|
|
||||||
if (DateTime.Now.CompareTo(checkTime) < 0)
|
|
||||||
{
|
|
||||||
Log.DebugFormat("No need to check RSS feed for updates, feed check will be after {0}", checkTime);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
Log.DebugFormat("Update check is due, last check was {0} check needs to be made after {1} (which is one {2} later)", CoreConfig.LastUpdateCheck, checkTime, CoreConfig.UpdateCheckInterval);
|
|
||||||
if (!RssHelper.IsRssModifiedAfter(CoreConfig.LastUpdateCheck))
|
|
||||||
{
|
|
||||||
Log.DebugFormat("RSS feed has not been updated since after {0}", CoreConfig.LastUpdateCheck);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Read the RSS feed to see if there is a Greenshot update
|
|
||||||
/// </summary>
|
|
||||||
public static void CheckAndAskForUpdate() {
|
|
||||||
lock (LockObject) {
|
|
||||||
Version currentVersion = Assembly.GetExecutingAssembly().GetName().Version;
|
|
||||||
// Test like this:
|
|
||||||
// currentVersion = new Version("0.8.1.1198");
|
|
||||||
|
|
||||||
// Make sure we update the LastUpdateCheck, in case an error occurs we should not retry every 5 minutes
|
|
||||||
// This actually prevents an update check, but rather one to less than many to many
|
|
||||||
CoreConfig.LastUpdateCheck = DateTime.Now;
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
_latestGreenshot = null;
|
|
||||||
ProcessRssInfo(currentVersion);
|
|
||||||
if (_latestGreenshot != null) {
|
|
||||||
var notifyIcon = SimpleServiceProvider.Current.GetInstance<NotifyIcon>();
|
|
||||||
|
|
||||||
notifyIcon.BalloonTipClicked += HandleBalloonTipClick;
|
|
||||||
notifyIcon.BalloonTipClosed += CleanupBalloonTipClick;
|
|
||||||
notifyIcon.ShowBalloonTip(10000, "Greenshot", Language.GetFormattedString(LangKey.update_found, "'" + _latestGreenshot.File + "'"), ToolTipIcon.Info);
|
|
||||||
}
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.Error("An error occured while checking for updates, the error will be ignored: ", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void CleanupBalloonTipClick(object sender, EventArgs e) {
|
|
||||||
var notifyIcon = SimpleServiceProvider.Current.GetInstance<NotifyIcon>();
|
|
||||||
notifyIcon.BalloonTipClicked -= HandleBalloonTipClick;
|
|
||||||
notifyIcon.BalloonTipClosed -= CleanupBalloonTipClick;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void HandleBalloonTipClick(object sender, EventArgs e) {
|
|
||||||
try {
|
|
||||||
if (_latestGreenshot != null) {
|
|
||||||
// "Direct" download link
|
|
||||||
// Process.Start(latestGreenshot.Link);
|
|
||||||
// Go to getgreenshot.org
|
|
||||||
Process.Start(_downloadLink);
|
|
||||||
}
|
|
||||||
} catch (Exception) {
|
|
||||||
MessageBox.Show(Language.GetFormattedString(LangKey.error_openlink, _downloadLink), Language.GetString(LangKey.error));
|
|
||||||
} finally {
|
|
||||||
CleanupBalloonTipClick(sender, e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void ProcessRssInfo(Version currentVersion) {
|
|
||||||
// Reset latest Greenshot
|
|
||||||
IList<RssFile> rssFiles = RssHelper.ReadRss();
|
|
||||||
|
|
||||||
if (rssFiles == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Retrieve the current and latest greenshot
|
|
||||||
foreach(RssFile rssFile in rssFiles) {
|
|
||||||
if (rssFile.File.StartsWith("Greenshot")) {
|
|
||||||
// check for exe
|
|
||||||
if (!rssFile.IsExe) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// do we have a version?
|
|
||||||
if (rssFile.Version == null) {
|
|
||||||
Log.DebugFormat("Skipping unversioned exe {0} which is published at {1} : {2}", rssFile.File, rssFile.Pubdate.ToLocalTime(), rssFile.Link);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the file is unstable, we will skip it when:
|
|
||||||
// the current version is a release or release candidate AND check unstable is turned off.
|
|
||||||
if (rssFile.IsUnstable) {
|
|
||||||
// Skip if we shouldn't check unstables
|
|
||||||
if ((CoreConfig.BuildState == BuildStates.RELEASE) && !CoreConfig.CheckForUnstable) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the file is a release candidate, we will skip it when:
|
|
||||||
// the current version is a release AND check unstable is turned off.
|
|
||||||
if (rssFile.IsReleaseCandidate) {
|
|
||||||
if (CoreConfig.BuildState == BuildStates.RELEASE && !CoreConfig.CheckForUnstable) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Compare versions
|
|
||||||
int versionCompare = rssFile.Version.CompareTo(currentVersion);
|
|
||||||
if (versionCompare > 0) {
|
|
||||||
Log.DebugFormat("Found newer Greenshot '{0}' with version {1} published at {2} : {3}", rssFile.File, rssFile.Version, rssFile.Pubdate.ToLocalTime(), rssFile.Link);
|
|
||||||
if (_latestGreenshot == null || rssFile.Version.CompareTo(_latestGreenshot.Version) > 0) {
|
|
||||||
_latestGreenshot = rssFile;
|
|
||||||
if (rssFile.IsReleaseCandidate || rssFile.IsUnstable) {
|
|
||||||
_downloadLink = VersionHistoryLink;
|
|
||||||
} else {
|
|
||||||
_downloadLink = StableDownloadLink;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (versionCompare < 0) {
|
|
||||||
Log.DebugFormat("Skipping older greenshot with version {0}", rssFile.Version);
|
|
||||||
} else if (versionCompare == 0) {
|
|
||||||
Log.DebugFormat("Found current version as exe {0} with version {1} published at {2} : {3}", rssFile.File, rssFile.Version, rssFile.Pubdate.ToLocalTime(), rssFile.Link);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
222
Greenshot/Helpers/UpdateService.cs
Normal file
222
Greenshot/Helpers/UpdateService.cs
Normal file
|
@ -0,0 +1,222 @@
|
||||||
|
// Greenshot - a free and open source screenshot tool
|
||||||
|
// Copyright (C) 2007-2020 Thomas Braun, Jens Klingen, Robin Krom
|
||||||
|
//
|
||||||
|
// For more information see: http://getgreenshot.org/
|
||||||
|
// The Greenshot project is hosted on GitHub https://github.com/greenshot/greenshot
|
||||||
|
//
|
||||||
|
// This program is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 1 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
|
using System.Threading;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Dapplo.HttpExtensions;
|
||||||
|
using Dapplo.HttpExtensions.JsonNet;
|
||||||
|
using Greenshot.Configuration;
|
||||||
|
using Greenshot.Helpers.Entities;
|
||||||
|
using GreenshotPlugin.Core;
|
||||||
|
using GreenshotPlugin.IniFile;
|
||||||
|
using GreenshotPlugin.Interfaces;
|
||||||
|
using log4net;
|
||||||
|
|
||||||
|
namespace Greenshot.Helpers
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// This processes the information, if there are updates available.
|
||||||
|
/// </summary>
|
||||||
|
public class UpdateService
|
||||||
|
{
|
||||||
|
private static readonly ILog Log = LogManager.GetLogger(typeof(UpdateService));
|
||||||
|
private static readonly CoreConfiguration CoreConfig = IniConfig.GetIniSection<CoreConfiguration>();
|
||||||
|
private static readonly Uri UpdateFeed = new Uri("https://getgreenshot.org/update-feed.json");
|
||||||
|
private static readonly Uri Downloads = new Uri("https://getgreenshot.org/downloads");
|
||||||
|
private readonly CancellationTokenSource _cancellationTokenSource = new CancellationTokenSource();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Provides the current version
|
||||||
|
/// </summary>
|
||||||
|
public Version CurrentVersion { get; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Provides the latest known version
|
||||||
|
/// </summary>
|
||||||
|
public Version LatestReleaseVersion { get; private set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The latest beta version
|
||||||
|
/// </summary>
|
||||||
|
public Version LatestBetaVersion { get; private set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Checks if there is an release update available
|
||||||
|
/// </summary>
|
||||||
|
public bool IsUpdateAvailable => LatestReleaseVersion > CurrentVersion;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Checks if there is an beta update available
|
||||||
|
/// </summary>
|
||||||
|
public bool IsBetaUpdateAvailable => LatestBetaVersion > CurrentVersion;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Keep track of when the update was shown, so it won't be every few minutes
|
||||||
|
/// </summary>
|
||||||
|
public DateTimeOffset LastUpdateShown = DateTimeOffset.MinValue;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Constructor with dependencies
|
||||||
|
/// </summary>
|
||||||
|
public UpdateService()
|
||||||
|
{
|
||||||
|
JsonNetJsonSerializer.RegisterGlobally();
|
||||||
|
var version = FileVersionInfo.GetVersionInfo(GetType().Assembly.Location);
|
||||||
|
LatestReleaseVersion = CurrentVersion = new Version(version.FileMajorPart, version.FileMinorPart, version.FileBuildPart);
|
||||||
|
CoreConfig.LastSaveWithVersion = CurrentVersion.ToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Start the background task which checks for updates
|
||||||
|
/// </summary>
|
||||||
|
public void Startup()
|
||||||
|
{
|
||||||
|
_ = BackgroundTask(() => TimeSpan.FromDays(CoreConfig.UpdateCheckInterval), UpdateCheck, _cancellationTokenSource.Token);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Stop the update checks
|
||||||
|
/// </summary>
|
||||||
|
public void Shutdown()
|
||||||
|
{
|
||||||
|
if (!_cancellationTokenSource.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
_cancellationTokenSource.Cancel();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This runs a periodic task in the background
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="intervalFactory">Func which returns a TimeSpan</param>
|
||||||
|
/// <param name="reoccurringTask">Func which returns a task</param>
|
||||||
|
/// <param name="cancellationToken">CancellationToken</param>
|
||||||
|
/// <returns>Task</returns>
|
||||||
|
private async Task BackgroundTask(Func<TimeSpan> intervalFactory, Func<CancellationToken, Task> reoccurringTask, CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
// Initial delay, to make sure this doesn't happen at the startup
|
||||||
|
await Task.Delay(20000, cancellationToken);
|
||||||
|
Log.Info("Starting background task to check for updates");
|
||||||
|
await Task.Run(async () =>
|
||||||
|
{
|
||||||
|
while (!cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
var interval = intervalFactory();
|
||||||
|
var task = reoccurringTask;
|
||||||
|
// If the check is disabled, handle that here
|
||||||
|
if (TimeSpan.Zero == interval)
|
||||||
|
{
|
||||||
|
interval = TimeSpan.FromMinutes(10);
|
||||||
|
task = c => Task.FromResult(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await task(cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Log.Error("Error occured when trying to check for updates.", ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await Task.Delay(interval, cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
catch (TaskCanceledException)
|
||||||
|
{
|
||||||
|
// Ignore, this always happens
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
Log.Error("Error occured await for the next update check.", ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}, cancellationToken).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Do the actual update check
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="cancellationToken">CancellationToken</param>
|
||||||
|
/// <returns>Task</returns>
|
||||||
|
private async Task UpdateCheck(CancellationToken cancellationToken = default)
|
||||||
|
{
|
||||||
|
Log.InfoFormat("Checking for updates from {0}", UpdateFeed);
|
||||||
|
var updateFeed = await UpdateFeed.GetAsAsync<UpdateFeed>(cancellationToken);
|
||||||
|
if (updateFeed == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
CoreConfig.LastUpdateCheck = DateTime.Now;
|
||||||
|
|
||||||
|
ProcessFeed(updateFeed);
|
||||||
|
|
||||||
|
// Only show if the update was shown >24 hours ago.
|
||||||
|
if (DateTimeOffset.Now.AddDays(-1) > LastUpdateShown)
|
||||||
|
{
|
||||||
|
if (IsBetaUpdateAvailable)
|
||||||
|
{
|
||||||
|
LastUpdateShown = DateTimeOffset.Now;
|
||||||
|
ShowUpdate(LatestBetaVersion);
|
||||||
|
} else if (IsUpdateAvailable)
|
||||||
|
{
|
||||||
|
LastUpdateShown = DateTimeOffset.Now;
|
||||||
|
ShowUpdate(LatestReleaseVersion);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// This takes care of creating the toast view model, publishing it, and disposing afterwards
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="newVersion">Version</param>
|
||||||
|
private void ShowUpdate(Version newVersion)
|
||||||
|
{
|
||||||
|
var notificationService = SimpleServiceProvider.Current.GetInstance<INotificationService>();
|
||||||
|
var message = Language.GetFormattedString(LangKey.update_found, newVersion.ToString());
|
||||||
|
notificationService.ShowInfoMessage(message, 10000, () => Process.Start(Downloads.AbsoluteUri));
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Process the update feed to get the latest version
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="updateFeed"></param>
|
||||||
|
private void ProcessFeed(UpdateFeed updateFeed)
|
||||||
|
{
|
||||||
|
var latestReleaseString = Regex.Replace(updateFeed.CurrentReleaseVersion, "[a-zA-Z\\-]*", "");
|
||||||
|
if (Version.TryParse(latestReleaseString, out var latestReleaseVersion))
|
||||||
|
{
|
||||||
|
LatestReleaseVersion = latestReleaseVersion;
|
||||||
|
}
|
||||||
|
var latestBetaString = Regex.Replace(updateFeed.CurrentBetaVersion, "[a-zA-Z\\-]*", "");
|
||||||
|
if (Version.TryParse(latestBetaString, out var latestBetaVersion))
|
||||||
|
{
|
||||||
|
LatestBetaVersion = latestBetaVersion;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,204 +0,0 @@
|
||||||
/*
|
|
||||||
* Greenshot - a free and open source screenshot tool
|
|
||||||
* Copyright (C) 2007-2020 Thomas Braun, Jens Klingen, Robin Krom
|
|
||||||
*
|
|
||||||
* For more information see: http://getgreenshot.org/
|
|
||||||
* The Greenshot project is hosted on GitHub https://github.com/greenshot/greenshot
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 1 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Globalization;
|
|
||||||
using System.Net;
|
|
||||||
using System.Text.RegularExpressions;
|
|
||||||
using System.Xml;
|
|
||||||
using log4net;
|
|
||||||
|
|
||||||
namespace GreenshotPlugin.Core {
|
|
||||||
public class RssFile {
|
|
||||||
public string File { get; }
|
|
||||||
private readonly DateTime _pubdate;
|
|
||||||
public DateTime Pubdate => _pubdate;
|
|
||||||
|
|
||||||
public string Link { get; }
|
|
||||||
public Version Version { get; set; }
|
|
||||||
|
|
||||||
public string Language { get; set; }
|
|
||||||
|
|
||||||
public bool IsExe {
|
|
||||||
get {
|
|
||||||
if (File != null) {
|
|
||||||
return File.ToLower().EndsWith(".exe");
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsUnstable {
|
|
||||||
get {
|
|
||||||
if (File != null) {
|
|
||||||
return File.ToLower().Contains("unstable");
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsReleaseCandidate {
|
|
||||||
get {
|
|
||||||
if (File != null) {
|
|
||||||
return Regex.IsMatch(File.ToLower(), "rc[0-9]+");
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public RssFile(string file, string pubdate, string link) {
|
|
||||||
File = file;
|
|
||||||
if (!DateTime.TryParse(pubdate, out _pubdate))
|
|
||||||
{
|
|
||||||
DateTime.TryParseExact(pubdate.Replace(" UT", ""), "ddd, dd MMM yyyy HH:mm:ss", CultureInfo.InvariantCulture, DateTimeStyles.AssumeLocal, out _pubdate);
|
|
||||||
}
|
|
||||||
Link = link;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Description of RssHelper.
|
|
||||||
/// </summary>
|
|
||||||
public class RssHelper {
|
|
||||||
private static readonly ILog Log = LogManager.GetLogger(typeof(RssHelper));
|
|
||||||
private const string Rssfeed = "http://getgreenshot.org/project-feed/";
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// This is using the HTTP HEAD Method to check if the RSS Feed is modified after the supplied date
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="updateTime">DateTime</param>
|
|
||||||
/// <returns>true if the feed is newer</returns>
|
|
||||||
public static bool IsRssModifiedAfter(DateTime updateTime) {
|
|
||||||
DateTime lastModified = NetworkHelper.GetLastModified(new Uri(Rssfeed));
|
|
||||||
if (lastModified == DateTime.MinValue)
|
|
||||||
{
|
|
||||||
// Time could not be read, just take now and add one hour to it.
|
|
||||||
// This assist BUG-1850
|
|
||||||
lastModified = DateTime.Now.AddHours(1);
|
|
||||||
}
|
|
||||||
return updateTime.CompareTo(lastModified) < 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Read the Greenshot RSS feed, so we can use this information to check for updates
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>List with RssFile(s)</returns>
|
|
||||||
public static IList<RssFile> ReadRss() {
|
|
||||||
XmlDocument rssDoc = new XmlDocument();
|
|
||||||
try {
|
|
||||||
HttpWebRequest webRequest = NetworkHelper.CreateWebRequest(Rssfeed);
|
|
||||||
XmlTextReader rssReader = new XmlTextReader(webRequest.GetResponse().GetResponseStream());
|
|
||||||
|
|
||||||
// Load the XML content into a XmlDocument
|
|
||||||
rssDoc.Load(rssReader);
|
|
||||||
} catch (Exception wE) {
|
|
||||||
Log.WarnFormat("Problem reading RSS from {0}", Rssfeed);
|
|
||||||
Log.Warn(wE.Message);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Loop for the <rss> tag
|
|
||||||
XmlNode nodeRss = null;
|
|
||||||
for (int i = 0; i < rssDoc.ChildNodes.Count; i++) {
|
|
||||||
// If it is the rss tag
|
|
||||||
if (rssDoc.ChildNodes[i].Name == "rss") {
|
|
||||||
// <rss> tag found
|
|
||||||
nodeRss = rssDoc.ChildNodes[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nodeRss == null) {
|
|
||||||
Log.Debug("No RSS Feed!");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Loop for the <channel> tag
|
|
||||||
XmlNode nodeChannel = null;
|
|
||||||
for (int i = 0; i < nodeRss.ChildNodes.Count; i++) {
|
|
||||||
// If it is the channel tag
|
|
||||||
if (nodeRss.ChildNodes[i].Name == "channel") {
|
|
||||||
// <channel> tag found
|
|
||||||
nodeChannel = nodeRss.ChildNodes[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (nodeChannel == null) {
|
|
||||||
Log.Debug("No channel in RSS feed!");
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
IList<RssFile> rssFiles = new List<RssFile>();
|
|
||||||
|
|
||||||
// Loop for the <title>, <link>, <description> and all the other tags
|
|
||||||
for (int i = 0; i < nodeChannel.ChildNodes.Count; i++) {
|
|
||||||
// If it is the item tag, then it has children tags which we will add as items to the ListView
|
|
||||||
|
|
||||||
if (nodeChannel.ChildNodes[i].Name != "item")
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
XmlNode nodeItem = nodeChannel.ChildNodes[i];
|
|
||||||
string link = nodeItem["link"]?.InnerText;
|
|
||||||
string pubdate = nodeItem["pubDate"]?.InnerText;
|
|
||||||
try {
|
|
||||||
if (link == null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
Match match = Regex.Match(Uri.UnescapeDataString(link), @"^.*\/(Greenshot.+)\/download$");
|
|
||||||
if (!match.Success)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
string file = match.Groups[1].Value;
|
|
||||||
|
|
||||||
RssFile rssFile = new RssFile(file, pubdate, link);
|
|
||||||
if (file.EndsWith(".exe") ||file.EndsWith(".zip")) {
|
|
||||||
string version = Regex.Replace(file, @".*[a-zA-Z_]\-", string.Empty);
|
|
||||||
version = version.Replace(@"\-[a-zA-Z]+.*","");
|
|
||||||
version = Regex.Replace(version, @"\.exe$", string.Empty);
|
|
||||||
version = Regex.Replace(version, @"\.zip$", string.Empty);
|
|
||||||
version = Regex.Replace(version, @"RC[0-9]+", string.Empty);
|
|
||||||
if (version.Trim().Length > 0) {
|
|
||||||
version = version.Replace('-','.');
|
|
||||||
version = version.Replace(',','.');
|
|
||||||
version = Regex.Replace(version, @"^[a-zA-Z_]*\.", string.Empty);
|
|
||||||
version = Regex.Replace(version, @"\.[a-zA-Z_]*$", string.Empty);
|
|
||||||
|
|
||||||
try {
|
|
||||||
rssFile.Version = new Version(version);
|
|
||||||
} catch (Exception) {
|
|
||||||
Log.DebugFormat("Found invalid version {0} in file {1}", version, file);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
rssFiles.Add(rssFile);
|
|
||||||
}
|
|
||||||
} catch (Exception ex) {
|
|
||||||
Log.WarnFormat("Couldn't read RSS entry for: {0}", nodeChannel["title"]?.InnerText);
|
|
||||||
Log.Warn("Reason: ", ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return rssFiles;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Add table
Add a link
Reference in a new issue