4

我需要一种方法让 viewmodel 指示XamDataGrid视图以最小的麻烦重新读取和重新绘制其单元格。我不想弄乱源并做一些不可持续的解决方法来提高其事件(源可能会改变)。

为了使其更易于理解,我有一个全局静态类,它包含一些不影响数据的视觉提示配置,但它们仅以它们在网格中表示的方式(缩放、格式化等)。视觉动作发生在IValueConverter附加到该字段的实现中,效果很好。当提示更改并且视图模型订阅它并且事件正确触发时,会触发一个静态事件。现在我只需要让事件处理程序导致网格重新绘制。

有什么建议么?

编辑:一些代码,如果有帮助:

转换器:

public class ScaleConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    {
        if (targetType == typeof(double) || targetType == typeof(double?))
        {
            if (value == null && targetType == typeof(double?)) return null;  // short circuit for null->null, nothing to scale
            if (value is double) return (double)value / DisplayScale.Scale; // shortcut for direct scaling, no need to waste time on conversion
            try
            {
                return System.Convert.ToDouble(value) / DisplayScale.Scale; // for convertible values, eat conversion exception
            }
            catch (Exception) {};
            // if all esle fails return NaN
            return double.NaN; 
        }
        // fallthrough, return null, this should not happen, if it does it means converter is incomplete
        return null;
    }
 ...
}

DisplayScale 全局

public class DisplayScale: NotificationObject
{
    private static KeyValuePair<string, double> _ActiveScaling;
    private static readonly object _lockHandle = new object(); // note: should not contest, but just to be on the safe side
    public static event Action ScalingChanged;

    public static List<KeyValuePair<string, double>> ScaleList { get; private set; }

    static DisplayScale()
    {
        ScaleList = new List<KeyValuePair<string, double>>() 
        { 
            new KeyValuePair<string, double>("No scaling (1's)", 1d),
            new KeyValuePair<string, double>("Thousands (1,000's)", 1000d),
            new KeyValuePair<string, double>("Millions (1,000,000's)", 1000000d),
            new KeyValuePair<string, double>("Billions (1,000,000,000's)", 1000000000d)
         };
        ActiveScaling = ScaleList.First();  // OPTION: could be in per-user config
    }

    public static double Scale { get { return ActiveScaling.Value; } }
    public static KeyValuePair<string, double> ActiveScaling
    {
        get
        {
            lock (_lockHandle) return _ActiveScaling;
        }
        set
        {
            lock (_lockHandle)
            {
                _ActiveScaling = value;
                var eventCall = ScalingChanged;
                if (eventCall != null) eventCall();
            }
        }
    }
}

字段定义为

// resource
<inf:ScaleConverter x:Key="ScaleConverter" />
// field
<igDP:Field Name="Deposit" Converter="{StaticResource ScaleConverter}">
4

2 回答 2

4

如果您有一个集合视图,而不是简单地调用 Refresh() 。

public class YourViewModel
{
  private ObservableCollection<YourDataClass> yourColl;

  public void YourViewModel()
  {
    yourColl = new ObservableCollection<YourDataClass>();
    YourCollectionView = CollectionViewSource.GetDefaultView(yourColl);
    DisplayScale.ScalingChanged += () => YourCollectionView.Refresh();
  }

  var ICollectionView yourCollView;
  public ICollectionView YourCollectionView
  {
    get { yourCollView; }
    set {
      yourCollView = value;
      RaisePropertyChanged("YourCollectionView");
    }
  }
}
于 2011-12-05T13:18:43.667 回答
0

我遇到了同样的问题,事实证明,即使我的 ViewModel 实现了 INotifyPropertyChanged,我也没有在我的 ObservableCollection 中网格行绑定到的类上实现 INotifyPropertyChanged(即punker76答案中的 YourDataClass)。一旦我在那个类上实现了它,一切都开始按预期更新。

我知道您提到“我不想弄乱源代码”,所以我不确定这个解决方案是否适合您,但我想我也会分享给其他找到这篇文章的人。

于 2013-10-11T18:47:19.123 回答