Итак, я просмотрел многие ответы, уже опубликованные в StackOverflow, и не могу найти ни одного подходящего.
Во-первых, позвольте мне объяснить свое приложение Xamarin Forms. У меня традиционная структура решения (библиотека классов .Net Standard 2.0 с проектами приложений Andoid и iOS). Приложение, над которым я работаю, является частью набора приложений, которые разделяют большой объем кода (верхний колонтитул, нижний колонтитул, обработка ошибок и т. Д.) Через дополнительную стандартную библиотеку классов .Net 2.0. В этом классе у меня есть модель просмотра для обработки исключений, а также стандартное представление содержимого заголовка, используемое во всех приложениях, входящих в комплект.
В заголовке я показываю значок ошибки в области заголовка, который пользователь может щелкнуть, чтобы сообщить об ошибках и т. Д., Когда обработчик исключений добавляет исключение. Я привязываю свойство IsVisible элемента управления изображением для значка ошибки к свойству в модели обработки исключений. Каждый раз, когда отображается моя страница (все страницы используют общий заголовок), вызывается get для свойства в модели представления, к которому привязано свойство IsVisible. Похоже, привязка настроена правильно.
Проблема в том, что когда я добавляю ошибку с помощью модели представления обработки исключений, а затем пытаюсь уведомить пользовательский интерфейс об изменении привязанного свойства, объект события PropertyChange определяется следующим образом:
public event PropertyChangedEventHandler PropertyChanged;
всегда равно нулю. Таким образом, get для связанного свойства никогда не вызывается, и мой пользовательский интерфейс не обновляется. Я надеюсь, что это что-то простое, но я не смог понять это.
ОБРАТИТЕ ВНИМАНИЕ !! Другое решение в пакете (только с заглушкой приложения UWP), в котором используются те же самые представления и код, работает точно так, как ожидалось. Другими словами, когда обработчик исключений добавляет ошибку, опрашивается связанное свойство и отображается значок ошибки, как и ожидалось.
Ниже мой код. Будем очень благодарны любой помощи.
КОД XAML ДЛЯ ПРОСМОТРА КОНТЕНТА APPHEADER
<?xml version="1.0" encoding="UTF-8"?>
<ContentView xmlns="http://xamarin.com/schemas/2014/forms"
xmlns:x="http://schemas.microsoft.com/winfx/2009/xaml"
xmlns:sfGrad ="clr-namespace:Syncfusion.XForms.Graphics;assembly=Syncfusion.Core.XForms"
BackgroundColor="#4D648D"
x:Class="FCISharedAll.AppHeader">
<ContentView.Content>
<StackLayout x:Name="slMaster" HorizontalOptions="Fill" VerticalOptions="Start" Orientation="Vertical" Margin="0,0,0,-6"
Spacing="0">
<Grid x:Name="grdGradient" HorizontalOptions="Fill" VerticalOptions="Start" HeightRequest="110" MinimumHeightRequest="110">
<sfGrad:SfGradientView>
<sfGrad:SfGradientView.BackgroundBrush>
<sfGrad:SfLinearGradientBrush StartPoint="0.5, 0" EndPoint="0.5, 1">
<sfGrad:SfLinearGradientBrush.GradientStops>
<sfGrad:SfGradientStop Color="#4D648D" Offset="0.0" />
<sfGrad:SfGradientStop Color="#283655" Offset="0.5" />
<sfGrad:SfGradientStop Color="#4D648D" Offset="1.0" />
</sfGrad:SfLinearGradientBrush.GradientStops>
</sfGrad:SfLinearGradientBrush>
</sfGrad:SfGradientView.BackgroundBrush>
</sfGrad:SfGradientView>
<Grid x:Name="grdHeader" HorizontalOptions="Fill" VerticalOptions="Start">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="*" />
</Grid.ColumnDefinitions>
<Image x:Name="imgHeader" Source="fciapp.png" HeightRequest="110" MinimumHeightRequest="110" HorizontalOptions="Fill"
VerticalOptions="Center" Aspect="AspectFit" Grid.Column="0" />
<Image x:Name="imgError" Source="error.png" HorizontalOptions="End" VerticalOptions="Center" Margin="8,8,8,8"
IsVisible="{Binding ShowErrorIcon}" Grid.Column="0">
<Image.GestureRecognizers>
<TapGestureRecognizer Tapped="imgError_Tapped" />
</Image.GestureRecognizers>
</Image>
</Grid>
</Grid>
</StackLayout>
</ContentView.Content>
</ContentView>
КОД C # ДЛЯ ПРОСМОТРА КОНТЕНТА APPHEADER
using Rg.Plugins.Popup.Services;
using System;
using Xamarin.Forms;
using Xamarin.Forms.Xaml;
namespace FCISharedAll
{
[XamlCompilation(XamlCompilationOptions.Compile)]
public partial class AppHeader : ContentView
{
public AppHeader()
{
InitializeComponent();
NavigationPage.SetHasNavigationBar(this, false);
imgError.BindingContext = FCISharedAllHelperClass.gvm_GlobalExceptionClass;
}
private async void imgError_Tapped(object sender, EventArgs e)
{
try
{
await PopupNavigation.Instance.PushAsync(new FCISharedAll.ExceptionDisplay(FCISharedAllHelperClass.gvm_GlobalExceptionClass));
}
catch (Exception ex)
{
SharedErrorHandler.ProcessException(ex);
}
}
}
}
КОД C # ДЛЯ EXCEPTION HANDLER VIEWMODEL
namespace FCISharedAll
{
public class CompleteException
{
public DateTime Error_DateTime { get; set; }
public Exception Error { get; set; }
}
public class ExceptionProcessor : INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
private List<CompleteException> iobj_ApplicationExceptions { get; set; }
public ExceptionProcessor()
{
try
{
if (iobj_ApplicationExceptions == null)
{
iobj_ApplicationExceptions = new List<CompleteException>();
}
}
catch (Exception ex)
{
iobj_ApplicationExceptions.Add(new CompleteException() { Error_DateTime = DateTime.Now, Error = ex });
}
}
public bool ShowErrorIcon
{
get
{
return (iobj_ApplicationExceptions == null || iobj_ApplicationExceptions.Count > 0);
}
}
public void AddExceptionToList(Exception pobj_Exception)
{
try
{
iobj_ApplicationExceptions.Add(new CompleteException() { Error_DateTime = DateTime.Now, Error = pobj_Exception });
//NotifyPropertyChanged(ge_PreferenceKey.ShowErrorIcon.ToString());
NotifyPropertyChanged("ShowErrorIcon");
}
catch (Exception ex)
{
iobj_ApplicationExceptions.Add(new CompleteException() { Error_DateTime = DateTime.Now, Error = ex });
}
}
public int ErrorCount
{
get
{
int li_ReturnValue = 0;
if (iobj_ApplicationExceptions != null)
{
li_ReturnValue = iobj_ApplicationExceptions.Count;
}
return li_ReturnValue;
}
}
public void RemoveProcessedExceptions(int pi_NumberOfExceptionsToInclude)
{
int li_ErrorsToProcess = 0;
try
{
li_ErrorsToProcess = (pi_NumberOfExceptionsToInclude <= iobj_ApplicationExceptions.Count ? pi_NumberOfExceptionsToInclude : iobj_ApplicationExceptions.Count);
for (int li_IX = 0; li_IX <= li_ErrorsToProcess - 1; li_IX++)
{
iobj_ApplicationExceptions.RemoveAt(0);
}
NotifyPropertyChanged(ge_PreferenceKey.ErrorCount.ToString());
NotifyPropertyChanged(ge_PreferenceKey.ShowErrorIcon.ToString());
}
catch (Exception ex)
{
iobj_ApplicationExceptions.Add(new CompleteException() { Error_DateTime = DateTime.Now, Error = ex });
}
}
public string PrepareErrorText(int pi_NumberOfExceptionsToInclude = 5)
{
string ls_EmailText = "";
int li_ErrorsToProcess = 0;
try
{
li_ErrorsToProcess = (pi_NumberOfExceptionsToInclude <= iobj_ApplicationExceptions.Count ? pi_NumberOfExceptionsToInclude : iobj_ApplicationExceptions.Count);
for (int li_IX = 0; li_IX <= li_ErrorsToProcess - 1; li_IX++)
{
ls_EmailText += FormatException(iobj_ApplicationExceptions[li_IX]) + System.Environment.NewLine;
}
}
catch (Exception ex)
{
iobj_ApplicationExceptions.Add(new CompleteException() { Error_DateTime = DateTime.Now, Error = ex });
}
return ls_EmailText;
}
private string FormatException(CompleteException pobj_CompleteException)
{
string ls_ReturnText = "";
ls_ReturnText = "Date: " + pobj_CompleteException.Error_DateTime + System.Environment.NewLine +
"Message: " + (pobj_CompleteException.Error.Message == null ? "" : pobj_CompleteException.Error.Message + System.Environment.NewLine) +
"Source: " + (pobj_CompleteException.Error.Source == null ? "" : pobj_CompleteException.Error.Source + System.Environment.NewLine) +
"Stack Trace: " + (pobj_CompleteException.Error.StackTrace == null ? "" : pobj_CompleteException.Error.StackTrace + System.Environment.NewLine);
if (pobj_CompleteException.Error.InnerException != null)
{
ls_ReturnText += "Inner Exception Message: " + (pobj_CompleteException.Error.InnerException.Message == null ? "" : pobj_CompleteException.Error.InnerException.Message + System.Environment.NewLine) +
"Inner Exception Source: " + (pobj_CompleteException.Error.InnerException.Source == null ? "" : pobj_CompleteException.Error.InnerException.Source + System.Environment.NewLine) +
"Inner Exception Stack Trace: " + (pobj_CompleteException.Error.InnerException.StackTrace == null ? "" : pobj_CompleteException.Error.InnerException.StackTrace);
}
if (ls_ReturnText.LastIndexOf(System.Environment.NewLine) > ls_ReturnText.Length - 5)
{
//Remove the last line linefeed
ls_ReturnText = ls_ReturnText.Substring(0, ls_ReturnText.LastIndexOf(System.Environment.NewLine));
}
return ls_ReturnText;
}
/// <summary>
/// Manual Notification to subscribers that a property has changed.
/// </summary>
private void NotifyPropertyChanged(String propertyName)
{
PropertyChangedEventHandler handler = PropertyChanged;
if (null != handler)
{
handler(this, new PropertyChangedEventArgs(propertyName));
}
}
protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null)
{
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
}
}
}
Вот код для вспомогательного класса в моем проекте общего кода, который создает экземпляр модели представления обработки исключений как статический объект.
using FCISharedAll.FCICommObjects;
using System;
namespace FCISharedAll
{
public static class FCISharedAllHelperClass
{
public static ExceptionProcessor gvm_GlobalExceptionClass = new ExceptionProcessor();
public static ThemeConfig gobj_ThemeConfig { get; set; }
public static bool IsCheckInPage { get; set; } = false;
//internal
private static KioskInfo iobj_KioskInfo = null;
public static KioskInfo HostKioskInfo
{
get
{
if (iobj_KioskInfo == null)
{
iobj_KioskInfo = new KioskInfo();
}
return iobj_KioskInfo;
}
set
{
iobj_KioskInfo = value;
}
}
public static void ConfigureTheme(bool pb_UseKioskTheme)
{
try
{
gobj_ThemeConfig = new ThemeConfig(pb_UseKioskTheme);
}
catch (Exception ex)
{
SharedErrorHandler.ProcessException(ex);
}
}
}
}
Я знаю это много, но я был бы признателен за любую помощь, которая может быть предоставлена.
ОБНОВИТЬ!!! ARG - вы знаете, что после того, как разместите вопрос, вы думаете, что разберетесь. Я считаю, что на самом деле я получал два разных экземпляра своего класса обработчика исключений, один из библиотеки классов Xamarin forms .net и один в моем общем коде. Я удаляю модель представления из моей библиотеки классов xamarin forms .net, чтобы увидеть, решит ли это проблему, и я обновлю.