UIOverride
No edit summary
(Automatically adding template at the end of the page.)
 
(26 intermediate revisions by 4 users not shown)
Line 1: Line 1:
[[File:2016-12-10 13h01 10.png|thumb|UIOverride]]
[[File:2016-12-10 13h01 10.png|thumb|UIOverride|241x241px]]
Working with MDriven viewmodels you expect the UI to be created by the MDriven logic for the given platform. This handles the standard grids, edits and picklists. But ever so often you want to be able to inject other controls and bind these to the data instead.
When working with MDriven ViewModels, you may expect the UI to be created by the MDriven logic for the given platform. This handles the standard grids, edits, and picklists. Every so often, you want to be able to inject other controls and bind these to the data instead.


You can replace the whole page/dialog/form but many times you will suffice with just injecting one or a few things into the standard UI.  
You can replace the whole page/dialog/form but many times, you will suffice by injecting one or more things into the standard UI.


To do this you check the "Content override" checkbox on the viewmodel column that is the root of your data for the override control. This will signal just as grey rectangle in the designer.
To do this, check the "Content override" checkbox on the ViewModel column that is the root of your data for the override control. This will signal just as a grey rectangle in the designer.  


Two new properties will show up in the propetry inspector: ContentOverrideType and ContentOverrideDesignTimePath. These are properties to point out a c# assembly that can render the control in design time. This is totally optional - and you do not need to do this in order to use the UIOverride on runtime. If you want to find out how to use it read more here: http://www.new.capableobjects.com/2013/06/10/custom-controls-in-viewmodel-aided-views/
Two new properties will show up in the property inspector: ContentOverrideType and ContentOverrideDesignTimePath. These are properties to point out a c# assembly that can render the control in design time. This is totally optional - and you do not need to do this in order to use the UIOverride on runtime. If you want to find out how to use it, read more here: [[Custom controls in ViewModel aided Views|Custom controls in ViewModel-aided Views]]


To get the UIOverride to work in runtime we need to do different things per target platform. All platforms support the UIOverride - but the used control must support the platform.
To get the UIOverride to work in runtime, we need to do different things per target platform. All platforms support the UIOverride - but the used control must support the platform.
{| class="wikitable"
{| class="wikitable"
!Platform
!Platform
Line 14: Line 14:
|-
|-
|WPF std MDriven Framework
|WPF std MDriven Framework
|Implement IExternalWECPOFUIComponent, [http://www.new.capableobjects.com/2013/06/10/custom-controls-in-viewmodel-aided-views/ read more here]
|Implement IExternalWECPOFUIComponent, [http://wiki.mdriven.net/index.php/Custom_controls_in_ViewModel_aided_Views read more here.]
|-
|-
|MDriven Turnkey WPF
|MDriven Turnkey WPF
|Currently you must implement the HandleContentOverride and merge your control with existing view
|Currently, you must implement the HandleContentOverride and merge your control with an existing view
|-
|-
|MDriven Turnkey AngularJS
|[[Plugins in Turnkey|MDriven Turnkey AngularJS]]
|
|
* Add AngularUIOverride tagged value on viewmodel column
* Add AngularUIOverride tagged value on the ViewModel column
* set the value to a file containing your control <yourcontrol>.cshtml
* Set the value to a file containing your control <yourcontrol>.cshtml
* Make sure the file exists on the server under EXT_OverridePages
* Make sure the file exists on the server under EXT_OverridePages (easiest done with [[AssetsTK]] )
* Make sure your control gets a name that is unique to your existing views in the app - or else the control will be picked up as a complete view override as well.
* Make sure your control gets a name that is unique to your existing views in the app - or else the control will be picked up as a complete view override as well.
|-
|-
|MDriven Turnkey MVC
|MDriven Turnkey MVC
|Paused Waiting for a use case and motivation to implement
|<Same instructions as for Angular> but the Cshtml must now be razor and your page must be set to use [[MVC]]
|}
|}


<html>
<html>
<h4> In the following video tutorial, the example of UIOverride, ContentOverride - creating and injecting your own controls in
  MDriven Turnkey generated UI - is given in a more practical way to facilitate your user experience.</h4>
<p class="video-warn">
  To make your experience smooth, we set the main tags mentioned in the video to the right bar menu of this mini-player. Choose an interesting subtitle on the list and immediately get to the exact theme navigation item place in the video. Now you can pick any topic to be instructed on without watching the whole video. </p>


<nowiki><h4> In the following video tutorial the example of UIOverride, ContentOverrid -creating and injecting your own controls in MDriven Turnkey generated UI is given in more practical way to faciliate your user experience.<h4></nowiki><p>
<em>To make your experience more comfortable, we set the main tags mentioned in the video to the right bar menu of this mini player. Choose the interesting subtitle on the list and immediately get to the exact theme timeplace in the video. Now you can pick any topic to be instructed without watching the whole video.</em>
<style type="text/css">
p {
opacity: 0.7;
text-align: justify;
width: 90%
}
</style>
</p>


<style>
<div class="video">
#video12 {
   <div class="video__wrapper">
  position: relative;
    <iframe src="https://www.youtube.com/embed/2Nfxqz8z1U0?rel=0&autoplay=0" frameborder="0" allowfullscreen></iframe>
  height: 500px;
   </div>
  width:560px;
  <div class="video__navigation">
  padding-bottom: 10px;
     <span class="navigation-item" data-video="2Nfxqz8z1U0" data-start="18" tabindex="0"> Content Override Checkbox</span>
}
     <span class="navigation-item" data-video="2Nfxqz8z1U0" data-start="45" tabindex="0"> WPF Turnkey Fat Client Look </span>
#video12 iframe {
     <span class="navigation-item" data-video="2Nfxqz8z1U0" data-start="208" tabindex="0"> Angular UI Override Control Implementation </span>
  position: absolute;
 
  min-height: auto;
     <span class="navigation-item" data-video="2Nfxqz8z1U0" data-start="270" tabindex="0"> Tagged Value </span>
  min-width: auto;
     <span class="navigation-item" data-video="2Nfxqz8z1U0" data-start="365" tabindex="0"> Data Change Reaction / Mutation Observer</span>
}
 
#video12 div {
     <span class="navigation-item" data-video="2Nfxqz8z1U0" data-start="441" tabindex="0"> Pattern Matching </span>
  position: absolute;
  </div>
  top: 0;
  left:760px;
  width: 260px;
  height: 100%;
  padding-left: 10px;
  overflow-y: auto;
}
span {
    font-size: 18;
    display:block;
  padding: 2px 10px 0.5px 10px;
    padding-bottom: 0.5;
    padding-top: 0.5;
  opacity: 0.7;
}
span:hover {
  color: #0000FF;
  cursor: pointer;
}
span:focus {
   color: blue;
</style>
<br>
<div id="video12">
<iframe width="740" height="500" src="https://www.youtube.com/embed/fu8VRHvabzk?rel=0&autoplay=0" frameborder="0" allowfullscreen></iframe>
   <div>
     <span data-video="fu8VRHvabzk" data-start="13" tabindex="0"> content override checkbox</span>
     <span data-video="fu8VRHvabzk" data-start="40" tabindex="0"> wpf turnkey fat client look </span>
     <span data-video="fu8VRHvabzk" data-start="203" tabindex="0"> angular UI override control implementation </span>
     <ul> <span data-video="fu8VRHvabzk" data-start="265" tabindex="0"> tag value  </span>  
     <span data-video="fu8VRHvabzk" data-start="360" tabindex="0"> data change reaction / mutation observer</span></ul>
     <span data-video="fu8VRHvabzk" data-start="436" tabindex="0"> pattern matching </span>
</div>
</div>
</div>


<script>
var IMG = document.querySelectorAll('#video12 span'),
    IFRAME = document.querySelector('#video12 iframe');
for (var i = 0; i < IMG.length; i++) {
  IMG[i].onclick = function() {
    IFRAME.src = 'http://www.youtube.com/embed/' + this.dataset.video + '?rel=0&autoplay=1';
    if(this.dataset.end) IFRAME.src = IFRAME.src.replace(/([\s\S]*)/g, '$1&end=' + this.dataset.end);
    if(this.dataset.start) IFRAME.src = IFRAME.src.replace(/([\s\S]*)/g, '$1&start=' + this.dataset.start);
    this.style.backgroundColor='rgba(0,0,0,.2)';
  }
}
</script>
</html>
</html>
In WPF-Turnkey Fat Client, do this:
    static WindowWecpofTest()
    {
      ViewMetaBase.ContentOverride += HandleContentOverride;
    }
    public static void HandleContentOverride(object sender, ContentOverrideEventArgs args)
    {
      if (args.ViewName == "UIOverride" && args.Owner== "UIOverride" && args.Name== "NumberOfLeaves")
      {
        args.ContinueWithOrg = false;
        ExampleUIOverrideTest myOverride = new ExampleUIOverrideTest();
        Grid.SetColumn(myOverride, args.X);
        Grid.SetRow(myOverride, args.Y);
        Grid.SetColumnSpan(myOverride, args.Colspan);
        Grid.SetRowSpan(myOverride, args.Rowspan);
        Binding b = new Binding("NumberOfLeaves");
        myOverride.SetBinding(ExampleUIOverrideTest.NumberOfLeafsProperty,b);
        (args.StreamCli.TargetForUI as Grid).Children.Add(myOverride);
      }
    }
In WPF-Turnkey Fat Client, you will likely work in Visual Studio with full access to the Turnkey client code. Your controls are then compiled and delivered along with the application client.
In MDrivenTurnkey for AngualarJS, you may or may not need access to the Visual Studio at all.
Instead, signal in the ContentOverride column’s tagged value what the Cshtml is named: [[File:ContentOverride.png|thumb|616x616px|none]]The important TaggedValue is AngularUIOverride and the file location should be <YourSite>\Views\EXT_OverridePages\AFlowerCtrlPartial.cshtml
In Angular, you have access to data from the scope variable named “data”. This is one example of how to bind:
<nowiki><h1>Control has access to data Im telling the truth! {{data.NumberOfLeaves}}</nowiki><nowiki></h1></nowiki>
In my example ctrl, I wanted to call a Javascript every time my data changed. Perhaps there are other ways to accomplish this, but I added a MutationObserver to a div that shows my data:
    var MutationObserver = window.MutationObserver ||window.WebKitMutationObserver;
    var myObserver = new MutationObserver(mutationHandler);
    var obsConfig = {
      childList: true,
      subtree: true};
    $(document).ready(function () {
      console.info("INIT mutationHandler");
      myObserver.observe(this, obsConfig);
      myObserver.observe(document.getElementById("TriggerToRedraw"), obsConfig);
    });
        function mutationHandler(mutationRecords) {
      console.info("mutationHandler:");
      var str = $("#TriggerToRedraw")[0].innerHTML;
      console.log("numleaves:" + str);
      var numleaves = Number(str);
      drawStar(100, 100, numleaves, 30, 15);
          };
Whenever the MutationObserver finds a “Mutation” on my div, it will call the script with the new content of that div.
[[Category:UI]]
[[Category:Content Override]]
[[Category:MDriven Turnkey]]
[[Category:MDriven Designer]]
[[Category:MDriven Framework]]
[[Category:MVC]]
[[Category:View Model]]
{{Edited|July|12|2024}}

Latest revision as of 15:49, 10 February 2024

UIOverride

When working with MDriven ViewModels, you may expect the UI to be created by the MDriven logic for the given platform. This handles the standard grids, edits, and picklists. Every so often, you want to be able to inject other controls and bind these to the data instead.

You can replace the whole page/dialog/form but many times, you will suffice by injecting one or more things into the standard UI.

To do this, check the "Content override" checkbox on the ViewModel column that is the root of your data for the override control. This will signal just as a grey rectangle in the designer.

Two new properties will show up in the property inspector: ContentOverrideType and ContentOverrideDesignTimePath. These are properties to point out a c# assembly that can render the control in design time. This is totally optional - and you do not need to do this in order to use the UIOverride on runtime. If you want to find out how to use it, read more here: Custom controls in ViewModel-aided Views

To get the UIOverride to work in runtime, we need to do different things per target platform. All platforms support the UIOverride - but the used control must support the platform.

Platform What you need to do
WPF std MDriven Framework Implement IExternalWECPOFUIComponent, read more here.
MDriven Turnkey WPF Currently, you must implement the HandleContentOverride and merge your control with an existing view
MDriven Turnkey AngularJS
  • Add AngularUIOverride tagged value on the ViewModel column
  • Set the value to a file containing your control <yourcontrol>.cshtml
  • Make sure the file exists on the server under EXT_OverridePages (easiest done with AssetsTK )
  • Make sure your control gets a name that is unique to your existing views in the app - or else the control will be picked up as a complete view override as well.
MDriven Turnkey MVC <Same instructions as for Angular> but the Cshtml must now be razor and your page must be set to use MVC

In the following video tutorial, the example of UIOverride, ContentOverride - creating and injecting your own controls in MDriven Turnkey generated UI - is given in a more practical way to facilitate your user experience.

To make your experience smooth, we set the main tags mentioned in the video to the right bar menu of this mini-player. Choose an interesting subtitle on the list and immediately get to the exact theme navigation item place in the video. Now you can pick any topic to be instructed on without watching the whole video.

Content Override Checkbox WPF Turnkey Fat Client Look Angular UI Override Control Implementation Tagged Value Data Change Reaction / Mutation Observer Pattern Matching

In WPF-Turnkey Fat Client, do this:

    static WindowWecpofTest()
    {
      ViewMetaBase.ContentOverride += HandleContentOverride;
    }
    public static void HandleContentOverride(object sender, ContentOverrideEventArgs args)
    {
      if (args.ViewName == "UIOverride" && args.Owner== "UIOverride" && args.Name== "NumberOfLeaves")
      {
        args.ContinueWithOrg = false;
        ExampleUIOverrideTest myOverride = new ExampleUIOverrideTest();
        Grid.SetColumn(myOverride, args.X);
        Grid.SetRow(myOverride, args.Y);
        Grid.SetColumnSpan(myOverride, args.Colspan);
        Grid.SetRowSpan(myOverride, args.Rowspan);
        Binding b = new Binding("NumberOfLeaves");
        myOverride.SetBinding(ExampleUIOverrideTest.NumberOfLeafsProperty,b);
        (args.StreamCli.TargetForUI as Grid).Children.Add(myOverride);
      }
    }

In WPF-Turnkey Fat Client, you will likely work in Visual Studio with full access to the Turnkey client code. Your controls are then compiled and delivered along with the application client.

In MDrivenTurnkey for AngualarJS, you may or may not need access to the Visual Studio at all.

Instead, signal in the ContentOverride column’s tagged value what the Cshtml is named:

ContentOverride.png

The important TaggedValue is AngularUIOverride and the file location should be <YourSite>\Views\EXT_OverridePages\AFlowerCtrlPartial.cshtml

In Angular, you have access to data from the scope variable named “data”. This is one example of how to bind:

<h1>Control has access to data Im telling the truth! {{data.NumberOfLeaves}}</h1>

In my example ctrl, I wanted to call a Javascript every time my data changed. Perhaps there are other ways to accomplish this, but I added a MutationObserver to a div that shows my data:

    var MutationObserver = window.MutationObserver ||window.WebKitMutationObserver;
    var myObserver = new MutationObserver(mutationHandler);
    var obsConfig = {
      childList: true,
      subtree: true};
    $(document).ready(function () {
      console.info("INIT mutationHandler");
      myObserver.observe(this, obsConfig);
      myObserver.observe(document.getElementById("TriggerToRedraw"), obsConfig);
    });
       function mutationHandler(mutationRecords) {
      console.info("mutationHandler:");
      var str = $("#TriggerToRedraw")[0].innerHTML;
      console.log("numleaves:" + str);
      var numleaves = Number(str);
      drawStar(100, 100, numleaves, 30, 15);

         };

Whenever the MutationObserver finds a “Mutation” on my div, it will call the script with the new content of that div.

This page was edited more than 11 months ago on 02/10/2024. What links here