Hi,
a user of my extension got the following error message.
Line 134: info = wallin.map(&:to_l)
wallin = [“0.15”, “3.0”, “0”, “0.05”, “0”, “0”, “0.01”, “0”, “0”]
Other users don’t have problem with this extension also it works well in my system. Can you help me to understand why it happen in my user system?
An input like “0.15” can only be converted correctly with .to_l on systems with a period as a decimal separator. As well as “0,15” only converts correctly with .to_l on systems with a comma as decimal separator.
For this conversion you will first have to look at which decimal separator the user uses.
You can then provide the values in the wallin Array with the correct decimal separator.
Now no ArgumentError will be caused because the correct decimal separator is applied.
I had such a problem before with Russian keyboards but “0.15” was created by my plugin, not by the user. By the way, I will ask the user to use “,” instead of “.” and will inform you of the result. Is there any way to know user system use “.” or “,” for decimal separator?
Avoid representing lengths as either Floats or Strings in your code base. Use Length whenever you can. Only use Strings at the UI layer and only use Floats for storing values where you can’t use Length, e.g. to a config file. Always use Lengths in between.
The Length/String conversion is aware and dependent on the user’s settings such as system decimal sign, model unit, display precision etc. Length/Float conversion is always the same and can safely be used to store data.
When the user has entered Strings, immediately turn them to Lengths before using them. If you need to save them to a config file, convert them to floats before doing so. Never save the user input Strings directly to a config file if, as they can’t be read back reliable if the user settings change in between. SketchUp lengths string representations are meant for humans, not machines.
This shouldn’t be needed if the right classes are used. Also this would break if any other decimal separator than “.” and “,” is ever added or if “.” is used denote other things than decimal separator in the input.
Usually, I do the following steps.
1- Read the last saved data and use them in UI.inputbox and let users change them as they wish.
2- Convert data to length from string. (to_l)
3- If cannot convert reset data.
4- Save data in the way the user entered.
for example, if users write 10.cm for a length next time also 10.cm will show them.
Would you please let me know how can we do it?
If I change data to float, I will lose the unit entered by users (cm in this example).
Thank you for your attention.
Do not save the string the user entered. Those strings only belong in the UI layer and are dead once they’ve been parsed to Lengths.
No SketchUp tool tries to show the value in the same unit as entered. If the user wants to read the value in cm, they can go to Window > Model Info > Units and set it to be cm.
It would more likely display as “~3.93”, rounded according to the user´s preferences.
In any case, saving the input string directly is more likely to cause inconsistency as one field could say “3m”, one say “10 cm” and one say “15”, instead of them saying “3000 mm”, “100 mm” and “15 mm”.
Saving the input as a string can also lead to data loss if you type “3” in one model being set to m and have it read back in a model set to mm.
For solving this problem I consider different saving spaces for different unit. So data that saved in meter unit is not same as data that saved for inch.
That alone wouldn’t guard against changing the system decimal separator, and it would prevent the previous input from being correctly formatted with the model unit if you mix m, cm and mm in your input. The string representations of Lengths just aren’t designed to be stored for later use this way.
You could still split up saving of values to avoid things like “~3.93"”, but for that splitting imperil from metric is probably better so a user keep their values when changing between say mm and cm.
Thank you for the very useful document. Following come from it.
" Because users have different decimal separators depending on their locale, never store unit data as formatted strings! Don’t write Length values out as length.to_s – instead convert it to a Float . That way you can be sure that you can read the data back regardless of the locale of the target system because Float.to_s always use period as decimal separator – and String.to_f always expect a period. To load a unit from a float stored as string correctly you use: string.to_f.to_l ."
I think “string.to_f.to_l” is really good because in this way we never have errors even wrong number format like “x10”.(“x10”.to_f.to_l = 0). (x10.to_l = error). I think the best way is to read numbers and units from the user, convert them to float, and save numbers and its unit in our format. is it ok?
I used the following ruby code and it works well. If you enter 5in or 5", “5.0 Inch” will save.
if Sketchup::RegionalSettings.decimal_separator == '.'
wallin = wallin.map {|s| s.sub(',', '.')}
else
wallin = wallin.map {|s| s.sub('.', ',')}
end
for i in 0...9
begin
Float(wallin[i]) # if string is not purely a number error will happen.
info[i] = wallin[i].to_l
wallin[i] = wallin[i].to_f.to_s
rescue
if wallin[i].to_f != 0 # if it is 0 means data is not a number.
if wallin[i].end_with?('"', "Inch", "inch", "INCH", "Inches", "INCHES", "inches", "in", "IN")
info[i] = wallin[i].to_f
wallin[i] = "#{wallin[i].to_f} Inch"
elsif wallin[i].end_with?("'", "Foot", "foot", "ft", "FT", "FOOT", "Feet", "feet", "FEET")
info[i] = wallin[i].to_f * 12
wallin[i] = "#{wallin[i].to_f} ft"
elsif wallin[i].end_with?("mm", "MM")
info[i] = wallin[i].to_f / 25.4
wallin[i] = "#{wallin[i].to_f} mm"
elsif wallin[i].end_with?("cm", "CM")
info[i] = wallin[i].to_f / 2.54
wallin[i] = "#{wallin[i].to_f} cm"
elsif wallin[i].end_with?("m", "M")
info[i] = wallin[i].to_f / 0.0254
wallin[i] = "#{wallin[i].to_f} m"
elsif wallin[i].end_with?("Yard", "yard", "YARD", "yd", "YD")
info[i] = wallin[i].to_f * 36.0
wallin[i] = "#{wallin[i].to_f} yd"
else
info[i] = 1000000 # This is out of range number and cause all data will reset.
end
else
info[i] = 1000000 # This is out of range number and cause all data will reset.
end
end
end
But for the record, when I posted that, I wasn’t giving a response based on best practices, but only directed at his specific use of an array of strings.
@Majid,
If you edit your plugin to follow Thomas’ (and Juila’s) advice and change the wallin array to hold Float values in inches (with period as the decimal separator,) then you will not need the character substitution.
Storing the values as Float, and convert to Length for the UI.inputbox call, it will automatically display and read user input in the user’s model units and return values as Length (decimal inches.)
Then you will not need the rigmarole you just posted at all.
This approach does not work. “10,5” would be red as 10, because String#to_f doesn’t honor the user’s chosen decimal sign. Also it doesn’t differentiate “10 m”, “10 cm” and “10””.
Only use strings in the UI layer. Only use floats to serialize the value for storage. In ALL other cases, use Length for length values.
Failing on invalid input is not a bad thing. That’s where you notify the user the input was invalid.
This is the reason at the beginning I used the following codes.
if Sketchup::RegionalSettings.decimal_separator == '.'
wallin = wallin.map {|s| s.sub(',', '.')}
else
wallin = wallin.map {|s| s.sub('.', ',')}
end
If your system decimal separator is “.” and you type 10,5 it will change to 10.5. As you can see in my codes, I only use string.to_f.to_l if it is a valid pure number.
Rather than trying to duplicate String#to_l you can just use String#to_l. It is less work, less error prone and it automatically works with any new units SketchUp may add in the future.