1 | using System;
|
---|
2 | using System.Collections.Generic;
|
---|
3 | using System.Collections.Specialized;
|
---|
4 | using System.ComponentModel;
|
---|
5 | using System.Diagnostics;
|
---|
6 | using System.IO;
|
---|
7 | using System.Linq;
|
---|
8 | using System.Windows;
|
---|
9 | using System.Windows.Automation.Peers;
|
---|
10 | using System.Windows.Controls;
|
---|
11 | using System.Windows.Markup;
|
---|
12 | using System.Windows.Media.Imaging;
|
---|
13 | using Microsoft.Research.DynamicDataDisplay.Common;
|
---|
14 | using Microsoft.Research.DynamicDataDisplay.Common.Auxiliary;
|
---|
15 | using Microsoft.Research.DynamicDataDisplay.Common.UndoSystem;
|
---|
16 | using Microsoft.Research.DynamicDataDisplay.Navigation;
|
---|
17 |
|
---|
18 | namespace Microsoft.Research.DynamicDataDisplay
|
---|
19 | {
|
---|
20 | /// <summary>Plotter is a base control for displaying various graphs. It provides
|
---|
21 | /// means to draw chart itself and side space for axes, annotations, etc</summary>
|
---|
22 | [ContentProperty("Children")]
|
---|
23 | [TemplatePart(Name = "PART_HeaderPanel", Type = typeof(StackPanel))]
|
---|
24 | [TemplatePart(Name = "PART_FooterPanel", Type = typeof(StackPanel))]
|
---|
25 |
|
---|
26 | [TemplatePart(Name = "PART_BottomPanel", Type = typeof(StackPanel))]
|
---|
27 | [TemplatePart(Name = "PART_LeftPanel", Type = typeof(StackPanel))]
|
---|
28 | [TemplatePart(Name = "PART_RightPanel", Type = typeof(StackPanel))]
|
---|
29 | [TemplatePart(Name = "PART_TopPanel", Type = typeof(StackPanel))]
|
---|
30 |
|
---|
31 | [TemplatePart(Name = "PART_MainCanvas", Type = typeof(Canvas))]
|
---|
32 | [TemplatePart(Name = "PART_CentralGrid", Type = typeof(Grid))]
|
---|
33 | [TemplatePart(Name = "PART_MainGrid", Type = typeof(Grid))]
|
---|
34 | [TemplatePart(Name = "PART_ContentsGrid", Type = typeof(Grid))]
|
---|
35 | [TemplatePart(Name = "PART_ParallelCanvas", Type = typeof(Canvas))]
|
---|
36 | public abstract class Plotter : ContentControl
|
---|
37 | {
|
---|
38 | private PlotterLoadMode loadMode = PlotterLoadMode.Normal;
|
---|
39 | protected PlotterLoadMode LoadMode
|
---|
40 | {
|
---|
41 | get { return loadMode; }
|
---|
42 | }
|
---|
43 |
|
---|
44 | private static Plotter current;
|
---|
45 | /// <summary>
|
---|
46 | /// Gets the current plotter. Used in VisualDebug.
|
---|
47 | /// </summary>
|
---|
48 | /// <value>The current.</value>
|
---|
49 | internal static Plotter Current
|
---|
50 | {
|
---|
51 | get { return Plotter.current; }
|
---|
52 | }
|
---|
53 |
|
---|
54 | protected Plotter() : this(PlotterLoadMode.Normal) { }
|
---|
55 |
|
---|
56 | /// <summary>
|
---|
57 | /// Initializes a new instance of the <see cref="Plotter"/> class.
|
---|
58 | /// </summary>
|
---|
59 | protected Plotter(PlotterLoadMode loadMode)
|
---|
60 | {
|
---|
61 | current = this;
|
---|
62 |
|
---|
63 | this.loadMode = loadMode;
|
---|
64 |
|
---|
65 | SetPlotter(this, this);
|
---|
66 |
|
---|
67 | if (loadMode == PlotterLoadMode.Normal)
|
---|
68 | {
|
---|
69 | UpdateUIParts();
|
---|
70 | }
|
---|
71 |
|
---|
72 | children = new PlotterChildrenCollection(this);
|
---|
73 | children.CollectionChanged += OnChildrenCollectionChanged;
|
---|
74 | Loaded += Plotter_Loaded;
|
---|
75 | Unloaded += Plotter_Unloaded;
|
---|
76 |
|
---|
77 | genericResources = (ResourceDictionary)Application.LoadComponent(new Uri("/DynamicDataDisplay;component/Themes/Generic.xaml", UriKind.Relative));
|
---|
78 |
|
---|
79 | ContextMenu = null;
|
---|
80 |
|
---|
81 | foreach (var panel in GetAllPanels().Where(panel => panel != null))
|
---|
82 | {
|
---|
83 | Plotter.SetIsDefaultPanel(panel, true);
|
---|
84 | }
|
---|
85 | }
|
---|
86 |
|
---|
87 | private void Plotter_Unloaded(object sender, RoutedEventArgs e)
|
---|
88 | {
|
---|
89 | OnUnloaded();
|
---|
90 | }
|
---|
91 |
|
---|
92 | protected virtual void OnUnloaded() { }
|
---|
93 |
|
---|
94 | protected override AutomationPeer OnCreateAutomationPeer()
|
---|
95 | {
|
---|
96 | return new PlotterAutomationPeer(this);
|
---|
97 | }
|
---|
98 |
|
---|
99 | [EditorBrowsable(EditorBrowsableState.Never)]
|
---|
100 | public override bool ShouldSerializeContent()
|
---|
101 | {
|
---|
102 | return false;
|
---|
103 | }
|
---|
104 |
|
---|
105 | protected override bool ShouldSerializeProperty(DependencyProperty dp)
|
---|
106 | {
|
---|
107 | // do not serialize context menu if it was created by DefaultContextMenu, because that context menu items contains references of plotter
|
---|
108 | if (dp == ContextMenuProperty && children.Any(el => el is DefaultContextMenu)) return false;
|
---|
109 | if (dp == TemplateProperty) return false;
|
---|
110 | if (dp == ContentProperty) return false;
|
---|
111 |
|
---|
112 | return base.ShouldSerializeProperty(dp);
|
---|
113 | }
|
---|
114 |
|
---|
115 | private const string templateKey = "defaultPlotterTemplate";
|
---|
116 | private const string styleKey = "defaultPlotterStyle";
|
---|
117 | private void UpdateUIParts()
|
---|
118 | {
|
---|
119 | ResourceDictionary dict = new ResourceDictionary
|
---|
120 | {
|
---|
121 | Source = new Uri("/DynamicDataDisplay;component/Common/PlotterStyle.xaml", UriKind.Relative)
|
---|
122 | };
|
---|
123 |
|
---|
124 | Resources.MergedDictionaries.Add(dict);
|
---|
125 |
|
---|
126 | Style = (Style)dict[styleKey];
|
---|
127 |
|
---|
128 | ControlTemplate template = (ControlTemplate)dict[templateKey];
|
---|
129 | Template = template;
|
---|
130 | ApplyTemplate();
|
---|
131 | }
|
---|
132 |
|
---|
133 | private ResourceDictionary genericResources;
|
---|
134 | protected ResourceDictionary GenericResources
|
---|
135 | {
|
---|
136 | get { return genericResources; }
|
---|
137 | }
|
---|
138 |
|
---|
139 | /// <summary>
|
---|
140 | /// Forces plotter to load.
|
---|
141 | /// </summary>
|
---|
142 | public void PerformLoad()
|
---|
143 | {
|
---|
144 | isLoadedIntensionally = true;
|
---|
145 | ApplyTemplate();
|
---|
146 |
|
---|
147 | Plotter_Loaded(null, null);
|
---|
148 | }
|
---|
149 |
|
---|
150 | private bool isLoadedIntensionally = false;
|
---|
151 | protected virtual bool IsLoadedInternal
|
---|
152 | {
|
---|
153 | get { return isLoadedIntensionally || IsLoaded; }
|
---|
154 | }
|
---|
155 |
|
---|
156 | private void Plotter_Loaded(object sender, RoutedEventArgs e)
|
---|
157 | {
|
---|
158 | ExecuteWaitingChildrenAdditions();
|
---|
159 |
|
---|
160 | OnLoaded();
|
---|
161 | }
|
---|
162 |
|
---|
163 | protected internal void ExecuteWaitingChildrenAdditions()
|
---|
164 | {
|
---|
165 | executedWaitingChildrenAdding = true;
|
---|
166 |
|
---|
167 | foreach (var pair in waitingForExecute)
|
---|
168 | {
|
---|
169 | pair.Value.Invoke();
|
---|
170 | }
|
---|
171 |
|
---|
172 | waitingForExecute.Clear();
|
---|
173 | }
|
---|
174 |
|
---|
175 | protected virtual void OnLoaded()
|
---|
176 | {
|
---|
177 | // this is done to enable keyboard shortcuts
|
---|
178 | Focus();
|
---|
179 | }
|
---|
180 |
|
---|
181 | protected override void OnTemplateChanged(ControlTemplate oldTemplate, ControlTemplate newTemplate)
|
---|
182 | {
|
---|
183 | base.OnTemplateChanged(oldTemplate, newTemplate);
|
---|
184 | }
|
---|
185 |
|
---|
186 | private Grid contentsGrid;
|
---|
187 | public override void OnApplyTemplate()
|
---|
188 | {
|
---|
189 | base.OnApplyTemplate();
|
---|
190 |
|
---|
191 | var headerPanel = GetPart<StackPanel>("PART_HeaderPanel");
|
---|
192 | MigrateChildren(this.headerPanel, headerPanel);
|
---|
193 | this.headerPanel = headerPanel;
|
---|
194 |
|
---|
195 | var footerPanel = GetPart<StackPanel>("PART_FooterPanel");
|
---|
196 | MigrateChildren(this.footerPanel, footerPanel);
|
---|
197 | this.footerPanel = footerPanel;
|
---|
198 |
|
---|
199 | var leftPanel = GetPart<StackPanel>("PART_LeftPanel");
|
---|
200 | MigrateChildren(this.leftPanel, leftPanel);
|
---|
201 | this.leftPanel = leftPanel;
|
---|
202 |
|
---|
203 | var bottomPanel = GetPart<StackPanel>("PART_BottomPanel");
|
---|
204 | MigrateChildren(this.bottomPanel, bottomPanel);
|
---|
205 | this.bottomPanel = bottomPanel;
|
---|
206 |
|
---|
207 | var rightPanel = GetPart<StackPanel>("PART_RightPanel");
|
---|
208 | MigrateChildren(this.rightPanel, rightPanel);
|
---|
209 | this.rightPanel = rightPanel;
|
---|
210 |
|
---|
211 | var topPanel = GetPart<StackPanel>("PART_TopPanel");
|
---|
212 | MigrateChildren(this.topPanel, topPanel);
|
---|
213 | this.topPanel = topPanel;
|
---|
214 |
|
---|
215 | var mainCanvas = GetPart<Canvas>("PART_MainCanvas");
|
---|
216 | MigrateChildren(this.mainCanvas, mainCanvas);
|
---|
217 | this.mainCanvas = mainCanvas;
|
---|
218 |
|
---|
219 | var centralGrid = GetPart<Grid>("PART_CentralGrid");
|
---|
220 | MigrateChildren(this.centralGrid, centralGrid);
|
---|
221 | this.centralGrid = centralGrid;
|
---|
222 |
|
---|
223 | var mainGrid = GetPart<Grid>("PART_MainGrid");
|
---|
224 | MigrateChildren(this.mainGrid, mainGrid);
|
---|
225 | this.mainGrid = mainGrid;
|
---|
226 |
|
---|
227 | var parallelCanvas = GetPart<Canvas>("PART_ParallelCanvas");
|
---|
228 | MigrateChildren(this.parallelCanvas, parallelCanvas);
|
---|
229 | this.parallelCanvas = parallelCanvas;
|
---|
230 |
|
---|
231 | var contentsGrid = GetPart<Grid>("PART_ContentsGrid");
|
---|
232 | MigrateChildren(this.contentsGrid, contentsGrid);
|
---|
233 | this.contentsGrid = contentsGrid;
|
---|
234 |
|
---|
235 | Content = contentsGrid;
|
---|
236 | AddLogicalChild(contentsGrid);
|
---|
237 |
|
---|
238 | if (contentsGrid != null)
|
---|
239 | ExecuteWaitingChildrenAdditions();
|
---|
240 | }
|
---|
241 |
|
---|
242 | private void MigrateChildren(Panel previousParent, Panel currentParent)
|
---|
243 | {
|
---|
244 | if (previousParent != null && currentParent != null)
|
---|
245 | {
|
---|
246 | UIElement[] children = new UIElement[previousParent.Children.Count];
|
---|
247 | previousParent.Children.CopyTo(children, 0);
|
---|
248 | previousParent.Children.Clear();
|
---|
249 |
|
---|
250 | foreach (var child in children)
|
---|
251 | {
|
---|
252 | bool isDefaultPanel = GetIsDefaultPanel(child);
|
---|
253 | if (!currentParent.Children.Contains(child) && !isDefaultPanel)
|
---|
254 | {
|
---|
255 | currentParent.Children.Add(child);
|
---|
256 | }
|
---|
257 | }
|
---|
258 | }
|
---|
259 | else if (previousParent != null)
|
---|
260 | {
|
---|
261 | previousParent.Children.Clear();
|
---|
262 | }
|
---|
263 | }
|
---|
264 |
|
---|
265 | internal virtual IEnumerable<Panel> GetAllPanels()
|
---|
266 | {
|
---|
267 | yield return headerPanel;
|
---|
268 | yield return footerPanel;
|
---|
269 |
|
---|
270 | yield return leftPanel;
|
---|
271 | yield return bottomPanel;
|
---|
272 | yield return rightPanel;
|
---|
273 | yield return topPanel;
|
---|
274 |
|
---|
275 | yield return mainCanvas;
|
---|
276 | yield return centralGrid;
|
---|
277 | yield return mainGrid;
|
---|
278 | yield return parallelCanvas;
|
---|
279 | yield return contentsGrid;
|
---|
280 | }
|
---|
281 |
|
---|
282 | private T GetPart<T>(string name)
|
---|
283 | {
|
---|
284 | return (T)Template.FindName(name, this);
|
---|
285 | }
|
---|
286 |
|
---|
287 | #region Children and add/removed events handling
|
---|
288 |
|
---|
289 | private readonly PlotterChildrenCollection children;
|
---|
290 |
|
---|
291 | /// <summary>
|
---|
292 | /// Provides access to Plotter's children charts.
|
---|
293 | /// </summary>
|
---|
294 | /// <value>The children.</value>
|
---|
295 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Content)]
|
---|
296 | public PlotterChildrenCollection Children
|
---|
297 | {
|
---|
298 | [DebuggerStepThrough]
|
---|
299 | get { return children; }
|
---|
300 | }
|
---|
301 |
|
---|
302 | private readonly Dictionary<IPlotterElement, Action> waitingForExecute = new Dictionary<IPlotterElement, Action>();
|
---|
303 |
|
---|
304 | bool executedWaitingChildrenAdding = false;
|
---|
305 | private void OnChildrenCollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
|
---|
306 | {
|
---|
307 | if (IsLoadedInternal && !executedWaitingChildrenAdding)
|
---|
308 | {
|
---|
309 | ExecuteWaitingChildrenAdditions();
|
---|
310 | executedWaitingChildrenAdding = true;
|
---|
311 | }
|
---|
312 |
|
---|
313 | if (e.NewItems != null)
|
---|
314 | {
|
---|
315 | foreach (IPlotterElement item in e.NewItems)
|
---|
316 | {
|
---|
317 | if (IsLoadedInternal)
|
---|
318 | {
|
---|
319 | OnChildAdded(item);
|
---|
320 | }
|
---|
321 | else
|
---|
322 | {
|
---|
323 | waitingForExecute.Remove(item);
|
---|
324 | waitingForExecute.Add(item, () => OnChildAdded(item));
|
---|
325 | }
|
---|
326 | }
|
---|
327 | }
|
---|
328 | if (e.OldItems != null)
|
---|
329 | {
|
---|
330 | foreach (IPlotterElement item in e.OldItems)
|
---|
331 | {
|
---|
332 | if (IsLoadedInternal)
|
---|
333 | {
|
---|
334 | OnChildRemoving(item);
|
---|
335 | }
|
---|
336 | else
|
---|
337 | {
|
---|
338 | waitingForExecute.Remove(item);
|
---|
339 | waitingForExecute.Add(item, () => OnChildRemoving(item));
|
---|
340 | }
|
---|
341 | }
|
---|
342 | }
|
---|
343 | }
|
---|
344 |
|
---|
345 | bool performChildChecks = true;
|
---|
346 | internal bool PerformChildChecks
|
---|
347 | {
|
---|
348 | get { return performChildChecks; }
|
---|
349 | set { performChildChecks = value; }
|
---|
350 | }
|
---|
351 |
|
---|
352 | private IPlotterElement currentChild = null;
|
---|
353 | protected IPlotterElement CurrentChild
|
---|
354 | {
|
---|
355 | get { return currentChild; }
|
---|
356 | }
|
---|
357 |
|
---|
358 | protected virtual void OnChildAdded(IPlotterElement child)
|
---|
359 | {
|
---|
360 | if (child != null)
|
---|
361 | {
|
---|
362 | currentChild = child;
|
---|
363 |
|
---|
364 | if (performChildChecks && child.Plotter != null)
|
---|
365 | {
|
---|
366 | throw new InvalidOperationException(Strings.Exceptions.PlotterElementAddedToAnotherPlotter);
|
---|
367 | }
|
---|
368 |
|
---|
369 | if (performChildChecks)
|
---|
370 | {
|
---|
371 | child.OnPlotterAttached(this);
|
---|
372 | if (child.Plotter != this)
|
---|
373 | {
|
---|
374 | throw new InvalidOperationException(Strings.Exceptions.InvalidParentPlotterValue);
|
---|
375 | }
|
---|
376 | }
|
---|
377 |
|
---|
378 | DependencyObject dependencyObject = child as DependencyObject;
|
---|
379 | if (dependencyObject != null)
|
---|
380 | {
|
---|
381 | SetPlotter(dependencyObject, this);
|
---|
382 | }
|
---|
383 | }
|
---|
384 | }
|
---|
385 |
|
---|
386 | protected virtual void OnChildRemoving(IPlotterElement child)
|
---|
387 | {
|
---|
388 | if (child != null)
|
---|
389 | {
|
---|
390 | // todo probably here child.Plotter can be null.
|
---|
391 | if (performChildChecks && child.Plotter != this)
|
---|
392 | {
|
---|
393 | //throw new InvalidOperationException(Strings.Exceptions.InvalidParentPlotterValueRemoving);
|
---|
394 | }
|
---|
395 |
|
---|
396 | if (performChildChecks)
|
---|
397 | {
|
---|
398 | if (child.Plotter != null)
|
---|
399 | child.OnPlotterDetaching(this);
|
---|
400 |
|
---|
401 | if (child.Plotter != null)
|
---|
402 | {
|
---|
403 | throw new InvalidOperationException(Strings.Exceptions.ParentPlotterNotNull);
|
---|
404 | }
|
---|
405 | }
|
---|
406 |
|
---|
407 | DependencyObject dependencyObject = child as DependencyObject;
|
---|
408 | if (dependencyObject != null)
|
---|
409 | {
|
---|
410 | SetPlotter(dependencyObject, null);
|
---|
411 | }
|
---|
412 | }
|
---|
413 | }
|
---|
414 |
|
---|
415 | #endregion
|
---|
416 |
|
---|
417 | #region Layout zones
|
---|
418 |
|
---|
419 | private Panel parallelCanvas = new Canvas();
|
---|
420 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
421 | public Panel ParallelCanvas
|
---|
422 | {
|
---|
423 | get { return parallelCanvas; }
|
---|
424 | protected set { parallelCanvas = value; }
|
---|
425 | }
|
---|
426 |
|
---|
427 | private Panel headerPanel = new StackPanel();
|
---|
428 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
429 | public Panel HeaderPanel
|
---|
430 | {
|
---|
431 | get { return headerPanel; }
|
---|
432 | protected set { headerPanel = value; }
|
---|
433 | }
|
---|
434 |
|
---|
435 | private Panel footerPanel = new StackPanel();
|
---|
436 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
437 | public Panel FooterPanel
|
---|
438 | {
|
---|
439 | get { return footerPanel; }
|
---|
440 | protected set { footerPanel = value; }
|
---|
441 | }
|
---|
442 |
|
---|
443 | private Panel leftPanel = new StackPanel();
|
---|
444 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
445 | public Panel LeftPanel
|
---|
446 | {
|
---|
447 | get { return leftPanel; }
|
---|
448 | protected set { leftPanel = value; }
|
---|
449 | }
|
---|
450 |
|
---|
451 | private Panel rightPanel = new StackPanel();
|
---|
452 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
453 | public Panel RightPanel
|
---|
454 | {
|
---|
455 | get { return rightPanel; }
|
---|
456 | protected set { rightPanel = value; }
|
---|
457 | }
|
---|
458 |
|
---|
459 | private Panel topPanel = new StackPanel();
|
---|
460 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
461 | public Panel TopPanel
|
---|
462 | {
|
---|
463 | get { return topPanel; }
|
---|
464 | protected set { topPanel = value; }
|
---|
465 | }
|
---|
466 |
|
---|
467 | private Panel bottomPanel = new StackPanel();
|
---|
468 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
469 | public Panel BottomPanel
|
---|
470 | {
|
---|
471 | get { return bottomPanel; }
|
---|
472 | protected set { bottomPanel = value; }
|
---|
473 | }
|
---|
474 |
|
---|
475 | private Panel mainCanvas = new Canvas();
|
---|
476 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
477 | public Panel MainCanvas
|
---|
478 | {
|
---|
479 | get { return mainCanvas; }
|
---|
480 | protected set { mainCanvas = value; }
|
---|
481 | }
|
---|
482 |
|
---|
483 | private Panel centralGrid = new Grid();
|
---|
484 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
485 | public Panel CentralGrid
|
---|
486 | {
|
---|
487 | get { return centralGrid; }
|
---|
488 | protected set { centralGrid = value; }
|
---|
489 | }
|
---|
490 |
|
---|
491 | private Panel mainGrid = new Grid();
|
---|
492 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
493 | public Panel MainGrid
|
---|
494 | {
|
---|
495 | get { return mainGrid; }
|
---|
496 | protected set { mainGrid = value; }
|
---|
497 | }
|
---|
498 |
|
---|
499 | #endregion
|
---|
500 |
|
---|
501 | #region Screenshots & copy to clipboard
|
---|
502 |
|
---|
503 | public BitmapSource CreateScreenshot()
|
---|
504 | {
|
---|
505 | UIElement parent = (UIElement)Parent;
|
---|
506 |
|
---|
507 | Rect renderBounds = new Rect(RenderSize);
|
---|
508 |
|
---|
509 | Point p1 = renderBounds.TopLeft;
|
---|
510 | Point p2 = renderBounds.BottomRight;
|
---|
511 |
|
---|
512 | if (parent != null)
|
---|
513 | {
|
---|
514 | //p1 = TranslatePoint(p1, parent);
|
---|
515 | //p2 = TranslatePoint(p2, parent);
|
---|
516 | }
|
---|
517 |
|
---|
518 | Int32Rect rect = new Rect(p1, p2).ToInt32Rect();
|
---|
519 |
|
---|
520 | return ScreenshotHelper.CreateScreenshot(this, rect);
|
---|
521 | }
|
---|
522 |
|
---|
523 |
|
---|
524 | /// <summary>Saves screenshot to file.</summary>
|
---|
525 | /// <param name="filePath">File path.</param>
|
---|
526 | public void SaveScreenshot(string filePath)
|
---|
527 | {
|
---|
528 | ScreenshotHelper.SaveBitmapToFile(CreateScreenshot(), filePath);
|
---|
529 | }
|
---|
530 |
|
---|
531 | /// <summary>
|
---|
532 | /// Saves screenshot to stream.
|
---|
533 | /// </summary>
|
---|
534 | /// <param name="stream">The stream.</param>
|
---|
535 | /// <param name="fileExtension">The file type extension.</param>
|
---|
536 | public void SaveScreenshotToStream(Stream stream, string fileExtension)
|
---|
537 | {
|
---|
538 | ScreenshotHelper.SaveBitmapToStream(CreateScreenshot(), stream, fileExtension);
|
---|
539 | }
|
---|
540 |
|
---|
541 | /// <summary>Copies the screenshot to clipboard.</summary>
|
---|
542 | public void CopyScreenshotToClipboard()
|
---|
543 | {
|
---|
544 | Clipboard.Clear();
|
---|
545 | Clipboard.SetImage(CreateScreenshot());
|
---|
546 | }
|
---|
547 |
|
---|
548 | #endregion
|
---|
549 |
|
---|
550 | #region IsDefaultElement attached property
|
---|
551 |
|
---|
552 | protected void SetAllChildrenAsDefault()
|
---|
553 | {
|
---|
554 | foreach (var child in Children.OfType<DependencyObject>())
|
---|
555 | {
|
---|
556 | child.SetValue(IsDefaultElementProperty, true);
|
---|
557 | }
|
---|
558 | }
|
---|
559 |
|
---|
560 | /// <summary>Gets a value whether specified graphics object is default to this plotter or not</summary>
|
---|
561 | /// <param name="obj">Graphics object to check</param>
|
---|
562 | /// <returns>True if it is default or false otherwise</returns>
|
---|
563 | public static bool GetIsDefaultElement(DependencyObject obj)
|
---|
564 | {
|
---|
565 | return (bool)obj.GetValue(IsDefaultElementProperty);
|
---|
566 | }
|
---|
567 |
|
---|
568 | public static void SetIsDefaultElement(DependencyObject obj, bool value)
|
---|
569 | {
|
---|
570 | obj.SetValue(IsDefaultElementProperty, value);
|
---|
571 | }
|
---|
572 |
|
---|
573 | public static readonly DependencyProperty IsDefaultElementProperty = DependencyProperty.RegisterAttached(
|
---|
574 | "IsDefaultElement",
|
---|
575 | typeof(bool),
|
---|
576 | typeof(Plotter),
|
---|
577 | new UIPropertyMetadata(false));
|
---|
578 |
|
---|
579 | /// <summary>Removes all user graphs from given UIElementCollection,
|
---|
580 | /// leaving only default graphs</summary>
|
---|
581 | protected static void RemoveUserElements(IList<IPlotterElement> elements)
|
---|
582 | {
|
---|
583 | int index = 0;
|
---|
584 |
|
---|
585 | while (index < elements.Count)
|
---|
586 | {
|
---|
587 | DependencyObject d = elements[index] as DependencyObject;
|
---|
588 | if (d != null && !GetIsDefaultElement(d))
|
---|
589 | {
|
---|
590 | elements.RemoveAt(index);
|
---|
591 | }
|
---|
592 | else
|
---|
593 | {
|
---|
594 | index++;
|
---|
595 | }
|
---|
596 | }
|
---|
597 | }
|
---|
598 |
|
---|
599 | public void RemoveUserElements()
|
---|
600 | {
|
---|
601 | RemoveUserElements(Children);
|
---|
602 | }
|
---|
603 |
|
---|
604 | #endregion
|
---|
605 |
|
---|
606 | #region IsDefaultAxis
|
---|
607 |
|
---|
608 | public static bool GetIsDefaultAxis(DependencyObject obj)
|
---|
609 | {
|
---|
610 | return (bool)obj.GetValue(IsDefaultAxisProperty);
|
---|
611 | }
|
---|
612 |
|
---|
613 | public static void SetIsDefaultAxis(DependencyObject obj, bool value)
|
---|
614 | {
|
---|
615 | obj.SetValue(IsDefaultAxisProperty, value);
|
---|
616 | }
|
---|
617 |
|
---|
618 | public static readonly DependencyProperty IsDefaultAxisProperty = DependencyProperty.RegisterAttached(
|
---|
619 | "IsDefaultAxis",
|
---|
620 | typeof(bool),
|
---|
621 | typeof(Plotter),
|
---|
622 | new UIPropertyMetadata(false, OnIsDefaultAxisChanged));
|
---|
623 |
|
---|
624 | private static void OnIsDefaultAxisChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
---|
625 | {
|
---|
626 | Plotter parentPlotter = null;
|
---|
627 | IPlotterElement plotterElement = d as IPlotterElement;
|
---|
628 | if (plotterElement != null)
|
---|
629 | {
|
---|
630 | parentPlotter = plotterElement.Plotter;
|
---|
631 |
|
---|
632 | if (parentPlotter != null)
|
---|
633 | {
|
---|
634 | parentPlotter.OnIsDefaultAxisChangedCore(d, e);
|
---|
635 | }
|
---|
636 | }
|
---|
637 | }
|
---|
638 |
|
---|
639 | protected virtual void OnIsDefaultAxisChangedCore(DependencyObject d, DependencyPropertyChangedEventArgs e) { }
|
---|
640 |
|
---|
641 | #endregion
|
---|
642 |
|
---|
643 | #region Undo
|
---|
644 |
|
---|
645 | private readonly UndoProvider undoProvider = new UndoProvider();
|
---|
646 | public UndoProvider UndoProvider
|
---|
647 | {
|
---|
648 | get { return undoProvider; }
|
---|
649 | }
|
---|
650 |
|
---|
651 | #endregion
|
---|
652 |
|
---|
653 | #region Plotter attached property
|
---|
654 |
|
---|
655 | [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
|
---|
656 | public static Plotter GetPlotter(DependencyObject obj)
|
---|
657 | {
|
---|
658 | return (Plotter)obj.GetValue(PlotterProperty);
|
---|
659 | }
|
---|
660 |
|
---|
661 | public static void SetPlotter(DependencyObject obj, Plotter value)
|
---|
662 | {
|
---|
663 | obj.SetValue(PlotterProperty, value);
|
---|
664 | }
|
---|
665 |
|
---|
666 | public static readonly DependencyProperty PlotterProperty = DependencyProperty.RegisterAttached(
|
---|
667 | "Plotter",
|
---|
668 | typeof(Plotter),
|
---|
669 | typeof(Plotter),
|
---|
670 | new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.Inherits, OnPlotterChanged));
|
---|
671 |
|
---|
672 | private static void OnPlotterChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
---|
673 | {
|
---|
674 | FrameworkElement element = d as FrameworkElement;
|
---|
675 | Plotter prevPlotter = (Plotter)e.OldValue;
|
---|
676 | Plotter currPlotter = (Plotter)e.NewValue;
|
---|
677 |
|
---|
678 | // raise Plotter[*] events, where * is Attached, Detaching, Changed.
|
---|
679 | if (element != null)
|
---|
680 | {
|
---|
681 | PlotterChangedEventArgs args = new PlotterChangedEventArgs(prevPlotter, currPlotter, PlotterDetachingEvent);
|
---|
682 |
|
---|
683 | if (currPlotter == null && prevPlotter != null)
|
---|
684 | {
|
---|
685 | RaisePlotterEvent(element, args);
|
---|
686 | }
|
---|
687 | else if (currPlotter != null)
|
---|
688 | {
|
---|
689 | args.RoutedEvent = PlotterAttachedEvent;
|
---|
690 | RaisePlotterEvent(element, args);
|
---|
691 | }
|
---|
692 |
|
---|
693 | args.RoutedEvent = PlotterChangedEvent;
|
---|
694 | RaisePlotterEvent(element, args);
|
---|
695 | }
|
---|
696 | }
|
---|
697 |
|
---|
698 | private static void RaisePlotterEvent(FrameworkElement element, PlotterChangedEventArgs args)
|
---|
699 | {
|
---|
700 | element.RaiseEvent(args);
|
---|
701 | PlotterEvents.Notify(element, args);
|
---|
702 | }
|
---|
703 |
|
---|
704 | #endregion
|
---|
705 |
|
---|
706 | #region Plotter routed events
|
---|
707 |
|
---|
708 | public static readonly RoutedEvent PlotterAttachedEvent = EventManager.RegisterRoutedEvent(
|
---|
709 | "PlotterAttached",
|
---|
710 | RoutingStrategy.Direct,
|
---|
711 | typeof(PlotterChangedEventHandler),
|
---|
712 | typeof(Plotter));
|
---|
713 |
|
---|
714 | public static readonly RoutedEvent PlotterDetachingEvent = EventManager.RegisterRoutedEvent(
|
---|
715 | "PlotterDetaching",
|
---|
716 | RoutingStrategy.Direct,
|
---|
717 | typeof(PlotterChangedEventHandler),
|
---|
718 | typeof(Plotter));
|
---|
719 |
|
---|
720 | public static readonly RoutedEvent PlotterChangedEvent = EventManager.RegisterRoutedEvent(
|
---|
721 | "PlotterChanged",
|
---|
722 | RoutingStrategy.Direct,
|
---|
723 | typeof(PlotterChangedEventHandler),
|
---|
724 | typeof(Plotter));
|
---|
725 |
|
---|
726 | #endregion
|
---|
727 |
|
---|
728 | #region DefaultPanel property
|
---|
729 |
|
---|
730 | /// <summary>
|
---|
731 | /// Gets the value indicating that this is a default panel.
|
---|
732 | /// Default panels are those that are contained in plotter by default, like MainGrid or CentralCanvas.
|
---|
733 | /// </summary>
|
---|
734 | /// <param name="obj">The obj.</param>
|
---|
735 | /// <returns></returns>
|
---|
736 | public static bool GetIsDefaultPanel(DependencyObject obj)
|
---|
737 | {
|
---|
738 | return (bool)obj.GetValue(IsDefaultPanelProperty);
|
---|
739 | }
|
---|
740 |
|
---|
741 | public static void SetIsDefaultPanel(DependencyObject obj, bool value)
|
---|
742 | {
|
---|
743 | obj.SetValue(IsDefaultPanelProperty, value);
|
---|
744 | }
|
---|
745 |
|
---|
746 | public static readonly DependencyProperty IsDefaultPanelProperty = DependencyProperty.RegisterAttached(
|
---|
747 | "IsDefaultPanel",
|
---|
748 | typeof(bool),
|
---|
749 | typeof(Plotter),
|
---|
750 | new FrameworkPropertyMetadata(false));
|
---|
751 |
|
---|
752 | #endregion
|
---|
753 | }
|
---|
754 |
|
---|
755 | public delegate void PlotterChangedEventHandler(object sender, PlotterChangedEventArgs e);
|
---|
756 | }
|
---|
757 |
|
---|