Arduino: Nokia LCD & Sensors

46K16314

Intro: Arduino: Nokia LCD & Sensors

Arduino: Nokia LCD & Sensors

PROBLEM: Making my Arduino, a Temperature-Relative Humidity sensor and a Nokia3310LCD screen work together.

Now, I’m a Lazy Old Geek, so what I wanted was an Arduino kit that would take shields. This Freeduino was the cheapest that I could find at the time.
http://www.seeedstudio.com/depot/freeduino-usb-complete-kit-p-58.html?cPath=79_82

Since I was at Seeedstudio, I also ordered an HSM-20G Temp-RH sensor as it was on closeout and I needed something to try out my Arduino with. Seeedstudio no longer carries these and they’re hard to find. But here’s one site that sells the HSM-20G. 
http://www.emartee.com/product/41488/#

My dog, Marcus and I walk every day, so I decided I needed a portable Arduino to display Temp-RH on our walks. My display choice was the Nokia3310LCD shield. Plus it has a cool, little joystick on it.
http://www.nuelectronics.com/estore/index.php?main_page=product_info&cPath=1&products_id=12

So here’s the problems and processes I went through to make it all work.

Parts:
USB Freeduino kit      $22
HSM-20G Temp-RH   $9 (Closeout)
Nokia3310LCD           $14.99

Prices US dollars August 2010

STEP 1: Connecting the HSM-20G

Here’s the suggested interface circuitry for the HSM-20G. See picture.

Problem 1: The Nokia3310 shield fits on the Freeduino but I didn’t want to connect the HSM-20G to the shield. There’s only three components in the interface but I didn’t have a protoshield or any prototype area on the Nokia3310LCD shield, so I decided to build the interface onto the HSM-20G PCB. This is reasonable as it will always be needed whenever I used the HSM-20G. So I soldered the 10K to the T output and to a ground on the top side of the PCB. See picture.
Caution: This method is not recommended for the inexperienced.
A ground pin can be found by following the Ground (-) on the connector and tracing it on the PCB. The best way to verify that it is ground is to set your DMM to ohms and measure between the (-) pin of the connector and the pin you suspect is ground. It should be shorted (less than an ohm).

Problem 2: Well, I didn’t have a 47uFd capacitor, but I scrounged a 22uFd capacitor from some scrapped printer PCBs. I can tell from the documentation, that all the capacitor does is smooth out the Relative Humidity voltage, so 22uFd should work just fine. So I solder the capacitor to the 100K resistor, then soldered the resistor lead to the H pin on the connector making sure the positive side of the electrolytic capacitor is on this side. Then solder the other side of the resistor to the 10K resistor lead that is soldered to ground. See picture.

Make sure all of the leads aren’t touching other components and there are no solder bridges.

Problem 3: There was no mating connector for the HSM-20G module or a part number. I emailed Seeeduino requesting the brand and part number and of course got no response. But I scrounged through my junk pile and found a connector that worked.

Problem 4: How do I connect it to the Arduino? The preferred way would be to get one of those stackable protoshields and connect the HSM-20G to it. You could also put the interface circuitry on the protoshield. But I didn’t have one.
NOT RECOMMENDED: So I soldered the connector directly to my Freeduino. The (+) goes to a 5V connection. The (-) goes to a ground. The ‘H’ connects to Analog 1 and the ‘T’ to Analog 2.

Caution: When connecting two different sensors or shields, you have to be careful about not overlapping Analog and Digital pins. I started an Excel spreadsheet that has all of my shields and the pins they use. The Nokia3310 shield uses Analog 0 for the joystick so Analog 1 and 2 are available for the HSM-20G.

Here is the site where I found the sample HSM-20G sketch. This person really seems to know what he’s talking about. I verified the formulas with the HSM-20G data sheet. They are very accurate. There are other sensor interfaces on this page, also.
http://sites.google.com/site/measuringstuff/more-sensor-examples#TOC-HSM-20G-Humidity-and-Temperature-Mi

So I was able to run this sample HSM-20G sketch and display temperature and humidity on the PC monitor.

STEP 2: Connecting the Nokia3310LCD With Joystick Shield

So, this is easy, just take the shield (see picture). Plug it in. Load the sample sketch and run it. Right?

Problem 5: Well, not so fast. When I first tried it, the LCD screen was dark and hard to read. When I powered it up with an external 9V battery, it was unreadable.
So I emailed nuelectronics with my problem and of course, got no response.

Solution 1?: I wondered why the LCD was better with USB power than with external power. So I measured the 5V. See picture. It was 4.84V. The power was coming from my PC but it was also going through an unpowered USB hub and 4.84V is acceptable. I measured the same place using a 9V battery connected to the 7805 regulator. The voltage was 5.03V, about what I expected.
The tiny LED in my head blinked. I decided the LCD would be more readable if I dropped the ‘5V’ voltage to the Freeduino and the Nokia LCD. 
My Freeduino has the typical power supply like the schematic. All I had to do was remove the PWR SEL jumper and replace it with a potentiometer (variable resistor) on pins 1 and 2. (or 2 and 3 if using USB power)
So I scrounged a three pin connector with wires on it and solder two wires to a scrounged 200 ohm potentiometer. For those of you not familiar with potentiometers, they are a resistor with a mechanical wiper that can move from 0 ohms to the maximum (200 in this case). Why 200, well it was the smallest resistance I could find. Potentiometers have three leads. You only need two connections, make sure you solder one to the variable pin. See picture. Then plug it in, see next picture.
So this worked pretty good. I could tweak the potentiometer to get a nice display on the screen. BUT. . .

STEP 3: Temperature and Humidity?

So now what’s wrong? I did some thinking (dangerous thing to do for an OLD man). I wonder what changing the 5V does to reading analog voltages. Sure enough, I ran the HSM-20G demo, the temperature and humidity varied with the DC voltage for the Arduino. For a full explanation, I put the Analog to Digital Conversion in another step so most of you can skip it.

Solution 2?: Well, I thought one option was to run the Arduino at 5V and adjust the voltage on the Nokia3310 shield. There didn’t seem any easy way to do that.

Solution 3?: There are ways to set up the Arduino Analog converters so it doesn’t need 5V but this would require a lot of software changes and experimenting, plus I had to consider the joystick was analog also and was already designed for 5V.

Solution 4: While searching around for solutions, I found this website:
http://blog.thiseldo.co.uk/?p=383

‘Andy’ rewrote the libraries for the Nokia3310. I downloaded them and took a look. One of the comments said the initialization of the LCD was updated for a newer version of the LCD that nuelectronics was using. AHA! So I haven’t figured out how Andy found this out from the nuelectronics website, I couldn’t but it would’ve been nice if they’d updated their libraries.

I took the potentiometer out and put the regular jumper back in. So all problems solved, right?

Problem 5: Of course, not. I loaded Andy’s software as best I knew how into the Arduino environment. Then I tried to run his demo program and kept getting compile errors. I took out his sensor stuff since it was different from mine but still kept getting errors.
One of the reasons I decided on the Arduino was so that I didn’t have to delve into .H, .C, .CPP files. I am LAZY and OLD so didn’t want to bother. Well, that didn’t last long. I’m still a GEEK, so with some tweaking to Andy’s files, I managed to get it running. I did my best to decipher the error codes and made changes. I tried to mark all the changes I made with a //MTS. One thing I know isn’t working is big fonts.
I was able to get the demo sensor code into the Nokia code, it is under examples: nokia_Temp_RH.pde
So I zipped up the code that basically works. You should unzip the files right into the Arduino environment libraries. 
Mine is \\MICHAEL-PC\Users\Michael\Documents\Arduino\arduino-0018\libraries

So it all works pretty good. 

If any of you have had good or bad experiences with the Nokia3310LCD module, I would appreciate a comment. If you have a working library, that would be better yet.

Practical Information: I worked many years in the electronic test and measurement industry. Temperature and humidity vary over distance and time. The sketch takes 10 samples and averages them. Even so, I would guess the temperature is only within +/- 3 degrees and +/- 5% RH. But do you really care? Can you tell the difference between 89 and 92 degrees F. I can’t.

There is one circumstance where temperature could be critical. That is the freezing point of water, 32 degrees F. If this is critical for your application, then you should calibrate your sensor at this point. Probably the easiest way would be to stick the sensor in a baggie and sticking it in a bag of ice. You may have to adjust your formulas to get this temperature more accurate.

By the way, the other ‘fixed’ temperature is the boiling point of water. 212 F. Beware, this is only at sea level. At my altitude (4653 feet) boiling point of water is 203.6 F. Here’s a website calculator to find your boiling point  (temperature).
http://www.csgnetwork.com/h2oboilcalc.html

I used Google Earth to find the altitude at my house.

Be very careful if using boiling point as a calibration point.

Here is a suggested method for a one point calibration of Relative Humidity. I am not sure why it should be 75%.
http://exoticpets.about.com/od/herpresources/ss/hygrometer.htm

STEP 4: Analog to Digital Conversion

Most of you will want to skip this step.

If you are curious, or having analog problems, or looking to design some analog circuits, this may be helpful.

The ATmega328 has six analog input pins (see pin mapping). I am assuming the chip has only one A/D (analog to digital) converter chip in it. It is called a ten bit converter because it uses ten bits, 210 or 1024. The next drawing is a pretty good representation of how the converter works. The description is pretty technical and not that important for most Arduino users. But look at the analog signal represented by the smoothly curving signal on the top left. The stair step picture on the bottom left is representative of the digital output. You should note that instead of a smoothly curving line like the analog signal, the digital output is very jagged, often called a stair step. For the Arduino, the digital input goes from 0000000000 to 1111111111 which is 1023 steps in decimal.

This digital output doesn’t come out of the ATmega directly. It is only used internally by the software.
For the standard Arduino if you have 5V on the analog pin, the analogRead() will be 1023. Since there are only 1023 different possibilities that means each step is 0.004888 V. What that means is if you have analogRead() of 1023, it might mean 5.000V or 4.996V or somewhere in between. So each stair step means that the actual voltage can be off by 0.004888 V.

Practically speaking: So many of you will say who cares and in most cases you are correct. The accuracy of this reading is dependent on the accuracy of Vcc. If the Arduino is using a 7805 voltage regulator, then the 5V is +/- 1.5% which is 0.075V. The actual voltage could be anywhere for 5.075V to 4.925V which is much greater that the tolerance of a step. Another factor is noise on Vcc and on the analog signal. With a PC USB, the 5V tolerance is usually worse. Mine measured 4.84V.

Practically speaking with the standard Arduino, your Analog accuracy is limited to +/- 1.5% of the 7805 anyway.

NOTE: I am not 100% sure the following is accurate.
Advanced information: The 5V equals 1023 is not always true for the Arduino. The 1023 value is the voltage on the AVcc pin (pin 20 of the ATmega) under most conditions. Usually AVcc is tied to Vcc and is 5V. However, if the sketch contains the function AnalogReference(INTERNAL), then an internal reference of 1.1V (for the ATmega 168 and 328) is used. I am fairly certain what this means is that if you have a 1.1V signal on an analog pin, then the analogRead will be 1023. What this means is each step equals 0.001075V which means you can get more accuracy but it also means your maximum analog input can only be 1.1V. Now AnalogReference() can also be set to EXTERNAL. What this means is the analog reference is equal to the voltage you put on the AREF pin.
WARINING: Be careful if you use EXTERNAL and an external voltage source; this could interfere with and damage the ATmega. If you run a different sketch, that doesn’t have AnalogReference(EXTERNAL) set, then the external voltage reference is going to be fighting the default voltage on AVcc. It’s like connecting two batteries of different voltages together. You have a complete circuit with voltage but zero resistance so lots of current. I think there is a way around this by connecting it through a resistor but am not sure.

The reason I mentioned AVcc and analog reference is that I was setting my Arduino-clone up to use the Nokia LCD. I noticed that LCD worked fairly well when I was using the USB power but it was very dark when I plugged in an Adafruit 9 volt battery adapter. So I grabbed my trusty little DMM and did some troubleshooting. With the USB power, the Vcc and the Aref pin on the ATmega were at 4.84V. With the 9 volt battery connected to the 7805 regulator the two pins were at 5.03V. I also had a temperature sensor attached and further testing verified that the temperature reading varied depending on what the AVcc voltage was.

So if you want to improve upon the accuracy of the ‘standard’ Arduino, there are ways to do it.
If you set AnalogReference(INTERNAL), the ATmega will use 1.1V as a reference, adjust your calculations accordingly.
TIP: Any time you change AnalogReference(), discard the first analog reading as it will be meaningless.

If you set AnalogReference(EXTERNAL), you can provide you own reference, assumed it has a tighter tolerance.
See WARNING above.
TIP: Any time you change AnalogReference(), discard the first analog reading as it will be meaningless.

Those same six analog pins use a different method for output. 
D/A (Digital to Analog): The Arduino does not use a D/A (Digital to Analog) converter. Instead it uses, something called:
Pulse Width Modulation (PWM). PWM is not an analog signal. It is a pulsed digital signal. The drawing is from http://arduino.cc/en/Tutorial/PWM.
What the Arduino sends out is a digital signal that pulses off and on at different rates.

Instruction               Output
analogWrite(0)         0 volts out
analogWrite(64)       see drawing
analogWrite(127)     see drawing
analogWrite(255)     5 volts out

The drawing depicts the output over a period of time. As the drawing shows as the analogWrite value goes farther above 0, the width of the positive pulse gets wider. In other words, the time the output stays at 5 volts is longer. Now this is hard to explain but if you think in terms of average voltage when analogWrite is 0 then the average voltage is 0 and when it is 255, then the average voltage is 5V. Now when it is 127, you can see that it is on ½ the time and off ½ the time so the average voltage is 2.5 volts. Now as the value goes larger than 127, you should see that the average voltage goes up.

For those of you who like real world “proof”, here is a way to demonstrate this. A capacitor smooths out DC voltages. That is why you will always see them on Vcc and power supply outputs. Now if you take a capacitor (I’m guessing a 1ufd or maybe .5ufd) from an analog pin to ground. If you write your sketch to set up the analog pin as output and send an analogWrite(127), your DMM should show the voltage at 2.5V. By the way, the mathematics of PWM is beyond me but I know it gets into AC circuitry which is way beyond what this Lazy Old Geek wants to get into.

To put it simply, PWM puts out a signal that simulates an analog signal for most purposes.
TIP: One thing to remember, while the analogRead goes from 0 to 1023, the analogWrite only goes from 0 to 255, a byte by the way.

Non-essential Info: Most adequately designed Arduinos and clones have a 0.01ufd capacitor from Vcc to GND and next to the ATmega chip. The reason is to filter noise from the digital ATmega. When running a sketch, most of the internal circuitry is trying to switch from 0 to 5 volts almost instantly. This produces a lot of noise, a lot of which is high frequency. The capacitor will absorb most of this and smooth it out, just like it did in our little experiment.

15 Comments

Great 'ible, very comprehensive!
Just one question;
Why does referencing 1.1v make the analog read more accurate?
Does it not also have 1.5% error?
Thanks
You are correct, there would still be a 1.5% error. But 1.5% of 1.1V is a smaller voltage than 1.5% of 5V. However, this is only speaking theoretically due to the inaccuracy of the A to D conversion.
One of the factors that will affect 'accuracy' is the tolerance of the 1.1V reference which is between 1.0 and 1.2V.

However, for real applications, this is pretty meaningless. Generally, I choose a reference voltage based on the sensors connected to the Arduino. If the maximum analog voltage is going to be 1.1V or less than choosing the internal 1.1V reference would probably be best.
I usually like to use 3.3V reference as shield compatible Arduinos have a 3.3V source and so do most USB to Arduino converters. These are usually much more accurate then the USB 5V or (I think) the 7805 regulators.

LOG
I am trying to hook up a nokia 5110 that is supposed to be the same pin mapping. Is this the pin mapping that the shield uses?

#ifdef PB1
#define LCD_RST PB1
#define SPI_CS PB2
#define SPI_MOSI PB3
#define SPI_SCK PB5
#define LCD_DC PB0
#define LCD_BL PD7
#else
#define LCD_RST PORTB1
#define SPI_CS PORTB2
#define SPI_MOSI PORTB3
#define SPI_SCK PORTB5
#define LCD_DC PORTB0
#define LCD_BL PORTD7
#endif

which when translated for the arduino pins this is how I believe it should be set
LCD_RST digital pin 1
SPI_CS digital pin 2
SPI_MOSI digital pin 3
SPI_SCK digital pin B5
LCD_DC digital pin 0
LCD_BL Pdigital pin 7

am I missing something?

Sorry, my ISP won't connect to Instructables.com.

I tried to design my own interface to a Nokia LCD but couldn't figure it out so that's why I bought one with software. So I can't really help with the code. I don't see anything obviously wrong with your code but don't know.

One thing I do remember was that the LCDs were not 5Vdc so a lot of the hacks had ways to convert the signals. I think the LCD expects 3Vdc signals.

Good luck.
Very well written! Nice and detailed!
Thanks. I get frustrated with internet articles that don't quite tell you what you're looking for so I try to make mine as complete as possible.

Lazy Old Geek
Do you have any experience getting this display shield to share the SPI bus with other shields? I'm trying to get it to play nicely with a Freetronics ethernet shield (which unlike most w5100 based shields, honors the cs line).

-- Mitch
Sorry, I haven't. This is my only experience so far.

LOG
I can tell that you are a very organized, grammatically correct, and SMART, lazy old man. You speak my language and I love how you actually got the results that you desired, good work, very resourceful and patient. Paying attention to the smaller details was always a key highlight in engineering.

To add furthermore on to your very eloquent and detailed instructable that you have created I would also like to add one small tidbit to your comment on discarding the first reading from the modified AnalogReference() id.

Your tip that explains not to include the first reading ( and I am speculating, I have never used an arduino or any of the devices listed here ) most likely is due to two consequences. Firstly, when you plug a device into any battery or power source, there will be an initial "pulse" high energy output that comes from the power source. One can interpret this as the phenomenon of kinetic energy from physics, i.e. - there are two different equations for the initial (or static) state of energy and when the energy is in motion. Molecules are put into a state of havoc then ultimately into a more steady state of flow, thus explaining the two different equations. Different batteries and alternative power sources that I know of have different "pulse" currents that initially "pulse" out, just like the Silver Surfer character in the second "Fantastic 4" comic movie. Secondly, the initial reading would be tossed out ( again here, speculating ) because the unit, when modified, might not use the same algorithm as the normal, debugged input. It might not be compensating for the initial "bounce effect", i.e. basketball effect of when a button is pushed. Digitally speaking, this means that whenever you are typing on your keyboard or pushing a button, there is a microcircuit with an algorithm automatically compensating for the impossibly small bounce that sends multiple "1's" to your next device. To fix this, the algorithm simply is written to ignore the initial bounce gap interval of "1's" and set the gap interval of "1's" to just plain and simply "1". That's it, and everything runs smoothly.

Did my comment overcompensate the "geek" effect in this article? ;)

I hope someone here understands it and appreciates it.

-Keith
Thank you for your comments. I do try to pay attention to details. Unfortunately, as I get older, it is harder to do. In my case, I think this comes from a couple of factors. One, I spent many years in electronics, mostly developing test systems where missing details could develop serious design flaws. Two, in this work, i had to write a lot of test programs. In debugging and troubleshooting programs you learn to pay attention to the littlest details that won't allow the program to run or won't run correctly.

I find your comments interesting. I do not fully understand them but definitely see and engineering background and probably highly educated.

Some of the initial power states, I am aware of as are most designers. I don't know how familiar you are with microcontrollers but at least in the earlier ones I used to work with there was always a resistor and capacitor attached to the the reset pin so there was always a delay that held the microcontroller idle until power was stable. As for the ATmega chip, apparently, there is a watchdog circuit that keeps the microcontroller idle until power is stable. I am not sure if this is what you were referring to or not.

Here is the data sheet for this microcontroller all 448 pages of it.
http://www.atmel.com/dyn/resources/prod_documents/doc8161.pdf

My suggestion to discard the first sample only refers to conditions where the Analog Reference type is changed and not to initial power up conditions.
From the datasheet:
'The first ADC conversion result after switching reference voltage source may
be inaccurate, and the user is advised to discard this result.'
Now, i am not sure but I believe it is using the same algorithm but the hardware comparator reference is changing. Perhaps the reference has residuals of the old reference until after the first conversion?

I am also familiar with the 'bounce effect' and there are hardware circuits to 'debounce' switches. In the Arduino world, though mostly what you see are little tiny buttons connected to an ATmega digital pin with a pullup resistor. So 'debounce' is usually done in software, if at all. If you do an internet search for Arduino debounce, you will find lots of suggestions and software that didn't work and fixes but it is very hard to find any information on theory and how to make it work if it doesn't.

After perusing the Arduino forum and playground areas for analog reference information, I was totally frustrated by the lack of understandable information that I could trust. It was probably there, but I have no good way of finding it.

That is one of the reasons why I chose Instructables to write about what I know, explain some of the theory and put the theory to work.

So, yes, I really appreciate your comments. It's good to know that someone is actually paying attention.

LOG
Well, it would appear that you are still as sharp as a tac! I get what you are saying by the initial "startup" states in your pdf now that you explained it.

I took 4 years of physics and 2 1/2 of electrical engineering, along with another 2 1/2 of advanced math and statistics/finance/economics courses. Let's just say I wanted to get the most for my money out of college ^^ :).

As far as the "not understanding" part, I was merely referring to patterns I saw while reading your statements in this instructable. There are many equations and laws of physics which can be referenced and applied to the problems that you were having, and apparently already knew of. I just found it fascinating that someone would actually put in as much work as you did in writing an article - and yes, I do appreciate it very much! I learned a lot from your article and it brought back memories from when I used microcontrollers and programs to implement circuit functions (and the headaches!!). But as you say, it is very hard work. I certainly wish I had you as my teacher instead of foreigners with thick accents that I could not even understand!

The capacitors are there to compensate for delay like you said and to provide open/close switch circuit operations for different "states" of digital electrical "time frames" - so to speak. For any given output at any given time, there must be a programmed circuit (both hardware and software) to fulfill all of the different combinatorial digital circuit logic functions that one may want, therefor, one must acquire capacitors and transistors to provide for those exact functional outputs. It is indeed very complicated and I did not want to pursue that stressful environment. The capacitors are almost always coupled with transistors in any circuit due to our "digital age".

ON....OFF....ON....OFF....etc...

The capacitor takes care of one part and the transistor takes care of the other. They pretty much work hand in hand, different properties and such with AC and DC current. I'm a sucker when it comes to "why" these things work the way they do. Technology is unbelievably mind boggling, it is very cool to see how you "hacked" the device to work for you and the very hard work you put into it to make it precisely the way you wanted it to be. I still could look at a circuit diagram drawing now and get lost in less than a second, and I'm only 26!! lol.

I make LED toys now ;). I've been working on new designs for over a year now and I am trying to start up a small business. I'm a geek too :).

Do you have any other instructables online?
Ha! Yes, I suspected you had a strong physics background!
My education was mostly practical, starting with training in the USAF. It was a rather ominous start. I was taking classes during the graveyard shift and all I remember was sitting around playing Battleships. They did get rid of that instructor. Like you I was a sucker for the ‘why’, so when I wasn’t drunk or sleeping (graveyard shift again) I probably applied my training better than most.
After getting out, I later took a correspondence course in microcomputers. Most of my experience came from working in electronics test and measurement for over 20 years. The Internet became such a significant factor in my life, instructor, reference, guru.

In some ways doing this project wasn’t that hard for me. It was mostly just applying some theory and a lot of experience. Probably the harder part was formatting the Instructable, trying to keep it interesting but informative and complete. I don’t think I’m ever satisfied with them but just decided to publish and hope I didn’t make any major mistakes.

This is actually my fifth:
Making a bedframe with lots of storage
Easy Laptop stand
Arduino-No Blinky
A 67 cent postal scale

You might like Arduino-No Blinky more than most. I don’t think it went over that well.
I actually was working on one called Arduino Basic Electronics but decided it was too theoretical for the more practical Instructable audience. I incorporated some of the material in the two Arduino Instructables.
I am currently working on one about GPS. Like you, I want to know how it works.

So good luck in your business. Let me know if I can help. I guess you can send me a Private Message.

LOG
Very cool btw, this must have taken you F-O-R-E-V-E-R! I appreciate the time that you spent to put together the grammar, organization, detailed information, and dummied down explanations. Your perspective writing is very well written and you have a keen sense of situational awareness. Very refreshing.
Thanks again. For a few years, I managed an electronics test group and had to do some instructing. More recently, I was an instructor in electronics and programming at a technical school. One of the things I learned was to know your audience and teach to them. Now the Instructable 'audience' varies all over the place.

Especially, in the Arduino Instructables, I notice a lot of authors, doing incredible, amazing projects with them and many seem to have a limited understanding of electronics and programming. Those are the ones who inspired me to do some of my Instructables. I hope to return the favor by helping them in future projects and giving them a better understanding of how it works.

Yes, these do take a while. But teaching is a learning experience. I'm always learning new stuff when I write one of these. It's good to keep the OLD brain cells active.

LOG
Thanks for the feedback. It is appreciated.

LOG