Part 5 Searching in databases
No edit summary
No edit summary
Line 52: Line 52:


<div id="video12">
<div id="video12">
  <iframe width="740" height="500" src="https://www.youtube.com/embed/BhKRj3-LoBI?rel=0&autoplay=1" frameborder="0" allowfullscreen></iframe>
  <iframe width="740" height="500" src="https://www.youtube.com/embed/BhKRj3-LoBI?rel=0&autoplay=0" frameborder="0" allowfullscreen></iframe>
   <div>
   <div>
   <strong>View Model Editor:</strong>
   <strong>View Model Editor:</strong>
Line 64: Line 64:
     <span data-video="BhKRj3-LoBI" data-start="2548" tabindex="0"> Validation Rules </span>
     <span data-video="BhKRj3-LoBI" data-start="2548" tabindex="0"> Validation Rules </span>
     <span data-video="BhKRj3-LoBI" data-start="2876" tabindex="0"> Saving despite error warnings </span>
     <span data-video="BhKRj3-LoBI" data-start="2876" tabindex="0"> Saving despite error warnings </span>
 
 
   </div>
   </div>
</div>
</div>
Line 84: Line 84:
'''Raw subtitles text'''
'''Raw subtitles text'''


Hi and welcome back to MDriven designer overview and this will be session five, so we ended off the last session with the creating a modal dialog that were allowed us to pick tenants to our house so we could pick persons that would create lease contract and when we used this dialogue "all persons" to to pick the tenants from and as you can imagine this strategy simply will not hold, if we have a million persons and this is good, when we have a hundred maybe, but we can't really trust on all instances to come to the rescue for every scenario, we really desperately need some way to vSeek the database without instanciating the objects in memory in their client during the prototyper, so how do we go about that in the view model pattern that we have we don't like this, we right click and among the nesting choices we have add add search expression and such expressions are expression step are supposed to be executed in persistent storage, so that will be a database if your back and is connected to it, to one or it will be the xml file in our case, where we prototype against the xml file and if we are in turn key will be the persistence mapper that in turn will question the backing database alright, so let's just go ahead and add a search expression or maybe better yet, we will do a new view model to keep the searching and we will column this "person search" and we type it to be "person" right click and in the nested view models add search expression and what happened here, was that three columns were added to the root and one of the columns was a variable called the vSeek result, this variable was created and added to the viewmodel well it wasn't because it's implicit, but it should be in the newer versions, it will be and so variables are just as they sound something that we can define in the view model and is only available in the viewmodel scope, so this one should be of collection person this is where our result is going to end up from a search and that one we connect to a search result grid, so we have another variable vSeekParam that is defined as a string, so we have a column in the UI that lets the user enter a criteria and we have his search button and the search button as the ocl expression "selfvm.search", "selfvm" is a special instance that is available in view models only from OCL, let's see what it contains, "selfvm." and from here we can do things as refresh, save, search, and a few other things, like PSexpression_refresh, we can come back to these in later sessions, but for now we just going to use the vm search, what the search operator does is it looks for search expressions in the view model and this is a search expression, it's just another view model nesting, but what separates this from the other nestings is that it is not connected to anything it's judt sits here in the middle, doesn't need to be in the middle because I said so I can right click this and move this one up, so as such expression is typed and it has criterias right here it's just columns you can have many criteria per search expression the criterias are merged together with the ocl operator intersection, so what if these criteria is true and we had another or this criteria returned five persons 12345 and we had another one that return two persons 45, the intersection will take what's equal between the two domains so 4 and 5, but the search expression the criteria is just another ocl, but what's different about this is that is the persistent storage subset of SQL Ocl-PS it sets up here in the edge when we entry and then we don't have access to every single operator, we can't concatenate strings etc. in this language its main purpose is to be easily transformed to SQL and executed as an SQL query to the database and the database will return a number of identities, that we will in turn interpret as persons and load personal objects, so what we do is select and we have a loop variable called A then the mechanism to create the search expression and all this for us it just took the first string attribute of the class that we had set person and that was first name and set, well it's not unlikely that is what you as a developer want to limit search of, this attribute as SqlLike(vSeekParam) and ends with a +% just like in Sql operator like, so if you want to find both and start, if you want to find values that equals in the middle that you would go something like this, so this will be a resulting string that's add into the SqlLike, that of course will be translated into a like operator in sequel, but as we are running against an xml file, it will do something similar just execute operator on that so is this really what we want well, we want to also check if it's the last name maybe, so I use "or" and I use "LastName", like that what should I show for a result I can show the things I have first name and last name so let's do one thing more in this view and if you don't find what you expect you might want to have somewhere where you can create a new person, so let's totally action definitions for this dialogue and let's do a viewmodel action and say that we should show it on the root and say new person and the execute expression on this one should be PersonCreate, having done that we can see, if we have, we can save this push the play button, start system, Run Model and continue, we never created an action to actually show the seeker view model, so person search I right click on this and say add a global action for show okay then it's shown here, but we probably want to sort into as menu group a double click it find it here set menu group-seekers maybe, so this is might be where I want to put all my search dialogues and who knows having done that and I head back to this one model refresh, seekers, show per person, search and I can search, but I only want those that has an "a" in them that's the only one, okay but what if someone's missing I can create a new person and call that Andersson Anders, now if I search for a I should find Anderson, but since my xml file is, I guess it is case sensitive it is case sensitive, so if we don't like that we should have used operator called .sqlLikeCaseInSensitive, where like so now we have the searcher let's go back to the house view where we added the pic tenants action so the pick tenant action and let's do a copy of this, where did he go let's do a copy of this, just going to do a copy and paste, and "pick tenants by search" and that one should bring up the person search instead and if it does it cannot call "check this" because that variable doesn't exist in that UI like instead it calls "SearchResultGrid" if that is empty ok and this expression won't go either because the same problem, like that and it adds the tenants if I have selected any so if we head back here we see that we have another action pick tenants by search of course in the end, I only want one of these and our most likely it's the "BySearch" that will survive and then maybe I will change its name, model refresh, houses and pick one here, this doesn't have any tenants, I'm going to pick tenants by search, I'm going to search on find this one and I add that one, so I find it rather disturbing that I don't see the name here this is what I talked about before, in session 2 or something first, when we do the UI and watch the information together that it's obvious that, of course, we need to see the name here as a new requirement, but it's obvious now that we see the result and the not a problem this is the grid, which is this one, which points to this one and let's add the columns and add a generic column and in that generic column sets an expression that follow the role to tenant and the first name that's the ocl editor acting up when you don't have a separator first name and you might have noticed that I quite a lot of times had a need to concatenate first and the last name because that's simply how we refer to persons, so this sort of this pattern comes back again and again and we're going to solve this in just a minute, just going to save the fact that we did a new column and we call that "tenant name" and we might want to give that a little more space like three columns and we might want to drag this one to the end here and over there, and this one and we added in summary session it's just we want so if I save and refresh and look at all the houses, was it this one and there's Andersson, but I don't want it there I want it first and let's show the gridlines, like that, refresh like that and that's pretty obvious and how come this isn't even showing, well it's too tight to show with this style, so how would you solve that well as you can see we are cramming everything into one grid cell here so we're gonna insert column and give this one more space for the presentation let's give it 2 and just to make things look straight you're going to get the rest of them 2 as well, head back, refresh, views, all houses, when I click it would still crap but you get the general idea I will just insert yet another column let's judt do that one for and refresh, like that so this is important how quickly we can go from feedback with our eyes that this isn't good back to design how should be instead back to verification, okay it's fine it really saves everyboy's time ok enough about that, and I said that this pattern is repeated again and again yes it is the first name and last name it's a really common scenario that we want to refer to persons by their full name, so what we probably would want to do is add an attribute, that is called full name, but that wouldn't be very normalized to store the first name and the last name and full name, so let's make that one right, that's another attribute mode, it's not persistent as it is not stored in the database, it's derived as it's calculated from other values that we do have, so I can paste in that one and now that was for the tenants and I don't want that because now we are in the scope of a person, so first name and last name this makes up the full name and now we have done this and we can make use of it in our view models or wherever we access model information, so instead of doing this we can just as well do the full name, like that and now it's grey I was expecting it to be pink, because it is derived I wonder why that was it's derived don't have an answer, but going to save what do you say? self is not... that was just wrong, save and head back and model refresh, and there's really no difference, but like here now that we have it let's use it, so tenance last name and tenance full name is full name we got refresh if I were to change the name the right attributes were like this there are always subscribed, so let's head back to this view and add it as well just to prove a point all persons add column full name,  give it some space when it's in grid too maybe we refresh, views, show all persons as soon as I change one of their values that build up the derivation, the derivation will kick in and invalidate as the UI looks at this value it will force it to refresh, getting the parts of the value and re-evaluate it, so when I write Hans it will be updated and if I were to change anything it will be updated and that might be Dick Smith, right, so derivation is also available for associations so what we could do, if we want to know where the person lives is of course to follow to see the rented apart and look up the street for that house but if that is a common operation that we want to do we could also add a association straight from person to streets and since I can rent many apartments, I potentially have many addresses, so if I want to derive this from available information, I was saying that the association is derived and I will take this end, will have no value so I will say that there it is non navigable, is navigable false, because if I would want to go from street to persons that would be another derivations that would have another expression, but this one should have a derivation expression, that it starts in person and it goes to here is the one that is derived it goes to owned house, no, to lease contract, and from lease contract rented apartments, and from rented apartments - the street I really must fix that, the street so this will be a collection of streets and if we were to show this in our UI, I guess we could add column and add generic column, and I follow the relation to streets and that has a name, but since this is a collection of strings, we need some way to display it in a single cell that we will do here in the grid and done an operator that's real handy for these cases as CommaList so it will take all the strings in the collection and append them as they with the comma in between, so lives at, so heading back to refresh, views or persons, so here we see the address is swelling ok, so good we have covered in a lot of ground we have mentioned derivations for attributes like the full night servations for associations like the street even if it was kind of silly, because we would probably would like the street number to get the precise address and that's not possible this way we have covered the idea of being able to vSeek in a lost database and then we also mentioned variables that are available in view models, so I'm going to show you another thing and what if we are in a situation where we actually need some test data, we need a lot of persons to verify that our idea to search in larger amount is valid and of course we could run the prototype print and press, we added the new person action here and we could press that a thousand times to enter the first and last name but is there a better way, you ask and since I talking about it you have probably guessed that there is and I go into the the Run button again because I need a context that we are in the XML persistence, we're going to start the system and we're going to go into the legacy debugger and in here we have a tab over here "import tab separated", so tab separated data import via named view model, so what it says he's given tab-separated data and the view model name assumes row 1 in data to be viewmodel column and creates instances of view model class so basically, if we have some data that we paste in here, we should be able to create objects of certain class, but it also says, that we need a view model, so let's create one, as you see the view models are used over and over for different purposes ImportPersons and type Person and let's assume the first name is the first value and the last name is the second value so that's about it and given that I should be able to say that this is the view model I want to use or maybe I need to reread this, I do start, lazy debugger, input tap separated and view model and get header from view model,and first-name lastname alright where can I get that, I found this site as google for a list of names and find anything on google list of random names let's do 50 and male and female and well no alliteration when you start the same ok now I want to grab these on listing text area, that's great, I wonder if this is tab separate it probably isn't and going to start notepad, alright so is that a tab no, I copy tab and I want to do is search and replace here, look for space and paste in my tab and replace it all and copy those and head back to my importer and paste those, and go, so that means I got a lot of dirty objects here that I will save like that and head back to my this one is going to restart, person search and I have some test data to work on oh let's see everyone that starts either their first loss name with "t" or has a "t" really because that was what we said in the criteria ok, so that means that when we do houses and we want to assign tenance by search and we do everyone with the "T" because it is case sensitive as we found out and select them and press ok then we have those as tenants alright moving ON as we see the tenants here, but they haven't got an apartment number...they haven't got a rent figure, but they really should this is important information it's so important, that we might want to add constraint on our class lease contract to make sure that the user of the system is made aware of the importance of these values so I select the class and I select constraints and I get a little dialogue where I can add constraints this constraint I give a name "must have apartment number" that's the name of my constraint so that I can refer to it and and I want to have a description as well and I'm going to make easy to myself is to do something like this, that's right the expression is more or less self.apartmentnumber and well it should be not null, right shouldn't be null and then now we got that symbol that says that this has one constraint, so when we look at our diagrams we will see that this is constrained in some way ok, so if we were to head back to our prototyper and show the house we see, must have apartment number must have apartment number, but there's really no information on who is missing the apartment number and let's head back to the constraint and in the description we can actually access attributes, so and we can go like this as string within the larger than and smaller than, greater than and tags to say that we want to describe the default presentation here, can you please inject that and what is the default presentation, default string representation on lease contract is none okay, but we might follow the link to the person and use the full name again we made use of the derived full name like that and save, and head back, refresh, view all houses, so Hans Karlsen must have an apartment number and Dick Smith must have apartment number and as I give our Hans Karlsen apartment number one that is removed and as I give  Dick's one 2 that one removed as well, so that's a good way to feedback information to the user about what needs to be filled in but it might not be enough for us, we might need to stress these even harder depending on use case so there's more ways to signal the user that something needs to be done if we check the house view here and we had the variables that we don't use in this view we use them in the seeker, but we also have the the constraints, so these are the constraints that we found in the classes that are used here that is just one and depending on use case we might not want to show it because who knows the view might not even contain their apartment number, so no need to stress the user about entering it if it isn't really possible, so that we could just opt it out, but we want to have it opt-in because it's good for us, but what we could do is something more precise as a validation rule let's right-click up in the top and add a validation rule, so we got rule one and then the rule one is that let's say it's the fact that the selected, what what I want to do is to highlight something here apartment number to make the user see this so what is this, this is here the lease contract, so what I want to do in my validation rule is to find a variable the current the lease contract and check that one that particular one that sunscreen and or selected at the moment, and check that for apartment number and verify that isn't null, and then I can write something please supply number like that, but we still haven't been able to say that this rule is supposed to show up for this field that I can select the field it's over here and I'm supposed to find it and validation rules, my menus really turn out the wrong end all the time, validation rules, rule one, please supply number ok, so that would be one way to further gearsteer our user to see something, but how does it look refresh, view, houses it doesn't show anything, but if I null that out first I get this one and I also get the red information here actually depending on what style you use it really renders differently, so that you can control this and we have examples, where you just have an "!" that was when you point that it states what's wrong with it so this is another important thing I should mention that,when you have errors and there's some something and updated in  the information and you try to save then it will ask you, there are errors in the information, do you really want to save anyway? "Hans Karlsen must have apartment number" and the system won't stop you from saving and this is an important principle I think, that you shouldn't stop the user from entering information, even if it isn't correct then it's better to to monitor all the information that is in incorrect, so that they can come back to it then and update it don't stop users from entering data and make a UI very clear, what you expect but you should always always honor the users' request to save and maybe you should work with the quality attributes like states that you can't go into approved, before you have apartment number or the rent in dollars, but you shouldn't stop them from saving ok, that's just my very clear view on this ok, so I'm going to save it anyway, but run nice wouldn't you say just having the area information this way, so whenever I enter something the error goes away and of course I don't get the warning, when I save, but that was not a very good side for this example, so switch back, so let's add another constraint and we make that not to error constraint, but rather look at the other types of constraints, so we have a constraint the "rent" and the description and we do as we before "<as string> lives free" and we don't do an error just the information expression would be self.rent->not null because when the expression is true then the constraint is fulfilled and we don't shoe anything, but as soon as the constraint is not true then it's broken and then we show warning signs etc. in this case information refresh and now it's a little white thing is Hans Karlsen lives free shouldn't be the case that let them say alright I wonder if we will have time to cramming reports in this, so that we could do rents and rents slips for these guys I think maybe we'll come back to that in the next session so we will round this off and thank you very much for following along with these tutorials and I'll be back
Hi and welcome back to MDriven designer overview
 
and this will be session five, so we
 
ended off the last session with the
 
creating a modal dialog that were
 
allowed us to pick tenants to our house
 
so we could pick persons that would
 
create lease contract and when we used
 
this dialogue "all persons" to to pick
 
the tenants from and as you can
 
imagine this strategy simply will not
 
hold, if we have a million persons and
 
this is good, when we have a hundred
 
maybe, but we can't really trust on all
 
instances to come to the rescue for
 
every scenario, we really desperately
 
need some way to vSeek the database
 
without instanciating the objects in
 
memory in their client during the
 
prototyper, so how do we go about that in
 
the view model pattern that we have
 
we don't like this, we right click and among the nesting
 
choices we have add add search expression
 
and such expressions are expression step
 
are supposed to be executed in
 
persistent storage, so that will be a
 
database if your back and is connected
 
to it, to one or it will be the xml file
 
in our case, where we prototype against
 
the xml file and if we are in turn
 
key will be the persistence mapper that
 
in turn will question the backing database
 
alright, so let's just go ahead and add
 
a search expression or maybe better
 
yet, we will do a new view model to keep
 
the searching and we will column this
 
"person search" and we type it to be "person"
 
right click and in the nested view models
 
add search expression and what happened
 
here, was that three columns were
 
added to the root and one of the
 
columns was a variable called the
 
vSeek result, this variable was created
 
and added to the viewmodel well it
 
wasn't because it's implicit, but it
 
should be in the newer versions, it will
 
be and so variables are just as they sound
 
something that we can define in
 
the view model and is only available
 
in the viewmodel scope, so this one
 
should be of collection person this is
 
where our result is going to end up from
 
a search and that one we connect to a
 
search result grid, so we have another
 
variable vSeekParam that is
 
defined as a string, so we have a column
 
in the UI that lets the user enter
 
a criteria and we have his search button
 
and the search button as the ocl
 
expression "selfvm.search", "selfvm" is
 
a special instance that is available in
 
view models only from OCL, let's see
 
what it contains, "selfvm." and from
 
here we can do things as
 
refresh, save, search, and a few other
 
things, like PSexpression_refresh, we can
 
come back to these in later sessions, but
 
for now we just going to use the vm
 
search, what the search operator does is
 
it looks for search expressions in the
 
view model and this is a search
 
expression, it's just another view model
 
nesting, but what separates this from the
 
other nestings is that it is not
 
connected to anything it's judt sits here
 
in the middle, doesn't need to be in the
 
middle because I said so I can right
 
click this and move this one up, so as
 
such expression is typed and it has
 
criterias right here it's just columns you
 
can have many criteria per search expression
 
the criterias are merged
 
together with the ocl operator
 
intersection, so what if these criteria
 
is true and we had another or this
 
criteria returned five persons 12345 and
 
we had another one that return two
 
persons 45, the intersection will take
 
what's equal between the two domains
 
so 4 and 5, but the search expression
 
the criteria is just another ocl, but
 
what's different about this is that is
 
the persistent storage subset of SQL Ocl-PS
 
it sets up here in the edge
 
when we entry and then we don't have access to every single
 
operator, we can't concatenate
 
strings etc. in this language
 
its main purpose is to be easily
 
transformed to SQL and executed as an
 
SQL query to the database and the
 
database will return a number of
 
identities, that we will in turn
 
interpret as persons and load personal
 
objects, so what we do is select and we
 
have a loop variable called A then
 
the mechanism to create the search
 
expression and all this for us it just
 
took the first string attribute of the
 
class that we had set person and that
 
was first name and set, well it's not
 
unlikely that is what you as a
 
developer want to limit search of,
 
this attribute as SqlLike(vSeekParam) and ends with a +% just like in
 
Sql operator like, so if you want to find
 
both and start, if you want to find
 
values that equals in the middle
 
that you would go something like this,
 
so this will be a resulting string that's
 
add into the SqlLike, that of course
 
will be translated into a like operator
 
in sequel, but as we are running against
 
an xml file, it will do something similar
 
just execute operator on that
 
so is this really what we want
 
well, we want to also check if it's the
 
last name maybe, so I use "or" and I use "LastName", like that
 
what should I show for a result I can show the things I have
 
first name and last name
 
so let's do one thing more in this view
 
and if you don't find what you expect
 
you might want to have somewhere where
 
you can create a new person, so let's
 
totally action definitions for this
 
dialogue and let's do a viewmodel action
 
and say that we should show it on the
 
root and say new person and the
 
execute expression on this one
 
should be PersonCreate, having done that
 
we can see, if we have, we can save this
 
push the play button, start system, Run Model
 
and continue, we never created an
 
action to actually show the seeker
 
view model, so person search
 
I right click on this and say add a
 
global action for show
 
okay then it's shown here, but we
 
probably want to sort into as menu group
 
a double click it
 
find it here set menu group-seekers
 
maybe, so this is might be where I want
 
to put all my search dialogues and who
 
knows
 
having done that and I head back to this one
 
model refresh, seekers, show per person,
 
search and I can search, but I only want those
 
that has an "a" in them
 
that's the only one, okay but
 
what if someone's missing
 
I can create a new person and call that
 
Andersson Anders, now if I search for a
 
I should find Anderson, but since
 
my xml file is, I guess it is case sensitive
 
it is case sensitive, so if we don't
 
like that we should have used operator
 
called .sqlLikeCaseInSensitive, where like
 
so now we have the searcher
 
let's go back to the house view
 
where we added the pic tenants action
 
so the pick tenant action and let's do
 
a copy of this, where did he go
 
let's do a copy of this, just going to do
 
a copy and paste, and "pick tenants by search"
 
and that one should bring up the
 
person search instead and if it does
 
it cannot call "check this" because that
 
variable doesn't exist in that UI like
 
instead it calls "SearchResultGrid" if that is empty
 
ok and this expression won't go either
 
because the same problem, like that
 
and it adds the tenants if I have selected any
 
so if we head back here we see that we have
 
another action pick tenants by search
 
of course in the end, I only want one of these
 
and our most likely it's the "BySearch"
 
that will survive and then maybe
 
I will change its name, model refresh,
 
houses and pick one here,
 
this doesn't have any tenants, I'm going to pick
 
tenants by search, I'm going to search on
 
find this one and I add that one, so I find it
 
rather disturbing that I don't see the name here
 
this is what I talked about
 
before, in session 2 or something
 
first, when we do the UI
 
and watch the information together
 
that it's obvious that, of course, we need to see
 
the name here as a new requirement, but
 
it's obvious now that we see the result and that's not a problem
 
this is the grid, which is this one, which
 
points to this one and let's add the columns
 
and add a generic column and in
 
that generic column sets an expression
 
that follow the role to tenant and the
 
first name that's the ocl editor acting up
 
when you don't have a separator
 
first name and you might have noticed that
 
I quite a lot of times had a need
 
to concatenate first and the last name
 
because that's simply how we
 
refer to persons, so this
 
sort of this pattern comes back
 
again and again and we're going to solve
 
this in just a minute, just going to save
 
the fact that we did a new column
 
and we call that "tenant name" and
 
we might want to give that a little more
 
space like three columns and we might
 
want to drag this one to the end
 
here and over there, and this one and we
 
added in summary session it's just we want
 
so if I save and refresh and look at all
 
the houses, was it this one and there's
 
Andersson, but I don't want it there
 
I want it first and let's show the
 
gridlines, like that, refresh like
 
that and that's pretty obvious and
 
how come this isn't even showing, well
 
it's too tight to show with this
 
style, so how would you solve that
 
well as you can see we are cramming
 
everything into one grid cell here
 
so we're gonna insert column and
 
give this one more space for the presentation
 
let's give it 2 and just to make things
 
look straight you're going to get the
 
rest of them 2 as well, head back,
 
refresh, views, all houses, when I click it
 
would still crap but you get the
 
general idea I will just insert yet another column
 
let's judt do that one for and refresh, like that
 
so this is important how quickly
 
we can go from feedback with our eyes
 
that this isn't good back to design how should be instead
 
back to verification, okay it's fine it
 
really saves everyboy's time
 
ok enough about that, and I said that
 
this pattern is repeated again and again
 
yes it is the first name and last name
 
it's a really common scenario that we
 
want to refer to persons by their full
 
name, so what we probably would want
 
to do is add an attribute, that is called
 
full name, but that wouldn't be very
 
normalized to store the first name and
 
the last name and full name, so let's
 
make that one right, that's another
 
attribute mode, it's not persistent as it
 
is not stored in the database, it's
 
derived as it's calculated from other
 
values that we do have, so I can paste in
 
that one and now that was for the
 
tenants and I don't want that
 
because now we are in the scope of a person,
 
so first name and last name
 
this makes up the full name and now we
 
have done this and we can make use of it
 
in our view models or wherever we access
 
model information, so instead of doing
 
this we can just as well do the full
 
name, like that and now it's grey I was
 
expecting it to be pink, because it is
 
derived I wonder why that was it's derived
 
don't have an answer, but going to save
 
what do you say? self is not...
 
that was just wrong, save and head back and
 
model refresh, and there's really
 
no difference, but like here now that we have it
 
let's use it, so tenance last name
 
and tenance full name is full name we got refresh
 
if I were to change the name
 
the right attributes were like this
 
there are always subscribed, so let's head
 
back to this view and add it as well
 
just to prove a point all persons add column
 
full name,  give it some space when it's in grid too
 
maybe we refresh, views, show all persons
 
as soon as I change one of their values
 
that build up the derivation, the
 
derivation will kick in and invalidate
 
as the UI looks at this value
 
it will force it to refresh, getting the
 
parts of the value and re-evaluate it,
 
so when I write Hans it will be updated
 
and if I were to change anything
 
it will be updated and that might be
 
Dick Smith, right, so derivation is also
 
available for associations
 
so what we could do, if we want to know
 
where the person lives is of course to
 
follow to see the rented apart and
 
look up the street for that house
 
but if that is a common operation
 
that we want to do we could
 
also add a association straight from
 
person to streets and since
 
I can rent many apartments, I potentially
 
have many addresses, so if I want to
 
derive this from available information,
 
I was saying that the association is derived
 
and I will take this end, will
 
have no value so I will say that there
 
it is non navigable, is navigable
 
false, because if I would want to go from
 
street to persons that would be another
 
derivations that would have another
 
expression, but this one should have
 
a derivation expression, that it starts
 
in person and it goes to
 
here is the one that is derived it
 
goes to owned house,
 
no, to lease contract, and from lease contract
 
rented apartments, and from rented
 
apartments - the street
 
I really must fix that, the street
 
so this will be a collection of
 
streets and if we were to show this in
 
our UI, I guess we could add column and
 
add generic column, and I follow the
 
relation to streets and that has a name,
 
but since this is a collection of
 
strings, we need some way to display it
 
in a single cell that we will do here in
 
the grid and done an operator that's
 
real handy for these cases as CommaList
 
so it will take all the strings in
 
the collection and append them as they
 
with the comma in between, so lives at, so heading
 
back to refresh, views or persons, so
 
here we see the address is swelling
 
ok, so good we have covered in a lot of
 
ground we have mentioned derivations
 
for attributes like the full night
 
servations for associations like the
 
street even if it was kind of silly,
 
because we would probably would like
 
the street number to get the precise
 
address and that's not possible this way
 
we have covered the idea of being able
 
to vSeek in a lost database and then
 
we also mentioned variables that are
 
available in view models, so I'm going to
 
show you another thing and what if we
 
are in a situation where we actually
 
need some test data, we need a lot of
 
persons to verify that our idea to search in
 
larger amount is valid and of course
 
we could run the prototype print
 
and press, we added the new person action
 
here and we could press that a thousand
 
times to enter the first and last name
 
but is there a better way, you ask
 
and since I talking about it you have
 
probably guessed that there is and I go
 
into the the Run button again because
 
I need a context that we are in the
 
XML persistence, we're going to start
 
the system and we're going to go into
 
the legacy debugger and in here we have
 
a tab over here
 
"import tab separated", so tab separated
 
data import via named view model, so what it
 
says he's given tab-separated data and
 
the view model name assumes row 1 in data
 
to be viewmodel column and creates
 
instances of view model class
 
so basically, if we have some data that
 
we paste in here, we should be able to
 
create objects of certain class, but it
 
also says, that we need a view model,
 
so let's create one, as you see the view
 
models are used over and over for different purposes
 
ImportPersons and type Person and let's assume
 
the first name is the first value and
 
the last name is the second value so
 
that's about it and given that I should be able to say
 
that this is the view model I want to
 
use or maybe I need to reread this, I do
 
start, lazy debugger,
 
input tap separated and view model and
 
get header from view model,and first-name
 
lastname alright, where can I get that, I found this site
 
as google for a list of names and find
 
anything on google list of random names
 
let's do 50 and male and female and well
 
no alliteration when you start the same ok
 
now I want to grab these on listing text
 
area, that's great, I wonder if this is
 
tab separate it probably isn't and going to
 
start notepad, alright so is that a tab
 
no, I copy tab and I want to
 
do is search and replace here, look
 
for space and paste in my tab and
 
replace it all and copy those and head
 
back to my importer and paste those, and
 
go, so that means I got a lot of dirty
 
objects here that I will save like that and
 
head back to my this one is going to
 
restart, person search and I have some
 
test data to work on
 
oh let's see everyone that starts either
 
their first loss name with "t" or has a "t"
 
really because that was what we said in the criteria
 
ok, so that means that when we do houses
 
and we want to assign tenance by search
 
and we do everyone with the "T"
 
because it is case sensitive as we
 
found out and select them and press ok
 
then we have those as tenants
 
alright moving ON as we see the
 
tenants here, but they haven't got
 
an apartment number...they
 
haven't got a rent figure, but they
 
really should this is important information
 
it's so important, that we might want to
 
add constraint on our class lease
 
contract to make sure that
 
the user of the system is made aware of
 
the importance of these values
 
so I select the class and I select
 
constraints and I get a little dialogue
 
where I can add constraints this
 
constraint I give a name "must have
 
apartment number" that's the name of my
 
constraint so that I can refer to it and
 
and I want to have a description as well
 
and I'm going to make easy to myself is
 
to do something like this, that's right
 
the expression is more or
 
less self.apartmentnumber and well it
 
should be not null, right
 
shouldn't be null and then now we got
 
that symbol that says that this has one
 
constraint, so when we look at our
 
diagrams we will see that this is
 
constrained in some way
 
ok, so if we were to head back to our
 
prototyper and show the house
 
we see, must have apartment number
 
must have apartment number, but there's
 
really no information on who is missing
 
the apartment number and let's head back
 
to the constraint and in the description we can actually
 
access attributes, so and we can
 
go like this as string within the
 
larger than and smaller than, greater than and
 
tags to say that we want to describe
 
the default presentation here, can you
 
please inject that and what is the
 
default presentation, default string
 
representation on lease contract is none
 
okay, but we might follow the link to the
 
person and use the full name again we
 
made use of the derived full name
 
like that and save, and head back, refresh, view
 
all houses, so Hans Karlsen must have an
 
apartment number and Dick Smith must
 
have apartment number and as I give our
 
Hans Karlsen apartment number one
 
that is removed and as I give  Dick's one 2
 
that one removed as well, so that's a
 
good way to feedback information to the
 
user about what needs to be filled in
 
but it might not be enough for us, we
 
might need to stress these even harder depending on use case
 
so there's more ways to signal the
 
user that something needs to be done
 
if we check the house view here and we had
 
the variables that we don't use in this
 
view we use them in the seeker, but we
 
also have the the constraints, so these
 
are the constraints that we found in the
 
classes that are used here that is just
 
one and depending on use case we might
 
not want to show it because who knows
 
the view might not even contain
 
their apartment number, so no need to
 
stress the user about entering it if it
 
isn't really possible, so that we could
 
just opt it out, but we want to have it
 
opt-in because it's good for us, but what
 
we could do is something more precise as a validation rule
 
let's right-click up in the top and add
 
a validation rule, so we got rule one and
 
then the rule one is that
 
let's say it's the fact that the selected, what
 
what I want to do is to highlight something here
 
apartment number to make the user see this
 
so what is this, this is here
 
the lease contract, so what I want to do in
 
my validation rule is to find a
 
variable the current
 
the lease contract and check that one
 
that particular one that sunscreen and
 
or selected at the moment, and check that
 
for apartment number and verify that
 
isn't null, and then I can write something
 
please supply number
 
like that, but we still haven't been able
 
to say that this rule
 
is supposed to show up for this field
 
that I can select the field it's over
 
here and I'm supposed to find it and
 
validation rules, my menus really turn
 
out the wrong end all the time,
 
validation rules, rule one, please supply number
 
ok, so that would be one way to
 
further gearsteer our user to see
 
something, but how does it look
 
refresh, view, houses
 
it doesn't show anything, but if I null
 
that out first I get this one and I also
 
get the red information here
 
actually depending on what style you use
 
it really renders differently, so that
 
you can control
 
this and we have examples, where you just
 
have an "!" that was when you point that
 
it states what's wrong with it
 
so this is another important thing I
 
should mention that,when you have errors
 
and there's some something and updated
 
in  the information and you try to
 
save then it will ask you, there are
 
errors in the information, do you really
 
want to save anyway? "Hans Karlsen must
 
have apartment number" and the system
 
won't stop you from saving and this is
 
an important principle I think, that you
 
shouldn't stop the user from entering
 
information, even if it isn't correct
 
then it's better to to monitor all the
 
information that is in incorrect, so that
 
they can come back to it then and update
 
it don't stop users from entering data
 
and make a UI very clear, what you expect
 
but you should always
 
always honor the users' request to
 
save and maybe you should work with
 
the quality attributes like states that
 
you can't go into approved, before you
 
have apartment number or the rent in
 
dollars, but you shouldn't stop them from saving
 
ok, that's just my very clear view on this
 
ok, so I'm going to save it anyway, but run
 
nice wouldn't you say
 
just having the area information this
 
way, so whenever I enter something the
 
error goes away and of course I don't
 
get the warning, when I save, but that was
 
not a very good side for this example, so
 
switch back, so let's add
 
another constraint and we make that
 
not to error constraint, but rather look at
 
the other types of constraints, so we
 
have a constraint the "rent" and
 
the description and we do as we before "<as
 
string> lives free" and
 
we don't do an error just the information
 
expression would be self.rent->not null
 
because when the expression
 
is true then the constraint is
 
fulfilled and we don't shoe anything, but
 
as soon as the constraint is not true
 
then it's broken and then we show warning signs etc.
 
in this case information
 
refresh and now it's a little white
 
thing is Hans Karlsen lives free
 
shouldn't be the case that let them say alright
 
I wonder if we will have time to
 
cramming reports in this, so that we
 
could do rents and rents slips for these guys
 
I think maybe we'll come back to
 
that in the next session so we will
 
round this off and thank you very much
 
for following along with these tutorials and I'll be back

Revision as of 19:31, 5 January 2017

We create a seeker that translates ocl to sql for searching in databases of any size. We look at derivations, constraints and validation rules.

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.

View Model Editor: Search expressions Variables and Validations Derivation Attribute Visible expressions Test data: Tab separated data import Constraints Validation Rules Saving despite error warnings

Raw subtitles text

Hi and welcome back to MDriven designer overview

and this will be session five, so we

ended off the last session with the

creating a modal dialog that were

allowed us to pick tenants to our house

so we could pick persons that would

create lease contract and when we used

this dialogue "all persons" to to pick

the tenants from and as you can

imagine this strategy simply will not

hold, if we have a million persons and

this is good, when we have a hundred

maybe, but we can't really trust on all

instances to come to the rescue for

every scenario, we really desperately

need some way to vSeek the database

without instanciating the objects in

memory in their client during the

prototyper, so how do we go about that in

the view model pattern that we have

we don't like this, we right click and among the nesting

choices we have add add search expression

and such expressions are expression step

are supposed to be executed in

persistent storage, so that will be a

database if your back and is connected

to it, to one or it will be the xml file

in our case, where we prototype against

the xml file and if we are in turn

key will be the persistence mapper that

in turn will question the backing database

alright, so let's just go ahead and add

a search expression or maybe better

yet, we will do a new view model to keep

the searching and we will column this

"person search" and we type it to be "person"

right click and in the nested view models

add search expression and what happened

here, was that three columns were

added to the root and one of the

columns was a variable called the

vSeek result, this variable was created

and added to the viewmodel well it

wasn't because it's implicit, but it

should be in the newer versions, it will

be and so variables are just as they sound

something that we can define in

the view model and is only available

in the viewmodel scope, so this one

should be of collection person this is

where our result is going to end up from

a search and that one we connect to a

search result grid, so we have another

variable vSeekParam that is

defined as a string, so we have a column

in the UI that lets the user enter

a criteria and we have his search button

and the search button as the ocl

expression "selfvm.search", "selfvm" is

a special instance that is available in

view models only from OCL, let's see

what it contains, "selfvm." and from

here we can do things as

refresh, save, search, and a few other

things, like PSexpression_refresh, we can

come back to these in later sessions, but

for now we just going to use the vm

search, what the search operator does is

it looks for search expressions in the

view model and this is a search

expression, it's just another view model

nesting, but what separates this from the

other nestings is that it is not

connected to anything it's judt sits here

in the middle, doesn't need to be in the

middle because I said so I can right

click this and move this one up, so as

such expression is typed and it has

criterias right here it's just columns you

can have many criteria per search expression

the criterias are merged

together with the ocl operator

intersection, so what if these criteria

is true and we had another or this

criteria returned five persons 12345 and

we had another one that return two

persons 45, the intersection will take

what's equal between the two domains

so 4 and 5, but the search expression

the criteria is just another ocl, but

what's different about this is that is

the persistent storage subset of SQL Ocl-PS

it sets up here in the edge

when we entry and then we don't have access to every single

operator, we can't concatenate

strings etc. in this language

its main purpose is to be easily

transformed to SQL and executed as an

SQL query to the database and the

database will return a number of

identities, that we will in turn

interpret as persons and load personal

objects, so what we do is select and we

have a loop variable called A then

the mechanism to create the search

expression and all this for us it just

took the first string attribute of the

class that we had set person and that

was first name and set, well it's not

unlikely that is what you as a

developer want to limit search of,

this attribute as SqlLike(vSeekParam) and ends with a +% just like in

Sql operator like, so if you want to find

both and start, if you want to find

values that equals in the middle

that you would go something like this,

so this will be a resulting string that's

add into the SqlLike, that of course

will be translated into a like operator

in sequel, but as we are running against

an xml file, it will do something similar

just execute operator on that

so is this really what we want

well, we want to also check if it's the

last name maybe, so I use "or" and I use "LastName", like that

what should I show for a result I can show the things I have

first name and last name

so let's do one thing more in this view

and if you don't find what you expect

you might want to have somewhere where

you can create a new person, so let's

totally action definitions for this

dialogue and let's do a viewmodel action

and say that we should show it on the

root and say new person and the

execute expression on this one

should be PersonCreate, having done that

we can see, if we have, we can save this

push the play button, start system, Run Model

and continue, we never created an

action to actually show the seeker

view model, so person search

I right click on this and say add a

global action for show

okay then it's shown here, but we

probably want to sort into as menu group

a double click it

find it here set menu group-seekers

maybe, so this is might be where I want

to put all my search dialogues and who

knows

having done that and I head back to this one

model refresh, seekers, show per person,

search and I can search, but I only want those

that has an "a" in them

that's the only one, okay but

what if someone's missing

I can create a new person and call that

Andersson Anders, now if I search for a

I should find Anderson, but since

my xml file is, I guess it is case sensitive

it is case sensitive, so if we don't

like that we should have used operator

called .sqlLikeCaseInSensitive, where like

so now we have the searcher

let's go back to the house view

where we added the pic tenants action

so the pick tenant action and let's do

a copy of this, where did he go

let's do a copy of this, just going to do

a copy and paste, and "pick tenants by search"

and that one should bring up the

person search instead and if it does

it cannot call "check this" because that

variable doesn't exist in that UI like

instead it calls "SearchResultGrid" if that is empty

ok and this expression won't go either

because the same problem, like that

and it adds the tenants if I have selected any

so if we head back here we see that we have

another action pick tenants by search

of course in the end, I only want one of these

and our most likely it's the "BySearch"

that will survive and then maybe

I will change its name, model refresh,

houses and pick one here,

this doesn't have any tenants, I'm going to pick

tenants by search, I'm going to search on

find this one and I add that one, so I find it

rather disturbing that I don't see the name here

this is what I talked about

before, in session 2 or something

first, when we do the UI

and watch the information together

that it's obvious that, of course, we need to see

the name here as a new requirement, but

it's obvious now that we see the result and that's not a problem

this is the grid, which is this one, which

points to this one and let's add the columns

and add a generic column and in

that generic column sets an expression

that follow the role to tenant and the

first name that's the ocl editor acting up

when you don't have a separator

first name and you might have noticed that

I quite a lot of times had a need

to concatenate first and the last name

because that's simply how we

refer to persons, so this

sort of this pattern comes back

again and again and we're going to solve

this in just a minute, just going to save

the fact that we did a new column

and we call that "tenant name" and

we might want to give that a little more

space like three columns and we might

want to drag this one to the end

here and over there, and this one and we

added in summary session it's just we want

so if I save and refresh and look at all

the houses, was it this one and there's

Andersson, but I don't want it there

I want it first and let's show the

gridlines, like that, refresh like

that and that's pretty obvious and

how come this isn't even showing, well

it's too tight to show with this

style, so how would you solve that

well as you can see we are cramming

everything into one grid cell here

so we're gonna insert column and

give this one more space for the presentation

let's give it 2 and just to make things

look straight you're going to get the

rest of them 2 as well, head back,

refresh, views, all houses, when I click it

would still crap but you get the

general idea I will just insert yet another column

let's judt do that one for and refresh, like that

so this is important how quickly

we can go from feedback with our eyes

that this isn't good back to design how should be instead

back to verification, okay it's fine it

really saves everyboy's time

ok enough about that, and I said that

this pattern is repeated again and again

yes it is the first name and last name

it's a really common scenario that we

want to refer to persons by their full

name, so what we probably would want

to do is add an attribute, that is called

full name, but that wouldn't be very

normalized to store the first name and

the last name and full name, so let's

make that one right, that's another

attribute mode, it's not persistent as it

is not stored in the database, it's

derived as it's calculated from other

values that we do have, so I can paste in

that one and now that was for the

tenants and I don't want that

because now we are in the scope of a person,

so first name and last name

this makes up the full name and now we

have done this and we can make use of it

in our view models or wherever we access

model information, so instead of doing

this we can just as well do the full

name, like that and now it's grey I was

expecting it to be pink, because it is

derived I wonder why that was it's derived

don't have an answer, but going to save

what do you say? self is not...

that was just wrong, save and head back and

model refresh, and there's really

no difference, but like here now that we have it

let's use it, so tenance last name

and tenance full name is full name we got refresh

if I were to change the name

the right attributes were like this

there are always subscribed, so let's head

back to this view and add it as well

just to prove a point all persons add column

full name,  give it some space when it's in grid too

maybe we refresh, views, show all persons

as soon as I change one of their values

that build up the derivation, the

derivation will kick in and invalidate

as the UI looks at this value

it will force it to refresh, getting the

parts of the value and re-evaluate it,

so when I write Hans it will be updated

and if I were to change anything

it will be updated and that might be

Dick Smith, right, so derivation is also

available for associations

so what we could do, if we want to know

where the person lives is of course to

follow to see the rented apart and

look up the street for that house

but if that is a common operation

that we want to do we could

also add a association straight from

person to streets and since

I can rent many apartments, I potentially

have many addresses, so if I want to

derive this from available information,

I was saying that the association is derived

and I will take this end, will

have no value so I will say that there

it is non navigable, is navigable

false, because if I would want to go from

street to persons that would be another

derivations that would have another

expression, but this one should have

a derivation expression, that it starts

in person and it goes to

here is the one that is derived it

goes to owned house,

no, to lease contract, and from lease contract

rented apartments, and from rented

apartments - the street

I really must fix that, the street

so this will be a collection of

streets and if we were to show this in

our UI, I guess we could add column and

add generic column, and I follow the

relation to streets and that has a name,

but since this is a collection of

strings, we need some way to display it

in a single cell that we will do here in

the grid and done an operator that's

real handy for these cases as CommaList

so it will take all the strings in

the collection and append them as they

with the comma in between, so lives at, so heading

back to refresh, views or persons, so

here we see the address is swelling

ok, so good we have covered in a lot of

ground we have mentioned derivations

for attributes like the full night

servations for associations like the

street even if it was kind of silly,

because we would probably would like

the street number to get the precise

address and that's not possible this way

we have covered the idea of being able

to vSeek in a lost database and then

we also mentioned variables that are

available in view models, so I'm going to

show you another thing and what if we

are in a situation where we actually

need some test data, we need a lot of

persons to verify that our idea to search in

larger amount is valid and of course

we could run the prototype print

and press, we added the new person action

here and we could press that a thousand

times to enter the first and last name

but is there a better way, you ask

and since I talking about it you have

probably guessed that there is and I go

into the the Run button again because

I need a context that we are in the

XML persistence, we're going to start

the system and we're going to go into

the legacy debugger and in here we have

a tab over here

"import tab separated", so tab separated

data import via named view model, so what it

says he's given tab-separated data and

the view model name assumes row 1 in data

to be viewmodel column and creates

instances of view model class

so basically, if we have some data that

we paste in here, we should be able to

create objects of certain class, but it

also says, that we need a view model,

so let's create one, as you see the view

models are used over and over for different purposes

ImportPersons and type Person and let's assume

the first name is the first value and

the last name is the second value so

that's about it and given that I should be able to say

that this is the view model I want to

use or maybe I need to reread this, I do

start, lazy debugger,

input tap separated and view model and

get header from view model,and first-name

lastname alright, where can I get that, I found this site

as google for a list of names and find

anything on google list of random names

let's do 50 and male and female and well

no alliteration when you start the same ok

now I want to grab these on listing text

area, that's great, I wonder if this is

tab separate it probably isn't and going to

start notepad, alright so is that a tab

no, I copy tab and I want to

do is search and replace here, look

for space and paste in my tab and

replace it all and copy those and head

back to my importer and paste those, and

go, so that means I got a lot of dirty

objects here that I will save like that and

head back to my this one is going to

restart, person search and I have some

test data to work on

oh let's see everyone that starts either

their first loss name with "t" or has a "t"

really because that was what we said in the criteria

ok, so that means that when we do houses

and we want to assign tenance by search

and we do everyone with the "T"

because it is case sensitive as we

found out and select them and press ok

then we have those as tenants

alright moving ON as we see the

tenants here, but they haven't got

an apartment number...they

haven't got a rent figure, but they

really should this is important information

it's so important, that we might want to

add constraint on our class lease

contract to make sure that

the user of the system is made aware of

the importance of these values

so I select the class and I select

constraints and I get a little dialogue

where I can add constraints this

constraint I give a name "must have

apartment number" that's the name of my

constraint so that I can refer to it and

and I want to have a description as well

and I'm going to make easy to myself is

to do something like this, that's right

the expression is more or

less self.apartmentnumber and well it

should be not null, right

shouldn't be null and then now we got

that symbol that says that this has one

constraint, so when we look at our

diagrams we will see that this is

constrained in some way

ok, so if we were to head back to our

prototyper and show the house

we see, must have apartment number

must have apartment number, but there's

really no information on who is missing

the apartment number and let's head back

to the constraint and in the description we can actually

access attributes, so and we can

go like this as string within the

larger than and smaller than, greater than and

tags to say that we want to describe

the default presentation here, can you

please inject that and what is the

default presentation, default string

representation on lease contract is none

okay, but we might follow the link to the

person and use the full name again we

made use of the derived full name

like that and save, and head back, refresh, view

all houses, so Hans Karlsen must have an

apartment number and Dick Smith must

have apartment number and as I give our

Hans Karlsen apartment number one

that is removed and as I give  Dick's one 2

that one removed as well, so that's a

good way to feedback information to the

user about what needs to be filled in

but it might not be enough for us, we

might need to stress these even harder depending on use case

so there's more ways to signal the

user that something needs to be done

if we check the house view here and we had

the variables that we don't use in this

view we use them in the seeker, but we

also have the the constraints, so these

are the constraints that we found in the

classes that are used here that is just

one and depending on use case we might

not want to show it because who knows

the view might not even contain

their apartment number, so no need to

stress the user about entering it if it

isn't really possible, so that we could

just opt it out, but we want to have it

opt-in because it's good for us, but what

we could do is something more precise as a validation rule

let's right-click up in the top and add

a validation rule, so we got rule one and

then the rule one is that

let's say it's the fact that the selected, what

what I want to do is to highlight something here

apartment number to make the user see this

so what is this, this is here

the lease contract, so what I want to do in

my validation rule is to find a

variable the current

the lease contract and check that one

that particular one that sunscreen and

or selected at the moment, and check that

for apartment number and verify that

isn't null, and then I can write something

please supply number

like that, but we still haven't been able

to say that this rule

is supposed to show up for this field

that I can select the field it's over

here and I'm supposed to find it and

validation rules, my menus really turn

out the wrong end all the time,

validation rules, rule one, please supply number

ok, so that would be one way to

further gearsteer our user to see

something, but how does it look

refresh, view, houses

it doesn't show anything, but if I null

that out first I get this one and I also

get the red information here

actually depending on what style you use

it really renders differently, so that

you can control

this and we have examples, where you just

have an "!" that was when you point that

it states what's wrong with it

so this is another important thing I

should mention that,when you have errors

and there's some something and updated

in  the information and you try to

save then it will ask you, there are

errors in the information, do you really

want to save anyway? "Hans Karlsen must

have apartment number" and the system

won't stop you from saving and this is

an important principle I think, that you

shouldn't stop the user from entering

information, even if it isn't correct

then it's better to to monitor all the

information that is in incorrect, so that

they can come back to it then and update

it don't stop users from entering data

and make a UI very clear, what you expect

but you should always

always honor the users' request to

save and maybe you should work with

the quality attributes like states that

you can't go into approved, before you

have apartment number or the rent in

dollars, but you shouldn't stop them from saving

ok, that's just my very clear view on this

ok, so I'm going to save it anyway, but run

nice wouldn't you say

just having the area information this

way, so whenever I enter something the

error goes away and of course I don't

get the warning, when I save, but that was

not a very good side for this example, so

switch back, so let's add

another constraint and we make that

not to error constraint, but rather look at

the other types of constraints, so we

have a constraint the "rent" and

the description and we do as we before "<as

string> lives free" and

we don't do an error just the information

expression would be self.rent->not null

because when the expression

is true then the constraint is

fulfilled and we don't shoe anything, but

as soon as the constraint is not true

then it's broken and then we show warning signs etc.

in this case information

refresh and now it's a little white

thing is Hans Karlsen lives free

shouldn't be the case that let them say alright

I wonder if we will have time to

cramming reports in this, so that we

could do rents and rents slips for these guys

I think maybe we'll come back to

that in the next session so we will

round this off and thank you very much

for following along with these tutorials and I'll be back

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