@user97622, you are looking for a cycle waster? Nope, you ain't go-n-a get one, no matter how much I'm stressed you (try to) pull us over on... But let's start over on the good foot:
We all - including me - we understand where you are coming from... But come on a ride - with a thread of thought and Espruino - something you will never ever regret.
Think about this scenario: instead of waiting and watching the clock painfully until it is time to do the next move, you set the timer / alarm clock and put your mind on something more enjoyable then constantly watching the hands moving or digits flipping and being terribly afraid to still miss the moment(s)...
Espruino is event driven like JavaScript in the browser. When you do nothing in a Web page on the browser, (usually) nothing is going on. But when you hit a button in the Web page, things move. The way you set that up is that you tell the button '... onClick="doThis();" ...', or generalized: ...onEventXyz(doThis();...
From your code, I cannot read what the application tries to get done. From what I though get is that the servo has to move, then wait for some time, and then move again...
Your implementation of waiting until it is time is understandable. It though suffocates Espruino. Matter of fact it kills the whole system: the (error) message you get are its lasts words: "Your wasting of my cycles 'to wait' make me so busy that I cannot even do my home / house work, such as tending to Wifi, for what you like me so much!" :}
The way out of this waiting business is to use one of these (time) event driven JavaScript language constructs:
setTimeout(thenDoThis,afterThisManyMilliseconds)
setIterval(doThis,everSoManyMilliseconds)
Both use timers internally that take care of the deferring / postponing the business and let you - and your processing resource of the hook - to do something else until then. This is also the reason that the control returns immediately and is not stuck in waiting until time has passed. When set time will come, your business is taken care of without you having to do something. I intentionally use deferring or postponing instead of waiting, because waiting implies sitting idle or twiddling your fingers until the time of your appointment comes to pass... and that's not what you do and more so cannot afford in real life.
Your requirement
this code must be stop the execution until the time is completed
gives me two options:
1. When a particular time - real time - has come, you want the servo to move.
2 . At a fixed time interval - for example every second - you want the servo to move.
For option 1, lets take your alarm clock and your favorite music you want to be woken up by... so no matter when you go to bed, you set your alarm clock... of course you have now a phone that does that job. When you set the alarm and save it, it will give you feedback about how much time will pass until the alarm will ring... that's exactly what my (Android) phone does, and that's also what will implement here:
Assume you have to get up at 6:30am... lets also assume you go to lala land
before midnight...and with that we know the 'next' 6:30am will be the next day... actually, it does not matter. We can manage without making it too complicated.
What we will do is to take the time now, calculate the time difference to 6:30am tomorrow, and tell Espruino to set the timer - timeout - for that time and upon timeout to turn the music on.To not get bogged down about how we would do the music thing, we just turn the red LED1 on.
To get there, we need to know how many milliseconds (ms or MS) it is from now to tomorrow 6:30am
Here comes the code... (c=`comment`; instead of // comment is used make to comment stick out vs. being dim and difficult to read. In real code you just use// comment and only where you think you need it to give a better understanding to the reader -will be yourself after a while you wrote the code....
var wakeupHr = 6, wakupMin = 30; var c=`6:30am wakeup time`;
var now = new Date(); c=`now is data and time, a Date class instance`;
var wakup = new Date(now.getTime()); c=`we start w/ a copy of now and remove and add time units to get to the targeted wakeup date and time. `;
wakeup.setHours(wakeupHr); c=`setting the hours right`;
wakup.setMinutes(wakeupTime); c=`setting the minutes right`;
wakup.setMilliseconds(0); c=`blow the milliseconds away`;
c=`---> wakeup date and time could be in the past... when going to bed between 6:30am midnight, but doing so between midnight and 6:30am, it is in the future. To make it right, we make former to be in the future by conditionally adjusting wakup (date and time), at the same time we convert date and times to milliseconds - the way we will need it after all to set the timer`;
var wakupInMS = wakup.getTime();
if ( (now.getHours() * 60 + now.getMinutes())
> (wakupHr * 60 + wakupMin) ) { c=`hr and min is good enough!`;
wakupInMS += 24*60*60*1000; }
var nowInMS = now.getTime(); c=`now in milliseconds`
c=`finally, we calculate the time to defer and set the timer:`;
var timerTimeoutTime = wakeupInMS - nowInMS;
var timeoutTimerHandle = setTimeout(function(){
LED1.set(); c=` turn red LED1 on in...
}, timerTimeoutTime); c=`timerTimeoutTime milliseconds...`;
And be assured that when power is not breaking down, your favorite music - sorry: just the red LED1 - will turn on 6:40am in the morning...
There is a bonus to catch: your boss - friend? - calls you few minutes after you set this all up and tells you: lets hit the gym for 30 minutes before work or school or what ever you planned to do, you can use the timeoutTimerHandle to 'clear' the set timeout and set it for half an hour earlier:
clearTimeout(timeoutTimerHandle);
c=` .... do the same procedure as above to set the alarm - oops, music delight - to 6:00am straight...`;
For option 2, you want to lite up the green LED2 for 5 seconds every minute and move the stepper motor 60 steps ahead. The code could look like below. To make it a bit better regarding CS in general and coding in particular, we define a function for what has to happen every minute:
function blinkAndMove() {
LED2.set(); // turn green LED2 on and...
setTimeout(function(){
LED2.reset(); // ...turn it off...
},5000); // ...after/in 5000 ms
// motor.stepForward(60); // do the 60 steps forward
}
var iId = setInterval(blinksAndMove, 60000); // get interval started
To stop this ongoing blinking and moving, you use the interval Id (or handle) you got from interval setting and use it to clear the interval:
clearInterval(iId); // stops the interval
Often, the code needs to know if a set timeout or interval is still going on in order to branch accordingly. For that reason the defining of the handle and clearing of it looks 'better' when done like this, because starts correctly initialized and also clears the variable from the then 'defunct' timer handle...
var intervalId; // declares it with value undefined
// some other code
intervalId = setInterval(someFunction, intervalTime);
// some more code
if (intervalId) intervalId = clearInterfal(intervalId)
And if you deal w/ a setTimeout and the timed function is relate to one ant the same timeout handletimeoutId, first thing in the that function you clear the handle: timeoutId = undefined;
Back to the way of coding events and times in Espruino:
And before you even notice, your thinking has changed to the exciting and very powerful event driven way of doing things versus always running in a circle - down the check list, go back to the top, run down again, and again, and again, and so on and so fort - like in the Arduino main loop with constantly checking all kinds of flags... (that the modules have set on the actual events). So:
Why not just attach the code to do to the actual event? There you go:
Use Espruino!
Driving directly by events is one of the reasons that things can run on a very small battery for a very very long time. The processor is active only when / while something is done...
Related to these concepts you will encounter terms such as call back and Promise... but let's save that talk for later.
I hope this helps you switch gears... enjoy, and come back and ask more questions, challenge the forum community, share your success to our joy, and pay forward...
Espruino is a JavaScript interpreter for low-power Microcontrollers. This site is both a support community for Espruino and a place to share what you are working on.
@user97622, you are looking for a cycle waster? Nope, you ain't go-n-a get one, no matter how much I'm stressed you (try to) pull us over on... But let's start over on the good foot:
Welcome @user97622 to Espruino and its forum!
We all - including me - we understand where you are coming from... But come on a ride - with a thread of thought and Espruino - something you will never ever regret.
Think about this scenario: instead of waiting and watching the clock painfully until it is time to do the next move, you set the timer / alarm clock and put your mind on something more enjoyable then constantly watching the hands moving or digits flipping and being terribly afraid to still miss the moment(s)...
Espruino is event driven like JavaScript in the browser. When you do nothing in a Web page on the browser, (usually) nothing is going on. But when you hit a button in the Web page, things move. The way you set that up is that you tell the button '... onClick="doThis();" ...', or generalized:
...onEventXyz(doThis();...
From your code, I cannot read what the application tries to get done. From what I though get is that the servo has to move, then wait for some time, and then move again...
Your implementation of waiting until it is time is understandable. It though suffocates Espruino. Matter of fact it kills the whole system: the (error) message you get are its lasts words: "Your wasting of my cycles 'to wait' make me so busy that I cannot even do my home / house work, such as tending to Wifi, for what you like me so much!" :}
The way out of this waiting business is to use one of these (time) event driven JavaScript language constructs:
setTimeout(thenDoThis,afterThisManyMilliseconds)
setIterval(doThis,everSoManyMilliseconds)
Both use timers internally that take care of the deferring / postponing the business and let you - and your processing resource of the hook - to do something else until then. This is also the reason that the control returns immediately and is not stuck in waiting until time has passed. When set time will come, your business is taken care of without you having to do something. I intentionally use deferring or postponing instead of waiting, because waiting implies sitting idle or twiddling your fingers until the time of your appointment comes to pass... and that's not what you do and more so cannot afford in real life.
Your requirement
gives me two options:
For option 1, lets take your alarm clock and your favorite music you want to be woken up by... so no matter when you go to bed, you set your alarm clock... of course you have now a phone that does that job. When you set the alarm and save it, it will give you feedback about how much time will pass until the alarm will ring... that's exactly what my (Android) phone does, and that's also what will implement here:
Assume you have to get up at 6:30am... lets also assume you go to lala land
before midnight...and with that we know the 'next' 6:30am will be the next day... actually, it does not matter. We can manage without making it too complicated.
What we will do is to take the time now, calculate the time difference to 6:30am tomorrow, and tell Espruino to set the timer - timeout - for that time and upon timeout to turn the music on.To not get bogged down about how we would do the music thing, we just turn the red LED1 on.
To get there, we need to know how many milliseconds (ms or MS) it is from now to tomorrow 6:30am
Here comes the code... (
c=`comment`;
instead of// comment
is used make to comment stick out vs. being dim and difficult to read. In real code you just use// comment
and only where you think you need it to give a better understanding to the reader -will be yourself after a while you wrote the code....And be assured that when power is not breaking down, your favorite music - sorry: just the red LED1 - will turn on 6:40am in the morning...
There is a bonus to catch: your boss - friend? - calls you few minutes after you set this all up and tells you: lets hit the gym for 30 minutes before work or school or what ever you planned to do, you can use the
timeoutTimerHandle
to 'clear' the set timeout and set it for half an hour earlier:For option 2, you want to lite up the green LED2 for 5 seconds every minute and move the stepper motor 60 steps ahead. The code could look like below. To make it a bit better regarding CS in general and coding in particular, we define a function for what has to happen every minute:
To stop this ongoing blinking and moving, you use the interval Id (or handle) you got from interval setting and use it to clear the interval:
Often, the code needs to know if a set timeout or interval is still going on in order to branch accordingly. For that reason the defining of the handle and clearing of it looks 'better' when done like this, because starts correctly initialized and also clears the variable from the then 'defunct' timer handle...
And if you deal w/ a setTimeout and the timed function is relate to one ant the same timeout handletimeoutId, first thing in the that function you clear the handle: timeoutId = undefined;
Back to the way of coding events and times in Espruino:
And before you even notice, your thinking has changed to the exciting and very powerful event driven way of doing things versus always running in a circle - down the check list, go back to the top, run down again, and again, and again, and so on and so fort - like in the Arduino main loop with constantly checking all kinds of flags... (that the modules have set on the actual events). So:
Why not just attach the code to do to the actual event? There you go:
Use Espruino!
Driving directly by events is one of the reasons that things can run on a very small battery for a very very long time. The processor is active only when / while something is done...
Related to these concepts you will encounter terms such as call back and Promise... but let's save that talk for later.
I hope this helps you switch gears... enjoy, and come back and ask more questions, challenge the forum community, share your success to our joy, and pay forward...